2015-12-08 04:51:55 +08:00
|
|
|
#pragma once
|
2015-09-13 17:34:17 +08:00
|
|
|
|
2015-10-08 06:58:29 +08:00
|
|
|
#include "transactions/transaction.hpp"
|
2015-09-13 17:34:17 +08:00
|
|
|
#include "mvcc/atom.hpp"
|
2015-10-08 06:58:29 +08:00
|
|
|
#include "mvcc/mvcc_error.hpp"
|
2015-09-13 17:34:17 +08:00
|
|
|
|
|
|
|
#include "data_structures/list/lockfree_list.hpp"
|
|
|
|
#include "utils/counters/atomic_counter.hpp"
|
|
|
|
|
|
|
|
// some interesting concepts described there, keep in mind for the future
|
|
|
|
// Serializable Isolation for Snapshot Databases, J. Cahill, et al.
|
|
|
|
|
|
|
|
namespace mvcc
|
|
|
|
{
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
class MvccStore
|
|
|
|
{
|
|
|
|
using list_t = lockfree::List<Atom<T>>;
|
|
|
|
|
|
|
|
public:
|
2015-10-08 06:58:29 +08:00
|
|
|
using read_iterator = typename list_t::read_iterator;
|
|
|
|
using read_write_iterator = typename list_t::read_write_iterator;
|
2015-09-13 17:34:17 +08:00
|
|
|
|
|
|
|
MvccStore() : counter(0) {}
|
|
|
|
|
2015-10-08 06:58:29 +08:00
|
|
|
read_iterator begin()
|
|
|
|
{
|
|
|
|
return data.begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
read_write_iterator rw_begin()
|
2015-09-13 17:34:17 +08:00
|
|
|
{
|
2015-10-08 06:58:29 +08:00
|
|
|
return data.rw_begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
Atom<T>* insert(const tx::Transaction& t)
|
|
|
|
{
|
|
|
|
// create a first version of the record
|
2015-09-13 17:34:17 +08:00
|
|
|
auto record = new T();
|
|
|
|
|
2015-10-08 06:58:29 +08:00
|
|
|
// mark the record as created by the transaction t
|
2015-09-13 17:34:17 +08:00
|
|
|
record->mark_created(t);
|
|
|
|
|
2015-10-08 06:58:29 +08:00
|
|
|
// create an Atom to put in the list
|
|
|
|
auto atom = new Atom<T>(counter.next(), record);
|
|
|
|
|
|
|
|
// put the atom with the record to the linked list
|
|
|
|
data.push_front(atom);
|
|
|
|
|
|
|
|
return atom;
|
2015-09-13 17:34:17 +08:00
|
|
|
}
|
|
|
|
|
2015-10-08 06:58:29 +08:00
|
|
|
T* update(Atom<T>& atom, T& record, const tx::Transaction& t)
|
2015-09-13 17:34:17 +08:00
|
|
|
{
|
|
|
|
auto guard = atom.acquire();
|
|
|
|
|
|
|
|
// if xmax is not zero, that means there is a newer version of this
|
2015-10-08 06:58:29 +08:00
|
|
|
// record or it has been deleted. we cannot do anything here until
|
|
|
|
// we implement some more intelligent locking mechanisms
|
2015-09-13 17:34:17 +08:00
|
|
|
if(record.tx.max())
|
|
|
|
throw MvccError("can't serialize due to concurrent operation(s)");
|
|
|
|
|
2015-10-08 06:58:29 +08:00
|
|
|
assert(atom.latest_visible(t) == &record);
|
|
|
|
assert(atom.latest_visible(t) == record.latest_visible(t));
|
|
|
|
|
2015-09-13 17:34:17 +08:00
|
|
|
// make a new version
|
|
|
|
auto updated = new T();
|
|
|
|
*updated = *record;
|
|
|
|
|
|
|
|
// mark the new version as created
|
|
|
|
updated->mark_created(t);
|
|
|
|
|
|
|
|
// mark the current version as deleted
|
|
|
|
record.mark_deleted(t);
|
|
|
|
record.newer(updated);
|
|
|
|
|
|
|
|
return updated;
|
|
|
|
}
|
|
|
|
|
2015-10-08 06:58:29 +08:00
|
|
|
void remove(Atom<T>& atom, T& record, const tx::Transaction& t)
|
2015-09-13 17:34:17 +08:00
|
|
|
{
|
|
|
|
auto guard = atom.acquire();
|
|
|
|
|
|
|
|
// if xmax is not zero, that means there is a newer version of this
|
|
|
|
// record or it has been deleted. we cannot do anything here
|
|
|
|
if(record.tx.max())
|
|
|
|
throw MvccError("can't serialize due to concurrent operation(s)");
|
|
|
|
|
|
|
|
record.mark_deleted(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
AtomicCounter<uint64_t> counter;
|
|
|
|
lockfree::List<Atom<T>> data;
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|