moved logic to shared-module and added documentation
This commit is contained in:
parent
748472de7a
commit
513253bc3f
@ -52,8 +52,6 @@ MCU_SUB_VARIANT = nrf52840
|
|||||||
# Fits on nrf52840 but space is tight on nrf52833.
|
# Fits on nrf52840 but space is tight on nrf52833.
|
||||||
CIRCUITPY_AESIO ?= 1
|
CIRCUITPY_AESIO ?= 1
|
||||||
|
|
||||||
CIRCUITPY_MSGPACK ?= 1
|
|
||||||
|
|
||||||
SD ?= s140
|
SD ?= s140
|
||||||
SOFTDEV_VERSION ?= 6.1.0
|
SOFTDEV_VERSION ?= 6.1.0
|
||||||
|
|
||||||
|
@ -465,6 +465,7 @@ SRC_SHARED_MODULE_ALL = \
|
|||||||
memorymonitor/AllocationAlarm.c \
|
memorymonitor/AllocationAlarm.c \
|
||||||
memorymonitor/AllocationSize.c \
|
memorymonitor/AllocationSize.c \
|
||||||
network/__init__.c \
|
network/__init__.c \
|
||||||
|
msgpack/__init__.c \
|
||||||
os/__init__.c \
|
os/__init__.c \
|
||||||
random/__init__.c \
|
random/__init__.c \
|
||||||
rgbmatrix/RGBMatrix.c \
|
rgbmatrix/RGBMatrix.c \
|
||||||
|
@ -815,6 +815,7 @@ extern const struct _mp_obj_module_t msgpack_module;
|
|||||||
_EVE_MODULE \
|
_EVE_MODULE \
|
||||||
MEMORYMONITOR_MODULE \
|
MEMORYMONITOR_MODULE \
|
||||||
MICROCONTROLLER_MODULE \
|
MICROCONTROLLER_MODULE \
|
||||||
|
MSGPACK_MODULE \
|
||||||
NEOPIXEL_WRITE_MODULE \
|
NEOPIXEL_WRITE_MODULE \
|
||||||
NETWORK_MODULE \
|
NETWORK_MODULE \
|
||||||
SOCKET_MODULE \
|
SOCKET_MODULE \
|
||||||
@ -846,7 +847,6 @@ extern const struct _mp_obj_module_t msgpack_module;
|
|||||||
USTACK_MODULE \
|
USTACK_MODULE \
|
||||||
WATCHDOG_MODULE \
|
WATCHDOG_MODULE \
|
||||||
WIFI_MODULE \
|
WIFI_MODULE \
|
||||||
MSGPACK_MODULE \
|
|
||||||
|
|
||||||
// If weak links are enabled, just include strong links in the main list of modules,
|
// If weak links are enabled, just include strong links in the main list of modules,
|
||||||
// and also include the underscore alternate names.
|
// and also include the underscore alternate names.
|
||||||
|
@ -297,5 +297,5 @@ CFLAGS += -DCIRCUITPY_WIFI=$(CIRCUITPY_WIFI)
|
|||||||
CIRCUITPY_ENABLE_MPY_NATIVE ?= 0
|
CIRCUITPY_ENABLE_MPY_NATIVE ?= 0
|
||||||
CFLAGS += -DCIRCUITPY_ENABLE_MPY_NATIVE=$(CIRCUITPY_ENABLE_MPY_NATIVE)
|
CFLAGS += -DCIRCUITPY_ENABLE_MPY_NATIVE=$(CIRCUITPY_ENABLE_MPY_NATIVE)
|
||||||
|
|
||||||
CIRCUITPY_MSGPACK ?= 0
|
CIRCUITPY_MSGPACK ?= $(CIRCUITPY_FULL_BUILD)
|
||||||
CFLAGS += -DCIRCUITPY_MSGPACK=$(CIRCUITPY_MSGPACK)
|
CFLAGS += -DCIRCUITPY_MSGPACK=$(CIRCUITPY_MSGPACK)
|
||||||
|
@ -24,359 +24,48 @@
|
|||||||
* THE SOFTWARE.
|
* THE SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <stdio.h>
|
#include "shared-bindings/msgpack/__init__.h"
|
||||||
|
#include "shared-module/msgpack/__init__.h"
|
||||||
|
|
||||||
#include "py/binary.h"
|
//| """Pack object in msgpack format
|
||||||
#include "py/objarray.h"
|
//|
|
||||||
#include "py/objlist.h"
|
//| The msgpack format is similar to json, except that the encoded data is binary.
|
||||||
#include "py/objstringio.h"
|
//| See https://msgpack.org for details.
|
||||||
#include "py/parsenum.h"
|
//|
|
||||||
#include "py/runtime.h"
|
//| Example:
|
||||||
#include "py/stream.h"
|
//| import msgpack
|
||||||
|
//| from io import StringIO
|
||||||
|
//|
|
||||||
|
//| s = StringIO()
|
||||||
|
//| msgpack.pack({'list': [True, False, None, 1, 'abc'], 'str': 'blah'}, s)
|
||||||
|
//| s.seek(0)
|
||||||
|
//| print(msgpack.unpack(s))"""
|
||||||
|
//|
|
||||||
|
|
||||||
#include "supervisor/shared/translate.h"
|
//| def pack(obj, stream):
|
||||||
|
//| """Pack obj to stream."""
|
||||||
|
//| ...
|
||||||
|
//|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////
|
|
||||||
// stream management
|
|
||||||
|
|
||||||
typedef struct _msgpack_stream_t {
|
|
||||||
mp_obj_t stream_obj;
|
|
||||||
mp_uint_t (*read)(mp_obj_t obj, void *buf, mp_uint_t size, int *errcode);
|
|
||||||
mp_uint_t (*write)(mp_obj_t obj, const void *buf, mp_uint_t size, int *errcode);
|
|
||||||
int errcode;
|
|
||||||
} msgpack_stream_t;
|
|
||||||
|
|
||||||
STATIC msgpack_stream_t get_stream(mp_obj_t stream_obj, int flags) {
|
|
||||||
const mp_stream_p_t *stream_p = mp_get_stream_raise(stream_obj, flags);
|
|
||||||
msgpack_stream_t s = {stream_obj, stream_p->read, stream_p->write, 0};
|
|
||||||
return s;
|
|
||||||
}
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////
|
|
||||||
// readers
|
|
||||||
|
|
||||||
STATIC void read(msgpack_stream_t *s, void *buf, mp_uint_t size) {
|
|
||||||
if (size == 0) return;
|
|
||||||
mp_uint_t ret = s->read(s->stream_obj, buf, size, &s->errcode);
|
|
||||||
if (s->errcode != 0) {
|
|
||||||
mp_raise_OSError(s->errcode);
|
|
||||||
}
|
|
||||||
if (ret == 0) {
|
|
||||||
mp_raise_msg(&mp_type_EOFError, NULL);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC uint32_t read_bytes(msgpack_stream_t *s, mp_uint_t n_bytes) {
|
|
||||||
uint32_t res = 0;
|
|
||||||
read(s, &res, n_bytes);
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t read_size(msgpack_stream_t *s, uint8_t len_index) {
|
|
||||||
size_t n_bytes = 4;
|
|
||||||
switch (len_index) {
|
|
||||||
case 0: n_bytes = 1; break;
|
|
||||||
case 1: n_bytes = 2; break;
|
|
||||||
case 2: n_bytes = 4; break;
|
|
||||||
}
|
|
||||||
size_t res = 0;
|
|
||||||
read(s, &res, n_bytes);
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////
|
|
||||||
// writers
|
|
||||||
|
|
||||||
STATIC void write(msgpack_stream_t *s, const void *buf, mp_uint_t size) {
|
|
||||||
mp_uint_t ret = s->write(s->stream_obj, buf, size, &s->errcode);
|
|
||||||
if (s->errcode != 0) {
|
|
||||||
mp_raise_OSError(s->errcode);
|
|
||||||
}
|
|
||||||
if (ret == 0) {
|
|
||||||
mp_raise_msg(&mp_type_EOFError, NULL);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
STATIC void write_bytes(msgpack_stream_t *s, mp_uint_t n_bytes, uint32_t obj) {
|
|
||||||
write(s, &obj, n_bytes);
|
|
||||||
}
|
|
||||||
|
|
||||||
void write_size(msgpack_stream_t *s, uint8_t code, size_t size) {
|
|
||||||
if ((uint8_t)size == size) {
|
|
||||||
write_bytes(s, 1, code);
|
|
||||||
write_bytes(s, 1, size);
|
|
||||||
} else if ((uint16_t)size == size) {
|
|
||||||
write_bytes(s, 1, code+1);
|
|
||||||
write_bytes(s, 2, size);
|
|
||||||
} else {
|
|
||||||
write_bytes(s, 1, code+2);
|
|
||||||
write_bytes(s, 4, size);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////
|
|
||||||
// packers
|
|
||||||
|
|
||||||
// This is a helper function to iterate through a dictionary. The state of
|
|
||||||
// the iteration is held in *cur and should be initialised with zero for the
|
|
||||||
// first call. Will return NULL when no more elements are available.
|
|
||||||
STATIC mp_map_elem_t *dict_iter_next(mp_obj_dict_t *dict, size_t *cur) {
|
|
||||||
size_t max = dict->map.alloc;
|
|
||||||
mp_map_t *map = &dict->map;
|
|
||||||
|
|
||||||
for (size_t i = *cur; i < max; i++) {
|
|
||||||
if (MP_MAP_SLOT_IS_FILLED(map, i)) {
|
|
||||||
*cur = i + 1;
|
|
||||||
return &(map->table[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
STATIC void pack_int(msgpack_stream_t *s, int32_t x) {
|
|
||||||
if (x > -32 && x < 128) {
|
|
||||||
write_bytes(s, 1, x);
|
|
||||||
} else if ((int8_t)x == x) {
|
|
||||||
write_bytes(s, 1, 0xd0);
|
|
||||||
write_bytes(s, 1, x);
|
|
||||||
} else if ((int16_t)x == x) {
|
|
||||||
write_bytes(s, 1, 0xd1);
|
|
||||||
write_bytes(s, 2, x);
|
|
||||||
} else {
|
|
||||||
write_bytes(s, 1, 0xd2);
|
|
||||||
write_bytes(s, 4, x);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void pack_bin(msgpack_stream_t *s, const uint8_t* data, size_t len) {
|
|
||||||
write_size(s, 0xc4, len);
|
|
||||||
for (size_t i=0; i<len; i++) {
|
|
||||||
write_bytes(s, 1, data[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void pack_str(msgpack_stream_t *s, const char* str, size_t len) {
|
|
||||||
// size_t len = strlen(str);
|
|
||||||
if (len < 32) {
|
|
||||||
write_bytes(s, 1, 0b10100000 | (uint8_t)len);
|
|
||||||
} else {
|
|
||||||
write_size(s, 0xd9, len);
|
|
||||||
}
|
|
||||||
for (size_t l=0; l<len; l++) {
|
|
||||||
write_bytes(s, 1, str[l]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void pack_array(msgpack_stream_t *s, size_t len) {
|
|
||||||
// only writes the header, manually write the objects after calling pack_array!
|
|
||||||
if (len < 16) {
|
|
||||||
write_bytes(s, 1, 0b10010000 | (uint8_t)len);
|
|
||||||
} else if (len < 0x10000) {
|
|
||||||
write_bytes(s, 1, 0xdc);
|
|
||||||
write_bytes(s, 2, len);
|
|
||||||
} else {
|
|
||||||
write_bytes(s, 1, 0xdd);
|
|
||||||
write_bytes(s, 4, len);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void pack_dict(msgpack_stream_t *s, size_t len) {
|
|
||||||
// only writes the header, manually write the objects after calling pack_array!
|
|
||||||
if (len < 16) {
|
|
||||||
write_bytes(s, 1, 0b10000000 | (uint8_t)len);
|
|
||||||
} else if (len < 0x10000) {
|
|
||||||
write_bytes(s, 1, 0xde);
|
|
||||||
write_bytes(s, 2, len);
|
|
||||||
} else {
|
|
||||||
write_bytes(s, 1, 0xdf);
|
|
||||||
write_bytes(s, 4, len);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write obj to stream in msgpack format
|
|
||||||
STATIC void _pack(mp_obj_t obj, msgpack_stream_t *s) {
|
|
||||||
if (MP_OBJ_IS_SMALL_INT(obj)) {
|
|
||||||
// int
|
|
||||||
int32_t x = MP_OBJ_SMALL_INT_VALUE(obj);
|
|
||||||
pack_int(s, x);
|
|
||||||
} else if (MP_OBJ_IS_STR(obj)) {
|
|
||||||
// string
|
|
||||||
size_t len;
|
|
||||||
const char *data = mp_obj_str_get_data(obj, &len);
|
|
||||||
pack_str(s, data, len);
|
|
||||||
} else if (MP_OBJ_IS_TYPE(obj, &mp_type_bytes)) {
|
|
||||||
// bytes
|
|
||||||
mp_buffer_info_t bufinfo;
|
|
||||||
mp_get_buffer_raise(obj, &bufinfo, MP_BUFFER_READ);
|
|
||||||
pack_bin(s, bufinfo.buf, bufinfo.len);
|
|
||||||
} else if (MP_OBJ_IS_TYPE(obj, &mp_type_tuple)) {
|
|
||||||
// tuple
|
|
||||||
mp_obj_tuple_t *self = MP_OBJ_TO_PTR(obj);
|
|
||||||
pack_array(s, self->len);
|
|
||||||
for (size_t i=0; i<self->len; i++) {
|
|
||||||
_pack(self->items[i], s);
|
|
||||||
}
|
|
||||||
} else if (MP_OBJ_IS_TYPE(obj, &mp_type_list)) {
|
|
||||||
// list (layout differs from tuple)
|
|
||||||
mp_obj_list_t *self = MP_OBJ_TO_PTR(obj);
|
|
||||||
pack_array(s, self->len);
|
|
||||||
for (size_t i=0; i<self->len; i++) {
|
|
||||||
_pack(self->items[i], s);
|
|
||||||
}
|
|
||||||
} else if (MP_OBJ_IS_TYPE(obj, &mp_type_dict)) {
|
|
||||||
// dict
|
|
||||||
mp_obj_dict_t *self = MP_OBJ_TO_PTR(obj);
|
|
||||||
pack_dict(s, self->map.used);
|
|
||||||
size_t cur = 0;
|
|
||||||
mp_map_elem_t *next = NULL;
|
|
||||||
while ((next = dict_iter_next(self, &cur)) != NULL) {
|
|
||||||
_pack(next->key, s);
|
|
||||||
_pack(next->value, s);
|
|
||||||
}
|
|
||||||
} else if (mp_obj_is_float(obj)) {
|
|
||||||
mp_float_t f = mp_obj_float_get(obj);
|
|
||||||
write_bytes(s, 1, 0xca);
|
|
||||||
write(s, &f, 4);
|
|
||||||
} else if (obj == mp_const_none) {
|
|
||||||
write_bytes(s, 1, 0xc0);
|
|
||||||
} else if (obj == mp_const_false) {
|
|
||||||
write_bytes(s, 1, 0xc2);
|
|
||||||
} else if (obj == mp_const_true) {
|
|
||||||
write_bytes(s, 1, 0xc3);
|
|
||||||
} else {
|
|
||||||
mp_raise_ValueError(translate("no packer"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write obj to stream in msgpack format
|
|
||||||
STATIC mp_obj_t mod_msgpack_pack(mp_obj_t obj, mp_obj_t stream_obj) {
|
STATIC mp_obj_t mod_msgpack_pack(mp_obj_t obj, mp_obj_t stream_obj) {
|
||||||
msgpack_stream_t s = get_stream(stream_obj, MP_STREAM_OP_WRITE);
|
common_hal_msgpack_pack(obj, stream_obj);
|
||||||
_pack(obj, &s);
|
|
||||||
return mp_const_none;
|
return mp_const_none;
|
||||||
}
|
}
|
||||||
STATIC MP_DEFINE_CONST_FUN_OBJ_2(mod_msgpack_pack_obj, mod_msgpack_pack);
|
STATIC MP_DEFINE_CONST_FUN_OBJ_2(mod_msgpack_pack_obj, mod_msgpack_pack);
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////
|
|
||||||
// un-packers
|
|
||||||
|
|
||||||
STATIC mp_obj_t _unpack(msgpack_stream_t *s) {
|
//| def pack(obj, stream) -> obj:
|
||||||
uint8_t code = read_bytes(s, 1);
|
//| """Unpack and return one object (in msgpack format) from stream.
|
||||||
if (((code & 0b10000000) == 0) || ((code & 0b11100000) == 0b11100000)) {
|
//| Call repeatedly to read multiple objects from the stream."""
|
||||||
// int
|
//| ...
|
||||||
return MP_OBJ_NEW_SMALL_INT((int8_t)code);
|
//|
|
||||||
}
|
|
||||||
if ((code & 0b11100000) == 0b10100000) {
|
|
||||||
// str
|
|
||||||
size_t len = code & 0b11111;
|
|
||||||
// allocate on stack; len < 32
|
|
||||||
char str[len];
|
|
||||||
read(s, &str, len);
|
|
||||||
return mp_obj_new_str(str, len);
|
|
||||||
}
|
|
||||||
if ((code & 0b11110000) == 0b10010000) {
|
|
||||||
// array (tuple)
|
|
||||||
size_t len = code & 0b1111;
|
|
||||||
mp_obj_tuple_t *t = MP_OBJ_TO_PTR(mp_obj_new_tuple(len, NULL));
|
|
||||||
for (size_t i=0; i<len; i++) {
|
|
||||||
t->items[i] = _unpack(s);
|
|
||||||
}
|
|
||||||
return MP_OBJ_FROM_PTR(t);
|
|
||||||
}
|
|
||||||
if ((code & 0b11110000) == 0b10000000) {
|
|
||||||
// map (dict)
|
|
||||||
size_t len = code & 0b1111;
|
|
||||||
mp_obj_dict_t *d = MP_OBJ_TO_PTR(mp_obj_new_dict(len));
|
|
||||||
for (size_t i=0; i<len; i++) {
|
|
||||||
mp_obj_dict_store(d, _unpack(s), _unpack(s));
|
|
||||||
}
|
|
||||||
return MP_OBJ_FROM_PTR(d);
|
|
||||||
}
|
|
||||||
switch (code) {
|
|
||||||
case 0xc0: return mp_const_none;
|
|
||||||
case 0xc2: return mp_const_false;
|
|
||||||
case 0xc3: return mp_const_true;
|
|
||||||
case 0xc4:
|
|
||||||
case 0xc5:
|
|
||||||
case 0xc6: {
|
|
||||||
// bin 8, 16, 32
|
|
||||||
size_t size = read_size(s, code-0xc4);
|
|
||||||
vstr_t vstr;
|
|
||||||
vstr_init_len(&vstr, size);
|
|
||||||
byte *p = (byte*)vstr.buf;
|
|
||||||
read(s, p, size);
|
|
||||||
return mp_obj_new_str_from_vstr(&mp_type_bytes, &vstr);
|
|
||||||
}
|
|
||||||
case 0xca: {
|
|
||||||
float f;
|
|
||||||
read(s, &f, 4);
|
|
||||||
return mp_obj_new_float(f);
|
|
||||||
}
|
|
||||||
case 0xd0:
|
|
||||||
return MP_OBJ_NEW_SMALL_INT((int8_t)read_bytes(s, 1));
|
|
||||||
case 0xd1:
|
|
||||||
return MP_OBJ_NEW_SMALL_INT((int16_t)read_bytes(s, 2));
|
|
||||||
case 0xd2:
|
|
||||||
return MP_OBJ_NEW_SMALL_INT((int32_t)read_bytes(s, 4));
|
|
||||||
case 0xd9:
|
|
||||||
case 0xda:
|
|
||||||
case 0xdb: {
|
|
||||||
// str 8, 16, 32
|
|
||||||
size_t size = read_size(s, code-0xd9);
|
|
||||||
vstr_t vstr;
|
|
||||||
vstr_init_len(&vstr, size);
|
|
||||||
byte *p = (byte*)vstr.buf;
|
|
||||||
read(s, p, size);
|
|
||||||
return mp_obj_new_str_from_vstr(&mp_type_str, &vstr);
|
|
||||||
}
|
|
||||||
case 0xde:
|
|
||||||
case 0xdf: {
|
|
||||||
// map 16 & 32
|
|
||||||
size_t len = read_size(s, code - 0xde + 1);
|
|
||||||
mp_obj_dict_t *d = MP_OBJ_TO_PTR(mp_obj_new_dict(len));
|
|
||||||
for (size_t i=0; i<len; i++) {
|
|
||||||
mp_obj_dict_store(d, _unpack(s), _unpack(s));
|
|
||||||
}
|
|
||||||
return MP_OBJ_FROM_PTR(d);
|
|
||||||
}
|
|
||||||
case 0xdc:
|
|
||||||
case 0xdd: {
|
|
||||||
// array 16 & 32
|
|
||||||
size_t len = read_size(s, code - 0xdc + 1);
|
|
||||||
mp_obj_tuple_t *t = MP_OBJ_TO_PTR(mp_obj_new_tuple(len, NULL));
|
|
||||||
for (size_t i=0; i<len; i++) {
|
|
||||||
t->items[i] = _unpack(s);
|
|
||||||
}
|
|
||||||
return MP_OBJ_FROM_PTR(t);
|
|
||||||
}
|
|
||||||
case 0xc1: // never used
|
|
||||||
case 0xc7: // ext 8
|
|
||||||
case 0xc8: // ext 16
|
|
||||||
case 0xc9: // ext 32
|
|
||||||
case 0xcb: // float 64
|
|
||||||
case 0xcc: // uint 8
|
|
||||||
case 0xcd: // uint 16
|
|
||||||
case 0xce: // uint 32
|
|
||||||
case 0xcf: // uint 64
|
|
||||||
case 0xd3: // int 64
|
|
||||||
case 0xd4: // fixenxt 1
|
|
||||||
case 0xd5: // fixenxt 2
|
|
||||||
case 0xd6: // fixenxt 4
|
|
||||||
case 0xd7: // fixenxt 8
|
|
||||||
case 0xd8: // fixenxt 16
|
|
||||||
default:
|
|
||||||
mp_raise_ValueError(translate("no unpacker found"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Read msgpack encoded object from stream
|
|
||||||
STATIC mp_obj_t mod_msgpack_unpack(mp_obj_t stream_obj) {
|
STATIC mp_obj_t mod_msgpack_unpack(mp_obj_t stream_obj) {
|
||||||
msgpack_stream_t s = get_stream(stream_obj, MP_STREAM_OP_READ);
|
return common_hal_msgpack_unpack(stream_obj);
|
||||||
return _unpack(&s);
|
|
||||||
}
|
}
|
||||||
STATIC MP_DEFINE_CONST_FUN_OBJ_1(mod_msgpack_unpack_obj, mod_msgpack_unpack);
|
STATIC MP_DEFINE_CONST_FUN_OBJ_1(mod_msgpack_unpack_obj, mod_msgpack_unpack);
|
||||||
|
|
||||||
|
|
||||||
STATIC const mp_rom_map_elem_t msgpack_module_globals_table[] = {
|
STATIC const mp_rom_map_elem_t msgpack_module_globals_table[] = {
|
||||||
{ MP_ROM_QSTR(MP_QSTR___name__), MP_ROM_QSTR(MP_QSTR_msgpack) },
|
{ MP_ROM_QSTR(MP_QSTR___name__), MP_ROM_QSTR(MP_QSTR_msgpack) },
|
||||||
{ MP_ROM_QSTR(MP_QSTR_pack), MP_ROM_PTR(&mod_msgpack_pack_obj) },
|
{ MP_ROM_QSTR(MP_QSTR_pack), MP_ROM_PTR(&mod_msgpack_pack_obj) },
|
||||||
|
372
shared-module/msgpack/__init__.c
Normal file
372
shared-module/msgpack/__init__.c
Normal file
@ -0,0 +1,372 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of the MicroPython project, http://micropython.org/
|
||||||
|
*
|
||||||
|
* The MIT License (MIT)
|
||||||
|
*
|
||||||
|
* Copyright (c) 2019 Bernhard Boser
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in
|
||||||
|
* all copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
* THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
#include "py/binary.h"
|
||||||
|
#include "py/objarray.h"
|
||||||
|
#include "py/objlist.h"
|
||||||
|
#include "py/objstringio.h"
|
||||||
|
#include "py/parsenum.h"
|
||||||
|
#include "py/runtime.h"
|
||||||
|
#include "py/stream.h"
|
||||||
|
|
||||||
|
#include "supervisor/shared/translate.h"
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////
|
||||||
|
// stream management
|
||||||
|
|
||||||
|
typedef struct _msgpack_stream_t {
|
||||||
|
mp_obj_t stream_obj;
|
||||||
|
mp_uint_t (*read)(mp_obj_t obj, void *buf, mp_uint_t size, int *errcode);
|
||||||
|
mp_uint_t (*write)(mp_obj_t obj, const void *buf, mp_uint_t size, int *errcode);
|
||||||
|
int errcode;
|
||||||
|
} msgpack_stream_t;
|
||||||
|
|
||||||
|
STATIC msgpack_stream_t get_stream(mp_obj_t stream_obj, int flags) {
|
||||||
|
const mp_stream_p_t *stream_p = mp_get_stream_raise(stream_obj, flags);
|
||||||
|
msgpack_stream_t s = {stream_obj, stream_p->read, stream_p->write, 0};
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////
|
||||||
|
// readers
|
||||||
|
|
||||||
|
STATIC void read(msgpack_stream_t *s, void *buf, mp_uint_t size) {
|
||||||
|
if (size == 0) return;
|
||||||
|
mp_uint_t ret = s->read(s->stream_obj, buf, size, &s->errcode);
|
||||||
|
if (s->errcode != 0) {
|
||||||
|
mp_raise_OSError(s->errcode);
|
||||||
|
}
|
||||||
|
if (ret == 0) {
|
||||||
|
mp_raise_msg(&mp_type_EOFError, NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
STATIC uint32_t read_bytes(msgpack_stream_t *s, mp_uint_t n_bytes) {
|
||||||
|
uint32_t res = 0;
|
||||||
|
read(s, &res, n_bytes);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t read_size(msgpack_stream_t *s, uint8_t len_index) {
|
||||||
|
size_t n_bytes = 4;
|
||||||
|
switch (len_index) {
|
||||||
|
case 0: n_bytes = 1; break;
|
||||||
|
case 1: n_bytes = 2; break;
|
||||||
|
case 2: n_bytes = 4; break;
|
||||||
|
}
|
||||||
|
size_t res = 0;
|
||||||
|
read(s, &res, n_bytes);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////
|
||||||
|
// writers
|
||||||
|
|
||||||
|
STATIC void write(msgpack_stream_t *s, const void *buf, mp_uint_t size) {
|
||||||
|
mp_uint_t ret = s->write(s->stream_obj, buf, size, &s->errcode);
|
||||||
|
if (s->errcode != 0) {
|
||||||
|
mp_raise_OSError(s->errcode);
|
||||||
|
}
|
||||||
|
if (ret == 0) {
|
||||||
|
mp_raise_msg(&mp_type_EOFError, NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
STATIC void write_bytes(msgpack_stream_t *s, mp_uint_t n_bytes, uint32_t obj) {
|
||||||
|
write(s, &obj, n_bytes);
|
||||||
|
}
|
||||||
|
|
||||||
|
void write_size(msgpack_stream_t *s, uint8_t code, size_t size) {
|
||||||
|
if ((uint8_t)size == size) {
|
||||||
|
write_bytes(s, 1, code);
|
||||||
|
write_bytes(s, 1, size);
|
||||||
|
} else if ((uint16_t)size == size) {
|
||||||
|
write_bytes(s, 1, code+1);
|
||||||
|
write_bytes(s, 2, size);
|
||||||
|
} else {
|
||||||
|
write_bytes(s, 1, code+2);
|
||||||
|
write_bytes(s, 4, size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////
|
||||||
|
// packers
|
||||||
|
|
||||||
|
// This is a helper function to iterate through a dictionary. The state of
|
||||||
|
// the iteration is held in *cur and should be initialised with zero for the
|
||||||
|
// first call. Will return NULL when no more elements are available.
|
||||||
|
STATIC mp_map_elem_t *dict_iter_next(mp_obj_dict_t *dict, size_t *cur) {
|
||||||
|
size_t max = dict->map.alloc;
|
||||||
|
mp_map_t *map = &dict->map;
|
||||||
|
|
||||||
|
for (size_t i = *cur; i < max; i++) {
|
||||||
|
if (MP_MAP_SLOT_IS_FILLED(map, i)) {
|
||||||
|
*cur = i + 1;
|
||||||
|
return &(map->table[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
STATIC void pack_int(msgpack_stream_t *s, int32_t x) {
|
||||||
|
if (x > -32 && x < 128) {
|
||||||
|
write_bytes(s, 1, x);
|
||||||
|
} else if ((int8_t)x == x) {
|
||||||
|
write_bytes(s, 1, 0xd0);
|
||||||
|
write_bytes(s, 1, x);
|
||||||
|
} else if ((int16_t)x == x) {
|
||||||
|
write_bytes(s, 1, 0xd1);
|
||||||
|
write_bytes(s, 2, x);
|
||||||
|
} else {
|
||||||
|
write_bytes(s, 1, 0xd2);
|
||||||
|
write_bytes(s, 4, x);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void pack_bin(msgpack_stream_t *s, const uint8_t* data, size_t len) {
|
||||||
|
write_size(s, 0xc4, len);
|
||||||
|
for (size_t i=0; i<len; i++) {
|
||||||
|
write_bytes(s, 1, data[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void pack_str(msgpack_stream_t *s, const char* str, size_t len) {
|
||||||
|
// size_t len = strlen(str);
|
||||||
|
if (len < 32) {
|
||||||
|
write_bytes(s, 1, 0b10100000 | (uint8_t)len);
|
||||||
|
} else {
|
||||||
|
write_size(s, 0xd9, len);
|
||||||
|
}
|
||||||
|
for (size_t l=0; l<len; l++) {
|
||||||
|
write_bytes(s, 1, str[l]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void pack_array(msgpack_stream_t *s, size_t len) {
|
||||||
|
// only writes the header, manually write the objects after calling pack_array!
|
||||||
|
if (len < 16) {
|
||||||
|
write_bytes(s, 1, 0b10010000 | (uint8_t)len);
|
||||||
|
} else if (len < 0x10000) {
|
||||||
|
write_bytes(s, 1, 0xdc);
|
||||||
|
write_bytes(s, 2, len);
|
||||||
|
} else {
|
||||||
|
write_bytes(s, 1, 0xdd);
|
||||||
|
write_bytes(s, 4, len);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void pack_dict(msgpack_stream_t *s, size_t len) {
|
||||||
|
// only writes the header, manually write the objects after calling pack_array!
|
||||||
|
if (len < 16) {
|
||||||
|
write_bytes(s, 1, 0b10000000 | (uint8_t)len);
|
||||||
|
} else if (len < 0x10000) {
|
||||||
|
write_bytes(s, 1, 0xde);
|
||||||
|
write_bytes(s, 2, len);
|
||||||
|
} else {
|
||||||
|
write_bytes(s, 1, 0xdf);
|
||||||
|
write_bytes(s, 4, len);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void pack(mp_obj_t obj, msgpack_stream_t *s) {
|
||||||
|
if (MP_OBJ_IS_SMALL_INT(obj)) {
|
||||||
|
// int
|
||||||
|
int32_t x = MP_OBJ_SMALL_INT_VALUE(obj);
|
||||||
|
pack_int(s, x);
|
||||||
|
} else if (MP_OBJ_IS_STR(obj)) {
|
||||||
|
// string
|
||||||
|
size_t len;
|
||||||
|
const char *data = mp_obj_str_get_data(obj, &len);
|
||||||
|
pack_str(s, data, len);
|
||||||
|
} else if (MP_OBJ_IS_TYPE(obj, &mp_type_bytes)) {
|
||||||
|
// bytes
|
||||||
|
mp_buffer_info_t bufinfo;
|
||||||
|
mp_get_buffer_raise(obj, &bufinfo, MP_BUFFER_READ);
|
||||||
|
pack_bin(s, bufinfo.buf, bufinfo.len);
|
||||||
|
} else if (MP_OBJ_IS_TYPE(obj, &mp_type_tuple)) {
|
||||||
|
// tuple
|
||||||
|
mp_obj_tuple_t *self = MP_OBJ_TO_PTR(obj);
|
||||||
|
pack_array(s, self->len);
|
||||||
|
for (size_t i=0; i<self->len; i++) {
|
||||||
|
pack(self->items[i], s);
|
||||||
|
}
|
||||||
|
} else if (MP_OBJ_IS_TYPE(obj, &mp_type_list)) {
|
||||||
|
// list (layout differs from tuple)
|
||||||
|
mp_obj_list_t *self = MP_OBJ_TO_PTR(obj);
|
||||||
|
pack_array(s, self->len);
|
||||||
|
for (size_t i=0; i<self->len; i++) {
|
||||||
|
pack(self->items[i], s);
|
||||||
|
}
|
||||||
|
} else if (MP_OBJ_IS_TYPE(obj, &mp_type_dict)) {
|
||||||
|
// dict
|
||||||
|
mp_obj_dict_t *self = MP_OBJ_TO_PTR(obj);
|
||||||
|
pack_dict(s, self->map.used);
|
||||||
|
size_t cur = 0;
|
||||||
|
mp_map_elem_t *next = NULL;
|
||||||
|
while ((next = dict_iter_next(self, &cur)) != NULL) {
|
||||||
|
pack(next->key, s);
|
||||||
|
pack(next->value, s);
|
||||||
|
}
|
||||||
|
} else if (mp_obj_is_float(obj)) {
|
||||||
|
mp_float_t f = mp_obj_float_get(obj);
|
||||||
|
write_bytes(s, 1, 0xca);
|
||||||
|
write(s, &f, 4);
|
||||||
|
} else if (obj == mp_const_none) {
|
||||||
|
write_bytes(s, 1, 0xc0);
|
||||||
|
} else if (obj == mp_const_false) {
|
||||||
|
write_bytes(s, 1, 0xc2);
|
||||||
|
} else if (obj == mp_const_true) {
|
||||||
|
write_bytes(s, 1, 0xc3);
|
||||||
|
} else {
|
||||||
|
mp_raise_ValueError(translate("no packer"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////
|
||||||
|
// unpacker
|
||||||
|
|
||||||
|
mp_obj_t unpack(msgpack_stream_t *s) {
|
||||||
|
uint8_t code = read_bytes(s, 1);
|
||||||
|
if (((code & 0b10000000) == 0) || ((code & 0b11100000) == 0b11100000)) {
|
||||||
|
// int
|
||||||
|
return MP_OBJ_NEW_SMALL_INT((int8_t)code);
|
||||||
|
}
|
||||||
|
if ((code & 0b11100000) == 0b10100000) {
|
||||||
|
// str
|
||||||
|
size_t len = code & 0b11111;
|
||||||
|
// allocate on stack; len < 32
|
||||||
|
char str[len];
|
||||||
|
read(s, &str, len);
|
||||||
|
return mp_obj_new_str(str, len);
|
||||||
|
}
|
||||||
|
if ((code & 0b11110000) == 0b10010000) {
|
||||||
|
// array (tuple)
|
||||||
|
size_t len = code & 0b1111;
|
||||||
|
mp_obj_tuple_t *t = MP_OBJ_TO_PTR(mp_obj_new_tuple(len, NULL));
|
||||||
|
for (size_t i=0; i<len; i++) {
|
||||||
|
t->items[i] = unpack(s);
|
||||||
|
}
|
||||||
|
return MP_OBJ_FROM_PTR(t);
|
||||||
|
}
|
||||||
|
if ((code & 0b11110000) == 0b10000000) {
|
||||||
|
// map (dict)
|
||||||
|
size_t len = code & 0b1111;
|
||||||
|
mp_obj_dict_t *d = MP_OBJ_TO_PTR(mp_obj_new_dict(len));
|
||||||
|
for (size_t i=0; i<len; i++) {
|
||||||
|
mp_obj_dict_store(d, unpack(s), unpack(s));
|
||||||
|
}
|
||||||
|
return MP_OBJ_FROM_PTR(d);
|
||||||
|
}
|
||||||
|
switch (code) {
|
||||||
|
case 0xc0: return mp_const_none;
|
||||||
|
case 0xc2: return mp_const_false;
|
||||||
|
case 0xc3: return mp_const_true;
|
||||||
|
case 0xc4:
|
||||||
|
case 0xc5:
|
||||||
|
case 0xc6: {
|
||||||
|
// bin 8, 16, 32
|
||||||
|
size_t size = read_size(s, code-0xc4);
|
||||||
|
vstr_t vstr;
|
||||||
|
vstr_init_len(&vstr, size);
|
||||||
|
byte *p = (byte*)vstr.buf;
|
||||||
|
read(s, p, size);
|
||||||
|
return mp_obj_new_str_from_vstr(&mp_type_bytes, &vstr);
|
||||||
|
}
|
||||||
|
case 0xca: {
|
||||||
|
float f;
|
||||||
|
read(s, &f, 4);
|
||||||
|
return mp_obj_new_float(f);
|
||||||
|
}
|
||||||
|
case 0xd0:
|
||||||
|
return MP_OBJ_NEW_SMALL_INT((int8_t)read_bytes(s, 1));
|
||||||
|
case 0xd1:
|
||||||
|
return MP_OBJ_NEW_SMALL_INT((int16_t)read_bytes(s, 2));
|
||||||
|
case 0xd2:
|
||||||
|
return MP_OBJ_NEW_SMALL_INT((int32_t)read_bytes(s, 4));
|
||||||
|
case 0xd9:
|
||||||
|
case 0xda:
|
||||||
|
case 0xdb: {
|
||||||
|
// str 8, 16, 32
|
||||||
|
size_t size = read_size(s, code-0xd9);
|
||||||
|
vstr_t vstr;
|
||||||
|
vstr_init_len(&vstr, size);
|
||||||
|
byte *p = (byte*)vstr.buf;
|
||||||
|
read(s, p, size);
|
||||||
|
return mp_obj_new_str_from_vstr(&mp_type_str, &vstr);
|
||||||
|
}
|
||||||
|
case 0xde:
|
||||||
|
case 0xdf: {
|
||||||
|
// map 16 & 32
|
||||||
|
size_t len = read_size(s, code - 0xde + 1);
|
||||||
|
mp_obj_dict_t *d = MP_OBJ_TO_PTR(mp_obj_new_dict(len));
|
||||||
|
for (size_t i=0; i<len; i++) {
|
||||||
|
mp_obj_dict_store(d, unpack(s), unpack(s));
|
||||||
|
}
|
||||||
|
return MP_OBJ_FROM_PTR(d);
|
||||||
|
}
|
||||||
|
case 0xdc:
|
||||||
|
case 0xdd: {
|
||||||
|
// array 16 & 32
|
||||||
|
size_t len = read_size(s, code - 0xdc + 1);
|
||||||
|
mp_obj_tuple_t *t = MP_OBJ_TO_PTR(mp_obj_new_tuple(len, NULL));
|
||||||
|
for (size_t i=0; i<len; i++) {
|
||||||
|
t->items[i] = unpack(s);
|
||||||
|
}
|
||||||
|
return MP_OBJ_FROM_PTR(t);
|
||||||
|
}
|
||||||
|
case 0xc1: // never used
|
||||||
|
case 0xc7: // ext 8
|
||||||
|
case 0xc8: // ext 16
|
||||||
|
case 0xc9: // ext 32
|
||||||
|
case 0xcb: // float 64
|
||||||
|
case 0xcc: // uint 8
|
||||||
|
case 0xcd: // uint 16
|
||||||
|
case 0xce: // uint 32
|
||||||
|
case 0xcf: // uint 64
|
||||||
|
case 0xd3: // int 64
|
||||||
|
case 0xd4: // fixenxt 1
|
||||||
|
case 0xd5: // fixenxt 2
|
||||||
|
case 0xd6: // fixenxt 4
|
||||||
|
case 0xd7: // fixenxt 8
|
||||||
|
case 0xd8: // fixenxt 16
|
||||||
|
default:
|
||||||
|
mp_raise_ValueError(translate("no unpacker found"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void common_hal_msgpack_pack(mp_obj_t obj, mp_obj_t stream_obj) {
|
||||||
|
msgpack_stream_t stream = get_stream(stream_obj, MP_STREAM_OP_WRITE);
|
||||||
|
pack(obj, &stream);
|
||||||
|
}
|
||||||
|
|
||||||
|
mp_obj_t common_hal_msgpack_unpack(mp_obj_t stream_obj) {
|
||||||
|
msgpack_stream_t stream = get_stream(stream_obj, MP_STREAM_OP_WRITE);
|
||||||
|
return unpack(&stream);
|
||||||
|
}
|
||||||
|
|
34
shared-module/msgpack/__init__.h
Normal file
34
shared-module/msgpack/__init__.h
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of the MicroPython project, http://micropython.org/
|
||||||
|
*
|
||||||
|
* The MIT License (MIT)
|
||||||
|
*
|
||||||
|
* Copyright (c) 2017 Scott Shawcroft for Adafruit Industries
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in
|
||||||
|
* all copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
* THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
#ifndef MICROPY_INCLUDED_SHARED_MODULE_MSGPACK___INIT___H
|
||||||
|
#define MICROPY_INCLUDED_SHARED_MODULE_MSGPACK___INIT___H
|
||||||
|
|
||||||
|
#include "py/stream.h"
|
||||||
|
|
||||||
|
void common_hal_msgpack_pack(mp_obj_t obj, mp_obj_t stream_obj);
|
||||||
|
mp_obj_t common_hal_msgpack_unpack(mp_obj_t stream_obj);
|
||||||
|
|
||||||
|
#endif
|
Loading…
x
Reference in New Issue
Block a user