1#include "prism/internal/buffer.h"
5#include "prism/internal/char.h"
6#include "prism/internal/allocator.h"
18pm_buffer_init(
pm_buffer_t *buffer,
size_t capacity) {
20 buffer->capacity = capacity;
22 buffer->value = (
char *)
xmalloc(capacity);
23 if (buffer->value == NULL) abort();
32 if (buffer == NULL) abort();
34 pm_buffer_init(buffer, 1024);
51 return buffer->length;
58pm_buffer_append_length(
pm_buffer_t *buffer,
size_t length) {
59 size_t next_length = buffer->length + length;
60 const size_t original_capacity = buffer->capacity;
62 if (next_length > buffer->capacity) {
63 if (buffer->capacity == 0) {
67 while (next_length > buffer->capacity) {
68 buffer->capacity *= 2;
71 buffer->value = xrealloc_sized(buffer->value, buffer->capacity, original_capacity);
72 if (buffer->value == NULL)
return false;
75 buffer->length = next_length;
83pm_buffer_append(
pm_buffer_t *buffer,
const void *source,
size_t length) {
84 size_t cursor = buffer->length;
85 if (pm_buffer_append_length(buffer, length)) {
86 memcpy(buffer->value + cursor, source, length);
94pm_buffer_append_zeroes(
pm_buffer_t *buffer,
size_t length) {
95 size_t cursor = buffer->length;
96 if (pm_buffer_append_length(buffer, length)) {
97 memset(buffer->value + cursor, 0, length);
105pm_buffer_append_format(
pm_buffer_t *buffer,
const char *format, ...) {
107 va_start(arguments, format);
108 int result = vsnprintf(NULL, 0, format, arguments);
111 if (result < 0)
return;
112 size_t length = (size_t) (result + 1);
114 size_t cursor = buffer->length;
115 if (pm_buffer_append_length(buffer, length)) {
116 va_start(arguments, format);
117 vsnprintf(buffer->value + cursor, length, format, arguments);
127pm_buffer_append_string(
pm_buffer_t *buffer,
const char *value,
size_t length) {
128 pm_buffer_append(buffer, value, length);
135pm_buffer_append_bytes(
pm_buffer_t *buffer,
const uint8_t *value,
size_t length) {
136 pm_buffer_append(buffer, (
const char *) value, length);
143pm_buffer_append_byte(
pm_buffer_t *buffer, uint8_t value) {
144 const void *source = &value;
145 pm_buffer_append(buffer, source,
sizeof(uint8_t));
152pm_buffer_append_varuint(
pm_buffer_t *buffer, uint32_t value) {
154 pm_buffer_append_byte(buffer, (uint8_t) value);
158 pm_buffer_append_byte(buffer, (uint8_t) (n | 128));
161 pm_buffer_append_byte(buffer, (uint8_t) n);
169pm_buffer_append_varsint(
pm_buffer_t *buffer, int32_t value) {
170 uint32_t unsigned_int = ((uint32_t)(value) << 1) ^ ((uint32_t)(value >> 31));
171 pm_buffer_append_varuint(buffer, unsigned_int);
178pm_buffer_append_double(
pm_buffer_t *buffer,
double value) {
179 const void *source = &value;
180 pm_buffer_append(buffer, source,
sizeof(
double));
187pm_buffer_append_unicode_codepoint(
pm_buffer_t *buffer, uint32_t value) {
189 pm_buffer_append_byte(buffer, (uint8_t) value);
191 }
else if (value <= 0x7FF) {
193 (uint8_t) (0xC0 | ((value >> 6) & 0x3F)),
194 (uint8_t) (0x80 | (value & 0x3F))
197 pm_buffer_append_bytes(buffer, bytes, 2);
199 }
else if (value <= 0xFFFF) {
201 (uint8_t) (0xE0 | ((value >> 12) & 0x3F)),
202 (uint8_t) (0x80 | ((value >> 6) & 0x3F)),
203 (uint8_t) (0x80 | (value & 0x3F))
206 pm_buffer_append_bytes(buffer, bytes, 3);
208 }
else if (value <= 0x10FFFF) {
210 (uint8_t) (0xF0 | ((value >> 18) & 0x3F)),
211 (uint8_t) (0x80 | ((value >> 12) & 0x3F)),
212 (uint8_t) (0x80 | ((value >> 6) & 0x3F)),
213 (uint8_t) (0x80 | (value & 0x3F))
216 pm_buffer_append_bytes(buffer, bytes, 4);
227pm_buffer_append_source(
pm_buffer_t *buffer,
const uint8_t *source,
size_t length, pm_buffer_escaping_t escaping) {
228 for (
size_t index = 0; index < length; index++) {
229 const uint8_t
byte = source[index];
231 if ((
byte <= 0x06) || (
byte >= 0x0E &&
byte <= 0x1F) || (
byte >= 0x7F)) {
232 if (escaping == PM_BUFFER_ESCAPING_RUBY) {
233 pm_buffer_append_format(buffer,
"\\x%02X",
byte);
235 pm_buffer_append_format(buffer,
"\\u%04X",
byte);
240 if (escaping == PM_BUFFER_ESCAPING_RUBY) {
241 pm_buffer_append_string(buffer,
"\\a", 2);
243 pm_buffer_append_format(buffer,
"\\u%04X",
byte);
247 pm_buffer_append_string(buffer,
"\\b", 2);
250 pm_buffer_append_string(buffer,
"\\t", 2);
253 pm_buffer_append_string(buffer,
"\\n", 2);
256 if (escaping == PM_BUFFER_ESCAPING_RUBY) {
257 pm_buffer_append_string(buffer,
"\\v", 2);
259 pm_buffer_append_format(buffer,
"\\u%04X",
byte);
263 pm_buffer_append_string(buffer,
"\\f", 2);
266 pm_buffer_append_string(buffer,
"\\r", 2);
269 pm_buffer_append_string(buffer,
"\\\"", 2);
272 if (escaping == PM_BUFFER_ESCAPING_RUBY && index + 1 < length) {
273 const uint8_t next_byte = source[index + 1];
274 if (next_byte ==
'{' || next_byte ==
'@' || next_byte ==
'$') {
275 pm_buffer_append_byte(buffer,
'\\');
279 pm_buffer_append_byte(buffer,
'#');
283 pm_buffer_append_string(buffer,
"\\\\", 2);
286 pm_buffer_append_byte(buffer,
byte);
297pm_buffer_prepend_string(
pm_buffer_t *buffer,
const char *value,
size_t length) {
298 size_t cursor = buffer->length;
299 if (pm_buffer_append_length(buffer, length)) {
300 memmove(buffer->value + length, buffer->value, cursor);
301 memcpy(buffer->value, value, length);
310 if (source->length > 0) {
311 pm_buffer_append(destination, source->value, source->length);
329 while (buffer->length > 0 && pm_char_is_whitespace((uint8_t) buffer->value[buffer->length - 1])) {
338pm_buffer_index(
const pm_buffer_t *buffer,
char value) {
339 const char *first = memchr(buffer->value, value, buffer->length);
340 return (first == NULL) ? SIZE_MAX : (size_t) (first - buffer->value);
347pm_buffer_insert(
pm_buffer_t *buffer,
size_t index,
const char *value,
size_t length) {
348 assert(index <= buffer->length);
350 if (index == buffer->length) {
351 pm_buffer_append_string(buffer, value, length);
353 pm_buffer_append_zeroes(buffer, length);
354 memmove(buffer->value + index + length, buffer->value + index, buffer->length - length - index);
355 memcpy(buffer->value + index, value, length);
364 xfree_sized(buffer->value, buffer->capacity);
372 pm_buffer_cleanup(buffer);
#define xmalloc
Old name of ruby_xmalloc.
#define PRISM_INLINE
Old Visual Studio versions do not support the inline keyword, so we need to define it to be __inline.