2017-08-08 19:28:59 +01:00
|
|
|
(library
|
|
|
|
(btree)
|
|
|
|
|
2017-08-11 15:41:57 +01:00
|
|
|
(export btree-value-type
|
|
|
|
btree-dev
|
|
|
|
btree-root
|
|
|
|
btree-open
|
2017-08-08 19:28:59 +01:00
|
|
|
btree-lookup
|
2017-08-10 15:07:20 +01:00
|
|
|
btree-each
|
|
|
|
le64-type)
|
2017-08-08 19:28:59 +01:00
|
|
|
|
|
|
|
(import (block-io)
|
|
|
|
(chezscheme)
|
|
|
|
(binary-format)
|
|
|
|
(list-utils))
|
|
|
|
|
|
|
|
;;; Unlike the kernel or c++ versions, I'm going to leave it to the hiogher
|
|
|
|
;;; levels to handle multi level btrees.
|
2017-08-11 15:41:57 +01:00
|
|
|
(binary-format node-header
|
2017-08-08 19:28:59 +01:00
|
|
|
(csum le32)
|
|
|
|
(flags le32)
|
|
|
|
(blocknr le64)
|
|
|
|
(nr-entries le32)
|
|
|
|
(max-entries le32)
|
|
|
|
(value-size le32)
|
|
|
|
(padding le32))
|
|
|
|
|
2017-08-11 15:41:57 +01:00
|
|
|
;;; (unpacker bv offset)
|
2017-08-08 19:28:59 +01:00
|
|
|
(define-record-type value-type (fields size unpacker))
|
|
|
|
|
|
|
|
(define (max-entries vt)
|
|
|
|
(/ (- metadata-block-size node-header-size)
|
2017-08-11 15:41:57 +01:00
|
|
|
(+ (size-type le64)
|
2017-08-08 19:28:59 +01:00
|
|
|
(value-type-size vt))))
|
|
|
|
|
|
|
|
(define (key-offset index)
|
2017-08-11 15:41:57 +01:00
|
|
|
(+ node-header-size (* (size-type le64) index)))
|
2017-08-08 19:28:59 +01:00
|
|
|
|
2017-08-11 15:41:57 +01:00
|
|
|
(define (value-base header)
|
2017-08-08 19:28:59 +01:00
|
|
|
(+ node-header-size
|
2017-08-11 15:41:57 +01:00
|
|
|
(* (node-header-max-entries header)
|
|
|
|
(size-type le64))))
|
2017-08-08 19:28:59 +01:00
|
|
|
|
2017-08-11 15:41:57 +01:00
|
|
|
(define (value-offset header vt index)
|
|
|
|
(+ (value-base header)
|
2017-08-08 19:28:59 +01:00
|
|
|
(* (value-type-size vt) index)))
|
|
|
|
|
|
|
|
(define-record-type btree
|
|
|
|
(fields value-type dev root))
|
|
|
|
|
|
|
|
(define (btree-open vt dev root)
|
2017-08-11 15:41:57 +01:00
|
|
|
(make-btree vt dev root))
|
2017-08-08 19:28:59 +01:00
|
|
|
|
|
|
|
(define le64-type
|
2017-08-11 15:41:57 +01:00
|
|
|
(make-value-type (size-type le64)
|
2017-08-08 19:28:59 +01:00
|
|
|
(lambda (bv offset)
|
|
|
|
(unpack-type bv offset le64))))
|
|
|
|
|
|
|
|
(define (internal-node? header)
|
2017-08-12 19:27:21 +01:00
|
|
|
(bitwise-bit-set? (node-header-flags header) 0))
|
2017-08-08 19:28:59 +01:00
|
|
|
|
|
|
|
(define (leaf-node? header)
|
2017-08-12 19:27:21 +01:00
|
|
|
(bitwise-bit-set? (node-header-flags header) 1))
|
2017-08-08 19:28:59 +01:00
|
|
|
|
|
|
|
(define (key-at node index)
|
2017-08-11 15:41:57 +01:00
|
|
|
(unpack-type node (key-offset index) le64))
|
2017-08-08 19:28:59 +01:00
|
|
|
|
2017-08-11 15:41:57 +01:00
|
|
|
(define (value-at header node index vt)
|
|
|
|
((value-type-unpacker vt) node (value-offset header vt index)))
|
2017-08-08 19:28:59 +01:00
|
|
|
|
|
|
|
;;; Performs a binary search looking for the key and returns the index of the
|
|
|
|
;;; lower bound.
|
|
|
|
(define (lower-bound node header key)
|
2017-08-10 15:07:20 +01:00
|
|
|
(let loop ((lo 0) (hi (node-header-nr-entries header)))
|
|
|
|
(if (<= (- hi lo) 1)
|
2017-08-08 19:28:59 +01:00
|
|
|
lo
|
|
|
|
(let* ((mid (+ lo (/ (- hi lo) 2)))
|
2017-08-10 15:07:20 +01:00
|
|
|
(k (key-at node mid)))
|
2017-08-08 19:28:59 +01:00
|
|
|
(cond
|
|
|
|
((= key k) mid)
|
|
|
|
((< k key) (loop mid hi))
|
2017-08-10 15:07:20 +01:00
|
|
|
(else (loop lo mid)))))))
|
2017-08-08 19:28:59 +01:00
|
|
|
|
|
|
|
;;;;----------------------------------------------
|
|
|
|
;;;; Lookup
|
|
|
|
;;;;----------------------------------------------
|
|
|
|
|
|
|
|
(define (btree-lookup tree key default)
|
|
|
|
(let ((dev (btree-dev tree))
|
|
|
|
(vt (btree-value-type tree)))
|
|
|
|
|
2017-08-14 10:05:38 +01:00
|
|
|
(let loop ((root (btree-root tree)))
|
|
|
|
(let* ((node (read-block dev root))
|
|
|
|
(header (node-header-unpack node 0))
|
|
|
|
(index (lower-bound node header key)))
|
|
|
|
(if (internal-node? header)
|
|
|
|
(loop (value-at header node index le64-type))
|
|
|
|
(if (= key (key-at node index))
|
|
|
|
(value-at header node index vt)
|
|
|
|
default))))))
|
2017-08-08 19:28:59 +01:00
|
|
|
|
|
|
|
;;;;----------------------------------------------
|
|
|
|
;;;; Walking the btree
|
|
|
|
;;;;----------------------------------------------
|
|
|
|
|
|
|
|
;;; Calls (fn key value) on every entry of the btree.
|
|
|
|
(define (btree-each tree fn)
|
|
|
|
(let ((vt (btree-value-type tree)))
|
|
|
|
|
|
|
|
(define (visit-leaf node header)
|
|
|
|
(let loop ((index 0))
|
|
|
|
(when (< index (node-header-nr-entries header))
|
2017-08-11 15:41:57 +01:00
|
|
|
(fn (key-at node index) (value-at header node index vt))
|
2017-08-08 19:28:59 +01:00
|
|
|
(loop (+ 1 index)))))
|
|
|
|
|
|
|
|
(define (visit-internal node header)
|
|
|
|
(let loop ((index 0))
|
|
|
|
(when (< index (node-header-nr-entries header))
|
2017-08-11 15:41:57 +01:00
|
|
|
(visit-node (value-at header node index le64-type))
|
2017-08-08 19:28:59 +01:00
|
|
|
(loop (+ 1 index)))))
|
|
|
|
|
|
|
|
(define (visit-node root)
|
2017-08-11 15:41:57 +01:00
|
|
|
(let* ((node (read-block (btree-dev tree) root))
|
|
|
|
(header (node-header-unpack node 0)))
|
2017-08-08 19:28:59 +01:00
|
|
|
((if (internal-node? header) visit-internal visit-leaf) node header)))
|
|
|
|
|
2017-08-11 15:41:57 +01:00
|
|
|
(visit-node (btree-root tree)))))
|
2017-08-08 19:28:59 +01:00
|
|
|
|