2015-11-22 02:16:19 +08:00
|
|
|
#pragma once
|
2015-07-04 17:51:33 +08:00
|
|
|
|
|
|
|
#include <atomic>
|
|
|
|
#include <vector>
|
|
|
|
|
2016-08-10 16:39:02 +08:00
|
|
|
#include "threading/sync/lockable.hpp"
|
|
|
|
#include "threading/sync/spinlock.hpp"
|
|
|
|
#include "transactions/commit_log.hpp"
|
2016-07-05 11:01:22 +08:00
|
|
|
#include "transactions/transaction.hpp"
|
|
|
|
#include "transactions/transaction_store.hpp"
|
2015-07-31 18:36:41 +08:00
|
|
|
#include "utils/counters/simple_counter.hpp"
|
2015-07-04 17:51:33 +08:00
|
|
|
|
2015-10-04 15:47:15 +08:00
|
|
|
namespace tx
|
|
|
|
{
|
|
|
|
|
|
|
|
class TransactionError : std::runtime_error
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
using std::runtime_error::runtime_error;
|
|
|
|
};
|
|
|
|
|
2015-10-12 02:59:27 +08:00
|
|
|
class Engine : Lockable<SpinLock>
|
2015-07-31 18:36:41 +08:00
|
|
|
{
|
2015-07-04 17:51:33 +08:00
|
|
|
public:
|
2015-10-12 02:59:27 +08:00
|
|
|
using sptr = std::shared_ptr<Engine>;
|
|
|
|
|
2015-12-06 23:42:47 +08:00
|
|
|
Engine() : counter(1) {}
|
2015-11-23 04:35:40 +08:00
|
|
|
|
2016-08-25 22:29:45 +08:00
|
|
|
// Begins transaction and runs given functions in same atomic step.
|
|
|
|
// Functions will be given Transaction&
|
|
|
|
template <class... F>
|
|
|
|
Transaction &begin(F... fun)
|
2015-07-04 17:51:33 +08:00
|
|
|
{
|
2015-10-08 06:58:29 +08:00
|
|
|
auto guard = this->acquire_unique();
|
2015-07-04 17:51:33 +08:00
|
|
|
|
2015-11-22 02:16:19 +08:00
|
|
|
auto id = Id(counter.next());
|
2015-12-06 23:42:47 +08:00
|
|
|
auto t = new Transaction(id, active, *this);
|
2015-07-04 17:51:33 +08:00
|
|
|
|
2015-11-22 02:16:19 +08:00
|
|
|
active.insert(id);
|
2015-12-06 23:42:47 +08:00
|
|
|
store.put(id, t);
|
2015-10-04 15:47:15 +08:00
|
|
|
|
2016-08-25 22:29:45 +08:00
|
|
|
call(*t, fun...);
|
|
|
|
|
2015-10-04 15:47:15 +08:00
|
|
|
return *t;
|
|
|
|
}
|
2015-07-04 17:51:33 +08:00
|
|
|
|
2016-08-10 16:39:02 +08:00
|
|
|
Transaction &advance(const Id &id)
|
2015-10-04 15:47:15 +08:00
|
|
|
{
|
2015-10-08 06:58:29 +08:00
|
|
|
auto guard = this->acquire_unique();
|
2015-10-04 15:47:15 +08:00
|
|
|
|
2016-08-10 16:39:02 +08:00
|
|
|
auto *t = store.get(id);
|
2015-10-04 15:47:15 +08:00
|
|
|
|
2016-08-10 16:39:02 +08:00
|
|
|
if (t == nullptr) throw TransactionError("transaction does not exist");
|
2015-10-04 15:47:15 +08:00
|
|
|
|
|
|
|
// this is a new command
|
|
|
|
t->cid++;
|
|
|
|
|
|
|
|
return *t;
|
2015-07-04 17:51:33 +08:00
|
|
|
}
|
|
|
|
|
2016-08-25 22:29:45 +08:00
|
|
|
// Returns copy of current snapshot
|
|
|
|
Snapshot<Id> snapshot()
|
|
|
|
{
|
|
|
|
auto guard = this->acquire_unique();
|
|
|
|
|
|
|
|
return active;
|
|
|
|
}
|
|
|
|
|
2016-08-10 16:39:02 +08:00
|
|
|
void commit(const Transaction &t)
|
2015-07-04 17:51:33 +08:00
|
|
|
{
|
2015-10-08 06:58:29 +08:00
|
|
|
auto guard = this->acquire_unique();
|
2016-03-12 19:26:56 +08:00
|
|
|
clog.set_committed(t.id);
|
2015-07-04 17:51:33 +08:00
|
|
|
|
2015-07-31 18:36:41 +08:00
|
|
|
finalize(t);
|
2015-07-04 17:51:33 +08:00
|
|
|
}
|
|
|
|
|
2016-08-10 16:39:02 +08:00
|
|
|
void abort(const Transaction &t)
|
2015-07-04 17:51:33 +08:00
|
|
|
{
|
2015-10-08 06:58:29 +08:00
|
|
|
auto guard = this->acquire_unique();
|
2016-03-12 19:26:56 +08:00
|
|
|
clog.set_aborted(t.id);
|
2015-11-23 04:35:40 +08:00
|
|
|
|
2015-07-31 18:36:41 +08:00
|
|
|
finalize(t);
|
2015-07-04 17:51:33 +08:00
|
|
|
}
|
|
|
|
|
2015-11-22 02:16:19 +08:00
|
|
|
Id last_known_active()
|
2015-07-04 17:51:33 +08:00
|
|
|
{
|
2015-10-08 06:58:29 +08:00
|
|
|
auto guard = this->acquire_unique();
|
2015-07-31 18:36:41 +08:00
|
|
|
return active.front();
|
2015-07-04 17:51:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// total number of transactions started from the beginning of time
|
2015-07-31 18:36:41 +08:00
|
|
|
uint64_t count()
|
2015-07-04 17:51:33 +08:00
|
|
|
{
|
2015-10-08 06:58:29 +08:00
|
|
|
auto guard = this->acquire_unique();
|
2015-07-31 18:36:41 +08:00
|
|
|
return counter.count();
|
2015-07-04 17:51:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// the number of currently active transactions
|
|
|
|
size_t size()
|
|
|
|
{
|
2015-10-08 06:58:29 +08:00
|
|
|
auto guard = this->acquire_unique();
|
2015-07-04 17:51:33 +08:00
|
|
|
return active.size();
|
|
|
|
}
|
|
|
|
|
2016-03-12 19:26:56 +08:00
|
|
|
CommitLog clog;
|
|
|
|
|
2015-07-04 17:51:33 +08:00
|
|
|
private:
|
2016-08-25 22:29:45 +08:00
|
|
|
template <class T, class... F>
|
|
|
|
void call(Transaction &t, T fun, F... funs)
|
|
|
|
{
|
|
|
|
call(t, fun);
|
|
|
|
call(t, funs...);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
void call(Transaction &t, T fun)
|
|
|
|
{
|
|
|
|
fun(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
void call(Transaction &t) {}
|
|
|
|
|
2016-08-10 16:39:02 +08:00
|
|
|
void finalize(const Transaction &t)
|
2015-07-04 17:51:33 +08:00
|
|
|
{
|
2015-11-22 02:16:19 +08:00
|
|
|
active.remove(t.id);
|
2015-10-04 15:47:15 +08:00
|
|
|
|
2015-12-06 23:42:47 +08:00
|
|
|
// remove transaction from store
|
|
|
|
store.del(t.id);
|
2015-07-04 17:51:33 +08:00
|
|
|
}
|
|
|
|
|
2015-07-31 18:36:41 +08:00
|
|
|
SimpleCounter<uint64_t> counter;
|
2015-11-22 02:16:19 +08:00
|
|
|
Snapshot<Id> active;
|
2015-12-06 23:42:47 +08:00
|
|
|
TransactionStore<uint64_t> store;
|
2015-07-04 17:51:33 +08:00
|
|
|
};
|
2015-10-04 15:47:15 +08:00
|
|
|
}
|