QCBOR
|
#include <stdint.h>
#include <string.h>
#include <stddef.h>
Go to the source code of this file.
Data Structures | |
struct | q_useful_buf_c |
struct | q_useful_buf |
struct | useful_out_buf |
struct | useful_input_buf |
Macros | |
#define | NULLUsefulBufC ((UsefulBufC) {NULL, 0}) |
#define | NULLUsefulBuf ((UsefulBuf) {NULL, 0}) |
#define | UsefulBuf_FROM_SZ_LITERAL(szString) |
#define | UsefulBuf_FROM_BYTE_ARRAY_LITERAL(pBytes) |
#define | UsefulBuf_MAKE_STACK_UB(name, size) |
#define | UsefulBuf_FROM_BYTE_ARRAY(pBytes) |
#define | UsefulBufC_NTH_BYTE(UBC, n) |
Get the nth byte from a UsefulBufC. | |
#define | SZLiteralToUsefulBufC(szString) |
#define | MakeUsefulBufOnStack(name, size) |
#define | ByteArrayLiteralToUsefulBufC(pBytes) |
#define | SizeCalculateUsefulBuf ((UsefulBuf) {NULL, SIZE_MAX}) |
#define | UsefulOutBuf_MakeOnStack(name, size) |
#define | UIB_MAGIC (0xB00F) |
Typedefs | |
typedef struct q_useful_buf_c | UsefulBufC |
typedef struct q_useful_buf | UsefulBuf |
typedef struct useful_out_buf | UsefulOutBuf |
typedef struct useful_input_buf | UsefulInputBuf |
Functions | |
static int | UsefulBuf_IsNULL (UsefulBuf UB) |
Check if a UsefulBuf is NULLUsefulBuf or not. | |
static int | UsefulBuf_IsNULLC (UsefulBufC UB) |
Check if a UsefulBufC is NULLUsefulBufC or not. | |
static int | UsefulBuf_IsEmpty (UsefulBuf UB) |
Check if a UsefulBuf is empty or not. | |
static int | UsefulBuf_IsEmptyC (UsefulBufC UB) |
Check if a UsefulBufC is empty or not. | |
static int | UsefulBuf_IsNULLOrEmpty (UsefulBuf UB) |
Check if a UsefulBuf is NULLUsefulBuf or empty. | |
static int | UsefulBuf_IsNULLOrEmptyC (UsefulBufC UB) |
Check if a UsefulBufC is NULLUsefulBufC or empty. | |
static UsefulBufC | UsefulBuf_Const (const UsefulBuf UB) |
Convert a non-const UsefulBuf to a const UsefulBufC. | |
static UsefulBuf | UsefulBuf_Unconst (const UsefulBufC UBC) |
Convert a const UsefulBufC to a non-const UsefulBuf. | |
static UsefulBufC | UsefulBuf_FromSZ (const char *szString) |
Convert a NULL-terminated string to a UsefulBufC. | |
UsefulBufC | UsefulBuf_CopyOffset (UsefulBuf Dest, size_t uOffset, const UsefulBufC Src) |
Copy one UsefulBuf into another at an offset. | |
static UsefulBufC | UsefulBuf_Copy (UsefulBuf Dest, const UsefulBufC Src) |
Copy one UsefulBuf into another. | |
static UsefulBufC | UsefulBuf_Set (UsefulBuf pDest, uint8_t value) |
Set all bytes in a UsefulBuf to a value, for example to 0. | |
static UsefulBufC | UsefulBuf_CopyPtr (UsefulBuf Dest, const void *ptr, size_t uLen) |
Copy a pointer into a UsefulBuf. | |
static UsefulBufC | UsefulBuf_Head (UsefulBufC UB, size_t uAmount) |
Returns a truncation of a UsefulBufC. | |
static UsefulBufC | UsefulBuf_Tail (UsefulBufC UB, size_t uAmount) |
Returns bytes from the end of a UsefulBufC. | |
int | UsefulBuf_Compare (const UsefulBufC UB1, const UsefulBufC UB2) |
Compare one UsefulBufC to another. | |
size_t | UsefulBuf_IsValue (const UsefulBufC UB, uint8_t uValue) |
Find first byte that is not a particular byte value. | |
size_t | UsefulBuf_FindBytes (UsefulBufC BytesToSearch, UsefulBufC BytesToFind) |
Find one UsefulBufC in another. | |
UsefulBufC | UsefulBuf_SkipLeading (UsefulBufC String, uint8_t uByte) |
Skip leading bytes of a particular value in a string. | |
static size_t | UsefulBuf_PointerToOffset (UsefulBufC UB, const void *p) |
Convert a pointer to an offset with bounds checking. | |
static const void * | UsefulBuf_OffsetToPointer (UsefulBufC UB, size_t uOffset) |
Convert an offset to a pointer with bounds checking. | |
static UsefulBuf | UsefulBufC_Unconst (const UsefulBufC UBC) |
static uint32_t | UsefulBufUtil_CopyFloatToUint32 (float f) |
Copy a float to a uint32_t . | |
static uint64_t | UsefulBufUtil_CopyDoubleToUint64 (double d) |
Copy a double to a uint64_t . | |
static float | UsefulBufUtil_CopyUint32ToFloat (uint32_t u32) |
Copy a uint32_t to a float . | |
static double | UsefulBufUtil_CopyUint64ToDouble (uint64_t u64) |
Copy a uint64_t to a double . | |
void | UsefulOutBuf_Init (UsefulOutBuf *pUOutBuf, UsefulBuf Storage) |
Initialize and supply the output buffer. | |
static void | UsefulOutBuf_Reset (UsefulOutBuf *pUOutBuf) |
Reset a UsefulOutBuf for re use. | |
static size_t | UsefulOutBuf_GetEndPosition (UsefulOutBuf *pUOutBuf) |
Returns position of end of data in the UsefulOutBuf. | |
static int | UsefulOutBuf_AtStart (UsefulOutBuf *pUOutBuf) |
Returns whether any data has been added to the UsefulOutBuf. | |
void | UsefulOutBuf_InsertUsefulBuf (UsefulOutBuf *pUOutBuf, UsefulBufC NewData, size_t uPos) |
Inserts bytes into the UsefulOutBuf. | |
static void | UsefulOutBuf_InsertData (UsefulOutBuf *pUOutBuf, const void *pBytes, size_t uLen, size_t uPos) |
Insert a data buffer into the UsefulOutBuf. | |
static void | UsefulOutBuf_InsertString (UsefulOutBuf *pUOutBuf, const char *szString, size_t uPos) |
Insert a NULL-terminated string into the UsefulOutBuf. | |
static void | UsefulOutBuf_InsertByte (UsefulOutBuf *pUOutBuf, uint8_t byte, size_t uPos) |
Insert a byte into the UsefulOutBuf. | |
static void | UsefulOutBuf_InsertUint16 (UsefulOutBuf *pUOutBuf, uint16_t uInteger16, size_t uPos) |
Insert a 16-bit integer into the UsefulOutBuf. | |
static void | UsefulOutBuf_InsertUint32 (UsefulOutBuf *pUOutBuf, uint32_t uInteger32, size_t uPos) |
Insert a 32-bit integer into the UsefulOutBuf. | |
static void | UsefulOutBuf_InsertUint64 (UsefulOutBuf *pUOutBuf, uint64_t uInteger64, size_t uPos) |
Insert a 64-bit integer into the UsefulOutBuf. | |
static void | UsefulOutBuf_InsertFloat (UsefulOutBuf *pUOutBuf, float f, size_t uPos) |
Insert a float into the UsefulOutBuf. | |
static void | UsefulOutBuf_InsertDouble (UsefulOutBuf *pUOutBuf, double d, size_t uPos) |
Insert a double into the UsefulOutBuf. | |
static void | UsefulOutBuf_AppendUsefulBuf (UsefulOutBuf *pUOutBuf, UsefulBufC NewData) |
Append a UsefulBuf into the UsefulOutBuf. | |
static void | UsefulOutBuf_AppendData (UsefulOutBuf *pUOutBuf, const void *pBytes, size_t uLen) |
Append bytes to the UsefulOutBuf. | |
static void | UsefulOutBuf_AppendString (UsefulOutBuf *pUOutBuf, const char *szString) |
Append a NULL-terminated string to the UsefulOutBuf. | |
static void | UsefulOutBuf_AppendByte (UsefulOutBuf *pUOutBuf, uint8_t byte) |
Append a byte to the UsefulOutBuf. | |
static void | UsefulOutBuf_AppendUint16 (UsefulOutBuf *pUOutBuf, uint16_t uInteger16) |
Append an integer to the UsefulOutBuf. | |
static void | UsefulOutBuf_AppendUint32 (UsefulOutBuf *pUOutBuf, uint32_t uInteger32) |
Append an integer to the UsefulOutBuf. | |
static void | UsefulOutBuf_AppendUint64 (UsefulOutBuf *pUOutBuf, uint64_t uInteger64) |
Append an integer to the UsefulOutBuf. | |
static void | UsefulOutBuf_AppendFloat (UsefulOutBuf *pUOutBuf, float f) |
Append a float to the UsefulOutBuf. | |
static void | UsefulOutBuf_AppendDouble (UsefulOutBuf *pUOutBuf, double d) |
Append a double to the UsefulOutBuf. | |
static int | UsefulOutBuf_GetError (UsefulOutBuf *pUOutBuf) |
Returns the current error status. | |
static size_t | UsefulOutBuf_RoomLeft (UsefulOutBuf *pUOutBuf) |
Returns number of bytes unused used in the output buffer. | |
static int | UsefulOutBuf_WillItFit (UsefulOutBuf *pUOutBuf, size_t uLen) |
Returns 1 if some number of bytes will fit in the UsefulOutBuf. | |
static int | UsefulOutBuf_IsBufferNULL (UsefulOutBuf *pUOutBuf) |
Returns 1 if buffer given to UsefulOutBuf_Init() was NULL . | |
static UsefulBuf | UsefulOutBuf_GetOutPlace (UsefulOutBuf *pUOutBuf) |
Returns pointer and length of the output buffer not yet used. | |
void | UsefulOutBuf_Advance (UsefulOutBuf *pUOutBuf, size_t uAmount) |
Advance the amount output assuming it was written by the caller. | |
UsefulBufC | UsefulOutBuf_OutUBuf (UsefulOutBuf *pUOutBuf) |
Returns the data put into a UsefulOutBuf. | |
UsefulBufC | UsefulOutBuf_CopyOut (UsefulOutBuf *pUOutBuf, UsefulBuf Dest) |
Copy out the data put into a UsefulOutBuf. | |
UsefulBufC | UsefulOutBuf_OutUBufOffset (UsefulOutBuf *pUOutBuf, size_t uOffset) |
Returns data starting at an offset that was put into a UsefulOutBuf. | |
UsefulBufC | UsefulOutBuf_SubString (UsefulOutBuf *pUOutBuf, const size_t uStart, const size_t uLen) |
Return a substring of the output data. | |
static UsefulBuf | UsefulOutBuf_RetrieveOutputStorage (UsefulOutBuf *pUOutBuf) |
Retrieve the storage buffer passed in to UsefulOutBuf_Init(). | |
int | UsefulOutBuf_Compare (UsefulOutBuf *pUOutBuf, size_t uStart1, size_t uLen1, size_t uStart2, size_t uLen2) |
Compare bytes at offsets. | |
void | UsefulOutBuf_Swap (UsefulOutBuf *pUOutBuf, size_t uStartOffset, size_t uPivotOffset, size_t uEndOffset) |
Swap two regions of output bytes. | |
static void | UsefulInputBuf_Init (UsefulInputBuf *pUInBuf, UsefulBufC UB) |
Initialize the UsefulInputBuf structure before use. | |
static size_t | UsefulInputBuf_Tell (UsefulInputBuf *pUInBuf) |
Returns current position in input buffer. | |
static void | UsefulInputBuf_Seek (UsefulInputBuf *pUInBuf, size_t uPos) |
Sets the current position in input buffer. | |
static size_t | UsefulInputBuf_BytesUnconsumed (UsefulInputBuf *pUInBuf) |
Returns the number of bytes from the cursor to the end of the buffer, the unconsumed bytes. | |
static int | UsefulInputBuf_BytesAvailable (UsefulInputBuf *pUInBuf, size_t uLen) |
Check if there are unconsumed bytes. | |
static size_t | UsefulInputBuf_PointerToOffset (UsefulInputBuf *pUInBuf, const void *p) |
Convert a pointer to an offset with bounds checking. | |
static const void * | UsefulInputBuf_OffsetToPointer (UsefulInputBuf *pUInBuf, size_t uOffset) |
Convert an offset to a pointer with bounds checking. | |
const void * | UsefulInputBuf_GetBytes (UsefulInputBuf *pUInBuf, size_t uNum) |
Get pointer to bytes out of the input buffer. | |
static UsefulBufC | UsefulInputBuf_GetUsefulBuf (UsefulInputBuf *pUInBuf, size_t uNum) |
Get UsefulBuf out of the input buffer. | |
static uint8_t | UsefulInputBuf_GetByte (UsefulInputBuf *pUInBuf) |
Get a byte out of the input buffer. | |
static uint16_t | UsefulInputBuf_GetUint16 (UsefulInputBuf *pUInBuf) |
Get a uint16_t out of the input buffer. | |
static uint32_t | UsefulInputBuf_GetUint32 (UsefulInputBuf *pUInBuf) |
Get a uint32_t out of the input buffer. | |
static uint64_t | UsefulInputBuf_GetUint64 (UsefulInputBuf *pUInBuf) |
Get a uint64_t out of the input buffer. | |
static float | UsefulInputBuf_GetFloat (UsefulInputBuf *pUInBuf) |
Get a float out of the input buffer. | |
static double | UsefulInputBuf_GetDouble (UsefulInputBuf *pUInBuf) |
Get a double out of the input buffer. | |
static int | UsefulInputBuf_GetError (UsefulInputBuf *pUInBuf) |
Get the error status. | |
static size_t | UsefulInputBuf_GetBufferLength (UsefulInputBuf *pUInBuf) |
Gets the input buffer length. | |
static void | UsefulInputBuf_SetBufferLength (UsefulInputBuf *pUInBuf, size_t uNewLen) |
Alters the input buffer length (use with caution). | |
static UsefulBufC | UsefulInputBuf_RetrieveUndecodedInput (UsefulInputBuf *pUInBuf) |
Retrieve the undecoded input buffer. | |
int | UsefulInputBuf_Compare (UsefulInputBuf *pUInBuf, const size_t uOffset1, const size_t uLen1, const size_t uOffset2, const size_t uLen2) |
Compare two ranges of bytes somewhere in the input buffer. | |
The goal of this code is to make buffer and pointer manipulation easier and safer when working with binary data.
The UsefulBuf, UsefulOutBuf and UsefulInputBuf structures are used to represent buffers rather than ad hoc pointers and lengths.
With these it is possible to write code that does little or no direct pointer manipulation for copying and formatting data. For example, the QCBOR encoder was written using these and has less pointer manipulation.
While it is true that object code using these functions will be a little larger and slower than a white-knuckle clever use of pointers might be, but not by that much or enough to have an effect for most use cases. For security-oriented code this is highly worthwhile. Clarity, simplicity, reviewability and are more important.
There are some extra sanity and double checks in this code to help catch coding errors and simple memory corruption. They are helpful, but not a substitute for proper code review, input validation and such.
This code consists of a lot of inline functions and a few that are not. It should not generate very much object code, especially with the optimizer turned up to -Os
or -O3
.
#define ByteArrayLiteralToUsefulBufC | ( | pBytes | ) |
Deprecated macro; use UsefulBuf_FROM_BYTE_ARRAY_LITERAL instead
#define MakeUsefulBufOnStack | ( | name, | |
size ) |
Deprecated macro; use UsefulBuf_MAKE_STACK_UB instead
#define NULLUsefulBuf ((UsefulBuf) {NULL, 0}) |
A null UsefulBuf is one that has no memory associated the same way NULL
points to nothing. It does not matter what len
is.
#define NULLUsefulBufC ((UsefulBufC) {NULL, 0}) |
A null UsefulBufC is one that has no value in the same way a NULL
pointer has no value. A UsefulBufC is NULL
when the ptr
field is NULL
. It doesn't matter what len
is. See UsefulBuf_IsEmpty() for the distinction between null and empty.
#define SizeCalculateUsefulBuf ((UsefulBuf) {NULL, SIZE_MAX}) |
This is a UsefulBuf value that can be passed to UsefulOutBuf_Init() to have it calculate the size of the output buffer needed. Pass this for Storage
, call all the append and insert functions normally, then call UsefulOutBuf_OutUBuf(). The returned UsefulBufC has the size.
As one can see, this is just a NULL pointer and very large size. The NULL pointer tells UsefulOutBuf to not copy any data.
#define SZLiteralToUsefulBufC | ( | szString | ) |
Deprecated macro; use UsefulBuf_FROM_SZ_LITERAL instead
#define UIB_MAGIC (0xB00F) |
The actual value of magic
in UsefulInputBuf.
#define UsefulBuf_FROM_BYTE_ARRAY | ( | pBytes | ) |
Make a byte array in to a UsefulBuf. This is usually used on stack variables or static variables. Also see UsefulBuf_MAKE_STACK_UB.
#define UsefulBuf_FROM_BYTE_ARRAY_LITERAL | ( | pBytes | ) |
Convert a literal byte array to a UsefulBufC.
pBytes
must be a literal string that sizeof()
works on. It will not work on non-literal arrays.
#define UsefulBuf_FROM_SZ_LITERAL | ( | szString | ) |
Convert a literal string to a UsefulBufC.
szString
must be a literal string that sizeof()
works on. This is better for literal strings than UsefulBuf_FromSZ() because it generates less code. It will not work on non-literal strings.
The terminating \0 (NULL) is NOT included in the length!
#define UsefulBuf_MAKE_STACK_UB | ( | name, | |
size ) |
#define UsefulBufC_NTH_BYTE | ( | UBC, | |
n ) |
Get the nth byte from a UsefulBufC.
[in] | UBC | UsefulBufC from which to get byte |
[in] | n | Index of byte to get |
WARNING: this doesn't check that is
within the UsefulBufC. This point of this is to have the ugly cast in just one place.
#define UsefulOutBuf_MakeOnStack | ( | name, | |
size ) |
Convenience macro to make a UsefulOutBuf on the stack and initialize it with a stack buffer of the given size. The variable will be named name
.
typedef struct q_useful_buf UsefulBuf |
This non-const UsefulBuf is typically used for some allocated memory that is to be filled in. The len
is the amount of memory, not the length of the valid data in the buffer.
typedef struct q_useful_buf_c UsefulBufC |
UsefulBufC and UsefulBuf are simple data structures to hold a pointer and length for binary data. In C99 this data structure can be passed on the stack making a lot of code cleaner than carrying around a pointer and length as two parameters.
This is also conducive to secure coding practice as the length is always carried with the pointer and the convention for handling a pointer and a length is clear.
While it might be possible to write buffer and pointer code more efficiently in some use cases, the thought is that unless there is an extreme need for performance (e.g., you are building a gigabit-per-second IP router), it is probably better to have cleaner code you can be most certain about the security of.
The non-const UsefulBuf is usually used to refer an empty buffer to be filled in. The length is the size of the buffer.
The const UsefulBufC is usually used to refer to some data that has been filled in. The length is amount of valid data pointed to.
A common use mode is to pass a UsefulBuf to a function, the function puts some data in it, then the function returns a UsefulBufC refering to the data. The UsefulBuf is a non-const "in" parameter and the UsefulBufC is a const "out" parameter so the constness stays correct. There is no single "in,out" parameter (if there was, it would have to be non-const). Note that the pointer returned in the UsefulBufC usually ends up being the same pointer passed in as a UsefulBuf, though this is not striclty required.
A UsefulBuf is null, it has no value, when ptr
in it is NULL
.
There are functions and macros for the following:
See also UsefulOutBuf. It is a richer structure that has both the size of the valid data and the size of the buffer.
UsefulBuf is only 16 or 8 bytes on a 64- or 32-bit machine so it can go on the stack and be a function parameter or return value.
Another way to look at it is this. C has the NULL-terminated string as a means for handling text strings, but no means or convention for binary strings. Other languages do have such means, Rust, an efficient compiled language, for example.
UsefulBuf is kind of like the Useful Pot Pooh gave Eeyore on his birthday. Eeyore's balloon fits beautifully, "it goes in and out like anything".
typedef struct useful_input_buf UsefulInputBuf |
UsefulInputBuf is the counterpart to UsefulOutBuf. It is for parsing data received. Initialize it with the data from the network. Then use the functions like UsefulInputBuf_GetBytes() to get data chunks of various types. A position cursor is maintained internally.
As long as the functions here are used, there will never be any reference off the end of the given buffer (except UsefulInputBuf_SetBufferLength()). This is true even if they are called incorrectly, an attempt is made to seek off the end of the buffer or such. This makes it easier to write safe and correct code. For example, the QCBOR decoder implementation is safer and easier to review through its use of UsefulInputBuf.
UsefulInputBuf maintains an internal error state. The intended use is fetching data chunks without any error checks until the end. If there was any error, such as an attempt to fetch data off the end, the error state is entered and no further data will be returned. In the error state the UsefulInputBuf_GetXxxx()
functions return 0, or NULL
or NULLUsefulBufC. As long as null is not dereferenced, the error check can be put off until the end, simplifying the calling code.
The integer and float parsing expects network byte order (big endian). Network byte order is what is used by TCP/IP, CBOR and most internet protocols.
Lots of inline functions are used to keep code size down. The optimizer, particularly with the -Os
or -O3
, also reduces code size a lot. The only non-inline code is UsefulInputBuf_GetBytes(). It is less than 100 bytes so use of UsefulInputBuf doesn't add much code for all the messy hard-to-get right issues with parsing binary protocols in C that it solves.
The parse context size is:
typedef struct useful_out_buf UsefulOutBuf |
UsefulOutBuf is a structure and functions (an object) for serializing data into a buffer to encode for a network protocol or write data to a file.
The main idea is that all the pointer manipulation is performed by UsefulOutBuf functions so the caller doesn't have to do any pointer manipulation. The pointer manipulation is centralized. This code has been reviewed and written carefully so it spares the caller of much of this work and results in safer code with less effort.
The UsefulOutBuf methods that add data to the output buffer always check the length and will never write off the end of the output buffer. If an attempt to add data that will not fit is made, an internal error flag will be set and further attempts to add data will not do anything.
There is no way to ever write off the end of that buffer when calling the UsefulOutBuf_AddXxx()
and UsefulOutBuf_InsertXxx()
functions.
The functions to add data do not report success of failure. The caller only needs to check for an error in the final call, either UsefulOutBuf_OutUBuf() or UsefulOutBuf_CopyOut() to get the result. This makes the calling code cleaner.
There is a utility function to get the error status anytime along the way for a special circumstance. There are functions to see how much room is left and see if some data will fit too, but their use is generally unnecessary.
The general call flow is:
UsefulOutBuf can be used in a mode to calculate the size of what would be output without actually outputting anything. This is useful to calculate the size of a buffer that is to be allocated to hold the output. See SizeCalculateUsefulBuf.
Methods like UsefulOutBuf_InsertUint64() always output in network byte order (big endian).
The possible errors are:
Some inexpensive simple sanity checks are performed before every data addition to guard against use of an uninitialized or corrupted UsefulOutBuf.
UsefulOutBuf has been used to create a CBOR encoder. The CBOR encoder has almost no pointer manipulation in it, is easier to read, and easier to review.
A UsefulOutBuf is small and can go on the stack:
int UsefulBuf_Compare | ( | const UsefulBufC | UB1, |
const UsefulBufC | UB2 ) |
Compare one UsefulBufC to another.
[in] | UB1 | The first buffer to compare. |
[in] | UB2 | The second buffer to compare. |
Returns a negative value if UB1
if is less than UB2
. UB1
is less than UB2
if it is shorter or the first byte that is not the same is less.
Returns 0 if the inputs are the same.
Returns a positive value if UB2
is less than UB1
.
All that is of significance is that the result is positive, negative or 0. (This doesn't return the difference between the first non-matching byte like memcmp()
).
|
inlinestatic |
Convert a non-const UsefulBuf to a const UsefulBufC.
[in] | UB | The UsefulBuf to convert. |
|
inlinestatic |
Copy one UsefulBuf into another.
[in] | Dest | The destination buffer to copy into. |
[out] | Src | The source to copy from. |
This fails if Src.len
is greater than Dest.len
.
Note that like memcpy()
, the pointers are not checked and this will crash rather than return NULLUsefulBufC if they are NULL
or invalid.
The results are undefined if Dest
and Src
overlap.
UsefulBufC UsefulBuf_CopyOffset | ( | UsefulBuf | Dest, |
size_t | uOffset, | ||
const UsefulBufC | Src ) |
Copy one UsefulBuf into another at an offset.
[in] | Dest | Destination buffer to copy into. |
[in] | uOffset | The byte offset in Dest at which to copy to. |
[in] | Src | The bytes to copy. |
This fails and returns NULLUsefulBufC if offset
is beyond the size of Dest
.
This fails and returns NULLUsefulBufC if the Src
length plus uOffset
is greater than the length of Dest
.
The results are undefined if Dest
and Src
overlap.
This assumes that there is valid data in Dest
up to uOffset
. The UsefulBufC returned starts at the beginning of Dest
and goes to Src.len
+
uOffset
.
|
inlinestatic |
Copy a pointer into a UsefulBuf.
[in,out] | Dest | The destination buffer to copy into. |
[in] | ptr | The source to copy from. |
[in] | uLen | Length of the source; amount to copy. |
This fails and returns NULLUsefulBufC if uLen
is greater than pDest->len
.
Note that like memcpy()
, the pointers are not checked and this will crash, rather than return 1 if they are NULL
or invalid.
size_t UsefulBuf_FindBytes | ( | UsefulBufC | BytesToSearch, |
UsefulBufC | BytesToFind ) |
Find one UsefulBufC in another.
[in] | BytesToSearch | Buffer to search through. |
[in] | BytesToFind | Buffer with bytes to be found. |
SIZE_MAX
if not found.
|
inlinestatic |
Convert a NULL-terminated string to a UsefulBufC.
[in] | szString | The string to convert. |
UsefulBufC.ptr
points to the string so its lifetime must be maintained.
The terminating \0 (NULL) is NOT included in the length.
|
inlinestatic |
Returns a truncation of a UsefulBufC.
[in] | UB | The buffer to get the head of. |
[in] | uAmount | The number of bytes in the head. |
|
inlinestatic |
Check if a UsefulBuf is empty or not.
[in] | UB | The UsefulBuf to check. |
An "empty" UsefulBuf is one that has a value and can be considered to be set, but that value is of zero length. It is empty when len
is zero. It doesn't matter what the ptr
is.
Many uses will not need to clearly distinguish a NULL
UsefulBuf from an empty one and can have the ptr
NULL
and the len
0. However if a use of UsefulBuf needs to make a distinction then ptr
should not be NULL
when the UsefulBuf is considered empty, but not NULL
.
|
inlinestatic |
Check if a UsefulBufC is empty or not.
[in] | UB | The UsefulBufC to check. |
|
inlinestatic |
Check if a UsefulBuf is NULLUsefulBuf or not.
[in] | UB | The UsefulBuf to check. |
|
inlinestatic |
Check if a UsefulBufC is NULLUsefulBufC or not.
[in] | UB | The UsefulBufC to check. |
NULLUsefulBufC
, 0 if not.
|
inlinestatic |
Check if a UsefulBuf is NULLUsefulBuf or empty.
[in] | UB | The UsefulBuf to check. |
|
inlinestatic |
Check if a UsefulBufC is NULLUsefulBufC or empty.
[in] | UB | The UsefulBufC to check. |
size_t UsefulBuf_IsValue | ( | const UsefulBufC | UB, |
uint8_t | uValue ) |
Find first byte that is not a particular byte value.
[in] | UB | The destination buffer for byte comparison. |
[in] | uValue | The byte value to compare to. |
uValue
or SIZE_MAX
if all bytes are uValue
.Note that unlike most comparison functions, 0 does not indicate a successful comparison, so the test for match is:
UsefulBuf_IsValue(...) == SIZE_MAX
If UB
is null or empty, there is no match and 0 is returned.
|
inlinestatic |
Convert an offset to a pointer with bounds checking.
[in] | UB | A UsefulBuf. |
[in] | uOffset | Offset in pUInBuf . |
NULL
if uOffset
is out of range, a pointer into the buffer if not.
|
inlinestatic |
Convert a pointer to an offset with bounds checking.
[in] | UB | A UsefulBuf. |
[in] | p | Pointer to convert to offset. |
p
is out of range, the byte offset if not.
|
inlinestatic |
Set all bytes in a UsefulBuf to a value, for example to 0.
[in] | pDest | The destination buffer to copy into. |
[in] | value | The value to set the bytes to. |
Note that like memset()
, the pointer in pDest
is not checked and this will crash if NULL
or invalid.
UsefulBufC UsefulBuf_SkipLeading | ( | UsefulBufC | String, |
uint8_t | uByte ) |
Skip leading bytes of a particular value in a string.
[in] | String | The input string. String.ptr must not be NULL . |
[in] | uByte | The byte value. |
uByte
removed.
|
inlinestatic |
Returns bytes from the end of a UsefulBufC.
[in] | UB | The buffer to get the tail of. |
[in] | uAmount | The offset from the start where the tail is to begin. |
UB
or NULLUsefulBufC if uAmount
is greater than the length of the UsefulBufC.If UB.ptr
is NULL
, but UB.len
is not zero, then the result will be a UsefulBufC with a NULL
ptr
and len
with the length of the tail.
|
inlinestatic |
Convert a const UsefulBufC to a non-const UsefulBuf.
[in] | UBC | The UsefulBuf to convert. |
Use of this is not necessary for the intended use mode of UsefulBufC and UsefulBuf. In that mode, the UsefulBuf is created to describe a buffer that has not had any data put in it. Then the data is put in it. Then a UsefulBufC is create to describe the part with the data in it. This goes from non-const to const, so this function is not needed.
If the -Wcast-qual warning is enabled, this function can be used to avoid that warning.
|
inlinestatic |
Deprecated function; use UsefulBuf_Unconst() instead
|
inlinestatic |
Copy a double
to a uint64_t
.
[in] | d | Double value to copy. |
uint64_t
with the double bits.Convenience function to avoid type punning, compiler warnings and such. The optimizer usually reduces this to a simple assignment. This is a crusty corner of C.
|
inlinestatic |
Copy a float
to a uint32_t
.
[in] | f | Float value to copy. |
uint32_t
with the float bits.Convenience function to avoid type punning, compiler warnings and such. The optimizer usually reduces this to a simple assignment. This is a crusty corner of C.
|
inlinestatic |
Copy a uint32_t
to a float
.
[in] | u32 | Integer value to copy. |
float
.Convenience function to avoid type punning, compiler warnings and such. The optimizer usually reduces this to a simple assignment. This is a crusty corner of C.
|
inlinestatic |
Copy a uint64_t
to a double
.
[in] | u64 | Integer value to copy. |
double
.Convenience function to avoid type punning, compiler warnings and such. The optimizer usually reduces this to a simple assignment. This is a crusty corner of C.
|
inlinestatic |
Check if there are unconsumed bytes.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
[in] | uLen | Number of bytes to check availability for. |
uLen
bytes are available after the cursor, and 0 if not.
|
inlinestatic |
Returns the number of bytes from the cursor to the end of the buffer, the unconsumed bytes.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
Returns 0 if the cursor is invalid or corruption of the UsefulInputBuf structure is detected.
int UsefulInputBuf_Compare | ( | UsefulInputBuf * | pUInBuf, |
const size_t | uOffset1, | ||
const size_t | uLen1, | ||
const size_t | uOffset2, | ||
const size_t | uLen2 ) |
Compare two ranges of bytes somewhere in the input buffer.
[in] | pUInBuf | The input buffer. |
[in] | uOffset1 | Offset of first range of bytes. |
[in] | uLen1 | Length of first range of bytes. |
[in] | uOffset2 | Offset of second range of bytes. |
[in] | uLen2 | Length of second range of bytes. |
This returns the same as UsefulBuf_Compare().
If the offset or the length plus offset or a range extends outside the input buffer, that range of bytes will be considered greater than the other string. If both are outside this is considered a degenerate condition and the first string is considered larger.
This is a somewhat odd function of UsefulInputBuf as it is not used for consuming data. QCBOR uses it for map order and duplicate checking.
|
inlinestatic |
Gets the input buffer length.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
This returns the length of the input buffer set by UsefulInputBuf_Init() or UsefulInputBuf_SetBufferLength().
|
inlinestatic |
Get a byte out of the input buffer.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
This consumes 1 byte from the input buffer, returns it and advances the position cursor by 1.
If there is not 1 byte in the buffer, 0 will be returned for the byte and the error state is entered. To know if the 0 returned was in error or the real value, the error state must be checked. If possible, put this off until all values are retrieved to have smaller and simpler code, but if not possible UsefulInputBuf_GetError() can be called. Also, in the error state UsefulInputBuf_GetBytes() returns NULL
*or the ptr
from UsefulInputBuf_GetUsefulBuf() is NULL
.
const void * UsefulInputBuf_GetBytes | ( | UsefulInputBuf * | pUInBuf, |
size_t | uNum ) |
Get pointer to bytes out of the input buffer.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
[in] | uNum | Number of bytes to get. |
This consumes uNum
bytes from the input buffer. This returns a pointer to the start of the uNum
bytes.
If there are not uNum
bytes in the input buffer, NULL
will be returned and the error state is entered.
This advances the position cursor by uNum
bytes.
|
inlinestatic |
Get a double out of the input buffer.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
See UsefulInputBuf_GetByte(). This works the same, except it returns a double and eight bytes are consumed.
The input bytes are interpreted in network order (big endian).
|
inlinestatic |
Get the error status.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
This returns whether the UsefulInputBuf is in the error state or not.
The error state is entered for one of these reasons:
Once in the error state, it can only be cleared by calling UsefulInputBuf_Init().
For many use cases, it is possible to only call this once after all the UsefulInputBuf_GetXxxx()
calls have been made. This is possible if no reference to the data returned are needed before the error state is checked.
In some cases UsefulInputBuf_GetUsefulBuf() or UsefulInputBuf_GetBytes() can stand in for this because they return NULL
if the error state has been entered. (The others can't stand in because they don't return a clearly distinct error value.)
|
inlinestatic |
Get a float out of the input buffer.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
See UsefulInputBuf_GetByte(). This works the same, except it returns a float and four bytes are consumed.
The input bytes are interpreted in network order (big endian).
|
inlinestatic |
Get a uint16_t
out of the input buffer.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
uint16_t
.See UsefulInputBuf_GetByte(). This works the same, except it returns a uint16_t
and two bytes are consumed.
The input bytes are interpreted in network order (big endian).
|
inlinestatic |
Get a uint32_t
out of the input buffer.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
uint32_t
.See UsefulInputBuf_GetByte(). This works the same, except it returns a uint32_t
and four bytes are consumed.
The input bytes are interpreted in network order (big endian).
|
inlinestatic |
Get a uint64_t
out of the input buffer.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
See UsefulInputBuf_GetByte(). This works the same, except it returns a uint64_t
and eight bytes are consumed.
The input bytes are interpreted in network order (big endian).
|
inlinestatic |
Get UsefulBuf out of the input buffer.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
[in] | uNum | Number of bytes to get. |
This consumes uNum
bytes from the input buffer and returns the pointer and length for them as a UsefulBufC. The length returned will always be uNum
. The position cursor is advanced by uNum
bytes.
If there are not uNum
bytes in the input buffer, NULLUsefulBufC will be returned and the error state is entered.
|
inlinestatic |
Initialize the UsefulInputBuf structure before use.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
[in] | UB | The data to parse. |
|
inlinestatic |
Convert an offset to a pointer with bounds checking.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
[in] | uOffset | Offset in pUInBuf . |
NULL
if uOffset
is out of range, a pointer into the buffer if not.
|
inlinestatic |
Convert a pointer to an offset with bounds checking.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
[in] | p | Pointer to convert to offset. |
p
is out of range, the byte offset if not.
|
inlinestatic |
Retrieve the undecoded input buffer.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
A simple convenience method, should it be useful to get the original input back.
|
inlinestatic |
Sets the current position in input buffer.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
[in] | uPos | Position to set to. |
If the position is off the end of the input buffer, the error state is entered.
Seeking to a valid position in the buffer will not reset the error state. Only re-initialization will do that.
|
inlinestatic |
Alters the input buffer length (use with caution).
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
[in] | uNewLen | The new length of the input buffer. |
This alters the internal remembered length of the input buffer set when UsefulInputBuf_Init() was called.
The new length given here should always be equal to or less than the length given when UsefulInputBuf_Init() was called. Making it larger allows UsefulInputBuf to run off the input buffer.
The typical use is to set a length shorter than that when initialized to constrain parsing. If UsefulInputBuf_GetBufferLength() was called before this, then the original length can be restored with another call to this.
This should be used with caution. It is the only UsefulInputBuf method that can violate the safety of input buffer parsing.
|
inlinestatic |
Returns current position in input buffer.
[in] | pUInBuf | Pointer to the UsefulInputBuf. |
The position that the next bytes will be returned from.
void UsefulOutBuf_Advance | ( | UsefulOutBuf * | pUOutBuf, |
size_t | uAmount ) |
Advance the amount output assuming it was written by the caller.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | uAmount | The amount to advance. |
This advances the position in the output buffer by uAmount
. This assumes that the caller has written uAmount
to the pointer obtained with UsefulOutBuf_GetOutPlace().
Warning: this bypasses the buffer safety provided by UsefulOutBuf!
|
inlinestatic |
Append a byte to the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | byte | Bytes to append. |
See UsefulOutBuf_InsertByte() for details. This does the same with the insertion point at the end of the valid data.
|
inlinestatic |
Append bytes to the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | pBytes | Pointer to bytes to append. |
[in] | uLen | Length of pBytes to append. |
See UsefulOutBuf_InsertData() for details. This does the same with the insertion point at the end of the valid data.
|
inlinestatic |
Append a double
to the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | d | double to append. |
See UsefulOutBuf_InsertDouble() for details. This does the same with the insertion point at the end of the valid data.
The double will be appended in network byte order (big endian).
|
inlinestatic |
Append a float
to the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | f | float to append. |
See UsefulOutBuf_InsertFloat() for details. This does the same with the insertion point at the end of the valid data.
The float will be appended in network byte order (big endian).
|
inlinestatic |
Append a NULL-terminated string to the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | szString | NULL-terminated string to append. |
|
inlinestatic |
Append an integer to the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | uInteger16 | Integer to append. |
See UsefulOutBuf_InsertUint16() for details. This does the same with the insertion point at the end of the valid data.
The integer will be appended in network byte order (big endian).
|
inlinestatic |
Append an integer to the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | uInteger32 | Integer to append. |
See UsefulOutBuf_InsertUint32() for details. This does the same with the insertion point at the end of the valid data.
The integer will be appended in network byte order (big endian).
|
inlinestatic |
Append an integer to the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | uInteger64 | Integer to append. |
See UsefulOutBuf_InsertUint64() for details. This does the same with the insertion point at the end of the valid data.
The integer will be appended in network byte order (big endian).
|
inlinestatic |
Append a UsefulBuf into the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | NewData | The UsefulBuf with the bytes to append. |
See UsefulOutBuf_InsertUsefulBuf() for details. This does the same with the insertion point at the end of the valid data.
|
inlinestatic |
Returns whether any data has been added to the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
int UsefulOutBuf_Compare | ( | UsefulOutBuf * | pUOutBuf, |
size_t | uStart1, | ||
size_t | uLen1, | ||
size_t | uStart2, | ||
size_t | uLen2 ) |
Compare bytes at offsets.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | uStart1 | Offset of first bytes to compare. |
[in] | uLen1 | Length of first bytes to compare. |
[in] | uStart2 | Offset of second bytes to compare. |
[in] | uLen2 | Length of second bytes to compare. |
This looks into bytes that have been output at the offsets start1
and start2
. It compares bytes at those two starting points until they are not equal or uLen1
or uLen2
is reached. If the length of the string given is off the end of the output data, the string will be effectively truncated to the data in the output buffer for the comparison.
This returns positive when uStart1
lexographically sorts ahead of uStart2
and vice versa. Zero is returned if the strings compare equally.
If lengths are unequal and the first bytes are an exact subset of the second string, then a positve value will be returned and vice versa.
If either start is past the end of data in the output buffer, 0 will be returned. It is the caller's responsibility to make sure the offsets are not off the end so that a comparison is actually being made. No data will ever be read off the end of the buffer so this safe no matter what offsets are passed.
This is a relatively odd function in that it works on data in the output buffer. It is employed by QCBOR to sort CBOR-encoded maps that are in the output buffer.
UsefulBufC UsefulOutBuf_CopyOut | ( | UsefulOutBuf * | pUOutBuf, |
UsefulBuf | Dest ) |
Copy out the data put into a UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[out] | Dest | The destination buffer to copy into. |
NULLUsefulBufC
if it will not fit in the Dest
buffer or the error state was entered.This is the same as UsefulOutBuf_OutUBuf() except it copies the data to Dest
rather than returning a pointer.
|
inlinestatic |
Returns position of end of data in the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
On a freshly initialized UsefulOutBuf with no data added, this will return 0. After 10 bytes have been added, it will return 10 and so on.
Generally, there is no need to call this for most uses of UsefulOutBuf.
|
inlinestatic |
Returns the current error status.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
This returns the error status since a call to either UsefulOutBuf_Reset() of UsefulOutBuf_Init(). Once a UsefulOutBuf goes into the error state, it will stay until one of those functions is called.
Possible error conditions are:
|
inlinestatic |
Returns pointer and length of the output buffer not yet used.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
This is an escape that allows the caller to write directly to the output buffer without any checks. This doesn't change the output buffer or state. It just returns a pointer and length of the bytes remaining.
This is useful to avoid having the bytes to be added all in a contiguous buffer. Its use can save memory. A good example is in the COSE encrypt implementation where the output of the symmetric cipher can go directly into the output buffer, rather than having to go into an intermediate buffer.
See UsefulOutBuf_Advance() which is used to tell UsefulOutBuf how much was written.
Warning: this bypasses the buffer safety provided by UsefulOutBuf!
void UsefulOutBuf_Init | ( | UsefulOutBuf * | pUOutBuf, |
UsefulBuf | Storage ) |
Initialize and supply the output buffer.
[out] | pUOutBuf | The UsefulOutBuf to initialize. |
[in] | Storage | Buffer to output into. |
This initializes the UsefulOutBuf with storage, sets the current position to the beginning of the buffer and clears the error state.
See SizeCalculateUsefulBuf for instructions on how to initialize a UsefulOutBuf to calculate the size that would be output without actually outputting.
This must be called before the UsefulOutBuf is used.
|
inlinestatic |
Insert a byte into the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | byte | Bytes to insert. |
[in] | uPos | Index in output buffer at which to insert. |
See UsefulOutBuf_InsertUsefulBuf() for details. This is the same with the difference being a single byte is to be inserted.
|
inlinestatic |
Insert a data buffer into the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | pBytes | Pointer to the bytes to insert |
[in] | uLen | Length of the bytes to insert |
[in] | uPos | Index in output buffer at which to insert |
See UsefulOutBuf_InsertUsefulBuf() for details. This is the same with the difference being a pointer and length is passed in rather than an UsefulBufC.
|
inlinestatic |
Insert a double
into the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | d | double to insert. |
[in] | uPos | Index in output buffer at which to insert. |
See UsefulOutBuf_InsertUsefulBuf() for details. This is the same with the difference being a double
is to be inserted.
The double
will be inserted in network byte order (big endian).
|
inlinestatic |
Insert a float
into the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | f | float to insert. |
[in] | uPos | Index in output buffer at which to insert. |
See UsefulOutBuf_InsertUsefulBuf() for details. This is the same with the difference being a float
is to be inserted.
The float
will be inserted in network byte order (big endian).
|
inlinestatic |
Insert a NULL-terminated string into the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | szString | NULL-terminated string to insert. |
[in] | uPos | Index in output buffer at which to insert. |
|
inlinestatic |
Insert a 16-bit integer into the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | uInteger16 | Integer to insert. |
[in] | uPos | Index in output buffer at which to insert. |
See UsefulOutBuf_InsertUsefulBuf() for details. This is the same with the difference being a two-byte integer is to be inserted.
The integer will be inserted in network byte order (big endian).
|
inlinestatic |
Insert a 32-bit integer into the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | uInteger32 | Integer to insert. |
[in] | uPos | Index in output buffer at which to insert. |
See UsefulOutBuf_InsertUsefulBuf() for details. This is the same with the difference being a four-byte integer is to be inserted.
The integer will be inserted in network byte order (big endian).
|
inlinestatic |
Insert a 64-bit integer into the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | uInteger64 | Integer to insert. |
[in] | uPos | Index in output buffer at which to insert. |
See UsefulOutBuf_InsertUsefulBuf() for details. This is the same with the difference being an eight-byte integer is to be inserted.
The integer will be inserted in network byte order (big endian).
void UsefulOutBuf_InsertUsefulBuf | ( | UsefulOutBuf * | pUOutBuf, |
UsefulBufC | NewData, | ||
size_t | uPos ) |
Inserts bytes into the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | NewData | The bytes to insert. |
[in] | uPos | Index in output buffer at which to insert. |
NewData
is the pointer and length for the bytes to be added to the output buffer. There must be room in the output buffer for all of NewData
or an error will occur.
The insertion point must be between 0 and the current valid data. If not, an error will occur. Appending data to the output buffer is achieved by inserting at the end of the valid data. This can be retrieved by calling UsefulOutBuf_GetEndPosition().
When insertion is performed, the bytes between the insertion point and the end of data previously added to the output buffer are slid to the right to make room for the new data.
Overlapping buffers are OK. NewData
can point to data in the output buffer.
NewData.len may be 0 in which case nothing will be inserted.
If an error occurs, an error state is set in the UsefulOutBuf. No error is returned. All subsequent attempts to add data will do nothing.
The intended use is that all additions are made without checking for an error. The error will be taken into account when UsefulOutBuf_OutUBuf() returns NullUsefulBufC
. UsefulOutBuf_GetError() can also be called to check for an error.
|
inlinestatic |
Returns 1 if buffer given to UsefulOutBuf_Init() was NULL
.
[in] | pUOutBuf | Pointer to the UsefulOutBuf |
NULL
.Giving a NULL
output buffer to UsefulOutBuf_Init() is used when just calculating the length of the encoded data.
UsefulBufC UsefulOutBuf_OutUBuf | ( | UsefulOutBuf * | pUOutBuf | ) |
Returns the data put into a UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
The storage for the returned data is the Storage
parameter passed to UsefulOutBuf_Init(). See also UsefulOutBuf_CopyOut().
This can be called anytime and many times to get intermediate results. It doesn't change the data or reset the current position, so further data can be added.
UsefulBufC UsefulOutBuf_OutUBufOffset | ( | UsefulOutBuf * | pUOutBuf, |
size_t | uOffset ) |
Returns data starting at an offset that was put into a UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | uOffset | Offset to bytes to return. |
This is the same as UsefulOutBuf_OutUBuf() except a starting offset maybe specified. It returns the bytes starting at uOffset
to the end of what was encoded so far. Calling this with uOffset
0 is equivalent to UsefulOutBuf_OutUBuf().
If there's nothing at uOffset
or it is past the in the output buffer, a NULLUsefulBufC is returned.
This is typically not needed in typical use. It is used by QCBOR along with UsefulOutBuf_Compare() and UsefulOutBuf_Swap() for sorting CBOR maps.
|
inlinestatic |
Reset a UsefulOutBuf for re use.
[in] | pUOutBuf | Pointer to the UsefulOutBuf |
This sets the amount of data in the output buffer to none and clears the error state.
The output buffer is still the same one and size as from the UsefulOutBuf_Init() call.
This doesn't zero the data, just resets to 0 bytes of valid data.
|
inlinestatic |
Retrieve the storage buffer passed in to UsefulOutBuf_Init().
[in] | pUOutBuf | The encoding context. |
This doesn't give any information about how much has been encoded or the error state. It just returns the exact UsefulOutBuf given to UsefulOutBuf_Init().
|
inlinestatic |
Returns number of bytes unused used in the output buffer.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
Because of the error handling strategy and checks in UsefulOutBuf_InsertUsefulBuf() it is usually not necessary to use this.
UsefulBufC UsefulOutBuf_SubString | ( | UsefulOutBuf * | pUOutBuf, |
const size_t | uStart, | ||
const size_t | uLen ) |
Return a substring of the output data.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | uStart | Offset of start of substring. |
[in] | uLen | Length of substring. |
This is the same as UsefulOutBuf_OutUBuf(), but returns a substring. NULLUsefulBufC
is returned if the requested substring is off the end of the output bytes or if in error state.
void UsefulOutBuf_Swap | ( | UsefulOutBuf * | pUOutBuf, |
size_t | uStartOffset, | ||
size_t | uPivotOffset, | ||
size_t | uEndOffset ) |
Swap two regions of output bytes.
[in] | pUOutBuf | Pointer to the UsefulOutBuf. |
[in] | uStartOffset | Offset to start of bytes to be swapped. |
[in] | uPivotOffset | Offset to pivot around which bytes are swapped. |
[in] | uEndOffset | Offset to end of region to be swappe. |
This reaches into bytes that have been output and swaps two adjacent regions.
If any of the offsets are outside the range of valid data, no swapping will be performed. If the start is not the smallest and the pivot is not in the middle no swapping will be performed.
The byte at uStartOffset
will participate in the swapping. The byte at uEndOffset
will not participate in the swapping, only the byte before it.
This is a relatively odd function in that it works on data in the output buffer. It is employed by QCBOR to bubble sort encoded CBOR maps.
|
inlinestatic |
Returns 1 if some number of bytes will fit in the UsefulOutBuf.
[in] | pUOutBuf | Pointer to the UsefulOutBuf |
[in] | uLen | Number of bytes for which to check |
uLen
bytes will fit, 0 if not.Because of the error handling strategy and checks in UsefulOutBuf_InsertUsefulBuf() it is usually not necessary to use this.