#include "ksbit.h" // keep in sync with github/pxt/pxtsim/libgeneric.ts enum class NumberFormat { Int8LE = 1, UInt8LE, Int16LE, UInt16LE, Int32LE, Int8BE, UInt8BE, Int16BE, UInt16BE, Int32BE, // UInt32, }; //% indexerGet=BufferMethods::getByte indexerSet=BufferMethods::setByte namespace BufferMethods { //% int getByte(Buffer buf, int off) { return max(ManagedBuffer(buf).getByte(off), 0); } //% void setByte(Buffer buf, int off, int v) { ManagedBuffer(buf).setByte(off, v); } //% uint8_t *getBytes(Buffer buf) { return buf->payload; } /** * Write a number in specified format in the buffer. */ //% void setNumber(Buffer buf, NumberFormat format, int offset, int value) { int8_t i8; uint8_t u8; int16_t i16; uint16_t u16; int32_t i32; ManagedBuffer b(buf); // Assume little endian #define WRITEBYTES(isz, swap) isz = value; b.writeBytes(offset, (uint8_t*)&isz, sizeof(isz), swap); break switch (format) { case NumberFormat::Int8LE: WRITEBYTES(i8, false); case NumberFormat::UInt8LE: WRITEBYTES(u8, false); case NumberFormat::Int16LE: WRITEBYTES(i16, false); case NumberFormat::UInt16LE: WRITEBYTES(u16, false); case NumberFormat::Int32LE: WRITEBYTES(i32, false); case NumberFormat::Int8BE: WRITEBYTES(i8, true); case NumberFormat::UInt8BE: WRITEBYTES(u8, true); case NumberFormat::Int16BE: WRITEBYTES(i16, true); case NumberFormat::UInt16BE: WRITEBYTES(u16, true); case NumberFormat::Int32BE: WRITEBYTES(i32, true); } } /** * Read a number in specified format from the buffer. */ //% int getNumber(Buffer buf, NumberFormat format, int offset) { int8_t i8; uint8_t u8; int16_t i16; uint16_t u16; int32_t i32; ManagedBuffer b(buf); // Assume little endian #define READBYTES(isz, swap) b.readBytes((uint8_t*)&isz, offset, sizeof(isz), swap); return isz switch (format) { case NumberFormat::Int8LE: READBYTES(i8, false); case NumberFormat::UInt8LE: READBYTES(u8, false); case NumberFormat::Int16LE: READBYTES(i16, false); case NumberFormat::UInt16LE: READBYTES(u16, false); case NumberFormat::Int32LE: READBYTES(i32, false); case NumberFormat::Int8BE: READBYTES(i8, true); case NumberFormat::UInt8BE: READBYTES(u8, true); case NumberFormat::Int16BE: READBYTES(i16, true); case NumberFormat::UInt16BE: READBYTES(u16, true); case NumberFormat::Int32BE: READBYTES(i32, true); } return 0; } /** Returns the length of a Buffer object. */ //% property int length(Buffer s) { return s->length; } /** * Fill (a fragment) of the buffer with given value. */ //% void fill(Buffer buf, int value, int offset = 0, int length = -1) { ManagedBuffer(buf).fill(value, offset, length); } /** * Return a copy of a fragment of a buffer. */ //% Buffer slice(Buffer buf, int offset = 0, int length = -1) { return ManagedBuffer(buf).slice(offset, length).leakData(); } /** * Shift buffer left in place, with zero padding. * @param offset number of bytes to shift; use negative value to shift right * @param start start offset in buffer. Default is 0. * @param length number of elements in buffer. If negative, length is set as the buffer length minus start. eg: -1 */ //% void shift(Buffer buf, int offset, int start = 0, int length = -1) { ManagedBuffer(buf).shift(offset, start, length); } /** * Rotate buffer left in place. * @param offset number of bytes to shift; use negative value to shift right * @param start start offset in buffer. Default is 0. * @param length number of elements in buffer. If negative, length is set as the buffer length minus start. eg: -1 */ //% void rotate(Buffer buf, int offset, int start = 0, int length = -1) { ManagedBuffer(buf).rotate(offset, start, length); } // int readBytes(uint8_t *dst, int offset, int length, bool swapBytes = false) const; // int writeBytes(int dstOffset, uint8_t *src, int length, bool swapBytes = false); /** * Write contents of `src` at `dstOffset` in current buffer. */ //% void write(Buffer buf, int dstOffset, Buffer src) { //Not supported, we only do up to 4 args :/ //void write(Buffer buf, int dstOffset, Buffer src, int srcOffset = 0, int length = -1) ManagedBuffer(buf).writeBuffer(dstOffset, ManagedBuffer(src), 0, -1); } }