gem5  v20.0.0.3
trie.hh
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2012 Google
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are
7  * met: redistributions of source code must retain the above copyright
8  * notice, this list of conditions and the following disclaimer;
9  * redistributions in binary form must reproduce the above copyright
10  * notice, this list of conditions and the following disclaimer in the
11  * documentation and/or other materials provided with the distribution;
12  * neither the name of the copyright holders nor the names of its
13  * contributors may be used to endorse or promote products derived from
14  * this software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28 
29 #ifndef __BASE_TRIE_HH__
30 #define __BASE_TRIE_HH__
31 
32 #include <cassert>
33 #include <iostream>
34 #include <type_traits>
35 
36 #include "base/cprintf.hh"
37 #include "base/logging.hh"
38 #include "base/types.hh"
39 
48 template <class Key, class Value>
49 class Trie
50 {
51  protected:
52  static_assert(std::is_integral<Key>::value,
53  "Key has to be an integral type");
54 
55  struct Node
56  {
57  Key key;
58  Key mask;
59 
60  bool
62  {
63  return (test & mask) == key;
64  }
65 
66  Value *value;
67 
69  Node *kids[2];
70 
71  Node(Key _key, Key _mask, Value *_val) :
72  key(_key & _mask), mask(_mask), value(_val),
73  parent(NULL)
74  {
75  kids[0] = NULL;
76  kids[1] = NULL;
77  }
78 
79  void
81  {
82  if (kids[1]) {
83  kids[1]->clear();
84  delete kids[1];
85  kids[1] = NULL;
86  }
87  if (kids[0]) {
88  kids[0]->clear();
89  delete kids[0];
90  kids[0] = NULL;
91  }
92  }
93 
94  void
95  dump(std::ostream &os, int level)
96  {
97  for (int i = 1; i < level; i++) {
98  ccprintf(os, "|");
99  }
100  if (level == 0)
101  ccprintf(os, "Root ");
102  else
103  ccprintf(os, "+ ");
104  ccprintf(os, "(%p, %p, %#X, %#X, %p)\n",
105  parent, this, key, mask, value);
106  if (kids[0])
107  kids[0]->dump(os, level + 1);
108  if (kids[1])
109  kids[1]->dump(os, level + 1);
110  }
111  };
112 
113  protected:
115 
116  public:
117  typedef Node *Handle;
118 
119  Trie() : head(0, 0, NULL)
120  {}
121 
122  static const unsigned MaxBits = sizeof(Key) * 8;
123 
124  private:
135  bool
136  goesAfter(Node **parent, Node *kid, Key key, Key new_mask)
137  {
138  if (kid && kid->matches(key) && (kid->mask & new_mask) == kid->mask) {
139  *parent = kid;
140  return true;
141  } else {
142  return false;
143  }
144  }
145 
154  Key
155  extendMask(Key orig)
156  {
157  // Just in case orig was 0.
158  const Key msb = ULL(1) << (MaxBits - 1);
159  return orig | (orig >> 1) | msb;
160  }
161 
169  Handle
171  {
172  Node *node = &head;
173  while (node) {
174  if (node->value)
175  return node;
176 
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];
181  else
182  node = NULL;
183  }
184 
185  return NULL;
186  }
187 
188  public:
196  Handle
197  insert(Key key, unsigned width, Value *val)
198  {
199  // We use NULL value pointers to mark internal nodes of the trie, so
200  // we don't allow inserting them as real values.
201  assert(val);
202 
203  // Build a mask which masks off all the bits we don't care about.
204  Key new_mask = ~(Key)0;
205  if (width < MaxBits)
206  new_mask <<= (MaxBits - width);
207  // Use it to tidy up the key.
208  key &= new_mask;
209 
210  // Walk past all the nodes this new node will be inserted after. They
211  // can be ignored for the purposes of this function.
212  Node *node = &head;
213  while (goesAfter(&node, node->kids[0], key, new_mask) ||
214  goesAfter(&node, node->kids[1], key, new_mask))
215  {}
216  assert(node);
217 
218  Key cur_mask = node->mask;
219  // If we're already where the value needs to be...
220  if (cur_mask == new_mask) {
221  assert(!node->value);
222  node->value = val;
223  return node;
224  }
225 
226  for (unsigned int i = 0; i < 2; i++) {
227  Node *&kid = node->kids[i];
228  Node *new_node;
229  if (!kid) {
230  // No kid. Add a new one.
231  new_node = new Node(key, new_mask, val);
232  new_node->parent = node;
233  kid = new_node;
234  return new_node;
235  }
236 
237  // Walk down the leg until something doesn't match or we run out
238  // of bits.
239  Key last_mask;
240  bool done;
241  do {
242  last_mask = cur_mask;
243  cur_mask = extendMask(cur_mask);
244  done = ((key & cur_mask) != (kid->key & cur_mask)) ||
245  last_mask == new_mask;
246  } while (!done);
247  cur_mask = last_mask;
248 
249  // If this isn't the right leg to go down at all, skip it.
250  if (cur_mask == node->mask)
251  continue;
252 
253  // At the point we walked to above, add a new node.
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;
258  kid = new_node;
259 
260  // If we ran out of bits, the value goes right here.
261  if (cur_mask == new_mask) {
262  new_node->value = val;
263  return new_node;
264  }
265 
266  // Still more bits to deal with, so add a new node for that path.
267  new_node = new Node(key, new_mask, val);
268  new_node->parent = kid;
269  kid->kids[1] = new_node;
270  return new_node;
271  }
272 
273  panic("Reached the end of the Trie insert function!\n");
274  return NULL;
275  }
276 
282  Value *
283  lookup(Key key)
284  {
285  Node *node = lookupHandle(key);
286  if (node)
287  return node->value;
288  else
289  return NULL;
290  }
291 
297  Value *
298  remove(Handle handle)
299  {
300  Node *node = handle;
301  Value *val = node->value;
302  if (node->kids[1]) {
303  assert(node->value);
304  node->value = NULL;
305  return val;
306  }
307  if (!node->parent)
308  panic("Trie: Can't remove root node.\n");
309 
310  Node *parent = node->parent;
311 
312  // If there's a kid, fix up it's parent pointer.
313  if (node->kids[0])
314  node->kids[0]->parent = parent;
315  // Figure out which kid we are, and update our parent's pointers.
316  if (parent->kids[0] == node)
317  parent->kids[0] = node->kids[0];
318  else if (parent->kids[1] == node)
319  parent->kids[1] = node->kids[0];
320  else
321  panic("Trie: Inconsistent parent/kid relationship.\n");
322  // Make sure if the parent only has one kid, it's kid[0].
323  if (parent->kids[1] && !parent->kids[0]) {
324  parent->kids[0] = parent->kids[1];
325  parent->kids[1] = NULL;
326  }
327 
328  // If the parent has less than two kids and no cargo and isn't the
329  // root, delete it too.
330  if (!parent->kids[1] && !parent->value && parent->parent)
331  remove(parent);
332  delete node;
333  return val;
334  }
335 
341  Value *
342  remove(Key key)
343  {
344  Handle handle = lookupHandle(key);
345  if (!handle)
346  return NULL;
347  return remove(handle);
348  }
349 
354  void
356  {
357  head.clear();
358  }
359 
364  void
365  dump(const char *title, std::ostream &os=std::cout)
366  {
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");
371  head.dump(os, 0);
372  }
373 };
374 
375 #endif
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:163
void ccprintf(cp::Print &print)
Definition: cprintf.hh:127
Handle insert(Key key, unsigned width, Value *val)
Method which inserts a key/value pair into the trie.
Definition: trie.hh:197
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...
Definition: trie.hh:136
Trie()
Definition: trie.hh:119
Bitfield< 7 > i
bool matches(Key test)
Definition: trie.hh:61
static const unsigned MaxBits
Definition: trie.hh:122
Value * value
Definition: trie.hh:66
Node(Key _key, Key _mask, Value *_val)
Definition: trie.hh:71
Key mask
Definition: trie.hh:58
void clear()
A method which removes all key/value pairs from the trie.
Definition: trie.hh:355
Handle lookupHandle(Key key)
Method which looks up the Handle corresponding to a particular key.
Definition: trie.hh:170
void dump(std::ostream &os, int level)
Definition: trie.hh:95
Node * kids[2]
Definition: trie.hh:69
Bitfield< 17 > os
Definition: misc.hh:803
Bitfield< 63 > val
Definition: misc.hh:769
Key extendMask(Key orig)
A utility method which extends a mask value one more bit towards the lsb.
Definition: trie.hh:155
void clear()
Definition: trie.hh:80
Value * lookup(Key key)
Method which looks up the Value corresponding to a particular key.
Definition: trie.hh:283
Node head
Definition: trie.hh:114
void dump(const char *title, std::ostream &os=std::cout)
A debugging method which prints the contents of this trie.
Definition: trie.hh:365
Defines global host-dependent types: Counter, Tick, and (indirectly) {int,uint}{8,16,32,64}_t.
#define ULL(N)
uint64_t constant
Definition: types.hh:48
Bitfield< 20 > level
Definition: intmessage.hh:47
Node * Handle
Definition: trie.hh:117
Definition: test.h:38
A trie is a tree-based data structure used for data retrieval.
Definition: trie.hh:49
Bitfield< 4 > width
Key key
Definition: trie.hh:57
Node * parent
Definition: trie.hh:68

Generated on Fri Jul 3 2020 15:52:59 for gem5 by doxygen 1.8.13