| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949 |
- /*
- * Copyright (c) 2020 Nordic Semiconductor ASA
- *
- * SPDX-License-Identifier: Apache-2.0
- */
- #include <stdint.h>
- #include <stdbool.h>
- #include <stddef.h>
- #include <string.h>
- #include "zcbor_decode.h"
- #include "zcbor_common.h"
- /** Return value length from additional value.
- */
- static uint_fast32_t additional_len(uint8_t additional)
- {
- if (ZCBOR_VALUE_IS_1_BYTE <= additional && additional <= ZCBOR_VALUE_IS_8_BYTES) {
- /* 24 => 1
- * 25 => 2
- * 26 => 4
- * 27 => 8
- */
- return 1U << (additional - ZCBOR_VALUE_IS_1_BYTE);
- }
- return 0;
- }
- /** Extract the major type, i.e. the first 3 bits of the header byte. */
- #define MAJOR_TYPE(header_byte) ((zcbor_major_type_t)(((header_byte) >> 5) & 0x7))
- /** Extract the additional info, i.e. the last 5 bits of the header byte. */
- #define ADDITIONAL(header_byte) ((header_byte) & 0x1F)
- #define FAIL_AND_DECR_IF(expr, err) \
- do {\
- if (expr) { \
- (state->payload)--; \
- ZCBOR_ERR(err); \
- } \
- } while(0)
- static bool initial_checks(zcbor_state_t *state)
- {
- ZCBOR_CHECK_ERROR();
- ZCBOR_CHECK_PAYLOAD();
- return true;
- }
- static bool type_check(zcbor_state_t *state, zcbor_major_type_t exp_major_type)
- {
- if (!initial_checks(state)) {
- ZCBOR_FAIL();
- }
- zcbor_major_type_t major_type = MAJOR_TYPE(*state->payload);
- if (major_type != exp_major_type) {
- ZCBOR_ERR(ZCBOR_ERR_WRONG_TYPE);
- }
- return true;
- }
- #define INITIAL_CHECKS() \
- do {\
- if (!initial_checks(state)) { \
- ZCBOR_FAIL(); \
- } \
- } while(0)
- #define INITIAL_CHECKS_WITH_TYPE(exp_major_type) \
- do {\
- if (!type_check(state, exp_major_type)) { \
- ZCBOR_FAIL(); \
- } \
- } while(0)
- #define ERR_RESTORE(err) \
- do { \
- state->payload = state->payload_bak; \
- state->elem_count++; \
- ZCBOR_ERR(err); \
- } while(0)
- #define FAIL_RESTORE() \
- do { \
- state->payload = state->payload_bak; \
- state->elem_count++; \
- ZCBOR_FAIL(); \
- } while(0)
- /** Get a single value.
- *
- * @details @p ppayload must point to the header byte. This function will
- * retrieve the value (either from within the additional info, or from
- * the subsequent bytes) and return it in the result. The result can
- * have arbitrary length.
- *
- * The function will also validate
- * - Min/max constraints on the value.
- * - That @p payload doesn't overrun past @p payload_end.
- * - That @p elem_count has not been exhausted.
- *
- * @p ppayload and @p elem_count are updated if the function
- * succeeds. If not, they are left unchanged.
- *
- * CBOR values are always big-endian, so this function converts from
- * big to little-endian if necessary (@ref CONFIG_BIG_ENDIAN).
- */
- static bool value_extract(zcbor_state_t *state,
- void *const result, uint_fast32_t result_len)
- {
- zcbor_trace();
- zcbor_assert_state(result_len != 0, "0-length result not supported.\r\n");
- zcbor_assert_state(result != NULL, NULL);
- INITIAL_CHECKS();
- ZCBOR_ERR_IF((state->elem_count == 0), ZCBOR_ERR_LOW_ELEM_COUNT);
- uint8_t *u8_result = (uint8_t *)result;
- uint8_t additional = ADDITIONAL(*state->payload);
- state->payload_bak = state->payload;
- (state->payload)++;
- memset(result, 0, result_len);
- if (additional <= ZCBOR_VALUE_IN_HEADER) {
- #ifdef CONFIG_BIG_ENDIAN
- u8_result[result_len - 1] = additional;
- #else
- u8_result[0] = additional;
- #endif /* CONFIG_BIG_ENDIAN */
- } else {
- uint_fast32_t len = additional_len(additional);
- FAIL_AND_DECR_IF(len > result_len, ZCBOR_ERR_INT_SIZE);
- FAIL_AND_DECR_IF(len == 0, ZCBOR_ERR_ADDITIONAL_INVAL); // additional_len() did not recognize the additional value.
- FAIL_AND_DECR_IF((state->payload + len) > state->payload_end,
- ZCBOR_ERR_NO_PAYLOAD);
- #ifdef CONFIG_BIG_ENDIAN
- memcpy(&u8_result[result_len - len], state->payload, len);
- #else
- for (uint_fast32_t i = 0; i < len; i++) {
- u8_result[i] = (state->payload)[len - i - 1];
- }
- #endif /* CONFIG_BIG_ENDIAN */
- (state->payload) += len;
- }
- (state->elem_count)--;
- return true;
- }
- bool zcbor_int_decode(zcbor_state_t *state, void *result_int, size_t int_size)
- {
- INITIAL_CHECKS();
- zcbor_major_type_t major_type = MAJOR_TYPE(*state->payload);
- uint8_t *result_uint8 = (uint8_t *)result_int;
- int8_t *result_int8 = (int8_t *)result_int;
- if (major_type != ZCBOR_MAJOR_TYPE_PINT
- && major_type != ZCBOR_MAJOR_TYPE_NINT) {
- /* Value to be read doesn't have the right type. */
- ZCBOR_ERR(ZCBOR_ERR_WRONG_TYPE);
- }
- if (!value_extract(state, result_int, int_size)) {
- ZCBOR_FAIL();
- }
- #ifdef CONFIG_BIG_ENDIAN
- if (result_int8[0] < 0) {
- #else
- if (result_int8[int_size - 1] < 0) {
- #endif
- /* Value is too large to fit in a signed integer. */
- ERR_RESTORE(ZCBOR_ERR_INT_SIZE);
- }
- if (major_type == ZCBOR_MAJOR_TYPE_NINT) {
- /* Convert from CBOR's representation by flipping all bits. */
- for (int i = 0; i < int_size; i++) {
- result_uint8[i] = (uint8_t)~result_uint8[i];
- }
- }
- return true;
- }
- bool zcbor_int32_decode(zcbor_state_t *state, int32_t *result)
- {
- return zcbor_int_decode(state, result, sizeof(*result));
- }
- bool zcbor_int64_decode(zcbor_state_t *state, int64_t *result)
- {
- return zcbor_int_decode(state, result, sizeof(*result));
- }
- bool zcbor_uint32_decode(zcbor_state_t *state, uint32_t *result)
- {
- INITIAL_CHECKS_WITH_TYPE(ZCBOR_MAJOR_TYPE_PINT);
- if (!value_extract(state, result, sizeof(*result))) {
- ZCBOR_FAIL();
- }
- return true;
- }
- bool zcbor_int32_expect_union(zcbor_state_t *state, int32_t result)
- {
- if (!zcbor_union_elem_code(state)) {
- ZCBOR_FAIL();
- }
- return zcbor_int32_expect(state, result);
- }
- bool zcbor_int64_expect_union(zcbor_state_t *state, int64_t result)
- {
- if (!zcbor_union_elem_code(state)) {
- ZCBOR_FAIL();
- }
- return zcbor_int64_expect(state, result);
- }
- bool zcbor_uint32_expect_union(zcbor_state_t *state, uint32_t result)
- {
- if (!zcbor_union_elem_code(state)) {
- ZCBOR_FAIL();
- }
- return zcbor_uint32_expect(state, result);
- }
- bool zcbor_uint64_expect_union(zcbor_state_t *state, uint64_t result)
- {
- if (!zcbor_union_elem_code(state)) {
- ZCBOR_FAIL();
- }
- return zcbor_uint64_expect(state, result);
- }
- bool zcbor_int32_expect(zcbor_state_t *state, int32_t result)
- {
- return zcbor_int64_expect(state, result);
- }
- bool zcbor_int64_expect(zcbor_state_t *state, int64_t result)
- {
- int64_t value;
- if (!zcbor_int64_decode(state, &value)) {
- ZCBOR_FAIL();
- }
- if (value != result) {
- zcbor_print("%" PRIi64 " != %" PRIi64 "\r\n", value, result);
- ERR_RESTORE(ZCBOR_ERR_WRONG_VALUE);
- }
- return true;
- }
- bool zcbor_uint64_decode(zcbor_state_t *state, uint64_t *result)
- {
- INITIAL_CHECKS_WITH_TYPE(ZCBOR_MAJOR_TYPE_PINT);
- if (!value_extract(state, result, sizeof(*result))) {
- ZCBOR_FAIL();
- }
- return true;
- }
- #ifdef ZCBOR_SUPPORTS_SIZE_T
- bool zcbor_size_decode(zcbor_state_t *state, size_t *result)
- {
- return value_extract(state, result, sizeof(size_t));
- }
- #endif
- bool zcbor_uint32_expect(zcbor_state_t *state, uint32_t result)
- {
- return zcbor_uint64_expect(state, result);
- }
- bool zcbor_uint64_expect(zcbor_state_t *state, uint64_t result)
- {
- uint64_t value;
- if (!zcbor_uint64_decode(state, &value)) {
- ZCBOR_FAIL();
- }
- if (value != result) {
- zcbor_print("%" PRIu64 " != %" PRIu64 "\r\n", value, result);
- ERR_RESTORE(ZCBOR_ERR_WRONG_VALUE);
- }
- return true;
- }
- #ifdef ZCBOR_SUPPORTS_SIZE_T
- bool zcbor_size_expect(zcbor_state_t *state, size_t result)
- {
- return zcbor_uint64_expect(state, result);
- }
- #endif
- static bool str_start_decode(zcbor_state_t *state,
- struct zcbor_string *result, zcbor_major_type_t exp_major_type)
- {
- INITIAL_CHECKS_WITH_TYPE(exp_major_type);
- if (!value_extract(state, &result->len, sizeof(result->len))) {
- ZCBOR_FAIL();
- }
- result->value = state->payload;
- return true;
- }
- static bool str_overflow_check(zcbor_state_t *state, struct zcbor_string *result)
- {
- if (result->len > (state->payload_end - state->payload)) {
- zcbor_print("error: 0x%zu > 0x%zu\r\n",
- result->len,
- (state->payload_end - state->payload));
- ERR_RESTORE(ZCBOR_ERR_NO_PAYLOAD);
- }
- return true;
- }
- bool zcbor_bstr_start_decode(zcbor_state_t *state, struct zcbor_string *result)
- {
- struct zcbor_string dummy;
- if (result == NULL) {
- result = &dummy;
- }
- if(!str_start_decode(state, result, ZCBOR_MAJOR_TYPE_BSTR)) {
- ZCBOR_FAIL();
- }
- if (!str_overflow_check(state, result)) {
- ZCBOR_FAIL();
- }
- if (!zcbor_new_backup(state, ZCBOR_MAX_ELEM_COUNT)) {
- FAIL_RESTORE();
- }
- state->payload_end = result->value + result->len;
- return true;
- }
- bool zcbor_bstr_end_decode(zcbor_state_t *state)
- {
- ZCBOR_ERR_IF(state->payload != state->payload_end, ZCBOR_ERR_PAYLOAD_NOT_CONSUMED);
- if (!zcbor_process_backup(state,
- ZCBOR_FLAG_RESTORE | ZCBOR_FLAG_CONSUME | ZCBOR_FLAG_TRANSFER_PAYLOAD,
- ZCBOR_MAX_ELEM_COUNT)) {
- ZCBOR_FAIL();
- }
- return true;
- }
- static void partition_fragment(const zcbor_state_t *state,
- struct zcbor_string_fragment *result)
- {
- result->fragment.len = MIN(result->fragment.len,
- (size_t)state->payload_end - (size_t)state->payload);
- }
- static bool start_decode_fragment(zcbor_state_t *state,
- struct zcbor_string_fragment *result,
- zcbor_major_type_t exp_major_type)
- {
- if(!str_start_decode(state, &result->fragment, exp_major_type)) {
- ZCBOR_FAIL();
- }
- result->offset = 0;
- result->total_len = result->fragment.len;
- partition_fragment(state, result);
- state->payload_end = state->payload + result->fragment.len;
- return true;
- }
- bool zcbor_bstr_start_decode_fragment(zcbor_state_t *state,
- struct zcbor_string_fragment *result)
- {
- if (!start_decode_fragment(state, result, ZCBOR_MAJOR_TYPE_BSTR)) {
- ZCBOR_FAIL();
- }
- if (!zcbor_new_backup(state, ZCBOR_MAX_ELEM_COUNT)) {
- FAIL_RESTORE();
- }
- return true;
- }
- void zcbor_next_fragment(zcbor_state_t *state,
- struct zcbor_string_fragment *prev_fragment,
- struct zcbor_string_fragment *result)
- {
- memcpy(result, prev_fragment, sizeof(*result));
- result->fragment.value = state->payload_mut;
- result->offset += prev_fragment->fragment.len;
- result->fragment.len = result->total_len - result->offset;
- partition_fragment(state, result);
- zcbor_print("New fragment length %zu\r\n", result->fragment.len);
- state->payload += result->fragment.len;
- }
- void zcbor_bstr_next_fragment(zcbor_state_t *state,
- struct zcbor_string_fragment *prev_fragment,
- struct zcbor_string_fragment *result)
- {
- memcpy(result, prev_fragment, sizeof(*result));
- result->fragment.value = state->payload_mut;
- result->offset += prev_fragment->fragment.len;
- result->fragment.len = result->total_len - result->offset;
- partition_fragment(state, result);
- zcbor_print("fragment length %zu\r\n", result->fragment.len);
- state->payload_end = state->payload + result->fragment.len;
- }
- bool zcbor_is_last_fragment(const struct zcbor_string_fragment *fragment)
- {
- return (fragment->total_len == (fragment->offset + fragment->fragment.len));
- }
- static bool str_decode(zcbor_state_t *state, struct zcbor_string *result,
- zcbor_major_type_t exp_major_type)
- {
- if (!str_start_decode(state, result, exp_major_type)) {
- ZCBOR_FAIL();
- }
- if (!str_overflow_check(state, result)) {
- ZCBOR_FAIL();
- }
- state->payload += result->len;
- return true;
- }
- static bool str_decode_fragment(zcbor_state_t *state, struct zcbor_string_fragment *result,
- zcbor_major_type_t exp_major_type)
- {
- if (!start_decode_fragment(state, result, exp_major_type)) {
- ZCBOR_FAIL();
- }
- (state->payload) += result->fragment.len;
- return true;
- }
- static bool str_expect(zcbor_state_t *state, struct zcbor_string *result,
- zcbor_major_type_t exp_major_type)
- {
- struct zcbor_string tmp_result;
- if (!str_decode(state, &tmp_result, exp_major_type)) {
- ZCBOR_FAIL();
- }
- if ((tmp_result.len != result->len)
- || memcmp(result->value, tmp_result.value, tmp_result.len)) {
- ERR_RESTORE(ZCBOR_ERR_WRONG_VALUE);
- }
- return true;
- }
- bool zcbor_bstr_decode(zcbor_state_t *state, struct zcbor_string *result)
- {
- return str_decode(state, result, ZCBOR_MAJOR_TYPE_BSTR);
- }
- bool zcbor_bstr_decode_fragment(zcbor_state_t *state, struct zcbor_string_fragment *result)
- {
- return str_decode_fragment(state, result, ZCBOR_MAJOR_TYPE_BSTR);
- }
- bool zcbor_bstr_expect(zcbor_state_t *state, struct zcbor_string *result)
- {
- return str_expect(state, result, ZCBOR_MAJOR_TYPE_BSTR);
- }
- bool zcbor_tstr_decode(zcbor_state_t *state, struct zcbor_string *result)
- {
- return str_decode(state, result, ZCBOR_MAJOR_TYPE_TSTR);
- }
- bool zcbor_tstr_decode_fragment(zcbor_state_t *state, struct zcbor_string_fragment *result)
- {
- return str_decode_fragment(state, result, ZCBOR_MAJOR_TYPE_TSTR);
- }
- bool zcbor_tstr_expect(zcbor_state_t *state, struct zcbor_string *result)
- {
- return str_expect(state, result, ZCBOR_MAJOR_TYPE_TSTR);
- }
- static bool list_map_start_decode(zcbor_state_t *state,
- zcbor_major_type_t exp_major_type)
- {
- uint_fast32_t new_elem_count;
- bool indefinite_length_array = false;
- INITIAL_CHECKS_WITH_TYPE(exp_major_type);
- if (ADDITIONAL(*state->payload) == ZCBOR_VALUE_IS_INDEFINITE_LENGTH) {
- /* Indefinite length array. */
- new_elem_count = ZCBOR_LARGE_ELEM_COUNT;
- ZCBOR_ERR_IF(state->elem_count == 0, ZCBOR_ERR_LOW_ELEM_COUNT);
- indefinite_length_array = true;
- state->payload++;
- state->elem_count--;
- } else {
- if (!value_extract(state, &new_elem_count, sizeof(new_elem_count))) {
- ZCBOR_FAIL();
- }
- }
- if (!zcbor_new_backup(state, new_elem_count)) {
- FAIL_RESTORE();
- }
- state->indefinite_length_array = indefinite_length_array;
- return true;
- }
- bool zcbor_list_start_decode(zcbor_state_t *state)
- {
- return list_map_start_decode(state, ZCBOR_MAJOR_TYPE_LIST);
- }
- bool zcbor_map_start_decode(zcbor_state_t *state)
- {
- bool ret = list_map_start_decode(state, ZCBOR_MAJOR_TYPE_MAP);
- if (ret && !state->indefinite_length_array) {
- if (state->elem_count >= (ZCBOR_MAX_ELEM_COUNT / 2)) {
- /* The new elem_count is too large. */
- ERR_RESTORE(ZCBOR_ERR_INT_SIZE);
- }
- state->elem_count *= 2;
- }
- return ret;
- }
- static bool array_end_expect(zcbor_state_t *state)
- {
- INITIAL_CHECKS();
- ZCBOR_ERR_IF(*state->payload != 0xFF, ZCBOR_ERR_WRONG_TYPE);
- state->payload++;
- return true;
- }
- static bool list_map_end_decode(zcbor_state_t *state)
- {
- uint_fast32_t max_elem_count = 0;
- if (state->indefinite_length_array) {
- if (!array_end_expect(state)) {
- ZCBOR_FAIL();
- }
- max_elem_count = ZCBOR_MAX_ELEM_COUNT;
- state->indefinite_length_array = false;
- }
- if (!zcbor_process_backup(state,
- ZCBOR_FLAG_RESTORE | ZCBOR_FLAG_CONSUME | ZCBOR_FLAG_TRANSFER_PAYLOAD,
- max_elem_count)) {
- ZCBOR_FAIL();
- }
- return true;
- }
- bool zcbor_list_end_decode(zcbor_state_t *state)
- {
- return list_map_end_decode(state);
- }
- bool zcbor_map_end_decode(zcbor_state_t *state)
- {
- return list_map_end_decode(state);
- }
- bool zcbor_list_map_end_force_decode(zcbor_state_t *state)
- {
- if (!zcbor_process_backup(state,
- ZCBOR_FLAG_RESTORE | ZCBOR_FLAG_CONSUME | ZCBOR_FLAG_TRANSFER_PAYLOAD,
- ZCBOR_MAX_ELEM_COUNT)) {
- ZCBOR_FAIL();
- }
- return true;
- }
- static bool primx_expect(zcbor_state_t *state, uint8_t result)
- {
- uint32_t value;
- INITIAL_CHECKS_WITH_TYPE(ZCBOR_MAJOR_TYPE_PRIM);
- if (!value_extract(state, &value, sizeof(value))) {
- ZCBOR_FAIL();
- }
- if (value != result) {
- return false;
- //ERR_RESTORE(ZCBOR_ERR_WRONG_VALUE);
- }
- return true;
- }
- bool zcbor_nil_expect(zcbor_state_t *state, void *unused)
- {
- if (!primx_expect(state, 22)) {
- ZCBOR_FAIL();
- }
- return true;
- }
- bool zcbor_undefined_expect(zcbor_state_t *state, void *unused)
- {
- if (!primx_expect(state, 23)) {
- ZCBOR_FAIL();
- }
- return true;
- }
- bool zcbor_bool_decode(zcbor_state_t *state, bool *result)
- {
- if (zcbor_bool_expect(state, false)) {
- *result = false;
- } else if (zcbor_bool_expect(state, true)) {
- *result = true;
- } else {
- ZCBOR_FAIL();
- }
- zcbor_print("boolval: %u\r\n", *result);
- return true;
- }
- bool zcbor_bool_expect(zcbor_state_t *state, bool result)
- {
- if (!primx_expect(state, (uint8_t)(!!result) + ZCBOR_BOOL_TO_PRIM)) {
- //ZCBOR_FAIL();
- return false;
- }
- return true;
- }
- bool zcbor_float32_decode(zcbor_state_t *state, float *result)
- {
- INITIAL_CHECKS_WITH_TYPE(ZCBOR_MAJOR_TYPE_PRIM);
- ZCBOR_ERR_IF(ADDITIONAL(*state->payload) != ZCBOR_VALUE_IS_4_BYTES, ZCBOR_ERR_FLOAT_SIZE);
- if (!value_extract(state, result, sizeof(*result))) {
- ZCBOR_FAIL();
- }
- return true;
- }
- bool zcbor_float32_expect(zcbor_state_t *state, float result)
- {
- float value;
- if (!zcbor_float32_decode(state, &value)) {
- ZCBOR_FAIL();
- }
- if (value != result) {
- ERR_RESTORE(ZCBOR_ERR_WRONG_VALUE);
- }
- return true;
- }
- bool zcbor_float64_decode(zcbor_state_t *state, double *result)
- {
- INITIAL_CHECKS_WITH_TYPE(ZCBOR_MAJOR_TYPE_PRIM);
- ZCBOR_ERR_IF(ADDITIONAL(*state->payload) != ZCBOR_VALUE_IS_8_BYTES, ZCBOR_ERR_FLOAT_SIZE);
- if (!value_extract(state, result, sizeof(*result))) {
- ZCBOR_FAIL();
- }
- return true;
- }
- bool zcbor_float64_expect(zcbor_state_t *state, double result)
- {
- double value;
- if (!zcbor_float64_decode(state, &value)) {
- ZCBOR_FAIL();
- }
- if (value != result) {
- ERR_RESTORE(ZCBOR_ERR_WRONG_VALUE);
- }
- return true;
- }
- bool zcbor_float_decode(zcbor_state_t *state, double *result)
- {
- float float_result;
- if (zcbor_float32_decode(state, &float_result)) {
- *result = (double)float_result;
- } else if (!zcbor_float64_decode(state, result)) {
- ZCBOR_FAIL();
- }
- return true;
- }
- bool zcbor_float_expect(zcbor_state_t *state, double result)
- {
- if (zcbor_float32_expect(state, (float)result)) {
- /* Do nothing */
- } else if (!zcbor_float64_expect(state, result)) {
- ZCBOR_FAIL();
- }
- return true;
- }
- bool zcbor_any_skip(zcbor_state_t *state, void *result)
- {
- zcbor_assert_state(result == NULL,
- "'any' type cannot be returned, only skipped.\r\n");
- INITIAL_CHECKS();
- zcbor_major_type_t major_type = MAJOR_TYPE(*state->payload);
- uint8_t additional = ADDITIONAL(*state->payload);
- uint_fast32_t value;
- uint_fast32_t num_decode;
- uint_fast32_t temp_elem_count;
- uint_fast32_t elem_count_bak = state->elem_count;
- uint8_t const *payload_bak = state->payload;
- uint64_t tag_dummy;
- payload_bak = state->payload;
- if (!zcbor_multi_decode(0, ZCBOR_LARGE_ELEM_COUNT, &num_decode,
- (zcbor_decoder_t *)zcbor_tag_decode, state,
- (void *)&tag_dummy, 0)) {
- state->elem_count = elem_count_bak;
- state->payload = payload_bak;
- ZCBOR_FAIL();
- }
- if ((major_type == ZCBOR_MAJOR_TYPE_MAP) || (major_type == ZCBOR_MAJOR_TYPE_LIST)) {
- if (additional == ZCBOR_VALUE_IS_INDEFINITE_LENGTH) {
- ZCBOR_ERR_IF(state->elem_count == 0, ZCBOR_ERR_LOW_ELEM_COUNT);
- state->payload++;
- state->elem_count--;
- temp_elem_count = state->elem_count;
- payload_bak = state->payload;
- state->elem_count = ZCBOR_LARGE_ELEM_COUNT;
- if (!zcbor_multi_decode(0, ZCBOR_LARGE_ELEM_COUNT, &num_decode,
- (zcbor_decoder_t *)zcbor_any_skip, state,
- NULL, 0)
- || (state->payload >= state->payload_end)
- || !(*(state->payload++) == 0xFF)) {
- state->elem_count = elem_count_bak;
- state->payload = payload_bak;
- ZCBOR_FAIL();
- }
- state->elem_count = temp_elem_count;
- return true;
- }
- }
- if (!value_extract(state, &value, sizeof(value))) {
- /* Can happen because of elem_count (or payload_end) */
- ZCBOR_FAIL();
- }
- switch (major_type) {
- case ZCBOR_MAJOR_TYPE_BSTR:
- case ZCBOR_MAJOR_TYPE_TSTR:
- /* 'value' is the length of the BSTR or TSTR */
- if (value > (state->payload_end - state->payload)) {
- ZCBOR_ERR(ZCBOR_ERR_NO_PAYLOAD);
- }
- (state->payload) += value;
- break;
- case ZCBOR_MAJOR_TYPE_MAP:
- value *= 2; /* Because all members have a key. */
- /* Fallthrough */
- case ZCBOR_MAJOR_TYPE_LIST:
- temp_elem_count = state->elem_count;
- state->elem_count = value;
- if (!zcbor_multi_decode(value, value, &num_decode,
- (zcbor_decoder_t *)zcbor_any_skip, state,
- NULL, 0)) {
- state->elem_count = elem_count_bak;
- state->payload = payload_bak;
- ZCBOR_FAIL();
- }
- state->elem_count = temp_elem_count;
- break;
- default:
- /* Do nothing */
- break;
- }
- return true;
- }
- bool zcbor_tag_decode(zcbor_state_t *state, uint32_t *result)
- {
- INITIAL_CHECKS_WITH_TYPE(ZCBOR_MAJOR_TYPE_TAG);
- if (!value_extract(state, result, sizeof(*result))) {
- ZCBOR_FAIL();
- }
- state->elem_count++;
- return true;
- }
- bool zcbor_tag_expect(zcbor_state_t *state, uint32_t result)
- {
- uint32_t tag_val;
- if (!zcbor_tag_decode(state, &tag_val)) {
- ZCBOR_FAIL();
- }
- if (tag_val != result) {
- ERR_RESTORE(ZCBOR_ERR_WRONG_VALUE);
- }
- return true;
- }
- bool zcbor_multi_decode(uint_fast32_t min_decode,
- uint_fast32_t max_decode,
- uint_fast32_t *num_decode,
- zcbor_decoder_t decoder,
- zcbor_state_t *state,
- void *result,
- uint_fast32_t result_len)
- {
- ZCBOR_CHECK_ERROR();
- for (uint_fast32_t i = 0; i < max_decode; i++) {
- uint8_t const *payload_bak = state->payload;
- uint_fast32_t elem_count_bak = state->elem_count;
- if (!decoder(state,
- (uint8_t *)result + i*result_len)) {
- *num_decode = i;
- state->payload = payload_bak;
- state->elem_count = elem_count_bak;
- ZCBOR_ERR_IF(i < min_decode, ZCBOR_ERR_ITERATIONS);
- zcbor_print("Found %" PRIuFAST32 " elements.\r\n", i);
- return true;
- }
- }
- zcbor_print("Found %" PRIuFAST32 " elements.\r\n", max_decode);
- *num_decode = max_decode;
- return true;
- }
- bool zcbor_present_decode(uint_fast32_t *present,
- zcbor_decoder_t decoder,
- zcbor_state_t *state,
- void *result)
- {
- uint_fast32_t num_decode;
- bool retval = zcbor_multi_decode(0, 1, &num_decode, decoder, state, result, 0);
- zcbor_assert_state(retval, "zcbor_multi_decode should not fail with these parameters.\r\n");
- *present = num_decode;
- return retval;
- }
- void zcbor_new_decode_state(zcbor_state_t *state_array, uint_fast32_t n_states,
- const uint8_t *payload, size_t payload_len, uint_fast32_t elem_count)
- {
- zcbor_new_state(state_array, n_states, payload, payload_len, elem_count);
- }
|