You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
129 lines
3.9 KiB
129 lines
3.9 KiB
#ifndef PAYLOAD_BUILDER_H
|
|
#define PAYLOAD_BUILDER_H
|
|
|
|
/**
|
|
* PayloadBuilder, part of the TinyFrame utilities collection
|
|
*
|
|
* (c) Ondřej Hruška, 2014-2022. MIT license.
|
|
*
|
|
* The builder supports big and little endian which is selected when
|
|
* initializing it or by accessing the bigendian struct field.
|
|
*
|
|
* This module helps you with building payloads (not only for TinyFrame)
|
|
*
|
|
* The builder performs bounds checking and calls the provided handler when
|
|
* the requested write wouldn't fit. Use the handler to realloc / flush the buffer
|
|
* or report an error.
|
|
*/
|
|
|
|
#include <stdint.h>
|
|
#include <stdbool.h>
|
|
#include <stddef.h>
|
|
#include "type_coerce.h"
|
|
|
|
typedef struct PayloadBuilder_ PayloadBuilder;
|
|
|
|
/**
|
|
* Full buffer handler.
|
|
*
|
|
* 'needed' more bytes should be written but the end of the buffer was reached.
|
|
*
|
|
* Return true if the problem was solved (e.g. buffer was flushed and the
|
|
* 'current' pointer moved to the beginning).
|
|
*
|
|
* If false is returned, the 'ok' flag on the struct is set to false
|
|
* and all following writes are discarded.
|
|
*/
|
|
typedef bool (*pb_full_handler)(PayloadBuilder *pb, uint32_t needed);
|
|
|
|
struct PayloadBuilder_ {
|
|
uint8_t *start; //!< Pointer to the beginning of the buffer
|
|
uint8_t *current; //!< Pointer to the next byte to be read
|
|
uint8_t *end; //!< Pointer to the end of the buffer (start + length)
|
|
pb_full_handler full_handler; //!< Callback for buffer overrun
|
|
bool bigendian; //!< Flag to use big-endian parsing
|
|
bool ok; //!< Indicates that all reads were successful
|
|
};
|
|
|
|
// --- initializer helper macros ---
|
|
|
|
/** Start the builder. */
|
|
#define pb_start_e(buf, capacity, bigendian, full_handler) ((PayloadBuilder){buf, buf, (buf)+(capacity), full_handler, bigendian, 1})
|
|
|
|
/** Start the builder in big-endian mode */
|
|
#define pb_start_be(buf, capacity, full_handler) pb_start_e(buf, capacity, 1, full_handler)
|
|
|
|
/** Start the builder in little-endian mode */
|
|
#define pb_start_le(buf, capacity, full_handler) pb_start_e(buf, capacity, 0, full_handler)
|
|
|
|
/** Start the parser in little-endian mode (default) */
|
|
#define pb_start(buf, capacity, full_handler) pb_start_le(buf, capacity, full_handler)
|
|
|
|
// --- utilities ---
|
|
|
|
/** Returns true if the parser is still in OK state (not overrun) */
|
|
#define pb_ok(pb) ((pb)->ok)
|
|
|
|
/** Get already used bytes count */
|
|
#define pb_length(pb) ((pb)->current - (pb)->start)
|
|
|
|
/** Reset the current pointer to start */
|
|
#define pb_rewind(pb) do { (pb)->current = (pb)->start; } while (0)
|
|
|
|
/** save & restore position */
|
|
typedef uint8_t* pb_mark_t;
|
|
#define pb_save(pb) ((pb)->current)
|
|
#define pb_restore(pb, mark) do { (pb)->current = (mark); } while (0)
|
|
|
|
/** Write from a buffer */
|
|
bool pb_buf(PayloadBuilder *pb, const uint8_t *buf, uint32_t len);
|
|
|
|
/** Write a zero terminated string */
|
|
bool pb_string(PayloadBuilder *pb, const char *str);
|
|
|
|
/** Write uint8_t to the buffer */
|
|
bool pb_u8(PayloadBuilder *pb, uint8_t byte);
|
|
|
|
/** Write boolean to the buffer. */
|
|
static inline bool pb_bool(PayloadBuilder *pb, bool b)
|
|
{
|
|
return pb_u8(pb, (uint8_t) b);
|
|
}
|
|
|
|
/** Write uint16_t to the buffer. */
|
|
bool pb_u16(PayloadBuilder *pb, uint16_t word);
|
|
|
|
/** Write uint32_t to the buffer. */
|
|
bool pb_u32(PayloadBuilder *pb, uint32_t word);
|
|
|
|
/** Write int8_t to the buffer. */
|
|
static inline bool pb_i8(PayloadBuilder *pb, int8_t byte)
|
|
{
|
|
return pb_u8(pb, ((union conv8) {.i8 = byte}).u8);
|
|
}
|
|
|
|
/** Write char (int8_t) to the buffer. */
|
|
static inline bool pb_char(PayloadBuilder *pb, char c)
|
|
{
|
|
return pb_i8(pb, c);
|
|
}
|
|
|
|
/** Write int16_t to the buffer. */
|
|
static inline bool pb_i16(PayloadBuilder *pb, int16_t word)
|
|
{
|
|
return pb_u16(pb, ((union conv16) {.i16 = word}).u16);
|
|
}
|
|
|
|
/** Write int32_t to the buffer. */
|
|
static inline bool pb_i32(PayloadBuilder *pb, int32_t word)
|
|
{
|
|
return pb_u32(pb, ((union conv32) {.i32 = word}).u32);
|
|
}
|
|
|
|
/** Write 4-byte float to the buffer. */
|
|
static inline bool pb_float(PayloadBuilder *pb, float f)
|
|
{
|
|
return pb_u32(pb, ((union conv32) {.f32 = f}).u32);
|
|
}
|
|
|
|
#endif // PAYLOAD_BUILDER_H
|
|
|