libblobpack/blob.c
2016-01-14 15:13:19 +01:00

323 lines
10 KiB
C

/*
* blob - library for generating/parsing tagged binary data
*
* Copyright (C) 2010 Felix Fietkau <nbd@openwrt.org>
* Copyright (C) 2015 Martin Schröder <mkschreder.uk@gmail.com>
*
* Permission to use, copy, modify, and/or distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
#include "blob.h"
#define pack754_32(f) (pack754((f), 32, 8))
#define pack754_64(f) (pack754((f), 64, 11))
#define unpack754_32(i) (unpack754((i), 32, 8))
#define unpack754_64(i) (unpack754((i), 64, 11))
extern void blob_field_fill_pad(struct blob_field *attr);
extern void blob_field_set_raw_len(struct blob_field *attr, unsigned int len);
uint64_t pack754(long double f, unsigned bits, unsigned expbits);
long double unpack754(uint64_t i, unsigned bits, unsigned expbits);
static inline struct blob_field *blob_offset_to_attr(struct blob *buf, blob_offset_t offset){
void *ptr = (char *)buf->buf + (size_t)offset;
return ptr;
}
static inline blob_offset_t blob_field_to_offset(struct blob *buf, struct blob_field *attr){
return (blob_offset_t)((char *)attr - (char *) buf->buf);
}
static void blob_field_init(struct blob_field *attr, int id, unsigned int len){
assert(attr);
memset(attr, 0, sizeof(struct blob_field));
//len += sizeof(struct blob_field);
len &= BLOB_FIELD_LEN_MASK;
len |= (id << BLOB_FIELD_ID_SHIFT) & BLOB_FIELD_ID_MASK;
attr->id_len = cpu_to_be32(len);
}
//! Attepts to reallocate the buffer to fit the new payload data
bool blob_resize(struct blob *buf, int minlen){
assert(minlen > 0 && minlen < BLOB_MAX_SIZE);
char *new = 0;
int newsize = ((minlen / 256) + 1) * 256;
int cur_size = blob_size(buf);
// reallocate the memory of the buffer if we no longer have any memory left
if(newsize > buf->memlen){
new = realloc(buf->buf, newsize);
if (new) {
buf->buf = new;
memset(buf->buf + cur_size, 0, newsize - cur_size);
buf->memlen = newsize;
} else {
return false;
}
}
blob_field_set_raw_len(blob_head(buf), minlen);
blob_field_fill_pad(blob_head(buf));
return true;
}
void blob_reset(struct blob *buf){
assert(buf);
assert(buf->buf);
if(buf->memlen)
memset(buf->buf, 0, buf->memlen);
blob_field_init(blob_head(buf), BLOB_FIELD_ARRAY, sizeof(struct blob_field));
}
void blob_init(struct blob *buf, const char *data, size_t size){
memset(buf, 0, sizeof(struct blob));
// default buffer is 256 bytes block with zero sized data
buf->memlen = (size > 0)?size:256;
buf->buf = malloc(buf->memlen);
assert(buf->buf);
if(data) {
memcpy(buf->buf, data, size);
//blob_field_init(blob_head(buf), BLOB_FIELD_ARRAY, sizeof(struct blob_field));
//blob_field_fill_pad(blob_head(buf));
} else {
blob_reset(buf);
}
}
void blob_free(struct blob *buf){
free(buf->buf);
buf->buf = NULL;
buf->memlen = 0;
}
static struct blob_field *blob_new_attr(struct blob *buf, int id, int payload){
//int namelen = 0;
int attr_raw_len = sizeof(struct blob_field) + payload;
/*if(name) {
namelen = strlen(name)+1;
attr_raw_len += blob_name_hdrlen(namelen);
}*/
int attr_pad_len = attr_raw_len + (BLOB_FIELD_ALIGN - ( attr_raw_len % BLOB_FIELD_ALIGN)) % BLOB_FIELD_ALIGN;
struct blob_field *head = blob_head(buf);
int cur_len = blob_field_raw_pad_len(head);
int req_len = cur_len + attr_pad_len;
//DEBUG("head pad len %d raw len %d payload %d attr_pad_len %d\n", blob_field_pad_len(head), blob_field_raw_len(head), payload, attr_pad_len);
if (!blob_resize(buf, req_len))
return NULL;
//DEBUG("adding attr at offset %d - ", cur_len);
struct blob_field *attr = (struct blob_field*)(buf->buf + cur_len);
blob_field_init(attr, id, attr_raw_len);
blob_field_fill_pad(attr);
//DEBUG("pad len %d, raw len: %d req len %d\n", blob_field_pad_len(attr), blob_field_raw_len(attr), req_len);
// update the length of the head element to enclose the whole buffer
blob_field_set_raw_len(blob_head(buf), req_len);
/*if(name){
attr->id_len |= be32_to_cpu(BLOB_FIELD_HAS_NAME);
struct blob_name_hdr *hdr = (struct blob_name_hdr*)attr->data;
hdr->namelen = cpu_to_be16(namelen);
strncpy((char *) hdr->name, (const char *)name, namelen);
}*/
return attr;
}
struct blob_field *blob_put_binary(struct blob *buf, const void *ptr, unsigned int len){
assert(ptr);
struct blob_field *attr;
if (len < sizeof(struct blob_field) || !ptr)
return NULL;
attr = blob_new_attr(buf, BLOB_FIELD_BINARY, len - sizeof(struct blob_field));
if (!attr)
return NULL;
memcpy(attr->data, ptr, len);
return attr;
}
static struct blob_field *blob_put(struct blob *buf, int id, const void *ptr, unsigned int len){
struct blob_field *attr;
attr = blob_new_attr(buf, id, len);
if (!attr) {
return NULL;
}
if (ptr)
memcpy(blob_field_data(attr), ptr, len);
return attr;
}
struct blob_field *blob_put_string(struct blob *buf, const char *str){
assert(str);
return blob_put(buf, BLOB_FIELD_STRING, str, strlen(str) + 1);
}
struct blob_field *blob_put_u8(struct blob *buf, uint8_t val){
return blob_put(buf, BLOB_FIELD_INT8, &val, sizeof(val));
}
struct blob_field *blob_put_u16(struct blob *buf, uint16_t val){
val = cpu_to_be16(val);
return blob_put(buf, BLOB_FIELD_INT16, &val, sizeof(val));
}
struct blob_field *blob_put_u32(struct blob *buf, uint32_t val){
val = cpu_to_be32(val);
return blob_put(buf, BLOB_FIELD_INT32, &val, sizeof(val));
}
struct blob_field *blob_put_u64(struct blob *buf, uint64_t val){
val = cpu_to_be64(val);
return blob_put(buf, BLOB_FIELD_INT64, &val, sizeof(val));
}
struct blob_field *blob_put_bool(struct blob *buf, bool val){
return blob_put_u8(buf, val);
}
struct blob_field *blob_put_int(struct blob *self, long long val){
if(val >= INT8_MIN && val <= INT8_MAX) return blob_put_u8(self, val);
if(val >= INT16_MIN && val <= INT16_MAX) return blob_put_u16(self, val);
if(val >= INT32_MIN && val <= INT32_MAX) return blob_put_u32(self, val);
if(val >= INT64_MIN && val <= INT64_MAX) return blob_put_u64(self, val);
return NULL;
}
blob_offset_t blob_open_array(struct blob *buf){
struct blob_field *attr = blob_new_attr(buf, BLOB_FIELD_ARRAY, 0);
return blob_field_to_offset(buf, attr);
}
void blob_close_array(struct blob *buf, blob_offset_t offset){
struct blob_field *attr = blob_offset_to_attr(buf, offset);
int len = (buf->buf + blob_field_raw_len(blob_head(buf))) - (void*)attr;
blob_field_set_raw_len(attr, len);
}
blob_offset_t blob_open_table(struct blob *buf){
struct blob_field *attr = blob_new_attr(buf, BLOB_FIELD_TABLE, 0);
return blob_field_to_offset(buf, attr);
}
void blob_close_table(struct blob *buf, blob_offset_t offset){
struct blob_field *attr = blob_offset_to_attr(buf, offset);
int len = (buf->buf + blob_field_raw_len(blob_head(buf))) - (void*)attr;
blob_field_set_raw_len(attr, len);
}
struct blob_field *blob_put_float(struct blob *buf, double value){
uint32_t val = cpu_to_be32(pack754_32((float)value));
return blob_put(buf, BLOB_FIELD_FLOAT32, &val, sizeof(val));
}
struct blob_field *blob_put_double(struct blob *buf, double value){
uint64_t val = cpu_to_be64(pack754_64(value));
return blob_put(buf, BLOB_FIELD_FLOAT64, &val, sizeof(val));
}
struct blob_field *blob_put_real(struct blob *buf, double value){
return blob_put_double(buf, value);
}
struct blob_field *blob_put_attr(struct blob *buf, struct blob_field *attr){
if(!attr) return NULL;
struct blob_field *head = blob_head(buf);
int cur_len = blob_field_raw_pad_len(head);
int attr_pad_len = blob_field_raw_pad_len(attr);
int req_len = cur_len + attr_pad_len;
if (!blob_resize(buf, req_len))
return NULL;
struct blob_field *ret = (struct blob_field*)(buf->buf + cur_len);
memcpy(ret, attr, attr_pad_len);
blob_field_set_raw_len(blob_head(buf), req_len);
return ret;
}
static void __attribute__((unused)) _blob_field_dump(struct blob_field *node, int indent){
static const char *names[] = {
[BLOB_FIELD_INVALID] = "BLOB_FIELD_INVALID",
[BLOB_FIELD_BINARY] = "BLOB_FIELD_BINARY",
[BLOB_FIELD_STRING] = "BLOB_FIELD_STRING",
[BLOB_FIELD_INT8] = "BLOB_FIELD_INT8",
[BLOB_FIELD_INT16] = "BLOB_FIELD_INT16",
[BLOB_FIELD_INT32] = "BLOB_FIELD_INT32",
[BLOB_FIELD_INT64] = "BLOB_FIELD_INT64",
[BLOB_FIELD_FLOAT32] = "BLOB_FIELD_FLOAT32",
[BLOB_FIELD_FLOAT64] = "BLOB_FIELD_FLOAT64",
[BLOB_FIELD_ARRAY] = "BLOB_FIELD_ARRAY",
[BLOB_FIELD_TABLE] = "BLOB_FIELD_TABLE"
};
for(struct blob_field *attr = blob_field_first_child(node); attr; attr = blob_field_next_child(node, attr)){
char *data = (char*)attr; //blob_field_data(attr);
int id = blob_field_type(attr);
int len = blob_field_raw_pad_len(attr);
int offset = (node)?((int)((char*)attr - (char*)node)):0;
for(int c = 0; c < indent; c++) printf("\t");
printf("[ field (");
for(int c = 0; c < sizeof(struct blob_field); c++){
printf("%02x", (int)*((char*)attr + c) & 0xff);
}
printf(") type=%s offset=%d full padded len: %d, header+data: %d, data len: %d ]\n", names[(id < BLOB_FIELD_LAST)?id:0], offset, len, blob_field_raw_len(attr), blob_field_data_len(attr));
if(id == BLOB_FIELD_ARRAY || id == BLOB_FIELD_TABLE) {
_blob_field_dump(attr, indent+1);
continue;
}
printf("\t");
for(int c = 0; c < blob_field_raw_pad_len(attr); c++){
if(c > 0 && c % 10 == 0)
printf("\n\t");
printf(" %02x(%c)", data[c] & 0xff, (data[c] > 0x10 && data[c] < 128)?data[c]:'.');
}
printf("\n");
}
}
void blob_field_dump(struct blob_field *self){
_blob_field_dump(self, 0);
}
void blob_dump(struct blob *self){
if(!self) return;
printf("=========== blob ===========\n");
printf("size: %d, memory: %d\n", (int)blob_size(self), (int)self->memlen);
for(int c = 0; c < 8; c++) printf("%02x ", ((char*)self->buf)[c] & 0xff);
printf("\n");
_blob_field_dump(blob_head(self), 0);
printf("============================\n");
}