/* vim: set expandtab ts=4 sw=4: */ /* * You may redistribute this program and/or modify it under the terms of * the GNU General Public License as published by the Free Software Foundation, * either version 3 of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef Message_H #define Message_H #include "exception/Except.h" #include #include "memory/Allocator.h" #include "util/Bits.h" #include "util/UniqueName.h" struct Message { /** The length of the message. */ int32_t length; /** The number of bytes of padding BEFORE where bytes begins. */ int32_t padding; /** The content. */ uint8_t* bytes; /** Amount of bytes of storage space available in the message. */ uint32_t capacity; /** The allocator which allocated space for this message. */ struct Allocator* alloc; }; #define Message_STACK(name, messageLength, amountOfPadding) \ uint8_t UniqueName_get()[messageLength + amountOfPadding]; \ name = &(struct Message){ \ .length = messageLength, \ .bytes = UniqueName_last() + amountOfPadding, \ .padding = amountOfPadding, \ .capacity = messageLength \ } static inline struct Message* Message_new(uint32_t messageLength, uint32_t amountOfPadding, struct Allocator* alloc) { uint8_t* buff = Allocator_malloc(alloc, messageLength + amountOfPadding); struct Message* out = Allocator_malloc(alloc, sizeof(struct Message)); out->bytes = &buff[amountOfPadding]; out->length = out->capacity = messageLength; out->padding = amountOfPadding; out->alloc = alloc; return out; } static inline struct Message* Message_clone(struct Message* toClone, struct Allocator* allocator) { uint32_t len = toClone->length + toClone->padding; if (len < (toClone->capacity + toClone->padding)) { len = (toClone->capacity + toClone->padding); } uint8_t* allocation = Allocator_malloc(allocator, len); Bits_memcpy(allocation, toClone->bytes - toClone->padding, len); return Allocator_clone(allocator, (&(struct Message) { .length = toClone->length, .padding = toClone->padding, .bytes = allocation + toClone->padding, .capacity = toClone->length, .alloc = allocator })); } static inline void Message_copyOver(struct Message* output, struct Message* input, struct Allocator* allocator) { size_t inTotalLength = input->length + input->padding; size_t outTotalLength = output->length + output->padding; uint8_t* allocation = output->bytes - output->padding; if (inTotalLength > outTotalLength) { allocation = Allocator_realloc(allocator, allocation, inTotalLength); } Bits_memcpy(allocation, input->bytes - input->padding, inTotalLength); output->bytes = allocation + input->padding; output->length = input->length; output->padding = input->padding; } /** * Pretend to shift the content forward by amount. * Really it shifts the bytes value backward. */ static inline int Message_shift(struct Message* toShift, int32_t amount, struct Except* eh) { if (amount > 0 && toShift->padding < amount) { Except_throw(eh, "buffer overflow"); } else if (toShift->length < (-amount)) { Except_throw(eh, "buffer underflow"); } toShift->length += amount; toShift->capacity += amount; toShift->bytes -= amount; toShift->padding -= amount; return 1; } static inline void Message_push(struct Message* restrict msg, const void* restrict object, size_t size, struct Except* eh) { Message_shift(msg, (int)size, eh); Bits_memcpy(msg->bytes, object, size); } static inline void Message_pop(struct Message* restrict msg, void* restrict object, size_t size, struct Except* eh) { Message_shift(msg, -((int)size), eh); Bits_memcpy(object, &msg->bytes[-((int)size)], size); } #define Message_popGeneric(size) \ static inline uint ## size ## _t Message_pop ## size (struct Message* msg, struct Except* eh) \ { \ uint ## size ## _t out; \ Message_pop(msg, &out, (size)/8, eh); \ return Endian_bigEndianToHost ## size (out); \ } Message_popGeneric(8) Message_popGeneric(16) Message_popGeneric(32) Message_popGeneric(64) #define Message_pushGeneric(size) \ static inline void Message_push ## size \ (struct Message* msg, uint ## size ## _t dat, struct Except* eh) \ { \ uint ## size ## _t x = Endian_hostToBigEndian ## size (dat); \ Message_push(msg, &x, (size)/8, eh); \ } Message_pushGeneric(8) Message_pushGeneric(16) Message_pushGeneric(32) Message_pushGeneric(64) #endif