29 #ifndef __BASE_TRIE_HH__ 30 #define __BASE_TRIE_HH__ 34 #include <type_traits> 48 template <
class Key,
class Value>
52 static_assert(std::is_integral<Key>::value,
53 "Key has to be an integral type");
63 return (test & mask) ==
key;
71 Node(Key _key, Key _mask, Value *_val) :
72 key(_key & _mask), mask(_mask), value(_val),
104 ccprintf(os,
"(%p, %p, %#X, %#X, %p)\n",
105 parent,
this, key, mask, value);
107 kids[0]->
dump(os, level + 1);
109 kids[1]->
dump(os, level + 1);
122 static const unsigned MaxBits =
sizeof(Key) * 8;
138 if (kid && kid->matches(key) && (kid->mask & new_mask) == kid->mask) {
158 const Key msb =
ULL(1) << (MaxBits - 1);
159 return orig | (orig >> 1) | msb;
177 if (node->kids[0] && node->kids[0]->matches(key))
178 node = node->
kids[0];
179 else if (node->kids[1] && node->kids[1]->matches(key))
180 node = node->
kids[1];
204 Key new_mask = ~(Key)0;
206 new_mask <<= (MaxBits -
width);
213 while (
goesAfter(&node, node->kids[0], key, new_mask) ||
214 goesAfter(&node, node->kids[1], key, new_mask))
218 Key cur_mask = node->mask;
220 if (cur_mask == new_mask) {
221 assert(!node->value);
226 for (
unsigned int i = 0;
i < 2;
i++) {
227 Node *&kid = node->kids[
i];
231 new_node =
new Node(key, new_mask, val);
232 new_node->parent = node;
242 last_mask = cur_mask;
244 done = ((key & cur_mask) != (kid->key & cur_mask)) ||
245 last_mask == new_mask;
247 cur_mask = last_mask;
250 if (cur_mask == node->mask)
254 new_node =
new Node(key, cur_mask, NULL);
255 new_node->parent = node;
256 kid->parent = new_node;
257 new_node->kids[0] = kid;
261 if (cur_mask == new_mask) {
262 new_node->value =
val;
267 new_node =
new Node(key, new_mask, val);
268 new_node->parent = kid;
269 kid->kids[1] = new_node;
273 panic(
"Reached the end of the Trie insert function!\n");
298 remove(Handle handle)
301 Value *
val = node->value;
308 panic(
"Trie: Can't remove root node.\n");
316 if (parent->kids[0] == node)
318 else if (parent->kids[1] == node)
321 panic(
"Trie: Inconsistent parent/kid relationship.\n");
323 if (parent->kids[1] && !parent->kids[0]) {
324 parent->kids[0] = parent->kids[1];
325 parent->kids[1] = NULL;
330 if (!parent->kids[1] && !parent->value && parent->parent)
347 return remove(handle);
365 dump(
const char *title, std::ostream &
os=std::cout)
367 ccprintf(
os,
"**************************************************\n");
368 ccprintf(
os,
"*** Start of Trie: %s\n", title);
369 ccprintf(
os,
"*** (parent, me, key, mask, value pointer)\n");
370 ccprintf(
os,
"**************************************************\n");
#define panic(...)
This implements a cprintf based panic() function.
void ccprintf(cp::Print &print)
Handle insert(Key key, unsigned width, Value *val)
Method which inserts a key/value pair into the trie.
bool goesAfter(Node **parent, Node *kid, Key key, Key new_mask)
A utility method which checks whether the key being looked up lies beyond the Node being examined...
static const unsigned MaxBits
Node(Key _key, Key _mask, Value *_val)
void clear()
A method which removes all key/value pairs from the trie.
Handle lookupHandle(Key key)
Method which looks up the Handle corresponding to a particular key.
void dump(std::ostream &os, int level)
Key extendMask(Key orig)
A utility method which extends a mask value one more bit towards the lsb.
Value * lookup(Key key)
Method which looks up the Value corresponding to a particular key.
void dump(const char *title, std::ostream &os=std::cout)
A debugging method which prints the contents of this trie.
Defines global host-dependent types: Counter, Tick, and (indirectly) {int,uint}{8,16,32,64}_t.
#define ULL(N)
uint64_t constant
A trie is a tree-based data structure used for data retrieval.