# # # The Nim Compiler # (c) Copyright 2018 Andreas Rumpf # # See the file "copying.txt", included in this # distribution, for details about the copyright. # ## BTree implementation with few features, but good enough for the ## Nim compiler's needs. const M = 512 # max children per B-tree node = M-1 # (must be even and greater than 2) Mhalf = M div 2 type Node[Key, Val] {.acyclic.} = ref object entries: int keys: array[M, Key] case isInternal: bool of false: vals: array[M, Val] of true: links: array[M, Node[Key, Val]] BTree*[Key, Val] = object root: Node[Key, Val] entries: int ## number of key-value pairs proc initBTree*[Key, Val](): BTree[Key, Val] = BTree[Key, Val](root: Node[Key, Val](entries: 0, isInternal: false)) template less(a, b): bool = cmp(a, b) < 0 template eq(a, b): bool = cmp(a, b) == 0 proc getOrDefault*[Key, Val](b: BTree[Key, Val], key: Key): Val = var x = b.root while x.isInternal: for j in 0.. 0: result.add(indent & "(" & $h.keys[j] & ")\n") toString(h.links[j], indent & " ", result) proc `$`[Key, Val](b: BTree[Key, Val]): string = result = "" toString(b.root, "", result) proc hasNext*[Key, Val](b: BTree[Key, Val]; index: int): bool = index < b.entries proc countSubTree[Key, Val](it: Node[Key, Val]): int = if it.isInternal: result = 0 for k in 0.. i: it = it.links[k] dec i, (sum - c) break result = (it.keys[i], it.vals[i], index+1) iterator pairs*[Key, Val](b: BTree[Key, Val]): (Key, Val) = var i = 0 while hasNext(b, i): let (k, v, i2) = next(b, i) i = i2 yield (k, v) proc len*[Key, Val](b: BTree[Key, Val]): int {.inline.} = b.entries