foundationdb/fdbserver/SkipList.cpp

1179 lines
34 KiB
C++
Raw Normal View History

2017-05-26 04:48:44 +08:00
/*
* SkipList.cpp
*
* This source file is part of the FoundationDB open source project
*
* Copyright 2013-2018 Apple Inc. and the FoundationDB project authors
*
2017-05-26 04:48:44 +08:00
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
2017-05-26 04:48:44 +08:00
* http://www.apache.org/licenses/LICENSE-2.0
*
2017-05-26 04:48:44 +08:00
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <stdint.h>
#include <memory.h>
#include <stdio.h>
#include <algorithm>
#include <numeric>
#include <string>
#include <vector>
2017-05-26 04:48:44 +08:00
#include "flow/Platform.h"
#include "fdbrpc/fdbrpc.h"
#include "fdbrpc/PerfMetric.h"
#include "fdbclient/FDBTypes.h"
#include "fdbclient/KeyRangeMap.h"
#include "fdbclient/SystemData.h"
#include "fdbserver/ConflictSet.h"
2017-05-26 04:48:44 +08:00
using std::max;
2020-01-31 06:24:37 +08:00
using std::min;
2017-05-26 04:48:44 +08:00
static std::vector<PerfDoubleCounter*> skc;
2017-05-26 04:48:44 +08:00
static thread_local uint32_t g_seed = 0;
static inline int skfastrand() {
g_seed = g_seed * 1664525L + 1013904223L;
return g_seed;
}
PerfDoubleCounter g_buildTest("Build", skc), g_add("Add", skc), g_detectConflicts("Detect", skc), g_sort("D.Sort", skc),
g_combine("D.Combine", skc), g_checkRead("D.CheckRead", skc), g_checkBatch("D.CheckIntraBatch", skc),
g_merge("D.MergeWrite", skc), g_removeBefore("D.RemoveBefore", skc);
2020-01-31 06:24:37 +08:00
static force_inline int compare(const StringRef& a, const StringRef& b) {
int c = memcmp(a.begin(), b.begin(), min(a.size(), b.size()));
if (c < 0)
return -1;
if (c > 0)
return +1;
if (a.size() < b.size())
return -1;
if (a.size() == b.size())
return 0;
2017-05-26 04:48:44 +08:00
return +1;
}
struct ReadConflictRange {
StringRef begin, end;
Version version;
int transaction;
int indexInTx;
VectorRef<int>* conflictingKeyRange;
Arena* cKRArena;
ReadConflictRange(StringRef begin,
StringRef end,
Version version,
int transaction,
int indexInTx,
VectorRef<int>* cKR = nullptr,
Arena* cKRArena = nullptr)
: begin(begin), end(end), version(version), transaction(transaction), indexInTx(indexInTx),
conflictingKeyRange(cKR), cKRArena(cKRArena) {}
2020-03-05 03:44:14 +08:00
bool operator<(const ReadConflictRange& rhs) const { return compare(begin, rhs.begin) < 0; }
2017-05-26 04:48:44 +08:00
};
struct KeyInfo {
StringRef key;
int* pIndex;
bool begin;
bool write;
int transaction;
KeyInfo() = default;
2020-02-06 00:08:41 +08:00
KeyInfo(StringRef key, bool begin, bool write, int transaction, int* pIndex)
: key(key), begin(begin), write(write), transaction(transaction), pIndex(pIndex) {}
2017-05-26 04:48:44 +08:00
};
2020-02-11 01:27:32 +08:00
force_inline int extra_ordering(const KeyInfo& ki) {
return ki.begin * 2 + (ki.write ^ ki.begin);
}
2017-05-26 04:48:44 +08:00
// returns true if done with string
2020-01-31 06:24:37 +08:00
force_inline bool getCharacter(const KeyInfo& ki, int character, int& outputCharacter) {
2017-05-26 04:48:44 +08:00
// normal case
2020-01-31 06:24:37 +08:00
if (character < ki.key.size()) {
2017-05-26 04:48:44 +08:00
outputCharacter = 5 + ki.key.begin()[character];
return false;
}
// termination
2020-01-31 06:24:37 +08:00
if (character == ki.key.size()) {
2017-05-26 04:48:44 +08:00
outputCharacter = 0;
2020-01-31 06:24:37 +08:00
return false;
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
if (character == ki.key.size() + 1) {
2017-05-26 04:48:44 +08:00
// end/begin+read/write relative sorting
2020-02-11 01:27:32 +08:00
outputCharacter = extra_ordering(ki);
2017-05-26 04:48:44 +08:00
return false;
}
outputCharacter = 0;
return true;
}
2020-01-31 06:24:37 +08:00
bool operator<(const KeyInfo& lhs, const KeyInfo& rhs) {
2017-05-26 04:48:44 +08:00
int i = min(lhs.key.size(), rhs.key.size());
2020-01-31 06:24:37 +08:00
int c = memcmp(lhs.key.begin(), rhs.key.begin(), i);
if (c != 0)
return c < 0;
2017-05-26 04:48:44 +08:00
2020-02-08 03:49:52 +08:00
// Always sort shorter keys before longer keys.
2020-02-11 01:27:32 +08:00
if (lhs.key.size() < rhs.key.size()) {
return true;
2017-05-26 04:48:44 +08:00
}
2020-02-11 01:27:32 +08:00
if (lhs.key.size() > rhs.key.size()) {
return false;
}
// When the keys are the same length, use the extra ordering constraint.
return extra_ordering(lhs) < extra_ordering(rhs);
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
bool operator==(const KeyInfo& lhs, const KeyInfo& rhs) {
return !(lhs < rhs || rhs < lhs);
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
void swapSort(std::vector<KeyInfo>& points, int a, int b) {
if (points[b] < points[a]) {
2017-05-26 04:48:44 +08:00
KeyInfo temp;
temp = points[a];
points[a] = points[b];
points[b] = temp;
}
}
2020-01-31 06:24:37 +08:00
void smallSort(std::vector<KeyInfo>& points, int start, int N) {
for (int i = 1; i < N; i++)
for (int j = i; j > 0; j -= 2)
swapSort(points, start + j - 1, start + j);
2020-01-31 06:24:37 +08:00
for (int i = N - 2; i > 0; i--)
for (int j = i; j > 0; j -= 2)
swapSort(points, start + j - 1, start + j);
2017-05-26 04:48:44 +08:00
}
struct SortTask {
int begin;
int size;
int character;
SortTask(int begin, int size, int character) : begin(begin), size(size), character(character) {}
};
2020-01-31 06:24:37 +08:00
void sortPoints(std::vector<KeyInfo>& points) {
std::vector<SortTask> tasks;
std::vector<KeyInfo> newPoints;
std::vector<int> counts;
2017-05-26 04:48:44 +08:00
tasks.emplace_back(0, points.size(), 0);
2017-05-26 04:48:44 +08:00
2020-01-31 06:24:37 +08:00
while (tasks.size()) {
2017-05-26 04:48:44 +08:00
SortTask st = tasks.back();
tasks.pop_back();
2020-01-31 06:24:37 +08:00
if (st.size < 10) {
// smallSort(points, st.begin, st.size);
std::sort(points.begin() + st.begin, points.begin() + st.begin + st.size);
2017-05-26 04:48:44 +08:00
continue;
}
newPoints.resize(st.size);
2020-01-31 06:24:37 +08:00
counts.assign(256 + 5, 0);
2017-05-26 04:48:44 +08:00
// get counts
int c;
bool allDone = true;
2020-01-31 06:24:37 +08:00
for (int i = st.begin; i < st.begin + st.size; i++) {
2017-05-26 04:48:44 +08:00
allDone &= getCharacter(points[i], st.character, c);
counts[c]++;
}
if (allDone)
continue;
2017-05-26 04:48:44 +08:00
// calculate offsets from counts and build next level of tasks
2020-01-31 06:24:37 +08:00
int total = 0;
for (int i = 0; i < counts.size(); i++) {
2017-05-26 04:48:44 +08:00
int temp = counts[i];
if (temp > 1)
tasks.emplace_back(st.begin + total, temp, st.character + 1);
2017-05-26 04:48:44 +08:00
counts[i] = total;
total += temp;
}
// put in their places
2020-01-31 06:24:37 +08:00
for (int i = st.begin; i < st.begin + st.size; i++) {
2017-05-26 04:48:44 +08:00
getCharacter(points[i], st.character, c);
newPoints[counts[c]++] = points[i];
}
2020-01-31 06:24:37 +08:00
// copy back into original points array
for (int i = 0; i < st.size; i++)
points[st.begin + i] = newPoints[i];
2017-05-26 04:48:44 +08:00
}
}
2020-01-31 06:24:37 +08:00
class SkipList : NonCopyable {
2017-05-26 04:48:44 +08:00
private:
2020-12-27 09:31:45 +08:00
static constexpr int MaxLevels = 26;
2017-05-26 04:48:44 +08:00
2020-12-27 09:31:45 +08:00
int randomLevel() const {
2020-01-31 06:24:37 +08:00
uint32_t i = uint32_t(skfastrand()) >> (32 - (MaxLevels - 1));
2017-05-26 04:48:44 +08:00
int level = 0;
2020-01-31 06:24:37 +08:00
while (i & 1) {
i >>= 1;
2017-05-26 04:48:44 +08:00
level++;
}
2020-01-31 06:24:37 +08:00
ASSERT(level < MaxLevels);
2017-05-26 04:48:44 +08:00
return level;
}
// Represent a node in the SkipList. The node has multiple (i.e., level) pointers to
// other nodes, and keeps a record of the max versions for each level.
2017-05-26 04:48:44 +08:00
struct Node {
2020-12-27 09:31:45 +08:00
int level() const { return nPointers - 1; }
2020-01-31 06:24:37 +08:00
uint8_t* value() { return end() + nPointers * (sizeof(Node*) + sizeof(Version)); }
2020-12-27 09:31:45 +08:00
int length() const { return valueLength; }
2017-05-26 04:48:44 +08:00
// Returns the next node pointer at the given level.
Node* getNext(int level) { return *((Node**)end() + level); }
// Sets the next node pointer at the given level.
void setNext(int level, Node* n) { *((Node**)end() + level) = n; }
// Returns the max version at the given level.
2020-12-27 09:31:45 +08:00
Version getMaxVersion(int i) const { return ((Version*)(end() + nPointers * sizeof(Node*)))[i]; }
// Sets the max version at the given level.
2020-01-31 06:24:37 +08:00
void setMaxVersion(int i, Version v) { ((Version*)(end() + nPointers * sizeof(Node*)))[i] = v; }
2017-05-26 04:48:44 +08:00
// Return a node with initialized value but uninitialized pointers
// Memory layout: *this, (level+1) Node*, (level+1) Version, value
2020-01-31 06:24:37 +08:00
static Node* create(const StringRef& value, int level) {
int nodeSize = sizeof(Node) + value.size() + (level + 1) * (sizeof(Node*) + sizeof(Version));
2017-05-26 04:48:44 +08:00
Node* n;
if (nodeSize <= 64) {
n = (Node*)FastAllocator<64>::allocate();
INSTRUMENT_ALLOCATE("SkipListNode64");
} else if (nodeSize <= 128) {
n = (Node*)FastAllocator<128>::allocate();
INSTRUMENT_ALLOCATE("SkipListNode128");
} else {
2020-01-31 06:24:37 +08:00
n = (Node*)new char[nodeSize];
2017-05-26 04:48:44 +08:00
INSTRUMENT_ALLOCATE("SkipListNodeLarge");
}
2020-01-31 06:24:37 +08:00
n->nPointers = level + 1;
2017-05-26 04:48:44 +08:00
n->valueLength = value.size();
if (value.size() > 0) {
memcpy(n->value(), value.begin(), value.size());
}
2017-05-26 04:48:44 +08:00
return n;
}
// pre: level>0, all lower level nodes between this and getNext(level) have correct maxversions
2020-01-31 06:24:37 +08:00
void calcVersionForLevel(int level) {
Node* end = getNext(level);
Version v = getMaxVersion(level - 1);
for (Node* x = getNext(level - 1); x != end; x = x->getNext(level - 1))
v = max(v, x->getMaxVersion(level - 1));
2017-05-26 04:48:44 +08:00
setMaxVersion(level, v);
}
void destroy() {
int nodeSize = getNodeSize();
if (nodeSize <= 64) {
FastAllocator<64>::release(this);
INSTRUMENT_RELEASE("SkipListNode64");
} else if (nodeSize <= 128) {
FastAllocator<128>::release(this);
INSTRUMENT_RELEASE("SkipListNode128");
} else {
2020-01-31 06:24:37 +08:00
delete[](char*) this;
2017-05-26 04:48:44 +08:00
INSTRUMENT_RELEASE("SkipListNodeLarge");
}
}
2020-01-31 06:24:37 +08:00
2017-05-26 04:48:44 +08:00
private:
2020-12-27 09:31:45 +08:00
int getNodeSize() const { return sizeof(Node) + valueLength + nPointers * (sizeof(Node*) + sizeof(Version)); }
// Returns the first Node* pointer
2020-01-31 06:24:37 +08:00
uint8_t* end() { return (uint8_t*)(this + 1); }
2020-12-27 09:31:45 +08:00
uint8_t const* end() const { return (uint8_t const*)(this + 1); }
2020-01-31 06:24:37 +08:00
int nPointers, valueLength;
2017-05-26 04:48:44 +08:00
};
static force_inline bool less(const uint8_t* a, int aLen, const uint8_t* b, int bLen) {
int c = memcmp(a, b, min(aLen, bLen));
if (c < 0)
return true;
if (c > 0)
return false;
2017-05-26 04:48:44 +08:00
return aLen < bLen;
}
2020-01-31 06:24:37 +08:00
Node* header;
2017-05-26 04:48:44 +08:00
void destroy() {
Node *next, *x;
2020-01-31 06:24:37 +08:00
for (x = header; x; x = next) {
2017-05-26 04:48:44 +08:00
next = x->getNext(0);
x->destroy();
}
}
2020-01-31 06:24:37 +08:00
2017-05-26 04:48:44 +08:00
public:
// Points the location (i.e., Node*) that value would appear in the SkipList.
// If the "value" is in the list, then finger[0] points to that exact node;
// otherwise, the finger points to Nodes that the value should be inserted before.
// Note the SkipList organizes all nodes at level 0, higher levels contain jump pointers.
2020-01-31 06:24:37 +08:00
struct Finger {
Node* finger[MaxLevels]; // valid for levels >= level
int level = MaxLevels;
Node* x = nullptr;
Node* alreadyChecked = nullptr;
2017-05-26 04:48:44 +08:00
StringRef value;
Finger() = default;
Finger(Node* header, const StringRef& ptr) : value(ptr), x(header) {}
2017-05-26 04:48:44 +08:00
2020-01-31 06:24:37 +08:00
void init(const StringRef& value, Node* header) {
2017-05-26 04:48:44 +08:00
this->value = value;
x = header;
2020-08-19 05:18:50 +08:00
alreadyChecked = nullptr;
2017-05-26 04:48:44 +08:00
level = MaxLevels;
}
// pre: !finished()
force_inline void prefetch() {
2020-01-31 06:24:37 +08:00
Node* next = x->getNext(level - 1);
_mm_prefetch((const char*)next, _MM_HINT_T0);
_mm_prefetch((const char*)next + 64, _MM_HINT_T0);
2017-05-26 04:48:44 +08:00
}
// pre: !finished()
// Advances the pointer at the current level to a Node that's >= finger's value
// if possible; or move to the next level (i.e., level--).
2017-05-26 04:48:44 +08:00
// Returns true if we have advanced to the next level
force_inline bool advance() {
2020-01-31 06:24:37 +08:00
Node* next = x->getNext(level - 1);
2017-05-26 04:48:44 +08:00
if (next == alreadyChecked || !less(next->value(), next->length(), value.begin(), value.size())) {
alreadyChecked = next;
level--;
finger[level] = x;
return true;
} else {
x = next;
return false;
}
}
// pre: !finished()
force_inline void nextLevel() {
2020-01-31 06:24:37 +08:00
while (!advance())
;
2017-05-26 04:48:44 +08:00
}
2020-12-27 09:31:45 +08:00
force_inline bool finished() const { return level == 0; }
2017-05-26 04:48:44 +08:00
// Returns if the finger value is found in the SkipList.
2017-05-26 04:48:44 +08:00
force_inline Node* found() const {
// valid after finished returns true
2020-01-31 06:24:37 +08:00
Node* n = finger[0]->getNext(0); // or alreadyChecked, but that is more easily invalidated
2017-05-26 04:48:44 +08:00
if (n && n->length() == value.size() && !memcmp(n->value(), value.begin(), value.size()))
return n;
2020-01-31 06:24:37 +08:00
else
2020-08-19 05:18:50 +08:00
return nullptr;
2017-05-26 04:48:44 +08:00
}
StringRef getValue() const {
Node* n = finger[0]->getNext(0);
2020-01-31 06:24:37 +08:00
return n ? StringRef(n->value(), n->length()) : StringRef();
2017-05-26 04:48:44 +08:00
}
};
// Returns the total number of nodes in the list.
int count() const {
2017-05-26 04:48:44 +08:00
int count = 0;
Node* x = header->getNext(0);
2020-01-31 06:24:37 +08:00
while (x) {
x = x->getNext(0);
count++;
2017-05-26 04:48:44 +08:00
}
return count;
}
2020-01-31 06:24:37 +08:00
explicit SkipList(Version version = 0) {
header = Node::create(StringRef(), MaxLevels - 1);
for (int l = 0; l < MaxLevels; l++) {
2020-08-19 05:18:50 +08:00
header->setNext(l, nullptr);
2017-05-26 04:48:44 +08:00
header->setMaxVersion(l, version);
}
}
2020-01-31 06:24:37 +08:00
~SkipList() { destroy(); }
2020-08-19 05:18:50 +08:00
SkipList(SkipList&& other) noexcept : header(other.header) { other.header = nullptr; }
2020-06-10 08:33:41 +08:00
void operator=(SkipList&& other) noexcept {
2017-05-26 04:48:44 +08:00
destroy();
header = other.header;
2020-08-19 05:18:50 +08:00
other.header = nullptr;
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
void swap(SkipList& other) { std::swap(header, other.header); }
2017-05-26 04:48:44 +08:00
2020-01-31 06:24:37 +08:00
void addConflictRanges(const Finger* fingers, int rangeCount, Version version) {
for (int r = rangeCount - 1; r >= 0; r--) {
const Finger& startF = fingers[r * 2];
const Finger& endF = fingers[r * 2 + 1];
2017-05-26 04:48:44 +08:00
if (endF.found() == nullptr)
insert(endF, endF.finger[0]->getMaxVersion(0));
2017-05-26 04:48:44 +08:00
2020-01-31 06:24:37 +08:00
remove(startF, endF);
insert(startF, version);
2017-05-26 04:48:44 +08:00
}
}
2020-01-31 06:24:37 +08:00
void detectConflicts(ReadConflictRange* ranges, int count, bool* transactionConflictStatus) {
2017-05-26 04:48:44 +08:00
const int M = 16;
int nextJob[M];
2020-01-31 06:24:37 +08:00
CheckMax inProgress[M];
if (!count)
return;
2017-05-26 04:48:44 +08:00
2020-03-05 03:44:14 +08:00
int started = min(M, count);
for (int i = 0; i < started; i++) {
inProgress[i].init(ranges[i],
header,
transactionConflictStatus,
ranges[i].indexInTx,
ranges[i].conflictingKeyRange,
ranges[i].cKRArena);
2020-03-05 03:44:14 +08:00
nextJob[i] = i + 1;
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
nextJob[started - 1] = 0;
2017-05-26 04:48:44 +08:00
2020-01-31 06:24:37 +08:00
int prevJob = started - 1;
2017-05-26 04:48:44 +08:00
int job = 0;
// vtune: 340 parts
while (true) {
if (inProgress[job].advance()) {
2020-01-31 06:24:37 +08:00
if (started == count) {
if (prevJob == job)
break;
2017-05-26 04:48:44 +08:00
nextJob[prevJob] = nextJob[job];
job = prevJob;
} else {
int temp = started++;
inProgress[job].init(ranges[temp],
header,
transactionConflictStatus,
ranges[temp].indexInTx,
ranges[temp].conflictingKeyRange,
ranges[temp].cKRArena);
}
2017-05-26 04:48:44 +08:00
}
prevJob = job;
job = nextJob[job];
}
}
// Splits the version history represented by this skiplist into separate key ranges
// delimited by the given array of keys. This SkipList is left empty. this->partition
// is intended to be followed by a call to this->concatenate() recombining the same
// partitions. In between, operations on each partition must not touch any keys outside
// the partition. Specifically, the partition to the left of 'key' must not have a range
// [...,key) inserted, since that would insert an entry at 'key'.
// Note this function is not used.
2020-01-31 06:24:37 +08:00
void partition(StringRef* begin, int splitCount, SkipList* output) {
for (int i = splitCount - 1; i >= 0; i--) {
Finger f(header, begin[i]);
while (!f.finished())
f.nextLevel();
2020-01-31 06:24:37 +08:00
split(f, output[i + 1]);
2017-05-26 04:48:44 +08:00
}
swap(output[0]);
}
// Concatenates multiple SkipList objects into one and stores in input[0].
// Note this function is not used.
2020-01-31 06:24:37 +08:00
void concatenate(SkipList* input, int count) {
std::vector<Finger> ends(count - 1);
for (int i = 0; i < ends.size(); i++)
input[i].getEnd(ends[i]);
2020-01-31 06:24:37 +08:00
for (int l = 0; l < MaxLevels; l++) {
for (int i = ends.size() - 1; i >= 0; i--) {
ends[i].finger[l]->setNext(l, input[i + 1].header->getNext(l));
if (l && (!i || ends[i].finger[l] != input[i].header))
ends[i].finger[l]->calcVersionForLevel(l);
2020-08-19 05:18:50 +08:00
input[i + 1].header->setNext(l, nullptr);
2017-05-26 04:48:44 +08:00
}
}
swap(input[0]);
}
2020-01-31 06:24:37 +08:00
void find(const StringRef* values, Finger* results, int* temp, int count) {
// Relying on the ordering of values, descend until the values aren't all in the
2017-05-26 04:48:44 +08:00
// same part of the tree
// vtune: 11 parts
2020-01-31 06:24:37 +08:00
results[0].init(values[0], header);
const StringRef& endValue = values[count - 1];
while (results[0].level > 1) {
2017-05-26 04:48:44 +08:00
results[0].nextLevel();
Node* ac = results[0].alreadyChecked;
if (ac && less(ac->value(), ac->length(), endValue.begin(), endValue.size()))
break;
2017-05-26 04:48:44 +08:00
}
// Init all the other fingers to start descending where we stopped
// the first one
// SOMEDAY: this loop showed up on vtune, could be faster?
// vtune: 8 parts
2020-01-31 06:24:37 +08:00
int startLevel = results[0].level + 1;
Node* x = startLevel < MaxLevels ? results[0].finger[startLevel] : header;
for (int i = 1; i < count; i++) {
2017-05-26 04:48:44 +08:00
results[i].level = startLevel;
results[i].x = x;
2020-08-19 05:18:50 +08:00
results[i].alreadyChecked = nullptr;
2017-05-26 04:48:44 +08:00
results[i].value = values[i];
for (int j = startLevel; j < MaxLevels; j++)
results[i].finger[j] = results[0].finger[j];
2017-05-26 04:48:44 +08:00
}
int* nextJob = temp;
for (int i = 0; i < count - 1; i++)
nextJob[i] = i + 1;
2020-01-31 06:24:37 +08:00
nextJob[count - 1] = 0;
2017-05-26 04:48:44 +08:00
2020-01-31 06:24:37 +08:00
int prevJob = count - 1;
2017-05-26 04:48:44 +08:00
int job = 0;
// vtune: 225 parts
while (true) {
Finger* f = &results[job];
f->advance();
if (f->finished()) {
if (prevJob == job)
break;
2017-05-26 04:48:44 +08:00
nextJob[prevJob] = nextJob[job];
2020-01-31 06:24:37 +08:00
} else {
2017-05-26 04:48:44 +08:00
f->prefetch();
prevJob = job;
}
job = nextJob[job];
}
}
2020-01-31 06:24:37 +08:00
int removeBefore(Version v, Finger& f, int nodeCount) {
2017-05-26 04:48:44 +08:00
// f.x, f.alreadyChecked?
int removedCount = 0;
bool wasAbove = true;
while (nodeCount--) {
Node* x = f.finger[0]->getNext(0);
if (!x)
break;
2020-01-31 06:24:37 +08:00
2017-05-26 04:48:44 +08:00
// double prefetch gives +25% speed (single threaded)
Node* next = x->getNext(0);
2020-01-31 06:24:37 +08:00
_mm_prefetch((const char*)next, _MM_HINT_T0);
2017-05-26 04:48:44 +08:00
next = x->getNext(1);
2020-01-31 06:24:37 +08:00
_mm_prefetch((const char*)next, _MM_HINT_T0);
2017-05-26 04:48:44 +08:00
bool isAbove = x->getMaxVersion(0) >= v;
2020-01-31 06:24:37 +08:00
if (isAbove || wasAbove) { // f.nextItem
for (int l = 0; l <= x->level(); l++)
f.finger[l] = x;
2020-01-31 06:24:37 +08:00
} else { // f.eraseItem
2017-05-26 04:48:44 +08:00
removedCount++;
for (int l = 0; l <= x->level(); l++)
f.finger[l]->setNext(l, x->getNext(l));
2020-01-31 06:24:37 +08:00
for (int i = 1; i <= x->level(); i++)
f.finger[i]->setMaxVersion(i, max(f.finger[i]->getMaxVersion(i), x->getMaxVersion(i)));
2017-05-26 04:48:44 +08:00
x->destroy();
}
wasAbove = isAbove;
}
return removedCount;
}
private:
2020-01-31 06:24:37 +08:00
void remove(const Finger& start, const Finger& end) {
if (start.finger[0] == end.finger[0])
return;
2017-05-26 04:48:44 +08:00
2020-01-31 06:24:37 +08:00
Node* x = start.finger[0]->getNext(0);
2017-05-26 04:48:44 +08:00
// vtune says: this loop is the expensive parts (6 parts)
2020-01-31 06:24:37 +08:00
for (int i = 0; i < MaxLevels; i++)
if (start.finger[i] != end.finger[i])
start.finger[i]->setNext(i, end.finger[i]->getNext(i));
2017-05-26 04:48:44 +08:00
while (true) {
Node* next = x->getNext(0);
x->destroy();
if (x == end.finger[0])
break;
2017-05-26 04:48:44 +08:00
x = next;
}
}
2020-01-31 06:24:37 +08:00
void insert(const Finger& f, Version version) {
2017-05-26 04:48:44 +08:00
int level = randomLevel();
2020-01-31 06:24:37 +08:00
// cout << std::string((const char*)value,length) << " level: " << level << endl;
Node* x = Node::create(f.value, level);
2017-05-26 04:48:44 +08:00
x->setMaxVersion(0, version);
2020-01-31 06:24:37 +08:00
for (int i = 0; i <= level; i++) {
2017-05-26 04:48:44 +08:00
x->setNext(i, f.finger[i]->getNext(i));
f.finger[i]->setNext(i, x);
}
// vtune says: this loop is the costly part of this function
2020-01-31 06:24:37 +08:00
for (int i = 1; i <= level; i++) {
2017-05-26 04:48:44 +08:00
f.finger[i]->calcVersionForLevel(i);
x->calcVersionForLevel(i);
}
2020-01-31 06:24:37 +08:00
for (int i = level + 1; i < MaxLevels; i++) {
2017-05-26 04:48:44 +08:00
Version v = f.finger[i]->getMaxVersion(i);
if (v >= version)
break;
2017-05-26 04:48:44 +08:00
f.finger[i]->setMaxVersion(i, version);
}
}
2020-01-31 06:24:37 +08:00
void insert(const StringRef& value, Version version) {
2017-05-26 04:48:44 +08:00
Finger f(header, value);
while (!f.finished())
f.nextLevel();
2017-05-26 04:48:44 +08:00
// SOMEDAY: equality?
2020-01-31 06:24:37 +08:00
insert(f, version);
2017-05-26 04:48:44 +08:00
}
struct CheckMax {
Finger start, end;
Version version;
2020-01-31 06:24:37 +08:00
bool* result;
2017-05-26 04:48:44 +08:00
int state;
int indexInTx;
VectorRef<int>* conflictingKeyRange; // nullptr if report_conflicting_keys is not enabled.
Arena* cKRArena; // nullptr if report_conflicting_keys is not enabled.
2017-05-26 04:48:44 +08:00
void init(const ReadConflictRange& r,
Node* header,
bool* tCS,
int indexInTx,
VectorRef<int>* cKR,
Arena* cKRArena) {
2020-03-05 03:44:14 +08:00
this->start.init(r.begin, header);
this->end.init(r.end, header);
2017-05-26 04:48:44 +08:00
this->version = r.version;
this->indexInTx = indexInTx;
this->cKRArena = cKRArena;
2020-03-05 03:44:14 +08:00
result = &tCS[r.transaction];
conflictingKeyRange = cKR;
2017-05-26 04:48:44 +08:00
this->state = 0;
}
2020-12-27 09:31:45 +08:00
bool noConflict() const { return true; }
2020-03-05 03:44:14 +08:00
bool conflict() {
*result = true;
if (conflictingKeyRange != nullptr)
conflictingKeyRange->push_back(*cKRArena, indexInTx);
return true;
}
2017-05-26 04:48:44 +08:00
// Return true if finished
force_inline bool advance() {
switch (state) {
case 0:
// find where start and end fingers diverge
while (true) {
if (!start.advance()) {
start.prefetch();
return false;
}
end.x = start.x;
2020-01-31 06:24:37 +08:00
while (!end.advance())
;
2017-05-26 04:48:44 +08:00
int l = start.level;
if (start.finger[l] != end.finger[l])
break;
2017-05-26 04:48:44 +08:00
// accept if the range spans the check range, but does not have a greater version
if (start.finger[l]->getMaxVersion(l) <= version)
return noConflict();
if (l == 0)
return conflict();
2017-05-26 04:48:44 +08:00
}
state = 1;
2020-01-31 06:24:37 +08:00
case 1: {
// check the end side of the pyramid
Node* e = end.finger[end.level];
while (e->getMaxVersion(end.level) > version) {
if (end.finished())
return conflict();
2020-01-31 06:24:37 +08:00
end.nextLevel();
Node* f = end.finger[end.level];
while (e != f) {
if (e->getMaxVersion(end.level) > version)
return conflict();
2020-01-31 06:24:37 +08:00
e = e->getNext(end.level);
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
}
2017-05-26 04:48:44 +08:00
2020-01-31 06:24:37 +08:00
// check the start side of the pyramid
Node* s = end.finger[start.level];
while (true) {
Node* nextS = start.finger[start.level]->getNext(start.level);
Node* p = nextS;
while (p != s) {
if (p->getMaxVersion(start.level) > version)
return conflict();
2020-01-31 06:24:37 +08:00
p = p->getNext(start.level);
}
if (start.finger[start.level]->getMaxVersion(start.level) <= version)
return noConflict();
2020-01-31 06:24:37 +08:00
s = nextS;
if (start.finished()) {
if (nextS->length() == start.value.size() &&
!memcmp(nextS->value(), start.value.begin(), start.value.size()))
2017-05-26 04:48:44 +08:00
return noConflict();
2020-01-31 06:24:37 +08:00
else
return conflict();
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
start.nextLevel();
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
}
2017-05-26 04:48:44 +08:00
default:
__assume(false);
}
}
};
// Splits the SkipLists so that those after finger is moved to "right".
2020-01-31 06:24:37 +08:00
void split(const Finger& f, SkipList& right) {
ASSERT(!right.header->getNext(0)); // right must be empty
2017-05-26 04:48:44 +08:00
right.header->setMaxVersion(0, f.finger[0]->getMaxVersion(0));
2020-01-31 06:24:37 +08:00
for (int l = 0; l < MaxLevels; l++) {
2017-05-26 04:48:44 +08:00
right.header->setNext(l, f.finger[l]->getNext(l));
2020-08-19 05:18:50 +08:00
f.finger[l]->setNext(l, nullptr);
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
}
2017-05-26 04:48:44 +08:00
// Sets end's finger to the last nodes at all levels.
2020-01-31 06:24:37 +08:00
void getEnd(Finger& end) {
2017-05-26 04:48:44 +08:00
Node* node = header;
2020-01-31 06:24:37 +08:00
for (int l = MaxLevels - 1; l >= 0; l--) {
2017-05-26 04:48:44 +08:00
Node* next;
while ((next = node->getNext(l)) != nullptr)
node = next;
2017-05-26 04:48:44 +08:00
end.finger[l] = node;
}
end.level = 0;
}
};
struct ConflictSet {
2020-05-23 00:25:32 +08:00
ConflictSet() : oldestVersion(0), removalKey(makeString(0)) {}
2020-02-01 01:33:24 +08:00
~ConflictSet() {}
2017-05-26 04:48:44 +08:00
SkipList versionHistory;
Key removalKey;
Version oldestVersion;
};
2020-01-31 06:24:37 +08:00
ConflictSet* newConflictSet() {
return new ConflictSet;
}
void clearConflictSet(ConflictSet* cs, Version v) {
SkipList(v).swap(cs->versionHistory);
2017-05-26 04:48:44 +08:00
}
void destroyConflictSet(ConflictSet* cs) {
delete cs;
}
ConflictBatch::ConflictBatch(ConflictSet* cs,
std::map<int, VectorRef<int>>* conflictingKeyRangeMap,
Arena* resolveBatchReplyArena)
: cs(cs), transactionCount(0), conflictingKeyRangeMap(conflictingKeyRangeMap),
resolveBatchReplyArena(resolveBatchReplyArena) {}
2017-05-26 04:48:44 +08:00
2020-01-31 06:24:37 +08:00
ConflictBatch::~ConflictBatch() {}
2017-05-26 04:48:44 +08:00
struct TransactionInfo {
2020-01-31 06:24:37 +08:00
VectorRef<std::pair<int, int>> readRanges;
VectorRef<std::pair<int, int>> writeRanges;
2017-05-26 04:48:44 +08:00
bool tooOld;
bool reportConflictingKeys;
2017-05-26 04:48:44 +08:00
};
2020-01-31 06:24:37 +08:00
void ConflictBatch::addTransaction(const CommitTransactionRef& tr) {
const int t = transactionCount++;
2017-05-26 04:48:44 +08:00
Arena& arena = transactionInfo.arena();
TransactionInfo* info = new (arena) TransactionInfo;
info->reportConflictingKeys = tr.report_conflicting_keys;
2017-05-26 04:48:44 +08:00
if (tr.read_snapshot < cs->oldestVersion && tr.read_conflict_ranges.size()) {
info->tooOld = true;
} else {
info->tooOld = false;
2020-01-31 06:24:37 +08:00
info->readRanges.resize(arena, tr.read_conflict_ranges.size());
info->writeRanges.resize(arena, tr.write_conflict_ranges.size());
2017-05-26 04:48:44 +08:00
2020-01-31 06:24:37 +08:00
for (int r = 0; r < tr.read_conflict_ranges.size(); r++) {
2017-05-26 04:48:44 +08:00
const KeyRangeRef& range = tr.read_conflict_ranges[r];
2020-02-06 00:08:41 +08:00
points.emplace_back(range.begin, true, false, t, &info->readRanges[r].first);
points.emplace_back(range.end, false, false, t, &info->readRanges[r].second);
combinedReadConflictRanges.emplace_back(range.begin,
range.end,
tr.read_snapshot,
t,
r,
tr.report_conflicting_keys ? &(*conflictingKeyRangeMap)[t]
: nullptr,
tr.report_conflicting_keys ? resolveBatchReplyArena : nullptr);
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
for (int r = 0; r < tr.write_conflict_ranges.size(); r++) {
2017-05-26 04:48:44 +08:00
const KeyRangeRef& range = tr.write_conflict_ranges[r];
2020-02-06 00:08:41 +08:00
points.emplace_back(range.begin, true, true, t, &info->writeRanges[r].first);
points.emplace_back(range.end, false, true, t, &info->writeRanges[r].second);
2017-05-26 04:48:44 +08:00
}
}
transactionInfo.push_back(arena, info);
2017-05-26 04:48:44 +08:00
}
2020-02-05 00:19:50 +08:00
// SOMEDAY: This should probably be replaced with a roaring bitmap.
2017-05-26 04:48:44 +08:00
class MiniConflictSet : NonCopyable {
std::vector<bool> values;
2017-05-26 04:48:44 +08:00
public:
2020-02-05 00:19:50 +08:00
explicit MiniConflictSet(int size) { values.assign(size, false); }
2020-01-31 06:24:37 +08:00
void set(int begin, int end) {
for (int i = begin; i < end; i++)
values[i] = true;
2017-05-26 04:48:44 +08:00
}
bool any(int begin, int end) {
2020-01-31 06:24:37 +08:00
for (int i = begin; i < end; i++)
if (values[i])
return true;
2017-05-26 04:48:44 +08:00
return false;
}
};
void ConflictBatch::checkIntraBatchConflicts() {
int index = 0;
for (int p = 0; p < points.size(); p++)
*points[p].pIndex = index++;
2017-05-26 04:48:44 +08:00
2020-01-31 06:24:37 +08:00
MiniConflictSet mcs(index);
for (int t = 0; t < transactionInfo.size(); t++) {
2017-05-26 04:48:44 +08:00
const TransactionInfo& tr = *transactionInfo[t];
if (transactionConflictStatus[t])
continue;
2017-05-26 04:48:44 +08:00
bool conflict = tr.tooOld;
for (int i = 0; i < tr.readRanges.size(); i++) {
2020-03-05 03:44:14 +08:00
if (mcs.any(tr.readRanges[i].first, tr.readRanges[i].second)) {
if (tr.reportConflictingKeys) {
(*conflictingKeyRangeMap)[t].push_back(*resolveBatchReplyArena, i);
}
2017-05-26 04:48:44 +08:00
conflict = true;
break;
}
}
2017-05-26 04:48:44 +08:00
transactionConflictStatus[t] = conflict;
if (!conflict)
for (int i = 0; i < tr.writeRanges.size(); i++)
mcs.set(tr.writeRanges[i].first, tr.writeRanges[i].second);
2017-05-26 04:48:44 +08:00
}
}
void ConflictBatch::GetTooOldTransactions(std::vector<int>& tooOldTransactions) {
2020-01-31 06:24:37 +08:00
for (int i = 0; i < transactionInfo.size(); i++) {
2017-05-26 04:48:44 +08:00
if (transactionInfo[i]->tooOld) {
tooOldTransactions.push_back(i);
}
}
}
void ConflictBatch::detectConflicts(Version now,
Version newOldestVersion,
std::vector<int>& nonConflicting,
2020-01-31 06:24:37 +08:00
std::vector<int>* tooOldTransactions) {
2017-05-26 04:48:44 +08:00
double t = timer();
2020-01-31 06:24:37 +08:00
sortPoints(points);
g_sort += timer() - t;
2017-05-26 04:48:44 +08:00
2020-01-31 06:24:37 +08:00
transactionConflictStatus = new bool[transactionCount];
memset(transactionConflictStatus, 0, transactionCount * sizeof(bool));
2017-05-26 04:48:44 +08:00
t = timer();
checkReadConflictRanges();
2020-01-31 06:24:37 +08:00
g_checkRead += timer() - t;
2017-05-26 04:48:44 +08:00
t = timer();
checkIntraBatchConflicts();
2020-01-31 06:24:37 +08:00
g_checkBatch += timer() - t;
2017-05-26 04:48:44 +08:00
t = timer();
combineWriteConflictRanges();
2020-01-31 06:24:37 +08:00
g_combine += timer() - t;
2017-05-26 04:48:44 +08:00
t = timer();
mergeWriteConflictRanges(now);
2020-01-31 06:24:37 +08:00
g_merge += timer() - t;
2020-01-31 06:24:37 +08:00
for (int i = 0; i < transactionCount; i++) {
2020-03-06 02:49:21 +08:00
if (tooOldTransactions && transactionInfo[i]->tooOld) {
2017-05-26 04:48:44 +08:00
tooOldTransactions->push_back(i);
} else if (!transactionConflictStatus[i]) {
nonConflicting.push_back(i);
2020-03-06 02:49:21 +08:00
}
2017-05-26 04:48:44 +08:00
}
delete[] transactionConflictStatus;
t = timer();
if (newOldestVersion > cs->oldestVersion) {
cs->oldestVersion = newOldestVersion;
2020-01-31 06:24:37 +08:00
SkipList::Finger finger;
2017-05-26 04:48:44 +08:00
int temp;
2020-01-31 06:24:37 +08:00
cs->versionHistory.find(&cs->removalKey, &finger, &temp, 1);
cs->versionHistory.removeBefore(cs->oldestVersion, finger, combinedWriteConflictRanges.size() * 3 + 10);
2017-05-26 04:48:44 +08:00
cs->removalKey = finger.getValue();
}
2020-01-31 06:24:37 +08:00
g_removeBefore += timer() - t;
2017-05-26 04:48:44 +08:00
}
void ConflictBatch::checkReadConflictRanges() {
if (combinedReadConflictRanges.empty())
return;
2017-05-26 04:48:44 +08:00
cs->versionHistory.detectConflicts(
&combinedReadConflictRanges[0], combinedReadConflictRanges.size(), transactionConflictStatus);
2017-05-26 04:48:44 +08:00
}
void ConflictBatch::addConflictRanges(Version now,
std::vector<std::pair<StringRef, StringRef>>::iterator begin,
std::vector<std::pair<StringRef, StringRef>>::iterator end,
SkipList* part) {
const int count = end - begin;
static_assert(sizeof(*begin) == sizeof(StringRef) * 2,
2020-01-31 06:24:37 +08:00
"Write Conflict Range type not convertible to two StringPtrs");
const StringRef* strings = reinterpret_cast<const StringRef*>(&*begin);
const int stringCount = count * 2;
2017-05-26 04:48:44 +08:00
const int stripeSize = 16;
2020-01-31 06:24:37 +08:00
SkipList::Finger fingers[stripeSize];
int temp[stripeSize];
int stripes = (stringCount + stripeSize - 1) / stripeSize;
int ss = stringCount - (stripes - 1) * stripeSize;
for (int s = stripes - 1; s >= 0; s--) {
part->find(&strings[s * stripeSize], fingers, temp, ss);
part->addConflictRanges(fingers, ss / 2, now);
2017-05-26 04:48:44 +08:00
ss = stripeSize;
}
}
void ConflictBatch::mergeWriteConflictRanges(Version now) {
if (combinedWriteConflictRanges.empty())
return;
2017-05-26 04:48:44 +08:00
2020-02-01 01:57:43 +08:00
addConflictRanges(now, combinedWriteConflictRanges.begin(), combinedWriteConflictRanges.end(), &cs->versionHistory);
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
void ConflictBatch::combineWriteConflictRanges() {
2017-05-26 04:48:44 +08:00
int activeWriteCount = 0;
for (const KeyInfo& point : points) {
2020-01-31 06:24:37 +08:00
if (point.write && !transactionConflictStatus[point.transaction]) {
2017-05-26 04:48:44 +08:00
if (point.begin) {
2020-01-31 06:24:37 +08:00
activeWriteCount++;
if (activeWriteCount == 1)
combinedWriteConflictRanges.emplace_back(point.key, KeyRef());
2017-05-26 04:48:44 +08:00
} else /*if (point.end)*/ {
activeWriteCount--;
if (activeWriteCount == 0)
combinedWriteConflictRanges.back().second = point.key;
2017-05-26 04:48:44 +08:00
}
}
}
}
namespace {
StringRef setK(Arena& arena, int i) {
char t[sizeof(i)];
*(int*)t = i;
const int keySize = 16;
char* ss = new (arena) char[keySize];
for (int c = 0; c < keySize - sizeof(i); c++)
ss[c] = '.';
for (int c = 0; c < sizeof(i); c++)
ss[c + keySize - sizeof(i)] = t[sizeof(i) - 1 - c];
return StringRef((const uint8_t*)ss, keySize);
}
2017-05-26 04:48:44 +08:00
void miniConflictSetTest() {
2020-01-31 06:24:37 +08:00
for (int i = 0; i < 2000000; i++) {
int size = 64 * 5; // Also run 64*64*5 to test multiple words of andValues and orValues
2017-05-26 04:48:44 +08:00
MiniConflictSet mini(size);
2020-01-31 06:24:37 +08:00
for (int j = 0; j < 2; j++) {
int a = deterministicRandom()->randomInt(0, size);
int b = deterministicRandom()->randomInt(a, size);
2020-01-31 06:24:37 +08:00
mini.set(a, b);
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
for (int j = 0; j < 4; j++) {
int a = deterministicRandom()->randomInt(0, size);
int b = deterministicRandom()->randomInt(a, size);
2020-01-31 06:24:37 +08:00
mini.any(a, b); // Tests correctness internally
2017-05-26 04:48:44 +08:00
}
}
printf("miniConflictSetTest complete\n");
}
2020-02-11 01:47:06 +08:00
void operatorLessThanTest() {
{ // Longer strings before shorter strings.
2020-02-11 02:48:34 +08:00
KeyInfo a(LiteralStringRef("hello"), /*begin=*/false, /*write=*/true, 0, nullptr);
KeyInfo b(LiteralStringRef("hello\0"), /*begin=*/false, /*write=*/false, 0, nullptr);
2020-02-11 01:47:06 +08:00
ASSERT(a < b);
ASSERT(!(b < a));
ASSERT(!(a == b));
}
2017-05-26 04:48:44 +08:00
2020-02-11 01:47:06 +08:00
{ // Reads before writes.
2020-02-11 02:48:34 +08:00
KeyInfo a(LiteralStringRef("hello"), /*begin=*/false, /*write=*/false, 0, nullptr);
KeyInfo b(LiteralStringRef("hello"), /*begin=*/false, /*write=*/true, 0, nullptr);
2020-02-11 01:47:06 +08:00
ASSERT(a < b);
ASSERT(!(b < a));
ASSERT(!(a == b));
}
{ // Begin reads after writes.
2020-02-11 02:48:34 +08:00
KeyInfo a(LiteralStringRef("hello"), /*begin=*/false, /*write=*/true, 0, nullptr);
KeyInfo b(LiteralStringRef("hello"), /*begin=*/true, /*write=*/false, 0, nullptr);
2020-02-11 01:47:06 +08:00
ASSERT(a < b);
ASSERT(!(b < a));
ASSERT(!(a == b));
}
2017-05-26 04:48:44 +08:00
2020-02-11 01:47:06 +08:00
{ // Begin writes after writes.
2020-02-11 02:48:34 +08:00
KeyInfo a(LiteralStringRef("hello"), /*begin=*/false, /*write=*/true, 0, nullptr);
KeyInfo b(LiteralStringRef("hello"), /*begin=*/true, /*write=*/true, 0, nullptr);
2020-02-11 01:47:06 +08:00
ASSERT(a < b);
ASSERT(!(b < a));
ASSERT(!(a == b));
}
}
} // namespace
2020-02-11 01:47:06 +08:00
2017-05-26 04:48:44 +08:00
void skipListTest() {
printf("Skip list test\n");
miniConflictSetTest();
2020-02-11 02:55:04 +08:00
operatorLessThanTest();
2017-05-26 04:48:44 +08:00
setAffinity(0);
double start;
ConflictSet* cs = newConflictSet();
Arena testDataArena;
2020-01-31 06:24:37 +08:00
VectorRef<VectorRef<KeyRangeRef>> testData;
const int batches = 500; // deterministicRandom()->randomInt(500, 5000);
const int data_per_batch = 5000;
testData.resize(testDataArena, batches);
std::vector<std::vector<uint8_t>> success(batches);
std::vector<std::vector<uint8_t>> success2(batches);
for (int i = 0; i < batches; i++) {
testData[i].resize(testDataArena, data_per_batch);
success[i].assign(data_per_batch, false);
success2[i].assign(data_per_batch, false);
for (int j = 0; j < data_per_batch; j++) {
int key = deterministicRandom()->randomInt(0, 20000000);
int key2 = key + 1 + deterministicRandom()->randomInt(0, 10);
2020-01-31 06:24:37 +08:00
testData[i][j] = KeyRangeRef(setK(testDataArena, key), setK(testDataArena, key2));
2017-05-26 04:48:44 +08:00
}
}
printf("Test data generated: %d batches, %d/batch\n", batches, data_per_batch);
2017-05-26 04:48:44 +08:00
printf("Running\n");
int readCount = 1, writeCount = 1;
int cranges = 0, tcount = 0;
start = timer();
std::vector<std::vector<int>> nonConflict(batches);
Version version = 0;
for (const auto& data : testData) {
2017-05-26 04:48:44 +08:00
Arena buf;
std::vector<CommitTransactionRef> trs;
2017-05-26 04:48:44 +08:00
double t = timer();
for (int j = 0; j + readCount + writeCount <= data.size(); j += readCount + writeCount) {
2017-05-26 04:48:44 +08:00
CommitTransactionRef tr;
2020-01-31 06:24:37 +08:00
for (int k = 0; k < readCount; k++) {
KeyRangeRef r(buf, data[j + k]);
2020-01-31 06:24:37 +08:00
tr.read_conflict_ranges.push_back(buf, r);
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
for (int k = 0; k < writeCount; k++) {
KeyRangeRef r(buf, data[j + readCount + k]);
2020-01-31 06:24:37 +08:00
tr.write_conflict_ranges.push_back(buf, r);
2017-05-26 04:48:44 +08:00
}
cranges += tr.read_conflict_ranges.size() + tr.write_conflict_ranges.size();
tr.read_snapshot = version;
2017-05-26 04:48:44 +08:00
trs.push_back(tr);
}
tcount += trs.size();
2020-01-31 06:24:37 +08:00
g_buildTest += timer() - t;
2017-05-26 04:48:44 +08:00
t = timer();
2020-01-31 06:24:37 +08:00
ConflictBatch batch(cs);
for (const auto& tr : trs) {
batch.addTransaction(tr);
}
2020-01-31 06:24:37 +08:00
g_add += timer() - t;
2017-05-26 04:48:44 +08:00
t = timer();
batch.detectConflicts(version + 50, version, nonConflict[version]);
2020-01-31 06:24:37 +08:00
g_detectConflicts += timer() - t;
version++;
2017-05-26 04:48:44 +08:00
}
2020-01-31 06:24:37 +08:00
double elapsed = timer() - start;
2017-05-26 04:48:44 +08:00
printf("New conflict set: %0.3f sec\n", elapsed);
2020-01-31 06:24:37 +08:00
printf(" %0.3f Mtransactions/sec\n", tcount / elapsed / 1e6);
printf(" %0.3f Mkeys/sec\n", cranges * 2 / elapsed / 1e6);
2017-05-26 04:48:44 +08:00
elapsed = g_detectConflicts.getValue();
printf("Detect only: %0.3f sec\n", elapsed);
2020-01-31 06:24:37 +08:00
printf(" %0.3f Mtransactions/sec\n", tcount / elapsed / 1e6);
printf(" %0.3f Mkeys/sec\n", cranges * 2 / elapsed / 1e6);
2017-05-26 04:48:44 +08:00
elapsed = g_checkRead.getValue() + g_merge.getValue();
printf("Skiplist only: %0.3f sec\n", elapsed);
2020-01-31 06:24:37 +08:00
printf(" %0.3f Mtransactions/sec\n", tcount / elapsed / 1e6);
printf(" %0.3f Mkeys/sec\n", cranges * 2 / elapsed / 1e6);
2017-05-26 04:48:44 +08:00
printf("Performance counters:\n");
for (const auto& counter : skc) {
printf("%20s: %s\n", counter->getMetric().name().c_str(), counter->getMetric().formatted().c_str());
2017-05-26 04:48:44 +08:00
}
printf("%d entries in version history\n", cs->versionHistory.count());
}