ref: 4fb339ca7d208f3ccc55af2c71b7149adfa1d697
author: Sigrid Haflínudóttir <ftrvxmtrx@gmail.com>
date: Thu Jun 25 05:38:38 EDT 2020
add qp tries
--- /dev/null
+++ b/LICENSE
@@ -1,0 +1,1 @@
+Public domain.
--- /dev/null
+++ b/README.md
@@ -1,0 +1,6 @@
+# Snippets
+
+Random snippets of code that are in public domain.
+Just copy and use for whatever you want.
+
+* `qt.[ch]` [QP tries](https://dotat.at/prog/qp/README.html)
--- /dev/null
+++ b/qp.c
@@ -1,0 +1,174 @@
+#include <u.h>
+#include <libc.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) >= (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, char **pv)
+{
+ Tbitmap b;
+
+ assert(k != nil && pk != nil && pv != nil);
+
+ 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;
+ uint 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;
+ }
+
+ return -1;
+}
+
+Trie *
+qpset(Trie *t, char *k, int len, void *v)
+{
+ Trie *t0, t1, t2;
+ uint 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));
+ 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 <= len && k[i] == t->leaf.k[i]; i++);
+ if(i == 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);
+ t->branch.x = (uvlong)f<<62 | (uvlong)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;
+}
--- /dev/null
+++ b/qp.h
@@ -1,0 +1,6 @@
+typedef union Trie Trie;
+#pragma incomplete Trie
+
+int qpget(Trie *t, char *k, int len, char **pk, char **pv);
+int qpnext(Trie *t, char **pk, int *plen, void **pv);
+Trie *qpset(Trie *t, char *k, int len, void *v);