Simplify firstGeq - make it not stepwise
All checks were successful
Tests / Release [gcc] total: 704, passed: 704
GNU C Compiler (gcc) |Total|New|Outstanding|Fixed|Trend
|:-:|:-:|:-:|:-:|:-:
|0|0|0|0|:clap:
Reference build: <a href="https://jenkins.weaselab.dev/job/weaselab/job/conflict-set/job/main/66//gcc">weaselab » conflict-set » main #66</a>
Tests / Release [gcc,aarch64] total: 703, passed: 703
Tests / Coverage total: 702, passed: 702
weaselab/conflict-set/pipeline/head This commit looks good
All checks were successful
Tests / Release [gcc] total: 704, passed: 704
GNU C Compiler (gcc) |Total|New|Outstanding|Fixed|Trend
|:-:|:-:|:-:|:-:|:-:
|0|0|0|0|:clap:
Reference build: <a href="https://jenkins.weaselab.dev/job/weaselab/job/conflict-set/job/main/66//gcc">weaselab » conflict-set » main #66</a>
Tests / Release [gcc,aarch64] total: 703, passed: 703
Tests / Coverage total: 702, passed: 702
weaselab/conflict-set/pipeline/head This commit looks good
This commit is contained in:
148
ConflictSet.cpp
148
ConflictSet.cpp
@@ -1741,109 +1741,69 @@ void addWriteRange(Node *&root, int64_t oldestVersion,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct FirstGeqStepwise {
|
Iterator firstGeq(Node *n, const std::span<const uint8_t> key) {
|
||||||
Node *n;
|
auto remaining = key;
|
||||||
std::span<const uint8_t> remaining;
|
for (;;) {
|
||||||
int cmp;
|
if (remaining.size() == 0) {
|
||||||
|
if (n->entryPresent) {
|
||||||
enum Phase {
|
return {n, 0};
|
||||||
Init,
|
|
||||||
// Being in this phase implies that the key matches the search path exactly
|
|
||||||
// up to this point
|
|
||||||
Search,
|
|
||||||
DownLeftSpine
|
|
||||||
};
|
|
||||||
Phase phase;
|
|
||||||
|
|
||||||
FirstGeqStepwise(Node *n, std::span<const uint8_t> remaining)
|
|
||||||
: n(n), remaining(remaining), phase(Init) {}
|
|
||||||
|
|
||||||
// Not being done implies that n is not the firstGeq
|
|
||||||
bool step() {
|
|
||||||
switch (phase) {
|
|
||||||
case Search: {
|
|
||||||
if (remaining.size() == 0) {
|
|
||||||
int c = getChildGeq(n, 0);
|
|
||||||
assert(c >= 0);
|
|
||||||
n = getChildExists(n, c);
|
|
||||||
return downLeftSpine();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
auto *child = getChild(n, remaining[0]);
|
|
||||||
if (child == nullptr) {
|
|
||||||
int c = getChildGeq(n, remaining[0]);
|
|
||||||
if (c >= 0) {
|
|
||||||
n = getChildExists(n, c);
|
|
||||||
return downLeftSpine();
|
|
||||||
} else {
|
|
||||||
n = nextSibling(n);
|
|
||||||
return downLeftSpine();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
n = child;
|
|
||||||
remaining = remaining.subspan(1, remaining.size() - 1);
|
|
||||||
|
|
||||||
if (n->partialKeyLen > 0) {
|
|
||||||
int commonLen = std::min<int>(n->partialKeyLen, remaining.size());
|
|
||||||
int i = longestCommonPrefixPartialKey(n->partialKey, remaining.data(),
|
|
||||||
commonLen);
|
|
||||||
if (i < commonLen) {
|
|
||||||
auto c = n->partialKey[i] <=> remaining[i];
|
|
||||||
if (c > 0) {
|
|
||||||
return downLeftSpine();
|
|
||||||
} else {
|
|
||||||
n = nextSibling(n);
|
|
||||||
return downLeftSpine();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (commonLen == n->partialKeyLen) {
|
|
||||||
// partial key matches
|
|
||||||
remaining =
|
|
||||||
remaining.subspan(commonLen, remaining.size() - commonLen);
|
|
||||||
} else if (n->partialKeyLen > int(remaining.size())) {
|
|
||||||
// n is the first physical node greater than remaining, and there's no
|
|
||||||
// eq node
|
|
||||||
return downLeftSpine();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
[[fallthrough]];
|
|
||||||
case Init:
|
|
||||||
phase = Search;
|
|
||||||
if (remaining.size() == 0 && n->entryPresent) {
|
|
||||||
cmp = 0;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
case DownLeftSpine:
|
|
||||||
int c = getChildGeq(n, 0);
|
int c = getChildGeq(n, 0);
|
||||||
assert(c >= 0);
|
assert(c >= 0);
|
||||||
n = getChildExists(n, c);
|
n = getChildExists(n, c);
|
||||||
if (n->entryPresent) {
|
goto downLeftSpine;
|
||||||
cmp = 1;
|
}
|
||||||
return true;
|
|
||||||
|
auto *child = getChild(n, remaining[0]);
|
||||||
|
if (child == nullptr) {
|
||||||
|
int c = getChildGeq(n, remaining[0]);
|
||||||
|
if (c >= 0) {
|
||||||
|
n = getChildExists(n, c);
|
||||||
|
goto downLeftSpine;
|
||||||
|
} else {
|
||||||
|
n = nextSibling(n);
|
||||||
|
goto downLeftSpine;
|
||||||
}
|
}
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
__builtin_unreachable(); // GCOVR_EXCL_LINE
|
|
||||||
}
|
|
||||||
|
|
||||||
bool downLeftSpine() {
|
n = child;
|
||||||
phase = DownLeftSpine;
|
remaining = remaining.subspan(1, remaining.size() - 1);
|
||||||
if (n == nullptr || n->entryPresent) {
|
|
||||||
cmp = 1;
|
if (n->partialKeyLen > 0) {
|
||||||
return true;
|
int commonLen = std::min<int>(n->partialKeyLen, remaining.size());
|
||||||
|
int i = longestCommonPrefixPartialKey(n->partialKey, remaining.data(),
|
||||||
|
commonLen);
|
||||||
|
if (i < commonLen) {
|
||||||
|
auto c = n->partialKey[i] <=> remaining[i];
|
||||||
|
if (c > 0) {
|
||||||
|
goto downLeftSpine;
|
||||||
|
} else {
|
||||||
|
n = nextSibling(n);
|
||||||
|
goto downLeftSpine;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (commonLen == n->partialKeyLen) {
|
||||||
|
// partial key matches
|
||||||
|
remaining = remaining.subspan(commonLen, remaining.size() - commonLen);
|
||||||
|
} else if (n->partialKeyLen > int(remaining.size())) {
|
||||||
|
// n is the first physical node greater than remaining, and there's no
|
||||||
|
// eq node
|
||||||
|
goto downLeftSpine;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return step();
|
|
||||||
}
|
}
|
||||||
};
|
downLeftSpine:
|
||||||
|
if (n == nullptr) {
|
||||||
Iterator firstGeq(Node *n, const std::span<const uint8_t> key) {
|
return {nullptr, 1};
|
||||||
FirstGeqStepwise stepwise{n, key};
|
}
|
||||||
while (!stepwise.step())
|
for (;;) {
|
||||||
;
|
if (n->entryPresent) {
|
||||||
return {stepwise.n, stepwise.cmp};
|
return {n, 1};
|
||||||
|
}
|
||||||
|
int c = getChildGeq(n, 0);
|
||||||
|
assert(c >= 0);
|
||||||
|
n = getChildExists(n, c);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct __attribute__((visibility("hidden"))) ConflictSet::Impl {
|
struct __attribute__((visibility("hidden"))) ConflictSet::Impl {
|
||||||
|
Reference in New Issue
Block a user