QCBOR
Loading...
Searching...
No Matches
qcbor_main_decode.h
Go to the documentation of this file.
1/* ===========================================================================
2 * qcbor_main_decode.h -- The main CBOR decoder.
3 *
4 * Copyright (c) 2016-2018, The Linux Foundation.
5 * Copyright (c) 2018-2024, Laurence Lundblade.
6 * Copyright (c) 2021, Arm Limited.
7 * All rights reserved.
8 *
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted provided that the following conditions are
11 * met:
12 * * Redistributions of source code must retain the above copyright
13 * notice, this list of conditions and the following disclaimer.
14 * * Redistributions in binary form must reproduce the above
15 * copyright notice, this list of conditions and the following
16 * disclaimer in the documentation and/or other materials provided
17 * with the distribution.
18 * * Neither the name of The Linux Foundation nor the names of its
19 * contributors, nor the name "Laurence Lundblade" may be used to
20 * endorse or promote products derived from this software without
21 * specific prior written permission.
22 *
23 * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
24 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
25 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
26 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
27 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
28 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
29 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
30 * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
31 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
32 * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
33 * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
34 * ========================================================================= */
35
36#ifndef qcbor_main_decode_h
37#define qcbor_main_decode_h
38
39
40#include "qcbor/qcbor_common.h"
41#include "qcbor/qcbor_private.h"
42#include <stdbool.h>
43
44
45#ifdef __cplusplus
46extern "C" {
47#if 0
48} /* Keep editor indention formatting happy */
49#endif
50#endif
51
52
286
287
288
293#define QCBOR_MAX_DECODE_INPUT_SIZE (UINT32_MAX - 2)
294
299#define QCBOR_MAX_TAGS_PER_ITEM QCBOR_MAX_TAGS_PER_ITEM1
300
301
302
303/* Do not renumber these. Code depends on some of these values. */
305#define QCBOR_TYPE_NONE 0
306
308#define QCBOR_TYPE_ANY 1
309
313#define QCBOR_TYPE_INT64 2
314
317#define QCBOR_TYPE_UINT64 3
318
320#define QCBOR_TYPE_ARRAY 4
321
323#define QCBOR_TYPE_MAP 5
324
326#define QCBOR_TYPE_BYTE_STRING 6
327
331#define QCBOR_TYPE_TEXT_STRING 7
332
336#define QCBOR_TYPE_POSBIGNUM 9
337
349#define QCBOR_TYPE_NEGBIGNUM 10
350
356#define QCBOR_TYPE_DATE_STRING 11
357
360#define QCBOR_TYPE_DATE_EPOCH 12
361
369#define QCBOR_TYPE_UKNOWN_SIMPLE 13
370
373#define QCBOR_TYPE_DECIMAL_FRACTION 14
374
378#define QCBOR_TYPE_DECIMAL_FRACTION_POS_BIGNUM 15
379
383#define QCBOR_TYPE_DECIMAL_FRACTION_NEG_BIGNUM 16
384
387#define QCBOR_TYPE_DECIMAL_FRACTION_POS_U64 79
388
392#define QCBOR_TYPE_DECIMAL_FRACTION_NEG_U64 80
393
397#define QCBOR_TYPE_BIGFLOAT 17
398
402#define QCBOR_TYPE_BIGFLOAT_POS_BIGMANTISSA 18
403#define QCBOR_TYPE_BIGFLOAT_POS_BIGNUM 18
404
408#define QCBOR_TYPE_BIGFLOAT_NEG_BIGMANTISSA 19
409#define QCBOR_TYPE_BIGFLOAT_NEG_BIGNUM 19
410
414#define QCBOR_TYPE_BIGFLOAT_POS_U64MANTISSA 82
415
419#define QCBOR_TYPE_BIGFLOAT_NEG_U64MANTISSA 83
420
422#define QCBOR_TYPE_FALSE 20
423
425#define QCBOR_TYPE_TRUE 21
426
428#define QCBOR_TYPE_NULL 22
429
431#define QCBOR_TYPE_UNDEF 23
432
434#define QCBOR_TYPE_FLOAT 26
435
437#define QCBOR_TYPE_DOUBLE 27
438
446#define QCBOR_TYPE_65BIT_NEG_INT 28
447
449#define QCBOR_TYPE_BREAK 31
450
453#define QCBOR_TYPE_MAP_AS_ARRAY 32
454
458#define QBCOR_TYPE_WRAPPED_CBOR 36
459
461#define QCBOR_TYPE_URI 44
462
465#define QCBOR_TYPE_BASE64URL 45
466
469#define QCBOR_TYPE_BASE64 46
470
472#define QCBOR_TYPE_REGEX 47
473
476#define QCBOR_TYPE_MIME 48
477
479#define QCBOR_TYPE_UUID 49
480
483#define QBCOR_TYPE_WRAPPED_CBOR_SEQUENCE 75
484
487#define QCBOR_TYPE_BINARY_MIME 76
488
492#define QCBOR_TYPE_DAYS_STRING 77
493
497#define QCBOR_TYPE_DAYS_EPOCH 78
498
499/* 79, 80, 82, 83 is used above for decimal fraction and big float */
500
501
503#define QCBOR_TYPE_TAG_NUMBER 127
504
507#define QCBOR_TYPE_START_USER_DEFINED 128
508
510#define QCBOR_TYPE_END_USER_DEFINED 255
511
512
517#define QCBOR_LAST_UNMAPPED_TAG (CBOR_TAG_INVALID16 - QCBOR_NUM_MAPPED_TAGS - 1)
518
519
554typedef struct {
555 int64_t nExponent;
556 union {
557 int64_t nInt;
558 uint64_t uInt;
559 UsefulBufC bigNum;
560 } Mantissa;
562
563
564
574typedef struct _QCBORItem {
577 uint8_t uDataType;
578
581 uint8_t uLabelType;
582
586
591
596 uint8_t uDataAlloc;
597
602 uint8_t uLabelAlloc;
603
606 union {
608 int64_t int64;
610 uint64_t uint64;
622 uint16_t uCount;
623#ifndef USEFULBUF_DISABLE_ALL_FLOAT
625 double dfnum;
627 float fnum;
628#endif /* USEFULBUF_DISABLE_ALL_FLOAT */
639 struct {
640 int64_t nSeconds;
641#ifndef USEFULBUF_DISABLE_ALL_FLOAT
642 double fSecondsFraction;
643#endif /* USEFULBUF_DISABLE_ALL_FLOAT */
645
648 int64_t epochDays;
649
653
655 uint8_t uSimple;
656
657#ifndef QCBOR_DISABLE_EXP_AND_MANTISSA
658 QCBORExpAndMantissa expAndMantissa;
659#endif /* ! QCBOR_DISABLE_EXP_AND_MANTISSA */
660
661 uint64_t uTagNumber; /* Used internally during decoding */
662
663 /* For use by user-defined tag content handlers */
664 uint8_t userDefined[24];
666
668 union {
670 int64_t int64;
671#ifndef QCBOR_DISABLE_NON_INTEGER_LABELS
673 uint64_t uint64;
677#endif /* ! QCBOR_DISABLE_NON_INTEGER_LABELS */
679
680#ifndef QCBOR_DISABLE_TAGS
689 QCBORMappedTagNumbers auTagNumbers;
690#endif
692
696#define QCBOR_COUNT_INDICATES_INDEFINITE_LENGTH UINT16_MAX
697
698
699
700
759typedef UsefulBuf (* QCBORStringAllocate)(void *pAllocateCxt,
760 void *pOldMem,
761 size_t uNewSize);
762
763
774#define QCBOR_DECODE_MIN_MEM_POOL_SIZE 8
775
776
777
778
785typedef struct _QCBORDecodeContext QCBORDecodeContext;
786
787
804void
806
807
856 UsefulBuf MemPool,
857 bool bAllStrings);
858
859
860#ifndef QCBOR_DISABLE_INDEFINITE_LENGTH_STRINGS
894static void
896 QCBORStringAllocate pfAllocateFunction,
897 void *pAllocateContext,
898 bool bAllStrings);
899
900#endif /* ! QCBOR_DISABLE_INDEFINITE_LENGTH_STRINGS */
901
1053void
1055
1056
1072
1073
1096void
1098
1099
1123void
1125
1126
1140
1141
1184static uint32_t
1186
1187
1202
1203
1204
1246
1247
1275
1276
1286static UsefulBufC
1288
1289
1317static QCBORError
1319
1320
1330static QCBORError
1332
1333
1340static bool
1342
1343
1363static bool
1365
1366
1387static void
1389
1390
1391
1392
1393/* ========================================================================= *
1394 * BEGINNING OF DEPRECATED FUNCTIONS *
1395 * *
1396 * There is no plan to remove these in future versions. *
1397 * They just have been replaced by something better. *
1398 * ========================================================================= */
1399
1425void
1427
1428
1429/* ========================================================================= *
1430 * END OF DEPRECATED FUNCTIONS *
1431 * ========================================================================= */
1432
1433
1434
1435
1436/* ========================================================================= *
1437 * BEGINNING OF PRIVATE AND INLINE IMPLEMENTATION *
1438 * ========================================================================= */
1439
1442QCBORDecode_Private_GetNextTagContent(QCBORDecodeContext *pMe,
1443 QCBORItem *pDecodedItem);
1444
1445
1447uint64_t
1448QCBORDecode_Private_UnMapTagNumber(const QCBORDecodeContext *pMe,
1449 const uint16_t uMappedTagNumber);
1450
1453QCBORDecode_Private_ConsumeItem(QCBORDecodeContext *pMe,
1454 const QCBORItem *pItemToConsume,
1455 bool *pbBreak,
1456 uint8_t *puNextNestLevel);
1457
1460QCBORDecode_Private_GetItemChecks(QCBORDecodeContext *pMe,
1461 QCBORError uErr,
1462 const size_t uOffset,
1463 QCBORItem *pDecodedItem);
1464
1467QCBORDecode_Private_NestLevelAscender(QCBORDecodeContext *pMe,
1468 bool bMarkEnd,
1469 bool *pbBreak);
1470
1471#ifndef QCBOR_DISABLE_INDEFINITE_LENGTH_STRINGS
1472static inline void
1474 QCBORStringAllocate pfAllocateFunction,
1475 void *pAllocateContext,
1476 bool bAllStrings)
1477{
1478 pMe->StringAllocator.pfAllocator = pfAllocateFunction;
1479 pMe->StringAllocator.pAllocateCxt = pAllocateContext;
1480 pMe->bStringAllocateAll = bAllStrings;
1481}
1482#endif /* ! QCBOR_DISABLE_INDEFINITE_LENGTH_STRINGS */
1483
1484
1485static inline uint32_t
1487{
1488 if(pMe->uLastError) {
1489 return UINT32_MAX;
1490 }
1491
1492 /* Cast is safe because decoder input size is restricted. */
1493 return (uint32_t)UsefulInputBuf_Tell(&(pMe->InBuf));
1494}
1495
1496static inline UsefulBufC
1501
1502static inline QCBORError
1504{
1505 return (QCBORError)pMe->uLastError;
1506}
1507
1508static inline QCBORError
1510{
1511 const QCBORError uReturn = (QCBORError)pMe->uLastError;
1512 pMe->uLastError = QCBOR_SUCCESS;
1513 return uReturn;
1514}
1515
1516static inline bool
1518{
1519 if(uErr >= QCBOR_START_OF_NOT_WELL_FORMED_ERRORS &&
1520 uErr <= QCBOR_END_OF_NOT_WELL_FORMED_ERRORS) {
1521 return true;
1522 } else {
1523 return false;
1524 }
1525}
1526
1527static inline bool
1529{
1530 if(uErr >= QCBOR_START_OF_UNRECOVERABLE_DECODE_ERRORS &&
1531 uErr <= QCBOR_END_OF_UNRECOVERABLE_DECODE_ERRORS) {
1532 return true;
1533 } else {
1534 return false;
1535 }
1536}
1537
1538
1539static inline void
1541{
1542 pMe->uLastError = (uint8_t)uError;
1543}
1544
1545
1547static inline void
1548QCBORDecode_Private_SaveTagNumbers(QCBORDecodeContext *pMe, const QCBORItem *pItem)
1549{
1550#ifndef QCBOR_DISABLE_TAGS
1551 memcpy(pMe->auLastTagNumbers,
1552 pItem->auTagNumbers,
1553 sizeof(pItem->auTagNumbers));
1554#else /* ! QCBOR_DISABLE_TAGS */
1555 (void)pMe;
1556 (void)pItem;
1557#endif /* ! QCBOR_DISABLE_TAGS */
1558}
1559
1561static inline void
1562QCBORDecode_Private_GetAndTell(QCBORDecodeContext *pMe, QCBORItem *Item, size_t *uOffset)
1563{
1564#ifndef QCBOR_DISABLE_TAGS
1565 if(pMe->uLastError != QCBOR_SUCCESS) {
1566 return;
1567 }
1568
1569 *uOffset = QCBORDecode_Tell(pMe);
1570#else /* ! QCBOR_DISABLE_TAGS */
1571 *uOffset = SIZE_MAX;
1572
1573#endif /* ! QCBOR_DISABLE_TAGS */
1574 pMe->uLastError = (uint8_t)QCBORDecode_Private_GetNextTagContent(pMe, Item);
1575}
1576
1577
1578/* ======================================================================== *
1579 * END OF PRIVATE INLINE IMPLEMENTATION *
1580 * ======================================================================== */
1581
1582
1583
1584
1585/* A few cross checks on size constants and special value lengths */
1586#if QCBOR_MAP_OFFSET_CACHE_INVALID < QCBOR_MAX_DECODE_INPUT_SIZE
1587#error QCBOR_MAP_OFFSET_CACHE_INVALID is too large
1588#endif
1589
1590#if QCBOR_NON_BOUNDED_OFFSET < QCBOR_MAX_DECODE_INPUT_SIZE
1591#error QCBOR_NON_BOUNDED_OFFSET is too large
1592#endif
1593
1594#ifdef __cplusplus
1595}
1596#endif
1597
1598#endif /* qcbor_main_decode_h */
struct q_useful_buf UsefulBuf
static UsefulBufC UsefulInputBuf_RetrieveUndecodedInput(UsefulInputBuf *pUInBuf)
Retrieve the undecoded input buffer.
Definition UsefulBuf.h:2708
static size_t UsefulInputBuf_Tell(UsefulInputBuf *pUInBuf)
Returns current position in input buffer.
Definition UsefulBuf.h:2450
QCBORError
Definition qcbor_common.h:260
@ QCBOR_SUCCESS
Definition qcbor_common.h:262
QCBORError QCBORDecode_PartialFinish(QCBORDecodeContext *pCtx, size_t *puConsumed)
Return number of bytes consumed so far.
void QCBORDecode_VGetNext(QCBORDecodeContext *pCtx, QCBORItem *pDecodedItem)
Get the next item (integer, byte string, array...) in the preorder traversal of the CBOR tree.
static void QCBORDecode_SetError(QCBORDecodeContext *pCtx, QCBORError uError)
Manually set error condition, or set user-defined error.
Definition qcbor_main_decode.h:1540
QCBORError QCBORDecode_EndCheck(QCBORDecodeContext *pCtx)
Tell whether cursor is at end of the input.
QCBORError QCBORDecode_GetNext(QCBORDecodeContext *pCtx, QCBORItem *pDecodedItem)
Preorder traversal like QCBORDecode_VGetNext() without use of internal error state.
static QCBORError QCBORDecode_GetAndResetError(QCBORDecodeContext *pCtx)
Get and reset the decoding error.
Definition qcbor_main_decode.h:1509
static bool QCBORDecode_IsUnrecoverableError(QCBORError uErr)
Whether a decoding error is recoverable.
Definition qcbor_main_decode.h:1528
void QCBORDecode_VPeekNext(QCBORDecodeContext *pCtx, QCBORItem *pDecodedItem)
Get the next data item without consuming it.
QCBORError QCBORDecode_Finish(QCBORDecodeContext *pCtx)
Check that a decode completed successfully.
static UsefulBufC QCBORDecode_RetrieveUndecodedInput(QCBORDecodeContext *pCtx)
Retrieve the undecoded input buffer.
Definition qcbor_main_decode.h:1497
void QCBORDecode_VGetNextConsume(QCBORDecodeContext *pCtx, QCBORItem *pDecodedItem)
Get the next item, fully consuming it if it is a map or array.
static void QCBORDecode_SetUpAllocator(QCBORDecodeContext *pCtx, QCBORStringAllocate pfAllocateFunction, void *pAllocateContext, bool bAllStrings)
Sets up a custom string allocator for indefinite-length strings.
Definition qcbor_main_decode.h:1473
UsefulBuf(* QCBORStringAllocate)(void *pAllocateCxt, void *pOldMem, size_t uNewSize)
Prototype for the implementation of a string allocator.
Definition qcbor_main_decode.h:759
struct _QCBORDecodeContext QCBORDecodeContext
Definition qcbor_main_decode.h:785
static bool QCBORDecode_IsNotWellFormedError(QCBORError uErr)
Whether an error indicates non-well-formed CBOR.
Definition qcbor_main_decode.h:1517
QCBORError QCBORDecode_PeekNext(QCBORDecodeContext *pCtx, QCBORItem *pDecodedItem)
Get the next data item without consuming it without use of internal error state.
static QCBORError QCBORDecode_GetError(QCBORDecodeContext *pCtx)
Get the decoding error.
Definition qcbor_main_decode.h:1503
QCBORError QCBORDecode_SetMemPool(QCBORDecodeContext *pCtx, UsefulBuf MemPool, bool bAllStrings)
Set up the MemPool string allocator for indefinite-length strings.
static uint32_t QCBORDecode_Tell(QCBORDecodeContext *pCtx)
Get the current traversal cursort offset in the input CBOR.
Definition qcbor_main_decode.h:1486
struct _QCBORItem QCBORItem
QCBORDecodeMode
Definition qcbor_main_decode.h:200
@ QCBOR_DECODE_MODE_DCBOR
Definition qcbor_main_decode.h:281
@ QCBOR_DECODE_NO_INDEF_LENGTH
Definition qcbor_main_decode.h:226
@ QCBOR_DECODE_ALLOW_UNPROCESSED_TAG_NUMBERS
Definition qcbor_main_decode.h:223
@ QCBOR_DECODE_MODE_NORMAL
Definition qcbor_main_decode.h:202
@ QCBOR_DECODE_MODE_MAP_STRINGS_ONLY
Definition qcbor_main_decode.h:206
@ QCBOR_DECODE_ONLY_PREFERRED_BIG_NUMBERS
Definition qcbor_main_decode.h:235
@ QCBOR_DECODE_ONLY_SORTED_MAPS
Definition qcbor_main_decode.h:240
@ QCBOR_DECODE_MODE_PREFERRED
Definition qcbor_main_decode.h:264
@ QCBOR_DECODE_ONLY_PREFERRED_NUMBERS
Definition qcbor_main_decode.h:230
@ QCBOR_DECODE_MODE_CDE
Definition qcbor_main_decode.h:276
@ QCBOR_DECODE_ONLY_REDUCED_FLOATS
Definition qcbor_main_decode.h:246
@ QCBOR_DECODE_DISALLOW_DCBOR_SIMPLES
Definition qcbor_main_decode.h:251
@ QCBOR_DECODE_MODE_MAP_AS_ARRAY
Definition qcbor_main_decode.h:217
void QCBORDecode_CompatibilityV1(QCBORDecodeContext *pCtx)
[Deprecated] Configure CBOR decoder context for QCBOR v1 compatibility.
void QCBORDecode_Init(QCBORDecodeContext *pCtx, UsefulBufC EncodedCBOR, QCBORDecodeMode uConfigFlags)
Definition qcbor_main_decode.h:574
union _QCBORItem::@2 label
UsefulBufC bigNum
Definition qcbor_main_decode.h:652
uint8_t uSimple
Definition qcbor_main_decode.h:655
UsefulBufC string
Definition qcbor_main_decode.h:615
uint8_t uLabelAlloc
Definition qcbor_main_decode.h:602
double dfnum
Definition qcbor_main_decode.h:625
int64_t epochDays
Definition qcbor_main_decode.h:648
float fnum
Definition qcbor_main_decode.h:627
QCBORMappedTagNumbers auTagNumbers
Definition qcbor_main_decode.h:689
int64_t int64
Definition qcbor_main_decode.h:608
uint64_t uint64
Definition qcbor_main_decode.h:610
uint8_t uDataAlloc
Definition qcbor_main_decode.h:596
union _QCBORItem::@1 val
uint8_t uDataType
Definition qcbor_main_decode.h:577
uint8_t uNestingLevel
Definition qcbor_main_decode.h:585
uint8_t uLabelType
Definition qcbor_main_decode.h:581
uint8_t uNextNestLevel
Definition qcbor_main_decode.h:590
struct _QCBORItem::@1::@3 epochDate
uint16_t uCount
Definition qcbor_main_decode.h:622
Definition qcbor_main_decode.h:554
Definition UsefulBuf.h:280
Definition UsefulBuf.h:291