summaryrefslogtreecommitdiffstats
path: root/ffi.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2017-06-27 06:39:28 -0700
committerKaz Kylheku <kaz@kylheku.com>2017-06-27 06:39:28 -0700
commit52870f75e9423ae12ac9b3254e4097520c20af86 (patch)
tree93eead0480d51fbc2f07bb4cd16dca3f802b51b5 /ffi.c
parent73874bab389305e76b8fa3726f4494a613735f5a (diff)
downloadtxr-52870f75e9423ae12ac9b3254e4097520c20af86.tar.gz
txr-52870f75e9423ae12ac9b3254e4097520c20af86.tar.bz2
txr-52870f75e9423ae12ac9b3254e4097520c20af86.zip
ffi: prepare for variably sized enum implementation.
This lays the groundwork for enumerations based off types other than int. All we need to do is implement the syntax for this. * ffi.c (ffi_enum_put, ffi_enum_get, ffi_enum_rput, ffi_enum_rget): Retrieve the element type descriptor and recurse into its put, get, rput and rget operation, respectively, instead of hard-coded calls to the int operations. (make_ffi_type_enum): New argument, base_type. Numerous properties of the enum type are taken from base_type: the FFI type, the alignment, size, and alloc, free and clone operations. The base type is installed as eltype. (ffi_type_lookup): Static function relocated. (ffi_type_compile): In the enum case, resolve the int type and pass to make_ffi_type_enum.
Diffstat (limited to 'ffi.c')
-rw-r--r--ffi.c43
1 files changed, 26 insertions, 17 deletions
diff --git a/ffi.c b/ffi.c
index 89602f55..ca8ab24b 100644
--- a/ffi.c
+++ b/ffi.c
@@ -2618,6 +2618,8 @@ static void ffi_carray_put(struct txr_ffi_type *tft, val carray, mem_t *dst,
static void ffi_enum_put(struct txr_ffi_type *tft, val n, mem_t *dst, val self)
{
+ struct txr_ffi_type *etft = ffi_type_struct(tft->eltype);
+
if (symbolp(n)) {
val n_num = gethash(tft->num_sym, n);
if (!n_num)
@@ -2625,12 +2627,13 @@ static void ffi_enum_put(struct txr_ffi_type *tft, val n, mem_t *dst, val self)
tft->syntax, n, nao);
n = n_num;
}
- ffi_int_put(tft, n, dst, self);
+ etft->put(tft, n, dst, self); /* tft deliberate */
}
static val ffi_enum_get(struct txr_ffi_type *tft, mem_t *src, val self)
{
- val n = ffi_int_get(tft, src, self);
+ struct txr_ffi_type *etft = ffi_type_struct(tft->eltype);
+ val n = etft->get(tft, src, self); /* tft deliberate */
val sym = gethash(tft->sym_num, n);
return if3(sym, sym, n);
}
@@ -2639,6 +2642,8 @@ static val ffi_enum_get(struct txr_ffi_type *tft, mem_t *src, val self)
static void ffi_enum_rput(struct txr_ffi_type *tft, val n, mem_t *dst, val self)
{
+ struct txr_ffi_type *etft = ffi_type_struct(tft->eltype);
+
if (symbolp(n)) {
val n_num = gethash(tft->num_sym, n);
if (!n_num)
@@ -2646,12 +2651,13 @@ static void ffi_enum_rput(struct txr_ffi_type *tft, val n, mem_t *dst, val self)
tft->syntax, n, nao);
n = n_num;
}
- ffi_int_rput(tft, n, dst, self);
+ etft->rput(tft, n, dst, self); /* tft deliberate */
}
static val ffi_enum_rget(struct txr_ffi_type *tft, mem_t *src, val self)
{
- val n = ffi_int_rget(tft, src, self);
+ struct txr_ffi_type *etft = ffi_type_struct(tft->eltype);
+ val n = etft->rget(tft, src, self); /* tft deliberate */
val sym = gethash(tft->sym_num, n);
return if3(sym, sym, n);
}
@@ -3111,10 +3117,12 @@ static val ffi_eval_expr(val expr, val menv, val env)
return eval(expr_ex, env, expr_ex);
}
-static val make_ffi_type_enum(val syntax, val enums, val self)
+static val make_ffi_type_enum(val syntax, val enums,
+ val base_type, val self)
{
struct txr_ffi_type *tft = coerce(struct txr_ffi_type *,
chk_calloc(1, sizeof *tft));
+ struct txr_ffi_type *btft = ffi_type_struct(base_type);
val obj = cobj(coerce(mem_t *, tft), ffi_type_s, &ffi_type_enum_ops);
cnum lowest = INT_PTR_MAX;
@@ -3128,20 +3136,21 @@ static val make_ffi_type_enum(val syntax, val enums, val self)
val shadow_menv = make_env(nil, nil, nil);
tft->self = obj;
- tft->ft = &ffi_type_sint;
+ tft->ft = btft->ft;
tft->syntax = syntax;
tft->lt = sym_s;
- tft->size = sizeof (int);
- tft->align = alignof (int);
- tft->clone = ffi_simple_clone;
+ tft->size = btft->size;
+ tft->align = btft->align;
+ tft->clone = btft->clone;
tft->put = ffi_enum_put;
tft->get = ffi_enum_get;
- tft->alloc = ffi_fixed_alloc;
- tft->free = free;
#if !HAVE_LITTLE_ENDIAN
tft->rput = ffi_enum_rput;
tft->rget = ffi_enum_rget;
#endif
+ tft->alloc = btft->alloc;
+ tft->free = btft->free;
+ tft->eltype = base_type;
for (iter = enums; !endp(iter); iter = cdr(iter), count++) {
val en = car(iter);
@@ -3233,6 +3242,11 @@ static val ffi_struct_compile(val membs, val *ptypes, val self)
return slots;
}
+static val ffi_type_lookup(val sym)
+{
+ return gethash(ffi_typedef_hash, sym);
+}
+
val ffi_type_compile(val syntax)
{
val self = lit("ffi-type-compile");
@@ -3472,7 +3486,7 @@ val ffi_type_compile(val syntax)
uw_throwf(error_s,
lit("~a: enum name ~s must be bindable symbol or nil"),
self, name, nao);
- return make_ffi_type_enum(xsyntax, enums, self);
+ return make_ffi_type_enum(xsyntax, enums, ffi_type_lookup(int_s), self);
} else if (sym == align_s) {
val align = ffi_eval_expr(cadr(syntax), nil, nil);
ucnum al = c_num(align);
@@ -3863,11 +3877,6 @@ static void ffi_init_types(void)
ffi_typedef(bool_s, ffi_type_compile(cons(bool_s, cons(uchar_s, nil))));
}
-static val ffi_type_lookup(val sym)
-{
- return gethash(ffi_typedef_hash, sym);
-}
-
static void ffi_init_extra_types(void)
{
val type_by_size[2][18] = { { 0 }, { 0 } };