aboutsummaryrefslogtreecommitdiffstats
path: root/prism/util/pm_buffer.c
blob: 0d84375767251cc1233f86c223b2e03367a6c507 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
#include "prism/util/pm_buffer.h"

#define PRISM_BUFFER_INITIAL_SIZE 1024

// Return the size of the pm_buffer_t struct.
size_t
pm_buffer_sizeof(void) {
    return sizeof(pm_buffer_t);
}

// Initialize a pm_buffer_t with its default values.
bool
pm_buffer_init(pm_buffer_t *buffer) {
    buffer->length = 0;
    buffer->capacity = PRISM_BUFFER_INITIAL_SIZE;

    buffer->value = (char *) malloc(PRISM_BUFFER_INITIAL_SIZE);
    return buffer->value != NULL;
}

#undef PRISM_BUFFER_INITIAL_SIZE

// Return the value of the buffer.
char *
pm_buffer_value(pm_buffer_t *buffer) {
    return buffer->value;
}

// Return the length of the buffer.
size_t
pm_buffer_length(pm_buffer_t *buffer) {
    return buffer->length;
}

// Append the given amount of space to the buffer.
static inline void
pm_buffer_append_length(pm_buffer_t *buffer, size_t length) {
    size_t next_length = buffer->length + length;

    if (next_length > buffer->capacity) {
        do {
            buffer->capacity *= 2;
        } while (next_length > buffer->capacity);

        buffer->value = realloc(buffer->value, buffer->capacity);
    }

    buffer->length = next_length;
}

// Append a generic pointer to memory to the buffer.
static inline void
pm_buffer_append(pm_buffer_t *buffer, const void *source, size_t length) {
    pm_buffer_append_length(buffer, length);
    memcpy(buffer->value + (buffer->length - length), source, length);
}

// Append the given amount of space as zeroes to the buffer.
void
pm_buffer_append_zeroes(pm_buffer_t *buffer, size_t length) {
    pm_buffer_append_length(buffer, length);
    memset(buffer->value + (buffer->length - length), 0, length);
}

// Append a string to the buffer.
void
pm_buffer_append_str(pm_buffer_t *buffer, const char *value, size_t length) {
    pm_buffer_append(buffer, value, length);
}

// Append a list of bytes to the buffer.
void
pm_buffer_append_bytes(pm_buffer_t *buffer, const uint8_t *value, size_t length) {
    pm_buffer_append(buffer, (const char *) value, length);
}

// Append a single byte to the buffer.
void
pm_buffer_append_u8(pm_buffer_t *buffer, uint8_t value) {
    const void *source = &value;
    pm_buffer_append(buffer, source, sizeof(uint8_t));
}

// Append a 32-bit unsigned integer to the buffer.
void
pm_buffer_append_u32(pm_buffer_t *buffer, uint32_t value) {
    if (value < 128) {
        pm_buffer_append_u8(buffer, (uint8_t) value);
    } else {
        uint32_t n = value;
        while (n >= 128) {
            pm_buffer_append_u8(buffer, (uint8_t) (n | 128));
            n >>= 7;
        }
        pm_buffer_append_u8(buffer, (uint8_t) n);
    }
}

// Free the memory associated with the buffer.
void
pm_buffer_free(pm_buffer_t *buffer) {
    free(buffer->value);
}