QCBOR
Loading...
Searching...
No Matches
UsefulBuf.h
Go to the documentation of this file.
1/* =========================================================================
2 * Copyright (c) 2016-2018, The Linux Foundation.
3 * Copyright (c) 2018-2024, Laurence Lundblade.
4 * Copyright (c) 2021, Arm Limited. All rights reserved.
5 * All rights reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions are
9 * met:
10 * * Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * * Redistributions in binary form must reproduce the above
13 * copyright notice, this list of conditions and the following
14 * disclaimer in the documentation and/or other materials provided
15 * with the distribution.
16 * * Neither the name of The Linux Foundation nor the names of its
17 * contributors, nor the name "Laurence Lundblade" may be used to
18 * endorse or promote products derived from this software without
19 * specific prior written permission.
20 *
21 * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
22 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
23 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
24 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
25 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
26 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
27 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
28 * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
29 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
30 * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
31 * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 * ========================================================================= */
33
34/*============================================================================
35 FILE: UsefulBuf.h
36
37 DESCRIPTION: General purpose input and output buffers
38
39 EDIT HISTORY FOR FILE:
40
41 This section contains comments describing changes made to the module.
42 Notice that changes are listed in reverse chronological order.
43
44 when who what, where, why
45 -------- ---- --------------------------------------------------
46 12/31/2024 llundblade Minor documentation tweaks for Doxygen.
47 08/31/2024 llundblade Add UsefulBufC_NTH_BYTE().
48 08/14/2024 llundblade Add UsefulOutBuf_RetrieveOutputStorage().
49 08/13/2024 llundblade Add UsefulInputBuf_RetrieveUndecodedInput().
50 8/10/2024 llundblade Add UsefulBuf_SkipLeading().
51 08/08/2024 llundblade Add UsefulOutBuf_SubString().
52 10/05/2024 llundblade Add Xxx_OffsetToPointer.
53 28/02/2024 llundblade Rearrange UsefulOutBuf_Compare().
54 1/7/2024 llundblade Add UsefulInputBuf_Compare().
55 19/11/2023 llundblade Add UsefulOutBuf_GetOutput().
56 19/11/2023 llundblade Add UsefulOutBuf_Swap().
57 19/11/2023 llundblade Add UsefulOutBuf_Compare().
58 19/12/2022 llundblade Document that adding empty data is allowed.
59 4/11/2022 llundblade Add GetOutPlace and Advance to UsefulOutBuf.
60 9/21/2021 llundbla Clarify UsefulOutBuf size calculation mode
61 8/8/2021 dthaler/llundbla Work with C++ without compiler extensions
62 5/11/2021 llundblade Improve comments and comment formatting.
63 3/6/2021 mcr/llundblade Fix warnings related to --Wcast-qual
64 2/17/2021 llundblade Add method to go from a pointer to an offset.
65 1/25/2020 llundblade Add some casts so static anlyzers don't complain.
66 5/21/2019 llundblade #define configs for efficient endianness handling.
67 5/16/2019 llundblade Add UsefulOutBuf_IsBufferNULL().
68 3/23/2019 llundblade Big documentation & style update. No interface
69 change.
70 3/6/2019 llundblade Add UsefulBuf_IsValue()
71 12/17/2018 llundblade Remove const from UsefulBuf and UsefulBufC .len
72 12/13/2018 llundblade Documentation improvements
73 09/18/2018 llundblade Cleaner distinction between UsefulBuf and
74 UsefulBufC.
75 02/02/18 llundbla Full support for integers in and out; fix pointer
76 alignment bug. Incompatible change: integers
77 in/out are now in network byte order.
78 08/12/17 llundbla Added UsefulOutBuf_AtStart and UsefulBuf_Find
79 06/27/17 llundbla Fix UsefulBuf_Compare() bug. Only affected
80 comparison for < or > for unequal length buffers.
81 Added UsefulBuf_Set() function.
82 05/30/17 llundbla Functions for NULL UsefulBufs and const / unconst
83 11/13/16 llundbla Initial Version.
84
85 =============================================================================*/
86
87#ifndef _UsefulBuf_h
88#define _UsefulBuf_h
89
90
91/*
92 * Endianness Configuration
93 *
94 * This code is written so it will work correctly on big- and
95 * little-endian CPUs without configuration or any auto-detection of
96 * endianness. All code here will run correctly regardless of the
97 * endianness of the CPU it is running on.
98 *
99 * There are four C preprocessor macros that can be set with #define
100 * to explicitly configure endianness handling. Setting them can
101 * reduce code size a little and improve efficiency a little.
102 *
103 * Note that most of QCBOR is unaffected by this configuration. Its
104 * endianness handling is integrated with the code that handles
105 * alignment and preferred serialization. This configuration does
106 * affect QCBOR's (planned) implementation of integer arrays (tagged
107 * arrays) and use of the functions here to serialize or deserialize
108 * integers and floating-point values.
109 *
110 * Following is the recipe for configuring the endianness-related
111 * #defines.
112 *
113 * The first option is to not define anything. This will work fine
114 * with all CPUs, OS's and compilers. The code for encoding integers
115 * may be a little larger and slower.
116 *
117 * If your CPU is big-endian then define
118 * USEFULBUF_CONFIG_BIG_ENDIAN. This will give the most efficient code
119 * for big-endian CPUs. It will be small and efficient because there
120 * will be no byte swapping.
121 *
122 * Try defining USEFULBUF_CONFIG_HTON. This will work on most CPUs,
123 * OS's and compilers, but not all. On big-endian CPUs this should
124 * give the most efficient code, the same as
125 * USEFULBUF_CONFIG_BIG_ENDIAN does. On little-endian CPUs it should
126 * call the system-defined byte swapping method which is presumably
127 * implemented efficiently. In some cases, this will be a dedicated
128 * byte swap instruction like Intel's bswap.
129 *
130 * If USEFULBUF_CONFIG_HTON works and you know your CPU is
131 * little-endian, it is also good to define
132 * USEFULBUF_CONFIG_LITTLE_ENDIAN.
133 *
134 * if USEFULBUF_CONFIG_HTON doesn't work and you know your system is
135 * little-endian, try defining both USEFULBUF_CONFIG_LITTLE_ENDIAN and
136 * USEFULBUF_CONFIG_BSWAP. This should call the most efficient
137 * system-defined byte swap method. However, note
138 * https://hardwarebug.org/2010/01/14/beware-the-builtins/. Perhaps
139 * this is fixed now. Often hton() and ntoh() will call the built-in
140 * __builtin_bswapXX()() function, so this size issue could affect
141 * USEFULBUF_CONFIG_HTON.
142 *
143 * Last, run the tests. They must all pass.
144 *
145 * These #define config options affect the inline implementation of
146 * UsefulOutBuf_InsertUint64() and UsefulInputBuf_GetUint64(). They
147 * also affect the 16-, 32-bit, float and double versions of these
148 * functions. Since they are inline, the size effect is not in the
149 * UsefulBuf object code, but in the calling code.
150 *
151 * Summary:
152 * USEFULBUF_CONFIG_BIG_ENDIAN -- Force configuration to big-endian.
153 * USEFULBUF_CONFIG_LITTLE_ENDIAN -- Force to little-endian.
154 * USEFULBUF_CONFIG_HTON -- Use hton(), htonl(), ntohl()... to
155 * handle big and little-endian with system option.
156 * USEFULBUF_CONFIG_BSWAP -- With USEFULBUF_CONFIG_LITTLE_ENDIAN,
157 * use __builtin_bswapXX().
158 *
159 * It is possible to run this code in environments where using floating point is
160 * not allowed. Defining USEFULBUF_DISABLE_ALL_FLOAT will disable all the code
161 * that is related to handling floating point types, along with related
162 * interfaces. This makes it possible to compile the code with the compile
163 * option -mgeneral-regs-only.
164 */
165
166#if defined(USEFULBUF_CONFIG_BIG_ENDIAN) && defined(USEFULBUF_CONFIG_LITTLE_ENDIAN)
167#error "Cannot define both USEFULBUF_CONFIG_BIG_ENDIAN and USEFULBUF_CONFIG_LITTLE_ENDIAN"
168#endif
169
170
171#include <stdint.h> /* for uint8_t, uint16_t.... */
172#include <string.h> /* for strlen, memcpy, memmove, memset */
173#include <stddef.h> /* for size_t */
174
175
176#ifdef USEFULBUF_CONFIG_HTON
177#include <arpa/inet.h> /* for htons, htonl, htonll, ntohs... */
178#endif
179
180#ifdef __cplusplus
181extern "C" {
182#if 0
183} /* Keep editor indention formatting happy */
184#endif
185#endif
186
280typedef struct q_useful_buf_c {
281 const void *ptr;
282 size_t len;
284
285
291typedef struct q_useful_buf {
292 void *ptr;
293 size_t len;
295
296
303/*
304 * NULLUsefulBufC and few other macros have to be
305 * definied differently in C than C++ because there
306 * is no common construct for a literal structure.
307 *
308 * In C compound literals are used.
309 *
310 * In C++ list initalization is used. This only works
311 * in C++11 and later.
312 *
313 * Note that some popular C++ compilers can handle compound
314 * literals with on-by-default extensions, however
315 * this code aims for full correctness with strict
316 * compilers so they are not used.
317 */
318#ifdef __cplusplus
319#define NULLUsefulBufC {NULL, 0}
320#else
321#define NULLUsefulBufC ((UsefulBufC) {NULL, 0})
322#endif
323
328#ifdef __cplusplus
329#define NULLUsefulBuf {NULL, 0}
330#else
331#define NULLUsefulBuf ((UsefulBuf) {NULL, 0})
332#endif
333
334
342static inline int UsefulBuf_IsNULL(UsefulBuf UB);
343
344
352static inline int UsefulBuf_IsNULLC(UsefulBufC UB);
353
354
372static inline int UsefulBuf_IsEmpty(UsefulBuf UB);
373
374
382static inline int UsefulBuf_IsEmptyC(UsefulBufC UB);
383
384
392static inline int UsefulBuf_IsNULLOrEmpty(UsefulBuf UB);
393
394
402static inline int UsefulBuf_IsNULLOrEmptyC(UsefulBufC UB);
403
404
412static inline UsefulBufC UsefulBuf_Const(const UsefulBuf UB);
413
414
432static inline UsefulBuf UsefulBuf_Unconst(const UsefulBufC UBC);
433
434
444#ifdef __cplusplus
445#define UsefulBuf_FROM_SZ_LITERAL(szString) {(szString), sizeof(szString)-1}
446#else
447#define UsefulBuf_FROM_SZ_LITERAL(szString) \
448 ((UsefulBufC) {(szString), sizeof(szString)-1})
449#endif
450
451
458#ifdef __cplusplus
459#define UsefulBuf_FROM_BYTE_ARRAY_LITERAL(pBytes) {(pBytes), sizeof(pBytes)}
460#else
461#define UsefulBuf_FROM_BYTE_ARRAY_LITERAL(pBytes) \
462 ((UsefulBufC) {(pBytes), sizeof(pBytes)})
463#endif
464
469#define UsefulBuf_MAKE_STACK_UB(name, size) \
470 uint8_t __pBuf##name[(size)];\
471 UsefulBuf name = {__pBuf##name , sizeof( __pBuf##name )}
472
473
479#ifdef __cplusplus
480#define UsefulBuf_FROM_BYTE_ARRAY(pBytes) {(pBytes), sizeof(pBytes)}
481#else
482#define UsefulBuf_FROM_BYTE_ARRAY(pBytes) \
483 ((UsefulBuf) {(pBytes), sizeof(pBytes)})
484#endif
485
486
499static inline UsefulBufC UsefulBuf_FromSZ(const char *szString);
500
501
511#define UsefulBufC_NTH_BYTE(UBC, n) (((const uint8_t *)(UBC.ptr))[n])
512
513
535UsefulBufC UsefulBuf_CopyOffset(UsefulBuf Dest, size_t uOffset, const UsefulBufC Src);
536
537
555static inline UsefulBufC UsefulBuf_Copy(UsefulBuf Dest, const UsefulBufC Src);
556
557
567static inline UsefulBufC UsefulBuf_Set(UsefulBuf pDest, uint8_t value);
568
569
586static inline UsefulBufC UsefulBuf_CopyPtr(UsefulBuf Dest,
587 const void *ptr,
588 size_t uLen);
589
590
599static inline UsefulBufC UsefulBuf_Head(UsefulBufC UB, size_t uAmount);
600
601
615static inline UsefulBufC UsefulBuf_Tail(UsefulBufC UB, size_t uAmount);
616
617
638int UsefulBuf_Compare(const UsefulBufC UB1, const UsefulBufC UB2);
639
640
659size_t UsefulBuf_IsValue(const UsefulBufC UB, uint8_t uValue);
660
661
670size_t UsefulBuf_FindBytes(UsefulBufC BytesToSearch, UsefulBufC BytesToFind);
671
672
682
683
692static inline size_t UsefulBuf_PointerToOffset(UsefulBufC UB, const void *p);
693
694
703static inline const void *UsefulBuf_OffsetToPointer(UsefulBufC UB, size_t uOffset);
704
705
706#ifndef USEFULBUF_DISABLE_DEPRECATED
708#define SZLiteralToUsefulBufC(szString) UsefulBuf_FROM_SZ_LITERAL(szString)
709
711#define MakeUsefulBufOnStack(name, size) \
712 uint8_t __pBuf##name[(size)];\
713 UsefulBuf name = {__pBuf##name , sizeof( __pBuf##name )}
714
716#define ByteArrayLiteralToUsefulBufC(pBytes) \
717 UsefulBuf_FROM_BYTE_ARRAY_LITERAL(pBytes)
718
721{
722 UsefulBuf UB;
723
724 /* See UsefulBuf_Unconst() implementation for comment */
725 UB.ptr = (void *)(uintptr_t)UBC.ptr;
726
727 UB.len = UBC.len;
728
729 return UB;
730}
731#endif /* USEFULBUF_DISABLE_DEPRECATED */
732
733
734
735
736#ifndef USEFULBUF_DISABLE_ALL_FLOAT
748static inline uint32_t UsefulBufUtil_CopyFloatToUint32(float f);
749
750
762static inline uint64_t UsefulBufUtil_CopyDoubleToUint64(double d);
763
764
776static inline float UsefulBufUtil_CopyUint32ToFloat(uint32_t u32);
777
778
790static inline double UsefulBufUtil_CopyUint64ToDouble(uint64_t u64);
791#endif /* USEFULBUF_DISABLE_ALL_FLOAT */
792
793
794
795
875typedef struct useful_out_buf {
877 UsefulBuf UB;
879 size_t data_len;
881 uint16_t magic;
883 uint8_t err;
885
886
897#ifdef __cplusplus
898#define SizeCalculateUsefulBuf {NULL, SIZE_MAX}
899#else
900#define SizeCalculateUsefulBuf ((UsefulBuf) {NULL, SIZE_MAX})
901#endif
902
903
920void UsefulOutBuf_Init(UsefulOutBuf *pUOutBuf, UsefulBuf Storage);
921
922
928#define UsefulOutBuf_MakeOnStack(name, size) \
929 uint8_t __pBuf##name[(size)];\
930 UsefulOutBuf name;\
931 UsefulOutBuf_Init(&(name), (UsefulBuf){__pBuf##name, (size)});
932
933
947static inline void UsefulOutBuf_Reset(UsefulOutBuf *pUOutBuf);
948
949
964static inline size_t UsefulOutBuf_GetEndPosition(UsefulOutBuf *pUOutBuf);
965
966
974static inline int UsefulOutBuf_AtStart(UsefulOutBuf *pUOutBuf);
975
976
1012 UsefulBufC NewData,
1013 size_t uPos);
1014
1015
1028static inline void UsefulOutBuf_InsertData(UsefulOutBuf *pUOutBuf,
1029 const void *pBytes,
1030 size_t uLen,
1031 size_t uPos);
1032
1033
1041static inline void UsefulOutBuf_InsertString(UsefulOutBuf *pUOutBuf,
1042 const char *szString,
1043 size_t uPos);
1044
1045
1056static inline void UsefulOutBuf_InsertByte(UsefulOutBuf *pUOutBuf,
1057 uint8_t byte,
1058 size_t uPos);
1059
1060
1073static inline void UsefulOutBuf_InsertUint16(UsefulOutBuf *pUOutBuf,
1074 uint16_t uInteger16,
1075 size_t uPos);
1076
1077
1090static inline void UsefulOutBuf_InsertUint32(UsefulOutBuf *pUOutBuf,
1091 uint32_t uInteger32,
1092 size_t uPos);
1093
1094
1107static inline void UsefulOutBuf_InsertUint64(UsefulOutBuf *pUOutBuf,
1108 uint64_t uInteger64,
1109 size_t uPos);
1110
1111
1112#ifndef USEFULBUF_DISABLE_ALL_FLOAT
1125static inline void UsefulOutBuf_InsertFloat(UsefulOutBuf *pUOutBuf,
1126 float f,
1127 size_t uPos);
1128
1129
1142static inline void UsefulOutBuf_InsertDouble(UsefulOutBuf *pUOutBuf,
1143 double d,
1144 size_t uPos);
1145#endif /* USEFULBUF_DISABLE_ALL_FLOAT */
1146
1147
1157static inline void UsefulOutBuf_AppendUsefulBuf(UsefulOutBuf *pUOutBuf,
1158 UsefulBufC NewData);
1159
1160
1171static inline void UsefulOutBuf_AppendData(UsefulOutBuf *pUOutBuf,
1172 const void *pBytes,
1173 size_t uLen);
1174
1175
1182static inline void UsefulOutBuf_AppendString(UsefulOutBuf *pUOutBuf,
1183 const char *szString);
1184
1185
1195static inline void UsefulOutBuf_AppendByte(UsefulOutBuf *pUOutBuf,
1196 uint8_t byte);
1197
1198
1210static inline void UsefulOutBuf_AppendUint16(UsefulOutBuf *pUOutBuf,
1211 uint16_t uInteger16);
1212
1213
1225static inline void UsefulOutBuf_AppendUint32(UsefulOutBuf *pUOutBuf,
1226 uint32_t uInteger32);
1227
1228
1240static inline void UsefulOutBuf_AppendUint64(UsefulOutBuf *pUOutBuf,
1241 uint64_t uInteger64);
1242
1243
1244#ifndef USEFULBUF_DISABLE_ALL_FLOAT
1256static inline void UsefulOutBuf_AppendFloat(UsefulOutBuf *pUOutBuf,
1257 float f);
1258
1259
1271static inline void UsefulOutBuf_AppendDouble(UsefulOutBuf *pUOutBuf,
1272 double d);
1273#endif /* USEFULBUF_DISABLE_ALL_FLOAT */
1274
1275
1294static inline int UsefulOutBuf_GetError(UsefulOutBuf *pUOutBuf);
1295
1296
1308static inline size_t UsefulOutBuf_RoomLeft(UsefulOutBuf *pUOutBuf);
1309
1310
1323static inline int UsefulOutBuf_WillItFit(UsefulOutBuf *pUOutBuf, size_t uLen);
1324
1325
1336static inline int UsefulOutBuf_IsBufferNULL(UsefulOutBuf *pUOutBuf);
1337
1338
1364static inline UsefulBuf
1366
1367
1382void
1383UsefulOutBuf_Advance(UsefulOutBuf *pUOutBuf, size_t uAmount);
1384
1385
1402
1403
1418
1419
1441UsefulOutBuf_OutUBufOffset(UsefulOutBuf *pUOutBuf, size_t uOffset);
1442
1443
1456 const size_t uStart,
1457 const size_t uLen);
1458
1459
1472
1473
1512 size_t uStart1, size_t uLen1,
1513 size_t uStart2, size_t uLen2);
1514
1539 size_t uStartOffset,
1540 size_t uPivotOffset,
1541 size_t uEndOffset);
1542
1543
1544
1545
1586typedef struct useful_input_buf {
1588 UsefulBufC UB;
1590 size_t cursor;
1592 uint16_t magic;
1594 uint8_t err;
1596
1598#define UIB_MAGIC (0xB00F)
1599
1600
1607static inline void UsefulInputBuf_Init(UsefulInputBuf *pUInBuf, UsefulBufC UB);
1608
1609
1619static size_t UsefulInputBuf_Tell(UsefulInputBuf *pUInBuf);
1620
1621
1634static void UsefulInputBuf_Seek(UsefulInputBuf *pUInBuf, size_t uPos);
1635
1636
1648static size_t UsefulInputBuf_BytesUnconsumed(UsefulInputBuf *pUInBuf);
1649
1650
1659static int UsefulInputBuf_BytesAvailable(UsefulInputBuf *pUInBuf, size_t uLen);
1660
1661
1670static size_t UsefulInputBuf_PointerToOffset(UsefulInputBuf *pUInBuf, const void *p);
1671
1672
1681static const void *UsefulInputBuf_OffsetToPointer(UsefulInputBuf *pUInBuf, size_t uOffset);
1682
1683
1700const void * UsefulInputBuf_GetBytes(UsefulInputBuf *pUInBuf, size_t uNum);
1701
1702
1719static inline UsefulBufC UsefulInputBuf_GetUsefulBuf(UsefulInputBuf *pUInBuf, size_t uNum);
1720
1721
1741static inline uint8_t UsefulInputBuf_GetByte(UsefulInputBuf *pUInBuf);
1742
1743
1756static inline uint16_t UsefulInputBuf_GetUint16(UsefulInputBuf *pUInBuf);
1757
1758
1771static uint32_t UsefulInputBuf_GetUint32(UsefulInputBuf *pUInBuf);
1772
1773
1786static uint64_t UsefulInputBuf_GetUint64(UsefulInputBuf *pUInBuf);
1787
1788
1789#ifndef USEFULBUF_DISABLE_ALL_FLOAT
1802static float UsefulInputBuf_GetFloat(UsefulInputBuf *pUInBuf);
1803
1804
1817static double UsefulInputBuf_GetDouble(UsefulInputBuf *pUInBuf);
1818#endif /* USEFULBUF_DISABLE_ALL_FLOAT */
1819
1820
1850static int UsefulInputBuf_GetError(UsefulInputBuf *pUInBuf);
1851
1852
1863static inline size_t UsefulInputBuf_GetBufferLength(UsefulInputBuf *pUInBuf);
1864
1865
1888static void UsefulInputBuf_SetBufferLength(UsefulInputBuf *pUInBuf, size_t uNewLen);
1889
1890
1900
1901
1922int
1924 const size_t uOffset1,
1925 const size_t uLen1,
1926 const size_t uOffset2,
1927 const size_t uLen2);
1928
1929
1930
1931
1932/*----------------------------------------------------------
1933 Inline implementations.
1934 */
1935static inline int UsefulBuf_IsNULL(UsefulBuf UB)
1936{
1937 return !UB.ptr;
1938}
1939
1940
1941static inline int UsefulBuf_IsNULLC(UsefulBufC UB)
1942{
1943 return !UB.ptr;
1944}
1945
1946
1947static inline int UsefulBuf_IsEmpty(UsefulBuf UB)
1948{
1949 return !UB.len;
1950}
1951
1952
1953static inline int UsefulBuf_IsEmptyC(UsefulBufC UB)
1954{
1955 return !UB.len;
1956}
1957
1958
1960{
1961 return UsefulBuf_IsEmpty(UB) || UsefulBuf_IsNULL(UB);
1962}
1963
1964
1966{
1967 return UsefulBuf_IsEmptyC(UB) || UsefulBuf_IsNULLC(UB);
1968}
1969
1970
1971static inline UsefulBufC UsefulBuf_Const(const UsefulBuf UB)
1972{
1973 UsefulBufC UBC;
1974 UBC.ptr = UB.ptr;
1975 UBC.len = UB.len;
1976
1977 return UBC;
1978}
1979
1981{
1982 UsefulBuf UB;
1983
1984 /* -Wcast-qual is a good warning flag to use in general. This is
1985 * the one place in UsefulBuf where it needs to be quieted.
1986 */
1987 UB.ptr = (void *)(uintptr_t)UBC.ptr;
1988
1989 UB.len = UBC.len;
1990
1991 return UB;
1992}
1993
1994
1995static inline UsefulBufC UsefulBuf_FromSZ(const char *szString)
1996{
1997 UsefulBufC UBC;
1998 UBC.ptr = szString;
1999 UBC.len = strlen(szString);
2000 return UBC;
2001}
2002
2003
2004static inline UsefulBufC UsefulBuf_Copy(UsefulBuf Dest, const UsefulBufC Src)
2005{
2006 return UsefulBuf_CopyOffset(Dest, 0, Src);
2007}
2008
2009
2010static inline UsefulBufC UsefulBuf_Set(UsefulBuf Dest, uint8_t value)
2011{
2012 memset(Dest.ptr, value, Dest.len);
2013
2014 UsefulBufC UBC;
2015 UBC.ptr = Dest.ptr;
2016 UBC.len = Dest.len;
2017
2018 return UBC;
2019}
2020
2021
2022static inline UsefulBufC UsefulBuf_CopyPtr(UsefulBuf Dest, const void *ptr, size_t len)
2023{
2024 UsefulBufC UBC;
2025 UBC.ptr = ptr;
2026 UBC.len = len;
2027 return UsefulBuf_Copy(Dest, UBC);
2028}
2029
2030
2031static inline UsefulBufC UsefulBuf_Head(UsefulBufC UB, size_t uAmount)
2032{
2033 if(uAmount > UB.len) {
2034 return NULLUsefulBufC;
2035 }
2036 UsefulBufC UBC;
2037
2038 UBC.ptr = UB.ptr;
2039 UBC.len = uAmount;
2040
2041 return UBC;
2042}
2043
2044
2045static inline UsefulBufC UsefulBuf_Tail(UsefulBufC UB, size_t uAmount)
2046{
2047 UsefulBufC ReturnValue;
2048
2049 if(uAmount > UB.len) {
2050 ReturnValue = NULLUsefulBufC;
2051 } else if(UB.ptr == NULL) {
2052 ReturnValue.ptr = NULL;
2053 ReturnValue.len = UB.len - uAmount;
2054 } else {
2055 ReturnValue.ptr = (const uint8_t *)UB.ptr + uAmount;
2056 ReturnValue.len = UB.len - uAmount;
2057 }
2058
2059 return ReturnValue;
2060}
2061
2062
2063static inline size_t UsefulBuf_PointerToOffset(UsefulBufC UB, const void *p)
2064{
2065 if(UB.ptr == NULL) {
2066 return SIZE_MAX;
2067 }
2068
2069 if(p < UB.ptr) {
2070 /* given pointer is before start of buffer */
2071 return SIZE_MAX;
2072 }
2073
2074 /* Cast to size_t (from ptrdiff_t) is OK because of check above */
2075 const size_t uOffset = (size_t)((const uint8_t *)p - (const uint8_t *)UB.ptr);
2076
2077 if(uOffset >= UB.len) {
2078 /* given pointer is off the end of the buffer */
2079 return SIZE_MAX;
2080 }
2081
2082 return uOffset;
2083}
2084
2085
2086static inline const void *UsefulBuf_OffsetToPointer(UsefulBufC UB, size_t uOffset)
2087{
2088 if(UsefulBuf_IsNULLC(UB) || uOffset >= UB.len) {
2089 return NULL;
2090 }
2091
2092 return (const uint8_t *)UB.ptr + uOffset;
2093}
2094
2095
2096
2097
2098#ifndef USEFULBUF_DISABLE_ALL_FLOAT
2099static inline uint32_t UsefulBufUtil_CopyFloatToUint32(float f)
2100{
2101 uint32_t u32;
2102 memcpy(&u32, &f, sizeof(uint32_t));
2103 return u32;
2104}
2105
2106static inline uint64_t UsefulBufUtil_CopyDoubleToUint64(double d)
2107{
2108 uint64_t u64;
2109 memcpy(&u64, &d, sizeof(uint64_t));
2110 return u64;
2111}
2112
2113static inline double UsefulBufUtil_CopyUint64ToDouble(uint64_t u64)
2114{
2115 double d;
2116 memcpy(&d, &u64, sizeof(uint64_t));
2117 return d;
2118}
2119
2120static inline float UsefulBufUtil_CopyUint32ToFloat(uint32_t u32)
2121{
2122 float f;
2123 memcpy(&f, &u32, sizeof(uint32_t));
2124 return f;
2125}
2126#endif /* USEFULBUF_DISABLE_ALL_FLOAT */
2127
2128
2129
2130
2131static inline void UsefulOutBuf_Reset(UsefulOutBuf *pMe)
2132{
2133 pMe->data_len = 0;
2134 pMe->err = 0;
2135}
2136
2137
2139{
2140 return pMe->data_len;
2141}
2142
2143
2144static inline int UsefulOutBuf_AtStart(UsefulOutBuf *pMe)
2145{
2146 return 0 == pMe->data_len;
2147}
2148
2149
2151 const void *pBytes,
2152 size_t uLen,
2153 size_t uPos)
2154{
2155 UsefulBufC Data = {pBytes, uLen};
2156 UsefulOutBuf_InsertUsefulBuf(pMe, Data, uPos);
2157}
2158
2159
2161 const char *szString,
2162 size_t uPos)
2163{
2164 UsefulBufC UBC;
2165 UBC.ptr = szString;
2166 UBC.len = strlen(szString);
2167
2168 UsefulOutBuf_InsertUsefulBuf(pMe, UBC, uPos);
2169}
2170
2171
2173 uint8_t byte,
2174 size_t uPos)
2175{
2176 UsefulOutBuf_InsertData(me, &byte, 1, uPos);
2177}
2178
2179
2181 uint16_t uInteger16,
2182 size_t uPos)
2183{
2184 /* See UsefulOutBuf_InsertUint64() for comments on this code */
2185
2186 const void *pBytes;
2187
2188#if defined(USEFULBUF_CONFIG_BIG_ENDIAN)
2189 pBytes = &uInteger16;
2190
2191#elif defined(USEFULBUF_CONFIG_HTON)
2192 uint16_t uTmp = htons(uInteger16);
2193 pBytes = &uTmp;
2194
2195#elif defined(USEFULBUF_CONFIG_LITTLE_ENDIAN) && defined(USEFULBUF_CONFIG_BSWAP)
2196 uint16_t uTmp = __builtin_bswap16(uInteger16);
2197 pBytes = &uTmp;
2198
2199#else
2200 uint8_t aTmp[2];
2201
2202 aTmp[0] = (uint8_t)((uInteger16 & 0xff00) >> 8);
2203 aTmp[1] = (uint8_t)(uInteger16 & 0xff);
2204
2205 pBytes = aTmp;
2206#endif
2207
2208 UsefulOutBuf_InsertData(me, pBytes, 2, uPos);
2209}
2210
2211
2213 uint32_t uInteger32,
2214 size_t uPos)
2215{
2216 /* See UsefulOutBuf_InsertUint64() for comments on this code */
2217
2218 const void *pBytes;
2219
2220#if defined(USEFULBUF_CONFIG_BIG_ENDIAN)
2221 pBytes = &uInteger32;
2222
2223#elif defined(USEFULBUF_CONFIG_HTON)
2224 uint32_t uTmp = htonl(uInteger32);
2225 pBytes = &uTmp;
2226
2227#elif defined(USEFULBUF_CONFIG_LITTLE_ENDIAN) && defined(USEFULBUF_CONFIG_BSWAP)
2228 uint32_t uTmp = __builtin_bswap32(uInteger32);
2229
2230 pBytes = &uTmp;
2231
2232#else
2233 uint8_t aTmp[4];
2234
2235 aTmp[0] = (uint8_t)((uInteger32 & 0xff000000) >> 24);
2236 aTmp[1] = (uint8_t)((uInteger32 & 0xff0000) >> 16);
2237 aTmp[2] = (uint8_t)((uInteger32 & 0xff00) >> 8);
2238 aTmp[3] = (uint8_t)(uInteger32 & 0xff);
2239
2240 pBytes = aTmp;
2241#endif
2242
2243 UsefulOutBuf_InsertData(pMe, pBytes, 4, uPos);
2244}
2245
2247 uint64_t uInteger64,
2248 size_t uPos)
2249{
2250 const void *pBytes;
2251
2252#if defined(USEFULBUF_CONFIG_BIG_ENDIAN)
2253 /* We have been told explicitly we are running on a big-endian
2254 * machine. Network byte order is big endian, so just copy. There
2255 * is no issue with alignment here because uInteger64 is always
2256 * aligned (and it doesn't matter if pBytes is aligned).
2257 */
2258 pBytes = &uInteger64;
2259
2260#elif defined(USEFULBUF_CONFIG_HTON)
2261 /* Use system function to handle big- and little-endian. This works
2262 * on both big- and little-endian machines, but hton() is not
2263 * always available or in a standard place so it is not used by
2264 * default. With some compilers and CPUs the code for this is very
2265 * compact through use of a special swap instruction and on
2266 * big-endian machines hton() will reduce to nothing.
2267 */
2268 uint64_t uTmp = htonll(uInteger64);
2269
2270 pBytes = &uTmp;
2271
2272#elif defined(USEFULBUF_CONFIG_LITTLE_ENDIAN) && defined(USEFULBUF_CONFIG_BSWAP)
2273 /* Use built-in function for byte swapping. This usually compiles
2274 * to an efficient special byte swap instruction. Unlike hton() it
2275 * does not do this conditionally on the CPU endianness, so this
2276 * code is also conditional on USEFULBUF_CONFIG_LITTLE_ENDIAN
2277 */
2278 uint64_t uTmp = __builtin_bswap64(uInteger64);
2279
2280 pBytes = &uTmp;
2281
2282#else
2283 /* Default which works on every CPU with no dependency on anything
2284 * from the CPU, compiler, libraries or OS. This always works, but
2285 * it is usually a little larger and slower than hton().
2286 */
2287 uint8_t aTmp[8];
2288
2289 aTmp[0] = (uint8_t)((uInteger64 & 0xff00000000000000) >> 56);
2290 aTmp[1] = (uint8_t)((uInteger64 & 0xff000000000000) >> 48);
2291 aTmp[2] = (uint8_t)((uInteger64 & 0xff0000000000) >> 40);
2292 aTmp[3] = (uint8_t)((uInteger64 & 0xff00000000) >> 32);
2293 aTmp[4] = (uint8_t)((uInteger64 & 0xff000000) >> 24);
2294 aTmp[5] = (uint8_t)((uInteger64 & 0xff0000) >> 16);
2295 aTmp[6] = (uint8_t)((uInteger64 & 0xff00) >> 8);
2296 aTmp[7] = (uint8_t)(uInteger64 & 0xff);
2297
2298 pBytes = aTmp;
2299#endif
2300
2301 /* Do the insert */
2302 UsefulOutBuf_InsertData(pMe, pBytes, sizeof(uint64_t), uPos);
2303}
2304
2305
2306#ifndef USEFULBUF_DISABLE_ALL_FLOAT
2308 float f,
2309 size_t uPos)
2310{
2312}
2313
2314
2316 double d,
2317 size_t uPos)
2318{
2320}
2321#endif /* USEFULBUF_DISABLE_ALL_FLOAT */
2322
2323
2325 UsefulBufC NewData)
2326{
2327 /* An append is just a insert at the end */
2329}
2330
2331
2333 const void *pBytes,
2334 size_t uLen)
2335{
2336 UsefulBufC Data = {pBytes, uLen};
2338}
2339
2340
2342 const char *szString)
2343{
2344 UsefulBufC UBC;
2345 UBC.ptr = szString;
2346 UBC.len = strlen(szString);
2347
2349}
2350
2351
2353 uint8_t byte)
2354{
2355 UsefulOutBuf_AppendData(pMe, &byte, 1);
2356}
2357
2358
2360 uint16_t uInteger16)
2361{
2363}
2364
2366 uint32_t uInteger32)
2367{
2369}
2370
2371
2373 uint64_t uInteger64)
2374{
2376}
2377
2378
2379#ifndef USEFULBUF_DISABLE_ALL_FLOAT
2381 float f)
2382{
2384}
2385
2386
2388 double d)
2389{
2391}
2392#endif /* USEFULBUF_DISABLE_ALL_FLOAT */
2393
2394
2396{
2397 return pMe->err;
2398}
2399
2400
2401static inline size_t UsefulOutBuf_RoomLeft(UsefulOutBuf *pMe)
2402{
2403 return pMe->UB.len - pMe->data_len;
2404}
2405
2406
2407static inline int UsefulOutBuf_WillItFit(UsefulOutBuf *pMe, size_t uLen)
2408{
2409 return uLen <= UsefulOutBuf_RoomLeft(pMe);
2410}
2411
2412
2414{
2415 return pMe->UB.ptr == NULL;
2416}
2417
2418
2420{
2421 UsefulBuf R;
2422
2423 R.len = UsefulOutBuf_RoomLeft(pUOutBuf);
2424 if(R.len > 0 && pUOutBuf->UB.ptr != NULL) {
2425 R.ptr = (uint8_t *)pUOutBuf->UB.ptr + pUOutBuf->data_len;
2426 } else {
2427 R.ptr = NULL;
2428 }
2429
2430 return R;
2431}
2432
2433
2435{
2436 return pMe->UB;
2437}
2438
2439
2440
2441
2443{
2444 pMe->cursor = 0;
2445 pMe->err = 0;
2446 pMe->magic = UIB_MAGIC;
2447 pMe->UB = UB;
2448}
2449
2450static inline size_t UsefulInputBuf_Tell(UsefulInputBuf *pMe)
2451{
2452 return pMe->cursor;
2453}
2454
2455
2457{
2458 return pMe->UB.len;
2459}
2460
2461
2462static inline void UsefulInputBuf_Seek(UsefulInputBuf *pMe, size_t uPos)
2463{
2464 if(uPos > pMe->UB.len) {
2465 pMe->err = 1;
2466 } else {
2467 pMe->cursor = uPos;
2468 }
2469}
2470
2471
2473{
2474 /* Code Reviewers: THIS FUNCTION DOES POINTER MATH */
2475
2476 /* Magic number is messed up. Either the structure got overwritten
2477 * or was never initialized.
2478 */
2479 if(pMe->magic != UIB_MAGIC) {
2480 return 0;
2481 }
2482
2483 /* The cursor is off the end of the input buffer given.
2484 * Presuming there are no bugs in this code, this should never happen.
2485 * If it is so, the struct was corrupted. The check is retained as
2486 * as a defense in case there is a bug in this code or the struct is
2487 * corrupted by an attacker or accidentally.
2488 */
2489 if(pMe->cursor > pMe->UB.len) {
2490 return 0;
2491 }
2492
2493 /* subtraction can't go negative because of check above */
2494 return pMe->UB.len - pMe->cursor;
2495}
2496
2497
2498static inline int UsefulInputBuf_BytesAvailable(UsefulInputBuf *pMe, size_t uLen)
2499{
2500 return UsefulInputBuf_BytesUnconsumed(pMe) >= uLen ? 1 : 0;
2501}
2502
2503
2504static inline size_t UsefulInputBuf_PointerToOffset(UsefulInputBuf *pUInBuf, const void *p)
2505{
2506 return UsefulBuf_PointerToOffset(pUInBuf->UB, p);
2507}
2508
2509
2510static inline const void *UsefulInputBuf_OffsetToPointer(UsefulInputBuf *pUInBuf, size_t uOffset)
2511 {
2512 return UsefulBuf_OffsetToPointer(pUInBuf->UB, uOffset);
2513 }
2514
2515
2517{
2518 const void *pResult = UsefulInputBuf_GetBytes(pMe, uNum);
2519 if(!pResult) {
2520 return NULLUsefulBufC;
2521 } else {
2522 UsefulBufC UBC;
2523 UBC.ptr = pResult;
2524 UBC.len = uNum;
2525 return UBC;
2526 }
2527}
2528
2529
2530static inline uint8_t UsefulInputBuf_GetByte(UsefulInputBuf *pMe)
2531{
2532 const void *pResult = UsefulInputBuf_GetBytes(pMe, sizeof(uint8_t));
2533
2534 /* The ternary operator is subject to integer promotion, because
2535 * the operands are smaller than int, so cast back to uint8_t is
2536 * needed to be completely explicit about types (for static
2537 * analyzers).
2538 */
2539 return (uint8_t)(pResult ? *(const uint8_t *)pResult : 0);
2540}
2541
2542static inline uint16_t UsefulInputBuf_GetUint16(UsefulInputBuf *pMe)
2543{
2544 const uint8_t *pResult = (const uint8_t *)UsefulInputBuf_GetBytes(pMe, sizeof(uint16_t));
2545
2546 if(!pResult) {
2547 return 0;
2548 }
2549
2550 /* See UsefulInputBuf_GetUint64() for comments on this code */
2551#if defined(USEFULBUF_CONFIG_BIG_ENDIAN) || defined(USEFULBUF_CONFIG_HTON) || defined(USEFULBUF_CONFIG_BSWAP)
2552 uint16_t uTmp;
2553 memcpy(&uTmp, pResult, sizeof(uint16_t));
2554
2555#if defined(USEFULBUF_CONFIG_BIG_ENDIAN)
2556 return uTmp;
2557
2558#elif defined(USEFULBUF_CONFIG_HTON)
2559 return ntohs(uTmp);
2560
2561#else
2562 return __builtin_bswap16(uTmp);
2563
2564#endif
2565
2566#else
2567
2568 /* The operations here are subject to integer promotion because the
2569 * operands are smaller than int. They will be promoted to unsigned
2570 * int for the shift and addition. The cast back to uint16_t is is
2571 * needed to be completely explicit about types (for static
2572 * analyzers).
2573 */
2574 return (uint16_t)((pResult[0] << 8) + pResult[1]);
2575
2576#endif
2577}
2578
2579
2580static inline uint32_t UsefulInputBuf_GetUint32(UsefulInputBuf *pMe)
2581{
2582 const uint8_t *pResult = (const uint8_t *)UsefulInputBuf_GetBytes(pMe, sizeof(uint32_t));
2583
2584 if(!pResult) {
2585 return 0;
2586 }
2587
2588 /* See UsefulInputBuf_GetUint64() for comments on this code */
2589#if defined(USEFULBUF_CONFIG_BIG_ENDIAN) || defined(USEFULBUF_CONFIG_HTON) || defined(USEFULBUF_CONFIG_BSWAP)
2590 uint32_t uTmp;
2591 memcpy(&uTmp, pResult, sizeof(uint32_t));
2592
2593#if defined(USEFULBUF_CONFIG_BIG_ENDIAN)
2594 return uTmp;
2595
2596#elif defined(USEFULBUF_CONFIG_HTON)
2597 return ntohl(uTmp);
2598
2599#else
2600 return __builtin_bswap32(uTmp);
2601
2602#endif
2603
2604#else
2605 return ((uint32_t)pResult[0]<<24) +
2606 ((uint32_t)pResult[1]<<16) +
2607 ((uint32_t)pResult[2]<<8) +
2608 (uint32_t)pResult[3];
2609#endif
2610}
2611
2612
2613static inline uint64_t UsefulInputBuf_GetUint64(UsefulInputBuf *pMe)
2614{
2615 const uint8_t *pResult = (const uint8_t *)UsefulInputBuf_GetBytes(pMe, sizeof(uint64_t));
2616
2617 if(!pResult) {
2618 return 0;
2619 }
2620
2621#if defined(USEFULBUF_CONFIG_BIG_ENDIAN) || defined(USEFULBUF_CONFIG_HTON) || defined(USEFULBUF_CONFIG_BSWAP)
2622 /* pResult will probably not be aligned. This memcpy() moves the
2623 * bytes into a temp variable safely for CPUs that can or can't do
2624 * unaligned memory access. Many compilers will optimize the
2625 * memcpy() into a simple move instruction.
2626 */
2627 uint64_t uTmp;
2628 memcpy(&uTmp, pResult, sizeof(uint64_t));
2629
2630#if defined(USEFULBUF_CONFIG_BIG_ENDIAN)
2631 /* We have been told expliclity this is a big-endian CPU. Since
2632 * network byte order is big-endian, there is nothing to do.
2633 */
2634
2635 return uTmp;
2636
2637#elif defined(USEFULBUF_CONFIG_HTON)
2638 /* We have been told to use ntoh(), the system function to handle
2639 * big- and little-endian. This works on both big- and
2640 * little-endian machines, but ntoh() is not always available or in
2641 * a standard place so it is not used by default. On some CPUs the
2642 * code for this is very compact through use of a special swap
2643 * instruction.
2644 */
2645
2646 return ntohll(uTmp);
2647
2648#else
2649 /* Little-endian (since it is not USEFULBUF_CONFIG_BIG_ENDIAN) and
2650 * USEFULBUF_CONFIG_BSWAP (since it is not USEFULBUF_CONFIG_HTON).
2651 * __builtin_bswap64() and friends are not conditional on CPU
2652 * endianness so this must only be used on little-endian machines.
2653 */
2654
2655 return __builtin_bswap64(uTmp);
2656
2657
2658#endif
2659
2660#else
2661 /* This is the default code that works on every CPU and every
2662 * endianness with no dependency on ntoh(). This works on CPUs
2663 * that either allow or do not allow unaligned access. It will
2664 * always work, but usually is a little less efficient than ntoh().
2665 */
2666
2667 return ((uint64_t)pResult[0]<<56) +
2668 ((uint64_t)pResult[1]<<48) +
2669 ((uint64_t)pResult[2]<<40) +
2670 ((uint64_t)pResult[3]<<32) +
2671 ((uint64_t)pResult[4]<<24) +
2672 ((uint64_t)pResult[5]<<16) +
2673 ((uint64_t)pResult[6]<<8) +
2674 (uint64_t)pResult[7];
2675#endif
2676}
2677
2678
2679#ifndef USEFULBUF_DISABLE_ALL_FLOAT
2681{
2682 uint32_t uResult = UsefulInputBuf_GetUint32(pMe);
2683
2684 return uResult ? UsefulBufUtil_CopyUint32ToFloat(uResult) : 0;
2685}
2686
2687
2689{
2690 uint64_t uResult = UsefulInputBuf_GetUint64(pMe);
2691
2692 return uResult ? UsefulBufUtil_CopyUint64ToDouble(uResult) : 0;
2693}
2694#endif /* USEFULBUF_DISABLE_ALL_FLOAT */
2695
2696
2698{
2699 return pMe->err;
2700}
2701
2702
2703static inline void UsefulInputBuf_SetBufferLength(UsefulInputBuf *pMe, size_t uNewLen)
2704{
2705 pMe->UB.len = uNewLen;
2706}
2707
2709{
2710 return pMe->UB;
2711}
2712
2713
2714#ifdef __cplusplus
2715}
2716#endif
2717
2718#endif /* _UsefulBuf_h */
2719
2720
static UsefulBufC UsefulInputBuf_GetUsefulBuf(UsefulInputBuf *pUInBuf, size_t uNum)
Get UsefulBuf out of the input buffer.
Definition UsefulBuf.h:2516
static void UsefulOutBuf_InsertByte(UsefulOutBuf *pUOutBuf, uint8_t byte, size_t uPos)
Insert a byte into the UsefulOutBuf.
Definition UsefulBuf.h:2172
size_t UsefulBuf_FindBytes(UsefulBufC BytesToSearch, UsefulBufC BytesToFind)
Find one UsefulBufC in another.
UsefulBufC UsefulOutBuf_OutUBuf(UsefulOutBuf *pUOutBuf)
Returns the data put into a UsefulOutBuf.
static int UsefulOutBuf_WillItFit(UsefulOutBuf *pUOutBuf, size_t uLen)
Returns 1 if some number of bytes will fit in the UsefulOutBuf.
Definition UsefulBuf.h:2407
void UsefulOutBuf_InsertUsefulBuf(UsefulOutBuf *pUOutBuf, UsefulBufC NewData, size_t uPos)
Inserts bytes into the UsefulOutBuf.
static uint8_t UsefulInputBuf_GetByte(UsefulInputBuf *pUInBuf)
Get a byte out of the input buffer.
Definition UsefulBuf.h:2530
static UsefulBuf UsefulBuf_Unconst(const UsefulBufC UBC)
Convert a const UsefulBufC to a non-const UsefulBuf.
Definition UsefulBuf.h:1980
static int UsefulInputBuf_BytesAvailable(UsefulInputBuf *pUInBuf, size_t uLen)
Check if there are unconsumed bytes.
Definition UsefulBuf.h:2498
static void UsefulOutBuf_InsertString(UsefulOutBuf *pUOutBuf, const char *szString, size_t uPos)
Insert a NULL-terminated string into the UsefulOutBuf.
Definition UsefulBuf.h:2160
static void UsefulOutBuf_InsertFloat(UsefulOutBuf *pUOutBuf, float f, size_t uPos)
Insert a float into the UsefulOutBuf.
Definition UsefulBuf.h:2307
static size_t UsefulOutBuf_GetEndPosition(UsefulOutBuf *pUOutBuf)
Returns position of end of data in the UsefulOutBuf.
Definition UsefulBuf.h:2138
static void UsefulOutBuf_AppendUsefulBuf(UsefulOutBuf *pUOutBuf, UsefulBufC NewData)
Append a UsefulBuf into the UsefulOutBuf.
Definition UsefulBuf.h:2324
static void UsefulOutBuf_InsertData(UsefulOutBuf *pUOutBuf, const void *pBytes, size_t uLen, size_t uPos)
Insert a data buffer into the UsefulOutBuf.
Definition UsefulBuf.h:2150
UsefulBufC UsefulOutBuf_CopyOut(UsefulOutBuf *pUOutBuf, UsefulBuf Dest)
Copy out the data put into a UsefulOutBuf.
struct q_useful_buf UsefulBuf
static UsefulBufC UsefulBuf_FromSZ(const char *szString)
Convert a NULL-terminated string to a UsefulBufC.
Definition UsefulBuf.h:1995
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.
static void UsefulInputBuf_SetBufferLength(UsefulInputBuf *pUInBuf, size_t uNewLen)
Alters the input buffer length (use with caution).
Definition UsefulBuf.h:2703
static uint32_t UsefulInputBuf_GetUint32(UsefulInputBuf *pUInBuf)
Get a uint32_t out of the input buffer.
Definition UsefulBuf.h:2580
void UsefulOutBuf_Swap(UsefulOutBuf *pUOutBuf, size_t uStartOffset, size_t uPivotOffset, size_t uEndOffset)
Swap two regions of output bytes.
static size_t UsefulInputBuf_BytesUnconsumed(UsefulInputBuf *pUInBuf)
Returns the number of bytes from the cursor to the end of the buffer, the unconsumed bytes.
Definition UsefulBuf.h:2472
static void UsefulInputBuf_Seek(UsefulInputBuf *pUInBuf, size_t uPos)
Sets the current position in input buffer.
Definition UsefulBuf.h:2462
static uint32_t UsefulBufUtil_CopyFloatToUint32(float f)
Copy a float to a uint32_t.
Definition UsefulBuf.h:2099
struct useful_input_buf UsefulInputBuf
static size_t UsefulBuf_PointerToOffset(UsefulBufC UB, const void *p)
Convert a pointer to an offset with bounds checking.
Definition UsefulBuf.h:2063
static void UsefulOutBuf_InsertUint32(UsefulOutBuf *pUOutBuf, uint32_t uInteger32, size_t uPos)
Insert a 32-bit integer into the UsefulOutBuf.
Definition UsefulBuf.h:2212
static UsefulBuf UsefulOutBuf_RetrieveOutputStorage(UsefulOutBuf *pUOutBuf)
Retrieve the storage buffer passed in to UsefulOutBuf_Init().
Definition UsefulBuf.h:2434
const void * UsefulInputBuf_GetBytes(UsefulInputBuf *pUInBuf, size_t uNum)
Get pointer to bytes out of the input buffer.
static void UsefulInputBuf_Init(UsefulInputBuf *pUInBuf, UsefulBufC UB)
Initialize the UsefulInputBuf structure before use.
Definition UsefulBuf.h:2442
static UsefulBufC UsefulBuf_CopyPtr(UsefulBuf Dest, const void *ptr, size_t uLen)
Copy a pointer into a UsefulBuf.
Definition UsefulBuf.h:2022
static void UsefulOutBuf_AppendDouble(UsefulOutBuf *pUOutBuf, double d)
Append a double to the UsefulOutBuf.
Definition UsefulBuf.h:2387
static int UsefulBuf_IsNULLC(UsefulBufC UB)
Check if a UsefulBufC is NULLUsefulBufC or not.
Definition UsefulBuf.h:1941
static uint64_t UsefulInputBuf_GetUint64(UsefulInputBuf *pUInBuf)
Get a uint64_t out of the input buffer.
Definition UsefulBuf.h:2613
static uint16_t UsefulInputBuf_GetUint16(UsefulInputBuf *pUInBuf)
Get a uint16_t out of the input buffer.
Definition UsefulBuf.h:2542
void UsefulOutBuf_Advance(UsefulOutBuf *pUOutBuf, size_t uAmount)
Advance the amount output assuming it was written by the caller.
static float UsefulBufUtil_CopyUint32ToFloat(uint32_t u32)
Copy a uint32_t to a float.
Definition UsefulBuf.h:2120
static int UsefulBuf_IsEmptyC(UsefulBufC UB)
Check if a UsefulBufC is empty or not.
Definition UsefulBuf.h:1953
static int UsefulBuf_IsNULL(UsefulBuf UB)
Check if a UsefulBuf is NULLUsefulBuf or not.
Definition UsefulBuf.h:1935
static float UsefulInputBuf_GetFloat(UsefulInputBuf *pUInBuf)
Get a float out of the input buffer.
Definition UsefulBuf.h:2680
static void UsefulOutBuf_AppendUint32(UsefulOutBuf *pUOutBuf, uint32_t uInteger32)
Append an integer to the UsefulOutBuf.
Definition UsefulBuf.h:2365
static void UsefulOutBuf_AppendData(UsefulOutBuf *pUOutBuf, const void *pBytes, size_t uLen)
Append bytes to the UsefulOutBuf.
Definition UsefulBuf.h:2332
static UsefulBufC UsefulBuf_Set(UsefulBuf pDest, uint8_t value)
Set all bytes in a UsefulBuf to a value, for example to 0.
Definition UsefulBuf.h:2010
static UsefulBufC UsefulInputBuf_RetrieveUndecodedInput(UsefulInputBuf *pUInBuf)
Retrieve the undecoded input buffer.
Definition UsefulBuf.h:2708
static uint64_t UsefulBufUtil_CopyDoubleToUint64(double d)
Copy a double to a uint64_t.
Definition UsefulBuf.h:2106
UsefulBufC UsefulOutBuf_SubString(UsefulOutBuf *pUOutBuf, const size_t uStart, const size_t uLen)
Return a substring of the output data.
static size_t UsefulInputBuf_Tell(UsefulInputBuf *pUInBuf)
Returns current position in input buffer.
Definition UsefulBuf.h:2450
static void UsefulOutBuf_AppendString(UsefulOutBuf *pUOutBuf, const char *szString)
Append a NULL-terminated string to the UsefulOutBuf.
Definition UsefulBuf.h:2341
static int UsefulInputBuf_GetError(UsefulInputBuf *pUInBuf)
Get the error status.
Definition UsefulBuf.h:2697
static UsefulBuf UsefulBufC_Unconst(const UsefulBufC UBC)
Definition UsefulBuf.h:720
static void UsefulOutBuf_InsertUint16(UsefulOutBuf *pUOutBuf, uint16_t uInteger16, size_t uPos)
Insert a 16-bit integer into the UsefulOutBuf.
Definition UsefulBuf.h:2180
UsefulBufC UsefulBuf_CopyOffset(UsefulBuf Dest, size_t uOffset, const UsefulBufC Src)
Copy one UsefulBuf into another at an offset.
struct useful_out_buf UsefulOutBuf
static void UsefulOutBuf_AppendUint16(UsefulOutBuf *pUOutBuf, uint16_t uInteger16)
Append an integer to the UsefulOutBuf.
Definition UsefulBuf.h:2359
static void UsefulOutBuf_AppendFloat(UsefulOutBuf *pUOutBuf, float f)
Append a float to the UsefulOutBuf.
Definition UsefulBuf.h:2380
static size_t UsefulInputBuf_PointerToOffset(UsefulInputBuf *pUInBuf, const void *p)
Convert a pointer to an offset with bounds checking.
Definition UsefulBuf.h:2504
static const void * UsefulInputBuf_OffsetToPointer(UsefulInputBuf *pUInBuf, size_t uOffset)
Convert an offset to a pointer with bounds checking.
Definition UsefulBuf.h:2510
static void UsefulOutBuf_InsertDouble(UsefulOutBuf *pUOutBuf, double d, size_t uPos)
Insert a double into the UsefulOutBuf.
Definition UsefulBuf.h:2315
void UsefulOutBuf_Init(UsefulOutBuf *pUOutBuf, UsefulBuf Storage)
Initialize and supply the output buffer.
static UsefulBuf UsefulOutBuf_GetOutPlace(UsefulOutBuf *pUOutBuf)
Returns pointer and length of the output buffer not yet used.
Definition UsefulBuf.h:2419
static const void * UsefulBuf_OffsetToPointer(UsefulBufC UB, size_t uOffset)
Convert an offset to a pointer with bounds checking.
Definition UsefulBuf.h:2086
static void UsefulOutBuf_AppendUint64(UsefulOutBuf *pUOutBuf, uint64_t uInteger64)
Append an integer to the UsefulOutBuf.
Definition UsefulBuf.h:2372
UsefulBufC UsefulOutBuf_OutUBufOffset(UsefulOutBuf *pUOutBuf, size_t uOffset)
Returns data starting at an offset that was put into a UsefulOutBuf.
static double UsefulBufUtil_CopyUint64ToDouble(uint64_t u64)
Copy a uint64_t to a double.
Definition UsefulBuf.h:2113
size_t UsefulBuf_IsValue(const UsefulBufC UB, uint8_t uValue)
Find first byte that is not a particular byte value.
UsefulBufC UsefulBuf_SkipLeading(UsefulBufC String, uint8_t uByte)
Skip leading bytes of a particular value in a string.
static size_t UsefulOutBuf_RoomLeft(UsefulOutBuf *pUOutBuf)
Returns number of bytes unused used in the output buffer.
Definition UsefulBuf.h:2401
int UsefulOutBuf_Compare(UsefulOutBuf *pUOutBuf, size_t uStart1, size_t uLen1, size_t uStart2, size_t uLen2)
Compare bytes at offsets.
static void UsefulOutBuf_Reset(UsefulOutBuf *pUOutBuf)
Reset a UsefulOutBuf for re use.
Definition UsefulBuf.h:2131
static int UsefulOutBuf_IsBufferNULL(UsefulOutBuf *pUOutBuf)
Returns 1 if buffer given to UsefulOutBuf_Init() was NULL.
Definition UsefulBuf.h:2413
int UsefulBuf_Compare(const UsefulBufC UB1, const UsefulBufC UB2)
Compare one UsefulBufC to another.
static void UsefulOutBuf_AppendByte(UsefulOutBuf *pUOutBuf, uint8_t byte)
Append a byte to the UsefulOutBuf.
Definition UsefulBuf.h:2352
static int UsefulBuf_IsNULLOrEmptyC(UsefulBufC UB)
Check if a UsefulBufC is NULLUsefulBufC or empty.
Definition UsefulBuf.h:1965
static UsefulBufC UsefulBuf_Copy(UsefulBuf Dest, const UsefulBufC Src)
Copy one UsefulBuf into another.
Definition UsefulBuf.h:2004
static double UsefulInputBuf_GetDouble(UsefulInputBuf *pUInBuf)
Get a double out of the input buffer.
Definition UsefulBuf.h:2688
static UsefulBufC UsefulBuf_Const(const UsefulBuf UB)
Convert a non-const UsefulBuf to a const UsefulBufC.
Definition UsefulBuf.h:1971
struct q_useful_buf_c UsefulBufC
static UsefulBufC UsefulBuf_Head(UsefulBufC UB, size_t uAmount)
Returns a truncation of a UsefulBufC.
Definition UsefulBuf.h:2031
static UsefulBufC UsefulBuf_Tail(UsefulBufC UB, size_t uAmount)
Returns bytes from the end of a UsefulBufC.
Definition UsefulBuf.h:2045
static void UsefulOutBuf_InsertUint64(UsefulOutBuf *pUOutBuf, uint64_t uInteger64, size_t uPos)
Insert a 64-bit integer into the UsefulOutBuf.
Definition UsefulBuf.h:2246
static int UsefulBuf_IsNULLOrEmpty(UsefulBuf UB)
Check if a UsefulBuf is NULLUsefulBuf or empty.
Definition UsefulBuf.h:1959
static int UsefulOutBuf_GetError(UsefulOutBuf *pUOutBuf)
Returns the current error status.
Definition UsefulBuf.h:2395
static int UsefulOutBuf_AtStart(UsefulOutBuf *pUOutBuf)
Returns whether any data has been added to the UsefulOutBuf.
Definition UsefulBuf.h:2144
#define NULLUsefulBufC
Definition UsefulBuf.h:321
#define UIB_MAGIC
Definition UsefulBuf.h:1598
static size_t UsefulInputBuf_GetBufferLength(UsefulInputBuf *pUInBuf)
Gets the input buffer length.
Definition UsefulBuf.h:2456
static int UsefulBuf_IsEmpty(UsefulBuf UB)
Check if a UsefulBuf is empty or not.
Definition UsefulBuf.h:1947
Definition UsefulBuf.h:280
Definition UsefulBuf.h:291
Definition UsefulBuf.h:1586
Definition UsefulBuf.h:875