diff options
author | Richard Henderson <rth@twiddle.net> | 2014-10-17 01:02:52 -0400 |
---|---|---|
committer | Richard Henderson <rth@twiddle.net> | 2014-11-12 09:32:13 +0100 |
commit | 0d39b4bb692b1a8e4f3354badcd9e123f8276bd4 (patch) | |
tree | 26ff1c3b316c4d1e1493eef5d3dd5064160ab31e /src/arm | |
parent | 57b24fb3f5471262ce57e1a912774d6e7de5ada7 (diff) | |
download | libffi-0d39b4bb692b1a8e4f3354badcd9e123f8276bd4.tar.gz |
arm: Deref ffi_put_arg arguments
Diffstat (limited to 'src/arm')
-rw-r--r-- | src/arm/ffi.c | 88 |
1 files changed, 38 insertions, 50 deletions
diff --git a/src/arm/ffi.c b/src/arm/ffi.c index 7d86e94..5e5ad0a 100644 --- a/src/arm/ffi.c +++ b/src/arm/ffi.c @@ -56,58 +56,46 @@ ffi_align (ffi_type *ty, void *p) } static size_t -ffi_put_arg (ffi_type **arg_type, void **arg, char *stack) +ffi_put_arg (ffi_type *ty, void *src, void *dst) { - register char *argp = stack; - register ffi_type **p_arg = arg_type; - register void **p_argv = arg; - register size_t z = (*p_arg)->size; + size_t z = ty->size; - if (z < sizeof (int)) + switch (ty->type) { - z = sizeof (int); - switch ((*p_arg)->type) - { - case FFI_TYPE_SINT8: - *(signed int *) argp = (signed int) *(SINT8 *) (*p_argv); - break; - - case FFI_TYPE_UINT8: - *(unsigned int *) argp = (unsigned int) *(UINT8 *) (*p_argv); - break; - - case FFI_TYPE_SINT16: - *(signed int *) argp = (signed int) *(SINT16 *) (*p_argv); - break; + case FFI_TYPE_SINT8: + *(UINT32 *)dst = *(SINT8 *)src; + break; + case FFI_TYPE_UINT8: + *(UINT32 *)dst = *(UINT8 *)src; + break; + case FFI_TYPE_SINT16: + *(UINT32 *)dst = *(SINT16 *)src; + break; + case FFI_TYPE_UINT16: + *(UINT32 *)dst = *(UINT16 *)src; + break; - case FFI_TYPE_UINT16: - *(unsigned int *) argp = (unsigned int) *(UINT16 *) (*p_argv); - break; + case FFI_TYPE_INT: + case FFI_TYPE_SINT32: + case FFI_TYPE_UINT32: + case FFI_TYPE_POINTER: + case FFI_TYPE_FLOAT: + *(UINT32 *)dst = *(UINT32 *)src; + break; - case FFI_TYPE_STRUCT: - memcpy (argp, *p_argv, (*p_arg)->size); - break; + case FFI_TYPE_SINT64: + case FFI_TYPE_UINT64: + case FFI_TYPE_DOUBLE: + *(UINT64 *)dst = *(UINT64 *)src; + break; - default: - FFI_ASSERT (0); - } - } - else if (z == sizeof (int)) - { - if ((*p_arg)->type == FFI_TYPE_FLOAT) - *(float *) argp = *(float *) (*p_argv); - else - *(unsigned int *) argp = (unsigned int) *(UINT32 *) (*p_argv); - } - else if (z == sizeof (double) && (*p_arg)->type == FFI_TYPE_DOUBLE) - { - *(double *) argp = *(double *) (*p_argv); - } - else - { - memcpy (argp, *p_argv, z); + case FFI_TYPE_STRUCT: + default: + memcpy (dst, src, z); + break; } - return z; + + return ALIGN (z, 4); } /* ffi_prep_args is called by the assembly routine once stack space @@ -138,7 +126,7 @@ ffi_prep_args_SYSV (char *stack, extended_cif *ecif, float *vfp_space) (i != 0); i--, p_arg++, p_argv++) { argp = ffi_align (*p_arg, argp); - argp += ffi_put_arg (p_arg, p_argv, argp); + argp += ffi_put_arg (*p_arg, *p_argv, argp); } return 0; @@ -182,7 +170,7 @@ ffi_prep_args_VFP (char *stack, extended_cif * ecif, float *vfp_space) if (vi < ecif->cif->vfp_nargs && is_vfp_type) { char *vfp_slot = (char *) (vfp_space + ecif->cif->vfp_args[vi++]); - ffi_put_arg (p_arg, p_argv, vfp_slot); + ffi_put_arg (*p_arg, *p_argv, vfp_slot); continue; } /* Try allocating in core registers. */ @@ -195,7 +183,7 @@ ffi_prep_args_VFP (char *stack, extended_cif * ecif, float *vfp_space) area to place the argument. */ if (tregp + size <= eo_regp) { - regp = tregp + ffi_put_arg (p_arg, p_argv, tregp); + regp = tregp + ffi_put_arg (*p_arg, *p_argv, tregp); done_with_regs = (regp == argp); // ensure we did not write into the stack area FFI_ASSERT (regp <= argp); @@ -208,7 +196,7 @@ ffi_prep_args_VFP (char *stack, extended_cif * ecif, float *vfp_space) { stack_used = 1; done_with_regs = 1; - argp = tregp + ffi_put_arg (p_arg, p_argv, tregp); + argp = tregp + ffi_put_arg (*p_arg, *p_argv, tregp); FFI_ASSERT (eo_regp < argp); continue; } @@ -216,7 +204,7 @@ ffi_prep_args_VFP (char *stack, extended_cif * ecif, float *vfp_space) /* Base case, arguments are passed on the stack */ stack_used = 1; argp = ffi_align (*p_arg, argp); - argp += ffi_put_arg (p_arg, p_argv, argp); + argp += ffi_put_arg (*p_arg, *p_argv, argp); } /* Indicate the VFP registers used. */ return ecif->cif->vfp_used; |