summaryrefslogtreecommitdiffstats
path: root/c
diff options
context:
space:
mode:
authorStephen Dolan <mu@netsoc.tcd.ie>2012-09-17 19:41:41 +0100
committerStephen Dolan <mu@netsoc.tcd.ie>2012-09-17 19:41:41 +0100
commit84ea129f26e46a0e38dd630f6d02cad41b983808 (patch)
tree2a018a2589a380a9112ce9555b16710247c04bcc /c
parentdf195b31873010f18883446c6e0e629a594badc5 (diff)
Better support for appending strings in JV.
Diffstat (limited to 'c')
-rw-r--r--c/jv.c59
-rw-r--r--c/jv.h2
-rw-r--r--c/testdata4
3 files changed, 48 insertions, 17 deletions
diff --git a/c/jv.c b/c/jv.c
index 786e7b26..595484e7 100644
--- a/c/jv.c
+++ b/c/jv.c
@@ -340,6 +340,7 @@ typedef struct {
// high 31 bits are length, low bit is a flag
// indicating whether hash has been computed.
uint32_t length_hashed;
+ uint32_t alloc_length;
char data[];
} jvp_string;
@@ -350,19 +351,19 @@ static jvp_string* jvp_string_ptr(jv_complex* a) {
static jvp_string* jvp_string_alloc(uint32_t size) {
jvp_string* s = malloc(sizeof(jvp_string) + size + 1);
s->refcnt.count = 1;
- s->length_hashed = size << 1;
+ s->alloc_length = size;
return s;
}
static jv_complex jvp_string_new(const char* data, uint32_t length) {
jvp_string* s = jvp_string_alloc(length);
+ s->length_hashed = length << 1;
memcpy(s->data, data, length);
s->data[length] = 0;
jv_complex r = {&s->refcnt, {0,0}};
return r;
}
-
static void jvp_string_free(jv_complex* s) {
if (jvp_refcnt_dec(s)) {
jvp_string* str = jvp_string_ptr(s);
@@ -385,16 +386,37 @@ static uint32_t jvp_string_length(jvp_string* s) {
return s->length_hashed >> 1;
}
-static jv_complex jvp_string_concat(jvp_string* a, jvp_string* b) {
- uint32_t la = jvp_string_length(a), lb = jvp_string_length(b);
- jvp_string* s = jvp_string_alloc(la + lb);
- memcpy(s->data, a->data, la);
- memcpy(s->data + la, b->data, lb);
- s->data[la + lb] = 0;
- jv_complex r = {&s->refcnt, {0,0}};
+static uint32_t jvp_string_remaining_space(jvp_string* s) {
+ uint32_t r = s->alloc_length - jvp_string_length(s);
+ assert(r >= 0);
return r;
}
+static void jvp_string_append(jv_complex* string, const char* data, uint32_t len) {
+ jvp_string* s = jvp_string_ptr(string);
+ uint32_t currlen = jvp_string_length(s);
+
+ if (jvp_refcnt_unshared(string) &&
+ jvp_string_remaining_space(s) >= len) {
+ // the next string fits at the end of a
+ memcpy(s->data + currlen, data, len);
+ s->data[currlen + len] = 0;
+ s->length_hashed = (currlen + len) << 1;
+ } else {
+ // allocate a bigger buffer and copy
+ uint32_t allocsz = (currlen + len) * 2;
+ if (allocsz < 32) allocsz = 32;
+ jvp_string* news = jvp_string_alloc(allocsz);
+ news->length_hashed = (currlen + len) << 1;
+ memcpy(news->data, s->data, currlen);
+ memcpy(news->data + currlen, data, len);
+ news->data[currlen + len] = 0;
+ jvp_string_free(string);
+ jv_complex r = {&news->refcnt, {0,0}};
+ *string = r;
+ }
+}
+
static const uint32_t HASH_SEED = 0x432A9843;
static uint32_t rotl32 (uint32_t x, int8_t r){
@@ -507,15 +529,22 @@ const char* jv_string_value(jv j) {
}
jv jv_string_concat(jv a, jv b) {
- jv j;
- j.kind = JV_KIND_STRING;
- j.val.complex = jvp_string_concat(jvp_string_ptr(&a.val.complex),
- jvp_string_ptr(&b.val.complex));
- jv_free(a);
+ jvp_string* sb = jvp_string_ptr(&b.val.complex);
+ jvp_string_append(&a.val.complex, sb->data, jvp_string_length(sb));
jv_free(b);
- return j;
+ return a;
}
+jv jv_string_append_buf(jv a, const char* buf, int len) {
+ jvp_string_append(&a.val.complex, buf, len);
+ return a;
+}
+
+jv jv_string_append_str(jv a, const char* str) {
+ return jv_string_append_buf(a, str, strlen(str));
+}
+
+
jv jv_string_fmt(const char* fmt, ...) {
int size = 1024;
while (1) {
diff --git a/c/jv.h b/c/jv.h
index ec23262c..295369e8 100644
--- a/c/jv.h
+++ b/c/jv.h
@@ -80,6 +80,8 @@ uint32_t jv_string_hash(jv);
const char* jv_string_value(jv);
jv jv_string_concat(jv, jv);
jv jv_string_fmt(const char*, ...);
+jv jv_string_append_buf(jv a, const char* buf, int len);
+jv jv_string_append_str(jv a, const char* str);
jv jv_object();
jv jv_object_get(jv object, jv key);
diff --git a/c/testdata b/c/testdata
index e17f2bb7..c0afbeae 100644
--- a/c/testdata
+++ b/c/testdata
@@ -159,9 +159,9 @@ null
"asdfasdf"
{"a":1, "b":2, "c":3}
-"asdf" + "jkl;" + .
+"asdf" + "jkl;" + . + . + .
"some string"
-"asdfjkl;some string"
+"asdfjkl;some stringsome stringsome string"
"\u0000\u0020\u0000" + .
"\u0000\u0020\u0000"