mirror of
https://gitlab.nic.cz/labs/bird.git
synced 2025-01-05 08:31:53 +00:00
272 lines
8.3 KiB
C
272 lines
8.3 KiB
C
#ifndef CBOR_H
|
|
#define CBOR_H
|
|
|
|
#include "nest/bird.h"
|
|
#include "lib/hash.h"
|
|
#include "lib/socket.h"
|
|
|
|
/**
|
|
* CBOR Commonalities
|
|
**/
|
|
|
|
enum cbor_basic_type {
|
|
CBOR_POSINT = 0,
|
|
CBOR_NEGINT = 1,
|
|
CBOR_BYTES = 2,
|
|
CBOR_TEXT = 3,
|
|
CBOR_ARRAY = 4,
|
|
CBOR_MAP = 5,
|
|
CBOR_TAG = 6,
|
|
CBOR_SPECIAL = 7,
|
|
};
|
|
|
|
const char *cbor_type_str(enum cbor_basic_type);
|
|
|
|
/**
|
|
* CBOR Writer
|
|
**/
|
|
|
|
struct cbor_writer {
|
|
buffer data;
|
|
uint stack_pos, stack_max; /* Nesting of CBOR_ARRAY / CBOR_MAP */
|
|
struct cbor_writer_stack_item {
|
|
u64 items;
|
|
byte *head;
|
|
} stack[0];
|
|
};
|
|
|
|
/* Initialization */
|
|
static inline struct cbor_writer *cbor_writer_init(struct cbor_writer *w, uint stack_max_depth, byte *buf, uint size)
|
|
{
|
|
*w = (struct cbor_writer) {
|
|
.data = {
|
|
.start = buf,
|
|
.pos = buf,
|
|
.end = buf + size,
|
|
},
|
|
.stack_max = stack_max_depth,
|
|
};
|
|
return w;
|
|
}
|
|
|
|
#define cbor_writer_new(p, smax, buf, size) cbor_writer_init(mb_alloc((p), sizeof(struct cbor_writer) + (smax) * sizeof(struct cbor_writer_stack_item)), (smax), (buf), (size))
|
|
|
|
|
|
/* Return how many items have been encoded */
|
|
static inline int cbor_writer_done(struct cbor_writer *w)
|
|
{
|
|
if (w->stack_pos > 0)
|
|
return -1;
|
|
else
|
|
return w->stack[0].items;
|
|
}
|
|
|
|
/* Integer types */
|
|
bool cbor_put(struct cbor_writer *w, enum cbor_basic_type type, u64 value);
|
|
#define cbor_put_posint(w,v) cbor_put((w), CBOR_POSINT, (v))
|
|
#define cbor_put_negint(w,v) cbor_put((w), CBOR_NEGINT, -1-(v))
|
|
bool cbor_put_int(struct cbor_writer *w, int64_t value);
|
|
|
|
/* String types */
|
|
bool cbor_put_raw_bytes(struct cbor_writer *w, enum cbor_basic_type type, const byte *block, u64 size);
|
|
#define cbor_put_bytes(w, b, s) cbor_put_raw_bytes((w), CBOR_BYTES, (b), (s))
|
|
#define cbor_put_text(w, b, s) cbor_put_raw_bytes((w), CBOR_TEXT, (b), (s))
|
|
#define cbor_put_string(w, s) cbor_put_raw_bytes((w), CBOR_TEXT, (s), strlen(s))
|
|
#define cbor_put_toks(w, s) cbor_put_raw_bytes((w), CBOR_TEXT, #s, sizeof #s)
|
|
|
|
/* Compound types */
|
|
bool cbor_put_open(struct cbor_writer *w, enum cbor_basic_type type);
|
|
bool cbor_put_close(struct cbor_writer *w, u64 actual_size, bool strict);
|
|
#define cbor_open_array(w) cbor_put_open((w), CBOR_ARRAY)
|
|
#define cbor_open_map(w) cbor_put_open((w), CBOR_MAP)
|
|
|
|
#define cbor_close_array(w) cbor_put_close((w), 0, 0)
|
|
#define cbor_close_map(w) cbor_put_close((w), 0, 0)
|
|
|
|
#define CBOR_PUT_ARRAY(w) for (struct cbor_writer *_w = w, *_ww = cbor_open_array(_w) ? (_w) : (bug("buffer overflow on CBOR_ARRAY"), NULL); (_w = NULL), _ww; cbor_close_array(_ww), _ww = NULL)
|
|
|
|
#define CBOR_PUT_MAP(w) for (struct cbor_writer *_w = w, *_ww = cbor_open_map(_w) ? (_w) : (bug("buffer overflow on CBOR_MAP"), NULL); (_w = NULL), _ww; cbor_close_map(_ww), _ww = NULL)
|
|
|
|
/* Specials */
|
|
#define cbor_put_false(w) cbor_put((w), CBOR_SPECIAL, 20);
|
|
#define cbor_put_true(w) cbor_put((w), CBOR_SPECIAL, 21);
|
|
#define cbor_put_null(w) cbor_put((w), CBOR_SPECIAL, 22);
|
|
#define cbor_put_undef(w) cbor_put((w), CBOR_SPECIAL, 23);
|
|
|
|
#if 0
|
|
void cbor_add_int(struct cbor_writer *writer, int64_t item);
|
|
|
|
void cbor_add_ipv4(struct cbor_writer *writer, ip4_addr);
|
|
|
|
void cbor_add_ipv6(struct cbor_writer *writer, ip6_addr);
|
|
|
|
void cbor_epoch_time(struct cbor_writer *writer, int64_t time, int shift);
|
|
|
|
void cbor_relativ_time(struct cbor_writer *writer, int64_t time, int shift);
|
|
|
|
void cbor_add_ipv4_prefix(struct cbor_writer *writer, net_addr_ip4 *n);
|
|
|
|
|
|
void cbor_add_ipv6_prefix(struct cbor_writer *writer, net_addr_ip6 *n);
|
|
|
|
|
|
void cbor_add_uint(struct cbor_writer *writer, uint64_t item);
|
|
|
|
void cbor_add_tag(struct cbor_writer *writer, int item);
|
|
|
|
void cbor_add_string(struct cbor_writer *writer, const char *string);
|
|
|
|
void cbor_nonterminated_string(struct cbor_writer *writer, const char *string, uint32_t length);
|
|
|
|
void write_item(struct cbor_writer *writer, uint8_t major, uint64_t num);
|
|
|
|
void cbor_write_item_with_constant_val_length_4(struct cbor_writer *writer, uint8_t major, uint64_t num);
|
|
|
|
void rewrite_4bytes_int(struct cbor_writer *writer, int pt, int num);
|
|
#endif
|
|
|
|
/*
|
|
* Parser bits
|
|
*/
|
|
|
|
struct cbor_parser_context {
|
|
/* Public part */
|
|
linpool *lp; /* Linpool for in-parser allocations */
|
|
|
|
byte type; /* Last parsed type */
|
|
enum {
|
|
CPT_VARLEN = 1,
|
|
} tflags; /* Additional flags for the type / value pair */
|
|
u64 value; /* Last parsed (integer) value */
|
|
|
|
byte *target_buf; /* Target buf for CBOR_BYTES or CBOR_TEXT */
|
|
uint target_len; /* Set how many bytes to store */
|
|
|
|
const char *error; /* Error message */
|
|
|
|
/* Private part */
|
|
lp_state *flush; /* Linpool reset pointer */
|
|
|
|
enum { /* Multi-byte reader */
|
|
CPE_TYPE = 0,
|
|
CPE_READ_INT,
|
|
CPE_COMPLETE_INT,
|
|
CPE_READ_BYTE,
|
|
CPE_ITEM_DONE,
|
|
CPE_EXIT,
|
|
} partial_state;
|
|
|
|
u64 partial_countdown; /* How many items remaining in CBOR_ARRAY / CBOR_MAP */
|
|
|
|
uint stack_pos, stack_max; /* Nesting of CBOR_ARRAY / CBOR_MAP */
|
|
u64 stack_countdown[0];
|
|
};
|
|
|
|
struct cbor_parser_context *cbor_parser_new(pool *, uint stack_max_depth);
|
|
static inline void cbor_parser_free(struct cbor_parser_context *ctx)
|
|
{ rfree(ctx->lp); }
|
|
void cbor_parser_reset(struct cbor_parser_context *ctx);
|
|
|
|
enum cbor_parse_result {
|
|
CPR_ERROR = 0,
|
|
CPR_MORE,
|
|
CPR_MAJOR,
|
|
CPR_STR_END,
|
|
CPR_BLOCK_END,
|
|
} cbor_parse_byte(struct cbor_parser_context *, const byte);
|
|
bool cbor_parse_block_end(struct cbor_parser_context *);
|
|
|
|
#define CBOR_PARSE_IF(_ctx, _type, _target) if (((_ctx)->type == CBOR_##_type) && CBOR_STORE_##_type((_ctx), _target))
|
|
#define CBOR_PARSE_ONLY(_ctx, _type, _target) CBOR_PARSE_IF(_ctx, _type, _target) {} else CBOR_PARSER_ERROR("Expected %s for %s, got %s", #_type, #_target, cbor_type_str((_ctx)->type))
|
|
|
|
#define CBOR_STORE_POSINT(_ctx, _target) ((_target = (_ctx)->value), 1)
|
|
#define CBOR_STORE_NEGINT(_ctx, _target) ((_target = -1LL-(_ctx)->value), 1)
|
|
#define CBOR_STORE_BYTES(_ctx, _target) ({ \
|
|
if ((_ctx)->tflags & CPT_VARLEN) CBOR_PARSER_ERROR("Variable length string not supported yet"); \
|
|
if ((_target)) CBOR_PARSER_ERROR("Duplicate argument %s", #_target); \
|
|
ASSERT_DIE(!(_ctx)->target_buf); \
|
|
_target = (_ctx)->target_buf = lp_alloc((_ctx)->lp, ((_ctx)->target_len = (_ctx)->value) + 1); \
|
|
1; })
|
|
#define CBOR_STORE_TEXT CBOR_STORE_BYTES
|
|
|
|
|
|
/*
|
|
* Message channels
|
|
*/
|
|
|
|
struct cbor_channel;
|
|
typedef enum cbor_parse_result (*cbor_stream_parse_fn)(struct cbor_channel *, enum cbor_parse_result);
|
|
|
|
struct cbor_stream {
|
|
HASH(struct cbor_channel) channels;
|
|
pool *p;
|
|
slab *slab;
|
|
sock *s;
|
|
cbor_stream_parse_fn parse;
|
|
void (*cancel)(struct cbor_stream *);
|
|
struct cbor_channel *cur_rx_channel;
|
|
u64 hmul;
|
|
enum {
|
|
CSTR_INIT,
|
|
CSTR_EXPECT_ID,
|
|
CSTR_MSG,
|
|
CSTR_FINISH,
|
|
CSTR_CLEANUP,
|
|
} state;
|
|
uint writer_depth;
|
|
struct cbor_parser_context parser;
|
|
};
|
|
|
|
#define CBOR_STREAM_EMBED(name, N) struct { \
|
|
struct cbor_stream name; \
|
|
u64 _##name##_stack_countdown[N]; \
|
|
}
|
|
|
|
#define CBOR_STREAM_INIT(up, name, chname, p, T) \
|
|
cbor_stream_init(&(up)->name, p, \
|
|
ARRAY_SIZE((up)->_##name##_stack_countdown), \
|
|
ARRAY_SIZE(((T *) NULL)->_##chname##_writer_stack), \
|
|
sizeof(T))
|
|
|
|
/* Init and cleanup of CBOR stream */
|
|
void cbor_stream_init(struct cbor_stream *stream, pool *p, uint parser_depth, uint writer_depth, uint channel_size);
|
|
void cbor_stream_attach(struct cbor_stream *, sock *);
|
|
void cbor_stream_cleanup(struct cbor_stream *);
|
|
|
|
struct cbor_channel {
|
|
struct cbor_channel *next_hash;
|
|
struct cbor_stream *stream;
|
|
cbor_stream_parse_fn parse;
|
|
void (*cancel)(struct cbor_channel *);
|
|
pool *p;
|
|
u64 id;
|
|
u64 idhash;
|
|
struct cbor_writer writer;
|
|
};
|
|
|
|
#define CBOR_CHANNEL_EMBED(name, N) struct { \
|
|
struct cbor_channel name; \
|
|
struct cbor_writer_stack_item _##name##_writer_stack[N]; \
|
|
}
|
|
|
|
extern struct cbor_channel cbor_channel_parse_error;
|
|
|
|
/* Locally define a new channel */
|
|
struct cbor_channel *cbor_channel_new(struct cbor_stream *);
|
|
|
|
/* Create a channel with a pre-determined ID.
|
|
* You have to check nonexistence manually. */
|
|
struct cbor_channel *cbor_channel_create(struct cbor_stream *stream, u64 id);
|
|
/* Find an existing channel */
|
|
struct cbor_channel *cbor_channel_find(struct cbor_stream *, u64 id);
|
|
|
|
/* Drop the channel */
|
|
void cbor_channel_done(struct cbor_channel *);
|
|
|
|
struct cbor_writer *cbor_reply_init(struct cbor_channel *);
|
|
void cbor_reply_send(struct cbor_channel *, struct cbor_writer *);
|
|
#define CBOR_REPLY(ch, cw) for (struct cbor_writer *cw = cbor_reply_init(ch); cw; cbor_reply_send(ch, cw), cw = NULL)
|
|
|
|
|
|
#endif
|