/*
 * Philosophy: FFI dispatch should be as fast as possible considering
 * reasonable trade offs.
 *
 *  - don't allocate memory for small things using `malloc`, instead use
 *    alloca on platforms that allow it (most modern platforms do).
 *  - don't make function calls.  You shouldn't have to make a function
 *    calls to call a function.  Exceptions are for custom types and
 *    some of the more esoteric types.
 *  - one way we avoid making function calls is by putting the FFI dispatch
 *    in this header file so that it can be "called" twice without an
 *    extra function call.  (`$ffi->function(...)->call(...)` and
 *    `$ffi->attach(foo => ...); foo(...)`).  This is obviously absurd.
 *
 * Maybe each of these weird trade offs each save only a few ms on
 * each call, but in the end the can add up.  As a result of this
 * priority set, FFI::Platypus does seem to perform considerably better
 * than any other FFI implementations available in Perl ( see
 * https://github.com/perl5-FFI/FFI-Performance ) and is even competitive
 * with XS tbh.
 */

    ffi_pl_heap *heap = NULL;

#if FFI_PL_CALL_NO_RECORD_VALUE
#define RESULT &result
    ffi_pl_result result;
#elif FFI_PL_CALL_RET_NO_NORMAL
#define RESULT result_ptr
    void *result_ptr;
    Newx_or_alloca(result_ptr, self->return_type->extra[0].record.size, char);
#else
#define RESULT result_ptr
    ffi_pl_result result;
    void *result_ptr;
    if(self->return_type->type_code == FFI_PL_TYPE_RECORD_VALUE
    || self->return_type->type_code == (FFI_PL_TYPE_RECORD_VALUE | FFI_PL_SHAPE_CUSTOM_PERL))
    {
      Newx_or_alloca(result_ptr, self->return_type->extra[0].record.size, char);
    }
    else
    {
      result_ptr = &result;
    }
#endif

    {
      /* buffer contains the memory required for the arguments structure */
      char *buffer;
      size_t buffer_size = sizeof(ffi_pl_argument) * self->ffi_cif.nargs +
                    sizeof(void*) * self->ffi_cif.nargs +
                    sizeof(ffi_pl_arguments);
      ffi_pl_heap_add(buffer, buffer_size, char);
      MY_CXT.current_argv = arguments = (ffi_pl_arguments*) buffer;
    }

    arguments->count = self->ffi_cif.nargs;
    argument_pointers = (void**) &arguments->slot[arguments->count];

/*
 * ARGUMENT IN
 */

    for(i=0, perl_arg_index=(EXTRA_ARGS); i < self->ffi_cif.nargs; i++, perl_arg_index++)
    {
      int type_code = self->argument_types[i]->type_code;
      argument_pointers[i] = (void*) &arguments->slot[i];

      arg = perl_arg_index < items ? ST(perl_arg_index) : &PL_sv_undef;

      int custom_flag = (type_code & FFI_PL_SHAPE_MASK) == FFI_PL_SHAPE_CUSTOM_PERL;
      if(custom_flag)
      {
        arg = ffi_pl_custom_perl(
          self->argument_types[i]->extra[0].custom_perl.perl_to_native,
          arg, i
        );
        if(arg == NULL)
        {
          int max = self->argument_types[i]->extra[0].custom_perl.argument_count;
          for(n=0; n < max; n++)
          {
            i++;
            argument_pointers[i] = &arguments->slot[i];
          }
          continue;
        }
        av_push(MY_CXT.custom_keepers, newRV_inc(arg));
        type_code ^= FFI_PL_SHAPE_CUSTOM_PERL;
      }

      switch(type_code)
      {

/*
 * ARGUMENT IN - SCALAR TYPES
 */

        case FFI_PL_TYPE_UINT8:
          ffi_pl_arguments_set_uint8(arguments, i, SvOK(arg) ? SvUV(arg) : 0);
          break;
        case FFI_PL_TYPE_SINT8:
          ffi_pl_arguments_set_sint8(arguments, i, SvOK(arg) ? SvIV(arg) : 0);
          break;
        case FFI_PL_TYPE_UINT16:
          ffi_pl_arguments_set_uint16(arguments, i, SvOK(arg) ? SvUV(arg) : 0);
          break;
        case FFI_PL_TYPE_SINT16:
          ffi_pl_arguments_set_sint16(arguments, i, SvOK(arg) ? SvIV(arg) : 0);
          break;
        case FFI_PL_TYPE_UINT32:
          ffi_pl_arguments_set_uint32(arguments, i, SvOK(arg) ? SvUV(arg) : 0);
          break;
        case FFI_PL_TYPE_SINT32:
          ffi_pl_arguments_set_sint32(arguments, i, SvOK(arg) ? SvIV(arg) : 0);
          break;
        case FFI_PL_TYPE_UINT64:
          ffi_pl_arguments_set_uint64(arguments, i, SvOK(arg) ? SvU64(arg) : 0);
          break;
        case FFI_PL_TYPE_SINT64:
          ffi_pl_arguments_set_sint64(arguments, i, SvOK(arg) ? SvI64(arg) : 0);
          break;
        case FFI_PL_TYPE_FLOAT:
          ffi_pl_arguments_set_float(arguments, i, SvOK(arg) ? SvNV(arg) : 0.0);
          break;
        case FFI_PL_TYPE_DOUBLE:
          ffi_pl_arguments_set_double(arguments, i, SvOK(arg) ? SvNV(arg) : 0.0);
          break;
        case FFI_PL_TYPE_OPAQUE:
          ffi_pl_arguments_set_pointer(arguments, i, SvOK(arg) ? INT2PTR(void*, SvIV(arg)) : NULL);
          break;
        case FFI_PL_TYPE_STRING:
          ffi_pl_arguments_set_string(arguments, i, SvOK(arg) ? SvPV_nolen(arg) : NULL);
          break;
#ifdef FFI_PL_PROBE_LONGDOUBLE
        case FFI_PL_TYPE_LONG_DOUBLE:
          {
            long double *ptr;
            Newx_or_alloca(ptr, 1, long double);
            argument_pointers[i] = ptr;
            ffi_pl_perl_to_long_double(arg, ptr);
          }
          break;
#endif
#ifdef FFI_PL_PROBE_COMPLEX
        case FFI_PL_TYPE_COMPLEX_FLOAT:
          {
            float *ptr;
            Newx_or_alloca(ptr, 2, float);
            argument_pointers[i] = ptr;
            ffi_pl_perl_to_complex_float(arg, ptr);
          }
          break;
        case FFI_PL_TYPE_COMPLEX_DOUBLE:
          {
            double *ptr;
            Newx_or_alloca(ptr, 2, double);
            argument_pointers[i] = ptr;
            ffi_pl_perl_to_complex_double(arg, ptr);
          }
          break;
#endif
        case FFI_PL_TYPE_RECORD:
          {
            void *ptr;
            STRLEN size;
            int expected;
            expected = self->argument_types[i]->extra[0].record.size;
            if(SvROK(arg))
            {
              SV *arg2 = SvRV(arg);
              ptr = SvOK(arg2) ? SvPV(arg2, size) : NULL;
            }
            else
            {
              ptr = SvOK(arg) ? SvPV(arg, size) : NULL;
            }
            if(ptr != NULL && expected != 0 && size != expected)
              warn("record argument %d has wrong size (is %d, expected %d)", i, (int)size, expected);
            ffi_pl_arguments_set_pointer(arguments, i, ptr);
          }
          break;
        case FFI_PL_TYPE_RECORD_VALUE:
          {
            const char *record_class = self->argument_types[i]->extra[0].record.class;
            /* TODO if object is read-onyl ? */
            if(sv_isobject(arg) && sv_derived_from(arg, record_class))
            {
              argument_pointers[i] = SvPV_nolen(SvRV(arg));
            }
            else
            {
              ffi_pl_croak("argument %d is not an instance of %s", i, record_class);
            }
          }
          break;
        case FFI_PL_TYPE_CLOSURE:
          {
            if(!SvROK(arg))
            {
              ffi_pl_arguments_set_pointer(arguments, i, SvOK(arg) ? INT2PTR(void*, SvIV(arg)) : NULL);
            }
            else
            {
              ffi_pl_closure *closure;
              ffi_status ffi_status;

              SvREFCNT_inc(arg);
              SvREFCNT_inc(SvRV(arg));

              closure = ffi_pl_closure_get_data(arg, self->argument_types[i]);
              if(closure != NULL)
              {
                ffi_pl_arguments_set_pointer(arguments, i, closure->function_pointer);
              }
              else
              {
                Newx(closure, 1, ffi_pl_closure);
                closure->ffi_closure = ffi_closure_alloc(sizeof(ffi_closure), &closure->function_pointer);
                if(closure->ffi_closure == NULL)
                {
                  Safefree(closure);
                  ffi_pl_arguments_set_pointer(arguments, i, NULL);
                  warn("unable to allocate memory for closure");
                }
                else
                {
                  closure->type = self->argument_types[i];

                  ffi_status = ffi_prep_closure_loc(
                    closure->ffi_closure,
                    &self->argument_types[i]->extra[0].closure.ffi_cif,
                    ffi_pl_closure_call,
                    closure,
                    closure->function_pointer
                  );
                  if(ffi_status != FFI_OK)
                  {
                    ffi_closure_free(closure->ffi_closure);
                    Safefree(closure);
                    ffi_pl_arguments_set_pointer(arguments, i, NULL);
                    warn("unable to create closure");
                  }
                  else
                  {
                    SV **svp;
                    svp = hv_fetch((HV *)SvRV(arg), "code", 4, 0);
                    if(svp != NULL)
                    {
                      closure->coderef = *svp;
                      SvREFCNT_inc(closure->coderef);
                      ffi_pl_closure_add_data(arg, closure);
                      ffi_pl_arguments_set_pointer(arguments, i, closure->function_pointer);
                    }
                    else
                    {
                      ffi_closure_free(closure->ffi_closure);
                      Safefree(closure);
                      ffi_pl_arguments_set_pointer(arguments, i, NULL);
                      warn("closure has no coderef");
                    }
                  }
                }
              }
            }
          }
          break;
        default:

          switch(type_code & FFI_PL_SHAPE_MASK)
          {

/*
 * ARGUMENT IN - POINTER & ARRAY TYPES
 */

            case FFI_PL_SHAPE_POINTER:
            case FFI_PL_SHAPE_ARRAY:
              {
                void *ptr = NULL;
                SSize_t count = 0;
                int is_pointer = (type_code & FFI_PL_SHAPE_MASK) == FFI_PL_SHAPE_POINTER;
                int is_bad = 0;
                if(SvROK(arg))
                {
                  SV *arg2 = SvRV(arg);
                  if(SvTYPE(arg2) < SVt_PVAV && is_pointer)
                  {
                    switch(type_code & (FFI_PL_BASE_MASK|FFI_PL_SIZE_MASK))
                    {
                      case FFI_PL_TYPE_UINT8:
                        Newx_or_alloca(ptr, 1, uint8_t);
                        *((uint8_t*)ptr) = SvOK(arg2) ? SvUV(arg2) : 0;
                        break;
                      case FFI_PL_TYPE_SINT8:
                        Newx_or_alloca(ptr, 1, int8_t);
                        *((int8_t*)ptr) = SvOK(arg2) ? SvIV(arg2) : 0;
                        break;
                      case FFI_PL_TYPE_UINT16:
                        Newx_or_alloca(ptr, 1, uint16_t);
                        *((uint16_t*)ptr) = SvOK(arg2) ? SvUV(arg2) : 0;
                        break;
                      case FFI_PL_TYPE_SINT16:
                        Newx_or_alloca(ptr, 1, int16_t);
                        *((int16_t*)ptr) = SvOK(arg2) ? SvIV(arg2) : 0;
                        break;
                      case FFI_PL_TYPE_UINT32:
                        Newx_or_alloca(ptr, 1, uint32_t);
                        *((uint32_t*)ptr) = SvOK(arg2) ? SvUV(arg2) : 0;
                        break;
                      case FFI_PL_TYPE_SINT32:
                        Newx_or_alloca(ptr, 1, int32_t);
                        *((int32_t*)ptr) = SvOK(arg2) ? SvIV(arg2) : 0;
                        break;
                      case FFI_PL_TYPE_UINT64:
                        Newx_or_alloca(ptr, 1, uint64_t);
                        *((uint64_t*)ptr) = SvOK(arg2) ? SvU64(arg2) : 0;
                        break;
                      case FFI_PL_TYPE_SINT64:
                        Newx_or_alloca(ptr, 1, int64_t);
                        *((int64_t*)ptr) = SvOK(arg2) ? SvI64(arg2) : 0;
                        break;
                      case FFI_PL_TYPE_FLOAT:
                        Newx_or_alloca(ptr, 1, float);
                        *((float*)ptr) = SvOK(arg2) ? SvNV(arg2) : 0.0;
                        break;
                      case FFI_PL_TYPE_DOUBLE:
                        Newx_or_alloca(ptr, 1, double);
                        *((double*)ptr) = SvOK(arg2) ? SvNV(arg2) : 0.0;
                        break;
                      case FFI_PL_TYPE_OPAQUE:
                        Newx_or_alloca(ptr, 1, void*);
                        {
                          SV *tmp = SvRV(arg);
                          *((void**)ptr) = SvOK(tmp) ? INT2PTR(void *, SvIV(tmp)) : NULL;
                        }
                        break;
#ifdef FFI_PL_PROBE_LONGDOUBLE
                      case FFI_PL_TYPE_LONG_DOUBLE:
                        Newx_or_alloca(ptr, 1, long double);
                        ffi_pl_perl_to_long_double(arg2, (long double*)ptr);
                        break;
#endif
#ifdef FFI_PL_PROBE_COMPLEX
                      case FFI_PL_TYPE_COMPLEX_FLOAT:
                        Newx_or_alloca(ptr, 1, float complex);
                        ffi_pl_perl_to_complex_float(arg2, (float *)ptr);
                        break;
                      case FFI_PL_TYPE_COMPLEX_DOUBLE:
                        Newx_or_alloca(ptr, 1, double complex);
                        ffi_pl_perl_to_complex_double(arg2, (double *)ptr);
                        break;
#endif
                      case FFI_PL_TYPE_STRING:
                        Newx_or_alloca(ptr, 1, char *);
                        if(SvOK(arg2))
                        {
                          char *pv;
                          STRLEN len;
                          char *str;
                          pv = SvPV(arg2, len);
                          /* TODO: this should probably be a malloc since it could be arbitrarily large */
                          Newx_or_alloca(str, len+1, char);
                          memcpy(str, pv, len+1);
                          *((char**)ptr) = str;
                        }
                        else
                        {
                          *((char**)ptr) = NULL;
                        }
                        break;
                      default:
                        warn("argument type not supported (%d)", i);
                        Newx_or_alloca(ptr, 1, void*);
                        *((void**)ptr) = NULL;
                        break;
                    }
                  }
                  else if(SvTYPE(arg2) == SVt_PVAV && (!is_pointer) || (is_pointer && self->platypus_api >= 2))
                  {
                    AV *av = (AV*) arg2;
                    if(!is_pointer)
                    {
                      count = self->argument_types[i]->extra[0].array.element_count;
                    }
                    if(count == 0)
                    {
                      count = av_len(av)+1;
                    }
                    switch(type_code & (FFI_PL_BASE_MASK|FFI_PL_SIZE_MASK))
                    {
                      case FFI_PL_TYPE_UINT8:
                        Newx(ptr, count, uint8_t);
                        for(n=0; n<count; n++)
                        {
                          ((uint8_t*)ptr)[n] = SvUV(*av_fetch(av, n, 1));
                        }
                        break;
                      case FFI_PL_TYPE_SINT8:
                        Newx(ptr, count, int8_t);
                        for(n=0; n<count; n++)
                        {
                          ((int8_t*)ptr)[n] = SvIV(*av_fetch(av, n, 1));
                        }
                        break;
                      case FFI_PL_TYPE_UINT16:
                        Newx(ptr, count, uint16_t);
                        for(n=0; n<count; n++)
                        {
                          ((uint16_t*)ptr)[n] = SvUV(*av_fetch(av, n, 1));
                        }
                        break;
                      case FFI_PL_TYPE_SINT16:
                        Newx(ptr, count, int16_t);
                        for(n=0; n<count; n++)
                        {
                          ((int16_t*)ptr)[n] = SvIV(*av_fetch(av, n, 1));
                        }
                        break;
                      case FFI_PL_TYPE_UINT32:
                        Newx(ptr, count, uint32_t);
                        for(n=0; n<count; n++)
                        {
                          ((uint32_t*)ptr)[n] = SvUV(*av_fetch(av, n, 1));
                        }
                        break;
                      case FFI_PL_TYPE_SINT32:
                        Newx(ptr, count, int32_t);
                        for(n=0; n<count; n++)
                        {
                          ((int32_t*)ptr)[n] = SvIV(*av_fetch(av, n, 1));
                        }
                        break;
                      case FFI_PL_TYPE_UINT64:
                        Newx(ptr, count, uint64_t);
                        for(n=0; n<count; n++)
                        {
                          ((uint64_t*)ptr)[n] = SvU64(*av_fetch(av, n, 1));
                        }
                        break;
                      case FFI_PL_TYPE_SINT64:
                        Newx(ptr, count, int64_t);
                        for(n=0; n<count; n++)
                        {
                          ((int64_t*)ptr)[n] = SvI64(*av_fetch(av, n, 1));
                        }
                        break;
                      case FFI_PL_TYPE_FLOAT:
                        Newx(ptr, count, float);
                        for(n=0; n<count; n++)
                        {
                          ((float*)ptr)[n] = SvNV(*av_fetch(av, n, 1));
                        }
                        break;
                      case FFI_PL_TYPE_DOUBLE:
                        Newx(ptr, count, double);
                        for(n=0; n<count; n++)
                        {
                          ((double*)ptr)[n] = SvNV(*av_fetch(av, n, 1));
                        }
                        break;
                      case FFI_PL_TYPE_OPAQUE:
                        Newx(ptr, count, void*);
                        for(n=0; n<count; n++)
                        {
                          SV *sv = *av_fetch(av, n, 1);
                          ((void**)ptr)[n] = SvOK(sv) ? INT2PTR(void*, SvIV(sv)) : NULL;
                        }
                        break;
#ifdef FFI_PL_PROBE_LONGDOUBLE
                      case FFI_PL_TYPE_LONG_DOUBLE:
                        /* gh#236: lets hope the compiler is smart enough to opitmize this */
                        if(sizeof(long double) >= 16)
                        {
                          Newx(ptr, count, long double);
                        }
                        else
                        {
                          Newx(ptr, count*16, char);
                        }
                        for(n=0; n<count; n++)
                        {
                          SV *sv = *av_fetch(av, n, 1);
                          ffi_pl_perl_to_long_double(sv, &((long double*)ptr)[n]);
                        }
                        break;
#endif
#ifdef FFI_PL_PROBE_COMPLEX
                      case FFI_PL_TYPE_COMPLEX_FLOAT:
                        Newx(ptr, count, float complex);
                        for(n=0; n<count; n++)
                        {
                          SV *sv = *av_fetch(av, n, 1);
                          ffi_pl_perl_to_complex_float(sv, &((float*)ptr)[n*2]);
                        }
                        break;
                      case FFI_PL_TYPE_COMPLEX_DOUBLE:
                        Newx(ptr, count, double complex);
                        for(n=0; n<count; n++)
                        {
                          SV *sv = *av_fetch(av, n, 1);
                          ffi_pl_perl_to_complex_double(sv, &((double*)ptr)[n*2]);
                        }
                        break;
#endif
                      case FFI_PL_TYPE_STRING:
                        Newx(ptr, count, char *);
                        for(n=0; n<count; n++)
                        {
                          SV *sv = *av_fetch(av, n, 1);
                          if(SvOK(sv))
                          {
                            char *str;
                            char *pv;
                            STRLEN len;
                            pv = SvPV(sv, len);
                            /* TODO: this should probably be a malloc since it could be arbitrarily large */
                            Newx_or_alloca(str, len+1, char);
                            memcpy(str, pv, len+1);
                            ((char**)ptr)[n] = str;
                          }
                          else
                          {
                            ((char**)ptr)[n] = NULL;
                          }
                        }
                        break;
                      default:
                        Newxz(ptr, count*(1 << ((type_code & FFI_PL_SIZE_MASK)-1)), char);
                        warn("argument type not supported (%d)", i);
                        break;
                    }
                    ffi_pl_heap_add_ptr(ptr);
                  }
                  else
                  {
                    is_bad = 1;
                  }
                }
                else
                {
                  if(is_pointer)
                  {
                    ptr = NULL;
                  }
                  else
                  {
                    is_bad = 1;
                  }
                }
                if(is_bad)
                {
                  if(is_pointer)
                  {
                    if(self->platypus_api >= 2)
                    {
                      warn("argument type not a reference to scalar or array (%d)", i);
                    }
                    else
                    {
                      warn("argument type not a reference to scalar (%d)", i);
                    }
                  }
                  else
                  {
                    warn("passing non array reference into ffi/platypus array argument type");
                    count = self->argument_types[i]->extra[0].array.element_count;
                    Newxz(ptr, count*(1 << ((type_code & FFI_PL_SIZE_MASK)-1)), char);
                    ffi_pl_heap_add_ptr(ptr);
                  }
                }
                ffi_pl_arguments_set_pointer(arguments, i, ptr);
              }
              break;

/*
 * ARGUMENT IN - OBJECT
 */

            case FFI_PL_SHAPE_OBJECT:
              {
                if(sv_isobject(arg) && sv_derived_from(arg, self->argument_types[i]->extra[0].object.class))
                {
                  SV *arg2 = SvRV(arg);
                  switch(type_code)
                  {
                    case FFI_PL_TYPE_UINT8 | FFI_PL_SHAPE_OBJECT:
                      ffi_pl_arguments_set_uint8(arguments, i, SvUV(arg2) );
                      break;
                    case FFI_PL_TYPE_SINT8 | FFI_PL_SHAPE_OBJECT:
                      ffi_pl_arguments_set_sint8(arguments, i, SvIV(arg2) );
                      break;
                    case FFI_PL_TYPE_UINT16 | FFI_PL_SHAPE_OBJECT:
                      ffi_pl_arguments_set_uint16(arguments, i, SvUV(arg2) );
                      break;
                    case FFI_PL_TYPE_SINT16 | FFI_PL_SHAPE_OBJECT:
                      ffi_pl_arguments_set_sint16(arguments, i, SvIV(arg2) );
                      break;
                    case FFI_PL_TYPE_UINT32 | FFI_PL_SHAPE_OBJECT:
                      ffi_pl_arguments_set_uint32(arguments, i, SvUV(arg2) );
                      break;
                    case FFI_PL_TYPE_SINT32 | FFI_PL_SHAPE_OBJECT:
                      ffi_pl_arguments_set_sint32(arguments, i, SvIV(arg2) );
                      break;
                    case FFI_PL_TYPE_UINT64 | FFI_PL_SHAPE_OBJECT:
                      ffi_pl_arguments_set_uint64(arguments, i, SvU64(arg2) );
                      break;
                    case FFI_PL_TYPE_SINT64 | FFI_PL_SHAPE_OBJECT:
                      ffi_pl_arguments_set_sint64(arguments, i, SvI64(arg2) );
                      break;
                    case FFI_PL_TYPE_OPAQUE | FFI_PL_SHAPE_OBJECT:
                      ffi_pl_arguments_set_pointer(arguments, i, SvOK(arg2) ? INT2PTR(void*, SvIV(arg2)) : NULL);
                      break;
                    default:
                      ffi_pl_croak("Object argument %d type not supported %d", i, type_code);
                  }
                }
                else
                {
                  ffi_pl_croak("Object argument %d must be an object of class %s", i, self->argument_types[i]->extra[0].object.class);
                }
              }
              break;

/*
 * ARGUMENT IN - UNSUPPORTED
 */

            default:
              warn("FFI::Platypus: argument %d type not supported (%04x)", i, type_code);
              break;
          }
      }

      if(custom_flag)
      {
        int max = self->argument_types[i]->extra[0].custom_perl.argument_count;
        SvREFCNT_dec(arg);
        for(n=0; n < max; n++)
        {
          i++;
          argument_pointers[i] = &arguments->slot[i];
        }
      }
    }

    /*
     * CALL
     */

#if 0
    fprintf(stderr, "# ===[%p]===\n", self->address);
    for(i=0; i < self->ffi_cif.nargs; i++)
    {
      fprintf(stderr, "# [%d] <%04x> %p %p",
        i,
        self->argument_types[i]->type_code,
        argument_pointers[i],
        &arguments->slot[i]
      );
      switch(self->argument_types[i]->type_code)
      {
        case FFI_PL_TYPE_LONG_DOUBLE:
          fprintf(stderr, " %Lg", *((long double*)argument_pointers[i]));
          break;
        case FFI_PL_TYPE_COMPLEX_FLOAT:
          fprintf(stderr, " %g + %g * i",
            crealf(*((float complex*)argument_pointers[i])),
            cimagf(*((float complex*)argument_pointers[i]))
          );
          break;
        case FFI_PL_TYPE_COMPLEX_DOUBLE:
          fprintf(stderr, " %g + %g * i",
            creal(*((double complex*)argument_pointers[i])),
            cimag(*((double complex*)argument_pointers[i]))
          );
          break;
        default:
          fprintf(stderr, "%016llx", ffi_pl_arguments_get_uint64(arguments, i));
          break;
      }
      fprintf(stderr, "\n");
    }
    fprintf(stderr, "# === ===\n");
    fflush(stderr);
#endif

    MY_CXT.current_argv = NULL;

    ffi_call(&self->ffi_cif, self->address, RESULT, ffi_pl_arguments_pointers(arguments));

/*
 * ARGUMENT OUT
 */

    MY_CXT.current_argv = arguments;

    for(i=self->ffi_cif.nargs-1,perl_arg_index--; i >= 0; i--, perl_arg_index--)
    {
      int type_code = self->argument_types[i]->type_code;

      switch(type_code)
      {

/*
 * ARGUMENT OUT - SCALAR TYPES
 */

        case FFI_PL_TYPE_CLOSURE:
          {
            arg = perl_arg_index < items ? ST(perl_arg_index) : &PL_sv_undef;
            if(SvROK(arg))
            {
              SvREFCNT_dec(arg);
              SvREFCNT_dec(SvRV(arg));
            }
          }
          break;

        default:
          switch(type_code & FFI_PL_SHAPE_MASK)
          {

/*
 * ARGUMENT OUT - POINTER & ARRAY TYPES
 */

            case FFI_PL_SHAPE_POINTER:
            case FFI_PL_SHAPE_ARRAY:
              {
                void *ptr = ffi_pl_arguments_get_pointer(arguments, i);
                arg = perl_arg_index < items ? ST(perl_arg_index) : &PL_sv_undef;
                if(ptr != NULL && SvOK(arg))
                {
                  SV *arg2 = SvROK(arg) ? SvRV(arg) : &PL_sv_undef;
                  if(SvTYPE(arg2) == SVt_PVAV)
                  {
                    SSize_t count = 0;
                    AV *av = (AV*)arg2;
                    if((type_code & FFI_PL_SHAPE_MASK) == FFI_PL_SHAPE_ARRAY)
                    {
                      count = self->argument_types[i]->extra[0].array.element_count;
                    }
                    if(count == 0)
                    {
                      count = av_len(av)+1;
                    }
                    switch(type_code & (FFI_PL_BASE_MASK|FFI_PL_SIZE_MASK))
                    {
                      case FFI_PL_TYPE_UINT8:
                        for(n=0; n<count; n++)
                        {
                          sv_setuv(*av_fetch(av, n, 1), ((uint8_t*)ptr)[n]);
                        }
                        break;
                      case FFI_PL_TYPE_SINT8:
                        for(n=0; n<count; n++)
                        {
                          sv_setiv(*av_fetch(av, n, 1), ((int8_t*)ptr)[n]);
                        }
                        break;
                      case FFI_PL_TYPE_UINT16:
                        for(n=0; n<count; n++)
                        {
                          sv_setuv(*av_fetch(av, n, 1), ((uint16_t*)ptr)[n]);
                        }
                        break;
                      case FFI_PL_TYPE_SINT16:
                        for(n=0; n<count; n++)
                        {
                          sv_setiv(*av_fetch(av, n, 1), ((int16_t*)ptr)[n]);
                        }
                        break;
                      case FFI_PL_TYPE_UINT32:
                        for(n=0; n<count; n++)
                        {
                          sv_setuv(*av_fetch(av, n, 1), ((uint32_t*)ptr)[n]);
                        }
                        break;
                      case FFI_PL_TYPE_SINT32:
                        for(n=0; n<count; n++)
                        {
                          sv_setiv(*av_fetch(av, n, 1), ((int32_t*)ptr)[n]);
                        }
                        break;
                      case FFI_PL_TYPE_UINT64:
                        for(n=0; n<count; n++)
                        {
                          sv_setu64(*av_fetch(av, n, 1), ((uint64_t*)ptr)[n]);
                        }
                        break;
                      case FFI_PL_TYPE_SINT64:
                        for(n=0; n<count; n++)
                        {
                          sv_seti64(*av_fetch(av, n, 1), ((int64_t*)ptr)[n]);
                        }
                        break;
                      case FFI_PL_TYPE_FLOAT:
                        for(n=0; n<count; n++)
                        {
                          sv_setnv(*av_fetch(av, n, 1), ((float*)ptr)[n]);
                        }
                        break;
                      case FFI_PL_TYPE_OPAQUE:
                      case FFI_PL_TYPE_STRING:
                        for(n=0; n<count; n++)
                        {
                          if( ((void**)ptr)[n] == NULL)
                          {
                            av_store(av, n, &PL_sv_undef);
                          }
                          else
                          {
                            switch(type_code) {
                              case FFI_PL_TYPE_OPAQUE | FFI_PL_SHAPE_ARRAY:
                                sv_setnv(*av_fetch(av,n,1), PTR2IV( ((void**)ptr)[n]) );
                                break;
                              case FFI_PL_TYPE_STRING | FFI_PL_SHAPE_ARRAY:
                                sv_setpv(*av_fetch(av,n,1), ((char**)ptr)[n] );
                                break;
                            }
                          }
                        }
                        break;
                      case FFI_PL_TYPE_DOUBLE:
                        for(n=0; n<count; n++)
                        {
                          sv_setnv(*av_fetch(av, n, 1), ((double*)ptr)[n]);
                        }
                        break;
#ifdef FFI_PL_PROBE_LONGDOUBLE
                      case FFI_PL_TYPE_LONG_DOUBLE:
                        for(n=0; n<count; n++)
                        {
                          SV *sv;
                          sv = *av_fetch(av, n, 1);
                          ffi_pl_long_double_to_perl(sv, &((long double*)ptr)[n]);
                        }
                        break;
#endif
#ifdef FFI_PL_PROBE_COMPLEX
                      case FFI_PL_TYPE_COMPLEX_DOUBLE:
                        for(n=0; n<count; n++)
                        {
                          SV *sv;
                          sv = *av_fetch(av, n, 1);
                          ffi_pl_complex_double_to_perl(sv, &((double*)ptr)[n*2]);
                        }
                        break;
                      case FFI_PL_TYPE_COMPLEX_FLOAT:
                        for(n=0; n<count; n++)
                        {
                          SV *sv;
                          sv = *av_fetch(av, n, 1);
                          ffi_pl_complex_float_to_perl(sv, &((float*)ptr)[n*2]);
                        }
                        break;
#endif
                    }
                  }
                  else if(SvTYPE(arg2) < SVt_PVAV && !SvREADONLY(arg2))
                  {
                    switch(type_code & (FFI_PL_BASE_MASK|FFI_PL_SIZE_MASK))
                    {
                      case FFI_PL_TYPE_UINT8:
                        sv_setuv(arg2, *((uint8_t*)ptr));
                        break;
                      case FFI_PL_TYPE_SINT8:
                        sv_setiv(arg2, *((int8_t*)ptr));
                        break;
                      case FFI_PL_TYPE_UINT16:
                        sv_setuv(arg2, *((uint16_t*)ptr));
                        break;
                      case FFI_PL_TYPE_SINT16:
                        sv_setiv(arg2, *((int16_t*)ptr));
                        break;
                      case FFI_PL_TYPE_UINT32:
                        sv_setuv(arg2, *((uint32_t*)ptr));
                        break;
                      case FFI_PL_TYPE_SINT32:
                        sv_setiv(arg2, *((int32_t*)ptr));
                        break;
                      case FFI_PL_TYPE_UINT64:
                        sv_setu64(arg2, *((uint64_t*)ptr));
                        break;
                      case FFI_PL_TYPE_SINT64:
                        sv_seti64(arg2, *((int64_t*)ptr));
                        break;
                      case FFI_PL_TYPE_FLOAT:
                        sv_setnv(arg2, *((float*)ptr));
                        break;
                      case FFI_PL_TYPE_OPAQUE:
                        if( *((void**)ptr) == NULL)
                          sv_setsv(arg2, &PL_sv_undef);
                        else
                          sv_setiv(arg2, PTR2IV(*((void**)ptr)));
                        break;
                      case FFI_PL_TYPE_DOUBLE:
                        sv_setnv(arg2, *((double*)ptr));
                        break;
#ifdef FFI_PL_PROBE_LONGDOUBLE
                      case FFI_PL_TYPE_LONG_DOUBLE:
                        ffi_pl_long_double_to_perl(arg2,(long double*)ptr);
                        break;
#endif
#ifdef FFI_PL_PROBE_COMPLEX
                      case FFI_PL_TYPE_COMPLEX_FLOAT:
                        ffi_pl_complex_float_to_perl(arg2, (float *)ptr);
                        break;
                      case FFI_PL_TYPE_COMPLEX_DOUBLE:
                        ffi_pl_complex_double_to_perl(arg2, (double *)ptr);
                        break;
#endif
                      case FFI_PL_TYPE_STRING:
                        {
                          char **pv = ptr;
                          if(*pv == NULL)
                          {
                            sv_setsv(arg2, &PL_sv_undef);
                          }
                          else
                          {
                            sv_setpv(arg2, *pv);
                          }
                        }
                        break;
                    }
                  }
                }
              }
              break;

/*
 * ARGUMENT OUT - CUSTOM TYPE
 */

            case FFI_PL_SHAPE_CUSTOM_PERL:
              {
                /* FIXME: need to fill out argument_types for skipping */
                i -= self->argument_types[i]->extra[0].custom_perl.argument_count;
                {
                  SV *coderef = self->argument_types[i]->extra[0].custom_perl.perl_to_native_post;
                  if(coderef != NULL)
                  {
                    arg = perl_arg_index < items ? ST(perl_arg_index) : &PL_sv_undef;
                    ffi_pl_custom_perl_cb(coderef, arg, i);
                  }
                }
                {
                  SV *sv = av_pop(MY_CXT.custom_keepers);
                  if(SvOK(sv))
                    SvREFCNT_dec(sv);
                }
              }
              break;

            default:
              break;
          }
      }
    }

    {

      int type_code = self->return_type->type_code;

      /*
       * TODO: This should always happen later if possible
       */
      if((type_code & FFI_PL_SHAPE_MASK) != FFI_PL_SHAPE_CUSTOM_PERL
      &&  type_code != FFI_PL_TYPE_RECORD_VALUE)
        ffi_pl_heap_free();

      MY_CXT.current_argv = NULL;

/*
 * RETURN VALUE
 */

      switch(type_code)
      {

/*
 * RETURN VALUE - TYPE SCALAR
 */


#if ! FFI_PL_CALL_NO_RECORD_VALUE
        case FFI_PL_TYPE_RECORD_VALUE:
          {
            SV *value, *ref;
            value = newSV(0);
            sv_setpvn(value, result_ptr, self->return_type->extra[0].record.size);
            ref = ST(0) = sv_2mortal(newRV_noinc(value));
            sv_bless(ref, gv_stashpv(self->return_type->extra[0].record.class, GV_ADD));
            ffi_pl_heap_free();
            XSRETURN(1);
          }
          break;
        case FFI_PL_TYPE_RECORD_VALUE | FFI_PL_SHAPE_CUSTOM_PERL:
          {
            SV *value, *ref;
            value = newSV(0);
            sv_setpvn(value, result_ptr, self->return_type->extra[0].record.size);
            ref = sv_2mortal(newRV_noinc(value));
            sv_bless(ref, gv_stashpv(self->return_type->extra[0].record.class, GV_ADD));

            MY_CXT.current_argv = arguments;

            ST(0) = ffi_pl_custom_perl(
              self->return_type->extra[0].custom_perl.native_to_perl,
              ref,
              -1
            );

            MY_CXT.current_argv = NULL;

            ffi_pl_heap_free();
            XSRETURN(1);
          }
          break;
#endif
#if ! FFI_PL_CALL_RET_NO_NORMAL
        case FFI_PL_TYPE_VOID:
          XSRETURN_EMPTY;
          break;
        case FFI_PL_TYPE_UINT8:
#if defined FFI_PL_PROBE_BIGENDIAN
          XSRETURN_UV(result.uint8_array[3]);
#elif defined FFI_PL_PROBE_BIGENDIAN64
          XSRETURN_UV(result.uint8_array[7]);
#else
          XSRETURN_UV(result.uint8);
#endif
          break;
        case FFI_PL_TYPE_SINT8:
#if defined FFI_PL_PROBE_BIGENDIAN
          XSRETURN_IV(result.sint8_array[3]);
#elif defined FFI_PL_PROBE_BIGENDIAN64
          XSRETURN_IV(result.sint8_array[7]);
#else
          XSRETURN_IV(result.sint8);
#endif
          break;
        case FFI_PL_TYPE_UINT16:
#if defined FFI_PL_PROBE_BIGENDIAN
          XSRETURN_UV(result.uint16_array[1]);
#elif defined FFI_PL_PROBE_BIGENDIAN64
          XSRETURN_UV(result.uint16_array[3]);
#else
          XSRETURN_UV(result.uint16);
#endif
          break;
        case FFI_PL_TYPE_SINT16:
#if defined FFI_PL_PROBE_BIGENDIAN
          XSRETURN_IV(result.sint16_array[1]);
#elif defined FFI_PL_PROBE_BIGENDIAN64
          XSRETURN_IV(result.sint16_array[3]);
#else
          XSRETURN_IV(result.sint16);
#endif
          break;
        case FFI_PL_TYPE_UINT32:
#if defined FFI_PL_PROBE_BIGENDIAN64
          XSRETURN_UV(result.uint32_array[1]);
#else
          XSRETURN_UV(result.uint32);
#endif
          break;
        case FFI_PL_TYPE_SINT32:
#if defined FFI_PL_PROBE_BIGENDIAN64
          XSRETURN_IV(result.sint32_array[1]);
#else
          XSRETURN_IV(result.sint32);
#endif
          break;
        case FFI_PL_TYPE_UINT64:
          XSRETURN_U64(result.uint64);
          break;
        case FFI_PL_TYPE_SINT64:
          XSRETURN_I64(result.sint64);
          break;
        case FFI_PL_TYPE_FLOAT:
          XSRETURN_NV(result.xfloat);
          break;
        case FFI_PL_TYPE_DOUBLE:
          XSRETURN_NV(result.xdouble);
          break;
        case FFI_PL_TYPE_OPAQUE:
        case FFI_PL_TYPE_STRING:
          if(result.pointer == NULL)
          {
            if(self->platypus_api >= 2)
            {
              XSRETURN_UNDEF;
            }
            else
            {
              XSRETURN_EMPTY;
            }
          }
          else
          {
            switch(type_code)
            {
              case FFI_PL_TYPE_OPAQUE:
                XSRETURN_IV(PTR2IV(result.pointer));
                break;
              case FFI_PL_TYPE_STRING:
                XSRETURN_PV(result.pointer);
                break;
            }
          }
          break;
#ifdef FFI_PL_PROBE_LONGDOUBLE
        case FFI_PL_TYPE_LONG_DOUBLE:
        {
#if !(defined(USE_LONG_DOUBLE) && defined(HAS_LONG_DOUBLE))
          if(MY_CXT.loaded_math_longdouble == 1)
          {
            SV *sv;
            long double *ptr;
            Newx(ptr, 1, long double);
            *ptr = result.longdouble;
            sv = sv_newmortal();
            sv_setref_pv(sv, "Math::LongDouble", (void*)ptr);
            ST(0) = sv;
            XSRETURN(1);
          }
          else
          {
#endif
            XSRETURN_NV((NV) result.longdouble);
#if !(defined(USE_LONG_DOUBLE) && defined(HAS_LONG_DOUBLE))
          }
#endif
        }
#endif
#ifdef FFI_PL_PROBE_COMPLEX
        case FFI_PL_TYPE_COMPLEX_FLOAT:
          {
            SV *c[2];
            AV *av;

            c[0] = sv_2mortal(newSVnv( ((float*)&result.complex_float)[0]) );
            c[1] = sv_2mortal(newSVnv( ((float*)&result.complex_float)[1]) );
            av = av_make(2,c);
            ST(0) = sv_2mortal(newRV_noinc((SV*) av));
            XSRETURN(1);
          }
          break;
        case FFI_PL_TYPE_COMPLEX_DOUBLE:
          {
            SV *c[2];
            AV *av;

            c[0] = sv_2mortal(newSVnv( ((double*)&result.complex_double)[0]) );
            c[1] = sv_2mortal(newSVnv( ((double*)&result.complex_double)[1]) );
            av = av_make(2,c);
            ST(0) = sv_2mortal(newRV_noinc((SV*) av));
            XSRETURN(1);
          }
          break;
#endif
        case FFI_PL_TYPE_RECORD:
        case FFI_PL_TYPE_RECORD | FFI_PL_SHAPE_CUSTOM_PERL:
          if(result.pointer == NULL)
          {
            if((type_code & FFI_PL_SHAPE_MASK) == FFI_PL_SHAPE_CUSTOM_PERL)
            {
              MY_CXT.current_argv = arguments;

              ST(0) = ffi_pl_custom_perl(
                self->return_type->extra[0].custom_perl.native_to_perl,
                &PL_sv_undef,
                -1
              );

              MY_CXT.current_argv = NULL;
              ffi_pl_heap_free();
              XSRETURN(1);
            }
            if(self->platypus_api >= 2)
            {
              XSRETURN_UNDEF;
            }
            else
            {
              XSRETURN_EMPTY;
            }
          }
          else
          {
            SV *ref;
            SV *value = newSV(0);
            sv_setpvn(value, result.pointer, self->return_type->extra[0].record.size);
            if(self->return_type->extra[0].record.class != NULL)
            {
              ref = sv_2mortal(newRV_noinc(value));
              sv_bless(ref, gv_stashpv(self->return_type->extra[0].record.class, GV_ADD));
            }
            else
            {
              ref = sv_2mortal(value);
            }

            if((type_code & FFI_PL_SHAPE_MASK) == FFI_PL_SHAPE_CUSTOM_PERL)
            {
              MY_CXT.current_argv = arguments;

              ST(0) = ffi_pl_custom_perl(
                self->return_type->extra[0].custom_perl.native_to_perl,
                ref,
                -1
              );

              MY_CXT.current_argv = NULL;
              ffi_pl_heap_free();
            }
            else
            {
              ST(0) = ref;
            }

            XSRETURN(1);
          }
          break;
        case FFI_PL_SHAPE_OBJECT | FFI_PL_TYPE_OPAQUE:
          if(result.pointer == NULL)
          {
            if(self->platypus_api >= 2)
            {
              XSRETURN_UNDEF;
            }
            else
            {
              XSRETURN_EMPTY;
            }
          }
          else
          {
            SV *ref;
            SV *value = newSV(0);
            sv_setiv(value, PTR2IV(((void*)result.pointer)));
            ref = ST(0) = sv_2mortal(newRV_noinc(value));
            sv_bless(ref, gv_stashpv(self->return_type->extra[0].object.class, GV_ADD));
            XSRETURN(1);
          }
          break;
        default:

          switch(type_code & FFI_PL_SHAPE_MASK)
          {

/*
 * RETURN VALUE - TYPE POINTER
 */

            case FFI_PL_SHAPE_POINTER:
              if(result.pointer == NULL)
              {
                if(self->platypus_api >= 2)
                {
                  XSRETURN_UNDEF;
                }
                else
                {
                  XSRETURN_EMPTY;
                }
              }
              else
              {
                SV *value;
                switch(type_code)
                {
                  case FFI_PL_TYPE_UINT8 | FFI_PL_SHAPE_POINTER:
                    value = newSV(0);
                    sv_setuv(value, *((uint8_t*) result.pointer));
                    break;
                  case FFI_PL_TYPE_SINT8 | FFI_PL_SHAPE_POINTER:
                    value = newSV(0);
                    sv_setiv(value, *((int8_t*) result.pointer));
                    break;
                  case FFI_PL_TYPE_UINT16 | FFI_PL_SHAPE_POINTER:
                    value = newSV(0);
                    sv_setuv(value, *((uint16_t*) result.pointer));
                    break;
                  case FFI_PL_TYPE_SINT16 | FFI_PL_SHAPE_POINTER:
                    value = newSV(0);
                    sv_setiv(value, *((int16_t*) result.pointer));
                    break;
                  case FFI_PL_TYPE_UINT32 | FFI_PL_SHAPE_POINTER:
                    value = newSV(0);
                    sv_setuv(value, *((uint32_t*) result.pointer));
                    break;
                  case FFI_PL_TYPE_SINT32 | FFI_PL_SHAPE_POINTER:
                    value = newSV(0);
                    sv_setiv(value, *((int32_t*) result.pointer));
                    break;
                  case FFI_PL_TYPE_UINT64 | FFI_PL_SHAPE_POINTER:
                    value = newSV(0);
                    sv_setu64(value, *((uint64_t*) result.pointer));
                    break;
                  case FFI_PL_TYPE_SINT64 | FFI_PL_SHAPE_POINTER:
                    value = newSV(0);
                    sv_seti64(value, *((int64_t*) result.pointer));
                    break;
                  case FFI_PL_TYPE_FLOAT | FFI_PL_SHAPE_POINTER:
                    value = newSV(0);
                    sv_setnv(value, *((float*) result.pointer));
                    break;
                  case FFI_PL_TYPE_DOUBLE | FFI_PL_SHAPE_POINTER:
                    value = newSV(0);
                    sv_setnv(value, *((double*) result.pointer));
                    break;
                  case FFI_PL_TYPE_OPAQUE | FFI_PL_SHAPE_POINTER:
                    if( *((void**)result.pointer) == NULL )
                      value = &PL_sv_undef;
                    else
                    {
                      value = newSV(0);
                      sv_setiv(value, PTR2IV(*((void**)result.pointer)));
                    }
                    break;
#ifdef FFI_PL_PROBE_LONGDOUBLE
                  case FFI_PL_TYPE_LONG_DOUBLE | FFI_PL_SHAPE_POINTER:
                    value = newSV(0);
                    ffi_pl_long_double_to_perl(value, (long double*)result.pointer);
                    break;
#endif
#ifdef FFI_PL_PROBE_COMPLEX
                  case FFI_PL_TYPE_COMPLEX_FLOAT | FFI_PL_SHAPE_POINTER:
                    {
                      SV *c[2];
                      AV *av;
                      c[0] = sv_2mortal(newSVnv( ((float*)result.pointer)[0] ));
                      c[1] = sv_2mortal(newSVnv( ((float*)result.pointer)[1] ));
                      av = av_make(2, c);
                      value = newRV_noinc((SV*)av);
                    }
                    break;
                  case FFI_PL_TYPE_COMPLEX_DOUBLE | FFI_PL_SHAPE_POINTER:
                    {
                      SV *c[2];
                      AV *av;
                      c[0] = sv_2mortal(newSVnv( ((double*)result.pointer)[0] ));
                      c[1] = sv_2mortal(newSVnv( ((double*)result.pointer)[1] ));
                      av = av_make(2, c);
                      value = newRV_noinc((SV*)av);
                    }
                    break;
#endif
                  case FFI_PL_TYPE_STRING | FFI_PL_SHAPE_POINTER:
                    value = newSV(0);
                    if( *((void**)result.pointer) == NULL )
                      value = &PL_sv_undef;
                    else
                      sv_setpv(value, (char*) result.pointer);
                    break;
                  default:
                    warn("return type not supported");
                    XSRETURN_EMPTY;
                }
                ST(0) = sv_2mortal(newRV_noinc(value));
                XSRETURN(1);
              }
              break;

/*
 * RETURN VALUE - TYPE ARRAY
 */

            case FFI_PL_SHAPE_ARRAY:
              if(result.pointer == NULL)
              {
                if(self->platypus_api >= 2)
                {
                  XSRETURN_UNDEF;
                }
                else
                {
                  XSRETURN_EMPTY;
                }
              }
              else
              {
                int count = self->return_type->extra[0].array.element_count;
                if(count == 0 && type_code & FFI_PL_TYPE_OPAQUE)
                {
                  while(((void**)result.pointer)[count] != NULL)
                    count++;
                }
                AV *av;
                SV **sv;
                Newx(sv, count, SV*);
                switch(type_code)
                {
                  case FFI_PL_TYPE_UINT8 | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      sv[i] = sv_2mortal( newSVuv( ((uint8_t*)result.pointer)[i] ));
                    }
                    break;
                  case FFI_PL_TYPE_SINT8 | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      sv[i] = sv_2mortal( newSViv( ((int8_t*)result.pointer)[i] ) );
                    }
                    break;
                  case FFI_PL_TYPE_UINT16 | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      sv[i] = sv_2mortal( newSVuv( ((uint16_t*)result.pointer)[i] ) );
                    }
                    break;
                  case FFI_PL_TYPE_SINT16 | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      sv[i] = sv_2mortal( newSViv( ((int16_t*)result.pointer)[i] ) );
                    }
                    break;
                  case FFI_PL_TYPE_UINT32 | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      sv[i] = sv_2mortal( newSVuv( ((uint32_t*)result.pointer)[i] ) );
                    }
                    break;
                  case FFI_PL_TYPE_SINT32 | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      sv[i] = sv_2mortal( newSViv( ((int32_t*)result.pointer)[i] ) );
                    }
                    break;
                  case FFI_PL_TYPE_UINT64 | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      sv[i] = sv_2mortal( newSVu64( ((uint64_t*)result.pointer)[i] ) );
                    }
                    break;
                  case FFI_PL_TYPE_SINT64 | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      sv[i] = sv_2mortal( newSVi64( ((int64_t*)result.pointer)[i] ) );
                    }
                    break;
                  case FFI_PL_TYPE_FLOAT | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      sv[i] = sv_2mortal( newSVnv( ((float*)result.pointer)[i] ) );
                    }
                    break;
                  case FFI_PL_TYPE_DOUBLE | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      sv[i] = sv_2mortal( newSVnv( ((double*)result.pointer)[i] ) );
                    }
                    break;
                  case FFI_PL_TYPE_STRING | FFI_PL_SHAPE_ARRAY:
                  case FFI_PL_TYPE_OPAQUE | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      if( ((void**)result.pointer)[i] == NULL)
                      {
                        sv[i] = &PL_sv_undef;
                      }
                      else
                      {
                        switch(type_code) {
                          case FFI_PL_TYPE_STRING | FFI_PL_SHAPE_ARRAY:
                            sv[i] = sv_2mortal( newSVpv( ((char**)result.pointer)[i], 0 ) );
                            break;
                          case FFI_PL_TYPE_OPAQUE | FFI_PL_SHAPE_ARRAY:
                            sv[i] = sv_2mortal( newSViv( PTR2IV( ((void**)result.pointer)[i] )) );
                            break;
                        }
                      }
                    }
                    break;
#ifdef FFI_PL_PROBE_LONGDOUBLE
                  case FFI_PL_TYPE_LONG_DOUBLE | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      sv[i] = sv_2mortal(newSV(0));
                      ffi_pl_long_double_to_perl(sv[i], &((long double*)result.pointer)[i]);
                    }
                    break;
#endif
#ifdef FFI_PL_PROBE_COMPLEX
                  case FFI_PL_TYPE_COMPLEX_FLOAT | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      SV *c[2];
                      AV *av;
                      c[0] = sv_2mortal(newSVnv(((float*)result.pointer)[i*2]));
                      c[1] = sv_2mortal(newSVnv(((float*)result.pointer)[i*2+1]));
                      av = av_make(2, c);
                      sv[i] = sv_2mortal(newRV_noinc((SV*)av));
                    }
                    break;
                  case FFI_PL_TYPE_COMPLEX_DOUBLE | FFI_PL_SHAPE_ARRAY:
                    for(i=0; i<count; i++)
                    {
                      SV *c[2];
                      AV *av;
                      c[0] = sv_2mortal(newSVnv(((double*)result.pointer)[i*2]));
                      c[1] = sv_2mortal(newSVnv(((double*)result.pointer)[i*2+1]));
                      av = av_make(2, c);
                      sv[i] = sv_2mortal(newRV_noinc((SV*)av));
                    }
                    break;
#endif
                  default:
                    warn("return type not supported");
                    XSRETURN_EMPTY;
                }
                av = av_make(count, sv);
                Safefree(sv);
                ST(0) = sv_2mortal(newRV_noinc((SV*)av));
                XSRETURN(1);
              }
              break;

/*
 * RETURN VALUE - CUSTOM PERL
 */

            case FFI_PL_SHAPE_CUSTOM_PERL:
              {
                SV *ret_in=NULL, *ret_out;
                switch(type_code)
                {
                  /* TODO: FFI_PL_BASE_VOID, FFI_PL_BASE_COMPLEX, FFI_PL_BASE_STRING, FFI_PL_BASE_CLOSURE, FFI_PL_BASE_RECORD */
                  case FFI_PL_TYPE_UINT8 | FFI_PL_SHAPE_CUSTOM_PERL:
#if defined FFI_PL_PROBE_BIGENDIAN
                    ret_in = newSVuv(result.uint8_array[3]);
#elif defined FFI_PL_PROBE_BIGENDIAN64
                    ret_in = newSVuv(result.uint8_array[7]);
#else
                    ret_in = newSVuv(result.uint8);
#endif
                    break;
                  case FFI_PL_TYPE_SINT8 | FFI_PL_SHAPE_CUSTOM_PERL:
#if defined FFI_PL_PROBE_BIGENDIAN
                    ret_in = newSViv(result.sint8_array[3]);
#elif defined FFI_PL_PROBE_BIGENDIAN64
                    ret_in = newSViv(result.sint8_array[7]);
#else
                    ret_in = newSViv(result.sint8);
#endif
                    break;
                  case FFI_PL_TYPE_UINT16 | FFI_PL_SHAPE_CUSTOM_PERL:
#if defined FFI_PL_PROBE_BIGENDIAN
                    ret_in = newSVuv(result.uint16_array[1]);
#elif defined FFI_PL_PROBE_BIGENDIAN64
                    ret_in = newSVuv(result.uint16_array[3]);
#else
                    ret_in = newSVuv(result.uint16);
#endif
                    break;
                  case FFI_PL_TYPE_SINT16 | FFI_PL_SHAPE_CUSTOM_PERL:
#if defined FFI_PL_PROBE_BIGENDIAN
                    ret_in = newSViv(result.sint16_array[1]);
#elif defined FFI_PL_PROBE_BIGENDIAN64
                    ret_in = newSViv(result.sint16_array[3]);
#else
                    ret_in = newSViv(result.sint16);
#endif
                    break;
                  case FFI_PL_TYPE_UINT32 | FFI_PL_SHAPE_CUSTOM_PERL:
#if defined FFI_PL_PROBE_BIGENDIAN64
                    ret_in = newSVuv(result.uint32_array[1]);
#else
                    ret_in = newSVuv(result.uint32);
#endif
                    break;
                  case FFI_PL_TYPE_SINT32 | FFI_PL_SHAPE_CUSTOM_PERL:
#if defined FFI_PL_PROBE_BIGENDIAN64
                    ret_in = newSViv(result.sint32_array[1]);
#else
                    ret_in = newSViv(result.sint32);
#endif
                    break;
                  case FFI_PL_TYPE_UINT64 | FFI_PL_SHAPE_CUSTOM_PERL:
                    ret_in = newSVu64(result.uint64);
                    break;
                  case FFI_PL_TYPE_SINT64 | FFI_PL_SHAPE_CUSTOM_PERL:
                    ret_in = newSVi64(result.sint64);
                    break;
                  case FFI_PL_TYPE_FLOAT | FFI_PL_SHAPE_CUSTOM_PERL:
                    ret_in = newSVnv(result.xfloat);
                    break;
                  case FFI_PL_TYPE_DOUBLE | FFI_PL_SHAPE_CUSTOM_PERL:
                    ret_in = newSVnv(result.xdouble);
                    break;
                  case FFI_PL_TYPE_OPAQUE | FFI_PL_SHAPE_CUSTOM_PERL:
                    if(result.pointer != NULL)
                      ret_in = newSViv(PTR2IV(result.pointer));
                    break;
                  default:
                    ffi_pl_heap_free();
                    warn("return type not supported");
                    XSRETURN_EMPTY;
                }

                MY_CXT.current_argv = arguments;

                ret_out = ffi_pl_custom_perl(
                  self->return_type->extra[0].custom_perl.native_to_perl,
                  ret_in != NULL ? ret_in : &PL_sv_undef,
                  -1
                );

                MY_CXT.current_argv = NULL;

                ffi_pl_heap_free();

                if(ret_in != NULL)
                {
                  SvREFCNT_dec(ret_in);
                }

                if(ret_out == NULL)
                {
                  if(self->platypus_api >= 2)
                  {
                    XSRETURN_UNDEF;
                  }
                  else
                  {
                    XSRETURN_EMPTY;
                  }
                }
                else
                {
                  ST(0) = sv_2mortal(ret_out);
                  XSRETURN(1);
                }

              }
              break;

            case FFI_PL_SHAPE_OBJECT:
              {
                SV *ref;
                SV *value = newSV(0);
                switch(type_code)
                {
                  case FFI_PL_TYPE_SINT8 | FFI_PL_SHAPE_OBJECT:
#if defined FFI_PL_PROBE_BIGENDIAN
                    sv_setiv(value, result.sint8_array[3]);
#elif defined FFI_PL_PROBE_BIGENDIAN64
                    sv_setiv(value, result.sint8_array[7]);
#else
                    sv_setiv(value, result.sint8);
#endif
                    break;
                  case FFI_PL_TYPE_UINT8 | FFI_PL_SHAPE_OBJECT:
#if defined FFI_PL_PROBE_BIGENDIAN
                    sv_setuv(value, result.uint8_array[3]);
#elif defined FFI_PL_PROBE_BIGENDIAN64
                    sv_setuv(value, result.uint8_array[7]);
#else
                    sv_setuv(value, result.uint8);
#endif
                    break;
                  case FFI_PL_TYPE_SINT16 | FFI_PL_SHAPE_OBJECT:
#if defined FFI_PL_PROBE_BIGENDIAN
                    sv_setiv(value, result.sint16_array[1]);
#elif defined FFI_PL_PROBE_BIGENDIAN64
                    sv_setiv(value, result.sint16_array[3]);
#else
                    sv_setiv(value, result.sint16);
#endif
                    break;
                  case FFI_PL_TYPE_UINT16 | FFI_PL_SHAPE_OBJECT:
#if defined FFI_PL_PROBE_BIGENDIAN
                    sv_setiv(value, result.uint16_array[1]);
#elif defined FFI_PL_PROBE_BIGENDIAN64
                    sv_setuv(value, result.uint16_array[3]);
#else
                    sv_setuv(value, result.uint16);
#endif
                    break;
                  case FFI_PL_TYPE_SINT32 | FFI_PL_SHAPE_OBJECT:
#if defined FFI_PL_PROBE_BIGENDIAN64
                    sv_setiv(value, result.sint32_array[1]);
#else
                    sv_setiv(value, result.sint32);
#endif
                    break;
                  case FFI_PL_TYPE_UINT32 | FFI_PL_SHAPE_OBJECT:
#if defined FFI_PL_PROBE_BIGENDIAN64
                    sv_setuv(value, result.uint32_array[1]);
#else
                    sv_setuv(value, result.uint32);
#endif
                    break;
                  case FFI_PL_TYPE_SINT64 | FFI_PL_SHAPE_OBJECT:
                    sv_seti64(value, result.sint64);
                    break;
                  case FFI_PL_TYPE_UINT64 | FFI_PL_SHAPE_OBJECT:
                    sv_setu64(value, result.uint64);
                    break;
                  default:
                    break;
                }
                ref = ST(0) = sv_2mortal(newRV_noinc(value));
                sv_bless(ref, gv_stashpv(self->return_type->extra[0].object.class, GV_ADD));
                XSRETURN(1);
              }
              break;

            default:
              warn("return type not supported");
              XSRETURN_EMPTY;
              break;
          }
#endif
      }
    }

    warn("return type not supported");
    XSRETURN_EMPTY;

#undef EXTRA_ARGS
#undef FFI_PL_CALL_NO_RECORD_VALUE
#undef FFI_PL_CALL_RET_NO_NORMAL
#undef RESULT