shithub: sl

Download patch

ref: fda389be7556bff1570f891c269df504b2d4850b
parent: df22ae34a79a76ed07090be41fce329b8d3cfe6c
author: Sigrid Solveig Haflínudóttir <sigrid@ftrv.se>
date: Thu Feb 6 19:28:39 EST 2025

allocate less cells

Reduce the number of cells allocated depending on 32 vs 64 bit.
Redefine what "number of words" for cvalue and cprim means.
Make cvalue->data point at the C builtin itself instead of storing
the pointer in data.

--- a/src/cvalues.c
+++ b/src/cvalues.c
@@ -7,15 +7,13 @@
 
 enum {
 	MAX_INL_SIZE = 384,
-	CVALUE_NWORDS = 4,
+	CVALUE_NWORDS = sizeof(cvalue_t)/sizeof(value_t),
 
 	CV_OWNED = 1<<0,
-	CV_PARENT = 1<<1,
 };
 
 #define owned(cv) ((uintptr_t)(cv)->type & CV_OWNED)
-#define hasparent(cv) ((uintptr_t)(cv)->type & CV_PARENT)
-#define isinlined(cv) ((cv)->data == &(cv)->_space[0])
+#define isinlined(cv) ((cv)->data == (cv)->_space)
 
 static void cvalue_init(fltype_t *type, value_t v, void *dest);
 
@@ -40,6 +38,7 @@
 #define SWAP_sf(a, b) (tmp = a, a = b, b = tmp, 1)
 	if(l == 0)
 		return;
+	bool exiting = FL(exiting);
 	do{
 		tmp = lst[n];
 		if(isforwarded((value_t)tmp)){
@@ -50,19 +49,13 @@
 			fltype_t *t = cv_class(tmp);
 			if(t->vtable != nil && t->vtable->finalize != nil)
 				t->vtable->finalize(tagptr(tmp, TAG_CVALUE));
-			if(!isinlined(tmp) && owned(tmp) && !FL(exiting)){
-				memset(cv_data(tmp), 0xbb, cv_len(tmp));
+			if(!isinlined(tmp) && owned(tmp) && !exiting)
 				MEM_FREE(cv_data(tmp));
-			}
 			ndel++;
 		}
 	}while((n < l-ndel) && SWAP_sf(lst[n], lst[n+ndel]));
 
 	FL(nfinalizers) -= ndel;
-#if defined(VERBOSEGC)
-	if(ndel > 0)
-		printf("GC: finalized %d objects\n", ndel);
-#endif
 
 	FL(malloc_pressure) = 0;
 }
@@ -71,13 +64,12 @@
 static size_t
 cv_nwords(cvalue_t *cv)
 {
-	if(isinlined(cv)){
-		size_t n = cv_len(cv);
-		if(n == 0 || cv_isstr(cv))
-			n++;
-		return CVALUE_NWORDS - 1 + NWORDS(n);
-	}
-	return CVALUE_NWORDS;
+	if(!isinlined(cv))
+		return CVALUE_NWORDS;
+	size_t n = cv_len(cv);
+	if(cv_isstr(cv))
+		n++;
+	return CVALUE_NWORDS + NWORDS(n);
 }
 
 void
@@ -92,7 +84,7 @@
 {
 	assert(!ismanaged((uintptr_t)type));
 	assert(sz == type->size);
-	cprim_t *pcp = alloc_words(CPRIM_NWORDS-1+NWORDS(sz));
+	cprim_t *pcp = alloc_words(CPRIM_NWORDS+NWORDS(sz));
 	pcp->type = type;
 	return tagptr(pcp, TAG_CPRIM);
 }
@@ -113,10 +105,10 @@
 	}
 	cvalue_t *pcv;
 	if(sz <= MAX_INL_SIZE){
-		size_t nw = CVALUE_NWORDS - 1 + NWORDS(sz) + (sz == 0 ? 1 : 0);
+		size_t nw = CVALUE_NWORDS + NWORDS(sz);
 		pcv = alloc_words(nw);
 		pcv->type = type;
-		pcv->data = &pcv->_space[0];
+		pcv->data = pcv->_space;
 		if(!nofinalize && type->vtable != nil && type->vtable->finalize != nil)
 			add_finalizer(pcv);
 	}else{
@@ -143,10 +135,9 @@
 // 'parent' is an optional cvalue that this pointer is known to point
 // into; NIL if none.
 value_t
-cvalue_from_ref(fltype_t *type, void *ptr, size_t sz, value_t parent)
+cvalue_from_ref(fltype_t *type, void *ptr, size_t sz)
 {
 	cvalue_t *pcv;
-	value_t cv;
 
 	assert(type != nil);
 	assert(ptr != nil);
@@ -154,12 +145,7 @@
 	pcv->data = ptr;
 	pcv->len = sz;
 	pcv->type = type;
-	if(parent != FL_nil){
-		pcv->type = (fltype_t*)(((uintptr_t)pcv->type) | CV_PARENT);
-		pcv->parent = parent;
-	}
-	cv = tagptr(pcv, TAG_CVALUE);
-	return cv;
+	return tagptr(pcv, TAG_CVALUE);
 }
 
 value_t
@@ -175,7 +161,7 @@
 {
 	if(*str == 0)
 		return FL(the_empty_string);
-	return cvalue_from_ref(FL(stringtype), (char*)str, strlen(str), FL_nil);
+	return cvalue_from_ref(FL(stringtype), (char*)str, strlen(str));
 }
 
 value_t
@@ -611,7 +597,7 @@
 	nv = alloc_words(nw);
 	memcpy(nv, cv, nw*sizeof(value_t));
 	if(isinlined(cv))
-		nv->data = &nv->_space[0];
+		nv->data = nv->_space;
 	ncv = tagptr(nv, TAG_CVALUE);
 	fltype_t *t = cv_class(cv);
 	if(t->vtable != nil && t->vtable->relocate != nil)
@@ -640,12 +626,8 @@
 		ncv->data = MEM_ALLOC(len);
 		memcpy(ncv->data, cv_data(cv), len);
 		cv_autorelease(ncv);
-		if(hasparent(cv)){
-			ncv->type = (fltype_t*)(((uintptr_t)ncv->type) & ~CV_PARENT);
-			ncv->parent = FL_nil;
-		}
 	}else{
-		ncv->data = &ncv->_space[0];
+		ncv->data = ncv->_space;
 	}
 
 	return tagptr(ncv, TAG_CVALUE);
@@ -807,12 +789,10 @@
 cbuiltin(const char *name, builtin_t f)
 {
 	cvalue_t *cv;
-	cv = MEM_CALLOC(CVALUE_NWORDS, sizeof(*cv));
+	cv = MEM_CALLOC(CVALUE_NWORDS-1, sizeof(*cv));
 	assert(cv != nil);
 	cv->type = FL(builtintype);
-	cv->data = &cv->_space[0];
-	cv->len = sizeof(value_t);
-	*(builtin_t*)cv->data = f;
+	cv->cbuiltin = f;
 
 	value_t sym = symbol(name, false);
 	symbol_t *s = ((symbol_t*)ptr(sym));
@@ -1375,6 +1355,6 @@
 	FL(mpinttype)->vtable = &mpint_vtable;
 
 	FL(stringtype) = get_type(symbol_value(FL_stringtypesym));
-	FL(the_empty_string) = cvalue_from_ref(FL(stringtype), (char*)"", 0, FL_nil);
+	FL(the_empty_string) = cvalue_from_ref(FL(stringtype), (char*)"", 0);
 	FL(runestringtype) = get_type(symbol_value(FL_runestringtypesym));
 }
--- a/src/cvalues.h
+++ b/src/cvalues.h
@@ -12,7 +12,7 @@
 value_t cvalue_(fltype_t *type, size_t sz, bool nofinalizer);
 #define cvalue(type, sz) cvalue_(type, sz, false)
 #define cvalue_nofinalizer(type, sz) cvalue_(type, sz, true)
-value_t cvalue_from_ref(fltype_t *type, void *ptr, size_t sz, value_t parent);
+value_t cvalue_from_ref(fltype_t *type, void *ptr, size_t sz);
 value_t cvalue_string(size_t sz);
 value_t cvalue_static_cstring(const char *str);
 value_t string_from_cstrn(char *str, size_t n);
--- a/src/flisp.c
+++ b/src/flisp.c
@@ -297,9 +297,13 @@
 {
 	value_t *first;
 
-	assert(n > 0);
-	if(n & 1) // only allocate multiples of 2 words
+#if defined(BITS64)
+	if(fl_unlikely(n < 2))
+		n = 2;
+#else
+	if(n & 1)
 		n++;
+#endif
 	if(fl_unlikely((value_t*)FL(curheap) > (value_t*)FL(lim)+2-n)){
 		fl_gc(0);
 		while(fl_unlikely((value_t*)FL(curheap) > ((value_t*)FL(lim))+2-n))
@@ -364,7 +368,7 @@
 			*pcdr = nc = tagptr((cons_t*)FL(curheap), TAG_CONS);
 			FL(curheap) += sizeof(cons_t);
 			cdr_(v) = nc;
-			car_(nc) = relocate(a);
+			car_(nc) = ismanaged(a) ? relocate(a) : a;
 			pcdr = &cdr_(nc);
 			v = d;
 		}while(iscons(v));
@@ -379,30 +383,6 @@
 
 	if(t == TAG_CVALUE)
 		return cvalue_relocate(v);
-	if(t == TAG_CPRIM){
-		cprim_t *pcp = ptr(v);
-		size_t nw = CPRIM_NWORDS-1+NWORDS(cp_class(pcp)->size);
-		cprim_t *ncp = alloc_words(nw);
-		while(nw--)
-			((value_t*)ncp)[nw] = ((value_t*)pcp)[nw];
-		nc = tagptr(ncp, TAG_CPRIM);
-		forward(v, nc);
-		return nc;
-	}
-	if(t == TAG_FUNCTION){
-		function_t *fn = ptr(v);
-		function_t *nfn = alloc_words(4);
-		nfn->bcode = fn->bcode;
-		nfn->vals = fn->vals;
-		nc = tagptr(nfn, TAG_FUNCTION);
-		forward(v, nc);
-		nfn->env = relocate(fn->env);
-		nfn->vals = relocate(nfn->vals);
-		nfn->bcode = relocate(nfn->bcode);
-		assert(!ismanaged(fn->name));
-		nfn->name = fn->name;
-		return nc;
-	}
 	if(t == TAG_VECTOR){
 		// N.B.: 0-length vectors secretly have space for a first element
 		size_t i, sz = vector_size(v);
@@ -423,6 +403,20 @@
 		}
 		return nc;
 	}
+	if(t == TAG_FUNCTION){
+		function_t *fn = ptr(v);
+		function_t *nfn = alloc_words(4);
+		nfn->bcode = fn->bcode;
+		nfn->vals = fn->vals;
+		nc = tagptr(nfn, TAG_FUNCTION);
+		forward(v, nc);
+		nfn->env = relocate(fn->env);
+		nfn->vals = relocate(nfn->vals);
+		nfn->bcode = relocate(nfn->bcode);
+		assert(!ismanaged(fn->name));
+		nfn->name = fn->name;
+		return nc;
+	}
 	if(t == TAG_SYM){
 		gensym_t *gs = ptr(v);
 		gensym_t *ng = alloc_words(sizeof(gensym_t)/sizeof(value_t));
@@ -435,6 +429,16 @@
 			ng->binding = relocate(ng->binding);
 		return nc;
 	}
+	if(t == TAG_CPRIM){
+		cprim_t *pcp = ptr(v);
+		size_t nw = CPRIM_NWORDS+NWORDS(cp_class(pcp)->size);
+		cprim_t *ncp = alloc_words(nw);
+		while(nw--)
+			((value_t*)ncp)[nw] = ((value_t*)pcp)[nw];
+		nc = tagptr(ncp, TAG_CPRIM);
+		forward(v, nc);
+		return nc;
+	}
 	return v;
 }
 
@@ -560,7 +564,7 @@
 	uint32_t saveSP = FL(sp);
 	value_t v;
 	if(iscbuiltin(f))
-		v = ((builtin_t*)ptr(f))[3](&FL(stack)[FL(sp)-n], n);
+		v = ((cvalue_t*)ptr(f))->cbuiltin(&FL(stack)[FL(sp)-n], n);
 	else if(isfunction(f))
 		v = apply_cl(n);
 	else if(fl_likely(isbuiltin(f))){
--- a/src/flisp.h
+++ b/src/flisp.h
@@ -293,12 +293,12 @@
 
 typedef struct {
 	fltype_t *type;
-	void *data;
-	size_t len;			// length of *data in bytes
 	union {
-		value_t parent;	// optional
-		uint8_t _space[1];	// variable size
+		void *data;
+		builtin_t cbuiltin;
 	};
+	size_t len; // length of *data in bytes
+	uint8_t _space[]; // variable size
 }fl_aligned(8) cvalue_t;
 
 typedef struct {
@@ -313,7 +313,7 @@
 	value_t name;
 }fl_aligned(8) function_t;
 
-#define CPRIM_NWORDS 2
+#define CPRIM_NWORDS sizeof(cprim_t)/sizeof(value_t)
 #define cv_class(cv) ((fltype_t*)(((uintptr_t)((cvalue_t*)cv)->type)&~(uintptr_t)3))
 #define cv_len(cv) (((cvalue_t*)(cv))->len)
 #define cv_type(cv) (cv_class(cv)->type)
--- a/src/iostream.c
+++ b/src/iostream.c
@@ -416,7 +416,7 @@
 		if(n == 0)
 			return FL(the_empty_string);
 		b[n] = '\0';
-		str = cvalue_from_ref(FL(stringtype), b, n, FL_nil);
+		str = cvalue_from_ref(FL(stringtype), b, n);
 		cv_autorelease(ptr(str));
 	}
 	return str;
@@ -447,7 +447,7 @@
 	FL_instrsym = symbol("*input-stream*", false);
 	FL_outstrsym = symbol("*output-stream*", false);
 	FL(iostreamtype) = define_opaque_type(FL_iostreamsym, sizeof(ios_t), &iostream_vtable, nil);
-	set(symbol("*stdout*", false), cvalue_from_ref(FL(iostreamtype), ios_stdout, sizeof(ios_t), FL_nil));
-	set(symbol("*stderr*", false), cvalue_from_ref(FL(iostreamtype), ios_stderr, sizeof(ios_t), FL_nil));
-	set(symbol("*stdin*", false), cvalue_from_ref(FL(iostreamtype), ios_stdin, sizeof(ios_t), FL_nil));
+	set(symbol("*stdout*", false), cvalue_from_ref(FL(iostreamtype), ios_stdout, sizeof(ios_t)));
+	set(symbol("*stderr*", false), cvalue_from_ref(FL(iostreamtype), ios_stderr, sizeof(ios_t)));
+	set(symbol("*stdin*", false), cvalue_from_ref(FL(iostreamtype), ios_stdin, sizeof(ios_t)));
 }
--- a/src/vm.inc
+++ b/src/vm.inc
@@ -86,7 +86,7 @@
 		}
 	}else if(fl_likely(iscbuiltin(func))){
 		s = FL(sp) - n;
-		v = (((builtin_t*)ptr(func))[3])(&FL(stack)[s], n);
+		v = ((cvalue_t*)ptr(func))->cbuiltin(&FL(stack)[s], n);
 		FL(sp) = s;
 		FL(stack)[s-1] = v;
 		NEXT_OP;