2015-12-08 04:51:55 +08:00
|
|
|
#pragma once
|
2015-10-08 06:58:29 +08:00
|
|
|
|
|
|
|
#include <atomic>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "transactions/commit_log.hpp"
|
|
|
|
|
|
|
|
namespace mvcc
|
|
|
|
{
|
|
|
|
|
2015-12-06 23:42:47 +08:00
|
|
|
// known committed and known aborted for both cre and exp
|
2015-10-08 06:58:29 +08:00
|
|
|
// this hints are used to quickly check the commit/abort status of the
|
|
|
|
// transaction that created this record. if these are not set, one should
|
|
|
|
// consult the commit log to find the status and update the status here
|
|
|
|
// more info https://wiki.postgresql.org/wiki/Hint_Bits
|
|
|
|
class Hints
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
union HintBits;
|
|
|
|
|
|
|
|
private:
|
2015-12-06 23:42:47 +08:00
|
|
|
enum Flags : uint8_t {
|
|
|
|
CRE_CMT = 0x01, // __01
|
|
|
|
CRE_ABT = 0x02, // __10
|
|
|
|
EXP_CMT = 0x04, // 01__
|
|
|
|
EXP_ABT = 0x08 // 10__
|
2015-10-08 06:58:29 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
template <Flags COMMITTED, Flags ABORTED>
|
|
|
|
class TxHints
|
|
|
|
{
|
|
|
|
using type = TxHints<COMMITTED, ABORTED>;
|
|
|
|
|
|
|
|
public:
|
2015-12-06 23:42:47 +08:00
|
|
|
TxHints(std::atomic<uint8_t>& bits) : bits(bits) {}
|
2015-10-08 06:58:29 +08:00
|
|
|
|
|
|
|
struct Value
|
|
|
|
{
|
|
|
|
bool is_committed() const
|
|
|
|
{
|
|
|
|
return bits & COMMITTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool is_aborted() const
|
|
|
|
{
|
|
|
|
return bits & ABORTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool is_unknown() const
|
|
|
|
{
|
|
|
|
return !(is_committed() || is_aborted());
|
|
|
|
}
|
2015-12-06 23:42:47 +08:00
|
|
|
|
2015-10-08 06:58:29 +08:00
|
|
|
uint8_t bits;
|
|
|
|
};
|
|
|
|
|
2015-12-06 23:42:47 +08:00
|
|
|
Value load(std::memory_order order = std::memory_order_acquire)
|
2015-10-08 06:58:29 +08:00
|
|
|
{
|
|
|
|
return Value { bits.load(order) };
|
|
|
|
}
|
|
|
|
|
2015-12-06 23:42:47 +08:00
|
|
|
void set_committed(std::memory_order order = std::memory_order_release)
|
2015-10-08 06:58:29 +08:00
|
|
|
{
|
2015-12-06 23:42:47 +08:00
|
|
|
bits.fetch_or(COMMITTED, order);
|
2015-10-08 06:58:29 +08:00
|
|
|
}
|
|
|
|
|
2015-12-06 23:42:47 +08:00
|
|
|
void set_aborted(std::memory_order order = std::memory_order_release)
|
2015-10-08 06:58:29 +08:00
|
|
|
{
|
2015-12-06 23:42:47 +08:00
|
|
|
bits.fetch_or(ABORTED, order);
|
2015-10-08 06:58:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::atomic<uint8_t>& bits;
|
|
|
|
};
|
|
|
|
|
2015-12-06 23:42:47 +08:00
|
|
|
struct Cre : public TxHints<CRE_CMT, CRE_ABT>
|
2015-10-08 06:58:29 +08:00
|
|
|
{
|
|
|
|
using TxHints::TxHints;
|
|
|
|
};
|
|
|
|
|
2015-12-06 23:42:47 +08:00
|
|
|
struct Exp : public TxHints<EXP_CMT, EXP_ABT>
|
2015-10-08 06:58:29 +08:00
|
|
|
{
|
|
|
|
using TxHints::TxHints;
|
|
|
|
};
|
|
|
|
|
|
|
|
public:
|
2015-12-06 23:42:47 +08:00
|
|
|
Hints() : cre(bits), exp(bits)
|
2015-10-08 06:58:29 +08:00
|
|
|
{
|
|
|
|
assert(bits.is_lock_free());
|
|
|
|
}
|
|
|
|
|
|
|
|
union HintBits
|
|
|
|
{
|
|
|
|
uint8_t bits;
|
|
|
|
|
2015-12-06 23:42:47 +08:00
|
|
|
Cre::Value cre;
|
|
|
|
Exp::Value exp;
|
2015-10-08 06:58:29 +08:00
|
|
|
};
|
|
|
|
|
2015-12-06 23:42:47 +08:00
|
|
|
HintBits load(std::memory_order order = std::memory_order_acquire)
|
2015-10-08 06:58:29 +08:00
|
|
|
{
|
|
|
|
return HintBits { bits.load(order) };
|
|
|
|
}
|
|
|
|
|
2015-12-06 23:42:47 +08:00
|
|
|
Cre cre;
|
|
|
|
Exp exp;
|
2015-10-08 06:58:29 +08:00
|
|
|
|
|
|
|
std::atomic<uint8_t> bits { 0 };
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|