shithub: femtolisp

ref: d60399e18345e69419489f201f27f19f1a5616b2
dir: /3rd/fn.h/

View raw version
// 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 <[email protected]>
// You may do anything with this. It has no warranty.
// <http://creativecommons.org/publicdomain/zero/1.0/>

#pragma once

typedef uint32_t Tbitmap;
typedef uint64_t Tindex;

#if defined(__plan9__)
static inline uint32_t
__builtin_popcount(Tbitmap w)
{
	w -= (w >> 1) & 0x55555555;
	w = (w & 0x33333333) + ((w >> 2) & 0x33333333);
	w = (w + (w >> 4)) & 0x0F0F0F0F;
	w = (w * 0x01010101) >> 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 *)(uint64_t), ptr,   const char *, key);
Tset_field((Tindex),           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*)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) \
	((uint32_t)(((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(uint32_t, shift);
Tindex_get(uint32_t, offset);
Tindex_get(Tbitmap, bitmap);

static inline Tindex
Tindex_new(uint32_t shift, uint32_t offset, Tbitmap bitmap)
{
	uint32_t 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));
}

// sanity checks!

#if !defined(static_assert)
#define static_assert_cat(a,b) a##b
#define static_assert_name(line) static_assert_cat(static_assert_,line)
#define static_assert(must_be_true, message) \
	static const void *static_assert_name(__LINE__) \
		[must_be_true ? 2 : -1] = { \
			message, \
			static_assert_name(__LINE__) }
#endif

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");

//  ..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 uint8_t
knybble(const char *key, uint32_t off, uint32_t shift)
{
	uint32_t word = key[off]<<8;
	if(word)
		word |= key[off+1];
	uint32_t right = 16 - 5 - shift;
	return (word >> right) & 0x1FU;
}

static inline uint8_t
nibble(Tindex i, const char *key, size_t len)
{
	uint32_t off = Tindex_offset(i);
	if(off >= len)
		return(0);
	else return knybble(key, off, Tindex_shift(i));
}

static inline Tbitmap
twigbit(Tindex i, const char *key, size_t len)
{
	return(1U << nibble(i, key, len));
}

static inline bool
hastwig(Tindex i, Tbitmap bit)
{
	return Tindex_bitmap(i) & bit;
}

static inline uint32_t
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)