2014-07-14 21:40:04 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Freie Universität Berlin
|
|
|
|
* Copyright (C) 2014 Kevin Funk <kfunk@kde.org>
|
|
|
|
* Copyright (C) 2014 Jana Cavojska <jana.cavojska9@gmail.com>
|
|
|
|
*
|
2014-08-23 15:43:13 +02:00
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup cbor
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @file
|
|
|
|
* @brief Implementation of a CBOR serializer/deserializer in C
|
|
|
|
*
|
|
|
|
* @author Kevin Funk <kfunk@kde.org>
|
|
|
|
* @author Jana Cavojska <jana.cavojska9@gmail.com>
|
2014-12-05 20:15:52 +01:00
|
|
|
* @author Oliver Hahm <oliver.hahm@inria.fr>
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* This is an implementation suited for constrained devices
|
|
|
|
* Characteristics:
|
2014-12-05 19:54:59 +01:00
|
|
|
* - No dynamic memory allocation (i.e. no calls to @e malloc, @e free) used
|
|
|
|
* throughout the implementation
|
|
|
|
* - User may allocate static buffers, this implementation uses the space
|
|
|
|
* provided by them (cf. @ref cbor_stream_t)
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* @par Supported types (categorized by major type (MT)):
|
|
|
|
*
|
|
|
|
* - Major type 0 (unsigned integer): Full support. Relevant functions:
|
|
|
|
* - cbor_serialize_int(), cbor_deserialize_int()
|
|
|
|
* - cbor_serialize_uint64_t(), cbor_deserialize_uint64_t()
|
|
|
|
*
|
|
|
|
* - Major type 1 (negative integer): Full support. Relevant functions:
|
|
|
|
* - cbor_serialize_int(), cbor_deserialize_int()
|
|
|
|
* - cbor_serialize_int64_t(), cbor_deserialize_int64_t()
|
|
|
|
*
|
|
|
|
* - Major type 2 (byte string): Full support. Relevant functions:
|
|
|
|
* - cbor_serialize_byte_string(), cbor_deserialize_byte_string()
|
|
|
|
*
|
|
|
|
* - Major type 3 (unicode string): Basic support (see below). Relevant functions:
|
|
|
|
* - cbor_serialize_unicode_string(), cbor_deserialize_unicode_string()
|
|
|
|
*
|
|
|
|
* - Major type 4 (array of data items): Full support. Relevant functions:
|
|
|
|
* - cbor_serialize_array(), cbor_deserialize_array()
|
2014-12-05 19:54:59 +01:00
|
|
|
* - cbor_serialize_indefinite_array(), cbor_deserialize_indefinite_array(),
|
|
|
|
* cbor_at_break()
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* - Major type 5 (map of pairs of data items): Full support. Relevant functions:
|
|
|
|
* - cbor_serialize_map(), cbor_deserialize_map()
|
2014-12-05 19:54:59 +01:00
|
|
|
* - cbor_serialize_indefinite_map(), cbor_deserialize_indefinite_map(),
|
|
|
|
* cbor_at_break()
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
2014-12-05 19:54:59 +01:00
|
|
|
* - Major type 6 (optional semantic tagging of other major types): Basic
|
|
|
|
* support (see below). Relevant functions:
|
2014-07-14 21:40:04 +02:00
|
|
|
* - cbor_write_tag()
|
|
|
|
* - cbor_deserialize_date_time()
|
|
|
|
* - cbor_serialize_date_time()
|
|
|
|
*
|
2014-12-05 19:54:59 +01:00
|
|
|
* - Major type 7 (floating-point numbers and values with no content): Basic
|
|
|
|
* support (see below). Relevant functions:
|
2014-07-14 21:40:04 +02:00
|
|
|
* - cbor_serialize_float_half(), cbor_deserialize_float_half()
|
|
|
|
* - cbor_serialize_float(), cbor_deserialize_float()
|
|
|
|
* - cbor_serialize_double(), cbor_deserialize_double()
|
|
|
|
* - cbor_serialize_bool(), cbor_deserialize_bool()
|
|
|
|
*
|
|
|
|
* @par Notes about major type 3:
|
2014-12-05 19:54:59 +01:00
|
|
|
* Since we do not have a standardised C type for representing Unicode code
|
|
|
|
* points, we just provide API to serialize/deserialize @e char* arrays. The
|
|
|
|
* user then has to transform that into a meaningful representation
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* @par Notes about major type 6 (cf. https://tools.ietf.org/html/rfc7049#section-2.4):
|
2014-12-05 19:54:59 +01:00
|
|
|
* Encoding date and time: date/time strings that follow the standard format
|
|
|
|
* described in Section 3.3 of [RFC3339]:
|
2014-07-14 21:40:04 +02:00
|
|
|
* 2003-12-13T18:30:02Z - supported
|
|
|
|
* 2003-12-13T18:30:02.25Z - not supported
|
|
|
|
* 2003-12-13T18:30:02+01:00 - not supported
|
|
|
|
* 2003-12-13T18:30:02.25+01:00 - not supported
|
|
|
|
* Since we do not have C types for representing bignums/bigfloats/decimal-fraction
|
|
|
|
* we do not provide API to serialize/deserialize them at all.
|
2014-12-05 19:54:59 +01:00
|
|
|
* You can still read out the actual data item behind the tag (via
|
|
|
|
* cbor_deserialize_byte_string()) and interpret it yourself.
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
2014-12-05 19:54:59 +01:00
|
|
|
* @par Notes about major type 7 and simple values
|
|
|
|
* (cf. https://tools.ietf.org/html/rfc7049#section-2.3)
|
2014-07-14 21:40:04 +02:00
|
|
|
* Simple values:
|
|
|
|
* - 0-19: (Unassigned) - No support
|
2014-12-05 19:54:59 +01:00
|
|
|
* - 20,21: True, False - Supported (see cbor_serialize_bool(),
|
|
|
|
* cbor_deserialize_bool())
|
2014-07-14 21:40:04 +02:00
|
|
|
* - 22,23: Null, Undefined - No support (what's the use-case?)
|
|
|
|
* - 24-31: (Reserved) - No support
|
|
|
|
* - 32-255: (Unassigned) - No support
|
|
|
|
*
|
2015-07-31 21:56:08 +02:00
|
|
|
* @todo API for Indefinite-Length Byte Strings and Text Strings
|
2014-07-14 21:40:04 +02:00
|
|
|
* (see https://tools.ietf.org/html/rfc7049#section-2.2.2)
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef CBOR_H
|
|
|
|
#define CBOR_H
|
|
|
|
|
|
|
|
#ifndef CBOR_NO_CTIME
|
|
|
|
/* 'strptime' is only declared when this macro is defined */
|
|
|
|
#define _XOPEN_SOURCE
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdlib.h>
|
2014-10-10 11:51:11 +02:00
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
#ifndef CBOR_NO_CTIME
|
|
|
|
#include <time.h>
|
|
|
|
#endif /* CBOR_NO_CTIME */
|
|
|
|
|
2014-10-10 11:51:11 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
/**
|
|
|
|
* @brief Struct containing CBOR-encoded data
|
|
|
|
*
|
|
|
|
* A typical usage of CBOR looks like:
|
|
|
|
* @code
|
|
|
|
* unsigned char data[1024];
|
|
|
|
* cbor_stream_t stream;
|
|
|
|
* cbor_init(&stream, data, sizeof(data));
|
|
|
|
*
|
|
|
|
* cbor_serialize_int(&stream, 5);
|
|
|
|
* (...)
|
|
|
|
* <data contains CBOR encoded items now>
|
|
|
|
*
|
|
|
|
* cbor_destroy(&stream);
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* @sa cbor_init
|
|
|
|
* @sa cbor_clear
|
|
|
|
* @sa cbor_destroy
|
|
|
|
*/
|
2016-04-07 21:24:33 +02:00
|
|
|
typedef struct {
|
2014-12-05 19:47:12 +01:00
|
|
|
/** Array containing CBOR encoded data */
|
2014-07-14 21:40:04 +02:00
|
|
|
unsigned char *data;
|
2014-12-05 19:47:12 +01:00
|
|
|
/** Size of the array */
|
2014-07-14 21:40:04 +02:00
|
|
|
size_t size;
|
2014-12-05 19:47:12 +01:00
|
|
|
/** Index to the next free byte */
|
2014-07-14 21:40:04 +02:00
|
|
|
size_t pos;
|
|
|
|
} cbor_stream_t;
|
|
|
|
|
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Initialize cbor struct
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* @note Does *not* take ownership of @p buffer
|
2014-12-05 19:47:12 +01:00
|
|
|
* @param[in] stream The cbor struct to initialize
|
|
|
|
* @param[in] buffer The buffer used for storing CBOR-encoded data
|
|
|
|
* @param[in] size The size of buffer @p buffer
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
|
|
|
void cbor_init(cbor_stream_t *stream, unsigned char *buffer, size_t size);
|
|
|
|
|
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Clear cbor struct
|
|
|
|
*
|
|
|
|
* Sets pos to zero
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
2014-12-05 19:47:12 +01:00
|
|
|
* @param[in, out] stream Pointer to the cbor struct
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
|
|
|
void cbor_clear(cbor_stream_t *stream);
|
|
|
|
|
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Destroy the cbor struct
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* @note Does *not* free data
|
2014-12-05 19:47:12 +01:00
|
|
|
*
|
|
|
|
* @param[in, out] stream Pointer to the cbor struct
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
|
|
|
void cbor_destroy(cbor_stream_t *stream);
|
|
|
|
|
|
|
|
#ifndef CBOR_NO_PRINT
|
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Print @p stream in hex representation
|
|
|
|
*
|
|
|
|
* @param[in] stream Pointer to the cbor struct
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
|
|
|
void cbor_stream_print(const cbor_stream_t *stream);
|
|
|
|
|
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Decode CBOR from @p stream
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
2014-12-05 19:54:59 +01:00
|
|
|
* This method interprets the data and prints each item in its natural
|
|
|
|
* representation
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* Example output:
|
|
|
|
* @code
|
|
|
|
* Data:
|
|
|
|
* (int, 1)
|
|
|
|
* (bool, 1)
|
|
|
|
* (float, 1.099609)
|
|
|
|
* (tag: 0, date/time string: "Mon Jul 14 19:07:40 2014")
|
|
|
|
* (tag: 1, date/time epoch: 1405357660)
|
|
|
|
* @endcode
|
2014-12-05 19:47:12 +01:00
|
|
|
*
|
|
|
|
* @param[in] stream Pointer to the cbor struct
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
|
|
|
void cbor_stream_decode(cbor_stream_t *stream);
|
|
|
|
#endif /* CBOR_NO_PRINT */
|
|
|
|
|
2014-12-05 20:15:52 +01:00
|
|
|
/**
|
|
|
|
* @brief Serializes an integer
|
|
|
|
*
|
|
|
|
* @param[out] stream The destination stream for serializing the array
|
|
|
|
* @param[in] val The integer to serialize
|
|
|
|
*
|
|
|
|
* @return Number of bytes written to stream @p stream
|
|
|
|
*/
|
|
|
|
size_t cbor_serialize_int(cbor_stream_t *stream, int val);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Deserialize integers from @p stream to @p val
|
|
|
|
*
|
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream where to start deserializing
|
|
|
|
* @param[out] val Pointer to destination array
|
|
|
|
*
|
2016-10-20 16:10:32 +02:00
|
|
|
* @return Number of bytes read from @p stream
|
2014-12-05 20:15:52 +01:00
|
|
|
*/
|
2014-12-05 19:54:59 +01:00
|
|
|
size_t cbor_deserialize_int(const cbor_stream_t *stream, size_t offset,
|
|
|
|
int *val);
|
2014-12-05 20:15:52 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Serializes an unsigned 64 bit value
|
|
|
|
*
|
|
|
|
* @param[out] stream The destination stream for serializing the array
|
|
|
|
* @param[in] val The 64 bit integer to serialize
|
|
|
|
*
|
|
|
|
* @return Number of bytes written to stream @p stream
|
|
|
|
*/
|
|
|
|
size_t cbor_serialize_uint64_t(cbor_stream_t *stream, uint64_t val);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Deserialize unsigned 64 bit values from @p stream to @p val
|
|
|
|
*
|
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream where to start deserializing
|
|
|
|
* @param[out] val Pointer to destination array
|
|
|
|
*
|
2016-10-20 16:10:32 +02:00
|
|
|
* @return Number of bytes read from @p stream
|
2014-12-05 20:15:52 +01:00
|
|
|
*/
|
2014-12-05 19:54:59 +01:00
|
|
|
size_t cbor_deserialize_uint64_t(const cbor_stream_t *stream, size_t offset,
|
|
|
|
uint64_t *val);
|
2014-12-05 20:15:52 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Serializes a signed 64 bit value
|
|
|
|
*
|
|
|
|
* @param[out] stream The destination stream for serializing the array
|
|
|
|
* @param[in] val The 64 bit integer to serialize
|
|
|
|
*
|
|
|
|
* @return Number of bytes written to stream @p stream
|
|
|
|
*/
|
|
|
|
size_t cbor_serialize_int64_t(cbor_stream_t *stream, int64_t val);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Deserialize signed 64 bit values from @p stream to @p val
|
|
|
|
*
|
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream where to start deserializing
|
|
|
|
* @param[out] val Pointer to destination array
|
|
|
|
*
|
2016-10-20 16:10:32 +02:00
|
|
|
* @return Number of bytes read from @p stream
|
2014-12-05 20:15:52 +01:00
|
|
|
*/
|
2014-12-05 19:54:59 +01:00
|
|
|
size_t cbor_deserialize_int64_t(const cbor_stream_t *stream, size_t offset,
|
|
|
|
int64_t *val);
|
2014-12-05 20:15:52 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Serializes a boolean value
|
|
|
|
*
|
|
|
|
* @param[out] stream The destination stream for serializing the array
|
|
|
|
* @param[in] val The boolean value to serialize
|
|
|
|
*
|
|
|
|
* @return Number of bytes written to stream @p stream
|
|
|
|
*/
|
|
|
|
size_t cbor_serialize_bool(cbor_stream_t *stream, bool val);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Deserialize boolean values from @p stream to @p val
|
|
|
|
*
|
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream where to start deserializing
|
|
|
|
* @param[out] val Pointer to destination array
|
|
|
|
*
|
2016-10-20 16:10:32 +02:00
|
|
|
* @return Number of bytes read from @p stream
|
2014-12-05 20:15:52 +01:00
|
|
|
*/
|
2014-12-05 19:54:59 +01:00
|
|
|
size_t cbor_deserialize_bool(const cbor_stream_t *stream, size_t offset,
|
|
|
|
bool *val);
|
2014-12-05 20:15:52 +01:00
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
#ifndef CBOR_NO_FLOAT
|
2014-12-05 20:15:52 +01:00
|
|
|
/**
|
|
|
|
* @brief Serializes a half-width floating point value
|
|
|
|
*
|
|
|
|
* @param[out] stream The destination stream for serializing the array
|
|
|
|
* @param[in] val The half-width floating point value to serialize
|
|
|
|
*
|
|
|
|
* @return Number of bytes written to stream @p stream
|
|
|
|
*/
|
|
|
|
size_t cbor_serialize_float_half(cbor_stream_t *stream, float val);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Deserialize half-width floating point values values from @p stream to
|
|
|
|
* @p val
|
|
|
|
*
|
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream where to start deserializing
|
|
|
|
* @param[out] val Pointer to destination array
|
|
|
|
*
|
2016-10-20 16:10:32 +02:00
|
|
|
* @return Number of bytes read from @p stream
|
2014-12-05 20:15:52 +01:00
|
|
|
*/
|
2014-12-05 19:54:59 +01:00
|
|
|
size_t cbor_deserialize_float_half(const cbor_stream_t *stream, size_t offset,
|
|
|
|
float *val);
|
2014-12-05 20:15:52 +01:00
|
|
|
/**
|
|
|
|
* @brief Serializes a floating point value
|
|
|
|
*
|
|
|
|
* @param[out] stream The destination stream for serializing the array
|
|
|
|
* @param[in] val The float to serialize
|
|
|
|
*
|
|
|
|
* @return Number of bytes written to stream @p stream
|
|
|
|
*/
|
|
|
|
size_t cbor_serialize_float(cbor_stream_t *stream, float val);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Deserialize floating point values values from @p stream to @p val
|
|
|
|
*
|
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream where to start deserializing
|
|
|
|
* @param[out] val Pointer to destination array
|
|
|
|
*
|
2016-10-20 16:10:32 +02:00
|
|
|
* @return Number of bytes read from @p stream
|
2014-12-05 20:15:52 +01:00
|
|
|
*/
|
2014-12-05 19:54:59 +01:00
|
|
|
size_t cbor_deserialize_float(const cbor_stream_t *stream, size_t offset,
|
|
|
|
float *val);
|
2014-12-05 20:15:52 +01:00
|
|
|
/**
|
|
|
|
* @brief Serializes a double precision floating point value
|
|
|
|
*
|
|
|
|
* @param[out] stream The destination stream for serializing the array
|
|
|
|
* @param[in] val The double to serialize
|
|
|
|
*
|
|
|
|
* @return Number of bytes written to stream @p stream
|
|
|
|
*/
|
|
|
|
size_t cbor_serialize_double(cbor_stream_t *stream, double val);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Deserialize double precision floating point valeus from @p stream to
|
|
|
|
* @p val
|
|
|
|
*
|
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream where to start deserializing
|
|
|
|
* @param[out] val Pointer to destination array
|
|
|
|
*
|
2016-10-20 16:10:32 +02:00
|
|
|
* @return Number of bytes read from @p stream
|
2014-12-05 20:15:52 +01:00
|
|
|
*/
|
2014-12-05 19:54:59 +01:00
|
|
|
size_t cbor_deserialize_double(const cbor_stream_t *stream, size_t offset,
|
|
|
|
double *val);
|
2014-07-14 21:40:04 +02:00
|
|
|
#endif /* CBOR_NO_FLOAT */
|
|
|
|
|
2014-12-05 20:15:52 +01:00
|
|
|
/**
|
|
|
|
* @brief Serializes a signed 64 bit value
|
|
|
|
*
|
|
|
|
* @param[out] stream The destination stream for serializing the array
|
|
|
|
* @param[in] val The 64 bit integer to serialize
|
|
|
|
*
|
|
|
|
* @return Number of bytes written to stream @p stream
|
|
|
|
*/
|
|
|
|
size_t cbor_serialize_byte_string(cbor_stream_t *stream, const char *val);
|
2014-12-05 19:47:12 +01:00
|
|
|
|
2015-08-06 22:19:52 +02:00
|
|
|
/**
|
|
|
|
* @brief Serializes an arbitrary byte string
|
|
|
|
*
|
|
|
|
* @param[out] stream The destination stream for serializing the byte stream
|
|
|
|
* @param[in] val The arbitrary byte string which may include null bytes
|
|
|
|
* @param[in] length The size of the byte string in bytes
|
|
|
|
*
|
|
|
|
* @return Number of bytes written to stream @p stream
|
|
|
|
*/
|
|
|
|
size_t cbor_serialize_byte_stringl(cbor_stream_t *stream, const char *val, size_t length);
|
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Deserialize bytes from @p stream to @p val
|
|
|
|
*
|
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream where to start deserializing
|
2014-12-05 20:15:52 +01:00
|
|
|
* @param[out] val Pointer to destination array
|
2014-12-05 19:47:12 +01:00
|
|
|
* @param[in] length Length of destination array
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
2016-10-20 16:10:32 +02:00
|
|
|
* @return Number of bytes read from @p stream
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
2014-12-05 19:54:59 +01:00
|
|
|
size_t cbor_deserialize_byte_string(const cbor_stream_t *stream, size_t offset,
|
|
|
|
char *val, size_t length);
|
2014-12-05 19:47:12 +01:00
|
|
|
|
|
|
|
size_t cbor_serialize_unicode_string(cbor_stream_t *stream, const char *val);
|
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Deserialize unicode string from @p stream to @p val
|
|
|
|
*
|
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream where to start deserializing
|
2014-12-05 20:15:52 +01:00
|
|
|
* @param[out] val Pointer to destination array
|
2014-12-05 19:47:12 +01:00
|
|
|
* @param[in] length Length of destination array
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
2016-10-20 16:10:32 +02:00
|
|
|
* @return Number of bytes read from @p stream
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
2014-12-05 19:54:59 +01:00
|
|
|
size_t cbor_deserialize_unicode_string(const cbor_stream_t *stream,
|
|
|
|
size_t offset, char *val, size_t length);
|
2014-07-14 21:40:04 +02:00
|
|
|
|
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Serialize array of length @p array_length
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* Basic usage:
|
|
|
|
* @code
|
|
|
|
* cbor_serialize_array(&stream, 2); // array of length 2 follows
|
|
|
|
* cbor_serialize_int(&stream, 1)); // write item 1
|
|
|
|
* cbor_serialize_int(&stream, 2)); // write item 2
|
|
|
|
* @endcode
|
|
|
|
*
|
|
|
|
* @note You have to make sure to serialize the correct amount of items.
|
|
|
|
* If you exceed the length @p array_length, items will just be appened as normal
|
|
|
|
*
|
2014-12-05 19:47:12 +01:00
|
|
|
* @param[out] stream The destination stream for serializing the array
|
|
|
|
* @param[in] array_length Length of the array of items which follows
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
2014-12-05 19:47:12 +01:00
|
|
|
* @return Number of bytes written to stream @p stream
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
2014-12-05 19:47:12 +01:00
|
|
|
size_t cbor_serialize_array(cbor_stream_t *stream, size_t array_length);
|
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Deserialize array of items
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* Basic usage:
|
|
|
|
* @code
|
|
|
|
* size_t array_length;
|
2014-12-05 19:47:12 +01:00
|
|
|
* // read out length of the array
|
|
|
|
* size_t offset = cbor_deserialize_array(&stream, 0, &array_length);
|
2014-07-14 21:40:04 +02:00
|
|
|
* int i1, i2;
|
|
|
|
* offset += cbor_deserialize_int(&stream, offset, &i1); // read item 1
|
|
|
|
* offset += cbor_deserialize_int(&stream, offset, &i2); // read item 2
|
|
|
|
* @endcode
|
|
|
|
*
|
2014-12-05 19:47:12 +01:00
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream
|
|
|
|
* @param[in] array_length Where the array length is stored
|
|
|
|
*
|
|
|
|
* @return Number of deserialized bytes from stream
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
2014-12-05 19:54:59 +01:00
|
|
|
size_t cbor_deserialize_array(const cbor_stream_t *stream, size_t offset,
|
|
|
|
size_t *array_length);
|
2014-07-14 21:40:04 +02:00
|
|
|
|
2014-12-05 20:15:52 +01:00
|
|
|
/**
|
|
|
|
* @brief Serialize array of infite length
|
|
|
|
*
|
|
|
|
* @param[out] stream The destination stream for serializing the array
|
|
|
|
*
|
|
|
|
* @return Number of bytes written to stream @p stream
|
|
|
|
*/
|
2014-12-05 19:47:12 +01:00
|
|
|
size_t cbor_serialize_array_indefinite(cbor_stream_t *stream);
|
|
|
|
|
2014-12-05 20:15:52 +01:00
|
|
|
/**
|
|
|
|
* @brief Deserialize array of items
|
|
|
|
*
|
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream
|
|
|
|
*
|
|
|
|
* @return Number of deserialized bytes from stream
|
|
|
|
*/
|
2014-12-05 19:47:12 +01:00
|
|
|
size_t cbor_deserialize_array_indefinite(const cbor_stream_t *stream, size_t offset);
|
2014-07-14 21:40:04 +02:00
|
|
|
|
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Serialize map of length @p map_length
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* Basic usage:
|
|
|
|
* @code
|
|
|
|
* cbor_serialize_map(&stream, 2); // map of length 2 follows
|
|
|
|
* cbor_serialize_int(&stream, 1)); // write key 1
|
|
|
|
* cbor_serialize_byte_string(&stream, "1")); // write value 1
|
|
|
|
* cbor_serialize_int(&stream, 2)); // write key 2
|
|
|
|
* cbor_serialize_byte_string(&stream, "2")); // write value 2
|
|
|
|
* @endcode
|
|
|
|
*
|
2014-12-05 19:47:12 +01:00
|
|
|
* @param[out] stream The destination stream for serializing the map
|
2014-07-14 21:40:04 +02:00
|
|
|
* @param map_length Length of the map of items which follows
|
2014-12-05 19:47:12 +01:00
|
|
|
*
|
|
|
|
* @return Number of bytes written to stream @p stream
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
2014-12-05 19:47:12 +01:00
|
|
|
size_t cbor_serialize_map(cbor_stream_t *stream, size_t map_length);
|
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Deserialize map of items
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* Basic usage:
|
|
|
|
* @code
|
|
|
|
* size_t map_length;
|
2014-12-05 19:54:59 +01:00
|
|
|
* // read out length of the map
|
|
|
|
* size_t offset = cbor_deserialize_map(&stream, 0, &map_length);
|
2014-07-14 21:40:04 +02:00
|
|
|
* int key1, key1;
|
|
|
|
* char value1[8], value2[8];
|
2014-12-05 19:54:59 +01:00
|
|
|
* // read key 1
|
|
|
|
* offset += cbor_deserialize_int(&stream, offset, &key1);
|
|
|
|
* // read value 1
|
|
|
|
* offset += cbor_deserialize_byte_string(&stream, offset, value1, sizeof(value));
|
|
|
|
* // read key 2
|
|
|
|
* offset += cbor_deserialize_int(&stream, offset, &key2);
|
|
|
|
* // read value 2
|
|
|
|
* offset += cbor_deserialize_byte_string(&stream, offset, value2, sizeof(value));
|
2014-07-14 21:40:04 +02:00
|
|
|
* @endcode
|
|
|
|
*
|
2014-12-05 19:47:12 +01:00
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream where to start deserializing
|
|
|
|
* @param[in] map_length Where the array length is stored
|
|
|
|
*
|
|
|
|
* @return Number of deserialized bytes from @p stream
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
2014-12-05 19:54:59 +01:00
|
|
|
size_t cbor_deserialize_map(const cbor_stream_t *stream, size_t offset,
|
|
|
|
size_t *map_length);
|
2014-07-14 21:40:04 +02:00
|
|
|
|
2014-12-05 20:15:52 +01:00
|
|
|
/**
|
|
|
|
* @brief Serialize map of infite length
|
|
|
|
*
|
|
|
|
* @param[out] stream The destination stream for serializing the map
|
|
|
|
*
|
|
|
|
* @return Number of bytes written to stream @p stream
|
|
|
|
*/
|
2014-12-05 19:47:12 +01:00
|
|
|
size_t cbor_serialize_map_indefinite(cbor_stream_t *stream);
|
|
|
|
|
2014-12-05 20:15:52 +01:00
|
|
|
/**
|
|
|
|
* @brief Deserialize map of items
|
|
|
|
*
|
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream
|
|
|
|
*
|
|
|
|
* @return Number of deserialized bytes from stream
|
|
|
|
*/
|
2014-12-05 19:47:12 +01:00
|
|
|
size_t cbor_deserialize_map_indefinite(const cbor_stream_t *stream, size_t offset);
|
2014-07-14 21:40:04 +02:00
|
|
|
|
|
|
|
#ifndef CBOR_NO_SEMANTIC_TAGGING
|
|
|
|
#ifndef CBOR_NO_CTIME
|
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Serialize date and time
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* Basic usage:
|
|
|
|
* @code
|
|
|
|
* struct tm val;
|
|
|
|
* val.tm_year = 114;
|
|
|
|
* val.tm_mon = 6;
|
|
|
|
* val.tm_mday = 1;
|
|
|
|
* val.tm_hour = 15;
|
|
|
|
* val.tm_min = 0;
|
|
|
|
* val.tm_sec = 0;
|
|
|
|
* mktime(&val);
|
|
|
|
* cbor_serialize_date_time(&stream, &val);
|
|
|
|
* @endcode
|
|
|
|
*
|
2014-12-05 19:47:12 +01:00
|
|
|
* @param[out] stream The destination stream for serializing the date_time
|
|
|
|
* @param[in] val tm struct containing the date/time info to be encoded
|
|
|
|
*
|
|
|
|
* @return Number of bytes written to stream @p stream
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
|
|
|
size_t cbor_serialize_date_time(cbor_stream_t *stream, struct tm *val);
|
2014-12-05 19:47:12 +01:00
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Deserialize date and time
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* Basic usage:
|
|
|
|
* @code
|
|
|
|
* struct tm val;
|
|
|
|
* cbor_deserialize_date_time(&stream, 0, &val);
|
|
|
|
* @endcode
|
|
|
|
*
|
2014-12-05 19:47:12 +01:00
|
|
|
* @param[in] stream The stream to deserialize
|
|
|
|
* @param[in] offset The offset within the stream where to start deserializing
|
2014-12-05 20:15:52 +01:00
|
|
|
* @param[out] val tm struct where the decoded date/time will be stored
|
2014-12-05 19:47:12 +01:00
|
|
|
*
|
|
|
|
* @return The number of deserialized bytes
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
|
|
|
size_t cbor_deserialize_date_time(const cbor_stream_t *stream, size_t offset, struct tm *val);
|
|
|
|
|
|
|
|
size_t cbor_serialize_date_time_epoch(cbor_stream_t *stream, time_t val);
|
2014-12-05 19:47:12 +01:00
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
size_t cbor_deserialize_date_time_epoch(const cbor_stream_t *stream, size_t offset, time_t *val);
|
2014-12-05 19:47:12 +01:00
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
#endif /* CBOR_NO_CTIME */
|
|
|
|
|
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Write a tag to give the next CBOR item additional semantics
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* Also see https://tools.ietf.org/html/rfc7049#section-2.4 (Optional Tagging of Items)
|
2014-12-05 19:47:12 +01:00
|
|
|
*
|
|
|
|
* @param[in, out] stream Pointer to the cbor struct
|
|
|
|
* @param[in] tag The tag to write
|
|
|
|
*
|
|
|
|
* @return Always 1
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
2014-12-05 19:47:12 +01:00
|
|
|
size_t cbor_write_tag(cbor_stream_t *stream, unsigned char tag);
|
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Whether we are at a tag symbol in stream @p stream at offset @p offset
|
|
|
|
*
|
|
|
|
* @param[in] stream Pointer to the cbor struct
|
|
|
|
* @param[in] offset The offset within @p stream
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* @return True in case there is a tag symbol at the current offset
|
|
|
|
*/
|
2014-12-05 19:47:12 +01:00
|
|
|
bool cbor_at_tag(const cbor_stream_t *stream, size_t offset);
|
|
|
|
|
|
|
|
#endif /* CBOR_NO_SEMANTIC_TAGGING */
|
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Write a break symbol at the current offset in stream @p stream
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* Used for marking the end of indefinite length CBOR items
|
2014-12-05 19:47:12 +01:00
|
|
|
*
|
|
|
|
* @param[in] stream Pointer to the cbor struct
|
|
|
|
*
|
|
|
|
* @return Always 1
|
2014-07-14 21:40:04 +02:00
|
|
|
*/
|
2014-12-05 19:47:12 +01:00
|
|
|
size_t cbor_write_break(cbor_stream_t *stream);
|
2014-07-14 21:40:04 +02:00
|
|
|
|
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Whether we are at a break symbol in stream @p stream at offset @p offset
|
|
|
|
*
|
|
|
|
* @param[in] stream Pointer to the cbor struct
|
|
|
|
* @param[in] offset The offset within @p stream
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* @return True in case the there is a break symbol at the current offset
|
|
|
|
*/
|
2014-12-05 19:47:12 +01:00
|
|
|
bool cbor_at_break(const cbor_stream_t *stream, size_t offset);
|
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
/**
|
2014-12-05 19:47:12 +01:00
|
|
|
* @brief Whether we are at the end of the stream @p stream at offset @p offset
|
2014-07-14 21:40:04 +02:00
|
|
|
*
|
|
|
|
* Useful for abort conditions in loops while deserializing CBOR items
|
|
|
|
*
|
2014-12-05 19:47:12 +01:00
|
|
|
* @param[in] stream Pointer to the cbor struct
|
|
|
|
* @param[in] offset The offset within @p stream
|
|
|
|
*
|
2014-07-14 21:40:04 +02:00
|
|
|
* @return True in case @p offset marks the end of the stream
|
|
|
|
*/
|
2014-12-05 19:47:12 +01:00
|
|
|
bool cbor_at_end(const cbor_stream_t *stream, size_t offset);
|
2014-07-14 21:40:04 +02:00
|
|
|
|
2014-10-10 11:51:11 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2014-07-14 21:40:04 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/** @} */
|