/* vim: set sw=8 ts=8 sts=8 noet: */ #ifndef CAPN_H #define CAPN_H #include #ifdef __cplusplus extern "C" { #endif #define CAPN_SEGID_LOCAL 0xFFFFFFFF /* struct capn is a common structure shared between segments in the same * session/context so that far pointers between the segments will be created. * * lookup is used to lookup segments by id when derefencing a far pointer * * create is used to create or lookup an alternate segment that has at least * sz available (ie returned seg->len + sz <= seg->cap) * * Allocated segments must be zero initialized. * * create and lookup can be NULL if you don't need multiple segments and don't * want to support copying * * create is also used to allocate room for the copy tree with id == * CAPN_SEGID_LOCAL. This data should be allocated in the local memory space * * seglist and copylist are linked lists which can be used to free up segments * on cleanup * * lookup, create, and user can be set by the user. Other values should be * zero initialized. */ struct capn { struct capn_segment *(*lookup)(void* /*user*/, uint32_t /*id */); struct capn_segment *(*create)(void* /*user*/, uint32_t /*id */, int /*sz*/); void *user; uint32_t segnum; struct capn_tree *copy; struct capn_tree *segtree; struct capn_segment *seglist, *lastseg; struct capn_segment *copylist; }; struct capn_tree { struct capn_tree *parent, *link[2]; unsigned int red : 1; }; /* struct capn_segment contains the information about a single segment. * capn should point to a struct capn that is shared between segments in the * same session * id specifies the segment id, used for far pointers * data specifies the segment data. This should not move after creation. * len specifies the current segment length. This should be 0 for a blank * segment. * cap specifies the segment capacity. * When creating new structures len will be incremented until it reaces cap, * at which point a new segment will be requested via capn->create. * * data, len, and cap must all by 8 byte aligned * * data, len, cap should all set by the user. Other values should be zero * initialized. */ struct capn_segment { struct capn_tree hdr; struct capn_segment *next; struct capn *capn; uint32_t id; char *data; int len, cap; }; enum CAPN_TYPE { CAPN_NULL = 0, CAPN_STRUCT = 1, CAPN_LIST = 2, CAPN_PTR_LIST = 3, CAPN_BIT_LIST = 4, CAPN_LIST_MEMBER = 5, }; struct capn_ptr { enum CAPN_TYPE type; int size; char *data; struct capn_segment *seg; uint32_t datasz; uint32_t ptrsz; }; struct capn_text { int size; const char *str; struct capn_segment *seg; }; struct capn_data { int size; const uint8_t *data; struct capn_segment *seg; }; union capn_iptr { struct capn_ptr c; uintptr_t u; void *p; }; struct capn_ret_vt { void (*free)(void*); }; /* capn_append_segment appends a segment to a session */ void capn_append_segment(struct capn*, struct capn_segment*); /* capn_root returns a fake pointer that can be used to get/set the session * root object using capn_(get|set)_ptr at index 0. The root is the object * pointed to by a ptr at offset 0 in segment 0. This will allocate room for * the root if not already. */ struct capn_ptr capn_root(struct capn*); /* capn_getp|setp functions get/set ptrs in list/structs * off is the list index or pointer index in a struct * capn_set_ptr will copy the data, create far pointers, etc if the target * is in a different segment/context. * Both of these will use/return inner pointers for composite lists. */ struct capn_ptr capn_getp(const struct capn_ptr *p, int off); int capn_setp(struct capn_ptr *p, int off, const struct capn_ptr *tgt); /* capn_to_string returns a pointer to a string * Use this instead of accessing the data directly as these checks that the * string is null terminated, the list type, etc. * psz is filled out with the string length if non NULL */ struct capn_text capn_get_text(const struct capn_ptr *p, int off); struct capn_data capn_get_data(const struct capn_ptr *p, int off); int capn_set_text(struct capn_ptr *p, int off, struct capn_text tgt); int capn_set_data(struct capn_ptr *p, int off, struct capn_data tgt); /* capn_get_* functions get data from a list * The length of the list is given by p->size * off specifies how far into the list to start * sz indicates the number of elements to get * The function returns the number of elements read or -1 on an error. * off must be byte aligned for capn_get1v */ int capn_get1(const struct capn_ptr *p, int off); uint8_t capn_get8(const struct capn_ptr *p, int off); uint16_t capn_get16(const struct capn_ptr *p, int off); uint32_t capn_get32(const struct capn_ptr *p, int off); uint64_t capn_get64(const struct capn_ptr *p, int off); int capn_getv1(const struct capn_ptr *p, int off, uint8_t *data, int sz); int capn_getv8(const struct capn_ptr *p, int off, uint8_t *data, int sz); int capn_getv16(const struct capn_ptr *p, int off, uint16_t *data, int sz); int capn_getv32(const struct capn_ptr *p, int off, uint32_t *data, int sz); int capn_getv64(const struct capn_ptr *p, int off, uint64_t *data, int sz); /* capn_set_* function set data in a list * off specifies how far into the list to start * sz indicates the number of elements to write * The function returns the number of elemnts written or -1 on an error. * off must be byte aligned for capn_set1v */ int capn_set1(struct capn_ptr *p, int off, int v); int capn_set8(struct capn_ptr *p, int off, uint8_t v); int capn_set16(struct capn_ptr *p, int off, uint16_t v); int capn_set32(struct capn_ptr *p, int off, uint32_t v); int capn_set64(struct capn_ptr *p, int off, uint64_t v); int capn_setv1(struct capn_ptr *p, int off, const uint8_t *data, int sz); int capn_setv8(struct capn_ptr *p, int off, const uint8_t *data, int sz); int capn_setv16(struct capn_ptr *p, int off, const uint16_t *data, int sz); int capn_setv32(struct capn_ptr *p, int off, const uint32_t *data, int sz); int capn_setv64(struct capn_ptr *p, int off, const uint64_t *data, int sz); /* capn_new_* functions create a new object * datasz is in bytes, ptrs is # of pointers, sz is # of elements in the list * If capn_new_string sz < 0, strlen is used to compute the string length * On an error a CAPN_NULL pointer is returned */ struct capn_ptr capn_new_struct(struct capn_segment *seg, int datasz, int ptrs); struct capn_ptr capn_new_list(struct capn_segment *seg, int sz, int datasz, int ptrs); struct capn_ptr capn_new_bit_list(struct capn_segment *seg, int sz); struct capn_ptr capn_new_ptr_list(struct capn_segment *seg, int sz); struct capn_ptr capn_new_string(struct capn_segment *seg, const char *str, int sz); #if defined(__cplusplus) || (defined(__STDC_VERSION__) && __STDC_VERSION__ >= 199901L) #define CAPN_INLINE static inline #else #define CAPN_INLINE static #endif /* capn_read|write_* functions read/write struct values * off is the offset into the structure in bytes * Rarely should these be called directly, instead use the generated code. * Data must be xored with the default value * These are inlined */ CAPN_INLINE uint8_t capn_read8(const struct capn_ptr *p, int off); CAPN_INLINE uint16_t capn_read16(const struct capn_ptr *p, int off); CAPN_INLINE uint32_t capn_read32(const struct capn_ptr *p, int off); CAPN_INLINE uint64_t capn_read64(const struct capn_ptr *p, int off); CAPN_INLINE int capn_write8(struct capn_ptr *p, int off, uint8_t val); CAPN_INLINE int capn_write16(struct capn_ptr *p, int off, uint16_t val); CAPN_INLINE int capn_write32(struct capn_ptr *p, int off, uint32_t val); CAPN_INLINE int capn_write64(struct capn_ptr *p, int off, uint64_t val); /* capn_init_malloc inits the capn struct with a create function which * allocates segments on the heap using malloc * * capn_free_all frees all the segment headers and data created by the create * function setup by capn_init_malloc */ void capn_init_malloc(struct capn *c); void capn_free_malloc(struct capn *c); /* capn_stream encapsulates the needed fields for capn_(deflate|inflate) in a * similar manner to z_stream from zlib * * The user should set next_in, avail_in, next_out, avail_out to the * available in/out buffers before calling capn_(deflate|inflate). * * Other fields should be zero initialized. */ struct capn_stream { uint8_t *next_in; int avail_in; uint8_t *next_out; int avail_out; int zeros, raw; }; #define CAPN_NEED_MORE_IN -1 #define CAPN_NEED_MORE_OUT -2 /* capn_deflate deflates a stream to the packed format * capn_inflate inflates a stream from the packed format * * They will return CAPN_NEED_MORE_(IN|OUT) as appropriate or 0 if the entire * input has been processed. */ int capn_deflate(struct capn_stream*); int capn_inflate(struct capn_stream*); int capn_marshal_iptr(const union capn_iptr*, struct capn_ptr*, int off); /* Inline functions */ #define T(IDX) s.v[IDX] = (uint8_t) (v >> (8*IDX)) CAPN_INLINE uint8_t capn_flip8(uint8_t v) { return v; } CAPN_INLINE uint16_t capn_flip16(uint16_t v) { union { uint16_t u; uint8_t v[2]; } s; T(0); T(1); return s.u; } CAPN_INLINE uint32_t capn_flip32(uint32_t v) { union { uint32_t u; uint8_t v[4]; } s; T(0); T(1); T(2); T(3); return s.u; } CAPN_INLINE uint64_t capn_flip64(uint64_t v) { union { uint64_t u; uint8_t v[8]; } s; T(0); T(1); T(2); T(3); T(4); T(5); T(6); T(7); return s.u; } #undef T CAPN_INLINE uint8_t capn_read8(const struct capn_ptr *p, int off) { return off+1 <= p->datasz ? capn_flip8(*(uint8_t*) (p->data+off)) : 0; } CAPN_INLINE int capn_write8(struct capn_ptr *p, int off, uint8_t val) { if (off+1 <= p->datasz) { *(uint8_t*) (p->data+off) = capn_flip8(val); return 0; } else { return -1; } } CAPN_INLINE uint16_t capn_read16(const struct capn_ptr *p, int off) { return off+2 <= p->datasz ? capn_flip16(*(uint16_t*) (p->data+off)) : 0; } CAPN_INLINE int capn_write16(struct capn_ptr *p, int off, uint16_t val) { if (off+2 <= p->datasz) { *(uint16_t*) (p->data+off) = capn_flip16(val); return 0; } else { return -1; } } CAPN_INLINE uint32_t capn_read32(const struct capn_ptr *p, int off) { return off+4 <= p->datasz ? capn_flip32(*(uint32_t*) (p->data+off)) : 0; } CAPN_INLINE int capn_write32(struct capn_ptr *p, int off, uint32_t val) { if (off+4 <= p->datasz) { *(uint32_t*) (p->data+off) = capn_flip32(val); return 0; } else { return -1; } } CAPN_INLINE uint64_t capn_read64(const struct capn_ptr *p, int off) { return off+8 <= p->datasz ? capn_flip64(*(uint64_t*) (p->data+off)) : 0; } CAPN_INLINE int capn_write64(struct capn_ptr *p, int off, uint64_t val) { if (off+8 <= p->datasz) { *(uint64_t*) (p->data+off) = capn_flip64(val); return 0; } else { return -1; } } CAPN_INLINE float capn_read_float(const struct capn_ptr *p, int off, float def) { union { float f; uint32_t u;} u; u.f = def; u.u ^= capn_read32(p, off); return u.f; } CAPN_INLINE int capn_write_float(struct capn_ptr *p, int off, float f, float def) { union { float f; uint32_t u;} u; union { float f; uint32_t u;} d; u.f = f; d.f = def; return capn_write32(p, off, u.u ^ d.u); } CAPN_INLINE double capn_read_double(const struct capn_ptr *p, int off, double def) { union { double f; uint64_t u;} u; u.f = def; u.u ^= capn_read64(p, off); return u.f; } CAPN_INLINE int capn_write_double(struct capn_ptr *p, int off, double f, double def) { union { double f; uint64_t u;} u; union { double f; uint64_t u;} d; d.f = f; u.f = f; return capn_write64(p, off, u.u ^ d.u); } #ifdef __cplusplus } #endif #endif