123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510 |
- /* -----------------------------------------------------------------*-C-*-
- libffi @VERSION@ - Copyright (c) 2011, 2014 Anthony Green
- - Copyright (c) 1996-2003, 2007, 2008 Red Hat, Inc.
- Permission is hereby granted, free of charge, to any person
- obtaining a copy of this software and associated documentation
- files (the ``Software''), to deal in the Software without
- restriction, including without limitation the rights to use, copy,
- modify, merge, publish, distribute, sublicense, and/or sell copies
- of the Software, and to permit persons to whom the Software is
- furnished to do so, subject to the following conditions:
- The above copyright notice and this permission notice shall be
- included in all copies or substantial portions of the Software.
- THE SOFTWARE IS PROVIDED ``AS IS'', WITHOUT WARRANTY OF ANY KIND,
- EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
- HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
- WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
- DEALINGS IN THE SOFTWARE.
- ----------------------------------------------------------------------- */
- /* -------------------------------------------------------------------
- The basic API is described in the README file.
- The raw API is designed to bypass some of the argument packing
- and unpacking on architectures for which it can be avoided.
- The closure API allows interpreted functions to be packaged up
- inside a C function pointer, so that they can be called as C functions,
- with no understanding on the client side that they are interpreted.
- It can also be used in other cases in which it is necessary to package
- up a user specified parameter and a function pointer as a single
- function pointer.
- The closure API must be implemented in order to get its functionality,
- e.g. for use by gij. Routines are provided to emulate the raw API
- if the underlying platform doesn't allow faster implementation.
- More details on the raw and cloure API can be found in:
- http://gcc.gnu.org/ml/java/1999-q3/msg00138.html
- and
- http://gcc.gnu.org/ml/java/1999-q3/msg00174.html
- -------------------------------------------------------------------- */
- #ifndef LIBFFI_H
- #define LIBFFI_H
- #ifdef __cplusplus
- extern "C" {
- #endif
- /* Specify which architecture libffi is configured for. */
- #ifndef @TARGET@
- #define @TARGET@
- #endif
- /* ---- System configuration information --------------------------------- */
- #include <ffitarget.h>
- #ifndef LIBFFI_ASM
- #if defined(_MSC_VER) && !defined(__clang__)
- #define __attribute__(X)
- #endif
- #include <stddef.h>
- #include <limits.h>
- /* LONG_LONG_MAX is not always defined (not if STRICT_ANSI, for example).
- But we can find it either under the correct ANSI name, or under GNU
- C's internal name. */
- #define FFI_64_BIT_MAX 9223372036854775807
- #ifdef LONG_LONG_MAX
- # define FFI_LONG_LONG_MAX LONG_LONG_MAX
- #else
- # ifdef LLONG_MAX
- # define FFI_LONG_LONG_MAX LLONG_MAX
- # ifdef _AIX52 /* or newer has C99 LLONG_MAX */
- # undef FFI_64_BIT_MAX
- # define FFI_64_BIT_MAX 9223372036854775807LL
- # endif /* _AIX52 or newer */
- # else
- # ifdef __GNUC__
- # define FFI_LONG_LONG_MAX __LONG_LONG_MAX__
- # endif
- # ifdef _AIX /* AIX 5.1 and earlier have LONGLONG_MAX */
- # ifndef __PPC64__
- # if defined (__IBMC__) || defined (__IBMCPP__)
- # define FFI_LONG_LONG_MAX LONGLONG_MAX
- # endif
- # endif /* __PPC64__ */
- # undef FFI_64_BIT_MAX
- # define FFI_64_BIT_MAX 9223372036854775807LL
- # endif
- # endif
- #endif
- /* The closure code assumes that this works on pointers, i.e. a size_t */
- /* can hold a pointer. */
- typedef struct _ffi_type
- {
- size_t size;
- unsigned short alignment;
- unsigned short type;
- struct _ffi_type **elements;
- } ffi_type;
- #ifndef LIBFFI_HIDE_BASIC_TYPES
- #if SCHAR_MAX == 127
- # define ffi_type_uchar ffi_type_uint8
- # define ffi_type_schar ffi_type_sint8
- #else
- #error "char size not supported"
- #endif
- #if SHRT_MAX == 32767
- # define ffi_type_ushort ffi_type_uint16
- # define ffi_type_sshort ffi_type_sint16
- #elif SHRT_MAX == 2147483647
- # define ffi_type_ushort ffi_type_uint32
- # define ffi_type_sshort ffi_type_sint32
- #else
- #error "short size not supported"
- #endif
- #if INT_MAX == 32767
- # define ffi_type_uint ffi_type_uint16
- # define ffi_type_sint ffi_type_sint16
- #elif INT_MAX == 2147483647
- # define ffi_type_uint ffi_type_uint32
- # define ffi_type_sint ffi_type_sint32
- #elif INT_MAX == 9223372036854775807
- # define ffi_type_uint ffi_type_uint64
- # define ffi_type_sint ffi_type_sint64
- #else
- #error "int size not supported"
- #endif
- #if LONG_MAX == 2147483647
- # if FFI_LONG_LONG_MAX != FFI_64_BIT_MAX
- #error "no 64-bit data type supported"
- # endif
- #elif LONG_MAX != FFI_64_BIT_MAX
- #error "long size not supported"
- #endif
- #if LONG_MAX == 2147483647
- # define ffi_type_ulong ffi_type_uint32
- # define ffi_type_slong ffi_type_sint32
- #elif LONG_MAX == FFI_64_BIT_MAX
- # define ffi_type_ulong ffi_type_uint64
- # define ffi_type_slong ffi_type_sint64
- #else
- #error "long size not supported"
- #endif
- /* Need minimal decorations for DLLs to works on Windows. */
- /* GCC has autoimport and autoexport. Rely on Libtool to */
- /* help MSVC export from a DLL, but always declare data */
- /* to be imported for MSVC clients. This costs an extra */
- /* indirection for MSVC clients using the static version */
- /* of the library, but don't worry about that. Besides, */
- /* as a workaround, they can define FFI_BUILDING if they */
- /* *know* they are going to link with the static library. */
- #if defined _MSC_VER && !defined FFI_BUILDING
- #define FFI_EXTERN extern __declspec(dllimport)
- #else
- #define FFI_EXTERN extern
- #endif
- /* These are defined in types.c */
- FFI_EXTERN ffi_type ffi_type_void;
- FFI_EXTERN ffi_type ffi_type_uint8;
- FFI_EXTERN ffi_type ffi_type_sint8;
- FFI_EXTERN ffi_type ffi_type_uint16;
- FFI_EXTERN ffi_type ffi_type_sint16;
- FFI_EXTERN ffi_type ffi_type_uint32;
- FFI_EXTERN ffi_type ffi_type_sint32;
- FFI_EXTERN ffi_type ffi_type_uint64;
- FFI_EXTERN ffi_type ffi_type_sint64;
- FFI_EXTERN ffi_type ffi_type_float;
- FFI_EXTERN ffi_type ffi_type_double;
- FFI_EXTERN ffi_type ffi_type_pointer;
- #if @HAVE_LONG_DOUBLE@
- FFI_EXTERN ffi_type ffi_type_longdouble;
- #else
- #define ffi_type_longdouble ffi_type_double
- #endif
- #ifdef FFI_TARGET_HAS_COMPLEX_TYPE
- FFI_EXTERN ffi_type ffi_type_complex_float;
- FFI_EXTERN ffi_type ffi_type_complex_double;
- #if @HAVE_LONG_DOUBLE@
- FFI_EXTERN ffi_type ffi_type_complex_longdouble;
- #else
- #define ffi_type_complex_longdouble ffi_type_complex_double
- #endif
- #endif
- #endif /* LIBFFI_HIDE_BASIC_TYPES */
- typedef enum {
- FFI_OK = 0,
- FFI_BAD_TYPEDEF,
- FFI_BAD_ABI
- } ffi_status;
- typedef unsigned FFI_TYPE;
- typedef struct {
- ffi_abi abi;
- unsigned nargs;
- ffi_type **arg_types;
- ffi_type *rtype;
- unsigned bytes;
- unsigned flags;
- #ifdef FFI_EXTRA_CIF_FIELDS
- FFI_EXTRA_CIF_FIELDS;
- #endif
- } ffi_cif;
- #if @HAVE_LONG_DOUBLE_VARIANT@
- /* Used to adjust size/alignment of ffi types. */
- void ffi_prep_types (ffi_abi abi);
- #endif
- /* Used internally, but overridden by some architectures */
- ffi_status ffi_prep_cif_core(ffi_cif *cif,
- ffi_abi abi,
- unsigned int isvariadic,
- unsigned int nfixedargs,
- unsigned int ntotalargs,
- ffi_type *rtype,
- ffi_type **atypes);
- /* ---- Definitions for the raw API -------------------------------------- */
- #ifndef FFI_SIZEOF_ARG
- # if LONG_MAX == 2147483647
- # define FFI_SIZEOF_ARG 4
- # elif LONG_MAX == FFI_64_BIT_MAX
- # define FFI_SIZEOF_ARG 8
- # endif
- #endif
- #ifndef FFI_SIZEOF_JAVA_RAW
- # define FFI_SIZEOF_JAVA_RAW FFI_SIZEOF_ARG
- #endif
- typedef union {
- ffi_sarg sint;
- ffi_arg uint;
- float flt;
- char data[FFI_SIZEOF_ARG];
- void* ptr;
- } ffi_raw;
- #if FFI_SIZEOF_JAVA_RAW == 4 && FFI_SIZEOF_ARG == 8
- /* This is a special case for mips64/n32 ABI (and perhaps others) where
- sizeof(void *) is 4 and FFI_SIZEOF_ARG is 8. */
- typedef union {
- signed int sint;
- unsigned int uint;
- float flt;
- char data[FFI_SIZEOF_JAVA_RAW];
- void* ptr;
- } ffi_java_raw;
- #else
- typedef ffi_raw ffi_java_raw;
- #endif
- void ffi_raw_call (ffi_cif *cif,
- void (*fn)(void),
- void *rvalue,
- ffi_raw *avalue);
- void ffi_ptrarray_to_raw (ffi_cif *cif, void **args, ffi_raw *raw);
- void ffi_raw_to_ptrarray (ffi_cif *cif, ffi_raw *raw, void **args);
- size_t ffi_raw_size (ffi_cif *cif);
- /* This is analogous to the raw API, except it uses Java parameter */
- /* packing, even on 64-bit machines. I.e. on 64-bit machines */
- /* longs and doubles are followed by an empty 64-bit word. */
- void ffi_java_raw_call (ffi_cif *cif,
- void (*fn)(void),
- void *rvalue,
- ffi_java_raw *avalue);
- void ffi_java_ptrarray_to_raw (ffi_cif *cif, void **args, ffi_java_raw *raw);
- void ffi_java_raw_to_ptrarray (ffi_cif *cif, ffi_java_raw *raw, void **args);
- size_t ffi_java_raw_size (ffi_cif *cif);
- /* ---- Definitions for closures ----------------------------------------- */
- #if FFI_CLOSURES
- #ifdef _MSC_VER
- __declspec(align(8))
- #endif
- typedef struct {
- #if @FFI_EXEC_TRAMPOLINE_TABLE@
- void *trampoline_table;
- void *trampoline_table_entry;
- #else
- char tramp[FFI_TRAMPOLINE_SIZE];
- #endif
- ffi_cif *cif;
- void (*fun)(ffi_cif*,void*,void**,void*);
- void *user_data;
- #ifdef __GNUC__
- } ffi_closure __attribute__((aligned (8)));
- #else
- } ffi_closure;
- # ifdef __sgi
- # pragma pack 0
- # endif
- #endif
- void *ffi_closure_alloc (size_t size, void **code);
- void ffi_closure_free (void *);
- ffi_status
- ffi_prep_closure (ffi_closure*,
- ffi_cif *,
- void (*fun)(ffi_cif*,void*,void**,void*),
- void *user_data);
- ffi_status
- ffi_prep_closure_loc (ffi_closure*,
- ffi_cif *,
- void (*fun)(ffi_cif*,void*,void**,void*),
- void *user_data,
- void*codeloc);
- #ifdef __sgi
- # pragma pack 8
- #endif
- typedef struct {
- #if @FFI_EXEC_TRAMPOLINE_TABLE@
- void *trampoline_table;
- void *trampoline_table_entry;
- #else
- char tramp[FFI_TRAMPOLINE_SIZE];
- #endif
- ffi_cif *cif;
- #if !FFI_NATIVE_RAW_API
- /* if this is enabled, then a raw closure has the same layout
- as a regular closure. We use this to install an intermediate
- handler to do the transaltion, void** -> ffi_raw*. */
- void (*translate_args)(ffi_cif*,void*,void**,void*);
- void *this_closure;
- #endif
- void (*fun)(ffi_cif*,void*,ffi_raw*,void*);
- void *user_data;
- } ffi_raw_closure;
- typedef struct {
- #if @FFI_EXEC_TRAMPOLINE_TABLE@
- void *trampoline_table;
- void *trampoline_table_entry;
- #else
- char tramp[FFI_TRAMPOLINE_SIZE];
- #endif
- ffi_cif *cif;
- #if !FFI_NATIVE_RAW_API
- /* if this is enabled, then a raw closure has the same layout
- as a regular closure. We use this to install an intermediate
- handler to do the transaltion, void** -> ffi_raw*. */
- void (*translate_args)(ffi_cif*,void*,void**,void*);
- void *this_closure;
- #endif
- void (*fun)(ffi_cif*,void*,ffi_java_raw*,void*);
- void *user_data;
- } ffi_java_raw_closure;
- ffi_status
- ffi_prep_raw_closure (ffi_raw_closure*,
- ffi_cif *cif,
- void (*fun)(ffi_cif*,void*,ffi_raw*,void*),
- void *user_data);
- ffi_status
- ffi_prep_raw_closure_loc (ffi_raw_closure*,
- ffi_cif *cif,
- void (*fun)(ffi_cif*,void*,ffi_raw*,void*),
- void *user_data,
- void *codeloc);
- ffi_status
- ffi_prep_java_raw_closure (ffi_java_raw_closure*,
- ffi_cif *cif,
- void (*fun)(ffi_cif*,void*,ffi_java_raw*,void*),
- void *user_data);
- ffi_status
- ffi_prep_java_raw_closure_loc (ffi_java_raw_closure*,
- ffi_cif *cif,
- void (*fun)(ffi_cif*,void*,ffi_java_raw*,void*),
- void *user_data,
- void *codeloc);
- #endif /* FFI_CLOSURES */
- #if FFI_GO_CLOSURES
- typedef struct {
- void *tramp;
- ffi_cif *cif;
- void (*fun)(ffi_cif*,void*,void**,void*);
- } ffi_go_closure;
- ffi_status ffi_prep_go_closure (ffi_go_closure*, ffi_cif *,
- void (*fun)(ffi_cif*,void*,void**,void*));
- void ffi_call_go (ffi_cif *cif, void (*fn)(void), void *rvalue,
- void **avalue, void *closure);
- #endif /* FFI_GO_CLOSURES */
- /* ---- Public interface definition -------------------------------------- */
- ffi_status ffi_prep_cif(ffi_cif *cif,
- ffi_abi abi,
- unsigned int nargs,
- ffi_type *rtype,
- ffi_type **atypes);
- ffi_status ffi_prep_cif_var(ffi_cif *cif,
- ffi_abi abi,
- unsigned int nfixedargs,
- unsigned int ntotalargs,
- ffi_type *rtype,
- ffi_type **atypes);
- void ffi_call(ffi_cif *cif,
- void (*fn)(void),
- void *rvalue,
- void **avalue);
- /* Useful for eliminating compiler warnings */
- #define FFI_FN(f) ((void (*)(void))f)
- /* ---- Definitions shared with assembly code ---------------------------- */
- #endif
- /* If these change, update src/mips/ffitarget.h. */
- #define FFI_TYPE_VOID 0
- #define FFI_TYPE_INT 1
- #define FFI_TYPE_FLOAT 2
- #define FFI_TYPE_DOUBLE 3
- #if @HAVE_LONG_DOUBLE@
- #define FFI_TYPE_LONGDOUBLE 4
- #else
- #define FFI_TYPE_LONGDOUBLE FFI_TYPE_DOUBLE
- #endif
- #define FFI_TYPE_UINT8 5
- #define FFI_TYPE_SINT8 6
- #define FFI_TYPE_UINT16 7
- #define FFI_TYPE_SINT16 8
- #define FFI_TYPE_UINT32 9
- #define FFI_TYPE_SINT32 10
- #define FFI_TYPE_UINT64 11
- #define FFI_TYPE_SINT64 12
- #define FFI_TYPE_STRUCT 13
- #define FFI_TYPE_POINTER 14
- #define FFI_TYPE_COMPLEX 15
- /* This should always refer to the last type code (for sanity checks) */
- /* ??? Ideally, anyway. There are assembly files that still depend
- on this not including COMPLEX. */
- #ifdef FFI_TARGET_HAS_COMPLEX_TYPE
- # define FFI_TYPE_LAST FFI_TYPE_COMPLEX
- #else
- # define FFI_TYPE_LAST FFI_TYPE_POINTER
- #endif
- #ifdef __cplusplus
- }
- #endif
- #endif
|