X-Git-Url: https://git.m6w6.name/?a=blobdiff_plain;f=src%2Flibffi.c;h=ac018a119640d8ea0deed03ba479cf1e6c18a260;hb=b0a8b273be0744ed445fbe6c935a04ee7a749f14;hp=8d41b7a192fa24287bffa03c522821d87bfea015;hpb=ad859923be758c210e2ad4226eb8e3b09bcf91eb;p=m6w6%2Fext-psi diff --git a/src/libffi.c b/src/libffi.c index 8d41b7a..ac018a1 100644 --- a/src/libffi.c +++ b/src/libffi.c @@ -91,6 +91,8 @@ static inline ffi_type *psi_ffi_token_type(token_t t) { return &ffi_type_float; case PSI_T_DOUBLE: return &ffi_type_double; + case PSI_T_POINTER: + return &ffi_type_pointer; } } static inline ffi_type *psi_ffi_impl_type(token_t impl_type) { @@ -107,8 +109,45 @@ static inline ffi_type *psi_ffi_impl_type(token_t impl_type) { EMPTY_SWITCH_DEFAULT_CASE(); } } +static void psi_ffi_struct_type_dtor(void *type) { + ffi_type *strct = type; + + if (strct->elements) { + free(strct->elements); + } + free(strct); +} +static inline ffi_type *psi_ffi_decl_arg_type(decl_arg *darg); +static ffi_type **psi_ffi_struct_type_elements(decl_struct *strct) { + ffi_type **els = calloc(strct->args->count + 1, sizeof(*els)); + size_t i; + + for (i = 0; i < strct->args->count; ++i) { + els[i] = psi_ffi_decl_arg_type(strct->args->args[i]); + } + els[i] = NULL; + + return els; +} static inline ffi_type *psi_ffi_decl_type(decl_type *type) { - return psi_ffi_token_type(real_decl_type(type)->type); + decl_type *real = real_decl_type(type); + + if (real->type == PSI_T_STRUCT) { + if (!real->strct->engine.type) { + ffi_type *strct = calloc(1, sizeof(ffi_type)); + + strct->type = FFI_TYPE_STRUCT; + strct->size = real->strct->size; + strct->alignment = 0; + strct->elements = psi_ffi_struct_type_elements(real->strct); + + real->strct->engine.type = strct; + real->strct->engine.dtor = psi_ffi_struct_type_dtor; + } + + return real->strct->engine.type; + } + return psi_ffi_token_type(real->type); } static inline ffi_type *psi_ffi_decl_arg_type(decl_arg *darg) { if (darg->var->pointer_level) {