main
parent 11a67b7b26
commit c61d6a0a96

104
ser.h

@ -4,65 +4,64 @@
typedef struct typedef struct
{ {
b8 failed; b8 failed;
String8 why; String8 why;
} SerError; } SerError;
typedef struct typedef struct
{ {
u8 *data; // set to 0 to dry run and get maximum size. max doesn't matter in this case u8 *data; // set to 0 to dry run and get maximum size. max doesn't matter in this case
u64 cur; u64 cur;
u64 max; u64 max;
Arena *arena; // allocate everything new on this, so that if serialization fails allocations can be undone Arena *arena; // allocate everything new on this, so that if serialization fails allocations can be undone
// Serializing should never allocate. So this can be null when you serialize // Serializing should never allocate. So this can be null when you serialize
int version; int version;
SerError cur_error; SerError cur_error;
Arena *error_arena; // all error messages are allocated here Arena *error_arena; // all error messages are allocated here
b8 serializing; b8 serializing;
} SerState; } SerState;
void ser_bytes(SerState *ser, u8 *bytes, u64 bytes_size) void ser_bytes(SerState *ser, u8 *bytes, u64 bytes_size)
{ {
if(!ser->data && !ser->serializing) if (!ser->data && !ser->serializing)
{
ser->cur_error = (SerError){.failed = true, .why = S8Lit("Deserializing but the data is null")};
}
if(!ser->cur_error.failed)
{
if(ser->data)
{
// maximum doesn't matter unless writing to data
if(ser->cur + bytes_size > ser->max)
{
ser->cur_error = (SerError){.failed = true, .why = S8Lit("Too big bro")};
}
else
{
if(ser->serializing)
{
memcpy(ser->data + ser->cur, bytes, bytes_size);
}
else
{ {
memcpy(bytes, ser->data + ser->cur, bytes_size); ser->cur_error = (SerError){.failed = true, .why = S8Lit("Deserializing but the data is null")};
} }
}
} if (!ser->cur_error.failed)
{
if (ser->data)
{
// maximum doesn't matter unless writing to data
if (ser->cur + bytes_size > ser->max)
{
ser->cur_error = (SerError){.failed = true, .why = S8Lit("Too big bro")};
}
else
{
if (ser->serializing)
{
memcpy(ser->data + ser->cur, bytes, bytes_size);
}
else
{
memcpy(bytes, ser->data + ser->cur, bytes_size);
}
}
}
ser->cur += bytes_size; ser->cur += bytes_size;
} }
} }
#define SER_MAKE_FOR_TYPE(type) void ser_##type(SerState *ser, type *into) \ #define SER_MAKE_FOR_TYPE(type) \
{ \ void ser_##type(SerState *ser, type *into) \
ser_bytes(ser, (u8*)into, sizeof(*into)); \ { \
} ser_bytes(ser, (u8 *)into, sizeof(*into)); \
}
SER_MAKE_FOR_TYPE(int); SER_MAKE_FOR_TYPE(int);
SER_MAKE_FOR_TYPE(u64); SER_MAKE_FOR_TYPE(u64);
@ -71,15 +70,14 @@ SER_MAKE_FOR_TYPE(u64);
// deserialized data onto a newly allocated buffer // deserialized data onto a newly allocated buffer
void ser_String8(SerState *ser, String8 *s, Arena *allocate_onto) void ser_String8(SerState *ser, String8 *s, Arena *allocate_onto)
{ {
ser_u64(ser, &s->size); ser_u64(ser, &s->size);
if(ser->serializing) if (ser->serializing)
{ {
ser_bytes(ser, s->str, s->size); ser_bytes(ser, s->str, s->size);
} }
else else
{ {
s->str = ArenaPush(allocate_onto, s->size); s->str = ArenaPush(allocate_onto, s->size);
ser_bytes(ser, s->str, s->size); ser_bytes(ser, s->str, s->size);
} }
} }

Loading…
Cancel
Save