Revision 4d745aa2f670491f3a93a7ed534dad367a7d7b84 authored by David Anthoff on 07 March 2020, 14:17:05 UTC, committed by GitHub on 07 March 2020, 14:17:05 UTC
* Add devcontainer and dockerfile * Add VS Code CPP extension * Use Julia base image for devcontainer * Add VS Code make task * Fix typo * Remove .vscode/tasks.json
1 parent a247038
cvalues.c
#ifdef _P64
#define NWORDS(sz) (((sz)+7)>>3)
#else
#define NWORDS(sz) (((sz)+3)>>2)
#endif
struct prim_int16{ char a; int16_t i; };
struct prim_int32{ char a; int32_t i; };
struct prim_int64{ char a; int64_t i; };
struct prim_ptr{ char a; void *i; };
// compute struct field alignment required for primitives
static const int ALIGN2 = sizeof(struct prim_int16) - 2;
static const int ALIGN4 = sizeof(struct prim_int32) - 4;
static const int ALIGN8 = sizeof(struct prim_int64) - 8;
static const int ALIGNPTR = sizeof(struct prim_ptr) - sizeof(void*);
static void cvalue_init(fl_context_t *fl_ctx, fltype_t *type, value_t v, void *dest);
// cvalues-specific builtins
value_t cvalue_new(fl_context_t *fl_ctx, value_t *args, uint32_t nargs);
value_t cvalue_sizeof(fl_context_t *fl_ctx, value_t *args, uint32_t nargs);
value_t cvalue_typeof(fl_context_t *fl_ctx, value_t *args, uint32_t nargs);
// trigger unconditional GC after this many bytes are allocated
#define ALLOC_LIMIT_TRIGGER 67108864
void add_finalizer(fl_context_t *fl_ctx, cvalue_t *cv)
{
if (fl_ctx->nfinalizers == fl_ctx->maxfinalizers) {
size_t nn = (fl_ctx->maxfinalizers==0 ? 256 : fl_ctx->maxfinalizers*2);
cvalue_t **temp = (cvalue_t**)realloc(fl_ctx->Finalizers, nn*sizeof(value_t));
if (temp == NULL)
lerror(fl_ctx, fl_ctx->OutOfMemoryError, "out of memory");
fl_ctx->Finalizers = temp;
fl_ctx->maxfinalizers = nn;
}
fl_ctx->Finalizers[fl_ctx->nfinalizers++] = cv;
}
// remove dead objects from finalization list in-place
static void sweep_finalizers(fl_context_t *fl_ctx)
{
cvalue_t **lst = fl_ctx->Finalizers;
size_t n=0, ndel=0, l=fl_ctx->nfinalizers;
cvalue_t *tmp;
#define SWAP_sf(a,b) (tmp=a,a=b,b=tmp,1)
if (l == 0)
return;
do {
tmp = lst[n];
if (isforwarded((value_t)tmp)) {
// object is alive
lst[n] = (cvalue_t*)ptr(forwardloc((value_t)tmp));
n++;
}
else {
fltype_t *t = cv_class(tmp);
if (t->vtable != NULL && t->vtable->finalize != NULL) {
t->vtable->finalize(fl_ctx, tagptr(tmp, TAG_CVALUE));
}
if (!isinlined(tmp) && owned(tmp)) {
#ifdef DEBUG
memset(cv_data(tmp), 0xbb, cv_len(tmp));
#endif
free(cv_data(tmp));
}
ndel++;
}
} while ((n < l-ndel) && SWAP_sf(lst[n],lst[n+ndel]));
fl_ctx->nfinalizers -= ndel;
#ifdef VERBOSEGC
if (ndel > 0)
printf("GC: finalized %d objects\n", ndel);
#endif
fl_ctx->malloc_pressure = 0;
}
// compute the size of the metadata object for a cvalue
static size_t cv_nwords(fl_context_t *fl_ctx, cvalue_t *cv)
{
if (isinlined(cv)) {
size_t n = cv_len(cv);
if (n==0 || cv_isstr(fl_ctx, cv))
n++;
return CVALUE_NWORDS - 1 + NWORDS(n);
}
return CVALUE_NWORDS;
}
static void autorelease(fl_context_t *fl_ctx, cvalue_t *cv)
{
cv->type = (fltype_t*)(((uintptr_t)cv->type) | CV_OWNED_BIT);
add_finalizer(fl_ctx, cv);
}
void cv_autorelease(fl_context_t *fl_ctx, cvalue_t *cv)
{
autorelease(fl_ctx, cv);
}
static value_t cprim(fl_context_t *fl_ctx, fltype_t *type, size_t sz)
{
cprim_t *pcp = (cprim_t*)alloc_words(fl_ctx, CPRIM_NWORDS-1+NWORDS(sz));
pcp->type = type;
return tagptr(pcp, TAG_CPRIM);
}
value_t cvalue(fl_context_t *fl_ctx, fltype_t *type, size_t sz)
{
cvalue_t *pcv;
int str=0;
if (valid_numtype(type->numtype)) {
return cprim(fl_ctx, type, sz);
}
if (type->eltype == fl_ctx->bytetype) {
if (sz == 0)
return symbol_value(fl_ctx->emptystringsym);
sz++;
str=1;
}
if (sz <= MAX_INL_SIZE) {
size_t nw = CVALUE_NWORDS - 1 + NWORDS(sz) + (sz==0 ? 1 : 0);
pcv = (cvalue_t*)alloc_words(fl_ctx, nw);
pcv->type = type;
pcv->data = &pcv->_space[0];
if (type->vtable != NULL && type->vtable->finalize != NULL)
add_finalizer(fl_ctx, pcv);
}
else {
if (fl_ctx->malloc_pressure > ALLOC_LIMIT_TRIGGER)
gc(fl_ctx, 0);
pcv = (cvalue_t*)alloc_words(fl_ctx, CVALUE_NWORDS);
pcv->type = type;
pcv->data = malloc(sz);
// TODO: if pcv->data == NULL
autorelease(fl_ctx, pcv);
fl_ctx->malloc_pressure += sz;
}
if (str) {
sz--;
((char*)pcv->data)[sz] = '\0';
}
pcv->len = sz;
return tagptr(pcv, TAG_CVALUE);
}
value_t cvalue_from_data(fl_context_t *fl_ctx, fltype_t *type, void *data, size_t sz)
{
value_t cv;
cv = cvalue(fl_ctx, type, sz);
memcpy(cptr(cv), data, sz);
return cv;
}
// this effectively dereferences a pointer
// just like *p in C, it only removes a level of indirection from the type,
// it doesn't copy any data.
// this method of creating a cvalue only allocates metadata.
// ptr is user-managed; we don't autorelease it unless the
// user explicitly calls (autorelease ) on the result of this function.
// 'parent' is an optional cvalue that this pointer is known to point
// into; fl_ctx->NIL if none.
value_t cvalue_from_ref(fl_context_t *fl_ctx, fltype_t *type, void *ptr, size_t sz, value_t parent)
{
cvalue_t *pcv;
value_t cv;
pcv = (cvalue_t*)alloc_words(fl_ctx, CVALUE_NWORDS);
pcv->data = ptr;
pcv->len = sz;
pcv->type = type;
if (parent != fl_ctx->NIL) {
pcv->type = (fltype_t*)(((uintptr_t)pcv->type) | CV_PARENT_BIT);
pcv->parent = parent;
}
cv = tagptr(pcv, TAG_CVALUE);
return cv;
}
value_t cvalue_string(fl_context_t *fl_ctx, size_t sz)
{
return cvalue(fl_ctx, fl_ctx->stringtype, sz);
}
value_t cvalue_static_cstrn(fl_context_t *fl_ctx, const char *str, size_t n)
{
return cvalue_from_ref(fl_ctx, fl_ctx->stringtype, (char*)str, n, fl_ctx->NIL);
}
value_t cvalue_static_cstring(fl_context_t *fl_ctx, const char *str)
{
return cvalue_static_cstrn(fl_ctx, str, strlen(str));
}
value_t string_from_cstrn(fl_context_t *fl_ctx, char *str, size_t n)
{
value_t v = cvalue_string(fl_ctx, n);
memcpy(cvalue_data(v), str, n);
return v;
}
value_t string_from_cstr(fl_context_t *fl_ctx, char *str)
{
return string_from_cstrn(fl_ctx, str, strlen(str));
}
int fl_isstring(fl_context_t *fl_ctx, value_t v)
{
return (iscvalue(v) && cv_isstr(fl_ctx, (cvalue_t*)ptr(v)));
}
// convert to malloc representation (fixed address)
void cv_pin(fl_context_t *fl_ctx, cvalue_t *cv)
{
if (!isinlined(cv))
return;
size_t sz = cv_len(cv);
if (cv_isstr(fl_ctx, cv)) sz++;
void *data = malloc(sz);
// TODO: if data == NULL
memcpy(data, cv_data(cv), sz);
cv->data = data;
autorelease(fl_ctx, cv);
}
#define num_init(ctype, cnvt, tag) \
static int cvalue_##ctype##_init(fl_context_t *fl_ctx, fltype_t *type, \
value_t arg, void *dest) \
{ \
fl_##ctype##_t n=0; \
(void)type; \
if (isfixnum(arg)) { \
n = numval(arg); \
} \
else if (iscprim(arg)) { \
cprim_t *cp = (cprim_t*)ptr(arg); \
void *p = cp_data(cp); \
n = (fl_##ctype##_t)conv_to_##cnvt(p, cp_numtype(cp)); \
} \
else { \
return 1; \
} \
memcpy(jl_assume_aligned(dest, sizeof(void*)), &n, \
sizeof(fl_##ctype##_t)); \
return 0; \
}
num_init(int8, int32, T_INT8)
num_init(uint8, uint32, T_UINT8)
num_init(int16, int32, T_INT16)
num_init(uint16, uint32, T_UINT16)
num_init(int32, int32, T_INT32)
num_init(uint32, uint32, T_UINT32)
num_init(int64, int64, T_INT64)
num_init(uint64, uint64, T_UINT64)
num_init(float, double, T_FLOAT)
num_init(double, double, T_DOUBLE)
#define num_ctor_init(typenam, ctype, tag) \
value_t cvalue_##typenam(fl_context_t *fl_ctx, value_t *args, uint32_t nargs) \
{ \
if (nargs==0) { PUSH(fl_ctx, fixnum(0)); args = &fl_ctx->Stack[fl_ctx->SP-1]; } \
value_t cp = cprim(fl_ctx, fl_ctx->typenam##type, sizeof(fl_##ctype##_t)); \
if (cvalue_##ctype##_init(fl_ctx, fl_ctx->typenam##type, \
args[0], cp_data((cprim_t*)ptr(cp)))) \
type_error(fl_ctx, #typenam, "number", args[0]); \
return cp; \
}
#define num_ctor_ctor(typenam, ctype, tag) \
value_t mk_##typenam(fl_context_t *fl_ctx, fl_##ctype##_t n) \
{ \
value_t cp = cprim(fl_ctx, fl_ctx->typenam##type, sizeof(fl_##ctype##_t)); \
*(fl_##ctype##_t*)cp_data((cprim_t*)ptr(cp)) = n; \
return cp; \
}
#define num_ctor(typenam, ctype, tag) \
num_ctor_init(typenam, ctype, tag) \
num_ctor_ctor(typenam, ctype, tag)
num_ctor(int8, int8, T_INT8)
num_ctor(uint8, uint8, T_UINT8)
num_ctor(int16, int16, T_INT16)
num_ctor(uint16, uint16, T_UINT16)
num_ctor(int32, int32, T_INT32)
num_ctor(uint32, uint32, T_UINT32)
num_ctor(int64, int64, T_INT64)
num_ctor(uint64, uint64, T_UINT64)
num_ctor(byte, uint8, T_UINT8)
num_ctor(wchar, int32, T_INT32)
#ifdef _P64
num_ctor(ptrdiff, int64, T_INT64)
num_ctor(size, uint64, T_UINT64)
#else
num_ctor(ptrdiff, int32, T_INT32)
num_ctor(size, uint32, T_UINT32)
#endif
num_ctor(float, float, T_FLOAT)
num_ctor(double, double, T_DOUBLE)
value_t size_wrap(fl_context_t *fl_ctx, size_t sz)
{
if (fits_fixnum(sz))
return fixnum(sz);
assert(sizeof(void*) == sizeof(size_t));
return mk_size(fl_ctx, sz);
}
size_t tosize(fl_context_t *fl_ctx, value_t n, const char *fname)
{
if (isfixnum(n))
return numval(n);
if (iscprim(n)) {
cprim_t *cp = (cprim_t*)ptr(n);
return conv_to_size(cp_data(cp), cp_numtype(cp));
}
type_error(fl_ctx, fname, "number", n);
return 0;
}
static int isarray(value_t v)
{
return iscvalue(v) && cv_class((cvalue_t*)ptr(v))->eltype != NULL;
}
static size_t predict_arraylen(fl_context_t *fl_ctx, value_t arg)
{
if (isvector(arg))
return vector_size(arg);
else if (iscons(arg))
return llength(arg);
else if (arg == fl_ctx->NIL)
return 0;
if (isarray(arg))
return cvalue_arraylen(arg);
return 1;
}
static int cvalue_array_init(fl_context_t *fl_ctx, fltype_t *ft, value_t arg, void *dest)
{
value_t type = ft->type;
size_t elsize, i, cnt, sz;
fltype_t *eltype = ft->eltype;
elsize = ft->elsz;
cnt = predict_arraylen(fl_ctx, arg);
if (iscons(cdr_(cdr_(type)))) {
size_t tc = tosize(fl_ctx, car_(cdr_(cdr_(type))), "array");
if (tc != cnt)
lerror(fl_ctx, fl_ctx->ArgError, "array: size mismatch");
}
sz = elsize * cnt;
if (isvector(arg)) {
for(i=0; i < cnt; i++) {
cvalue_init(fl_ctx, eltype, vector_elt(arg,i), dest);
dest = (char *)dest + elsize;
}
return 0;
}
else if (iscons(arg) || arg==fl_ctx->NIL) {
i = 0;
while (iscons(arg)) {
if (i == cnt) { i++; break; } // trigger error
cvalue_init(fl_ctx, eltype, car_(arg), dest);
i++;
dest = (char *)dest + elsize;
arg = cdr_(arg);
}
if (i != cnt)
lerror(fl_ctx, fl_ctx->ArgError, "array: size mismatch");
return 0;
}
else if (iscvalue(arg)) {
cvalue_t *cv = (cvalue_t*)ptr(arg);
if (isarray(arg)) {
fltype_t *aet = cv_class(cv)->eltype;
if (aet == eltype) {
if (cv_len(cv) == sz)
memcpy(dest, cv_data(cv), sz);
else
lerror(fl_ctx, fl_ctx->ArgError, "array: size mismatch");
return 0;
}
else {
// TODO: initialize array from different type elements
lerror(fl_ctx, fl_ctx->ArgError, "array: element type mismatch");
}
}
}
if (cnt == 1)
cvalue_init(fl_ctx, eltype, arg, dest);
else
type_error(fl_ctx, "array", "sequence", arg);
return 0;
}
value_t cvalue_array(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
{
size_t elsize, cnt, sz, i;
value_t arg;
if (nargs < 1)
argcount(fl_ctx, "array", nargs, 1);
cnt = nargs - 1;
fltype_t *type = get_array_type(fl_ctx, args[0]);
elsize = type->elsz;
sz = elsize * cnt;
value_t cv = cvalue(fl_ctx, type, sz);
char *dest = (char*)cv_data((cvalue_t*)ptr(cv));
FOR_ARGS(i,1,arg,args) {
cvalue_init(fl_ctx, type->eltype, arg, dest);
dest += elsize;
}
return cv;
}
// NOTE: v must be an array
size_t cvalue_arraylen(value_t v)
{
cvalue_t *cv = (cvalue_t*)ptr(v);
return cv_len(cv)/(cv_class(cv)->elsz);
}
// *palign is an output argument giving the alignment required by type
size_t ctype_sizeof(fl_context_t *fl_ctx, value_t type, int *palign)
{
if (type == fl_ctx->int8sym || type == fl_ctx->uint8sym || type == fl_ctx->bytesym) {
*palign = 1;
return 1;
}
if (type == fl_ctx->int16sym || type == fl_ctx->uint16sym) {
*palign = ALIGN2;
return 2;
}
if (type == fl_ctx->int32sym || type == fl_ctx->uint32sym || type == fl_ctx->wcharsym ||
type == fl_ctx->floatsym) {
*palign = ALIGN4;
return 4;
}
if (type == fl_ctx->int64sym || type == fl_ctx->uint64sym || type == fl_ctx->doublesym) {
*palign = ALIGN8;
return 8;
}
if (type == fl_ctx->ptrdiffsym || type == fl_ctx->sizesym) {
#ifdef _P64
*palign = ALIGN8;
return 8;
#else
*palign = ALIGN4;
return 4;
#endif
}
if (iscons(type)) {
value_t hed = car_(type);
if (hed == fl_ctx->pointersym || hed == fl_ctx->cfunctionsym) {
*palign = ALIGNPTR;
return sizeof(void*);
}
if (hed == fl_ctx->arraysym) {
value_t t = car(fl_ctx, cdr_(type));
if (!iscons(cdr_(cdr_(type))))
lerror(fl_ctx, fl_ctx->ArgError, "sizeof: incomplete type");
value_t n = car_(cdr_(cdr_(type)));
size_t sz = tosize(fl_ctx, n, "sizeof");
return sz * ctype_sizeof(fl_ctx, t, palign);
}
}
lerror(fl_ctx, fl_ctx->ArgError, "sizeof: invalid c type");
return 0;
}
// get pointer and size for any plain-old-data value
void to_sized_ptr(fl_context_t *fl_ctx, value_t v, const char *fname, char **pdata, size_t *psz)
{
if (iscvalue(v)) {
cvalue_t *pcv = (cvalue_t*)ptr(v);
ios_t *x = value2c(ios_t*,v);
if (cv_class(pcv) == fl_ctx->iostreamtype && (x->bm == bm_mem)) {
*pdata = x->buf;
*psz = (size_t)x->size;
return;
}
else if (cv_isPOD(pcv)) {
*pdata = (char*)cv_data(pcv);
*psz = cv_len(pcv);
return;
}
}
else if (iscprim(v)) {
cprim_t *pcp = (cprim_t*)ptr(v);
*pdata = cp_data(pcp);
*psz = cp_class(pcp)->size;
return;
}
type_error(fl_ctx, fname, "plain-old-data", v);
}
value_t cvalue_sizeof(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
{
argcount(fl_ctx, "sizeof", nargs, 1);
if (issymbol(args[0]) || iscons(args[0])) {
int a;
return size_wrap(fl_ctx, ctype_sizeof(fl_ctx, args[0], &a));
}
size_t n; char *data;
to_sized_ptr(fl_ctx, args[0], "sizeof", &data, &n);
return size_wrap(fl_ctx, n);
}
value_t cvalue_typeof(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
{
argcount(fl_ctx, "typeof", nargs, 1);
switch(tag(args[0])) {
case TAG_CONS: return fl_ctx->pairsym;
case TAG_NUM1:
case TAG_NUM: return fl_ctx->fixnumsym;
case TAG_SYM: return fl_ctx->symbolsym;
case TAG_VECTOR: return fl_ctx->vectorsym;
case TAG_FUNCTION:
if (args[0] == fl_ctx->T || args[0] == fl_ctx->F)
return fl_ctx->booleansym;
if (args[0] == fl_ctx->NIL)
return fl_ctx->nullsym;
if (args[0] == fl_ctx->FL_EOF)
return symbol(fl_ctx, "eof-object");
if (isbuiltin(args[0]))
return fl_ctx->builtinsym;
return fl_ctx->FUNCTION;
}
return cv_type((cvalue_t*)ptr(args[0]));
}
static value_t cvalue_relocate(fl_context_t *fl_ctx, value_t v)
{
size_t nw;
cvalue_t *cv = (cvalue_t*)ptr(v);
cvalue_t *nv;
value_t ncv;
nw = cv_nwords(fl_ctx, cv);
nv = (cvalue_t*)alloc_words(fl_ctx, nw);
memcpy(nv, cv, nw*sizeof(value_t));
if (isinlined(cv))
nv->data = &nv->_space[0];
ncv = tagptr(nv, TAG_CVALUE);
fltype_t *t = cv_class(cv);
if (t->vtable != NULL && t->vtable->relocate != NULL)
t->vtable->relocate(fl_ctx, v, ncv);
forward(v, ncv);
return ncv;
}
value_t cvalue_copy(fl_context_t *fl_ctx, value_t v)
{
assert(iscvalue(v));
PUSH(fl_ctx, v);
cvalue_t *cv = (cvalue_t*)ptr(v);
size_t nw = cv_nwords(fl_ctx, cv);
cvalue_t *ncv = (cvalue_t*)alloc_words(fl_ctx, nw);
v = POP(fl_ctx); cv = (cvalue_t*)ptr(v);
memcpy(ncv, cv, nw * sizeof(value_t));
if (!isinlined(cv)) {
size_t len = cv_len(cv);
if (cv_isstr(fl_ctx, cv)) len++;
ncv->data = malloc(len);
// TODO: if ncv->data == NULL
memcpy(ncv->data, cv_data(cv), len);
autorelease(fl_ctx, ncv);
if (hasparent(cv)) {
ncv->type = (fltype_t*)(((uintptr_t)ncv->type) & ~CV_PARENT_BIT);
ncv->parent = fl_ctx->NIL;
}
}
else {
ncv->data = &ncv->_space[0];
}
return tagptr(ncv, TAG_CVALUE);
}
value_t fl_copy(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
{
argcount(fl_ctx, "copy", nargs, 1);
if (iscons(args[0]) || isvector(args[0]))
lerror(fl_ctx, fl_ctx->ArgError, "copy: argument must be a leaf atom");
if (!iscvalue(args[0]))
return args[0];
if (!cv_isPOD((cvalue_t*)ptr(args[0])))
lerror(fl_ctx, fl_ctx->ArgError, "copy: argument must be a plain-old-data type");
return cvalue_copy(fl_ctx, args[0]);
}
value_t fl_podp(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
{
argcount(fl_ctx, "plain-old-data?", nargs, 1);
return (iscprim(args[0]) ||
(iscvalue(args[0]) && cv_isPOD((cvalue_t*)ptr(args[0])))) ?
fl_ctx->T : fl_ctx->F;
}
static void cvalue_init(fl_context_t *fl_ctx, fltype_t *type, value_t v, void *dest)
{
cvinitfunc_t f=type->init;
if (f == NULL)
lerror(fl_ctx, fl_ctx->ArgError, "c-value: invalid c type");
f(fl_ctx, type, v, dest);
}
static numerictype_t sym_to_numtype(fl_context_t *fl_ctx, value_t type)
{
if (type == fl_ctx->int8sym)
return T_INT8;
else if (type == fl_ctx->uint8sym || type == fl_ctx->bytesym)
return T_UINT8;
else if (type == fl_ctx->int16sym)
return T_INT16;
else if (type == fl_ctx->uint16sym)
return T_UINT16;
#ifdef _P64
else if (type == fl_ctx->int32sym || type == fl_ctx->wcharsym)
#else
else if (type == fl_ctx->int32sym || type == fl_ctx->wcharsym || type == fl_ctx->ptrdiffsym)
#endif
return T_INT32;
#ifdef _P64
else if (type == fl_ctx->uint32sym)
#else
else if (type == fl_ctx->uint32sym || type == fl_ctx->sizesym)
#endif
return T_UINT32;
#ifdef _P64
else if (type == fl_ctx->int64sym || type == fl_ctx->ptrdiffsym)
#else
else if (type == fl_ctx->int64sym)
#endif
return T_INT64;
#ifdef _P64
else if (type == fl_ctx->uint64sym || type == fl_ctx->sizesym)
#else
else if (type == fl_ctx->uint64sym)
#endif
return T_UINT64;
else if (type == fl_ctx->floatsym)
return T_FLOAT;
else if (type == fl_ctx->doublesym)
return T_DOUBLE;
return (numerictype_t)N_NUMTYPES;
}
// (new type . args)
// this provides (1) a way to allocate values with a shared type for
// efficiency, (2) a uniform interface for allocating cvalues of any
// type, including user-defined.
value_t cvalue_new(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
{
if (nargs < 1 || nargs > 2)
argcount(fl_ctx, "c-value", nargs, 2);
value_t type = args[0];
fltype_t *ft = get_type(fl_ctx, type);
value_t cv;
if (ft->eltype != NULL) {
// special case to handle incomplete array types bla[]
size_t elsz = ft->elsz;
size_t cnt;
if (iscons(cdr_(cdr_(type))))
cnt = tosize(fl_ctx, car_(cdr_(cdr_(type))), "array");
else if (nargs == 2)
cnt = predict_arraylen(fl_ctx, args[1]);
else
cnt = 0;
cv = cvalue(fl_ctx, ft, elsz * cnt);
if (nargs == 2)
cvalue_array_init(fl_ctx, ft, args[1], cv_data((cvalue_t*)ptr(cv)));
}
else {
cv = cvalue(fl_ctx, ft, ft->size);
if (nargs == 2)
cvalue_init(fl_ctx, ft, args[1], cptr(cv));
}
return cv;
}
// NOTE: this only compares lexicographically; it ignores numeric formats
value_t cvalue_compare(value_t a, value_t b)
{
cvalue_t *ca = (cvalue_t*)ptr(a);
cvalue_t *cb = (cvalue_t*)ptr(b);
char *adata = (char*)cv_data(ca);
char *bdata = (char*)cv_data(cb);
size_t asz = cv_len(ca);
size_t bsz = cv_len(cb);
size_t minsz = asz < bsz ? asz : bsz;
int diff = memcmp(adata, bdata, minsz);
if (diff == 0) {
if (asz > bsz)
return fixnum(1);
else if (asz < bsz)
return fixnum(-1);
}
return fixnum(diff);
}
static void check_addr_args(fl_context_t *fl_ctx, const char *fname, value_t arr,
value_t ind, char **data, size_t *index)
{
size_t numel;
cvalue_t *cv = (cvalue_t*)ptr(arr);
*data = (char*)cv_data(cv);
numel = cv_len(cv)/(cv_class(cv)->elsz);
*index = tosize(fl_ctx, ind, fname);
if (*index >= numel)
bounds_error(fl_ctx, fname, arr, ind);
}
static value_t cvalue_array_aref(fl_context_t *fl_ctx, value_t *args)
{
char *data; size_t index;
fltype_t *eltype = cv_class((cvalue_t*)ptr(args[0]))->eltype;
value_t el = 0;
numerictype_t nt = eltype->numtype;
if (nt >= T_INT32)
el = cvalue(fl_ctx, eltype, eltype->size);
check_addr_args(fl_ctx, "aref", args[0], args[1], &data, &index);
if (nt < T_INT32) {
if (nt == T_INT8)
return fixnum((int8_t)data[index]);
else if (nt == T_UINT8)
return fixnum((uint8_t)data[index]);
else if (nt == T_INT16)
return fixnum(((int16_t*)data)[index]);
return fixnum(((uint16_t*)data)[index]);
}
char *dest = (char*)cptr(el);
size_t sz = eltype->size;
if (sz == 1)
*dest = data[index];
else if (sz == 2)
*(int16_t*)dest = ((int16_t*)data)[index];
else if (sz == 4)
*(int32_t*)dest = ((int32_t*)data)[index];
else if (sz == 8)
*(int64_t*)dest = ((int64_t*)data)[index];
else
memcpy(dest, data + index*sz, sz);
return el;
}
static value_t cvalue_array_aset(fl_context_t *fl_ctx, value_t *args)
{
char *data; size_t index;
fltype_t *eltype = cv_class((cvalue_t*)ptr(args[0]))->eltype;
check_addr_args(fl_ctx, "aset!", args[0], args[1], &data, &index);
char *dest = data + index*eltype->size;
cvalue_init(fl_ctx, eltype, args[2], dest);
return args[2];
}
value_t fl_builtin(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
{
argcount(fl_ctx, "builtin", nargs, 1);
symbol_t *name = tosymbol(fl_ctx, args[0], "builtin");
cvalue_t *cv;
if (ismanaged(fl_ctx, args[0]) || (cv=(cvalue_t*)name->dlcache) == NULL) {
lerrorf(fl_ctx, fl_ctx->ArgError, "builtin: function %s not found", name->name);
}
return tagptr(cv, TAG_CVALUE);
}
value_t cbuiltin(fl_context_t *fl_ctx, const char *name, builtin_t f)
{
cvalue_t *cv = (cvalue_t*)malloc(CVALUE_NWORDS * sizeof(value_t));
// TODO: if cv->data == NULL
cv->type = fl_ctx->builtintype;
cv->data = &cv->_space[0];
cv->len = sizeof(value_t);
*(void**)cv->data = (void*)(uintptr_t)f;
value_t sym = symbol(fl_ctx, name);
((symbol_t*)ptr(sym))->dlcache = cv;
ptrhash_put(&fl_ctx->reverse_dlsym_lookup_table, cv, (void*)sym);
return tagptr(cv, TAG_CVALUE);
}
static value_t fl_logand(fl_context_t *fl_ctx, value_t *args, uint32_t nargs);
static value_t fl_logior(fl_context_t *fl_ctx, value_t *args, uint32_t nargs);
static value_t fl_logxor(fl_context_t *fl_ctx, value_t *args, uint32_t nargs);
static value_t fl_lognot(fl_context_t *fl_ctx, value_t *args, uint32_t nargs);
static value_t fl_ash(fl_context_t *fl_ctx, value_t *args, uint32_t nargs);
static const builtinspec_t cvalues_builtin_info[] = {
{ "c-value", cvalue_new },
{ "typeof", cvalue_typeof },
{ "sizeof", cvalue_sizeof },
{ "builtin", fl_builtin },
{ "copy", fl_copy },
{ "plain-old-data?", fl_podp },
{ "logand", fl_logand },
{ "logior", fl_logior },
{ "logxor", fl_logxor },
{ "lognot", fl_lognot },
{ "ash", fl_ash },
// todo: autorelease
{ NULL, NULL }
};
#define cv_intern(fl_ctx, tok) fl_ctx->tok##sym = symbol(fl_ctx, #tok)
#define ctor_cv_intern(fl_ctx, tok) \
cv_intern(fl_ctx, tok); \
set(fl_ctx->tok##sym, cbuiltin(fl_ctx, #tok, cvalue_##tok))
#define mk_primtype(fl_ctx, name) \
fl_ctx->name##type = get_type(fl_ctx, fl_ctx->name##sym); \
fl_ctx->name##type->init = &cvalue_##name##_init
#define mk_primtype_(fl_ctx, name, ctype) \
fl_ctx->name##type = get_type(fl_ctx, fl_ctx->name##sym); \
fl_ctx->name##type->init = &cvalue_##ctype##_init
static void cvalues_init(fl_context_t *fl_ctx)
{
fl_ctx->malloc_pressure = 0;
fl_ctx->Finalizers = NULL;
fl_ctx->nfinalizers = 0;
fl_ctx->maxfinalizers = 0;
htable_new(&fl_ctx->TypeTable, 256);
htable_new(&fl_ctx->reverse_dlsym_lookup_table, 256);
fl_ctx->builtintype = define_opaque_type(fl_ctx->builtinsym, sizeof(builtin_t), NULL, NULL);
ctor_cv_intern(fl_ctx, int8);
ctor_cv_intern(fl_ctx, uint8);
ctor_cv_intern(fl_ctx, int16);
ctor_cv_intern(fl_ctx, uint16);
ctor_cv_intern(fl_ctx, int32);
ctor_cv_intern(fl_ctx, uint32);
ctor_cv_intern(fl_ctx, int64);
ctor_cv_intern(fl_ctx, uint64);
ctor_cv_intern(fl_ctx, byte);
ctor_cv_intern(fl_ctx, wchar);
ctor_cv_intern(fl_ctx, ptrdiff);
ctor_cv_intern(fl_ctx, size);
ctor_cv_intern(fl_ctx, float);
ctor_cv_intern(fl_ctx, double);
ctor_cv_intern(fl_ctx, array);
cv_intern(fl_ctx, pointer);
cv_intern(fl_ctx, void);
fl_ctx->cfunctionsym = symbol(fl_ctx, "c-function");
assign_global_builtins(fl_ctx, cvalues_builtin_info);
fl_ctx->stringtypesym = symbol(fl_ctx, "*string-type*");
setc(fl_ctx->stringtypesym, fl_list2(fl_ctx, fl_ctx->arraysym, fl_ctx->bytesym));
fl_ctx->wcstringtypesym = symbol(fl_ctx, "*wcstring-type*");
setc(fl_ctx->wcstringtypesym, fl_list2(fl_ctx, fl_ctx->arraysym, fl_ctx->wcharsym));
mk_primtype(fl_ctx, int8);
mk_primtype(fl_ctx, uint8);
mk_primtype(fl_ctx, int16);
mk_primtype(fl_ctx, uint16);
mk_primtype(fl_ctx, int32);
mk_primtype(fl_ctx, uint32);
mk_primtype(fl_ctx, int64);
mk_primtype(fl_ctx, uint64);
#ifdef _P64
mk_primtype_(fl_ctx, ptrdiff, int64);
mk_primtype_(fl_ctx, size, uint64);
#else
mk_primtype_(fl_ctx, ptrdiff, int32);
mk_primtype_(fl_ctx, size, uint32);
#endif
mk_primtype_(fl_ctx, byte, uint8);
mk_primtype_(fl_ctx, wchar, int32);
mk_primtype(fl_ctx, float);
mk_primtype(fl_ctx, double);
fl_ctx->stringtype = get_type(fl_ctx, symbol_value(fl_ctx->stringtypesym));
fl_ctx->wcstringtype = get_type(fl_ctx, symbol_value(fl_ctx->wcstringtypesym));
fl_ctx->emptystringsym = symbol(fl_ctx, "*empty-string*");
setc(fl_ctx->emptystringsym, cvalue_static_cstring(fl_ctx, ""));
}
#define RETURN_NUM_AS(fl_ctx, var, type) return(mk_##type(fl_ctx, (fl_##type##_t)var))
value_t return_from_uint64(fl_context_t *fl_ctx, uint64_t Uaccum)
{
if (fits_fixnum(Uaccum)) {
return fixnum((fixnum_t)Uaccum);
}
if (Uaccum > (uint64_t)S64_MAX) {
RETURN_NUM_AS(fl_ctx, Uaccum, uint64);
}
else if (Uaccum > (uint64_t)INT_MAX) {
RETURN_NUM_AS(fl_ctx, Uaccum, int64);
}
RETURN_NUM_AS(fl_ctx, Uaccum, int32);
}
value_t return_from_int64(fl_context_t *fl_ctx, int64_t Saccum)
{
if (fits_fixnum(Saccum)) {
return fixnum((fixnum_t)Saccum);
}
if (Saccum > (int64_t)INT_MAX || Saccum < (int64_t)INT_MIN) {
RETURN_NUM_AS(fl_ctx, Saccum, int64);
}
RETURN_NUM_AS(fl_ctx, Saccum, int32);
}
static value_t fl_add_any(fl_context_t *fl_ctx, value_t *args, uint32_t nargs, fixnum_t carryIn)
{
uint64_t Uaccum=0;
int64_t Saccum = carryIn;
double Faccum=0;
uint32_t i;
value_t arg=fl_ctx->NIL;
FOR_ARGS(i,0,arg,args) {
if (isfixnum(arg)) {
Saccum += numval(arg);
continue;
}
else if (iscprim(arg)) {
cprim_t *cp = (cprim_t*)ptr(arg);
void *a = cp_data(cp);
int64_t i64;
switch(cp_numtype(cp)) {
case T_INT8: Saccum += *(int8_t*)a; break;
case T_UINT8: Saccum += *(uint8_t*)a; break;
case T_INT16: Saccum += *(int16_t*)a; break;
case T_UINT16: Saccum += *(uint16_t*)a; break;
case T_INT32: Saccum += *(int32_t*)a; break;
case T_UINT32: Saccum += *(uint32_t*)a; break;
case T_INT64:
i64 = *(int64_t*)a;
if (i64 > 0)
Uaccum += (uint64_t)i64;
else
Saccum += i64;
break;
case T_UINT64: Uaccum += *(uint64_t*)a; break;
case T_FLOAT: Faccum += *(float*)a; break;
case T_DOUBLE: Faccum += *(double*)a; break;
default:
goto add_type_error;
}
continue;
}
add_type_error:
type_error(fl_ctx, "+", "number", arg);
}
if (Faccum != 0) {
Faccum += Uaccum;
Faccum += Saccum;
return mk_double(fl_ctx, Faccum);
}
else if (Saccum < 0) {
uint64_t negpart = (uint64_t)(-Saccum);
if (negpart > Uaccum) {
Saccum += (int64_t)Uaccum;
// return value in Saccum
if (Saccum >= INT_MIN) {
if (fits_fixnum(Saccum)) {
return fixnum((fixnum_t)Saccum);
}
RETURN_NUM_AS(fl_ctx, Saccum, int32);
}
RETURN_NUM_AS(fl_ctx, Saccum, int64);
}
Uaccum -= negpart;
}
else {
Uaccum += (uint64_t)Saccum;
}
// return value in Uaccum
return return_from_uint64(fl_ctx, Uaccum);
}
static value_t fl_neg(fl_context_t *fl_ctx, value_t n)
{
if (isfixnum(n)) {
return fixnum(-numval(n));
}
else if (iscprim(n)) {
cprim_t *cp = (cprim_t*)ptr(n);
void *a = cp_data(cp);
uint32_t ui32;
int32_t i32;
int64_t i64;
switch(cp_numtype(cp)) {
case T_INT8: return fixnum(-(int32_t)*(int8_t*)a);
case T_UINT8: return fixnum(-(int32_t)*(uint8_t*)a);
case T_INT16: return fixnum(-(int32_t)*(int16_t*)a);
case T_UINT16: return fixnum(-(int32_t)*(uint16_t*)a);
case T_INT32:
i32 = *(int32_t*)a;
if (i32 == (int32_t)BIT31)
return mk_uint32(fl_ctx, (uint32_t)BIT31);
return mk_int32(fl_ctx, -i32);
case T_UINT32:
ui32 = *(uint32_t*)a;
if (ui32 <= ((uint32_t)INT_MAX)+1) return mk_int32(fl_ctx, -(int32_t)ui32);
return mk_int64(fl_ctx, -(int64_t)ui32);
case T_INT64:
i64 = *(int64_t*)a;
if (i64 == (int64_t)BIT63)
return mk_uint64(fl_ctx, (uint64_t)BIT63);
return mk_int64(fl_ctx, -i64);
case T_UINT64: return mk_int64(fl_ctx, -(int64_t)*(uint64_t*)a);
case T_FLOAT: return mk_float(fl_ctx, -*(float*)a);
case T_DOUBLE: return mk_double(fl_ctx, -*(double*)a);
break;
}
}
type_error(fl_ctx, "-", "number", n);
}
static value_t fl_mul_any(fl_context_t *fl_ctx, value_t *args, uint32_t nargs, int64_t Saccum)
{
uint64_t Uaccum=1;
double Faccum=1;
uint32_t i;
value_t arg=fl_ctx->NIL;
FOR_ARGS(i,0,arg,args) {
if (isfixnum(arg)) {
Saccum *= numval(arg);
continue;
}
else if (iscprim(arg)) {
cprim_t *cp = (cprim_t*)ptr(arg);
void *a = cp_data(cp);
int64_t i64;
switch(cp_numtype(cp)) {
case T_INT8: Saccum *= *(int8_t*)a; break;
case T_UINT8: Saccum *= *(uint8_t*)a; break;
case T_INT16: Saccum *= *(int16_t*)a; break;
case T_UINT16: Saccum *= *(uint16_t*)a; break;
case T_INT32: Saccum *= *(int32_t*)a; break;
case T_UINT32: Saccum *= *(uint32_t*)a; break;
case T_INT64:
i64 = *(int64_t*)a;
if (i64 > 0)
Uaccum *= (uint64_t)i64;
else
Saccum *= i64;
break;
case T_UINT64: Uaccum *= *(uint64_t*)a; break;
case T_FLOAT: Faccum *= *(float*)a; break;
case T_DOUBLE: Faccum *= *(double*)a; break;
default:
goto mul_type_error;
}
continue;
}
mul_type_error:
type_error(fl_ctx, "*", "number", arg);
}
if (Faccum != 1) {
Faccum *= Uaccum;
Faccum *= Saccum;
return mk_double(fl_ctx, Faccum);
}
else if (Saccum < 0) {
Saccum *= (int64_t)Uaccum;
if (Saccum >= INT_MIN) {
if (fits_fixnum(Saccum)) {
return fixnum((fixnum_t)Saccum);
}
RETURN_NUM_AS(fl_ctx, Saccum, int32);
}
RETURN_NUM_AS(fl_ctx, Saccum, int64);
}
else {
Uaccum *= (uint64_t)Saccum;
}
return return_from_uint64(fl_ctx, Uaccum);
}
static int num_to_ptr(value_t a, fixnum_t *pi, numerictype_t *pt, void **pp)
{
cprim_t *cp;
if (isfixnum(a)) {
*pi = numval(a);
*pp = pi;
*pt = T_FIXNUM;
}
else if (iscprim(a)) {
cp = (cprim_t*)ptr(a);
*pp = cp_data(cp);
*pt = cp_numtype(cp);
}
else {
return 0;
}
return 1;
}
/*
returns -1, 0, or 1 based on ordering of a and b
eq: consider equality only, returning 0 or nonzero
eqnans: NaNs considered equal to each other
-0.0 not considered equal to 0.0
inexact not considered equal to exact
fname: if not NULL, throws type errors, else returns 2 for type errors
*/
int numeric_compare(fl_context_t *fl_ctx, value_t a, value_t b, int eq, int eqnans, char *fname)
{
int_t ai, bi;
numerictype_t ta, tb;
void *aptr, *bptr;
if (bothfixnums(a,b)) {
if (a==b) return 0;
if (numval(a) < numval(b)) return -1;
return 1;
}
if (!num_to_ptr(a, &ai, &ta, &aptr)) {
if (fname) type_error(fl_ctx, fname, "number", a); else return 2;
}
if (!num_to_ptr(b, &bi, &tb, &bptr)) {
if (fname) type_error(fl_ctx, fname, "number", b); else return 2;
}
if (eq && eqnans && ((ta >= T_FLOAT) != (tb >= T_FLOAT)))
return 1;
if (cmp_eq(aptr, ta, bptr, tb, eqnans))
return 0;
if (eq) return 1;
if (cmp_lt(aptr, ta, bptr, tb))
return -1;
return 1;
}
#if defined(_OS_WINDOWS_)
__declspec(noreturn) static void DivideByZeroError(fl_context_t *fl_ctx);
#else
static void DivideByZeroError(fl_context_t *fl_ctx) __attribute__ ((__noreturn__));
#endif
static void DivideByZeroError(fl_context_t *fl_ctx)
{
lerror(fl_ctx, fl_ctx->DivideError, "/: division by zero");
}
static value_t fl_div2(fl_context_t *fl_ctx, value_t a, value_t b)
{
double da, db;
int_t ai, bi;
numerictype_t ta, tb;
void *aptr, *bptr;
if (!num_to_ptr(a, &ai, &ta, &aptr))
type_error(fl_ctx, "/", "number", a);
if (!num_to_ptr(b, &bi, &tb, &bptr))
type_error(fl_ctx, "/", "number", b);
da = conv_to_double(aptr, ta);
db = conv_to_double(bptr, tb);
if (db == 0 && tb < T_FLOAT) // exact 0
DivideByZeroError(fl_ctx);
da = da/db;
if (ta < T_FLOAT && tb < T_FLOAT && (double)(int64_t)da == da)
return return_from_int64(fl_ctx, (int64_t)da);
return mk_double(fl_ctx, da);
}
static value_t fl_idiv2(fl_context_t *fl_ctx, value_t a, value_t b)
{
int_t ai, bi;
numerictype_t ta, tb;
void *aptr, *bptr;
int64_t a64, b64;
if (!num_to_ptr(a, &ai, &ta, &aptr))
type_error(fl_ctx, "div0", "number", a);
if (!num_to_ptr(b, &bi, &tb, &bptr))
type_error(fl_ctx, "div0", "number", b);
if (ta == T_UINT64) {
if (tb == T_UINT64) {
if (*(uint64_t*)bptr == 0) goto div_error;
return return_from_uint64(fl_ctx, *(uint64_t*)aptr / *(uint64_t*)bptr);
}
b64 = conv_to_int64(bptr, tb);
if (b64 < 0) {
return return_from_int64(fl_ctx, -(int64_t)(*(uint64_t*)aptr /
(uint64_t)(-b64)));
}
if (b64 == 0)
goto div_error;
return return_from_uint64(fl_ctx, *(uint64_t*)aptr / (uint64_t)b64);
}
if (tb == T_UINT64) {
if (*(uint64_t*)bptr == 0) goto div_error;
a64 = conv_to_int64(aptr, ta);
if (a64 < 0) {
return return_from_int64(fl_ctx, -((int64_t)((uint64_t)(-a64) /
*(uint64_t*)bptr)));
}
return return_from_uint64(fl_ctx, (uint64_t)a64 / *(uint64_t*)bptr);
}
b64 = conv_to_int64(bptr, tb);
if (b64 == 0) goto div_error;
return return_from_int64(fl_ctx, conv_to_int64(aptr, ta) / b64);
div_error:
DivideByZeroError(fl_ctx);
}
static value_t fl_bitwise_op(fl_context_t *fl_ctx, value_t a, value_t b, int opcode, char *fname)
{
int_t ai, bi;
numerictype_t ta, tb, itmp;
void *aptr=NULL, *bptr=NULL, *ptmp;
int64_t b64;
if (!num_to_ptr(a, &ai, &ta, &aptr) || ta >= T_FLOAT)
type_error(fl_ctx, fname, "integer", a);
if (!num_to_ptr(b, &bi, &tb, &bptr) || tb >= T_FLOAT)
type_error(fl_ctx, fname, "integer", b);
if (ta < tb) {
itmp = ta; ta = tb; tb = itmp;
ptmp = aptr; aptr = bptr; bptr = ptmp;
}
// now a's type is larger than or same as b's
b64 = conv_to_int64(bptr, tb);
switch (opcode) {
case 0:
switch (ta) {
case T_INT8: return fixnum( *(int8_t *)aptr & (int8_t )b64);
case T_UINT8: return fixnum( *(uint8_t *)aptr & (uint8_t )b64);
case T_INT16: return fixnum( *(int16_t*)aptr & (int16_t )b64);
case T_UINT16: return fixnum( *(uint16_t*)aptr & (uint16_t)b64);
case T_INT32: return mk_int32(fl_ctx, *(int32_t*)aptr & (int32_t )b64);
case T_UINT32: return mk_uint32(fl_ctx, *(uint32_t*)aptr & (uint32_t)b64);
case T_INT64: return mk_int64(fl_ctx, *(int64_t*)aptr & (int64_t )b64);
case T_UINT64: return mk_uint64(fl_ctx, *(uint64_t*)aptr & (uint64_t)b64);
case T_FLOAT:
case T_DOUBLE: assert(0);
}
break;
case 1:
switch (ta) {
case T_INT8: return fixnum( *(int8_t *)aptr | (int8_t )b64);
case T_UINT8: return fixnum( *(uint8_t *)aptr | (uint8_t )b64);
case T_INT16: return fixnum( *(int16_t*)aptr | (int16_t )b64);
case T_UINT16: return fixnum( *(uint16_t*)aptr | (uint16_t)b64);
case T_INT32: return mk_int32(fl_ctx, *(int32_t*)aptr | (int32_t )b64);
case T_UINT32: return mk_uint32(fl_ctx, *(uint32_t*)aptr | (uint32_t)b64);
case T_INT64: return mk_int64(fl_ctx, *(int64_t*)aptr | (int64_t )b64);
case T_UINT64: return mk_uint64(fl_ctx, *(uint64_t*)aptr | (uint64_t)b64);
case T_FLOAT:
case T_DOUBLE: assert(0);
}
break;
case 2:
switch (ta) {
case T_INT8: return fixnum( *(int8_t *)aptr ^ (int8_t )b64);
case T_UINT8: return fixnum( *(uint8_t *)aptr ^ (uint8_t )b64);
case T_INT16: return fixnum( *(int16_t*)aptr ^ (int16_t )b64);
case T_UINT16: return fixnum( *(uint16_t*)aptr ^ (uint16_t)b64);
case T_INT32: return mk_int32(fl_ctx, *(int32_t*)aptr ^ (int32_t )b64);
case T_UINT32: return mk_uint32(fl_ctx, *(uint32_t*)aptr ^ (uint32_t)b64);
case T_INT64: return mk_int64(fl_ctx, *(int64_t*)aptr ^ (int64_t )b64);
case T_UINT64: return mk_uint64(fl_ctx, *(uint64_t*)aptr ^ (uint64_t)b64);
case T_FLOAT:
case T_DOUBLE: assert(0);
}
}
assert(0);
return fl_ctx->NIL;
}
static value_t fl_logand(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
{
value_t v, e;
int i;
if (nargs == 0)
return fixnum(-1);
v = args[0];
FOR_ARGS(i,1,e,args) {
if (bothfixnums(v, e))
v = v & e;
else
v = fl_bitwise_op(fl_ctx, v, e, 0, "logand");
}
return v;
}
static value_t fl_logior(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
{
value_t v, e;
int i;
if (nargs == 0)
return fixnum(0);
v = args[0];
FOR_ARGS(i,1,e,args) {
if (bothfixnums(v, e))
v = v | e;
else
v = fl_bitwise_op(fl_ctx, v, e, 1, "logior");
}
return v;
}
static value_t fl_logxor(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
{
value_t v, e;
int i;
if (nargs == 0)
return fixnum(0);
v = args[0];
FOR_ARGS(i,1,e,args) {
if (bothfixnums(v, e))
v = fixnum(numval(v) ^ numval(e));
else
v = fl_bitwise_op(fl_ctx, v, e, 2, "logxor");
}
return v;
}
static value_t fl_lognot(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
{
argcount(fl_ctx, "lognot", nargs, 1);
value_t a = args[0];
if (isfixnum(a))
return fixnum(~numval(a));
cprim_t *cp;
int ta;
void *aptr;
if (iscprim(a)) {
cp = (cprim_t*)ptr(a);
ta = cp_numtype(cp);
aptr = cp_data(cp);
switch (ta) {
case T_INT8: return fixnum(~*(int8_t *)aptr);
case T_UINT8: return fixnum(~*(uint8_t *)aptr);
case T_INT16: return fixnum(~*(int16_t *)aptr);
case T_UINT16: return fixnum(~*(uint16_t*)aptr);
case T_INT32: return mk_int32(fl_ctx, ~*(int32_t *)aptr);
case T_UINT32: return mk_uint32(fl_ctx, ~*(uint32_t*)aptr);
case T_INT64: return mk_int64(fl_ctx, ~*(int64_t *)aptr);
case T_UINT64: return mk_uint64(fl_ctx, ~*(uint64_t*)aptr);
}
}
type_error(fl_ctx, "lognot", "integer", a);
}
static value_t fl_ash(fl_context_t *fl_ctx, value_t *args, uint32_t nargs)
{
fixnum_t n;
int64_t accum;
argcount(fl_ctx, "ash", nargs, 2);
value_t a = args[0];
n = tofixnum(fl_ctx, args[1], "ash");
if (isfixnum(a)) {
if (n <= 0)
return fixnum(numval(a)>>(-n));
accum = ((int64_t)numval(a))<<n;
if (fits_fixnum(accum))
return fixnum(accum);
else
return return_from_int64(fl_ctx, accum);
}
cprim_t *cp;
int ta;
void *aptr;
if (iscprim(a)) {
if (n == 0) return a;
cp = (cprim_t*)ptr(a);
ta = cp_numtype(cp);
aptr = cp_data(cp);
if (n < 0) {
n = -n;
switch (ta) {
case T_INT8: return fixnum((*(int8_t *)aptr) >> n);
case T_UINT8: return fixnum((*(uint8_t *)aptr) >> n);
case T_INT16: return fixnum((*(int16_t *)aptr) >> n);
case T_UINT16: return fixnum((*(uint16_t*)aptr) >> n);
case T_INT32: return mk_int32(fl_ctx, (*(int32_t *)aptr) >> n);
case T_UINT32: return mk_uint32(fl_ctx, (*(uint32_t*)aptr) >> n);
case T_INT64: return mk_int64(fl_ctx, (*(int64_t *)aptr) >> n);
case T_UINT64: return mk_uint64(fl_ctx, (*(uint64_t*)aptr) >> n);
}
}
else {
if (ta == T_UINT64)
return return_from_uint64(fl_ctx, (*(uint64_t*)aptr)<<n);
else if (ta < T_FLOAT) {
int64_t i64 = conv_to_int64(aptr, (numerictype_t)ta);
return return_from_int64(fl_ctx, i64<<n);
}
}
}
type_error(fl_ctx, "ash", "integer", a);
return fl_ctx->NIL;
}
Computing file changes ...