Skip to content

Commit

Permalink
Implement string reference counting in cnex
Browse files Browse the repository at this point in the history
  • Loading branch information
gitlarryf committed Feb 28, 2024
1 parent bf5bb08 commit 77277c4
Show file tree
Hide file tree
Showing 5 changed files with 118 additions and 27 deletions.
22 changes: 19 additions & 3 deletions exec/cnex/cell.c
Original file line number Diff line number Diff line change
Expand Up @@ -229,7 +229,8 @@ Cell *cell_fromCell(const Cell *c)
x->other = NULL;
break;
case cBytes:
x->string = string_copyString(c->string);
x->string = c->string;
x->string->refcount++;
x->boolean = FALSE;
x->number = number_from_uint32(0);
x->object = NULL;
Expand Down Expand Up @@ -270,7 +271,8 @@ Cell *cell_fromCell(const Cell *c)
x->other = NULL;
break;
case cString:
x->string = string_copyString(c->string);
x->string = c->string;
x->string->refcount++;
x->address = NULL;
x->number = number_from_uint32(0);
x->object = NULL;
Expand Down Expand Up @@ -332,6 +334,16 @@ Cell *cell_fromCellValue(const Cell* c)
x->array = NULL;
x->other = NULL;
break;
case cString:
x->string = string_copyString(c->string);
x->address = NULL;
x->number = number_from_uint32(0);
x->object = NULL;
x->boolean = FALSE;
x->array = NULL;
x->dictionary = NULL;
x->other = NULL;
break;
default:
cell_copyCell(x, c);
break;
Expand Down Expand Up @@ -618,6 +630,9 @@ void cell_valueCopyCell(Cell *dest, const Cell *source)
case cDictionary:
dest->dictionary = dictionary_copyDictionary(source->dictionary);
break;
case cString:
dest->string = string_copyString(source->string);
break;
default:
cell_copyCell(dest, source);
break;
Expand All @@ -634,7 +649,8 @@ void cell_copyCell(Cell *dest, const Cell *source)
dest->number = source->number;
// ToDo: Split strings and bytes into separate entities; once we implement actual UTF8 strings.
if ((source->type == cString || source->type == cBytes) && source->string != NULL) {
dest->string = string_copyString(source->string);
dest->string = source->string;
dest->string->refcount++;
} else {
dest->string = NULL;
}
Expand Down
2 changes: 1 addition & 1 deletion exec/cnex/cnex.c
Original file line number Diff line number Diff line change
Expand Up @@ -823,7 +823,7 @@ void exec_STORES(TExecutor *self)
{
self->ip++;
Cell *addr = top(self->stack)->address; pop(self->stack);
cell_copyCell(addr, top(self->stack)); pop(self->stack);
cell_valueCopyCell(addr, top(self->stack)); pop(self->stack);
}

void exec_STOREY(TExecutor *self)
Expand Down
4 changes: 2 additions & 2 deletions exec/cnex/global.c
Original file line number Diff line number Diff line change
Expand Up @@ -1537,7 +1537,7 @@ void pointer__toString(TExecutor *exec)

void string__append(TExecutor *exec)
{
Cell *b = cell_fromCell(top(exec->stack)); pop(exec->stack);
Cell *b = cell_fromCellValue(top(exec->stack)); pop(exec->stack);
Cell *addr = top(exec->stack)->address; pop(exec->stack);

addr->string->data = realloc(addr->string->data, addr->string->length + b->string->length);
Expand Down Expand Up @@ -1581,7 +1581,7 @@ void string__toString(TExecutor *exec)
void string__index(TExecutor *exec)
{
Number index = top(exec->stack)->number; pop(exec->stack);
Cell *a = cell_fromCell(top(exec->stack)); pop(exec->stack);
Cell *a = cell_fromCellValue(top(exec->stack)); pop(exec->stack);

if (!number_is_integer(index)) {
char buf[100];
Expand Down
116 changes: 95 additions & 21 deletions exec/cnex/nstring.c
Original file line number Diff line number Diff line change
Expand Up @@ -49,24 +49,29 @@ TString *string_newString(void)
}

c->data = NULL;
c->refcount = 1;
c->length = 0;
return c;
}

void string_freeString(TString *s)
{
if (s) {
free(s->data);
s->data = NULL;
s->length = 0;
s->refcount--;
if (s->refcount <= 0) {
free(s->data);
s->data = NULL;
s->length = 0;
free(s);
}
}
free(s);
}

void string_clearString(TString *s)
{
if (s) {
if (s->data) {
assert(s->refcount == 1);
free(s->data);
s->data = NULL;
s->length = 0;
Expand All @@ -76,6 +81,7 @@ void string_clearString(TString *s)

TString *string_copyString(TString *s)
{
// Note: In the (new) copied string, the refcount is set back to 1, since this is a new string.
TString *r = string_newString();

if (s->data) {
Expand All @@ -97,6 +103,14 @@ TString *string_fromString(TString *s)
TString *r = string_newString();

if (s != NULL) {
if (s->refcount == 1) {
s->refcount++;
r->data = s->data;
r->length = s->length;
r->refcount = s->refcount;
return r;
}

r->length = s->length;
r->data = malloc(r->length);
if (r->data == NULL) {
Expand All @@ -110,6 +124,10 @@ TString *string_fromString(TString *s)
int string_compareString(TString *lhs, TString *rhs)
{
int r = 0;
// If the two pointers are the same, then it's equal.
if (lhs == rhs) {
return r;
}
uint64_t size = lhs->length;

if (lhs->length > rhs->length) {
Expand Down Expand Up @@ -181,20 +199,42 @@ TString *string_appendCString(TString *s, const char *ns)
return s;
}

// NOTE: This function can possibly return a new string, and the string passed in
// should never be used! Only the returned string should be referenced after
// a call to this function!
TString *string_appendChar(TString *s, char c)
{
s->data = realloc(s->data, s->length + 1);
if (s->data == NULL) {
TString *r = s;
if (r->refcount > 1) {
// We have to create a new string, since more than one string is pointing to it.
r = string_copyString(s);
// Then remove our reference to the (old) string.
string_freeString(s);
}

r->data = realloc(r->data, r->length + 1);
if (r->data == NULL) {
fatal_error("Could not reallocate data for appended character.");
}

s->length++;
s->data[s->length-1] = c;
return s;
r->length++;
r->data[r->length-1] = c;
return r;
}

// NOTE: This function can possibly return a new string, and the string passed in
// should never be used! Only the returned string should be referenced after
// a call to this function!
TString *string_appendCodePoint(TString *s, uint32_t cp)
{
TString *r = s;
if (r->refcount > 1) {
// We have to create a new string, since more than one string is pointing to it.
r = string_copyString(s);
// Then remove our reference to the (old) string.
string_freeString(s);
}

char val[4];
uint32_t lead_byte = 0x7F;
int index = 0;
Expand All @@ -205,21 +245,33 @@ TString *string_appendCodePoint(TString *s, uint32_t cp)
}
val[index++] = (cp & lead_byte) | (~lead_byte << 1);
while (index) {
s = string_appendChar(s, val[--index]);
r = string_appendChar(s, val[--index]);
}
return s;
return r;
}

// NOTE: This function can possibly return a new string, and the string passed in
// should never be used! Only the returned string should be referenced after
// a call to this function!
TString *string_appendData(TString *s, char *buf, size_t len)
{
s->data = realloc(s->data, s->length + len);
if (s->data == NULL) {
TString *r = s;
if (s->refcount > 1) {
assert(s->refcount > 1);
// We have to create a new string, since more than one string is pointing to it.
r = string_copyString(s);
// Then remove our reference to the (old) string.
string_freeString(s);
}

r->data = realloc(r->data, r->length + len);
if (r->data == NULL) {
fatal_error("Could not reallocate string for append data.");
}

memcpy(&s->data[s->length], buf, len);
s->length += len;
return s;
memcpy(&r->data[r->length], buf, len);
r->length += len;
return r;
}

void string_resizeString(TString *s, size_t n)
Expand All @@ -233,14 +285,22 @@ void string_resizeString(TString *s, size_t n)

TString *string_appendString(TString *s, TString *ns)
{
s->data = realloc(s->data, s->length + ns->length);
if (s->data == NULL) {
TString *r = s;
if (s->refcount > 1) {
assert(s->refcount > 1);
// We have to create a new string, since more than one string is pointing to it.
r = string_copyString(s);
// Then remove our reference to the (old) string.
string_freeString(s);
}
r->data = realloc(r->data, r->length + ns->length);
if (r->data == NULL) {
fatal_error("Could not allocate appended TString data.");
}

memcpy(&s->data[s->length], ns->data, ns->length);
s->length += ns->length;
return s;
memcpy(&r->data[r->length], ns->data, ns->length);
r->length += ns->length;
return r;
}

/* NOTE: The caller is responsible for freeing the string created from this function! */
Expand Down Expand Up @@ -486,6 +546,7 @@ const char *string_ensureNullTerminated(TString *s)

// NOTE: The following functions return NEW string objects from existing strings.
// Return the middle part of a string, starting at pos, for count number of bytes.
// NOTE: New string returned, so refcount will be set to 1 on the new string.
TString *string_subString(TString *s, int64_t pos, int64_t count)
{
int64_t newLen = count;
Expand All @@ -505,6 +566,7 @@ TString *string_subString(TString *s, int64_t pos, int64_t count)

TString *string_toLowerCase(TString *s)
{
// Note: returned string will always have a refcount of 1.
TString *r = string_createString(s->length);

for (size_t i = 0; i < s->length; i++) {
Expand All @@ -516,6 +578,7 @@ TString *string_toLowerCase(TString *s)

TString *string_toUpperCase(TString *s)
{
// Note: returned string will always have a refcount of 1.
TString *r = string_createString(s->length);

for (size_t i = 0; i < s->length; i++) {
Expand All @@ -532,6 +595,8 @@ TString *string_toUpperCase(TString *s)
TString *string_quoteInPlace(TString *s)
{
TString *r = string_quote(s);

// Do the modification on the provided string.
free(s->data);
s->data = r->data;
s->length = r->length;
Expand All @@ -542,6 +607,9 @@ TString *string_quoteInPlace(TString *s)
// This may not be the best place for this, because it is not part of
// the implementation of TString, but is just a specific string utility
// function. But there wasn't an obviously better place to put it.
// NOTE: The returned string will have a refcount of 1, so a new string
// is returned, in all cases. This means that we may have to call
// string_freeString() on it when we're done with it.
TString *string_quote(TString *s)
{
TString *r = string_newString();
Expand Down Expand Up @@ -573,6 +641,7 @@ TString *string_quote(TString *s)
i++;
if ((s->data[i] & 0xc0) != 0x80) {
// Garbage data, give up.
// Should this raise an exception?
break;
}
c = (c << 6) | (s->data[i] & 0x3f);
Expand Down Expand Up @@ -606,6 +675,9 @@ TString *string_quote(TString *s)
}
}
string_appendChar(r, '"');
if (s->refcount > 1) {
string_freeString(s);
}
return r;
}

Expand Down Expand Up @@ -642,6 +714,7 @@ size_t string_getLength(TString *s)

TString *string_index(TString *s, size_t index)
{
// Returned string will have a refcount of 1;
TString *r = string_createString(0);

size_t idx = 0;
Expand All @@ -666,6 +739,7 @@ TString *string_index(TString *s, size_t index)
c &= 0x01;
n = 6;
} else {
// Should this exit cnex? What happens in neonx.exe?
fatal_error("Invalid UTF8 in string: %02x", s->data[i]);
}
} else {
Expand Down
1 change: 1 addition & 0 deletions exec/cnex/nstring.h
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@

typedef struct tagTString {
size_t length;
int refcount;
char *data;
} TString;

Expand Down

0 comments on commit 77277c4

Please sign in to comment.