forked from kohler/masstree-beta
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmasstree_get.hh
128 lines (115 loc) · 3.65 KB
/
masstree_get.hh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
/* Masstree
* Eddie Kohler, Yandong Mao, Robert Morris
* Copyright (c) 2012-2014 President and Fellows of Harvard College
* Copyright (c) 2012-2014 Massachusetts Institute of Technology
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, subject to the conditions
* listed in the Masstree LICENSE file. These conditions include: you must
* preserve this copyright notice, and you cannot mention the copyright
* holders in advertising related to the Software without their permission.
* The Software is provided WITHOUT ANY WARRANTY, EXPRESS OR IMPLIED. This
* notice is a summary of the Masstree LICENSE file; the license in that file
* is legally binding.
*/
#ifndef MASSTREE_GET_HH
#define MASSTREE_GET_HH
#include "masstree_tcursor.hh"
#include "masstree_key.hh"
namespace Masstree {
template <typename P>
bool unlocked_tcursor<P>::find_unlocked(threadinfo& ti)
{
int match;
key_indexed_position kx;
node_base<P>* root = const_cast<node_base<P>*>(root_);
retry:
n_ = root->reach_leaf(ka_, v_, ti);
forward:
if (v_.deleted())
goto retry;
n_->prefetch();
perm_ = n_->permutation();
kx = leaf<P>::bound_type::lower(ka_, *this);
if (kx.p >= 0) {
lv_ = n_->lv_[kx.p];
lv_.prefetch(n_->keylenx_[kx.p]);
match = n_->ksuf_matches(kx.p, ka_);
} else
match = 0;
if constexpr (P::track_nodes) {
if (match == 0) {
// Store a snapshot of the node tracker
// (TicTocVersion in our case)
tkr_[0] = *n_->get_aux_tracker();
}
}
if (n_->has_changed(v_)) {
ti.mark(threadcounter(tc_stable_leaf_insert + n_->simple_has_split(v_)));
n_ = n_->advance_to_key(ka_, v_, ti);
goto forward;
}
if (match < 0) {
ka_.shift_by(-match);
root = lv_.layer();
goto retry;
} else
return match;
}
template <typename P>
inline bool basic_table<P>::get(Str key, value_type &value,
threadinfo& ti) const
{
unlocked_tcursor<P> lp(*this, key);
bool found = lp.find_unlocked(ti);
if (found)
value = lp.value();
return found;
}
template <typename P>
bool tcursor<P>::find_locked(threadinfo& ti)
{
node_base<P>* root = const_cast<node_base<P>*>(root_);
nodeversion_type v;
permuter_type perm;
retry:
n_ = root->reach_leaf(ka_, v, ti);
forward:
if (v.deleted())
goto retry;
n_->prefetch();
perm = n_->permutation();
fence();
kx_ = leaf<P>::bound_type::lower(ka_, *n_);
if (kx_.p >= 0) {
leafvalue<P> lv = n_->lv_[kx_.p];
lv.prefetch(n_->keylenx_[kx_.p]);
state_ = n_->ksuf_matches(kx_.p, ka_);
if (state_ < 0 && !n_->has_changed(v) && lv.layer()->is_root()) {
ka_.shift_by(-state_);
root = lv.layer();
goto retry;
}
} else
state_ = 0;
n_->lock(v, ti.lock_fence(tc_leaf_lock));
if (n_->has_changed(v) || n_->permutation() != perm) {
ti.mark(threadcounter(tc_stable_leaf_insert + n_->simple_has_split(v)));
n_->unlock();
n_ = n_->advance_to_key(ka_, v, ti);
goto forward;
} else if (unlikely(state_ < 0)) {
ka_.shift_by(-state_);
n_->lv_[kx_.p] = root = n_->lv_[kx_.p].layer()->maybe_parent();
n_->unlock();
goto retry;
} else if (unlikely(n_->deleted_layer())) {
ka_.unshift_all();
root = const_cast<node_base<P>*>(root_);
goto retry;
}
return state_;
}
} // namespace Masstree
#endif