mirror of
https://github.com/vale981/libblobpack
synced 2025-03-05 09:51:42 -05:00
323 lines
10 KiB
C
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");
|
|
}
|