diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2017-06-27 06:39:28 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2017-06-27 06:39:28 -0700 |
commit | 52870f75e9423ae12ac9b3254e4097520c20af86 (patch) | |
tree | 93eead0480d51fbc2f07bb4cd16dca3f802b51b5 /ffi.c | |
parent | 73874bab389305e76b8fa3726f4494a613735f5a (diff) | |
download | txr-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.c | 43 |
1 files changed, 26 insertions, 17 deletions
@@ -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 } }; |