ref: f5ad0ff02ba05f2f5e0926b85668968fbebeb721
dir: /qp.c/
#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, 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;
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));
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 <= 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);
assert(t->branch.twigs != nil);
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;
}