shithub: qk1

Download patch

ref: 0667b932941239bd7a561e4ff17e1f1307ab9acb
parent: 6405b2a349f9ceb8b8678c81535e72843b848b6e
author: Sigrid Solveig Haflínudóttir <sigrid@ftrv.se>
date: Sun Jan 12 19:17:17 EST 2025

qp: replace with 5-bit qp tries (more portable)

--- a/cmd.c
+++ b/cmd.c
@@ -222,7 +222,7 @@
 }
 
 static void
-Cmd_AliasList(char *name, void *v, void *aux)
+Cmd_AliasList(const char *name, void *v, void *aux)
 {
 	cmd_t *a;
 
--- a/console.c
+++ b/console.c
@@ -1,5 +1,5 @@
 #include "quakedef.h"
-#include "qp.h"
+#include "tbl.h"
 
 static int con_linewidth;
 
@@ -36,31 +36,26 @@
 
 extern void M_Menu_Main_f (cmd_t *c);
 
-static Trie *conobj;
+static Tbl *conobj;
 
 void
 Con_AddObject(char *name, void *obj)
 {
-	conobj = qpset(conobj, name, 0, obj);
+	conobj = Tset(conobj, name, obj);
 }
 
 void *
 Con_FindObject(char *name)
 {
-	char *k;
-	void *v;
-
-	if(qpget(conobj, name, 0, &k, &v) != 0)
-		v = nil;
-	return v;
+	return Tget(conobj, name);
 }
 
 int
-Con_SearchObject(char *prefix, int len0, void (*f)(char *name, void *obj, void *aux), void *aux)
+Con_SearchObject(const char *prefix, int len0, void (*f)(const char *name, void *obj, void *aux), void *aux)
 {
-	char *k;
+	const char *k;
 	void *v;
-	int n, len, c;
+	int n, c, len;
 
 	if(conobj == nil)
 		return 0;
@@ -69,7 +64,7 @@
 	v = nil;
 	len = 0;
 	for(n = 0;;){
-		if(qpnext(conobj, &k, &len, &v) < 0)
+		if(!Tnextl(conobj, &k, &len, &v))
 			break;
 		c = -1;
 		if(len0 == 0 || (len >= len0 && (c = strncmp(k, prefix, len0)) == 0)){
@@ -259,9 +254,9 @@
 static void
 Con_Print(char *txt)
 {
+	static bool cr;
 	int		y;
 	int		c, l;
-	static int	cr;
 	int		mask;
 
 	con_backscroll = 0;
@@ -312,13 +307,10 @@
 
 		switch (c)
 		{
+		case '\r':
+			cr = true;
 		case '\n':
 			con_x = 0;
-			break;
-
-		case '\r':
-			con_x = 0;
-			cr = 1;
 			break;
 
 		default:	// display character and advance
--- a/console.h
+++ b/console.h
@@ -24,4 +24,4 @@
 
 void Con_AddObject(char *name, void *obj);
 void *Con_FindObject(char *name);
-int Con_SearchObject(char *prefix, int len, void (*f)(char *, void *, void *), void *aux);
+int Con_SearchObject(const char *prefix, int len, void (*f)(const char *, void *, void *), void *aux);
--- /dev/null
+++ b/fn.c
@@ -1,0 +1,205 @@
+// fn.c: quintet bit popcount patricia tries, new version
+//
+// Written by Tony Finch <dot@dotat.at>
+// You may do anything with this. It has no warranty.
+// <http://creativecommons.org/publicdomain/zero/1.0/>
+
+#include "platform.h"
+#include "tbl.h"
+#include "fn.h"
+
+bool
+Tgetkv(Tbl *t, const char *key, int len, const char **pkey, void **pval)
+{
+	if(t == nil)
+		return false;
+	while(isbranch(t)){
+		Tindex i = t->index;
+		Tbitmap b = twigbit(i, key, len);
+		if(!hastwig(i, b))
+			return false;
+		t = Tbranch_twigs(t) + twigoff(i, b);
+	}
+	if(strcmp(key, Tleaf_key(t)) != 0)
+		return false;
+	*pkey = Tleaf_key(t);
+	*pval = Tleaf_val(t);
+	return true;
+}
+
+static bool
+next_rec(Trie *t, const char **pkey, int *plen, void **pval)
+{
+	Tindex i = t->index;
+	if(Tindex_branch(i)){
+		// Recurse to find either this leaf (*pkey != nil)
+		// or the next one (*pkey == nil).
+		Tbitmap b = twigbit(i, *pkey, *plen);
+		u32int s, m; TWIGOFFMAX(s, m, i, b);
+		for(; s < m; s++){
+			if(next_rec(Tbranch_twigs(t)+s, pkey, plen, pval))
+				return true;
+		}
+		return false;
+	}
+	// We have found the next leaf.
+	if(*pkey == nil){
+		*pkey = Tleaf_key(t);
+		*plen = strlen(*pkey);
+		*pval = Tleaf_val(t);
+		return true;
+	}
+	// We have found this leaf, so start looking for the next one.
+	if(strcmp(*pkey, Tleaf_key(t)) == 0){
+		*pkey = nil;
+		*plen = 0;
+		return false;
+	}
+	// No match.
+	return false;
+}
+
+bool
+Tnextl(Tbl *tbl, const char **pkey, int *plen, void **pval)
+{
+	if(tbl == nil){
+		*pkey = nil;
+		*plen = 0;
+		return false;
+	}
+	return next_rec(tbl, pkey, plen, pval);
+}
+
+Tbl *
+Tdelkv(Tbl *tbl, const char *key, int len, const char **pkey, void **pval)
+{
+	if(tbl == nil)
+		return nil;
+	Trie *t = tbl, *p = nil;
+	Tindex i = 0;
+	Tbitmap b = 0;
+	while(isbranch(t)){
+		i = t->index;
+		b = twigbit(i, key, len);
+		if(!hastwig(i, b))
+			return tbl;
+		p = t; t = Tbranch_twigs(t) + twigoff(i, b);
+	}
+	if(strcmp(key, Tleaf_key(t)) != 0)
+		return tbl;
+	*pkey = Tleaf_key(t);
+	*pval = Tleaf_val(t);
+	if(p == nil){
+		free(tbl);
+		return nil;
+	}
+	Trie *twigs = Tbranch_twigs(p);
+	u32int m = __builtin_popcount(Tindex_bitmap(i));
+	assert(twigs <= t && t < twigs+m);
+	if(m == 2){
+		// Move the other twig to the parent branch.
+		*p = twigs[twigs == t];
+		free(twigs);
+		return tbl;
+	}
+	memmove(t, t+1, ((twigs + m) - (t + 1)) * sizeof(Trie));
+	p->index = Tbitmap_del(i, b);
+	// We have now correctly removed the twig from the trie, so if
+	// realloc() fails we can ignore it and continue to use the
+	// slightly oversized twig array.
+	twigs = realloc(twigs, sizeof(Trie) * (m - 1));
+	if(twigs != nil)
+		Tset_twigs(p, twigs);
+	return tbl;
+}
+
+Tbl *
+Tsetl(Tbl *tbl, const char *key, int len, void *val)
+{
+	assert(!Tindex_branch((Tindex)(uintptr)val) && len <= (int)Tmaxlen);
+	if(val == nil)
+		return Tdell(tbl, key, len);
+	// First leaf in an empty tbl?
+	if(tbl == nil){
+		tbl = malloc(sizeof(*tbl));
+		if(tbl == nil)
+			return nil;
+		Tset_key(tbl, key);
+		Tset_val(tbl, val);
+		return tbl;
+	}
+	Trie *t = tbl;
+	// Find the most similar leaf node in the trie. We will compare
+	// its key with our new key to find the first differing nibble,
+	// which can be at a lower index than the point at which we
+	// detect a difference.
+	while(isbranch(t)){
+		Tindex i = t->index;
+		Tbitmap b = twigbit(i, key, len);
+		// Even if our key is missing from this branch we need to
+		// keep iterating down to a leaf. It doesn't matter which
+		// twig we choose since the keys are all the same up to this
+		// index. Note that blindly using twigoff(t, b) can cause
+		// an out-of-bounds index if it equals twigmax(t).
+		u32int s = hastwig(i, b) ? twigoff(i, b) : 0;
+		t = Tbranch_twigs(t) + s;
+	}
+	// Do the keys differ, and if so, where?
+	u32int off, xor, shf;
+	const char *tkey = Tleaf_key(t);
+	for(off = 0; off <= (u32int)len; off++){
+		xor = (u8int)key[off] ^ (u8int)tkey[off];
+		if(xor != 0)
+			goto newkey;
+	}
+	Tset_val(t, val);
+	return tbl;
+newkey:; // We have the branch's byte index; what is its chunk index?
+	u32int bit = off * 8 + qclz(xor) + 8 - sizeof(u32int) * 8;
+	u32int qo = bit / 5;
+	off = qo * 5 / 8;
+	shf = qo * 5 % 8;
+	// re-index keys with adjusted offset
+	Tbitmap nb = 1U << knybble(key,off,shf);
+	Tbitmap tb = 1U << knybble(tkey,off,shf);
+	// Prepare the new leaf.
+	Trie nt;
+	Tset_key(&nt, key);
+	Tset_val(&nt, val);
+	// Find where to insert a branch or grow an existing branch.
+	t = tbl;
+	Tindex i;
+	while(isbranch(t)){
+		i = t->index;
+		if(off == Tindex_offset(i) && shf == Tindex_shift(i))
+			goto growbranch;
+		if(off == Tindex_offset(i) && shf < Tindex_shift(i))
+			goto newbranch;
+		if(off < Tindex_offset(i))
+			goto newbranch;
+		Tbitmap b = twigbit(i, key, len);
+		assert(hastwig(i, b));
+		t = Tbranch_twigs(t) + twigoff(i, b);
+	}
+newbranch:;
+	Trie *twigs = malloc(sizeof(Trie) * 2);
+	if(twigs == nil)
+		return nil;
+	i = Tindex_new(shf, off, nb | tb);
+	twigs[twigoff(i, nb)] = nt;
+	twigs[twigoff(i, tb)] = *t;
+	Tset_twigs(t, twigs);
+	Tset_index(t, i);
+	return tbl;
+growbranch:
+	assert(!hastwig(i, nb));
+	u32int s, m; TWIGOFFMAX(s, m, i, nb);
+	twigs = realloc(Tbranch_twigs(t), sizeof(Trie) * (m + 1));
+	if(twigs == nil)
+		return nil;
+	memmove(twigs+s+1, twigs+s, sizeof(Trie) * (m - s));
+	memmove(twigs+s, &nt, sizeof(Trie));
+	Tset_twigs(t, twigs);
+	Tset_index(t, Tbitmap_add(i, nb));
+	return tbl;
+}
--- /dev/null
+++ b/fn.h
@@ -1,0 +1,200 @@
+// fn.h: quintet bit popcount patricia tries, new version
+//
+// This version uses somewhat different terminology than older
+// variants. The location of a quintet in the key is now called its
+// "offset", and the whole word containing the offset, bitmap, and tag
+// bit is called the "index word" (by analogy with a database index).
+// The precise quintet location is represented as a byte offset and a
+// shift. Previously a flags field contained the isbranch tag and shift,
+// but these are now separate.
+//
+// Instead of trying to use bit fields, this code uses accessor
+// functions to split up a pair of words into their constituent parts.
+// This should improve portability to machines with varying endianness
+// and/or word size.
+//
+// Written by Tony Finch <dot@dotat.at>
+// You may do anything with this. It has no warranty.
+// <http://creativecommons.org/publicdomain/zero/1.0/>
+
+#pragma once
+
+typedef u32int Tbitmap;
+typedef u64int Tindex;
+
+#ifdef __plan9__
+static inline u32int
+__builtin_popcount(Tbitmap w)
+{
+	w -= (w >> 1) & 0x55555555U;
+	w = (w & 0x33333333U) + ((w >> 2) & 0x33333333U);
+	w = (w + (w >> 4)) & 0x0F0F0F0FU;
+	w = (w * 0x01010101U) >> 24;
+	return w;
+}
+#endif
+
+typedef struct Tbl {
+	Tindex index;
+	void *ptr;
+}Trie;
+
+// accessor functions, except for the index word
+
+#define Tset_field(cast, elem, type, field) \
+	static inline void \
+	Tset_##field(Trie *t, type field) \
+	{ \
+		t->elem = cast field; \
+	} \
+	struct dummy
+
+Tset_field((void *),          ptr,   Trie *,       twigs);
+Tset_field((Tindex),          index, Tindex,       index);
+Tset_field((void *)(uintptr), ptr,   const char *, key);
+Tset_field((Tindex)(uintptr), index, void *,       val);
+
+static inline bool Tindex_branch(Tindex i);
+
+static inline bool isbranch(Trie *t)
+{
+	return Tindex_branch(t->index);
+}
+
+#define Tbranch(t) assert(isbranch(t))
+#define Tleaf(t)  assert(!isbranch(t))
+
+#define Tcheck_get(type, tag, field, expr) \
+	static inline type \
+	tag##_##field(Trie *t) \
+	{ \
+		tag(t); \
+		return expr; \
+	} \
+	struct dummy
+
+Tcheck_get(Trie *,       Tbranch, twigs, t->ptr);
+Tcheck_get(const char *, Tleaf,   key,   t->ptr);
+Tcheck_get(void *,       Tleaf,   val,   (void*)(uintptr)t->index);
+
+// index word layout
+
+#define Tix_width_branch 1
+#define Tix_width_shift  3
+#define Tix_width_offset 28
+#define Tix_width_bitmap 32
+
+#define Tix_base_branch 0
+#define Tix_base_shift  (Tix_base_branch + Tix_width_branch)
+#define Tix_base_offset (Tix_base_shift  + Tix_width_shift)
+#define Tix_base_bitmap (Tix_base_offset + Tix_width_offset)
+
+#define Tix_place(field) ((Tindex)(field) << Tix_base_##field)
+
+#define Tix_mask(field) ((1ULL << Tix_width_##field) - 1ULL)
+
+#define Tunmask(field,index) \
+	((u32int)(((index) >> Tix_base_##field) & Tix_mask(field)))
+
+#define Tmaxlen Tix_mask(offset)
+
+// index word accessor functions
+
+#define Tindex_get(type, field) \
+	static inline type \
+	Tindex_##field(Tindex i) \
+	{ \
+		return Tunmask(field, i); \
+	} \
+	struct dummy
+
+Tindex_get(bool, branch);
+Tindex_get(u32int, shift);
+Tindex_get(u32int, offset);
+Tindex_get(Tbitmap, bitmap);
+
+static inline Tindex
+Tindex_new(u32int shift, u32int offset, Tbitmap bitmap)
+{
+	u32int branch = 1;
+	return
+		Tix_place(branch) |
+		Tix_place(shift)  |
+		Tix_place(offset) |
+		Tix_place(bitmap) ;
+}
+
+static inline Tindex
+Tbitmap_add(Tindex i, Tbitmap bitmap)
+{
+	return i | Tix_place(bitmap);
+}
+
+static inline Tindex
+Tbitmap_del(Tindex i, Tbitmap bitmap)
+{
+	return i & ~Tix_place(bitmap);
+}
+
+#if 0
+// sanity checks!
+static_assert(Tix_base_bitmap + Tix_width_bitmap == 64,
+	      "index fields must fill a 64 bit word");
+
+static_assert(Tunmask(bitmap,0x1234567800000000ULL) == 0x12345678,
+	      "extracting the bitmap works");
+
+static_assert(Tunmask(offset,0x0420ULL) == 0x42,
+	      "extracting the offset works");
+
+static_assert(Tunmask(shift,0xFEDCBAULL) == 5,
+	      "extracting the shift works");
+#endif
+
+//  ..key[o%5==0].. ..key[o%5==1].. ..key[o%5==2].. ..key[o%5==3].. ..key[o%5==4]..
+// |               |               |               |               |               |
+//  7 6 5 4 3 2 1 0 7 6 5 4 3 2 1 0 7 6 5 4 3 2 1 0 7 6 5 4 3 2 1 0 7 6 5 4 3 2 1 0
+// |         |         |         |         |         |         |         |         |
+//  shift=0   shift=5   shift=2   shift=7   shift=4   shift=1   shift=6   shift=3
+
+static inline u8int
+knybble(const char *key, u32int off, u32int shift)
+{
+	u32int word = (u8int)key[off]<<8;
+	if(word)
+		word |= key[off+1];
+	u32int right = 16 - 5 - shift;
+	return (word >> right) & 0x1FU;
+}
+
+static inline u8int
+nibble(Tindex i, const char *key, int len)
+{
+	u32int off = Tindex_offset(i);
+	if(off >= (u32int)len)
+		return 0;
+	return knybble(key, off, Tindex_shift(i));
+}
+
+static inline Tbitmap
+twigbit(Tindex i, const char *key, int len)
+{
+	return 1U << nibble(i, key, len);
+}
+
+static inline bool
+hastwig(Tindex i, Tbitmap bit)
+{
+	return Tindex_bitmap(i) & bit;
+}
+
+static inline u32int
+twigoff(Tindex i, Tbitmap bit)
+{
+	return __builtin_popcount(Tindex_bitmap(i) & (bit-1));
+}
+
+#define TWIGOFFMAX(off, max, i, b) do{ \
+		off = twigoff(i, b); \
+		max = __builtin_popcount(Tindex_bitmap(i)); \
+	}while(0)
--- a/fs.c
+++ b/fs.c
@@ -471,7 +471,7 @@
 }
 
 static void
-dumpcvar(char *name, void *o, void *bf)
+dumpcvar(const char *name, void *o, void *bf)
 {
 	cvar_t *c;
 
--- a/keys.c
+++ b/keys.c
@@ -129,7 +129,7 @@
 */
 
 static void
-Key_Complete(char *name, void *o, void *aux)
+Key_Complete(const char *name, void *o, void *aux)
 {
 	static int printed = 0;
 	int n;
--- a/meson.build
+++ b/meson.build
@@ -48,6 +48,7 @@
 	'd_surf.c',
 	'd_vars.c',
 	'draw.c',
+	'fn.c',
 	'fs.c',
 	'host.c',
 	'host_cmd.c',
@@ -73,7 +74,6 @@
 	'pr_edict.c',
 	'pr_exec.c',
 	'protocol.c',
-	'qp.c',
 	'r_aclip.c',
 	'r_alias.c',
 	'r_bsp.c',
@@ -96,6 +96,7 @@
 	'sv_move.c',
 	'sv_phys.c',
 	'sv_user.c',
+	'tbl.c',
 	'view.c',
 	'wav.c',
 	'world.c',
--- a/mkfile
+++ b/mkfile
@@ -31,6 +31,7 @@
 	d_surf.$O\
 	d_vars.$O\
 	draw.$O\
+	fn.$O\
 	fs.$O\
 	host.$O\
 	host_cmd.$O\
@@ -62,7 +63,6 @@
 	pr_edict.$O\
 	pr_exec.$O\
 	protocol.$O\
-	qp.$O\
 	r_aclip.$O\
 	r_alias.$O\
 	r_bsp.$O\
@@ -87,6 +87,7 @@
 	sv_phys.$O\
 	sv_user.$O\
 	sys_plan9.$O\
+	tbl.$O\
 	vid_plan9.$O\
 	view.$O\
 	wav.$O\
@@ -107,6 +108,7 @@
 	d_local.h\
 	dat.h\
 	draw.h\
+	fn.h\
 	fns.h\
 	i_tga.h\
 	i_wad.h\
@@ -122,7 +124,6 @@
 	progdefs.h\
 	progs.h\
 	protocol.h\
-	qp.h\
 	quakedef.h\
 	r_local.h\
 	r_shared.h\
@@ -132,6 +133,7 @@
 	server.h\
 	softfloat.h\
 	spritegn.h\
+	tbl.h\
 	vid.h\
 	view.h\
 	world.h\
@@ -147,3 +149,6 @@
 
 i_resize.$O: i_resize.c
 	$CC $CFLAGS -p i_resize.c
+
+fn.$O: fn.c
+	$CC $CFLAGS -p fn.c
--- a/plan9/platform.h
+++ b/plan9/platform.h
@@ -45,6 +45,7 @@
 static double ln2c;
 #define exp2f(x) (exp((x) * (ln2c ? ln2c : (ln2c = log(2.0)))))
 
+int qclz(unsigned);
 int qctz(unsigned);
 
 float DotProduct(const float v1[3], const float v2[3]);
--- a/posix/platform.h
+++ b/posix/platform.h
@@ -41,6 +41,7 @@
 #define setrealloctag(p, t) do{USED(p); USED(t);}while(0)
 #define isNaNf isnan
 
+#define qclz(x) __builtin_clz(x)
 #define qctz(x) __builtin_ctz(x)
 
 #ifdef HAVE_ENDIAN_H
--- a/qp.c
+++ /dev/null
@@ -1,230 +1,0 @@
-#include "quakedef.h"
-#include "qp.h"
-
-typedef u16int Tbitmap;
-typedef struct Tbranch Tbranch;
-typedef struct Tleaf Tleaf;
-
-struct Tleaf {
-	char *k;
-	void *v;
-};
-
-struct Tbranch {
-	Trie *twigs;
-	u64int x;
-};
-
-union Trie {
-	Tleaf leaf;
-	Tbranch branch;
-};
-
-#define x_flags(x) ((x)>>62)
-#define x_index(x) ((x)>>16 & ((1ULL<<46)-1ULL))
-#define x_bitmap(x) ((x) & 0xffff)
-
-#define isbranch(t) (x_flags((t)->branch.x) != 0)
-#define hastwig(t, b) (x_bitmap((t)->branch.x) & (b))
-#define twigoff(t, b) popcount(x_bitmap((t)->branch.x) & ((b)-1))
-#define twig(t, i) (&(t)->branch.twigs[(i)])
-#define twigoffmax(off, max, t, b) do{ off = twigoff(t, b); max = popcount(x_bitmap((t)->branch.x)); }while(0)
-#define nibbit(k, f) (1 << (((k) & (((f)-2) ^ 0x0f) & 0xff) >> ((2-(f))<<2)))
-#define twigbit(t, k, len) \
-	(x_index((t)->branch.x) >= (u64int)(len) ? \
-	1 : \
-	nibbit((u8int)(k)[x_index((t)->branch.x)], x_flags((t)->branch.x)))
-
-/*
-// need some speed? just say no.
-#define POPCNT	BYTE $0xf3; BYTE $0x0f; BYTE $0xb8
-
-TEXT popcount(SB),$0
-	MOVL b+0(FP), AX
-	POPCNT
-	RET
-*/
-static u16int
-popcount(u16int b)
-{
-	b -= b>>1 & 0x5555;
-	b = (b & 0x3333) + ((b>>2) & 0x3333);
-	b = (b + (b>>4)) & 0x0f0f;
-	b = (b + (b>>8)) & 0x00ff;
-	return b;
-}
-
-int
-qpget(Trie *t, char *k, int len, char **pk, void **pv)
-{
-	Tbitmap b;
-
-	assert(k != nil && pk != nil && pv != nil);
-
-	if(len < 1)
-		len = strlen(k);
-	if(t == nil)
-		return -1;
-	for(; isbranch(t); t = twig(t, twigoff(t, b))){
-		b = twigbit(t, k, len);
-		if(!hastwig(t, b))
-			return -1;
-	}
-	if(strncmp(k, t->leaf.k, len) != 0)
-		return -1;
-	*pk = t->leaf.k;
-	*pv = t->leaf.v;
-
-	return 0;
-}
-
-int
-qpnext(Trie *t, char **pk, int *plen, void **pv)
-{
-	Tbitmap b;
-	unsigned s, m;
-
-	assert(pk != nil && plen != nil && pv != nil);
-
-	if(isbranch(t)){
-		b = twigbit(t, *pk, *plen);
-		twigoffmax(s, m, t, b);
-		for(; s < m; s++){
-			if(qpnext(twig(t, s), pk, plen, pv) == 0)
-				return 0;
-		}
-		return -1;
-	}
-
-	if(*pk == nil){
-		*pk = t->leaf.k;
-		*plen = strlen(*pk);
-		*pv = t->leaf.v;
-		return 0;
-	}
-	if(*pk == t->leaf.k || strcmp(*pk, t->leaf.k) == 0){
-		*pk = nil;
-		*plen = 0;
-		*pv = nil;
-		return -1;
-	}
-
-	return -1;
-}
-
-Trie *
-qpdel(Trie *t, char *k, int len, char **pk, void **pv)
-{
-	Trie *p, *twigs;
-	Tbitmap b;
-	unsigned s, m;
-
-	if(t == nil)
-		return nil;
-	assert(k != nil && pk != nil && pv != nil);
-	if(len < 1)
-		len = strlen(k);
-	assert(len > 0);
-
-	for(b = 0, p = nil; isbranch(t); p = t, t = twig(t, twigoff(t, b))){
-		b = twigbit(t, k, len);
-		if(!hastwig(t, b))
-			return t;
-	}
-	
-	if(strncmp(k, t->leaf.k, len) != 0)
-		return t;
-	*pk = t->leaf.k;
-	*pv = t->leaf.v;
-
-	if(p == nil){
-		free(t);
-		return nil;
-	}
-	t = p;
-
-	twigoffmax(s, m, t, b);
-	if(m == 2){
-		twigs = t->branch.twigs;
-		*t = *twig(t, !s);
-		free(twigs);
-		return t;
-	}
-	memmove(t->branch.twigs+s, t->branch.twigs+s+1, sizeof(*t)*(m-s-1));
-	t->branch.x &= ~(u64int)b;
-
-	t->branch.twigs = realloc(t->branch.twigs, sizeof(*t)*(m-1));
-	return t;
-}
-
-Trie *
-qpset(Trie *t, char *k, int len, void *v)
-{
-	Trie *t0, t1, t2;
-	unsigned i, s, m;
-	u8int f;
-	Tbitmap b, b1, b2;
-	u8int k2;
-
-	assert(k != nil && v != nil);
-	if(len < 1)
-		len = strlen(k);
-	assert(len > 0);
-
-	if(t == nil){
-		t = malloc(sizeof(*t));
-		assert(t != nil);
-		t->leaf.k = k;
-		t->leaf.v = v;
-		return t;
-	}
-
-	t0 = t;
-	for(; isbranch(t); t = twig(t, i)){
-		b = twigbit(t, k, len);
-		i = hastwig(t, b) ? twigoff(t, b) : 0;
-	}
-	for(i = 0; i <= (unsigned)len && k[i] == t->leaf.k[i]; i++);
-	if(i == (unsigned)len+1){
-		t->leaf.v = v;
-		return t0;
-	}
-
-	k2 = (u8int)t->leaf.k[i];
-	f = (((u8int)k[i] ^ k2) & 0xf0) ? 1 : 2;
-	b1 = nibbit(k[i], f);
-	t1.leaf.k = k;
-	t1.leaf.v = v;
-	for(t = t0; isbranch(t); t = twig(t, twigoff(t, b))){
-		if(i == x_index(t->branch.x)){
-			if(f == x_flags(t->branch.x))
-				goto growbranch;
-			if(f < x_flags(t->branch.x))
-				break;
-		}else if(i < x_index(t->branch.x)){
-			break;
-		}
-		b = twigbit(t, k, len);
-		assert(hastwig(t, b));
-	}
-
-	t2 = *t;
-	b2 = nibbit(k2, f);
-	t->branch.twigs = malloc(sizeof(*t)*2);
-	assert(t->branch.twigs != nil);
-	t->branch.x = (u64int)f<<62 | (u64int)i<<16 | b1 | b2;
-	*twig(t, twigoff(t, b1)) = t1;
-	*twig(t, twigoff(t, b2)) = t2;
-
-	return t0;
-
-growbranch:
-	assert(!hastwig(t, b1));
-	twigoffmax(s, m, t, b1);
-	t->branch.twigs = realloc(t->branch.twigs, sizeof(*t)*(m+1));
-	memmove(t->branch.twigs+s+1, t->branch.twigs+s, sizeof(*t)*(m-s));
-	memmove(t->branch.twigs+s, &t1, sizeof(t1));
-	t->branch.x |= b1;
-
-	return t0;
-}
--- a/qp.h
+++ /dev/null
@@ -1,7 +1,0 @@
-typedef union Trie Trie;
-#pragma incomplete Trie
-
-int qpget(Trie *t, char *k, int len, char **pk, void **pv);
-int qpnext(Trie *t, char **pk, int *plen, void **pv);
-Trie *qpdel(Trie *t, char *k, int len, char **pk, void **pv);
-Trie *qpset(Trie *t, char *k, int len, void *v);
--- a/sys_plan9.c
+++ b/sys_plan9.c
@@ -9,6 +9,16 @@
 static int ndisabled;
 
 int
+qclz(unsigned x)
+{
+	unsigned r;
+	if(x == 0)
+		return 32;
+	for(r = 0; (x & (1UL<<31)) == 0; x <<= 1, r++);
+	return r;
+}
+
+int
 qctz(unsigned x)
 {
 	unsigned r;
--- /dev/null
+++ b/tbl.c
@@ -1,0 +1,57 @@
+// tbl.c: simpler wrappers for core table functions
+//
+// Written by Tony Finch <dot@dotat.at>
+// You may do anything with this. It has no warranty.
+// <http://creativecommons.org/publicdomain/zero/1.0/>
+
+#include "platform.h"
+#include "tbl.h"
+
+void *
+Tgetl(Tbl *tbl, const char *key, int len)
+{
+	const char *rkey = nil;
+	void *rval = nil;
+	return Tgetkv(tbl, key, len, &rkey, &rval) ? rval : nil;
+}
+
+void *
+Tget(Tbl *tbl, const char *key)
+{
+	return Tgetl(tbl, key, strlen(key));
+}
+
+Tbl *
+Tset(Tbl *tbl, const char *key, void *value)
+{
+	return Tsetl(tbl, key, strlen(key), value);
+}
+
+Tbl *
+Tdell(Tbl *tbl, const char *key, int len)
+{
+	const char *rkey = nil;
+	void *rval = nil;
+	return Tdelkv(tbl, key, len, &rkey, &rval);
+}
+
+Tbl *
+Tdel(Tbl *tbl, const char *key)
+{
+	return Tdell(tbl, key, strlen(key));
+}
+
+bool
+Tnext(Tbl *tbl, const char **pkey, void **pvalue)
+{
+	int len = *pkey == nil ? 0 : strlen(*pkey);
+	return Tnextl(tbl, pkey, &len, pvalue);
+}
+
+const char *
+Tnxt(Tbl *tbl, const char *key)
+{
+	void *value = nil;
+	Tnext(tbl, &key, &value);
+	return key;
+}
--- /dev/null
+++ b/tbl.h
@@ -1,0 +1,56 @@
+// tbl.h: an abstract API for tables with string keys.
+//
+// Written by Tony Finch <dot@dotat.at>
+// You may do anything with this. It has no warranty.
+// <http://creativecommons.org/publicdomain/zero/1.0/>
+
+#pragma once
+
+// A table is represented by a pointer to this incomplete struct type.
+// You initialize an empty table by setting the pointer to NULL.
+//
+typedef struct Tbl Tbl;
+#ifdef __plan9__
+#pragma incomplete Tbl
+#endif
+
+// Get the value associated with a key.
+// Returns NULL if the key is not in the Table.
+//
+void *Tgetl(Tbl *tbl, const char *key, int klen);
+void *Tget(Tbl *tbl, const char *key);
+
+// Returns false if the key is not found, otherwise returns true and
+// sets *rkey and *rval to the table's key and value pointers.
+//
+bool Tgetkv(Tbl *tbl, const char *key, int klen, const char **rkey, void **rval);
+
+// Associate a key with a value in a table. Returns a new pointer to
+// the modified table. If there is an error it sets errno and returns
+// NULL. To delete a key, set its value to NULL. When the last key is
+// deleted, Tset() returns NULL without setting errno. The key and
+// value are borrowed not copied.
+//
+// Errors:
+// EINVAL - value pointer is not word-aligned
+// ENOMEM - allocation failed
+//
+Tbl *Tsetl(Tbl *tbl, const char *key, int klen, void *value);
+Tbl *Tset(Tbl *tbl, const char *key, void *value);
+Tbl *Tdell(Tbl *tbl, const char *key, int klen);
+Tbl *Tdel(Tbl *tbl, const char *key);
+
+// Deletes an entry from the table as above, and sets *rkey and *rval
+// to the removed key and value pointers.
+//
+Tbl *Tdelkv(Tbl *tbl, const char *key, int klen, const char **rkey, void **rval);
+
+// Find the next item in the table. The p... arguments are in/out
+// parameters. To find the first key, pass *pkey=NULL and *pklen=0.
+// For subsequent keys, *pkey must be present in the table and is
+// updated to the lexicographically following key. Returns false or
+// NULL when there are no more keys.
+//
+bool Tnextl(Tbl *tbl, const char **pkey, int *pklen, void **pvalue);
+bool Tnext(Tbl *tbl, const char **pkey, void **pvalue);
+const char *Tnxt(Tbl *tbl, const char *key);