remogatto-ffi 0.5.0
Sign up to get free protection for your applications and to get access to all the features.
- data/LICENSE +51 -0
- data/README.rdoc +69 -0
- data/Rakefile +144 -0
- data/ext/ffi_c/AbstractMemory.c +489 -0
- data/ext/ffi_c/AbstractMemory.h +160 -0
- data/ext/ffi_c/ArrayType.c +129 -0
- data/ext/ffi_c/ArrayType.h +58 -0
- data/ext/ffi_c/AutoPointer.c +61 -0
- data/ext/ffi_c/AutoPointer.h +18 -0
- data/ext/ffi_c/Buffer.c +187 -0
- data/ext/ffi_c/Call.c +431 -0
- data/ext/ffi_c/Call.h +77 -0
- data/ext/ffi_c/DynamicLibrary.c +216 -0
- data/ext/ffi_c/DynamicLibrary.h +22 -0
- data/ext/ffi_c/Function.c +498 -0
- data/ext/ffi_c/Function.h +73 -0
- data/ext/ffi_c/FunctionInfo.c +213 -0
- data/ext/ffi_c/LastError.c +159 -0
- data/ext/ffi_c/LastError.h +18 -0
- data/ext/ffi_c/MemoryPointer.c +178 -0
- data/ext/ffi_c/MemoryPointer.h +20 -0
- data/ext/ffi_c/MethodHandle.c +356 -0
- data/ext/ffi_c/MethodHandle.h +53 -0
- data/ext/ffi_c/Platform.c +59 -0
- data/ext/ffi_c/Platform.h +16 -0
- data/ext/ffi_c/Pointer.c +217 -0
- data/ext/ffi_c/Pointer.h +49 -0
- data/ext/ffi_c/Struct.c +770 -0
- data/ext/ffi_c/Struct.h +80 -0
- data/ext/ffi_c/StructByValue.c +140 -0
- data/ext/ffi_c/StructByValue.h +53 -0
- data/ext/ffi_c/StructLayout.c +450 -0
- data/ext/ffi_c/Type.c +329 -0
- data/ext/ffi_c/Type.h +57 -0
- data/ext/ffi_c/Types.c +103 -0
- data/ext/ffi_c/Types.h +85 -0
- data/ext/ffi_c/Variadic.c +260 -0
- data/ext/ffi_c/compat.h +72 -0
- data/ext/ffi_c/endian.h +40 -0
- data/ext/ffi_c/extconf.rb +34 -0
- data/ext/ffi_c/ffi.c +80 -0
- data/ext/ffi_c/libffi.bsd.mk +34 -0
- data/ext/ffi_c/libffi.darwin.mk +75 -0
- data/ext/ffi_c/libffi.gnu.mk +29 -0
- data/ext/ffi_c/libffi.mk +13 -0
- data/ext/ffi_c/libffi/ChangeLog +3243 -0
- data/ext/ffi_c/libffi/ChangeLog.libffi +347 -0
- data/ext/ffi_c/libffi/ChangeLog.libgcj +40 -0
- data/ext/ffi_c/libffi/ChangeLog.v1 +764 -0
- data/ext/ffi_c/libffi/LICENSE +21 -0
- data/ext/ffi_c/libffi/Makefile.am +177 -0
- data/ext/ffi_c/libffi/Makefile.in +1640 -0
- data/ext/ffi_c/libffi/README +328 -0
- data/ext/ffi_c/libffi/TODO +1 -0
- data/ext/ffi_c/libffi/acinclude.m4 +92 -0
- data/ext/ffi_c/libffi/aclocal.m4 +7516 -0
- data/ext/ffi_c/libffi/compile +142 -0
- data/ext/ffi_c/libffi/config.guess +1516 -0
- data/ext/ffi_c/libffi/config.sub +1626 -0
- data/ext/ffi_c/libffi/configure +24414 -0
- data/ext/ffi_c/libffi/configure.ac +365 -0
- data/ext/ffi_c/libffi/configure.host +11 -0
- data/ext/ffi_c/libffi/depcomp +584 -0
- data/ext/ffi_c/libffi/doc/libffi.info +533 -0
- data/ext/ffi_c/libffi/doc/libffi.texi +541 -0
- data/ext/ffi_c/libffi/doc/stamp-vti +4 -0
- data/ext/ffi_c/libffi/doc/version.texi +4 -0
- data/ext/ffi_c/libffi/fficonfig.h.in +160 -0
- data/ext/ffi_c/libffi/include/Makefile.am +9 -0
- data/ext/ffi_c/libffi/include/Makefile.in +422 -0
- data/ext/ffi_c/libffi/include/ffi.h.in +393 -0
- data/ext/ffi_c/libffi/include/ffi_common.h +98 -0
- data/ext/ffi_c/libffi/install-sh +323 -0
- data/ext/ffi_c/libffi/libffi.pc.in +10 -0
- data/ext/ffi_c/libffi/libtool-version +29 -0
- data/ext/ffi_c/libffi/ltcf-c.sh +861 -0
- data/ext/ffi_c/libffi/ltcf-cxx.sh +1069 -0
- data/ext/ffi_c/libffi/ltcf-gcj.sh +700 -0
- data/ext/ffi_c/libffi/ltconfig +2862 -0
- data/ext/ffi_c/libffi/ltmain.sh +6930 -0
- data/ext/ffi_c/libffi/man/Makefile.am +8 -0
- data/ext/ffi_c/libffi/man/Makefile.in +395 -0
- data/ext/ffi_c/libffi/man/ffi.3 +31 -0
- data/ext/ffi_c/libffi/man/ffi_call.3 +103 -0
- data/ext/ffi_c/libffi/man/ffi_prep_cif.3 +66 -0
- data/ext/ffi_c/libffi/mdate-sh +201 -0
- data/ext/ffi_c/libffi/missing +353 -0
- data/ext/ffi_c/libffi/mkinstalldirs +158 -0
- data/ext/ffi_c/libffi/src/alpha/ffi.c +284 -0
- data/ext/ffi_c/libffi/src/alpha/ffitarget.h +48 -0
- data/ext/ffi_c/libffi/src/alpha/osf.S +366 -0
- data/ext/ffi_c/libffi/src/arm/ffi.c +309 -0
- data/ext/ffi_c/libffi/src/arm/ffitarget.h +49 -0
- data/ext/ffi_c/libffi/src/arm/sysv.S +299 -0
- data/ext/ffi_c/libffi/src/closures.c +596 -0
- data/ext/ffi_c/libffi/src/cris/ffi.c +383 -0
- data/ext/ffi_c/libffi/src/cris/ffitarget.h +51 -0
- data/ext/ffi_c/libffi/src/cris/sysv.S +215 -0
- data/ext/ffi_c/libffi/src/debug.c +59 -0
- data/ext/ffi_c/libffi/src/dlmalloc.c +5099 -0
- data/ext/ffi_c/libffi/src/frv/eabi.S +128 -0
- data/ext/ffi_c/libffi/src/frv/ffi.c +292 -0
- data/ext/ffi_c/libffi/src/frv/ffitarget.h +61 -0
- data/ext/ffi_c/libffi/src/ia64/ffi.c +580 -0
- data/ext/ffi_c/libffi/src/ia64/ffitarget.h +50 -0
- data/ext/ffi_c/libffi/src/ia64/ia64_flags.h +40 -0
- data/ext/ffi_c/libffi/src/ia64/unix.S +560 -0
- data/ext/ffi_c/libffi/src/java_raw_api.c +359 -0
- data/ext/ffi_c/libffi/src/m32r/ffi.c +232 -0
- data/ext/ffi_c/libffi/src/m32r/ffitarget.h +48 -0
- data/ext/ffi_c/libffi/src/m32r/sysv.S +121 -0
- data/ext/ffi_c/libffi/src/m68k/ffi.c +278 -0
- data/ext/ffi_c/libffi/src/m68k/ffitarget.h +49 -0
- data/ext/ffi_c/libffi/src/m68k/sysv.S +234 -0
- data/ext/ffi_c/libffi/src/mips/ffi.c +926 -0
- data/ext/ffi_c/libffi/src/mips/ffitarget.h +202 -0
- data/ext/ffi_c/libffi/src/mips/n32.S +534 -0
- data/ext/ffi_c/libffi/src/mips/o32.S +381 -0
- data/ext/ffi_c/libffi/src/pa/ffi.c +709 -0
- data/ext/ffi_c/libffi/src/pa/ffitarget.h +77 -0
- data/ext/ffi_c/libffi/src/pa/hpux32.S +368 -0
- data/ext/ffi_c/libffi/src/pa/linux.S +357 -0
- data/ext/ffi_c/libffi/src/powerpc/aix.S +225 -0
- data/ext/ffi_c/libffi/src/powerpc/aix_closure.S +247 -0
- data/ext/ffi_c/libffi/src/powerpc/asm.h +125 -0
- data/ext/ffi_c/libffi/src/powerpc/darwin.S +245 -0
- data/ext/ffi_c/libffi/src/powerpc/darwin_closure.S +317 -0
- data/ext/ffi_c/libffi/src/powerpc/ffi.c +1429 -0
- data/ext/ffi_c/libffi/src/powerpc/ffi_darwin.c +800 -0
- data/ext/ffi_c/libffi/src/powerpc/ffitarget.h +118 -0
- data/ext/ffi_c/libffi/src/powerpc/linux64.S +187 -0
- data/ext/ffi_c/libffi/src/powerpc/linux64_closure.S +236 -0
- data/ext/ffi_c/libffi/src/powerpc/ppc_closure.S +327 -0
- data/ext/ffi_c/libffi/src/powerpc/sysv.S +230 -0
- data/ext/ffi_c/libffi/src/prep_cif.c +174 -0
- data/ext/ffi_c/libffi/src/raw_api.c +254 -0
- data/ext/ffi_c/libffi/src/s390/ffi.c +780 -0
- data/ext/ffi_c/libffi/src/s390/ffitarget.h +60 -0
- data/ext/ffi_c/libffi/src/s390/sysv.S +434 -0
- data/ext/ffi_c/libffi/src/sh/ffi.c +716 -0
- data/ext/ffi_c/libffi/src/sh/ffitarget.h +49 -0
- data/ext/ffi_c/libffi/src/sh/sysv.S +850 -0
- data/ext/ffi_c/libffi/src/sh64/ffi.c +453 -0
- data/ext/ffi_c/libffi/src/sh64/ffitarget.h +53 -0
- data/ext/ffi_c/libffi/src/sh64/sysv.S +530 -0
- data/ext/ffi_c/libffi/src/sparc/ffi.c +610 -0
- data/ext/ffi_c/libffi/src/sparc/ffitarget.h +66 -0
- data/ext/ffi_c/libffi/src/sparc/v8.S +272 -0
- data/ext/ffi_c/libffi/src/sparc/v9.S +307 -0
- data/ext/ffi_c/libffi/src/types.c +77 -0
- data/ext/ffi_c/libffi/src/x86/darwin.S +443 -0
- data/ext/ffi_c/libffi/src/x86/darwin64.S +416 -0
- data/ext/ffi_c/libffi/src/x86/ffi.c +475 -0
- data/ext/ffi_c/libffi/src/x86/ffi64.c +572 -0
- data/ext/ffi_c/libffi/src/x86/ffitarget.h +90 -0
- data/ext/ffi_c/libffi/src/x86/freebsd.S +458 -0
- data/ext/ffi_c/libffi/src/x86/sysv.S +437 -0
- data/ext/ffi_c/libffi/src/x86/unix64.S +418 -0
- data/ext/ffi_c/libffi/src/x86/win32.S +391 -0
- data/ext/ffi_c/libffi/testsuite/Makefile.am +71 -0
- data/ext/ffi_c/libffi/testsuite/Makefile.in +447 -0
- data/ext/ffi_c/libffi/testsuite/config/default.exp +1 -0
- data/ext/ffi_c/libffi/testsuite/lib/libffi-dg.exp +289 -0
- data/ext/ffi_c/libffi/testsuite/lib/target-libpath.exp +263 -0
- data/ext/ffi_c/libffi/testsuite/lib/wrapper.exp +45 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/call.exp +36 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn0.c +97 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn1.c +89 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn2.c +89 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn3.c +90 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn4.c +97 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn5.c +99 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn6.c +98 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/closure_stdcall.c +72 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_12byte.c +102 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_16byte.c +103 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_18byte.c +104 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_19byte.c +110 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_1_1byte.c +97 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_20byte.c +99 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_20byte1.c +101 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_24byte.c +121 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_2byte.c +98 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_3_1byte.c +103 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_3byte1.c +98 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_3byte2.c +98 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_4_1byte.c +106 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_4byte.c +98 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_5_1_byte.c +117 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_5byte.c +106 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_64byte.c +132 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_6_1_byte.c +121 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_6byte.c +107 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_7_1_byte.c +125 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_7byte.c +105 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_8byte.c +96 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_9byte1.c +98 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_9byte2.c +99 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_double.c +101 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_float.c +99 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_longdouble.c +100 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_pointer.c +101 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_sint16.c +99 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_sint32.c +99 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_sint64.c +99 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_uint16.c +99 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_uint32.c +99 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_uint64.c +100 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_double.c +51 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_float.c +51 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_schar.c +82 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_sshort.c +82 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_sshortchar.c +94 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_uchar.c +99 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_ushort.c +82 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_ushortchar.c +94 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_schar.c +52 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_sint.c +50 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_sshort.c +50 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_uchar.c +50 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_uint.c +51 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_ulonglong.c +54 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/cls_ushort.c +51 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/ffitest.h +86 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/float.c +59 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/float1.c +58 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/float2.c +57 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/float3.c +72 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/float4.c +62 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/many.c +69 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/many_win32.c +63 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/negint.c +53 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct.c +160 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct1.c +169 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct10.c +141 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct2.c +118 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct3.c +119 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct4.c +119 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct5.c +120 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct6.c +139 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct7.c +119 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct8.c +139 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct9.c +139 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/problem1.c +98 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/promotion.c +59 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/pyobjc-tc.c +114 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_dbl.c +35 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_dbl1.c +43 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_dbl2.c +42 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_fl.c +35 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_fl1.c +36 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_fl2.c +49 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_fl3.c +42 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_ldl.c +34 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_ll.c +41 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_ll1.c +42 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_sc.c +36 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_sl.c +38 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_uc.c +38 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/return_ul.c +38 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/strlen.c +44 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/strlen_win32.c +44 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/struct1.c +65 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/struct2.c +67 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/struct3.c +59 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/struct4.c +63 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/struct5.c +65 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/struct6.c +64 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/struct7.c +74 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/struct8.c +80 -0
- data/ext/ffi_c/libffi/testsuite/libffi.call/struct9.c +67 -0
- data/ext/ffi_c/libffi/testsuite/libffi.special/ffitestcxx.h +86 -0
- data/ext/ffi_c/libffi/testsuite/libffi.special/special.exp +38 -0
- data/ext/ffi_c/libffi/testsuite/libffi.special/unwindtest.cc +123 -0
- data/ext/ffi_c/libffi/testsuite/libffi.special/unwindtest_ffi_call.cc +53 -0
- data/ext/ffi_c/libffi/texinfo.tex +7482 -0
- data/ext/ffi_c/rbffi.h +26 -0
- data/gen/Rakefile +12 -0
- data/lib/ffi.rb +11 -0
- data/lib/ffi/autopointer.rb +61 -0
- data/lib/ffi/buffer.rb +0 -0
- data/lib/ffi/callback.rb +10 -0
- data/lib/ffi/enum.rb +78 -0
- data/lib/ffi/errno.rb +8 -0
- data/lib/ffi/ffi.rb +99 -0
- data/lib/ffi/io.rb +21 -0
- data/lib/ffi/library.rb +218 -0
- data/lib/ffi/managedstruct.rb +55 -0
- data/lib/ffi/memorypointer.rb +73 -0
- data/lib/ffi/platform.rb +78 -0
- data/lib/ffi/pointer.rb +119 -0
- data/lib/ffi/struct.rb +164 -0
- data/lib/ffi/tools/const_generator.rb +177 -0
- data/lib/ffi/tools/generator.rb +58 -0
- data/lib/ffi/tools/generator_task.rb +35 -0
- data/lib/ffi/tools/struct_generator.rb +194 -0
- data/lib/ffi/tools/types_generator.rb +123 -0
- data/lib/ffi/types.rb +153 -0
- data/lib/ffi/union.rb +12 -0
- data/lib/ffi/variadic.rb +25 -0
- data/spec/ffi/bool_spec.rb +24 -0
- data/spec/ffi/buffer_spec.rb +202 -0
- data/spec/ffi/callback_spec.rb +591 -0
- data/spec/ffi/enum_spec.rb +164 -0
- data/spec/ffi/errno_spec.rb +13 -0
- data/spec/ffi/function_spec.rb +47 -0
- data/spec/ffi/library_spec.rb +144 -0
- data/spec/ffi/managed_struct_spec.rb +56 -0
- data/spec/ffi/number_spec.rb +231 -0
- data/spec/ffi/pointer_spec.rb +195 -0
- data/spec/ffi/rbx/attach_function_spec.rb +27 -0
- data/spec/ffi/rbx/memory_pointer_spec.rb +102 -0
- data/spec/ffi/rbx/spec_helper.rb +1 -0
- data/spec/ffi/rbx/struct_spec.rb +13 -0
- data/spec/ffi/spec_helper.rb +17 -0
- data/spec/ffi/string_spec.rb +103 -0
- data/spec/ffi/struct_callback_spec.rb +62 -0
- data/spec/ffi/struct_initialize_spec.rb +30 -0
- data/spec/ffi/struct_spec.rb +529 -0
- data/spec/ffi/typedef_spec.rb +48 -0
- data/spec/ffi/union_spec.rb +60 -0
- data/spec/ffi/variadic_spec.rb +84 -0
- data/spec/spec.opts +4 -0
- metadata +432 -0
@@ -0,0 +1,216 @@
|
|
1
|
+
#include <sys/types.h>
|
2
|
+
#include <stdio.h>
|
3
|
+
#include <stdint.h>
|
4
|
+
#if defined(_WIN32) || defined(__WIN32__)
|
5
|
+
# include <windows.h>
|
6
|
+
#else
|
7
|
+
# include <dlfcn.h>
|
8
|
+
#endif
|
9
|
+
#include <ruby.h>
|
10
|
+
|
11
|
+
#include <ffi.h>
|
12
|
+
|
13
|
+
#include "rbffi.h"
|
14
|
+
#include "compat.h"
|
15
|
+
#include "AbstractMemory.h"
|
16
|
+
#include "Pointer.h"
|
17
|
+
#include "DynamicLibrary.h"
|
18
|
+
|
19
|
+
typedef struct LibrarySymbol_ {
|
20
|
+
AbstractMemory memory;
|
21
|
+
VALUE library;
|
22
|
+
VALUE name;
|
23
|
+
} LibrarySymbol;
|
24
|
+
|
25
|
+
static VALUE library_initialize(VALUE self, VALUE libname, VALUE libflags);
|
26
|
+
static void library_free(Library* lib);
|
27
|
+
|
28
|
+
|
29
|
+
static VALUE symbol_allocate(VALUE klass);
|
30
|
+
static VALUE symbol_new(VALUE library, void* address, VALUE name);
|
31
|
+
static void symbol_mark(LibrarySymbol* sym);
|
32
|
+
|
33
|
+
static VALUE LibraryClass = Qnil, SymbolClass = Qnil;
|
34
|
+
|
35
|
+
#if defined(_WIN32) || defined(__WIN32__)
|
36
|
+
static void* dl_open(const char* name, int flags);
|
37
|
+
static void dl_error(char* buf, int size);
|
38
|
+
#define dl_sym(handle, name) GetProcAddress(handle, name)
|
39
|
+
#define dl_close(handle) FreeLibrary(handle)
|
40
|
+
enum { RTLD_LAZY=1, RTLD_NOW, RTLD_GLOBAL, RTLD_LOCAL };
|
41
|
+
#else
|
42
|
+
# define dl_open(name, flags) dlopen(name, flags != 0 ? flags : RTLD_LAZY)
|
43
|
+
# define dl_error(buf, size) do { snprintf(buf, size, "%s", dlerror()); } while(0)
|
44
|
+
# define dl_sym(handle, name) dlsym(handle, name)
|
45
|
+
# define dl_close(handle) dlclose(handle)
|
46
|
+
#ifndef RTLD_LOCAL
|
47
|
+
# define RTLD_LOCAL 8
|
48
|
+
#endif
|
49
|
+
#endif
|
50
|
+
|
51
|
+
static VALUE
|
52
|
+
library_allocate(VALUE klass)
|
53
|
+
{
|
54
|
+
Library* library;
|
55
|
+
return Data_Make_Struct(klass, Library, NULL, library_free, library);
|
56
|
+
}
|
57
|
+
|
58
|
+
static VALUE
|
59
|
+
library_open(VALUE klass, VALUE libname, VALUE libflags)
|
60
|
+
{
|
61
|
+
return library_initialize(library_allocate(klass), libname, libflags);
|
62
|
+
}
|
63
|
+
|
64
|
+
static VALUE
|
65
|
+
library_initialize(VALUE self, VALUE libname, VALUE libflags)
|
66
|
+
{
|
67
|
+
Library* library;
|
68
|
+
int flags;
|
69
|
+
|
70
|
+
Check_Type(libflags, T_FIXNUM);
|
71
|
+
|
72
|
+
Data_Get_Struct(self, Library, library);
|
73
|
+
flags = libflags != Qnil ? NUM2UINT(libflags) : 0;
|
74
|
+
|
75
|
+
library->handle = dl_open(libname != Qnil ? StringValueCStr(libname) : NULL, flags);
|
76
|
+
if (library->handle == NULL) {
|
77
|
+
char errmsg[1024];
|
78
|
+
dl_error(errmsg, sizeof(errmsg));
|
79
|
+
rb_raise(rb_eLoadError, "Could not open library '%s': %s",
|
80
|
+
libname != Qnil ? StringValueCStr(libname) : "[current process]",
|
81
|
+
errmsg);
|
82
|
+
}
|
83
|
+
rb_iv_set(self, "@name", libname != Qnil ? libname : rb_str_new2("[current process]"));
|
84
|
+
return self;
|
85
|
+
}
|
86
|
+
|
87
|
+
static VALUE
|
88
|
+
library_dlsym(VALUE self, VALUE name)
|
89
|
+
{
|
90
|
+
Library* library;
|
91
|
+
void* address = NULL;
|
92
|
+
Check_Type(name, T_STRING);
|
93
|
+
|
94
|
+
Data_Get_Struct(self, Library, library);
|
95
|
+
address = dl_sym(library->handle, StringValueCStr(name));
|
96
|
+
|
97
|
+
return address != NULL ? symbol_new(self, address, name) : Qnil;
|
98
|
+
}
|
99
|
+
|
100
|
+
static VALUE
|
101
|
+
library_dlerror(VALUE self)
|
102
|
+
{
|
103
|
+
char errmsg[1024];
|
104
|
+
dl_error(errmsg, sizeof(errmsg));
|
105
|
+
return rb_tainted_str_new2(errmsg);
|
106
|
+
}
|
107
|
+
|
108
|
+
static void
|
109
|
+
library_free(Library* library)
|
110
|
+
{
|
111
|
+
// dlclose() on MacOS tends to segfault - avoid it
|
112
|
+
#ifndef __APPLE__
|
113
|
+
if (library->handle != NULL) {
|
114
|
+
dl_close(library->handle);
|
115
|
+
}
|
116
|
+
#endif
|
117
|
+
xfree(library);
|
118
|
+
}
|
119
|
+
|
120
|
+
#if defined(_WIN32) || defined(__WIN32__)
|
121
|
+
static void*
|
122
|
+
dl_open(const char* name, int flags)
|
123
|
+
{
|
124
|
+
if (name == NULL) {
|
125
|
+
return GetModuleHandle(NULL);
|
126
|
+
} else {
|
127
|
+
return LoadLibraryEx(name, NULL, LOAD_WITH_ALTERED_SEARCH_PATH);
|
128
|
+
}
|
129
|
+
}
|
130
|
+
|
131
|
+
static void
|
132
|
+
dl_error(char* buf, int size)
|
133
|
+
{
|
134
|
+
FormatMessageA(FORMAT_MESSAGE_FROM_SYSTEM, NULL, GetLastError(),
|
135
|
+
0, buf, size, NULL);
|
136
|
+
}
|
137
|
+
#endif
|
138
|
+
|
139
|
+
static VALUE
|
140
|
+
symbol_allocate(VALUE klass)
|
141
|
+
{
|
142
|
+
LibrarySymbol* sym;
|
143
|
+
VALUE obj = Data_Make_Struct(klass, LibrarySymbol, NULL, -1, sym);
|
144
|
+
sym->name = Qnil;
|
145
|
+
sym->library = Qnil;
|
146
|
+
|
147
|
+
return obj;
|
148
|
+
}
|
149
|
+
|
150
|
+
static VALUE
|
151
|
+
symbol_new(VALUE library, void* address, VALUE name)
|
152
|
+
{
|
153
|
+
LibrarySymbol* sym;
|
154
|
+
VALUE obj = Data_Make_Struct(SymbolClass, LibrarySymbol, symbol_mark, -1, sym);
|
155
|
+
|
156
|
+
sym->memory.address = address;
|
157
|
+
sym->memory.size = LONG_MAX;
|
158
|
+
sym->memory.typeSize = 1;
|
159
|
+
sym->memory.access = MEM_RD | MEM_WR;
|
160
|
+
sym->memory.ops = &rbffi_AbstractMemoryOps;
|
161
|
+
sym->library = library;
|
162
|
+
sym->name = name;
|
163
|
+
|
164
|
+
return obj;
|
165
|
+
}
|
166
|
+
|
167
|
+
static void
|
168
|
+
symbol_mark(LibrarySymbol* sym)
|
169
|
+
{
|
170
|
+
rb_gc_mark(sym->library);
|
171
|
+
rb_gc_mark(sym->name);
|
172
|
+
}
|
173
|
+
|
174
|
+
static VALUE
|
175
|
+
symbol_inspect(VALUE self)
|
176
|
+
{
|
177
|
+
LibrarySymbol* sym;
|
178
|
+
char buf[256];
|
179
|
+
|
180
|
+
Data_Get_Struct(self, LibrarySymbol, sym);
|
181
|
+
snprintf(buf, sizeof(buf), "#<FFI::Library::Symbol name=%s address=%p>",
|
182
|
+
StringValueCStr(sym->name), sym->memory.address);
|
183
|
+
return rb_str_new2(buf);
|
184
|
+
}
|
185
|
+
|
186
|
+
void
|
187
|
+
rbffi_DynamicLibrary_Init(VALUE moduleFFI)
|
188
|
+
{
|
189
|
+
LibraryClass = rb_define_class_under(moduleFFI, "DynamicLibrary", rb_cObject);
|
190
|
+
rb_global_variable(&LibraryClass);
|
191
|
+
SymbolClass = rb_define_class_under(LibraryClass, "Symbol", rbffi_PointerClass);
|
192
|
+
rb_global_variable(&SymbolClass);
|
193
|
+
|
194
|
+
rb_define_const(moduleFFI, "NativeLibrary", LibraryClass); // backwards compat library
|
195
|
+
rb_define_alloc_func(LibraryClass, library_allocate);
|
196
|
+
rb_define_singleton_method(LibraryClass, "open", library_open, 2);
|
197
|
+
rb_define_singleton_method(LibraryClass, "last_error", library_dlerror, 0);
|
198
|
+
rb_define_method(LibraryClass, "initialize", library_initialize, 2);
|
199
|
+
rb_define_method(LibraryClass, "find_symbol", library_dlsym, 1);
|
200
|
+
rb_define_method(LibraryClass, "find_function", library_dlsym, 1);
|
201
|
+
rb_define_method(LibraryClass, "find_variable", library_dlsym, 1);
|
202
|
+
rb_define_method(LibraryClass, "last_error", library_dlerror, 0);
|
203
|
+
rb_define_attr(LibraryClass, "name", 1, 0);
|
204
|
+
|
205
|
+
rb_define_alloc_func(SymbolClass, symbol_allocate);
|
206
|
+
rb_undef_method(SymbolClass, "new");
|
207
|
+
rb_define_method(SymbolClass, "inspect", symbol_inspect, 0);
|
208
|
+
|
209
|
+
#define DEF(x) rb_define_const(LibraryClass, "RTLD_" #x, UINT2NUM(RTLD_##x))
|
210
|
+
DEF(LAZY);
|
211
|
+
DEF(NOW);
|
212
|
+
DEF(GLOBAL);
|
213
|
+
DEF(LOCAL);
|
214
|
+
|
215
|
+
}
|
216
|
+
|
@@ -0,0 +1,22 @@
|
|
1
|
+
|
2
|
+
#ifndef _LIBRARY_H
|
3
|
+
#define _LIBRARY_H
|
4
|
+
|
5
|
+
#ifdef __cplusplus
|
6
|
+
extern "C" {
|
7
|
+
#endif
|
8
|
+
|
9
|
+
|
10
|
+
|
11
|
+
typedef struct Library {
|
12
|
+
void* handle;
|
13
|
+
} Library;
|
14
|
+
|
15
|
+
extern void rbffi_DynamicLibrary_Init(VALUE ffiModule);
|
16
|
+
|
17
|
+
#ifdef __cplusplus
|
18
|
+
}
|
19
|
+
#endif
|
20
|
+
|
21
|
+
#endif /* _LIBRARY_H */
|
22
|
+
|
@@ -0,0 +1,498 @@
|
|
1
|
+
/*
|
2
|
+
* Copyright (c) 2009, Wayne Meissner
|
3
|
+
* All rights reserved.
|
4
|
+
*
|
5
|
+
* Redistribution and use in source and binary forms, with or without
|
6
|
+
* modification, are permitted provided that the following conditions are met:
|
7
|
+
*
|
8
|
+
* * Redistributions of source code must retain the above copyright notice, this
|
9
|
+
* list of conditions and the following disclaimer.
|
10
|
+
* * Redistributions in binary form must reproduce the above copyright notice
|
11
|
+
* this list of conditions and the following disclaimer in the documentation
|
12
|
+
* and/or other materials provided with the distribution.
|
13
|
+
* * The name of the author or authors may not be used to endorse or promote
|
14
|
+
* products derived from this software without specific prior written permission.
|
15
|
+
*
|
16
|
+
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
17
|
+
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
18
|
+
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
19
|
+
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
|
20
|
+
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
21
|
+
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
22
|
+
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
23
|
+
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
24
|
+
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
25
|
+
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
26
|
+
*/
|
27
|
+
|
28
|
+
#include "MethodHandle.h"
|
29
|
+
|
30
|
+
|
31
|
+
#include <sys/param.h>
|
32
|
+
#include <sys/types.h>
|
33
|
+
#ifndef _WIN32
|
34
|
+
# include <sys/mman.h>
|
35
|
+
#endif
|
36
|
+
#include <stdio.h>
|
37
|
+
#include <stdint.h>
|
38
|
+
#include <stdbool.h>
|
39
|
+
#include <ruby.h>
|
40
|
+
|
41
|
+
#include <ffi.h>
|
42
|
+
#include "rbffi.h"
|
43
|
+
#include "compat.h"
|
44
|
+
|
45
|
+
#include "AbstractMemory.h"
|
46
|
+
#include "Pointer.h"
|
47
|
+
#include "Struct.h"
|
48
|
+
#include "Platform.h"
|
49
|
+
#include "Type.h"
|
50
|
+
#include "LastError.h"
|
51
|
+
#include "Call.h"
|
52
|
+
#include "Function.h"
|
53
|
+
|
54
|
+
#if defined(HAVE_LIBFFI) && !defined(HAVE_FFI_CLOSURE_ALLOC)
|
55
|
+
static void* ffi_closure_alloc(size_t size, void** code);
|
56
|
+
static void ffi_closure_free(void* ptr);
|
57
|
+
ffi_status ffi_prep_closure_loc(ffi_closure* closure, ffi_cif* cif,
|
58
|
+
void (*fun)(ffi_cif*, void*, void**, void*),
|
59
|
+
void* user_data, void* code);
|
60
|
+
#endif /* HAVE_FFI_CLOSURE_ALLOC */
|
61
|
+
|
62
|
+
typedef struct Function_ {
|
63
|
+
AbstractMemory memory;
|
64
|
+
FunctionType* info;
|
65
|
+
MethodHandle* methodHandle;
|
66
|
+
bool autorelease;
|
67
|
+
ffi_closure* ffiClosure;
|
68
|
+
VALUE rbProc;
|
69
|
+
VALUE rbFunctionInfo;
|
70
|
+
} Function;
|
71
|
+
|
72
|
+
static void function_mark(Function *);
|
73
|
+
static void function_free(Function *);
|
74
|
+
static VALUE function_init(VALUE self, VALUE rbFunctionInfo, VALUE rbProc);
|
75
|
+
static void callback_invoke(ffi_cif* cif, void* retval, void** parameters, void* user_data);
|
76
|
+
|
77
|
+
VALUE rbffi_FunctionClass = Qnil;
|
78
|
+
|
79
|
+
static ID id_call = 0, id_cbtable = 0;
|
80
|
+
|
81
|
+
static VALUE
|
82
|
+
function_allocate(VALUE klass)
|
83
|
+
{
|
84
|
+
Function *fn;
|
85
|
+
VALUE obj;
|
86
|
+
|
87
|
+
obj = Data_Make_Struct(klass, Function, function_mark, function_free, fn);
|
88
|
+
|
89
|
+
fn->memory.access = MEM_RD;
|
90
|
+
fn->memory.ops = &rbffi_AbstractMemoryOps;
|
91
|
+
|
92
|
+
fn->rbProc = Qnil;
|
93
|
+
fn->rbFunctionInfo = Qnil;
|
94
|
+
fn->autorelease = true;
|
95
|
+
|
96
|
+
return obj;
|
97
|
+
}
|
98
|
+
|
99
|
+
static void
|
100
|
+
function_mark(Function *fn)
|
101
|
+
{
|
102
|
+
rb_gc_mark(fn->rbProc);
|
103
|
+
rb_gc_mark(fn->rbFunctionInfo);
|
104
|
+
}
|
105
|
+
|
106
|
+
static void
|
107
|
+
function_free(Function *fn)
|
108
|
+
{
|
109
|
+
if (fn->methodHandle != NULL) {
|
110
|
+
rbffi_MethodHandle_Free(fn->methodHandle);
|
111
|
+
}
|
112
|
+
|
113
|
+
if (fn->ffiClosure != NULL && fn->autorelease) {
|
114
|
+
ffi_closure_free(fn->ffiClosure);
|
115
|
+
}
|
116
|
+
|
117
|
+
xfree(fn);
|
118
|
+
}
|
119
|
+
|
120
|
+
static VALUE
|
121
|
+
function_initialize(int argc, VALUE* argv, VALUE self)
|
122
|
+
{
|
123
|
+
|
124
|
+
VALUE rbReturnType = Qnil, rbParamTypes = Qnil, rbProc = Qnil, rbOptions = Qnil;
|
125
|
+
VALUE rbFunctionInfo = Qnil;
|
126
|
+
VALUE infoArgv[3];
|
127
|
+
int nargs;
|
128
|
+
|
129
|
+
nargs = rb_scan_args(argc, argv, "22", &rbReturnType, &rbParamTypes, &rbProc, &rbOptions);
|
130
|
+
|
131
|
+
//
|
132
|
+
// Callback with block,
|
133
|
+
// e.g. Function.new(:int, [ :int ]) { |i| blah }
|
134
|
+
// or Function.new(:int, [ :int ], { :convention => :stdcall }) { |i| blah }
|
135
|
+
//
|
136
|
+
if (rb_block_given_p()) {
|
137
|
+
if (nargs > 3) {
|
138
|
+
rb_raise(rb_eArgError, "cannot create function with both proc/address and block");
|
139
|
+
}
|
140
|
+
rbOptions = rbProc;
|
141
|
+
rbProc = rb_block_proc();
|
142
|
+
} else {
|
143
|
+
// Callback with proc, or Function with address
|
144
|
+
// e.g. Function.new(:int, [ :int ], Proc.new { |i| })
|
145
|
+
// Function.new(:int, [ :int ], Proc.new { |i| }, { :convention => :stdcall })
|
146
|
+
// Function.new(:int, [ :int ], addr)
|
147
|
+
// Function.new(:int, [ :int ], addr, { :convention => :stdcall })
|
148
|
+
}
|
149
|
+
|
150
|
+
infoArgv[0] = rbReturnType;
|
151
|
+
infoArgv[1] = rbParamTypes;
|
152
|
+
infoArgv[2] = rbOptions;
|
153
|
+
rbFunctionInfo = rb_class_new_instance(rbOptions != Qnil ? 3 : 2, infoArgv, rbffi_FunctionTypeClass);
|
154
|
+
|
155
|
+
function_init(self, rbFunctionInfo, rbProc);
|
156
|
+
|
157
|
+
return self;
|
158
|
+
}
|
159
|
+
|
160
|
+
VALUE
|
161
|
+
rbffi_Function_NewInstance(VALUE rbFunctionInfo, VALUE rbProc)
|
162
|
+
{
|
163
|
+
return function_init(function_allocate(rbffi_FunctionClass), rbFunctionInfo, rbProc);
|
164
|
+
}
|
165
|
+
|
166
|
+
VALUE
|
167
|
+
rbffi_Function_ForProc(VALUE rbFunctionInfo, VALUE proc)
|
168
|
+
{
|
169
|
+
VALUE callback;
|
170
|
+
VALUE cbTable = RTEST(rb_ivar_defined(proc, id_cbtable)) ? rb_ivar_get(proc, id_cbtable) : Qnil;
|
171
|
+
|
172
|
+
if (cbTable == Qnil) {
|
173
|
+
cbTable = rb_hash_new();
|
174
|
+
rb_ivar_set(proc, id_cbtable, cbTable);
|
175
|
+
}
|
176
|
+
|
177
|
+
callback = rb_hash_aref(cbTable, rbFunctionInfo);
|
178
|
+
if (callback != Qnil) {
|
179
|
+
return callback;
|
180
|
+
}
|
181
|
+
|
182
|
+
callback = rbffi_Function_NewInstance(rbFunctionInfo, proc);
|
183
|
+
rb_hash_aset(cbTable, rbFunctionInfo, callback);
|
184
|
+
return callback;
|
185
|
+
}
|
186
|
+
|
187
|
+
static VALUE
|
188
|
+
function_init(VALUE self, VALUE rbFunctionInfo, VALUE rbProc)
|
189
|
+
{
|
190
|
+
Function* fn = NULL;
|
191
|
+
|
192
|
+
Data_Get_Struct(self, Function, fn);
|
193
|
+
|
194
|
+
fn->rbFunctionInfo = rbFunctionInfo;
|
195
|
+
|
196
|
+
Data_Get_Struct(fn->rbFunctionInfo, FunctionType, fn->info);
|
197
|
+
|
198
|
+
if (rb_obj_is_kind_of(rbProc, rbffi_PointerClass)) {
|
199
|
+
AbstractMemory* memory;
|
200
|
+
Data_Get_Struct(rbProc, AbstractMemory, memory);
|
201
|
+
fn->memory = *memory;
|
202
|
+
|
203
|
+
} else if (rb_obj_is_kind_of(rbProc, rb_cProc) || rb_respond_to(rbProc, id_call)) {
|
204
|
+
void* code;
|
205
|
+
ffi_status status;
|
206
|
+
fn->ffiClosure = ffi_closure_alloc(sizeof(*fn->ffiClosure), &code);
|
207
|
+
if (fn->ffiClosure == NULL) {
|
208
|
+
rb_raise(rb_eNoMemError, "Failed to allocate libffi closure");
|
209
|
+
}
|
210
|
+
|
211
|
+
status = ffi_prep_closure_loc(fn->ffiClosure, &fn->info->ffi_cif,
|
212
|
+
callback_invoke, fn, code);
|
213
|
+
if (status != FFI_OK) {
|
214
|
+
rb_raise(rb_eArgError, "ffi_prep_closure_loc failed");
|
215
|
+
}
|
216
|
+
|
217
|
+
fn->memory.address = code;
|
218
|
+
fn->memory.size = sizeof(*fn->ffiClosure);
|
219
|
+
fn->autorelease = true;
|
220
|
+
|
221
|
+
} else {
|
222
|
+
rb_raise(rb_eTypeError, "wrong argument type. Expected pointer or proc");
|
223
|
+
}
|
224
|
+
|
225
|
+
fn->rbProc = rbProc;
|
226
|
+
|
227
|
+
return self;
|
228
|
+
}
|
229
|
+
|
230
|
+
static VALUE
|
231
|
+
function_call(int argc, VALUE* argv, VALUE self)
|
232
|
+
{
|
233
|
+
Function* fn;
|
234
|
+
|
235
|
+
Data_Get_Struct(self, Function, fn);
|
236
|
+
|
237
|
+
return rbffi_CallFunction(argc, argv, fn->memory.address, fn->info);
|
238
|
+
}
|
239
|
+
|
240
|
+
static VALUE
|
241
|
+
function_attach(VALUE self, VALUE module, VALUE name)
|
242
|
+
{
|
243
|
+
Function* fn;
|
244
|
+
char var[1024];
|
245
|
+
|
246
|
+
Data_Get_Struct(self, Function, fn);
|
247
|
+
|
248
|
+
if (fn->info->parameterCount == -1) {
|
249
|
+
rb_raise(rb_eRuntimeError, "Cannot attach variadic functions");
|
250
|
+
}
|
251
|
+
|
252
|
+
if (fn->methodHandle == NULL) {
|
253
|
+
fn->methodHandle = rbffi_MethodHandle_Alloc(fn->info, fn->memory.address);
|
254
|
+
}
|
255
|
+
|
256
|
+
//
|
257
|
+
// Stash the Function in a module variable so it does not get garbage collected
|
258
|
+
//
|
259
|
+
snprintf(var, sizeof(var), "@@%s", StringValueCStr(name));
|
260
|
+
rb_cv_set(module, var, self);
|
261
|
+
|
262
|
+
rb_define_module_function(module, StringValueCStr(name),
|
263
|
+
rbffi_MethodHandle_CodeAddress(fn->methodHandle), -1);
|
264
|
+
|
265
|
+
|
266
|
+
rb_define_method(module, StringValueCStr(name),
|
267
|
+
rbffi_MethodHandle_CodeAddress(fn->methodHandle), -1);
|
268
|
+
|
269
|
+
return self;
|
270
|
+
}
|
271
|
+
|
272
|
+
static VALUE
|
273
|
+
function_set_autorelease(VALUE self, VALUE autorelease)
|
274
|
+
{
|
275
|
+
Function* fn;
|
276
|
+
|
277
|
+
Data_Get_Struct(self, Function, fn);
|
278
|
+
|
279
|
+
fn->autorelease = RTEST(autorelease);
|
280
|
+
|
281
|
+
return self;
|
282
|
+
}
|
283
|
+
|
284
|
+
static VALUE
|
285
|
+
function_autorelease_p(VALUE self)
|
286
|
+
{
|
287
|
+
Function* fn;
|
288
|
+
|
289
|
+
Data_Get_Struct(self, Function, fn);
|
290
|
+
|
291
|
+
return fn->autorelease ? Qtrue : Qfalse;
|
292
|
+
}
|
293
|
+
|
294
|
+
static VALUE
|
295
|
+
function_release(VALUE self)
|
296
|
+
{
|
297
|
+
Function* fn;
|
298
|
+
|
299
|
+
Data_Get_Struct(self, Function, fn);
|
300
|
+
|
301
|
+
if (fn->ffiClosure == NULL) {
|
302
|
+
rb_raise(rb_eRuntimeError, "cannot free function which was not allocated");
|
303
|
+
}
|
304
|
+
|
305
|
+
ffi_closure_free(fn->ffiClosure);
|
306
|
+
fn->ffiClosure = NULL;
|
307
|
+
|
308
|
+
return self;
|
309
|
+
}
|
310
|
+
|
311
|
+
static void
|
312
|
+
callback_invoke(ffi_cif* cif, void* retval, void** parameters, void* user_data)
|
313
|
+
{
|
314
|
+
Function* fn = (Function *) user_data;
|
315
|
+
FunctionType *cbInfo = fn->info;
|
316
|
+
VALUE* rbParams;
|
317
|
+
VALUE rbReturnValue;
|
318
|
+
int i;
|
319
|
+
|
320
|
+
rbParams = ALLOCA_N(VALUE, cbInfo->parameterCount);
|
321
|
+
for (i = 0; i < cbInfo->parameterCount; ++i) {
|
322
|
+
VALUE param;
|
323
|
+
switch (cbInfo->parameterTypes[i]->nativeType) {
|
324
|
+
case NATIVE_INT8:
|
325
|
+
param = INT2NUM(*(int8_t *) parameters[i]);
|
326
|
+
break;
|
327
|
+
case NATIVE_UINT8:
|
328
|
+
param = UINT2NUM(*(uint8_t *) parameters[i]);
|
329
|
+
break;
|
330
|
+
case NATIVE_INT16:
|
331
|
+
param = INT2NUM(*(int16_t *) parameters[i]);
|
332
|
+
break;
|
333
|
+
case NATIVE_UINT16:
|
334
|
+
param = UINT2NUM(*(uint16_t *) parameters[i]);
|
335
|
+
break;
|
336
|
+
case NATIVE_INT32:
|
337
|
+
param = INT2NUM(*(int32_t *) parameters[i]);
|
338
|
+
break;
|
339
|
+
case NATIVE_UINT32:
|
340
|
+
param = UINT2NUM(*(uint32_t *) parameters[i]);
|
341
|
+
break;
|
342
|
+
case NATIVE_INT64:
|
343
|
+
param = LL2NUM(*(int64_t *) parameters[i]);
|
344
|
+
break;
|
345
|
+
case NATIVE_UINT64:
|
346
|
+
param = ULL2NUM(*(uint64_t *) parameters[i]);
|
347
|
+
break;
|
348
|
+
case NATIVE_FLOAT32:
|
349
|
+
param = rb_float_new(*(float *) parameters[i]);
|
350
|
+
break;
|
351
|
+
case NATIVE_FLOAT64:
|
352
|
+
param = rb_float_new(*(double *) parameters[i]);
|
353
|
+
break;
|
354
|
+
case NATIVE_STRING:
|
355
|
+
param = (*(void **) parameters[i] != NULL) ? rb_tainted_str_new2(*(char **) parameters[i]) : Qnil;
|
356
|
+
break;
|
357
|
+
case NATIVE_POINTER:
|
358
|
+
param = rbffi_Pointer_NewInstance(*(void **) parameters[i]);
|
359
|
+
break;
|
360
|
+
case NATIVE_BOOL:
|
361
|
+
param = (*(void **) parameters[i]) ? Qtrue : Qfalse;
|
362
|
+
break;
|
363
|
+
|
364
|
+
case NATIVE_FUNCTION:
|
365
|
+
case NATIVE_CALLBACK:
|
366
|
+
param = rbffi_NativeValue_ToRuby(cbInfo->parameterTypes[i],
|
367
|
+
rb_ary_entry(cbInfo->rbParameterTypes, i), parameters[i], Qnil);
|
368
|
+
break;
|
369
|
+
default:
|
370
|
+
param = Qnil;
|
371
|
+
break;
|
372
|
+
}
|
373
|
+
rbParams[i] = param;
|
374
|
+
}
|
375
|
+
rbReturnValue = rb_funcall2(fn->rbProc, id_call, cbInfo->parameterCount, rbParams);
|
376
|
+
if (rbReturnValue == Qnil || TYPE(rbReturnValue) == T_NIL) {
|
377
|
+
memset(retval, 0, cbInfo->ffiReturnType->size);
|
378
|
+
} else switch (cbInfo->returnType->nativeType) {
|
379
|
+
case NATIVE_INT8:
|
380
|
+
case NATIVE_INT16:
|
381
|
+
case NATIVE_INT32:
|
382
|
+
*((ffi_sarg *) retval) = NUM2INT(rbReturnValue);
|
383
|
+
break;
|
384
|
+
case NATIVE_UINT8:
|
385
|
+
case NATIVE_UINT16:
|
386
|
+
case NATIVE_UINT32:
|
387
|
+
*((ffi_arg *) retval) = NUM2UINT(rbReturnValue);
|
388
|
+
break;
|
389
|
+
case NATIVE_INT64:
|
390
|
+
*((int64_t *) retval) = NUM2LL(rbReturnValue);
|
391
|
+
break;
|
392
|
+
case NATIVE_UINT64:
|
393
|
+
*((uint64_t *) retval) = NUM2ULL(rbReturnValue);
|
394
|
+
break;
|
395
|
+
case NATIVE_FLOAT32:
|
396
|
+
*((float *) retval) = (float) NUM2DBL(rbReturnValue);
|
397
|
+
break;
|
398
|
+
case NATIVE_FLOAT64:
|
399
|
+
*((double *) retval) = NUM2DBL(rbReturnValue);
|
400
|
+
break;
|
401
|
+
case NATIVE_POINTER:
|
402
|
+
if (TYPE(rbReturnValue) == T_DATA && rb_obj_is_kind_of(rbReturnValue, rbffi_PointerClass)) {
|
403
|
+
*((void **) retval) = ((AbstractMemory *) DATA_PTR(rbReturnValue))->address;
|
404
|
+
} else {
|
405
|
+
// Default to returning NULL if not a value pointer object. handles nil case as well
|
406
|
+
*((void **) retval) = NULL;
|
407
|
+
}
|
408
|
+
break;
|
409
|
+
case NATIVE_BOOL:
|
410
|
+
*((ffi_sarg *) retval) = TYPE(rbReturnValue) == T_TRUE ? 1 : 0;
|
411
|
+
break;
|
412
|
+
|
413
|
+
case NATIVE_FUNCTION:
|
414
|
+
case NATIVE_CALLBACK:
|
415
|
+
if (TYPE(rbReturnValue) == T_DATA && rb_obj_is_kind_of(rbReturnValue, rbffi_PointerClass)) {
|
416
|
+
|
417
|
+
*((void **) retval) = ((AbstractMemory *) DATA_PTR(rbReturnValue))->address;
|
418
|
+
|
419
|
+
} else if (rb_obj_is_kind_of(rbReturnValue, rb_cProc) || rb_respond_to(rbReturnValue, id_call)) {
|
420
|
+
VALUE function;
|
421
|
+
|
422
|
+
function = rbffi_Function_ForProc(cbInfo->rbReturnType, rbReturnValue);
|
423
|
+
|
424
|
+
*((void **) retval) = ((AbstractMemory *) DATA_PTR(function))->address;
|
425
|
+
} else {
|
426
|
+
*((void **) retval) = NULL;
|
427
|
+
}
|
428
|
+
break;
|
429
|
+
|
430
|
+
default:
|
431
|
+
*((ffi_arg *) retval) = 0;
|
432
|
+
break;
|
433
|
+
}
|
434
|
+
}
|
435
|
+
|
436
|
+
|
437
|
+
#if defined(HAVE_LIBFFI) && !defined(HAVE_FFI_CLOSURE_ALLOC)
|
438
|
+
/*
|
439
|
+
* versions of ffi_closure_alloc, ffi_closure_free and ffi_prep_closure_loc for older
|
440
|
+
* system libffi versions.
|
441
|
+
*/
|
442
|
+
static void*
|
443
|
+
ffi_closure_alloc(size_t size, void** code)
|
444
|
+
{
|
445
|
+
void* closure;
|
446
|
+
closure = mmap(NULL, size, PROT_READ | PROT_WRITE,
|
447
|
+
MAP_ANON | MAP_PRIVATE, -1, 0);
|
448
|
+
if (closure == (void *) -1) {
|
449
|
+
return NULL;
|
450
|
+
}
|
451
|
+
memset(closure, 0, size);
|
452
|
+
*code = closure;
|
453
|
+
return closure;
|
454
|
+
}
|
455
|
+
|
456
|
+
static void
|
457
|
+
ffi_closure_free(void* ptr)
|
458
|
+
{
|
459
|
+
if (ptr != NULL && ptr != (void *) -1) {
|
460
|
+
munmap(ptr, sizeof(ffi_closure));
|
461
|
+
}
|
462
|
+
}
|
463
|
+
|
464
|
+
ffi_status
|
465
|
+
ffi_prep_closure_loc(ffi_closure* closure, ffi_cif* cif,
|
466
|
+
void (*fun)(ffi_cif*, void*, void**, void*),
|
467
|
+
void* user_data, void* code)
|
468
|
+
{
|
469
|
+
ffi_status retval = ffi_prep_closure(closure, cif, fun, user_data);
|
470
|
+
if (retval == FFI_OK) {
|
471
|
+
mprotect(closure, sizeof(ffi_closure), PROT_READ | PROT_EXEC);
|
472
|
+
}
|
473
|
+
return retval;
|
474
|
+
}
|
475
|
+
|
476
|
+
#endif /* HAVE_FFI_CLOSURE_ALLOC */
|
477
|
+
|
478
|
+
void
|
479
|
+
rbffi_Function_Init(VALUE moduleFFI)
|
480
|
+
{
|
481
|
+
rbffi_FunctionInfo_Init(moduleFFI);
|
482
|
+
rbffi_FunctionClass = rb_define_class_under(moduleFFI, "Function", rbffi_PointerClass);
|
483
|
+
|
484
|
+
rb_global_variable(&rbffi_FunctionClass);
|
485
|
+
rb_define_alloc_func(rbffi_FunctionClass, function_allocate);
|
486
|
+
|
487
|
+
rb_define_method(rbffi_FunctionClass, "initialize", function_initialize, -1);
|
488
|
+
rb_define_method(rbffi_FunctionClass, "call", function_call, -1);
|
489
|
+
rb_define_method(rbffi_FunctionClass, "attach", function_attach, 2);
|
490
|
+
rb_define_method(rbffi_FunctionClass, "free", function_release, 0);
|
491
|
+
rb_define_method(rbffi_FunctionClass, "autorelease=", function_set_autorelease, 1);
|
492
|
+
rb_define_method(rbffi_FunctionClass, "autorelease", function_autorelease_p, 0);
|
493
|
+
rb_define_method(rbffi_FunctionClass, "autorelease?", function_autorelease_p, 0);
|
494
|
+
|
495
|
+
id_call = rb_intern("call");
|
496
|
+
id_cbtable = rb_intern("@__ffi_callback_table__");
|
497
|
+
}
|
498
|
+
|