Microsoft Visual C port
authorAnthony Green <green@gmachine.(none)>
Fri, 15 Jan 2010 15:46:51 +0000 (10:46 -0500)
committerAnthony Green <green@gmachine.(none)>
Fri, 15 Jan 2010 15:46:51 +0000 (10:46 -0500)
14 files changed:
.pc/applied-patches
.pc/msvc/.timestamp [new file with mode: 0644]
.pc/msvc/ChangeLog.libffi [new file with mode: 0644]
.pc/msvc/include/ffi.h.in [new file with mode: 0644]
.pc/msvc/src/prep_cif.c [new file with mode: 0644]
.pc/msvc/src/x86/ffi.c [new file with mode: 0644]
.pc/msvc/src/x86/win32.S [new file with mode: 0644]
ChangeLog.libffi
include/ffi.h.in
patches/msvc [new file with mode: 0644]
patches/series
src/prep_cif.c
src/x86/ffi.c
src/x86/win32.S

index 060b184..829923e 100644 (file)
@@ -1,3 +1,4 @@
 stand-alone
 closure-api-example-doc
 sunstudio
+msvc
diff --git a/.pc/msvc/.timestamp b/.pc/msvc/.timestamp
new file mode 100644 (file)
index 0000000..e69de29
diff --git a/.pc/msvc/ChangeLog.libffi b/.pc/msvc/ChangeLog.libffi
new file mode 100644 (file)
index 0000000..3ea21e1
--- /dev/null
@@ -0,0 +1,590 @@
+2010-01-15  Oliver Kiddle  <okiddle@yahoo.co.uk>
+
+       * src/x86/ffitarget.h (ffi_abi): Check for __i386 and __amd64 for
+       Sun Studio compiler compatibility. 
+
+2010-01-12  Conrad Irwin <conrad.irwin@gmail.com>
+
+       * doc/libffi.texi: Add closure example.
+       * doc/libffi.info: Rebuilt.
+
+2009-12-25  Samuli Suominen  <ssuominen@gentoo.org>
+
+       * configure.ac: Undefine _AC_ARG_VAR_PRECIOUS for autoconf 2.64.
+       * configure: Rebuilt.
+       * fficonfig.h.in: Rebuilt.
+
+2009-06-16  Andrew Haley  <aph@redhat.com>
+
+       * testsuite/libffi.call/cls_align_sint64.c,
+       testsuite/libffi.call/cls_align_uint64.c,
+       testsuite/libffi.call/cls_longdouble_va.c,
+       testsuite/libffi.call/cls_ulonglong.c,
+       testsuite/libffi.call/return_ll1.c,
+       testsuite/libffi.call/stret_medium2.c: Fix printf format
+       specifiers.
+       * testsuite/libffi.call/huge_struct.c: Ad x86 XFAILs.
+       * testsuite/libffi.call/float2.c: Fix dg-excess-errors.
+       * testsuite/libffi.call/ffitest.h,
+       testsuite/libffi.special/ffitestcxx.h (PRIdLL, PRIuLL): Define.
+
+2009-06-12  Andrew Haley  <aph@redhat.com>
+
+       * testsuite/libffi.call/cls_align_sint64.c,
+       testsuite/libffi.call/cls_align_uint64.c,
+       testsuite/libffi.call/cls_ulonglong.c,
+       testsuite/libffi.call/return_ll1.c,
+       testsuite/libffi.call/stret_medium2.c: Fix printf format
+       specifiers.
+       testsuite/libffi.special/unwindtest.cc: include stdint.h.
+
+2009-06-11  Timothy Wall  <twall@users.sf.net>
+
+       * Makefile.am,
+        configure.ac,
+        include/ffi.h.in,
+        include/ffi_common.h,
+        src/closures.c,
+        src/dlmalloc.c,
+        src/x86/ffi.c,
+        src/x86/ffitarget.h,
+        src/x86/win64.S (new),
+       README: Added win64 support (mingw or MSVC)
+        * Makefile.in,
+        include/Makefile.in,
+        man/Makefile.in,
+        testsuite/Makefile.in,
+        configure,
+        aclocal.m4: Regenerated
+        * ltcf-c.sh: properly escape cygwin/w32 path
+        * man/ffi_call.3: Clarify size requirements for return value.
+        * src/x86/ffi64.c: Fix filename in comment.
+        * src/x86/win32.S: Remove unused extern.
+
+        * testsuite/libffi.call/closure_fn0.c,
+        testsuite/libffi.call/closure_fn1.c,
+        testsuite/libffi.call/closure_fn2.c,
+        testsuite/libffi.call/closure_fn3.c,
+        testsuite/libffi.call/closure_fn4.c,
+        testsuite/libffi.call/closure_fn5.c,
+        testsuite/libffi.call/closure_fn6.c,
+       testsuite/libffi.call/closure_stdcall.c,
+       testsuite/libffi.call/cls_12byte.c,
+       testsuite/libffi.call/cls_16byte.c,
+       testsuite/libffi.call/cls_18byte.c,
+       testsuite/libffi.call/cls_19byte.c,
+       testsuite/libffi.call/cls_1_1byte.c,
+       testsuite/libffi.call/cls_20byte.c,
+       testsuite/libffi.call/cls_20byte1.c,
+       testsuite/libffi.call/cls_24byte.c,
+       testsuite/libffi.call/cls_2byte.c,
+       testsuite/libffi.call/cls_3_1byte.c,
+       testsuite/libffi.call/cls_3byte1.c,
+       testsuite/libffi.call/cls_3byte2.c,
+       testsuite/libffi.call/cls_4_1byte.c,
+       testsuite/libffi.call/cls_4byte.c,
+       testsuite/libffi.call/cls_5_1_byte.c,
+       testsuite/libffi.call/cls_5byte.c,
+       testsuite/libffi.call/cls_64byte.c,
+       testsuite/libffi.call/cls_6_1_byte.c,
+       testsuite/libffi.call/cls_6byte.c,
+       testsuite/libffi.call/cls_7_1_byte.c,
+       testsuite/libffi.call/cls_7byte.c,
+       testsuite/libffi.call/cls_8byte.c,
+       testsuite/libffi.call/cls_9byte1.c,
+       testsuite/libffi.call/cls_9byte2.c,
+       testsuite/libffi.call/cls_align_double.c,
+       testsuite/libffi.call/cls_align_float.c,
+       testsuite/libffi.call/cls_align_longdouble.c,
+       testsuite/libffi.call/cls_align_longdouble_split.c,
+       testsuite/libffi.call/cls_align_longdouble_split2.c,
+       testsuite/libffi.call/cls_align_pointer.c,
+       testsuite/libffi.call/cls_align_sint16.c,
+       testsuite/libffi.call/cls_align_sint32.c,
+       testsuite/libffi.call/cls_align_sint64.c,
+       testsuite/libffi.call/cls_align_uint16.c,
+       testsuite/libffi.call/cls_align_uint32.c,
+       testsuite/libffi.call/cls_align_uint64.c,
+       testsuite/libffi.call/cls_dbls_struct.c,
+       testsuite/libffi.call/cls_double.c,
+       testsuite/libffi.call/cls_double_va.c,
+       testsuite/libffi.call/cls_float.c,
+       testsuite/libffi.call/cls_longdouble.c,
+       testsuite/libffi.call/cls_longdouble_va.c,
+       testsuite/libffi.call/cls_multi_schar.c,
+       testsuite/libffi.call/cls_multi_sshort.c,
+       testsuite/libffi.call/cls_multi_sshortchar.c,
+       testsuite/libffi.call/cls_multi_uchar.c,
+       testsuite/libffi.call/cls_multi_ushort.c,
+       testsuite/libffi.call/cls_multi_ushortchar.c,
+       testsuite/libffi.call/cls_pointer.c,
+       testsuite/libffi.call/cls_pointer_stack.c,
+       testsuite/libffi.call/cls_schar.c,
+       testsuite/libffi.call/cls_sint.c,
+       testsuite/libffi.call/cls_sshort.c,
+       testsuite/libffi.call/cls_uchar.c,
+       testsuite/libffi.call/cls_uint.c,
+       testsuite/libffi.call/cls_ulonglong.c,
+       testsuite/libffi.call/cls_ushort.c,
+       testsuite/libffi.call/err_bad_abi.c,
+       testsuite/libffi.call/err_bad_typedef.c,
+       testsuite/libffi.call/float2.c,
+       testsuite/libffi.call/huge_struct.c,
+       testsuite/libffi.call/nested_struct.c,
+       testsuite/libffi.call/nested_struct1.c,
+       testsuite/libffi.call/nested_struct10.c,
+       testsuite/libffi.call/nested_struct2.c,
+       testsuite/libffi.call/nested_struct3.c,
+       testsuite/libffi.call/nested_struct4.c,
+       testsuite/libffi.call/nested_struct5.c,
+       testsuite/libffi.call/nested_struct6.c,
+       testsuite/libffi.call/nested_struct7.c,
+       testsuite/libffi.call/nested_struct8.c,
+       testsuite/libffi.call/nested_struct9.c,
+       testsuite/libffi.call/problem1.c,
+       testsuite/libffi.call/return_ldl.c,
+       testsuite/libffi.call/return_ll1.c,
+       testsuite/libffi.call/stret_large.c,
+       testsuite/libffi.call/stret_large2.c,
+       testsuite/libffi.call/stret_medium.c,
+       testsuite/libffi.call/stret_medium2.c,
+        testsuite/libffi.special/unwindtest.cc: use ffi_closure_alloc instead
+        of checking for MMAP.  Use intptr_t instead of long casts.
+
+2009-06-04  Andrew Haley  <aph@redhat.com>
+
+       * src/powerpc/ffitarget.h: Fix misapplied merge from gcc.
+
+2009-06-04  Andrew Haley  <aph@redhat.com>
+
+       * src/mips/o32.S,
+       src/mips/n32.S: Fix licence formatting.
+
+2009-06-04  Andrew Haley  <aph@redhat.com>
+
+       * src/x86/darwin.S: Fix licence formatting.
+       src/x86/win32.S: Likewise.
+       src/sh64/sysv.S: Likewise.
+       src/sh/sysv.S: Likewise.
+
+2009-06-04  Andrew Haley  <aph@redhat.com>
+
+       * src/sh64/ffi.c: Remove lint directives.  Was missing from merge
+       of Andreas Tobler's patch from 2006-04-22.
+       
+2009-06-04  Andrew Haley  <aph@redhat.com>
+
+       * src/sh/ffi.c: Apply missing hunk from Alexandre Oliva's patch of
+       2007-03-07.
+
+2008-12-26  Timothy Wall  <twall@users.sf.net>
+
+       * testsuite/libffi.call/cls_longdouble.c,
+        testsuite/libffi.call/cls_longdouble_va.c,
+        testsuite/libffi.call/cls_align_longdouble.c,
+        testsuite/libffi.call/cls_align_longdouble_split.c,
+        testsuite/libffi.call/cls_align_longdouble_split2.c: mark expected
+        failures on x86_64 cygwin/mingw.
+
+2008-12-22  Timothy Wall  <twall@users.sf.net>
+
+       * testsuite/libffi.call/closure_fn0.c,
+        testsuite/libffi.call/closure_fn1.c,    
+        testsuite/libffi.call/closure_fn2.c,    
+        testsuite/libffi.call/closure_fn3.c,    
+        testsuite/libffi.call/closure_fn4.c,    
+        testsuite/libffi.call/closure_fn5.c,    
+        testsuite/libffi.call/closure_fn6.c,    
+        testsuite/libffi.call/closure_loc_fn0.c,    
+        testsuite/libffi.call/closure_stdcall.c,    
+        testsuite/libffi.call/cls_align_pointer.c,    
+        testsuite/libffi.call/cls_pointer.c,    
+        testsuite/libffi.call/cls_pointer_stack.c: use portable cast from
+        pointer to integer (intptr_t).
+        * testsuite/libffi.call/cls_longdouble.c: disable for win64.
+       
+2008-12-19  Anthony Green  <green@redhat.com>
+
+       * configure.ac: Bump version to 3.0.8.
+       * configure, doc/stamp-vti, doc/version.texi: Rebuilt.
+       * libtool-version: Increment revision.
+       * README: Update for new release.
+
+2008-11-11  Anthony Green  <green@redhat.com>
+
+       * configure.ac: Bump version to 3.0.7.
+       * configure, doc/stamp-vti, doc/version.texi: Rebuilt.
+       * libtool-version: Increment revision.
+       * README: Update for new release.
+
+2008-08-25  Andreas Tobler  <a.tobler@schweiz.org>
+
+       * src/powerpc/ffitarget.h (ffi_abi): Add FFI_LINUX and
+       FFI_LINUX_SOFT_FLOAT to the POWERPC_FREEBSD enum.
+       Add note about flag bits used for FFI_SYSV_TYPE_SMALL_STRUCT.
+       Adjust copyright notice.
+       * src/powerpc/ffi.c: Add two new flags to indicate if we have one
+       register or two register to use for FFI_SYSV structs.
+       (ffi_prep_cif_machdep): Pass the right register flag introduced above.
+       (ffi_closure_helper_SYSV): Fix the return type for
+       FFI_SYSV_TYPE_SMALL_STRUCT. Comment.
+       Adjust copyright notice.
+
+2008-07-24  Anthony Green  <green@redhat.com>
+
+       * testsuite/libffi.call/cls_dbls_struct.c,
+       testsuite/libffi.call/cls_double_va.c,
+       testsuite/libffi.call/cls_longdouble.c,
+       testsuite/libffi.call/cls_longdouble_va.c,
+       testsuite/libffi.call/cls_pointer.c,
+       testsuite/libffi.call/cls_pointer_stack.c,
+       testsuite/libffi.call/err_bad_abi.c: Clean up failures from
+       compiler warnings.
+
+2008-07-17  Anthony Green  <green@redhat.com>
+
+       * configure.ac: Bump version to 3.0.6.
+       * configure, doc/stamp-vti, doc/version.texi: Rebuilt.
+       * libtool-version: Increment revision.  Add documentation.
+       * README: Update for new release.
+
+2008-07-16  Kaz Kojima  <kkojima@gcc.gnu.org>
+
+       * src/sh/ffi.c (ffi_prep_closure_loc): Turn INSN into an unsigned
+       int.
+
+2008-07-16  Kaz Kojima  <kkojima@gcc.gnu.org>
+
+       * src/sh/sysv.S: Add .note.GNU-stack on Linux.
+       * src/sh64/sysv.S: Likewise.
+
+2008-04-03  Anthony Green  <green@redhat.com>
+
+       * libffi.pc.in (Libs): Add -L${libdir}.
+       * configure.ac: Bump version to 3.0.5.
+       * configure, doc/stamp-vti, doc/version.texi: Rebuilt.
+       * libtool-version: Increment revision.
+       * README: Update for new release.
+
+2008-04-03  Anthony Green  <green@redhat.com>
+           Xerces Ranby  <xerxes@zafena.se>
+
+       * include/ffi.h.in: Wrap definition of target architecture to
+       protect from double definitions.
+
+2008-03-22  Moriyoshi Koizumi  <moriyoshi@gmail.com>
+
+       * src/x86/ffi.c (ffi_prep_closure_loc): Fix for bug revealed in
+       closure_loc_fn0.c.
+       * testsuite/libffi.call/closure_loc_fn0.c (closure_loc_test_fn0):
+       New test.
+
+2008-03-04  Anthony Green  <green@redhat.com>
+           Blake Chaffin
+           hos@tamanegi.org
+
+       * testsuite/libffi.call/cls_align_longdouble_split2.c
+          testsuite/libffi.call/cls_align_longdouble_split.c
+          testsuite/libffi.call/cls_dbls_struct.c
+          testsuite/libffi.call/cls_double_va.c
+          testsuite/libffi.call/cls_longdouble.c
+          testsuite/libffi.call/cls_longdouble_va.c
+          testsuite/libffi.call/cls_pointer.c
+          testsuite/libffi.call/cls_pointer_stack.c
+          testsuite/libffi.call/err_bad_abi.c
+          testsuite/libffi.call/err_bad_typedef.c
+          testsuite/libffi.call/huge_struct.c
+          testsuite/libffi.call/stret_large2.c
+          testsuite/libffi.call/stret_large.c
+          testsuite/libffi.call/stret_medium2.c
+          testsuite/libffi.call/stret_medium.c: New tests from Apple.
+
+2008-02-26  Jakub Jelinek  <jakub@redhat.com>
+            Anthony Green  <green@redhat.com>
+
+       * src/alpha/osf.S: Add .note.GNU-stack on Linux.
+       * src/s390/sysv.S: Likewise.
+       * src/powerpc/linux64.S: Likewise.
+       * src/powerpc/linux64_closure.S: Likewise.
+       * src/powerpc/ppc_closure.S: Likewise.
+       * src/powerpc/sysv.S: Likewise.
+       * src/x86/unix64.S: Likewise.
+       * src/x86/sysv.S: Likewise.
+       * src/sparc/v8.S: Likewise.
+       * src/sparc/v9.S: Likewise.
+       * src/m68k/sysv.S: Likewise.
+       * src/ia64/unix.S: Likewise.
+       * src/arm/sysv.S: Likewise.
+
+2008-02-26  Anthony Green  <green@redhat.com>
+            Thomas Heller  <theller@ctypes.org>
+
+       * src/x86/ffi.c (ffi_closure_SYSV_inner): Change C++ comment to C
+       comment.
+
+2008-02-26  Anthony Green  <green@redhat.org>
+            Thomas Heller  <theller@ctypes.org>
+
+       * include/ffi.h.in: Change void (*)() to void (*)(void).
+
+2008-02-26  Anthony Green  <green@redhat.org>
+            Thomas Heller  <theller@ctypes.org>
+
+       * src/alpha/ffi.c: Change void (*)() to void (*)(void).
+       src/alpha/osf.S, src/arm/ffi.c, src/frv/ffi.c, src/ia64/ffi.c,
+       src/ia64/unix.S, src/java_raw_api.c, src/m32r/ffi.c,
+       src/mips/ffi.c, src/pa/ffi.c, src/pa/hpux32.S, src/pa/linux.S,
+       src/powerpc/ffi.c, src/powerpc/ffi_darwin.c, src/raw_api.c,
+       src/s390/ffi.c, src/sh/ffi.c, src/sh64/ffi.c, src/sparc/ffi.c,
+       src/x86/ffi.c, src/x86/unix64.S, src/x86/darwin64.S,
+       src/x86/ffi64.c: Ditto.
+
+2008-02-24  Anthony Green  <green@redhat.org>
+
+       * configure.ac: Accept openbsd*, not just openbsd.
+       Bump version to 3.0.4.
+       * configure, doc/stamp-vti, doc/version.texi: Rebuilt.
+       * libtool-version: Increment revision.
+       * README: Update for new release.
+
+2008-02-22  Anthony Green  <green@redhat.com>
+
+       * README: Clean up list of tested platforms.
+
+2008-02-22  Anthony Green  <green@redhat.com>
+
+       * configure.ac: Bump version to 3.0.3.
+       * configure, doc/stamp-vti, doc/version.texi: Rebuilt.
+       * libtool-version: Increment revision.
+       * README: Update for new release.  Clean up test docs.
+
+2008-02-22  Bjoern Koenig  <bkoenig@alpha-tierchen.de>
+           Andreas Tobler  <a.tobler@schweiz.org>
+
+       * configure.ac: Add amd64-*-freebsd* target.
+       * configure: Regenerate.
+
+2008-02-22  Thomas Heller <theller@ctypes.org>
+
+       * configure.ac: Add x86 OpenBSD support.
+       * configure: Rebuilt.
+
+2008-02-21  Thomas Heller <theller@ctypes.org>
+
+       * README: Change "make test" to "make check".
+
+2008-02-21  Anthony Green  <green@redhat.com>
+
+       * configure.ac: Bump version to 3.0.2.
+       * configure, doc/stamp-vti, doc/version.texi: Rebuilt.
+       * libtool-version: Increment revision.
+       * README: Update for new release.
+
+2008-02-21  Björn König <bkoenig@alpha-tierchen.de>
+
+       * src/x86/freebsd.S: New file.
+       * configure.ac: Add x86 FreeBSD support.
+       * Makefile.am: Ditto.
+
+2008-02-15  Anthony Green  <green@redhat.com>
+
+       * configure.ac: Bump version to 3.0.1.
+       * configure, doc/stamp-vti, doc/version.texi: Rebuilt.
+       * libtool-version: Increment revision.
+       * README: Update for new release.
+
+2008-02-15  David Daney         <ddaney@avtrex.com>
+
+       * src/mips/ffi.c: Remove extra '>' from include directive.
+       (ffi_prep_closure_loc): Use clear_location instead of tramp.
+
+2008-02-15  Anthony Green  <green@redhat.com>
+
+       * configure.ac: Bump version to 3.0.0.
+       * configure, doc/stamp-vti, doc/version.texi: Rebuilt.
+
+2008-02-15  David Daney         <ddaney@avtrex.com>
+
+       * src/mips/ffi.c (USE__BUILTIN___CLEAR_CACHE):
+       Define (conditionally), and use it to include cachectl.h.
+       (ffi_prep_closure_loc): Fix cache flushing.
+       * src/mips/ffitarget.h (_ABIN32, _ABI64, _ABIO32): Define.
+
+2008-02-15  Anthony Green  <green@redhat.com>
+
+        * man/ffi_call.3, man/ffi_prep_cif.3, man/ffi.3:
+       Update dates and remove all references to ffi_prep_closure.
+       * configure.ac: Bump version to 2.99.9.
+       * configure, doc/stamp-vti, doc/version.texi: Rebuilt.
+
+2008-02-15  Anthony Green  <green@redhat.com>
+
+       * man/ffi_prep_closure.3: Delete.
+       * man/Makefile.am (EXTRA_DIST): Remove ffi_prep_closure.3.
+       (man_MANS): Ditto.
+       * man/Makefile.in: Rebuilt.
+       * configure.ac: Bump version to 2.99.8.
+       * configure, doc/stamp-vti, doc/version.texi: Rebuilt.
+
+2008-02-14  Anthony Green  <green@redhat.com>
+
+       * configure.ac: Bump version to 2.99.7.
+       * configure, doc/stamp-vti, doc/version.texi: Rebuilt.
+       * include/ffi.h.in LICENSE src/debug.c src/closures.c
+          src/ffitest.c src/s390/sysv.S src/s390/ffitarget.h
+          src/types.c src/m68k/ffitarget.h src/raw_api.c src/frv/ffi.c
+          src/frv/ffitarget.h src/sh/ffi.c src/sh/sysv.S
+          src/sh/ffitarget.h src/powerpc/ffitarget.h src/pa/ffi.c
+          src/pa/ffitarget.h src/pa/linux.S src/java_raw_api.c
+          src/cris/ffitarget.h src/x86/ffi.c src/x86/sysv.S
+          src/x86/unix64.S src/x86/win32.S src/x86/ffitarget.h
+          src/x86/ffi64.c src/x86/darwin.S src/ia64/ffi.c
+          src/ia64/ffitarget.h src/ia64/ia64_flags.h src/ia64/unix.S
+          src/sparc/ffi.c src/sparc/v9.S src/sparc/ffitarget.h
+          src/sparc/v8.S src/alpha/ffi.c src/alpha/ffitarget.h
+          src/alpha/osf.S src/sh64/ffi.c src/sh64/sysv.S
+          src/sh64/ffitarget.h src/mips/ffi.c src/mips/ffitarget.h
+          src/mips/n32.S src/mips/o32.S src/arm/ffi.c src/arm/sysv.S
+          src/arm/ffitarget.h src/prep_cif.c: Update license text.
+
+2008-02-14  Anthony Green  <green@redhat.com>
+
+       * README: Update tested platforms.
+       * configure.ac: Bump version to 2.99.6.
+       * configure: Rebuilt.
+
+2008-02-14  Anthony Green  <green@redhat.com>
+
+       * configure.ac: Bump version to 2.99.5.
+       * configure: Rebuilt.
+       * Makefile.am (EXTRA_DIST): Add darwin64.S
+       * Makefile.in: Rebuilt.
+       * testsuite/lib/libffi-dg.exp: Remove libstdc++ bits from GCC tree.
+       * LICENSE: Update WARRANTY.
+
+2008-02-14  Anthony Green  <green@redhat.com>
+
+       * libffi.pc.in (libdir): Fix libdir definition.
+       * configure.ac: Bump version to 2.99.4.
+       * configure: Rebuilt.
+
+2008-02-14  Anthony Green  <green@redhat.com>
+
+       * README: Update.
+       * libffi.info: New file.
+       * doc/stamp-vti: New file.
+       * configure.ac: Bump version to 2.99.3.
+       * configure: Rebuilt.
+
+2008-02-14  Anthony Green  <green@redhat.com>
+
+       * Makefile.am (SUBDIRS): Add man dir.
+       * Makefile.in: Rebuilt.
+       * configure.ac: Create Makefile.
+       * configure: Rebuilt.
+        * man/ffi_call.3 man/ffi_prep_cif.3 man/ffi_prep_closure.3
+          man/Makefile.am man/Makefile.in: New files.
+
+2008-02-14  Tom Tromey  <tromey@redhat.com>
+
+       * aclocal.m4, Makefile.in, configure, fficonfig.h.in: Rebuilt.
+       * mdate-sh, texinfo.tex: New files.
+       * Makefile.am (info_TEXINFOS): New variable.
+       * doc/libffi.texi: New file.
+       * doc/version.texi: Likewise.
+
+2008-02-14  Anthony Green  <green@redhat.com>
+
+       * Makefile.am (AM_CFLAGS): Don't compile with -D$(TARGET).
+       (lib_LTLIBRARIES): Define.
+       (toolexeclib_LIBRARIES): Undefine.
+       * Makefile.in: Rebuilt.
+       * configure.ac: Reset version to 2.99.1.
+       * configure.in: Rebuilt.
+
+2008-02-14  Anthony Green  <green@redhat.com>
+
+       * libffi.pc.in: Use @PACKAGE_NAME@ and @PACKAGE_VERSION@.
+       * configure.ac: Reset version to 2.99.1.
+       * configure.in: Rebuilt.
+       * Makefile.am (EXTRA_DIST): Add ChangeLog.libffi.
+       * Makefile.in: Rebuilt.
+       * LICENSE: Update copyright notice.
+
+2008-02-14  Anthony Green  <green@redhat.com>
+
+       * include/Makefile.am (nodist_includes_HEADERS): Define.  Don't
+       distribute ffitarget.h or ffi.h from the build include dir.
+       * Makefile.in: Rebuilt.
+
+2008-02-14  Anthony Green  <green@redhat.com>
+
+       * include/Makefile.am (includesdir): Install headers under libdir.
+       (pkgconfigdir): Define. Install libffi.pc.
+       * include/Makefile.in: Rebuilt.
+       * libffi.pc.in: Create.
+       * libtool-version: Increment CURRENT
+       * configure.ac: Add libffi.pc.in
+       * configure: Rebuilt.
+
+2008-02-03  Anthony Green  <green@redhat.com>
+
+       * include/Makefile.am (includesdir): Fix header install with
+       DESTDIR.
+       * include/Makefile.in: Rebuilt.
+
+2008-02-03  Timothy Wall  <twall@users.sf.net>
+
+       * src/x86/ffi.c (FFI_INIT_TRAMPOLINE_STDCALL): Calculate jump return
+          offset based on code pointer, not data pointer.
+
+2008-02-01  Anthony Green  <green@redhat.com>
+
+       * include/Makefile.am: Fix header installs.
+       * Makefile.am: Ditto.
+       * include/Makefile.in: Rebuilt.
+       * Makefile.in: Ditto.
+
+2008-02-01  Anthony Green  <green@redhat.com>
+
+       * src/x86/ffi.c (FFI_INIT_TRAMPOLINE_STDCALL,
+       FFI_INIT_TRAMPOLINE): Revert my broken changes to twall's last
+       patch.
+
+2008-01-31  Anthony Green  <green@redhat.com>
+
+       * Makefile.am (EXTRA_DIST): Add missing files.
+       * testsuite/Makefile.am: Ditto.
+       * Makefile.in, testsuite/Makefile.in: Rebuilt.
+
+2008-01-31  Timothy Wall <twall@users.sf.net>
+
+       * testsuite/libffi.call/closure_stdcall.c: Add test for stdcall
+       closures.
+       * src/x86/ffitarget.h: Increase size of trampoline for stdcall
+       closures.
+       * src/x86/win32.S: Add assembly for stdcall closure.
+       * src/x86/ffi.c: Initialize stdcall closure trampoline.
+
+2008-01-30  H.J. Lu <hongjiu.lu@intel.com>
+
+       PR libffi/34612
+       * src/x86/sysv.S (ffi_closure_SYSV): Pop 4 byte from stack when
+       returning struct.
+
+       * testsuite/libffi.call/call.exp: Add "-O2 -fomit-frame-pointer"
+       tests.
+
+2008-01-30  Anthony Green  <green@redhat.com>
+
+       * Makefile.am, include/Makefile.am: Move headers to
+       libffi_la_SOURCES for new automake.
+       * Makefile.in, include/Makefile.in: Rebuilt.
+       
+       * testsuite/lib/wrapper.exp: Copied from gcc tree to allow for 
+       execution outside of gcc tree.
+       * testsuite/lib/target-libpath.exp: Ditto.
+
+       * testsuite/lib/libffi-dg.exp: Many changes to allow for execution
+       outside of gcc tree.
+
diff --git a/.pc/msvc/include/ffi.h.in b/.pc/msvc/include/ffi.h.in
new file mode 100644 (file)
index 0000000..32f6103
--- /dev/null
@@ -0,0 +1,399 @@
+/* -----------------------------------------------------------------*-C-*-
+   libffi @VERSION@ - 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. */
+#define @TARGET@
+
+/* ---- System configuration information --------------------------------- */
+
+#include <ffitarget.h>
+
+#ifndef LIBFFI_ASM
+
+#ifdef _MSC_VER
+#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.  */
+#ifdef LONG_LONG_MAX
+# define FFI_LONG_LONG_MAX LONG_LONG_MAX
+#else
+# ifdef LLONG_MAX
+#  define FFI_LONG_LONG_MAX LLONG_MAX
+# else
+#  ifdef __GNUC__
+#   define FFI_LONG_LONG_MAX __LONG_LONG_MAX__
+#  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 != 9223372036854775807
+ #error "no 64-bit data type supported"
+# endif
+#elif LONG_MAX != 9223372036854775807
+ #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 == 9223372036854775807
+# define ffi_type_ulong        ffi_type_uint64
+# define ffi_type_slong        ffi_type_sint64
+#else
+ #error "long size not supported"
+#endif
+
+/* These are defined in types.c */
+extern ffi_type ffi_type_void;
+extern ffi_type ffi_type_uint8;
+extern ffi_type ffi_type_sint8;
+extern ffi_type ffi_type_uint16;
+extern ffi_type ffi_type_sint16;
+extern ffi_type ffi_type_uint32;
+extern ffi_type ffi_type_sint32;
+extern ffi_type ffi_type_uint64;
+extern ffi_type ffi_type_sint64;
+extern ffi_type ffi_type_float;
+extern ffi_type ffi_type_double;
+extern ffi_type ffi_type_pointer;
+
+#if @HAVE_LONG_DOUBLE@
+extern ffi_type ffi_type_longdouble;
+#else
+#define ffi_type_longdouble ffi_type_double
+#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;
+
+/* ---- Definitions for the raw API -------------------------------------- */
+
+#ifndef FFI_SIZEOF_ARG
+# if LONG_MAX == 2147483647
+#  define FFI_SIZEOF_ARG        4
+# elif LONG_MAX == 9223372036854775807
+#  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
+
+typedef struct {
+  char tramp[FFI_TRAMPOLINE_SIZE];
+  ffi_cif   *cif;
+  void     (*fun)(ffi_cif*,void*,void**,void*);
+  void      *user_data;
+#ifdef __GNUC__
+} ffi_closure __attribute__((aligned (8)));
+#else
+} ffi_closure;
+#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);
+
+typedef struct {
+  char tramp[FFI_TRAMPOLINE_SIZE];
+
+  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 {
+  char tramp[FFI_TRAMPOLINE_SIZE];
+
+  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 */
+
+/* ---- Public interface definition -------------------------------------- */
+
+ffi_status ffi_prep_cif(ffi_cif *cif,
+                       ffi_abi abi,
+                       unsigned int nargs,
+                       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
+
+/* This should always refer to the last type code (for sanity checks) */
+#define FFI_TYPE_LAST       FFI_TYPE_POINTER
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/.pc/msvc/src/prep_cif.c b/.pc/msvc/src/prep_cif.c
new file mode 100644 (file)
index 0000000..eb78f9a
--- /dev/null
@@ -0,0 +1,174 @@
+/* -----------------------------------------------------------------------
+   prep_cif.c - Copyright (c) 1996, 1998, 2007  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.
+   ----------------------------------------------------------------------- */
+
+#include <ffi.h>
+#include <ffi_common.h>
+#include <stdlib.h>
+
+/* Round up to FFI_SIZEOF_ARG. */
+
+#define STACK_ARG_SIZE(x) ALIGN(x, FFI_SIZEOF_ARG)
+
+/* Perform machine independent initialization of aggregate type
+   specifications. */
+
+static ffi_status initialize_aggregate(ffi_type *arg)
+{
+  ffi_type **ptr;
+
+  FFI_ASSERT(arg != NULL);
+
+  FFI_ASSERT(arg->elements != NULL);
+  FFI_ASSERT(arg->size == 0);
+  FFI_ASSERT(arg->alignment == 0);
+
+  ptr = &(arg->elements[0]);
+
+  while ((*ptr) != NULL)
+    {
+      if (((*ptr)->size == 0) && (initialize_aggregate((*ptr)) != FFI_OK))
+       return FFI_BAD_TYPEDEF;
+
+      /* Perform a sanity check on the argument type */
+      FFI_ASSERT_VALID_TYPE(*ptr);
+
+      arg->size = ALIGN(arg->size, (*ptr)->alignment);
+      arg->size += (*ptr)->size;
+
+      arg->alignment = (arg->alignment > (*ptr)->alignment) ?
+       arg->alignment : (*ptr)->alignment;
+
+      ptr++;
+    }
+
+  /* Structure size includes tail padding.  This is important for
+     structures that fit in one register on ABIs like the PowerPC64
+     Linux ABI that right justify small structs in a register.
+     It's also needed for nested structure layout, for example
+     struct A { long a; char b; }; struct B { struct A x; char y; };
+     should find y at an offset of 2*sizeof(long) and result in a
+     total size of 3*sizeof(long).  */
+  arg->size = ALIGN (arg->size, arg->alignment);
+
+  if (arg->size == 0)
+    return FFI_BAD_TYPEDEF;
+  else
+    return FFI_OK;
+}
+
+#ifndef __CRIS__
+/* The CRIS ABI specifies structure elements to have byte
+   alignment only, so it completely overrides this functions,
+   which assumes "natural" alignment and padding.  */
+
+/* Perform machine independent ffi_cif preparation, then call
+   machine dependent routine. */
+
+ffi_status ffi_prep_cif(ffi_cif *cif, ffi_abi abi, unsigned int nargs,
+                       ffi_type *rtype, ffi_type **atypes)
+{
+  unsigned bytes = 0;
+  unsigned int i;
+  ffi_type **ptr;
+
+  FFI_ASSERT(cif != NULL);
+  FFI_ASSERT((abi > FFI_FIRST_ABI) && (abi <= FFI_DEFAULT_ABI));
+
+  cif->abi = abi;
+  cif->arg_types = atypes;
+  cif->nargs = nargs;
+  cif->rtype = rtype;
+
+  cif->flags = 0;
+
+  /* Initialize the return type if necessary */
+  if ((cif->rtype->size == 0) && (initialize_aggregate(cif->rtype) != FFI_OK))
+    return FFI_BAD_TYPEDEF;
+
+  /* Perform a sanity check on the return type */
+  FFI_ASSERT_VALID_TYPE(cif->rtype);
+
+  /* x86-64 and s390 stack space allocation is handled in prep_machdep.  */
+#if !defined M68K && !defined __x86_64__ && !defined S390 && !defined PA
+  /* Make space for the return structure pointer */
+  if (cif->rtype->type == FFI_TYPE_STRUCT
+#ifdef SPARC
+      && (cif->abi != FFI_V9 || cif->rtype->size > 32)
+#endif
+#ifdef X86_DARWIN
+      && (cif->rtype->size > 8)
+#endif
+     )
+    bytes = STACK_ARG_SIZE(sizeof(void*));
+#endif
+
+  for (ptr = cif->arg_types, i = cif->nargs; i > 0; i--, ptr++)
+    {
+
+      /* Initialize any uninitialized aggregate type definitions */
+      if (((*ptr)->size == 0) && (initialize_aggregate((*ptr)) != FFI_OK))
+       return FFI_BAD_TYPEDEF;
+
+      /* Perform a sanity check on the argument type, do this
+        check after the initialization.  */
+      FFI_ASSERT_VALID_TYPE(*ptr);
+
+#if !defined __x86_64__ && !defined S390 && !defined PA
+#ifdef SPARC
+      if (((*ptr)->type == FFI_TYPE_STRUCT
+          && ((*ptr)->size > 16 || cif->abi != FFI_V9))
+         || ((*ptr)->type == FFI_TYPE_LONGDOUBLE
+             && cif->abi != FFI_V9))
+       bytes += sizeof(void*);
+      else
+#endif
+       {
+         /* Add any padding if necessary */
+         if (((*ptr)->alignment - 1) & bytes)
+           bytes = ALIGN(bytes, (*ptr)->alignment);
+
+         bytes += STACK_ARG_SIZE((*ptr)->size);
+       }
+#endif
+    }
+
+  cif->bytes = bytes;
+
+  /* Perform machine dependent cif processing */
+  return ffi_prep_cif_machdep(cif);
+}
+#endif /* not __CRIS__ */
+
+#if FFI_CLOSURES
+
+ffi_status
+ffi_prep_closure (ffi_closure* closure,
+                 ffi_cif* cif,
+                 void (*fun)(ffi_cif*,void*,void**,void*),
+                 void *user_data)
+{
+  return ffi_prep_closure_loc (closure, cif, fun, user_data, closure);
+}
+
+#endif
diff --git a/.pc/msvc/src/x86/ffi.c b/.pc/msvc/src/x86/ffi.c
new file mode 100644 (file)
index 0000000..594235a
--- /dev/null
@@ -0,0 +1,682 @@
+/* -----------------------------------------------------------------------
+   ffi.c - Copyright (c) 1996, 1998, 1999, 2001, 2007, 2008  Red Hat, Inc.
+           Copyright (c) 2002  Ranjit Mathew
+           Copyright (c) 2002  Bo Thorsen
+           Copyright (c) 2002  Roger Sayle
+           Copyright (C) 2008  Free Software Foundation, Inc.
+
+   x86 Foreign Function Interface
+
+   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.
+   ----------------------------------------------------------------------- */
+
+#if !defined(__x86_64__) || defined(_WIN64)
+
+#ifdef _WIN64
+#include <windows.h>
+#endif
+
+#include <ffi.h>
+#include <ffi_common.h>
+
+#include <stdlib.h>
+
+/* ffi_prep_args is called by the assembly routine once stack space
+   has been allocated for the function's arguments */
+
+void ffi_prep_args(char *stack, extended_cif *ecif)
+{
+  register unsigned int i;
+  register void **p_argv;
+  register char *argp;
+  register ffi_type **p_arg;
+
+  argp = stack;
+
+  if (ecif->cif->flags == FFI_TYPE_STRUCT
+#ifdef X86_WIN64
+      && (ecif->cif->rtype->size != 1 && ecif->cif->rtype->size != 2
+          && ecif->cif->rtype->size != 4 && ecif->cif->rtype->size != 8)
+#endif
+      )
+    {
+      *(void **) argp = ecif->rvalue;
+      argp += sizeof(void*);
+    }
+
+  p_argv = ecif->avalue;
+
+  for (i = ecif->cif->nargs, p_arg = ecif->cif->arg_types;
+       i != 0;
+       i--, p_arg++)
+    {
+      size_t z;
+
+      /* Align if necessary */
+      if ((sizeof(void*) - 1) & (size_t) argp)
+        argp = (char *) ALIGN(argp, sizeof(void*));
+
+      z = (*p_arg)->size;
+#ifdef X86_WIN64
+      if (z > sizeof(ffi_arg)
+          || ((*p_arg)->type == FFI_TYPE_STRUCT
+              && (z != 1 && z != 2 && z != 4 && z != 8))
+#if FFI_TYPE_DOUBLE != FFI_TYPE_LONGDOUBLE
+          || ((*p_arg)->type == FFI_TYPE_LONGDOUBLE)
+#endif
+          )
+        {
+          z = sizeof(ffi_arg);
+          *(void **)argp = *p_argv;
+        }
+      else if ((*p_arg)->type == FFI_TYPE_FLOAT)
+        {
+          memcpy(argp, *p_argv, z);
+        }
+      else
+#endif
+      if (z < sizeof(ffi_arg))
+        {
+          z = sizeof(ffi_arg);
+          switch ((*p_arg)->type)
+            {
+            case FFI_TYPE_SINT8:
+              *(ffi_sarg *) argp = (ffi_sarg)*(SINT8 *)(* p_argv);
+              break;
+
+            case FFI_TYPE_UINT8:
+              *(ffi_arg *) argp = (ffi_arg)*(UINT8 *)(* p_argv);
+              break;
+
+            case FFI_TYPE_SINT16:
+              *(ffi_sarg *) argp = (ffi_sarg)*(SINT16 *)(* p_argv);
+              break;
+
+            case FFI_TYPE_UINT16:
+              *(ffi_arg *) argp = (ffi_arg)*(UINT16 *)(* p_argv);
+              break;
+
+            case FFI_TYPE_SINT32:
+              *(ffi_sarg *) argp = (ffi_sarg)*(SINT32 *)(* p_argv);
+              break;
+
+            case FFI_TYPE_UINT32:
+              *(ffi_arg *) argp = (ffi_arg)*(UINT32 *)(* p_argv);
+              break;
+
+            case FFI_TYPE_STRUCT:
+              *(ffi_arg *) argp = *(ffi_arg *)(* p_argv);
+              break;
+
+            default:
+              FFI_ASSERT(0);
+            }
+        }
+      else
+        {
+          memcpy(argp, *p_argv, z);
+        }
+      p_argv++;
+#ifdef X86_WIN64
+      argp += (z + sizeof(void*) - 1) & ~(sizeof(void*) - 1);
+#else
+      argp += z;
+#endif
+    }
+  
+  return;
+}
+
+/* Perform machine dependent cif processing */
+ffi_status ffi_prep_cif_machdep(ffi_cif *cif)
+{
+  /* Set the return type flag */
+  switch (cif->rtype->type)
+    {
+    case FFI_TYPE_VOID:
+#ifdef X86
+    case FFI_TYPE_STRUCT:
+#endif
+#if defined(X86) || defined (X86_WIN32) || defined(X86_FREEBSD) || defined(X86_DARWIN) || defined(X86_WIN64)
+    case FFI_TYPE_UINT8:
+    case FFI_TYPE_UINT16:
+    case FFI_TYPE_SINT8:
+    case FFI_TYPE_SINT16:
+#endif
+#ifdef X86_WIN64
+    case FFI_TYPE_UINT32:
+    case FFI_TYPE_SINT32:
+#endif
+
+    case FFI_TYPE_SINT64:
+    case FFI_TYPE_FLOAT:
+    case FFI_TYPE_DOUBLE:
+#ifndef X86_WIN64
+#if FFI_TYPE_DOUBLE != FFI_TYPE_LONGDOUBLE
+    case FFI_TYPE_LONGDOUBLE:
+#endif
+#endif
+      cif->flags = (unsigned) cif->rtype->type;
+      break;
+
+    case FFI_TYPE_UINT64:
+#ifdef X86_WIN64
+    case FFI_TYPE_POINTER:
+#endif
+      cif->flags = FFI_TYPE_SINT64;
+      break;
+
+#ifndef X86
+    case FFI_TYPE_STRUCT:
+      if (cif->rtype->size == 1)
+        {
+          cif->flags = FFI_TYPE_SMALL_STRUCT_1B; /* same as char size */
+        }
+      else if (cif->rtype->size == 2)
+        {
+          cif->flags = FFI_TYPE_SMALL_STRUCT_2B; /* same as short size */
+        }
+      else if (cif->rtype->size == 4)
+        {
+#ifdef X86_WIN64
+          cif->flags = FFI_TYPE_SMALL_STRUCT_4B;
+#else
+          cif->flags = FFI_TYPE_INT; /* same as int type */
+#endif
+        }
+      else if (cif->rtype->size == 8)
+        {
+          cif->flags = FFI_TYPE_SINT64; /* same as int64 type */
+        }
+      else
+        {
+          cif->flags = FFI_TYPE_STRUCT;
+#ifdef X86_WIN64
+          // allocate space for return value pointer
+          cif->bytes += ALIGN(sizeof(void*), FFI_SIZEOF_ARG);
+#endif
+        }
+      break;
+#endif
+
+    default:
+#ifdef X86_WIN64
+      cif->flags = FFI_TYPE_SINT64;
+      break;
+    case FFI_TYPE_INT:
+      cif->flags = FFI_TYPE_SINT32;
+#else
+      cif->flags = FFI_TYPE_INT;
+#endif
+      break;
+    }
+
+#ifdef X86_DARWIN
+  cif->bytes = (cif->bytes + 15) & ~0xF;
+#endif
+
+#ifdef X86_WIN64
+  {
+    unsigned int i;
+    ffi_type **ptr;
+
+    for (ptr = cif->arg_types, i = cif->nargs; i > 0; i--, ptr++)
+      {
+        if (((*ptr)->alignment - 1) & cif->bytes)
+          cif->bytes = ALIGN(cif->bytes, (*ptr)->alignment);
+        cif->bytes += ALIGN((*ptr)->size, FFI_SIZEOF_ARG);
+      }
+  }
+  // ensure space for storing four registers
+  cif->bytes += 4 * sizeof(ffi_arg);
+#endif
+
+  return FFI_OK;
+}
+
+extern void ffi_call_SYSV(void (*)(char *, extended_cif *), extended_cif *,
+                          unsigned, unsigned, unsigned *, void (*fn)(void));
+
+#ifdef X86_WIN32
+extern void ffi_call_STDCALL(void (*)(char *, extended_cif *), extended_cif *,
+                          unsigned, unsigned, unsigned *, void (*fn)(void));
+
+#endif /* X86_WIN32 */
+#ifdef X86_WIN64
+extern int
+ffi_call_win64(void (*)(char *, extended_cif *), extended_cif *,
+               unsigned, unsigned, unsigned *, void (*fn)(void));
+#endif
+
+void ffi_call(ffi_cif *cif, void (*fn)(void), void *rvalue, void **avalue)
+{
+  extended_cif ecif;
+
+  ecif.cif = cif;
+  ecif.avalue = avalue;
+  
+  /* If the return value is a struct and we don't have a return */
+  /* value address then we need to make one                     */
+
+#ifdef X86_WIN64
+  if (rvalue == NULL
+      && cif->flags == FFI_TYPE_STRUCT
+      && cif->rtype->size != 1 && cif->rtype->size != 2
+      && cif->rtype->size != 4 && cif->rtype->size != 8)
+    {
+      ecif.rvalue = alloca((cif->rtype->size + 0xF) & ~0xF);
+    }
+#else
+  if (rvalue == NULL
+      && cif->flags == FFI_TYPE_STRUCT)
+    {
+      ecif.rvalue = alloca(cif->rtype->size);
+    }
+#endif
+  else
+    ecif.rvalue = rvalue;
+    
+  
+  switch (cif->abi) 
+    {
+#ifdef X86_WIN64
+    case FFI_WIN64:
+      {
+        // Make copies of all struct arguments
+        // NOTE: not sure if responsibility should be here or in caller
+        unsigned int i;
+        for (i=0; i < cif->nargs;i++) {
+          size_t size = cif->arg_types[i]->size;
+          if ((cif->arg_types[i]->type == FFI_TYPE_STRUCT
+               && (size != 1 && size != 2 && size != 4 && size != 8))
+#if FFI_TYPE_LONGDOUBLE != FFI_TYPE_DOUBLE
+              || cif->arg_types[i]->type == FFI_TYPE_LONGDOUBLE
+#endif
+              )
+            {
+              void *local = alloca(size);
+              memcpy(local, avalue[i], size);
+              avalue[i] = local;
+            }
+        }
+        ffi_call_win64(ffi_prep_args, &ecif, cif->bytes,
+                       cif->flags, ecif.rvalue, fn);
+      }
+      break;
+#else
+    case FFI_SYSV:
+      ffi_call_SYSV(ffi_prep_args, &ecif, cif->bytes, cif->flags, ecif.rvalue,
+                    fn);
+      break;
+#ifdef X86_WIN32
+    case FFI_STDCALL:
+      ffi_call_STDCALL(ffi_prep_args, &ecif, cif->bytes, cif->flags,
+                       ecif.rvalue, fn);
+      break;
+#endif /* X86_WIN32 */
+#endif /* X86_WIN64 */
+    default:
+      FFI_ASSERT(0);
+      break;
+    }
+}
+
+
+/** private members **/
+
+static void ffi_prep_incoming_args_SYSV (char *stack, void **ret,
+                                         void** args, ffi_cif* cif);
+void FFI_HIDDEN ffi_closure_SYSV (ffi_closure *)
+     __attribute__ ((regparm(1)));
+unsigned int FFI_HIDDEN ffi_closure_SYSV_inner (ffi_closure *, void **, void *)
+     __attribute__ ((regparm(1)));
+void FFI_HIDDEN ffi_closure_raw_SYSV (ffi_raw_closure *)
+     __attribute__ ((regparm(1)));
+#ifdef X86_WIN32
+void FFI_HIDDEN ffi_closure_STDCALL (ffi_closure *)
+     __attribute__ ((regparm(1)));
+#endif
+#ifdef X86_WIN64
+void FFI_HIDDEN ffi_closure_win64 (ffi_closure *);
+#endif
+
+/* This function is jumped to by the trampoline */
+
+#ifdef X86_WIN64
+void * FFI_HIDDEN
+ffi_closure_win64_inner (ffi_closure *closure, void *args) {
+  ffi_cif       *cif;
+  void         **arg_area;
+  void          *result;
+  void          *resp = &result;
+
+  cif         = closure->cif;
+  arg_area    = (void**) alloca (cif->nargs * sizeof (void*));  
+
+  /* this call will initialize ARG_AREA, such that each
+   * element in that array points to the corresponding 
+   * value on the stack; and if the function returns
+   * a structure, it will change RESP to point to the
+   * structure return address.  */
+
+  ffi_prep_incoming_args_SYSV(args, &resp, arg_area, cif);
+  
+  (closure->fun) (cif, resp, arg_area, closure->user_data);
+
+  /* The result is returned in rax.  This does the right thing for
+     result types except for floats; we have to 'mov xmm0, rax' in the
+     caller to correct this.
+     TODO: structure sizes of 3 5 6 7 are returned by reference, too!!!
+  */
+  return cif->rtype->size > sizeof(void *) ? resp : *(void **)resp;
+}
+
+#else
+unsigned int FFI_HIDDEN
+ffi_closure_SYSV_inner (closure, respp, args)
+     ffi_closure *closure;
+     void **respp;
+     void *args;
+{
+  /* our various things...  */
+  ffi_cif       *cif;
+  void         **arg_area;
+
+  cif         = closure->cif;
+  arg_area    = (void**) alloca (cif->nargs * sizeof (void*));  
+
+  /* this call will initialize ARG_AREA, such that each
+   * element in that array points to the corresponding 
+   * value on the stack; and if the function returns
+   * a structure, it will change RESP to point to the
+   * structure return address.  */
+
+  ffi_prep_incoming_args_SYSV(args, respp, arg_area, cif);
+
+  (closure->fun) (cif, *respp, arg_area, closure->user_data);
+
+  return cif->flags;
+}
+#endif /* !X86_WIN64 */
+
+static void
+ffi_prep_incoming_args_SYSV(char *stack, void **rvalue, void **avalue,
+                            ffi_cif *cif)
+{
+  register unsigned int i;
+  register void **p_argv;
+  register char *argp;
+  register ffi_type **p_arg;
+
+  argp = stack;
+
+#ifdef X86_WIN64
+  if (cif->rtype->size > sizeof(ffi_arg)
+      || (cif->flags == FFI_TYPE_STRUCT
+          && (cif->rtype->size != 1 && cif->rtype->size != 2
+              && cif->rtype->size != 4 && cif->rtype->size != 8))) {
+    *rvalue = *(void **) argp;
+    argp += sizeof(void *);
+  }
+#else
+  if ( cif->flags == FFI_TYPE_STRUCT ) {
+    *rvalue = *(void **) argp;
+    argp += sizeof(void *);
+  }
+#endif
+
+  p_argv = avalue;
+
+  for (i = cif->nargs, p_arg = cif->arg_types; (i != 0); i--, p_arg++)
+    {
+      size_t z;
+
+      /* Align if necessary */
+      if ((sizeof(void*) - 1) & (size_t) argp) {
+        argp = (char *) ALIGN(argp, sizeof(void*));
+      }
+
+#ifdef X86_WIN64
+      if ((*p_arg)->size > sizeof(ffi_arg)
+          || ((*p_arg)->type == FFI_TYPE_STRUCT
+              && ((*p_arg)->size != 1 && (*p_arg)->size != 2
+                  && (*p_arg)->size != 4 && (*p_arg)->size != 8)))
+        {
+          z = sizeof(void *);
+          *p_argv = *(void **)argp;
+        }
+      else
+#endif
+        {
+          z = (*p_arg)->size;
+          
+          /* because we're little endian, this is what it turns into.   */
+          
+          *p_argv = (void*) argp;
+        }
+          
+      p_argv++;
+#ifdef X86_WIN64
+      argp += (z + sizeof(void*) - 1) & ~(sizeof(void*) - 1);
+#else
+      argp += z;
+#endif
+    }
+  
+  return;
+}
+
+#define FFI_INIT_TRAMPOLINE_WIN64(TRAMP,FUN,CTX,MASK) \
+{ unsigned char *__tramp = (unsigned char*)(TRAMP); \
+   void*  __fun = (void*)(FUN); \
+   void*  __ctx = (void*)(CTX); \
+   *(unsigned char*) &__tramp[0] = 0x41; \
+   *(unsigned char*) &__tramp[1] = 0xbb; \
+   *(unsigned int*) &__tramp[2] = MASK; /* mov $mask, %r11 */ \
+   *(unsigned char*) &__tramp[6] = 0x48; \
+   *(unsigned char*) &__tramp[7] = 0xb8; \
+   *(void**) &__tramp[8] = __ctx; /* mov __ctx, %rax */ \
+   *(unsigned char *)  &__tramp[16] = 0x49; \
+   *(unsigned char *)  &__tramp[17] = 0xba; \
+   *(void**) &__tramp[18] = __fun; /* mov __fun, %r10 */ \
+   *(unsigned char *)  &__tramp[26] = 0x41; \
+   *(unsigned char *)  &__tramp[27] = 0xff; \
+   *(unsigned char *)  &__tramp[28] = 0xe2; /* jmp %r10 */ \
+ }
+
+/* How to make a trampoline.  Derived from gcc/config/i386/i386.c. */
+
+#define FFI_INIT_TRAMPOLINE(TRAMP,FUN,CTX) \
+({ unsigned char *__tramp = (unsigned char*)(TRAMP); \
+   unsigned int  __fun = (unsigned int)(FUN); \
+   unsigned int  __ctx = (unsigned int)(CTX); \
+   unsigned int  __dis = __fun - (__ctx + 10);  \
+   *(unsigned char*) &__tramp[0] = 0xb8; \
+   *(unsigned int*)  &__tramp[1] = __ctx; /* movl __ctx, %eax */ \
+   *(unsigned char *)  &__tramp[5] = 0xe9; \
+   *(unsigned int*)  &__tramp[6] = __dis; /* jmp __fun  */ \
+ })
+
+#define FFI_INIT_TRAMPOLINE_STDCALL(TRAMP,FUN,CTX,SIZE)  \
+({ unsigned char *__tramp = (unsigned char*)(TRAMP); \
+   unsigned int  __fun = (unsigned int)(FUN); \
+   unsigned int  __ctx = (unsigned int)(CTX); \
+   unsigned int  __dis = __fun - (__ctx + 10); \
+   unsigned short __size = (unsigned short)(SIZE); \
+   *(unsigned char*) &__tramp[0] = 0xb8; \
+   *(unsigned int*)  &__tramp[1] = __ctx; /* movl __ctx, %eax */ \
+   *(unsigned char *)  &__tramp[5] = 0xe8; \
+   *(unsigned int*)  &__tramp[6] = __dis; /* call __fun  */ \
+   *(unsigned char *)  &__tramp[10] = 0xc2; \
+   *(unsigned short*)  &__tramp[11] = __size; /* ret __size  */ \
+ })
+
+/* the cif must already be prep'ed */
+
+ffi_status
+ffi_prep_closure_loc (ffi_closure* closure,
+                      ffi_cif* cif,
+                      void (*fun)(ffi_cif*,void*,void**,void*),
+                      void *user_data,
+                      void *codeloc)
+{
+#ifdef X86_WIN64
+#define ISFLOAT(IDX) (cif->arg_types[IDX]->type == FFI_TYPE_FLOAT || cif->arg_types[IDX]->type == FFI_TYPE_DOUBLE)
+#define FLAG(IDX) (cif->nargs>(IDX)&&ISFLOAT(IDX)?(1<<(IDX)):0)
+  if (cif->abi == FFI_WIN64) 
+    {
+      int mask = FLAG(0)|FLAG(1)|FLAG(2)|FLAG(3);
+      FFI_INIT_TRAMPOLINE_WIN64 (&closure->tramp[0],
+                                 &ffi_closure_win64,
+                                 codeloc, mask);
+      /* make sure we can execute here */
+    }
+#else
+  if (cif->abi == FFI_SYSV)
+    {
+      FFI_INIT_TRAMPOLINE (&closure->tramp[0],
+                           &ffi_closure_SYSV,
+                           (void*)codeloc);
+    }
+#ifdef X86_WIN32
+  else if (cif->abi == FFI_STDCALL)
+    {
+      FFI_INIT_TRAMPOLINE_STDCALL (&closure->tramp[0],
+                                   &ffi_closure_STDCALL,
+                                   (void*)codeloc, cif->bytes);
+    }
+#endif /* X86_WIN32 */
+#endif /* !X86_WIN64 */
+  else
+    {
+      return FFI_BAD_ABI;
+    }
+    
+  closure->cif  = cif;
+  closure->user_data = user_data;
+  closure->fun  = fun;
+
+  return FFI_OK;
+}
+
+/* ------- Native raw API support -------------------------------- */
+
+#if !FFI_NO_RAW_API
+
+ffi_status
+ffi_prep_raw_closure_loc (ffi_raw_closure* closure,
+                          ffi_cif* cif,
+                          void (*fun)(ffi_cif*,void*,ffi_raw*,void*),
+                          void *user_data,
+                          void *codeloc)
+{
+  int i;
+
+  if (cif->abi != FFI_SYSV) {
+    return FFI_BAD_ABI;
+  }
+
+  // we currently don't support certain kinds of arguments for raw
+  // closures.  This should be implemented by a separate assembly language
+  // routine, since it would require argument processing, something we
+  // don't do now for performance.
+
+  for (i = cif->nargs-1; i >= 0; i--)
+    {
+      FFI_ASSERT (cif->arg_types[i]->type != FFI_TYPE_STRUCT);
+      FFI_ASSERT (cif->arg_types[i]->type != FFI_TYPE_LONGDOUBLE);
+    }
+  
+
+  FFI_INIT_TRAMPOLINE (&closure->tramp[0], &ffi_closure_raw_SYSV,
+                       codeloc);
+    
+  closure->cif  = cif;
+  closure->user_data = user_data;
+  closure->fun  = fun;
+
+  return FFI_OK;
+}
+
+static void 
+ffi_prep_args_raw(char *stack, extended_cif *ecif)
+{
+  memcpy (stack, ecif->avalue, ecif->cif->bytes);
+}
+
+/* we borrow this routine from libffi (it must be changed, though, to
+ * actually call the function passed in the first argument.  as of
+ * libffi-1.20, this is not the case.)
+ */
+
+extern void
+ffi_call_SYSV(void (*)(char *, extended_cif *), extended_cif *, unsigned, 
+              unsigned, unsigned *, void (*fn)(void));
+
+#ifdef X86_WIN32
+extern void
+ffi_call_STDCALL(void (*)(char *, extended_cif *), extended_cif *, unsigned,
+                 unsigned, unsigned *, void (*fn)(void));
+#endif /* X86_WIN32 */
+
+void
+ffi_raw_call(ffi_cif *cif, void (*fn)(void), void *rvalue, ffi_raw *fake_avalue)
+{
+  extended_cif ecif;
+  void **avalue = (void **)fake_avalue;
+
+  ecif.cif = cif;
+  ecif.avalue = avalue;
+  
+  /* If the return value is a struct and we don't have a return */
+  /* value address then we need to make one                     */
+
+  if ((rvalue == NULL) && 
+      (cif->rtype->type == FFI_TYPE_STRUCT))
+    {
+      ecif.rvalue = alloca(cif->rtype->size);
+    }
+  else
+    ecif.rvalue = rvalue;
+    
+  
+  switch (cif->abi) 
+    {
+    case FFI_SYSV:
+      ffi_call_SYSV(ffi_prep_args_raw, &ecif, cif->bytes, cif->flags,
+                    ecif.rvalue, fn);
+      break;
+#ifdef X86_WIN32
+    case FFI_STDCALL:
+      ffi_call_STDCALL(ffi_prep_args_raw, &ecif, cif->bytes, cif->flags,
+                       ecif.rvalue, fn);
+      break;
+#endif /* X86_WIN32 */
+    default:
+      FFI_ASSERT(0);
+      break;
+    }
+}
+
+#endif
+
+#endif /* !__x86_64__  || X86_WIN64 */
+
diff --git a/.pc/msvc/src/x86/win32.S b/.pc/msvc/src/x86/win32.S
new file mode 100644 (file)
index 0000000..7489dab
--- /dev/null
@@ -0,0 +1,877 @@
+/* -----------------------------------------------------------------------
+   win32.S - Copyright (c) 1996, 1998, 2001, 2002, 2009  Red Hat, Inc.
+            Copyright (c) 2001  John Beniton
+            Copyright (c) 2002  Ranjit Mathew
+                       
+   X86 Foreign Function Interface
+   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.
+   -----------------------------------------------------------------------
+   */
+#define LIBFFI_ASM
+#include <fficonfig.h>
+#include <ffi.h>
+       .text
+        # This assumes we are using gas.
+        .balign 16
+       .globl  _ffi_call_SYSV
+       .def    _ffi_call_SYSV; .scl    2;      .type   32;     .endef
+_ffi_call_SYSV:
+.LFB1:
+        pushl %ebp
+.LCFI0:
+        movl  %esp,%ebp
+.LCFI1:
+        # Make room for all of the new args.
+        movl  16(%ebp),%ecx                                                     
+        subl  %ecx,%esp
+        movl  %esp,%eax
+        # Place all of the ffi_prep_args in position
+        pushl 12(%ebp)
+        pushl %eax
+        call  *8(%ebp)
+        # Return stack to previous state and call the function
+        addl  $8,%esp
+        # FIXME: Align the stack to a 128-bit boundary to avoid
+        # potential performance hits.
+
+       call  *28(%ebp)
+        # Load %ecx with the return type code
+        movl  20(%ebp),%ecx
+        # If the return value pointer is NULL, assume no return value.
+        cmpl  $0,24(%ebp)
+        jne   0f
+        # Even if there is no space for the return value, we are
+        # obliged to handle floating-point values.
+        cmpl  $FFI_TYPE_FLOAT,%ecx
+        jne   .Lnoretval
+        fstp  %st(0)
+        jmp   .Lepilogue
+
+0:
+       call    1f
+       # Do not insert anything here between the call and the jump table.
+.Lstore_table:
+       .long   .Lnoretval              /* FFI_TYPE_VOID */
+       .long   .Lretint                /* FFI_TYPE_INT */
+       .long   .Lretfloat              /* FFI_TYPE_FLOAT */
+       .long   .Lretdouble             /* FFI_TYPE_DOUBLE */
+       .long   .Lretlongdouble         /* FFI_TYPE_LONGDOUBLE */
+       .long   .Lretuint8              /* FFI_TYPE_UINT8 */
+       .long   .Lretsint8              /* FFI_TYPE_SINT8 */
+       .long   .Lretuint16             /* FFI_TYPE_UINT16 */
+       .long   .Lretsint16             /* FFI_TYPE_SINT16 */
+       .long   .Lretint                /* FFI_TYPE_UINT32 */
+       .long   .Lretint                /* FFI_TYPE_SINT32 */
+       .long   .Lretint64              /* FFI_TYPE_UINT64 */
+       .long   .Lretint64              /* FFI_TYPE_SINT64 */
+       .long   .Lretstruct             /* FFI_TYPE_STRUCT */
+       .long   .Lretint                /* FFI_TYPE_POINTER */
+       .long   .Lretstruct1b           /* FFI_TYPE_SMALL_STRUCT_1B */
+       .long   .Lretstruct2b           /* FFI_TYPE_SMALL_STRUCT_2B */
+       .long   .Lretstruct4b           /* FFI_TYPE_SMALL_STRUCT_4B */
+1:
+       add     %ecx, %ecx
+       add     %ecx, %ecx
+       add     (%esp),%ecx
+       add     $4, %esp
+       jmp     *(%ecx)
+
+       /* Sign/zero extend as appropriate.  */
+.Lretsint8:
+       movsbl  %al, %eax
+       jmp     .Lretint
+
+.Lretsint16:
+       movswl  %ax, %eax
+       jmp     .Lretint
+
+.Lretuint8:
+       movzbl  %al, %eax
+       jmp     .Lretint
+
+.Lretuint16:
+       movzwl  %ax, %eax
+       jmp     .Lretint
+
+.Lretint:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        movl  %eax,0(%ecx)
+        jmp   .Lepilogue
+.Lretfloat:
+         # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        fstps (%ecx)
+        jmp   .Lepilogue
+.Lretdouble:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        fstpl (%ecx)
+        jmp   .Lepilogue
+.Lretlongdouble:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        fstpt (%ecx)
+        jmp   .Lepilogue
+.Lretint64:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        movl  %eax,0(%ecx)
+        movl  %edx,4(%ecx)
+       jmp   .Lepilogue
+
+.Lretstruct1b:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        movb  %al,0(%ecx)
+        jmp   .Lepilogue
+.Lretstruct2b:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        movw  %ax,0(%ecx)
+        jmp   .Lepilogue
+
+.Lretstruct4b:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        movl  %eax,0(%ecx)
+        jmp   .Lepilogue
+
+.Lretstruct:
+        # Nothing to do!
+.Lnoretval:
+.Lepilogue:
+        movl %ebp,%esp
+        popl %ebp
+        ret
+.ffi_call_SYSV_end:
+.LFE1:
+
+        # This assumes we are using gas.
+        .balign 16
+       .globl  _ffi_call_STDCALL
+       .def    _ffi_call_STDCALL;      .scl    2;      .type   32;     .endef
+_ffi_call_STDCALL:
+.LFB2:
+        pushl %ebp
+.LCFI2:
+        movl  %esp,%ebp
+.LCFI3:
+        # Make room for all of the new args.
+        movl  16(%ebp),%ecx 
+        subl  %ecx,%esp
+
+        movl  %esp,%eax
+
+        # Place all of the ffi_prep_args in position
+        pushl 12(%ebp)
+        pushl %eax
+        call  *8(%ebp)
+
+        # Return stack to previous state and call the function
+        addl  $8,%esp
+
+        # FIXME: Align the stack to a 128-bit boundary to avoid
+        # potential performance hits.
+
+        call  *28(%ebp)
+
+        # stdcall functions pop arguments off the stack themselves
+
+        # Load %ecx with the return type code
+        movl  20(%ebp),%ecx
+
+        # If the return value pointer is NULL, assume no return value.
+        cmpl  $0,24(%ebp)
+        jne   0f
+
+        # Even if there is no space for the return value, we are
+        # obliged to handle floating-point values.
+        cmpl  $FFI_TYPE_FLOAT,%ecx
+        jne   .Lsc_noretval
+        fstp  %st(0)
+
+        jmp   .Lsc_epilogue
+
+0:
+       call    1f
+       # Do not insert anything here between the call and the jump table.
+.Lsc_store_table:
+       .long   .Lsc_noretval           /* FFI_TYPE_VOID */
+       .long   .Lsc_retint             /* FFI_TYPE_INT */
+       .long   .Lsc_retfloat           /* FFI_TYPE_FLOAT */
+       .long   .Lsc_retdouble          /* FFI_TYPE_DOUBLE */
+       .long   .Lsc_retlongdouble      /* FFI_TYPE_LONGDOUBLE */
+       .long   .Lsc_retuint8           /* FFI_TYPE_UINT8 */
+       .long   .Lsc_retsint8           /* FFI_TYPE_SINT8 */
+       .long   .Lsc_retuint16          /* FFI_TYPE_UINT16 */
+       .long   .Lsc_retsint16          /* FFI_TYPE_SINT16 */
+       .long   .Lsc_retint             /* FFI_TYPE_UINT32 */
+       .long   .Lsc_retint             /* FFI_TYPE_SINT32 */
+       .long   .Lsc_retint64           /* FFI_TYPE_UINT64 */
+       .long   .Lsc_retint64           /* FFI_TYPE_SINT64 */
+       .long   .Lsc_retstruct          /* FFI_TYPE_STRUCT */
+       .long   .Lsc_retint             /* FFI_TYPE_POINTER */
+       .long   .Lsc_retstruct1b        /* FFI_TYPE_SMALL_STRUCT_1B */
+       .long   .Lsc_retstruct2b        /* FFI_TYPE_SMALL_STRUCT_2B */
+       .long   .Lsc_retstruct4b        /* FFI_TYPE_SMALL_STRUCT_4B */
+
+1:
+       add     %ecx, %ecx
+       add     %ecx, %ecx
+       add     (%esp),%ecx
+       add     $4, %esp
+       jmp     *(%ecx)
+
+       /* Sign/zero extend as appropriate.  */
+.Lsc_retsint8:
+       movsbl  %al, %eax
+       jmp     .Lsc_retint
+
+.Lsc_retsint16:
+       movswl  %ax, %eax
+       jmp     .Lsc_retint
+
+.Lsc_retuint8:
+       movzbl  %al, %eax
+       jmp     .Lsc_retint
+
+.Lsc_retuint16:
+       movzwl  %ax, %eax
+       jmp     .Lsc_retint
+
+.Lsc_retint:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        movl  %eax,0(%ecx)
+        jmp   .Lsc_epilogue
+
+.Lsc_retfloat:
+         # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        fstps (%ecx)
+        jmp   .Lsc_epilogue
+
+.Lsc_retdouble:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        fstpl (%ecx)
+        jmp   .Lsc_epilogue
+
+.Lsc_retlongdouble:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        fstpt (%ecx)
+        jmp   .Lsc_epilogue
+
+.Lsc_retint64:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        movl  %eax,0(%ecx)
+        movl  %edx,4(%ecx)
+       jmp   .Lsc_epilogue
+
+.Lsc_retstruct1b:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        movb  %al,0(%ecx)
+        jmp   .Lsc_epilogue
+
+.Lsc_retstruct2b:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        movw  %ax,0(%ecx)
+        jmp   .Lsc_epilogue
+
+.Lsc_retstruct4b:
+        # Load %ecx with the pointer to storage for the return value
+        movl  24(%ebp),%ecx
+        movl  %eax,0(%ecx)
+        jmp   .Lsc_epilogue
+
+.Lsc_retstruct:
+        # Nothing to do!
+
+.Lsc_noretval:
+.Lsc_epilogue:
+        movl %ebp,%esp
+        popl %ebp
+        ret
+.ffi_call_STDCALL_end:
+.LFE2:
+
+        # This assumes we are using gas.
+        .balign 16
+       .globl  _ffi_closure_SYSV
+       .def    _ffi_closure_SYSV;      .scl    2;      .type   32;     .endef
+_ffi_closure_SYSV:
+.LFB3:
+       pushl   %ebp
+.LCFI4:
+       movl    %esp, %ebp
+.LCFI5:
+       subl    $40, %esp
+       leal    -24(%ebp), %edx
+       movl    %edx, -12(%ebp) /* resp */
+       leal    8(%ebp), %edx
+       movl    %edx, 4(%esp)   /* args = __builtin_dwarf_cfa () */
+       leal    -12(%ebp), %edx
+       movl    %edx, (%esp)    /* &resp */
+       call    _ffi_closure_SYSV_inner
+       movl    -12(%ebp), %ecx
+
+0:
+       call    1f
+       # Do not insert anything here between the call and the jump table.
+.Lcls_store_table:
+       .long   .Lcls_noretval          /* FFI_TYPE_VOID */
+       .long   .Lcls_retint            /* FFI_TYPE_INT */
+       .long   .Lcls_retfloat          /* FFI_TYPE_FLOAT */
+       .long   .Lcls_retdouble         /* FFI_TYPE_DOUBLE */
+       .long   .Lcls_retldouble        /* FFI_TYPE_LONGDOUBLE */
+       .long   .Lcls_retuint8          /* FFI_TYPE_UINT8 */
+       .long   .Lcls_retsint8          /* FFI_TYPE_SINT8 */
+       .long   .Lcls_retuint16         /* FFI_TYPE_UINT16 */
+       .long   .Lcls_retsint16         /* FFI_TYPE_SINT16 */
+       .long   .Lcls_retint            /* FFI_TYPE_UINT32 */
+       .long   .Lcls_retint            /* FFI_TYPE_SINT32 */
+       .long   .Lcls_retllong          /* FFI_TYPE_UINT64 */
+       .long   .Lcls_retllong          /* FFI_TYPE_SINT64 */
+       .long   .Lcls_retstruct         /* FFI_TYPE_STRUCT */
+       .long   .Lcls_retint            /* FFI_TYPE_POINTER */
+       .long   .Lcls_retstruct1        /* FFI_TYPE_SMALL_STRUCT_1B */
+       .long   .Lcls_retstruct2        /* FFI_TYPE_SMALL_STRUCT_2B */
+       .long   .Lcls_retstruct4        /* FFI_TYPE_SMALL_STRUCT_4B */
+
+1:
+       add     %eax, %eax
+       add     %eax, %eax
+       add     (%esp),%eax
+       add     $4, %esp
+       jmp     *(%eax)
+
+       /* Sign/zero extend as appropriate.  */
+.Lcls_retsint8:
+       movsbl  (%ecx), %eax
+       jmp     .Lcls_epilogue
+
+.Lcls_retsint16:
+       movswl  (%ecx), %eax
+       jmp     .Lcls_epilogue
+
+.Lcls_retuint8:
+       movzbl  (%ecx), %eax
+       jmp     .Lcls_epilogue
+
+.Lcls_retuint16:
+       movzwl  (%ecx), %eax
+       jmp     .Lcls_epilogue
+
+.Lcls_retint:
+       movl    (%ecx), %eax
+       jmp     .Lcls_epilogue
+
+.Lcls_retfloat:
+       flds    (%ecx)
+       jmp     .Lcls_epilogue
+
+.Lcls_retdouble:
+       fldl    (%ecx)
+       jmp     .Lcls_epilogue
+
+.Lcls_retldouble:
+       fldt    (%ecx)
+       jmp     .Lcls_epilogue
+
+.Lcls_retllong:
+       movl    (%ecx), %eax
+       movl    4(%ecx), %edx
+       jmp     .Lcls_epilogue
+
+.Lcls_retstruct1:
+       movsbl  (%ecx), %eax
+       jmp     .Lcls_epilogue
+
+.Lcls_retstruct2:
+       movswl  (%ecx), %eax
+       jmp     .Lcls_epilogue
+
+.Lcls_retstruct4:
+       movl    (%ecx), %eax
+       jmp     .Lcls_epilogue
+
+.Lcls_retstruct:
+        # Caller expects us to pop struct return value pointer hidden arg.
+       movl    %ebp, %esp
+       popl    %ebp
+       ret     $0x4
+
+.Lcls_noretval:
+.Lcls_epilogue:
+       movl    %ebp, %esp
+       popl    %ebp
+       ret
+.ffi_closure_SYSV_end:
+.LFE3:
+
+#if !FFI_NO_RAW_API
+
+#define RAW_CLOSURE_CIF_OFFSET ((FFI_TRAMPOLINE_SIZE + 3) & ~3)
+#define RAW_CLOSURE_FUN_OFFSET (RAW_CLOSURE_CIF_OFFSET + 4)
+#define RAW_CLOSURE_USER_DATA_OFFSET (RAW_CLOSURE_FUN_OFFSET + 4)
+#define CIF_FLAGS_OFFSET 20
+
+        # This assumes we are using gas.
+        .balign 16
+       .globl  _ffi_closure_raw_SYSV
+       .def    _ffi_closure_raw_SYSV;  .scl    2;      .type   32;     .endef
+_ffi_closure_raw_SYSV:
+.LFB4:
+       pushl   %ebp
+.LCFI6:
+       movl    %esp, %ebp
+.LCFI7:
+       pushl   %esi
+.LCFI8:
+       subl    $36, %esp
+       movl    RAW_CLOSURE_CIF_OFFSET(%eax), %esi       /* closure->cif */
+       movl    RAW_CLOSURE_USER_DATA_OFFSET(%eax), %edx /* closure->user_data */
+       movl    %edx, 12(%esp)  /* user_data */
+       leal    8(%ebp), %edx   /* __builtin_dwarf_cfa () */
+       movl    %edx, 8(%esp)   /* raw_args */
+       leal    -24(%ebp), %edx
+       movl    %edx, 4(%esp)   /* &res */
+       movl    %esi, (%esp)    /* cif */
+       call    *RAW_CLOSURE_FUN_OFFSET(%eax)            /* closure->fun */
+       movl    CIF_FLAGS_OFFSET(%esi), %eax             /* rtype */
+0:
+       call    1f
+       # Do not insert anything here between the call and the jump table.
+.Lrcls_store_table:
+       .long   .Lrcls_noretval         /* FFI_TYPE_VOID */
+       .long   .Lrcls_retint           /* FFI_TYPE_INT */
+       .long   .Lrcls_retfloat         /* FFI_TYPE_FLOAT */
+       .long   .Lrcls_retdouble        /* FFI_TYPE_DOUBLE */
+       .long   .Lrcls_retldouble       /* FFI_TYPE_LONGDOUBLE */
+       .long   .Lrcls_retuint8         /* FFI_TYPE_UINT8 */
+       .long   .Lrcls_retsint8         /* FFI_TYPE_SINT8 */
+       .long   .Lrcls_retuint16        /* FFI_TYPE_UINT16 */
+       .long   .Lrcls_retsint16        /* FFI_TYPE_SINT16 */
+       .long   .Lrcls_retint           /* FFI_TYPE_UINT32 */
+       .long   .Lrcls_retint           /* FFI_TYPE_SINT32 */
+       .long   .Lrcls_retllong         /* FFI_TYPE_UINT64 */
+       .long   .Lrcls_retllong         /* FFI_TYPE_SINT64 */
+       .long   .Lrcls_retstruct        /* FFI_TYPE_STRUCT */
+       .long   .Lrcls_retint           /* FFI_TYPE_POINTER */
+       .long   .Lrcls_retstruct1       /* FFI_TYPE_SMALL_STRUCT_1B */
+       .long   .Lrcls_retstruct2       /* FFI_TYPE_SMALL_STRUCT_2B */
+       .long   .Lrcls_retstruct4       /* FFI_TYPE_SMALL_STRUCT_4B */
+1:
+       add     %eax, %eax
+       add     %eax, %eax
+       add     (%esp),%eax
+       add     $4, %esp
+       jmp     *(%eax)
+
+       /* Sign/zero extend as appropriate.  */
+.Lrcls_retsint8:
+       movsbl  -24(%ebp), %eax
+       jmp     .Lrcls_epilogue
+
+.Lrcls_retsint16:
+       movswl  -24(%ebp), %eax
+       jmp     .Lrcls_epilogue
+
+.Lrcls_retuint8:
+       movzbl  -24(%ebp), %eax
+       jmp     .Lrcls_epilogue
+
+.Lrcls_retuint16:
+       movzwl  -24(%ebp), %eax
+       jmp     .Lrcls_epilogue
+
+.Lrcls_retint:
+       movl    -24(%ebp), %eax
+       jmp     .Lrcls_epilogue
+
+.Lrcls_retfloat:
+       flds    -24(%ebp)
+       jmp     .Lrcls_epilogue
+
+.Lrcls_retdouble:
+       fldl    -24(%ebp)
+       jmp     .Lrcls_epilogue
+
+.Lrcls_retldouble:
+       fldt    -24(%ebp)
+       jmp     .Lrcls_epilogue
+
+.Lrcls_retllong:
+       movl    -24(%ebp), %eax
+       movl    -20(%ebp), %edx
+       jmp     .Lrcls_epilogue
+
+.Lrcls_retstruct1:
+       movsbl  -24(%ebp), %eax
+       jmp     .Lrcls_epilogue
+
+.Lrcls_retstruct2:
+       movswl  -24(%ebp), %eax
+       jmp     .Lrcls_epilogue
+
+.Lrcls_retstruct4:
+       movl    -24(%ebp), %eax
+       jmp     .Lrcls_epilogue
+
+.Lrcls_retstruct:
+       # Nothing to do!
+
+.Lrcls_noretval:
+.Lrcls_epilogue:
+       addl    $36, %esp
+       popl    %esi
+       popl    %ebp
+       ret
+.ffi_closure_raw_SYSV_end:
+.LFE4:
+
+#endif /* !FFI_NO_RAW_API */
+
+        # This assumes we are using gas.
+       .balign 16
+       .globl  _ffi_closure_STDCALL
+       .def    _ffi_closure_STDCALL;   .scl    2;      .type   32;     .endef
+_ffi_closure_STDCALL:
+.LFB5:
+       pushl   %ebp
+.LCFI9:
+       movl    %esp, %ebp
+.LCFI10:
+       subl    $40, %esp
+       leal    -24(%ebp), %edx
+       movl    %edx, -12(%ebp) /* resp */
+       leal    12(%ebp), %edx  /* account for stub return address on stack */
+       movl    %edx, 4(%esp)   /* args */
+       leal    -12(%ebp), %edx
+       movl    %edx, (%esp)    /* &resp */
+       call    _ffi_closure_SYSV_inner
+       movl    -12(%ebp), %ecx
+0:
+       call    1f
+       # Do not insert anything here between the call and the jump table.
+.Lscls_store_table:
+       .long   .Lscls_noretval         /* FFI_TYPE_VOID */
+       .long   .Lscls_retint           /* FFI_TYPE_INT */
+       .long   .Lscls_retfloat         /* FFI_TYPE_FLOAT */
+       .long   .Lscls_retdouble        /* FFI_TYPE_DOUBLE */
+       .long   .Lscls_retldouble       /* FFI_TYPE_LONGDOUBLE */
+       .long   .Lscls_retuint8         /* FFI_TYPE_UINT8 */
+       .long   .Lscls_retsint8         /* FFI_TYPE_SINT8 */
+       .long   .Lscls_retuint16        /* FFI_TYPE_UINT16 */
+       .long   .Lscls_retsint16        /* FFI_TYPE_SINT16 */
+       .long   .Lscls_retint           /* FFI_TYPE_UINT32 */
+       .long   .Lscls_retint           /* FFI_TYPE_SINT32 */
+       .long   .Lscls_retllong         /* FFI_TYPE_UINT64 */
+       .long   .Lscls_retllong         /* FFI_TYPE_SINT64 */
+       .long   .Lscls_retstruct        /* FFI_TYPE_STRUCT */
+       .long   .Lscls_retint           /* FFI_TYPE_POINTER */
+       .long   .Lscls_retstruct1       /* FFI_TYPE_SMALL_STRUCT_1B */
+       .long   .Lscls_retstruct2       /* FFI_TYPE_SMALL_STRUCT_2B */
+       .long   .Lscls_retstruct4       /* FFI_TYPE_SMALL_STRUCT_4B */
+1:
+       add     %eax, %eax
+       add     %eax, %eax
+       add     (%esp),%eax
+       add     $4, %esp
+       jmp     *(%eax)
+
+       /* Sign/zero extend as appropriate.  */
+.Lscls_retsint8:
+       movsbl  (%ecx), %eax
+       jmp     .Lscls_epilogue
+
+.Lscls_retsint16:
+       movswl  (%ecx), %eax
+       jmp     .Lscls_epilogue
+
+.Lscls_retuint8:
+       movzbl  (%ecx), %eax
+       jmp     .Lscls_epilogue
+
+.Lscls_retuint16:
+       movzwl  (%ecx), %eax
+       jmp     .Lscls_epilogue
+
+.Lscls_retint:
+       movl    (%ecx), %eax
+       jmp     .Lscls_epilogue
+
+.Lscls_retfloat:
+       flds    (%ecx)
+       jmp     .Lscls_epilogue
+
+.Lscls_retdouble:
+       fldl    (%ecx)
+       jmp     .Lscls_epilogue
+
+.Lscls_retldouble:
+       fldt    (%ecx)
+       jmp     .Lscls_epilogue
+
+.Lscls_retllong:
+       movl    (%ecx), %eax
+       movl    4(%ecx), %edx
+       jmp     .Lscls_epilogue
+
+.Lscls_retstruct1:
+       movsbl  (%ecx), %eax
+       jmp     .Lscls_epilogue
+
+.Lscls_retstruct2:
+       movswl  (%ecx), %eax
+       jmp     .Lscls_epilogue
+
+.Lscls_retstruct4:
+       movl    (%ecx), %eax
+       jmp     .Lscls_epilogue
+
+.Lscls_retstruct:
+       # Nothing to do!
+
+.Lscls_noretval:
+.Lscls_epilogue:
+       movl    %ebp, %esp
+       popl    %ebp
+       ret
+.ffi_closure_STDCALL_end:
+.LFE5:
+
+       .section        .eh_frame,"w"
+.Lframe1:
+.LSCIE1:
+       .long   .LECIE1-.LASCIE1  /* Length of Common Information Entry */
+.LASCIE1:
+       .long   0x0     /* CIE Identifier Tag */
+       .byte   0x1     /* CIE Version */
+#ifdef __PIC__
+       .ascii "zR\0"   /* CIE Augmentation */
+#else
+       .ascii "\0"     /* CIE Augmentation */
+#endif
+       .byte   0x1     /* .uleb128 0x1; CIE Code Alignment Factor */
+       .byte   0x7c    /* .sleb128 -4; CIE Data Alignment Factor */
+       .byte   0x8     /* CIE RA Column */
+#ifdef __PIC__
+       .byte   0x1     /* .uleb128 0x1; Augmentation size */
+       .byte   0x1b    /* FDE Encoding (pcrel sdata4) */
+#endif
+       .byte   0xc     /* DW_CFA_def_cfa CFA = r4 + 4 = 4(%esp) */
+       .byte   0x4     /* .uleb128 0x4 */
+       .byte   0x4     /* .uleb128 0x4 */
+       .byte   0x88    /* DW_CFA_offset, column 0x8 %eip at CFA + 1 * -4 */
+       .byte   0x1     /* .uleb128 0x1 */
+       .align 4
+.LECIE1:
+
+.LSFDE1:
+       .long   .LEFDE1-.LASFDE1        /* FDE Length */
+.LASFDE1:
+       .long   .LASFDE1-.Lframe1       /* FDE CIE offset */
+#if defined __PIC__ && defined HAVE_AS_X86_PCREL
+       .long   .LFB1-. /* FDE initial location */
+#else
+       .long   .LFB1
+#endif
+       .long   .LFE1-.LFB1     /* FDE address range */
+#ifdef __PIC__
+       .byte   0x0     /* .uleb128 0x0; Augmentation size */
+#endif
+       /* DW_CFA_xxx CFI instructions go here.  */
+
+       .byte   0x4     /* DW_CFA_advance_loc4 */
+       .long   .LCFI0-.LFB1
+       .byte   0xe     /* DW_CFA_def_cfa_offset CFA = r4 + 8 = 8(%esp) */
+       .byte   0x8     /* .uleb128 0x8 */
+       .byte   0x85    /* DW_CFA_offset, column 0x5 %ebp at CFA + 2 * -4 */
+       .byte   0x2     /* .uleb128 0x2 */
+
+       .byte   0x4     /* DW_CFA_advance_loc4 */
+       .long   .LCFI1-.LCFI0
+       .byte   0xd     /* DW_CFA_def_cfa_register CFA = r5 = %ebp */
+       .byte   0x5     /* .uleb128 0x5 */
+
+       /* End of DW_CFA_xxx CFI instructions.  */
+       .align 4
+.LEFDE1:
+
+
+.LSFDE2:
+       .long   .LEFDE2-.LASFDE2        /* FDE Length */
+.LASFDE2:
+       .long   .LASFDE2-.Lframe1       /* FDE CIE offset */
+#if defined __PIC__ && defined HAVE_AS_X86_PCREL
+       .long   .LFB2-. /* FDE initial location */
+#else
+       .long   .LFB2
+#endif
+       .long   .LFE2-.LFB2     /* FDE address range */
+#ifdef __PIC__
+       .byte   0x0     /* .uleb128 0x0; Augmentation size */
+#endif
+       /* DW_CFA_xxx CFI instructions go here.  */
+
+       .byte   0x4     /* DW_CFA_advance_loc4 */
+       .long   .LCFI2-.LFB2
+       .byte   0xe     /* DW_CFA_def_cfa_offset CFA = r4 + 8 = 8(%esp) */
+       .byte   0x8     /* .uleb128 0x8 */
+       .byte   0x85    /* DW_CFA_offset, column 0x5 %ebp at CFA + 2 * -4 */
+       .byte   0x2     /* .uleb128 0x2 */
+
+       .byte   0x4     /* DW_CFA_advance_loc4 */
+       .long   .LCFI3-.LCFI2
+       .byte   0xd     /* DW_CFA_def_cfa_register CFA = r5 = %ebp */
+       .byte   0x5     /* .uleb128 0x5 */
+
+       /* End of DW_CFA_xxx CFI instructions.  */
+       .align 4
+.LEFDE2:
+
+
+.LSFDE3:
+       .long   .LEFDE3-.LASFDE3        /* FDE Length */
+.LASFDE3:
+       .long   .LASFDE3-.Lframe1       /* FDE CIE offset */
+#if defined __PIC__ && defined HAVE_AS_X86_PCREL
+       .long   .LFB3-. /* FDE initial location */
+#else
+       .long   .LFB3
+#endif
+       .long   .LFE3-.LFB3     /* FDE address range */
+#ifdef __PIC__
+       .byte   0x0     /* .uleb128 0x0; Augmentation size */
+#endif
+       /* DW_CFA_xxx CFI instructions go here.  */
+
+       .byte   0x4     /* DW_CFA_advance_loc4 */
+       .long   .LCFI4-.LFB3
+       .byte   0xe     /* DW_CFA_def_cfa_offset CFA = r4 + 8 = 8(%esp) */
+       .byte   0x8     /* .uleb128 0x8 */
+       .byte   0x85    /* DW_CFA_offset, column 0x5 %ebp at CFA + 2 * -4 */
+       .byte   0x2     /* .uleb128 0x2 */
+
+       .byte   0x4     /* DW_CFA_advance_loc4 */
+       .long   .LCFI5-.LCFI4
+       .byte   0xd     /* DW_CFA_def_cfa_register CFA = r5 = %ebp */
+       .byte   0x5     /* .uleb128 0x5 */
+
+       /* End of DW_CFA_xxx CFI instructions.  */
+       .align 4
+.LEFDE3:
+
+#if !FFI_NO_RAW_API
+
+.LSFDE4:
+       .long   .LEFDE4-.LASFDE4        /* FDE Length */
+.LASFDE4:
+       .long   .LASFDE4-.Lframe1       /* FDE CIE offset */
+#if defined __PIC__ && defined HAVE_AS_X86_PCREL
+       .long   .LFB4-. /* FDE initial location */
+#else
+       .long   .LFB4
+#endif
+       .long   .LFE4-.LFB4     /* FDE address range */
+#ifdef __PIC__
+       .byte   0x0     /* .uleb128 0x0; Augmentation size */
+#endif
+       /* DW_CFA_xxx CFI instructions go here.  */
+
+       .byte   0x4     /* DW_CFA_advance_loc4 */
+       .long   .LCFI6-.LFB4
+       .byte   0xe     /* DW_CFA_def_cfa_offset CFA = r4 + 8 = 8(%esp) */
+       .byte   0x8     /* .uleb128 0x8 */
+       .byte   0x85    /* DW_CFA_offset, column 0x5 %ebp at CFA + 2 * -4 */
+       .byte   0x2     /* .uleb128 0x2 */
+
+       .byte   0x4     /* DW_CFA_advance_loc4 */
+       .long   .LCFI7-.LCFI6
+       .byte   0xd     /* DW_CFA_def_cfa_register CFA = r5 = %ebp */
+       .byte   0x5     /* .uleb128 0x5 */
+
+       .byte   0x4     /* DW_CFA_advance_loc4 */
+       .long   .LCFI8-.LCFI7
+       .byte   0x86    /* DW_CFA_offset, column 0x6 %esi at CFA + 3 * -4 */
+       .byte   0x3     /* .uleb128 0x3 */
+
+       /* End of DW_CFA_xxx CFI instructions.  */
+       .align 4
+.LEFDE4:
+
+#endif /* !FFI_NO_RAW_API */
+
+.LSFDE5:
+       .long   .LEFDE5-.LASFDE5        /* FDE Length */
+.LASFDE5:
+       .long   .LASFDE5-.Lframe1       /* FDE CIE offset */
+#if defined __PIC__ && defined HAVE_AS_X86_PCREL
+       .long   .LFB5-. /* FDE initial location */
+#else
+       .long   .LFB5
+#endif
+       .long   .LFE5-.LFB5     /* FDE address range */
+#ifdef __PIC__
+       .byte   0x0     /* .uleb128 0x0; Augmentation size */
+#endif
+       /* DW_CFA_xxx CFI instructions go here.  */
+
+       .byte   0x4     /* DW_CFA_advance_loc4 */
+       .long   .LCFI9-.LFB5
+       .byte   0xe     /* DW_CFA_def_cfa_offset CFA = r4 + 8 = 8(%esp) */
+       .byte   0x8     /* .uleb128 0x8 */
+       .byte   0x85    /* DW_CFA_offset, column 0x5 %ebp at CFA + 2 * -4 */
+       .byte   0x2     /* .uleb128 0x2 */
+
+       .byte   0x4     /* DW_CFA_advance_loc4 */
+       .long   .LCFI10-.LCFI9
+       .byte   0xd     /* DW_CFA_def_cfa_register CFA = r5 = %ebp */
+       .byte   0x5     /* .uleb128 0x5 */
+
+       /* End of DW_CFA_xxx CFI instructions.  */
+       .align 4
+.LEFDE5:
index 3ea21e1..b3c333c 100644 (file)
@@ -1,3 +1,20 @@
+2010-01-15  Daniel Witte  <dwitte@mozilla.com>
+
+       * src/x86/win32.S: Port assembly routines to MSVC and #ifdef.
+       * src/x86/ffi.c: Tweak function declaration and remove excess
+       parens.
+       * include/ffi.h.in: Add __declspec(align(8)) to typedef struct
+       ffi_closure.
+
+       * src/x86/ffi.c: Merge ffi_call_SYSV and ffi_call_STDCALL into new
+       function ffi_call_win32 on X86_WIN32.
+       * src/x86/win32.S (ffi_call_SYSV): Rename to ffi_call_win32.
+       (ffi_call_STDCALL): Remove.
+
+       * src/prep_cif.c (ffi_prep_cif): Move stack space allocation code
+       to ffi_prep_cif_machdep for x86.
+       * src/x86/ffi.c (ffi_prep_cif_machdep): To here.
+
 2010-01-15  Oliver Kiddle  <okiddle@yahoo.co.uk>
 
        * src/x86/ffitarget.h (ffi_abi): Check for __i386 and __amd64 for
index 32f6103..df7d2cd 100644 (file)
@@ -251,6 +251,9 @@ size_t ffi_java_raw_size (ffi_cif *cif);
 
 #if FFI_CLOSURES
 
+#ifdef _MSC_VER
+__declspec(align(8))
+#endif
 typedef struct {
   char tramp[FFI_TRAMPOLINE_SIZE];
   ffi_cif   *cif;
diff --git a/patches/msvc b/patches/msvc
new file mode 100644 (file)
index 0000000..1856fd5
--- /dev/null
@@ -0,0 +1,920 @@
+Index: libffi/ChangeLog.libffi
+===================================================================
+--- libffi.orig/ChangeLog.libffi
++++ libffi/ChangeLog.libffi
+@@ -1,3 +1,20 @@
++2010-01-15  Daniel Witte  <dwitte@mozilla.com>
++
++      * src/x86/win32.S: Port assembly routines to MSVC and #ifdef.
++      * src/x86/ffi.c: Tweak function declaration and remove excess
++      parens.
++      * include/ffi.h.in: Add __declspec(align(8)) to typedef struct
++      ffi_closure.
++
++      * src/x86/ffi.c: Merge ffi_call_SYSV and ffi_call_STDCALL into new
++      function ffi_call_win32 on X86_WIN32.
++      * src/x86/win32.S (ffi_call_SYSV): Rename to ffi_call_win32.
++      (ffi_call_STDCALL): Remove.
++
++      * src/prep_cif.c (ffi_prep_cif): Move stack space allocation code
++      to ffi_prep_cif_machdep for x86.
++      * src/x86/ffi.c (ffi_prep_cif_machdep): To here.
++
+ 2010-01-15  Oliver Kiddle  <okiddle@yahoo.co.uk>
+       * src/x86/ffitarget.h (ffi_abi): Check for __i386 and __amd64 for
+Index: libffi/include/ffi.h.in
+===================================================================
+--- libffi.orig/include/ffi.h.in
++++ libffi/include/ffi.h.in
+@@ -251,6 +251,9 @@ size_t ffi_java_raw_size (ffi_cif *cif);
+ #if FFI_CLOSURES
++#ifdef _MSC_VER
++__declspec(align(8))
++#endif
+ typedef struct {
+   char tramp[FFI_TRAMPOLINE_SIZE];
+   ffi_cif   *cif;
+Index: libffi/src/prep_cif.c
+===================================================================
+--- libffi.orig/src/prep_cif.c
++++ libffi/src/prep_cif.c
+@@ -109,16 +109,13 @@ ffi_status ffi_prep_cif(ffi_cif *cif, ff
+   /* Perform a sanity check on the return type */
+   FFI_ASSERT_VALID_TYPE(cif->rtype);
+-  /* x86-64 and s390 stack space allocation is handled in prep_machdep.  */
+-#if !defined M68K && !defined __x86_64__ && !defined S390 && !defined PA
++  /* x86, x86-64 and s390 stack space allocation is handled in prep_machdep. */
++#if !defined M68K && !defined __i386__ && !defined __x86_64__ && !defined S390 && !defined PA
+   /* Make space for the return structure pointer */
+   if (cif->rtype->type == FFI_TYPE_STRUCT
+ #ifdef SPARC
+       && (cif->abi != FFI_V9 || cif->rtype->size > 32)
+ #endif
+-#ifdef X86_DARWIN
+-      && (cif->rtype->size > 8)
+-#endif
+      )
+     bytes = STACK_ARG_SIZE(sizeof(void*));
+ #endif
+@@ -134,7 +131,7 @@ ffi_status ffi_prep_cif(ffi_cif *cif, ff
+        check after the initialization.  */
+       FFI_ASSERT_VALID_TYPE(*ptr);
+-#if !defined __x86_64__ && !defined S390 && !defined PA
++#if !defined __i386__ && !defined __x86_64__ && !defined S390 && !defined PA
+ #ifdef SPARC
+       if (((*ptr)->type == FFI_TYPE_STRUCT
+          && ((*ptr)->size > 16 || cif->abi != FFI_V9))
+Index: libffi/src/x86/ffi.c
+===================================================================
+--- libffi.orig/src/x86/ffi.c
++++ libffi/src/x86/ffi.c
+@@ -148,13 +148,13 @@ void ffi_prep_args(char *stack, extended
+ /* Perform machine dependent cif processing */
+ ffi_status ffi_prep_cif_machdep(ffi_cif *cif)
+ {
++  unsigned int i;
++  ffi_type **ptr;
++
+   /* Set the return type flag */
+   switch (cif->rtype->type)
+     {
+     case FFI_TYPE_VOID:
+-#ifdef X86
+-    case FFI_TYPE_STRUCT:
+-#endif
+ #if defined(X86) || defined (X86_WIN32) || defined(X86_FREEBSD) || defined(X86_DARWIN) || defined(X86_WIN64)
+     case FFI_TYPE_UINT8:
+     case FFI_TYPE_UINT16:
+@@ -165,7 +165,6 @@ ffi_status ffi_prep_cif_machdep(ffi_cif 
+     case FFI_TYPE_UINT32:
+     case FFI_TYPE_SINT32:
+ #endif
+-
+     case FFI_TYPE_SINT64:
+     case FFI_TYPE_FLOAT:
+     case FFI_TYPE_DOUBLE:
+@@ -184,8 +183,8 @@ ffi_status ffi_prep_cif_machdep(ffi_cif 
+       cif->flags = FFI_TYPE_SINT64;
+       break;
+-#ifndef X86
+     case FFI_TYPE_STRUCT:
++#ifndef X86
+       if (cif->rtype->size == 1)
+         {
+           cif->flags = FFI_TYPE_SMALL_STRUCT_1B; /* same as char size */
+@@ -207,15 +206,13 @@ ffi_status ffi_prep_cif_machdep(ffi_cif 
+           cif->flags = FFI_TYPE_SINT64; /* same as int64 type */
+         }
+       else
++#endif
+         {
+           cif->flags = FFI_TYPE_STRUCT;
+-#ifdef X86_WIN64
+           // allocate space for return value pointer
+           cif->bytes += ALIGN(sizeof(void*), FFI_SIZEOF_ARG);
+-#endif
+         }
+       break;
+-#endif
+     default:
+ #ifdef X86_WIN64
+@@ -229,41 +226,36 @@ ffi_status ffi_prep_cif_machdep(ffi_cif 
+       break;
+     }
+-#ifdef X86_DARWIN
+-  cif->bytes = (cif->bytes + 15) & ~0xF;
+-#endif
++  for (ptr = cif->arg_types, i = cif->nargs; i > 0; i--, ptr++)
++    {
++      if (((*ptr)->alignment - 1) & cif->bytes)
++        cif->bytes = ALIGN(cif->bytes, (*ptr)->alignment);
++      cif->bytes += ALIGN((*ptr)->size, FFI_SIZEOF_ARG);
++    }
+ #ifdef X86_WIN64
+-  {
+-    unsigned int i;
+-    ffi_type **ptr;
+-
+-    for (ptr = cif->arg_types, i = cif->nargs; i > 0; i--, ptr++)
+-      {
+-        if (((*ptr)->alignment - 1) & cif->bytes)
+-          cif->bytes = ALIGN(cif->bytes, (*ptr)->alignment);
+-        cif->bytes += ALIGN((*ptr)->size, FFI_SIZEOF_ARG);
+-      }
+-  }
+   // ensure space for storing four registers
+   cif->bytes += 4 * sizeof(ffi_arg);
+ #endif
++#ifdef X86_DARWIN
++  cif->bytes = (cif->bytes + 15) & ~0xF;
++#endif
++
+   return FFI_OK;
+ }
+-extern void ffi_call_SYSV(void (*)(char *, extended_cif *), extended_cif *,
+-                          unsigned, unsigned, unsigned *, void (*fn)(void));
+-
+-#ifdef X86_WIN32
+-extern void ffi_call_STDCALL(void (*)(char *, extended_cif *), extended_cif *,
+-                          unsigned, unsigned, unsigned *, void (*fn)(void));
+-
+-#endif /* X86_WIN32 */
+ #ifdef X86_WIN64
+ extern int
+ ffi_call_win64(void (*)(char *, extended_cif *), extended_cif *,
+                unsigned, unsigned, unsigned *, void (*fn)(void));
++#elif defined(X86_WIN32)
++extern void
++ffi_call_win32(void (*)(char *, extended_cif *), extended_cif *,
++               unsigned, unsigned, unsigned *, void (*fn)(void));
++#else
++extern void ffi_call_SYSV(void (*)(char *, extended_cif *), extended_cif *,
++                          unsigned, unsigned, unsigned *, void (*fn)(void));
+ #endif
+ void ffi_call(ffi_cif *cif, void (*fn)(void), void *rvalue, void **avalue)
+@@ -321,18 +313,18 @@ void ffi_call(ffi_cif *cif, void (*fn)(v
+                        cif->flags, ecif.rvalue, fn);
+       }
+       break;
++#elif defined(X86_WIN32)
++    case FFI_SYSV:
++    case FFI_STDCALL:
++      ffi_call_win32(ffi_prep_args, &ecif, cif->bytes, cif->flags,
++                     ecif.rvalue, fn);
++      break;
+ #else
+     case FFI_SYSV:
+       ffi_call_SYSV(ffi_prep_args, &ecif, cif->bytes, cif->flags, ecif.rvalue,
+                     fn);
+       break;
+-#ifdef X86_WIN32
+-    case FFI_STDCALL:
+-      ffi_call_STDCALL(ffi_prep_args, &ecif, cif->bytes, cif->flags,
+-                       ecif.rvalue, fn);
+-      break;
+-#endif /* X86_WIN32 */
+-#endif /* X86_WIN64 */
++#endif
+     default:
+       FFI_ASSERT(0);
+       break;
+@@ -342,6 +334,8 @@ void ffi_call(ffi_cif *cif, void (*fn)(v
+ /** private members **/
++/* The following __attribute__((regparm(1))) decorations will have no effect
++   on MSVC - standard cdecl convention applies. */
+ static void ffi_prep_incoming_args_SYSV (char *stack, void **ret,
+                                          void** args, ffi_cif* cif);
+ void FFI_HIDDEN ffi_closure_SYSV (ffi_closure *)
+@@ -390,11 +384,8 @@ ffi_closure_win64_inner (ffi_closure *cl
+ }
+ #else
+-unsigned int FFI_HIDDEN
+-ffi_closure_SYSV_inner (closure, respp, args)
+-     ffi_closure *closure;
+-     void **respp;
+-     void *args;
++unsigned int FFI_HIDDEN __attribute__ ((regparm(1)))
++ffi_closure_SYSV_inner (ffi_closure *closure, void **respp, void *args)
+ {
+   /* our various things...  */
+   ffi_cif       *cif;
+@@ -505,7 +496,7 @@ ffi_prep_incoming_args_SYSV(char *stack,
+ /* How to make a trampoline.  Derived from gcc/config/i386/i386.c. */
+ #define FFI_INIT_TRAMPOLINE(TRAMP,FUN,CTX) \
+-({ unsigned char *__tramp = (unsigned char*)(TRAMP); \
++{ unsigned char *__tramp = (unsigned char*)(TRAMP); \
+    unsigned int  __fun = (unsigned int)(FUN); \
+    unsigned int  __ctx = (unsigned int)(CTX); \
+    unsigned int  __dis = __fun - (__ctx + 10);  \
+@@ -513,10 +504,10 @@ ffi_prep_incoming_args_SYSV(char *stack,
+    *(unsigned int*)  &__tramp[1] = __ctx; /* movl __ctx, %eax */ \
+    *(unsigned char *)  &__tramp[5] = 0xe9; \
+    *(unsigned int*)  &__tramp[6] = __dis; /* jmp __fun  */ \
+- })
++ }
+ #define FFI_INIT_TRAMPOLINE_STDCALL(TRAMP,FUN,CTX,SIZE)  \
+-({ unsigned char *__tramp = (unsigned char*)(TRAMP); \
++{ unsigned char *__tramp = (unsigned char*)(TRAMP); \
+    unsigned int  __fun = (unsigned int)(FUN); \
+    unsigned int  __ctx = (unsigned int)(CTX); \
+    unsigned int  __dis = __fun - (__ctx + 10); \
+@@ -527,7 +518,7 @@ ffi_prep_incoming_args_SYSV(char *stack,
+    *(unsigned int*)  &__tramp[6] = __dis; /* call __fun  */ \
+    *(unsigned char *)  &__tramp[10] = 0xc2; \
+    *(unsigned short*)  &__tramp[11] = __size; /* ret __size  */ \
+- })
++ }
+ /* the cif must already be prep'ed */
+@@ -627,16 +618,6 @@ ffi_prep_args_raw(char *stack, extended_
+  * libffi-1.20, this is not the case.)
+  */
+-extern void
+-ffi_call_SYSV(void (*)(char *, extended_cif *), extended_cif *, unsigned, 
+-              unsigned, unsigned *, void (*fn)(void));
+-
+-#ifdef X86_WIN32
+-extern void
+-ffi_call_STDCALL(void (*)(char *, extended_cif *), extended_cif *, unsigned,
+-                 unsigned, unsigned *, void (*fn)(void));
+-#endif /* X86_WIN32 */
+-
+ void
+ ffi_raw_call(ffi_cif *cif, void (*fn)(void), void *rvalue, ffi_raw *fake_avalue)
+ {
+@@ -660,16 +641,18 @@ ffi_raw_call(ffi_cif *cif, void (*fn)(vo
+   
+   switch (cif->abi) 
+     {
++#ifdef X86_WIN32
++    case FFI_SYSV:
++    case FFI_STDCALL:
++      ffi_call_win32(ffi_prep_args_raw, &ecif, cif->bytes, cif->flags,
++                     ecif.rvalue, fn);
++      break;
++#else
+     case FFI_SYSV:
+       ffi_call_SYSV(ffi_prep_args_raw, &ecif, cif->bytes, cif->flags,
+                     ecif.rvalue, fn);
+       break;
+-#ifdef X86_WIN32
+-    case FFI_STDCALL:
+-      ffi_call_STDCALL(ffi_prep_args_raw, &ecif, cif->bytes, cif->flags,
+-                       ecif.rvalue, fn);
+-      break;
+-#endif /* X86_WIN32 */
++#endif
+     default:
+       FFI_ASSERT(0);
+       break;
+Index: libffi/src/x86/win32.S
+===================================================================
+--- libffi.orig/src/x86/win32.S
++++ libffi/src/x86/win32.S
+@@ -2,6 +2,7 @@
+    win32.S - Copyright (c) 1996, 1998, 2001, 2002, 2009  Red Hat, Inc.
+            Copyright (c) 2001  John Beniton
+            Copyright (c) 2002  Ranjit Mathew
++           Copyright (c) 2009  Daniel Witte
+                       
+  
+    X86 Foreign Function Interface
+@@ -31,14 +32,371 @@
+ #define LIBFFI_ASM
+ #include <fficonfig.h>
+ #include <ffi.h>
+- 
++
++#ifdef _MSC_VER
++
++.386
++.MODEL FLAT, C
++
++EXTRN ffi_closure_SYSV_inner:NEAR
++
++_TEXT SEGMENT
++
++ffi_call_win32 PROC NEAR,
++    ffi_prep_args : NEAR PTR DWORD,
++    ecif          : NEAR PTR DWORD,
++    cif_bytes     : DWORD,
++    cif_flags     : DWORD,
++    rvalue        : NEAR PTR DWORD,
++    fn            : NEAR PTR DWORD
++
++        ;; Make room for all of the new args.
++        mov  ecx, cif_bytes
++        sub  esp, ecx
++
++        mov  eax, esp
++
++        ;; Place all of the ffi_prep_args in position
++        push ecif
++        push eax
++        call ffi_prep_args
++
++        ;; Return stack to previous state and call the function
++        add  esp, 8
++
++        call fn
++
++        ;; cdecl:   we restore esp in the epilogue, so there's no need to
++        ;;          remove the space we pushed for the args.
++        ;; stdcall: the callee has already cleaned the stack.
++
++        ;; Load ecx with the return type code
++        mov  ecx, cif_flags
++
++        ;; If the return value pointer is NULL, assume no return value.
++        cmp  rvalue, 0
++        jne  ca_jumptable
++
++        ;; Even if there is no space for the return value, we are
++        ;; obliged to handle floating-point values.
++        cmp  ecx, FFI_TYPE_FLOAT
++        jne  ca_epilogue
++        fstp st(0)
++
++        jmp  ca_epilogue
++
++ca_jumptable:
++        jmp  [ca_jumpdata + 4 * ecx]
++ca_jumpdata:
++        ;; Do not insert anything here between label and jump table.
++        dd offset ca_epilogue       ;; FFI_TYPE_VOID
++        dd offset ca_retint         ;; FFI_TYPE_INT
++        dd offset ca_retfloat       ;; FFI_TYPE_FLOAT
++        dd offset ca_retdouble      ;; FFI_TYPE_DOUBLE
++        dd offset ca_retlongdouble  ;; FFI_TYPE_LONGDOUBLE
++        dd offset ca_retint8        ;; FFI_TYPE_UINT8
++        dd offset ca_retint8        ;; FFI_TYPE_SINT8
++        dd offset ca_retint16       ;; FFI_TYPE_UINT16
++        dd offset ca_retint16       ;; FFI_TYPE_SINT16
++        dd offset ca_retint         ;; FFI_TYPE_UINT32
++        dd offset ca_retint         ;; FFI_TYPE_SINT32
++        dd offset ca_retint64       ;; FFI_TYPE_UINT64
++        dd offset ca_retint64       ;; FFI_TYPE_SINT64
++        dd offset ca_epilogue       ;; FFI_TYPE_STRUCT
++        dd offset ca_retint         ;; FFI_TYPE_POINTER
++        dd offset ca_retint8        ;; FFI_TYPE_SMALL_STRUCT_1B
++        dd offset ca_retint16       ;; FFI_TYPE_SMALL_STRUCT_2B
++        dd offset ca_retint         ;; FFI_TYPE_SMALL_STRUCT_4B
++
++ca_retint8:
++        ;; Load %ecx with the pointer to storage for the return value
++        mov   ecx, rvalue
++        mov   [ecx + 0], al
++        jmp   ca_epilogue
++
++ca_retint16:
++        ;; Load %ecx with the pointer to storage for the return value
++        mov   ecx, rvalue
++        mov   [ecx + 0], ax
++        jmp   ca_epilogue
++
++ca_retint:
++        ;; Load %ecx with the pointer to storage for the return value
++        mov   ecx, rvalue
++        mov   [ecx + 0], eax
++        jmp   ca_epilogue
++
++ca_retint64:
++        ;; Load %ecx with the pointer to storage for the return value
++        mov   ecx, rvalue
++        mov   [ecx + 0], eax
++        mov   [ecx + 4], edx
++        jmp   ca_epilogue
++
++ca_retfloat:
++        ;; Load %ecx with the pointer to storage for the return value
++        mov   ecx, rvalue
++        fstp  DWORD PTR [ecx]
++        jmp   ca_epilogue
++
++ca_retdouble:
++        ;; Load %ecx with the pointer to storage for the return value
++        mov   ecx, rvalue
++        fstp  QWORD PTR [ecx]
++        jmp   ca_epilogue
++
++ca_retlongdouble:
++        ;; Load %ecx with the pointer to storage for the return value
++        mov   ecx, rvalue
++        fstp  TBYTE PTR [ecx]
++        jmp   ca_epilogue
++
++ca_epilogue:
++        ;; Epilogue code is autogenerated.
++        ret
++ffi_call_win32 ENDP
++
++ffi_closure_SYSV PROC NEAR FORCEFRAME
++    ;; the ffi_closure ctx is passed in eax by the trampoline.
++
++        sub  esp, 40
++        lea  edx, [ebp - 24]
++        mov  [ebp - 12], edx         ;; resp
++        lea  edx, [ebp + 8]
++        mov  [esp + 8], edx          ;; args
++        lea  edx, [ebp - 12]
++        mov  [esp + 4], edx          ;; &resp
++        mov  [esp], eax              ;; closure
++        call ffi_closure_SYSV_inner
++        mov  ecx, [ebp - 12]
++
++cs_jumptable:
++        jmp  [cs_jumpdata + 4 * eax]
++cs_jumpdata:
++        ;; Do not insert anything here between the label and jump table.
++        dd offset cs_epilogue       ;; FFI_TYPE_VOID
++        dd offset cs_retint         ;; FFI_TYPE_INT
++        dd offset cs_retfloat       ;; FFI_TYPE_FLOAT
++        dd offset cs_retdouble      ;; FFI_TYPE_DOUBLE
++        dd offset cs_retlongdouble  ;; FFI_TYPE_LONGDOUBLE
++        dd offset cs_retint8        ;; FFI_TYPE_UINT8
++        dd offset cs_retint8        ;; FFI_TYPE_SINT8
++        dd offset cs_retint16       ;; FFI_TYPE_UINT16
++        dd offset cs_retint16       ;; FFI_TYPE_SINT16
++        dd offset cs_retint         ;; FFI_TYPE_UINT32
++        dd offset cs_retint         ;; FFI_TYPE_SINT32
++        dd offset cs_retint64       ;; FFI_TYPE_UINT64
++        dd offset cs_retint64       ;; FFI_TYPE_SINT64
++        dd offset cs_retstruct      ;; FFI_TYPE_STRUCT
++        dd offset cs_retint         ;; FFI_TYPE_POINTER
++        dd offset cs_retint8        ;; FFI_TYPE_SMALL_STRUCT_1B
++        dd offset cs_retint16       ;; FFI_TYPE_SMALL_STRUCT_2B
++        dd offset cs_retint         ;; FFI_TYPE_SMALL_STRUCT_4B
++
++cs_retint8:
++        mov   al, [ecx]
++        jmp   cs_epilogue
++
++cs_retint16:
++        mov   ax, [ecx]
++        jmp   cs_epilogue
++
++cs_retint:
++        mov   eax, [ecx]
++        jmp   cs_epilogue
++
++cs_retint64:
++        mov   eax, [ecx + 0]
++        mov   edx, [ecx + 4]
++        jmp   cs_epilogue
++
++cs_retfloat:
++        fld   DWORD PTR [ecx]
++        jmp   cs_epilogue
++
++cs_retdouble:
++        fld   QWORD PTR [ecx]
++        jmp   cs_epilogue
++
++cs_retlongdouble:
++        fld   TBYTE PTR [ecx]
++        jmp   cs_epilogue
++
++cs_retstruct:
++        ;; Caller expects us to pop struct return value pointer hidden arg.
++        ;; Epilogue code is autogenerated.
++        ret   4
++
++cs_epilogue:
++        ;; Epilogue code is autogenerated.
++        ret
++ffi_closure_SYSV ENDP
++
++#if !FFI_NO_RAW_API
++
++#define RAW_CLOSURE_CIF_OFFSET ((FFI_TRAMPOLINE_SIZE + 3) AND NOT 3)
++#define RAW_CLOSURE_FUN_OFFSET (RAW_CLOSURE_CIF_OFFSET + 4)
++#define RAW_CLOSURE_USER_DATA_OFFSET (RAW_CLOSURE_FUN_OFFSET + 4)
++#define CIF_FLAGS_OFFSET 20
++
++ffi_closure_raw_SYSV PROC NEAR USES esi
++    ;; the ffi_closure ctx is passed in eax by the trampoline.
++
++        sub  esp, 40
++        mov  esi, [eax + RAW_CLOSURE_CIF_OFFSET]        ;; closure->cif
++        mov  edx, [eax + RAW_CLOSURE_USER_DATA_OFFSET]  ;; closure->user_data
++        mov  [esp + 12], edx                            ;; user_data
++        lea  edx, [ebp + 8]
++        mov  [esp + 8], edx                             ;; raw_args
++        lea  edx, [ebp - 24]
++        mov  [esp + 4], edx                             ;; &res
++        mov  [esp], esi                                 ;; cif
++        call DWORD PTR [eax + RAW_CLOSURE_FUN_OFFSET]   ;; closure->fun
++        mov  eax, [esi + CIF_FLAGS_OFFSET]              ;; cif->flags
++        lea  ecx, [ebp - 24]
++
++cr_jumptable:
++        jmp  [cr_jumpdata + 4 * eax]
++cr_jumpdata:
++        ;; Do not insert anything here between the label and jump table.
++        dd offset cr_epilogue       ;; FFI_TYPE_VOID
++        dd offset cr_retint         ;; FFI_TYPE_INT
++        dd offset cr_retfloat       ;; FFI_TYPE_FLOAT
++        dd offset cr_retdouble      ;; FFI_TYPE_DOUBLE
++        dd offset cr_retlongdouble  ;; FFI_TYPE_LONGDOUBLE
++        dd offset cr_retint8        ;; FFI_TYPE_UINT8
++        dd offset cr_retint8        ;; FFI_TYPE_SINT8
++        dd offset cr_retint16       ;; FFI_TYPE_UINT16
++        dd offset cr_retint16       ;; FFI_TYPE_SINT16
++        dd offset cr_retint         ;; FFI_TYPE_UINT32
++        dd offset cr_retint         ;; FFI_TYPE_SINT32
++        dd offset cr_retint64       ;; FFI_TYPE_UINT64
++        dd offset cr_retint64       ;; FFI_TYPE_SINT64
++        dd offset cr_epilogue       ;; FFI_TYPE_STRUCT
++        dd offset cr_retint         ;; FFI_TYPE_POINTER
++        dd offset cr_retint8        ;; FFI_TYPE_SMALL_STRUCT_1B
++        dd offset cr_retint16       ;; FFI_TYPE_SMALL_STRUCT_2B
++        dd offset cr_retint         ;; FFI_TYPE_SMALL_STRUCT_4B
++
++cr_retint8:
++        mov   al, [ecx]
++        jmp   cr_epilogue
++
++cr_retint16:
++        mov   ax, [ecx]
++        jmp   cr_epilogue
++
++cr_retint:
++        mov   eax, [ecx]
++        jmp   cr_epilogue
++
++cr_retint64:
++        mov   eax, [ecx + 0]
++        mov   edx, [ecx + 4]
++        jmp   cr_epilogue
++
++cr_retfloat:
++        fld   DWORD PTR [ecx]
++        jmp   cr_epilogue
++
++cr_retdouble:
++        fld   QWORD PTR [ecx]
++        jmp   cr_epilogue
++
++cr_retlongdouble:
++        fld   TBYTE PTR [ecx]
++        jmp   cr_epilogue
++
++cr_epilogue:
++        ;; Epilogue code is autogenerated.
++        ret
++ffi_closure_raw_SYSV ENDP
++
++#endif /* !FFI_NO_RAW_API */
++
++ffi_closure_STDCALL PROC NEAR FORCEFRAME
++    ;; the ffi_closure ctx is passed in eax by the trampoline.
++
++        sub  esp, 40
++        lea  edx, [ebp - 24]
++        mov  [ebp - 12], edx         ;; resp
++        lea  edx, [ebp + 12]         ;; account for stub return address on stack
++        mov  [esp + 8], edx          ;; args
++        lea  edx, [ebp - 12]
++        mov  [esp + 4], edx          ;; &resp
++        mov  [esp], eax              ;; closure
++        call ffi_closure_SYSV_inner
++        mov  ecx, [ebp - 12]
++
++cd_jumptable:
++        jmp  [cd_jumpdata + 4 * eax]
++cd_jumpdata:
++        ;; Do not insert anything here between the label and jump table.
++        dd offset cd_epilogue       ;; FFI_TYPE_VOID
++        dd offset cd_retint         ;; FFI_TYPE_INT
++        dd offset cd_retfloat       ;; FFI_TYPE_FLOAT
++        dd offset cd_retdouble      ;; FFI_TYPE_DOUBLE
++        dd offset cd_retlongdouble  ;; FFI_TYPE_LONGDOUBLE
++        dd offset cd_retint8        ;; FFI_TYPE_UINT8
++        dd offset cd_retint8        ;; FFI_TYPE_SINT8
++        dd offset cd_retint16       ;; FFI_TYPE_UINT16
++        dd offset cd_retint16       ;; FFI_TYPE_SINT16
++        dd offset cd_retint         ;; FFI_TYPE_UINT32
++        dd offset cd_retint         ;; FFI_TYPE_SINT32
++        dd offset cd_retint64       ;; FFI_TYPE_UINT64
++        dd offset cd_retint64       ;; FFI_TYPE_SINT64
++        dd offset cd_epilogue       ;; FFI_TYPE_STRUCT
++        dd offset cd_retint         ;; FFI_TYPE_POINTER
++        dd offset cd_retint8        ;; FFI_TYPE_SMALL_STRUCT_1B
++        dd offset cd_retint16       ;; FFI_TYPE_SMALL_STRUCT_2B
++        dd offset cd_retint         ;; FFI_TYPE_SMALL_STRUCT_4B
++
++cd_retint8:
++        mov   al, [ecx]
++        jmp   cd_epilogue
++
++cd_retint16:
++        mov   ax, [ecx]
++        jmp   cd_epilogue
++
++cd_retint:
++        mov   eax, [ecx]
++        jmp   cd_epilogue
++
++cd_retint64:
++        mov   eax, [ecx + 0]
++        mov   edx, [ecx + 4]
++        jmp   cd_epilogue
++
++cd_retfloat:
++        fld   DWORD PTR [ecx]
++        jmp   cd_epilogue
++
++cd_retdouble:
++        fld   QWORD PTR [ecx]
++        jmp   cd_epilogue
++
++cd_retlongdouble:
++        fld   TBYTE PTR [ecx]
++        jmp   cd_epilogue
++
++cd_epilogue:
++        ;; Epilogue code is autogenerated.
++        ret
++ffi_closure_STDCALL ENDP
++
++_TEXT ENDS
++END
++
++#else
++
+       .text
+  
+         # This assumes we are using gas.
+         .balign 16
+-      .globl  _ffi_call_SYSV
+-      .def    _ffi_call_SYSV; .scl    2;      .type   32;     .endef
+-_ffi_call_SYSV:
++      .globl  _ffi_call_win32
++      .def    _ffi_call_win32;        .scl    2;      .type   32;     .endef
++_ffi_call_win32:
+ .LFB1:
+         pushl %ebp
+ .LCFI0:
+@@ -61,8 +419,10 @@ _ffi_call_SYSV:
+         # FIXME: Align the stack to a 128-bit boundary to avoid
+         # potential performance hits.
+-      call  *28(%ebp)
++        call  *28(%ebp)
+  
++        # stdcall functions pop arguments off the stack themselves
++
+         # Load %ecx with the return type code
+         movl  20(%ebp),%ecx
+  
+@@ -181,164 +541,11 @@ _ffi_call_SYSV:
+         movl %ebp,%esp
+         popl %ebp
+         ret
+-.ffi_call_SYSV_end:
++.ffi_call_win32_end:
+ .LFE1:
+         # This assumes we are using gas.
+         .balign 16
+-      .globl  _ffi_call_STDCALL
+-      .def    _ffi_call_STDCALL;      .scl    2;      .type   32;     .endef
+-_ffi_call_STDCALL:
+-.LFB2:
+-        pushl %ebp
+-.LCFI2:
+-        movl  %esp,%ebp
+-.LCFI3:
+-        # Make room for all of the new args.
+-        movl  16(%ebp),%ecx 
+-        subl  %ecx,%esp
+-
+-        movl  %esp,%eax
+-
+-        # Place all of the ffi_prep_args in position
+-        pushl 12(%ebp)
+-        pushl %eax
+-        call  *8(%ebp)
+-
+-        # Return stack to previous state and call the function
+-        addl  $8,%esp
+-
+-        # FIXME: Align the stack to a 128-bit boundary to avoid
+-        # potential performance hits.
+-
+-        call  *28(%ebp)
+-
+-        # stdcall functions pop arguments off the stack themselves
+-
+-        # Load %ecx with the return type code
+-        movl  20(%ebp),%ecx
+-
+-        # If the return value pointer is NULL, assume no return value.
+-        cmpl  $0,24(%ebp)
+-        jne   0f
+-
+-        # Even if there is no space for the return value, we are
+-        # obliged to handle floating-point values.
+-        cmpl  $FFI_TYPE_FLOAT,%ecx
+-        jne   .Lsc_noretval
+-        fstp  %st(0)
+-
+-        jmp   .Lsc_epilogue
+-
+-0:
+-      call    1f
+-      # Do not insert anything here between the call and the jump table.
+-.Lsc_store_table:
+-      .long   .Lsc_noretval           /* FFI_TYPE_VOID */
+-      .long   .Lsc_retint             /* FFI_TYPE_INT */
+-      .long   .Lsc_retfloat           /* FFI_TYPE_FLOAT */
+-      .long   .Lsc_retdouble          /* FFI_TYPE_DOUBLE */
+-      .long   .Lsc_retlongdouble      /* FFI_TYPE_LONGDOUBLE */
+-      .long   .Lsc_retuint8           /* FFI_TYPE_UINT8 */
+-      .long   .Lsc_retsint8           /* FFI_TYPE_SINT8 */
+-      .long   .Lsc_retuint16          /* FFI_TYPE_UINT16 */
+-      .long   .Lsc_retsint16          /* FFI_TYPE_SINT16 */
+-      .long   .Lsc_retint             /* FFI_TYPE_UINT32 */
+-      .long   .Lsc_retint             /* FFI_TYPE_SINT32 */
+-      .long   .Lsc_retint64           /* FFI_TYPE_UINT64 */
+-      .long   .Lsc_retint64           /* FFI_TYPE_SINT64 */
+-      .long   .Lsc_retstruct          /* FFI_TYPE_STRUCT */
+-      .long   .Lsc_retint             /* FFI_TYPE_POINTER */
+-      .long   .Lsc_retstruct1b        /* FFI_TYPE_SMALL_STRUCT_1B */
+-      .long   .Lsc_retstruct2b        /* FFI_TYPE_SMALL_STRUCT_2B */
+-      .long   .Lsc_retstruct4b        /* FFI_TYPE_SMALL_STRUCT_4B */
+-
+-1:
+-      add     %ecx, %ecx
+-      add     %ecx, %ecx
+-      add     (%esp),%ecx
+-      add     $4, %esp
+-      jmp     *(%ecx)
+-
+-      /* Sign/zero extend as appropriate.  */
+-.Lsc_retsint8:
+-      movsbl  %al, %eax
+-      jmp     .Lsc_retint
+-
+-.Lsc_retsint16:
+-      movswl  %ax, %eax
+-      jmp     .Lsc_retint
+-
+-.Lsc_retuint8:
+-      movzbl  %al, %eax
+-      jmp     .Lsc_retint
+-
+-.Lsc_retuint16:
+-      movzwl  %ax, %eax
+-      jmp     .Lsc_retint
+-
+-.Lsc_retint:
+-        # Load %ecx with the pointer to storage for the return value
+-        movl  24(%ebp),%ecx
+-        movl  %eax,0(%ecx)
+-        jmp   .Lsc_epilogue
+-
+-.Lsc_retfloat:
+-         # Load %ecx with the pointer to storage for the return value
+-        movl  24(%ebp),%ecx
+-        fstps (%ecx)
+-        jmp   .Lsc_epilogue
+-
+-.Lsc_retdouble:
+-        # Load %ecx with the pointer to storage for the return value
+-        movl  24(%ebp),%ecx
+-        fstpl (%ecx)
+-        jmp   .Lsc_epilogue
+-
+-.Lsc_retlongdouble:
+-        # Load %ecx with the pointer to storage for the return value
+-        movl  24(%ebp),%ecx
+-        fstpt (%ecx)
+-        jmp   .Lsc_epilogue
+-
+-.Lsc_retint64:
+-        # Load %ecx with the pointer to storage for the return value
+-        movl  24(%ebp),%ecx
+-        movl  %eax,0(%ecx)
+-        movl  %edx,4(%ecx)
+-      jmp   .Lsc_epilogue
+-
+-.Lsc_retstruct1b:
+-        # Load %ecx with the pointer to storage for the return value
+-        movl  24(%ebp),%ecx
+-        movb  %al,0(%ecx)
+-        jmp   .Lsc_epilogue
+-
+-.Lsc_retstruct2b:
+-        # Load %ecx with the pointer to storage for the return value
+-        movl  24(%ebp),%ecx
+-        movw  %ax,0(%ecx)
+-        jmp   .Lsc_epilogue
+-
+-.Lsc_retstruct4b:
+-        # Load %ecx with the pointer to storage for the return value
+-        movl  24(%ebp),%ecx
+-        movl  %eax,0(%ecx)
+-        jmp   .Lsc_epilogue
+-
+-.Lsc_retstruct:
+-        # Nothing to do!
+-
+-.Lsc_noretval:
+-.Lsc_epilogue:
+-        movl %ebp,%esp
+-        popl %ebp
+-        ret
+-.ffi_call_STDCALL_end:
+-.LFE2:
+-
+-        # This assumes we are using gas.
+-        .balign 16
+       .globl  _ffi_closure_SYSV
+       .def    _ffi_closure_SYSV;      .scl    2;      .type   32;     .endef
+ _ffi_closure_SYSV:
+@@ -742,38 +949,6 @@ _ffi_closure_STDCALL:
+ .LEFDE1:
+-.LSFDE2:
+-      .long   .LEFDE2-.LASFDE2        /* FDE Length */
+-.LASFDE2:
+-      .long   .LASFDE2-.Lframe1       /* FDE CIE offset */
+-#if defined __PIC__ && defined HAVE_AS_X86_PCREL
+-      .long   .LFB2-. /* FDE initial location */
+-#else
+-      .long   .LFB2
+-#endif
+-      .long   .LFE2-.LFB2     /* FDE address range */
+-#ifdef __PIC__
+-      .byte   0x0     /* .uleb128 0x0; Augmentation size */
+-#endif
+-      /* DW_CFA_xxx CFI instructions go here.  */
+-
+-      .byte   0x4     /* DW_CFA_advance_loc4 */
+-      .long   .LCFI2-.LFB2
+-      .byte   0xe     /* DW_CFA_def_cfa_offset CFA = r4 + 8 = 8(%esp) */
+-      .byte   0x8     /* .uleb128 0x8 */
+-      .byte   0x85    /* DW_CFA_offset, column 0x5 %ebp at CFA + 2 * -4 */
+-      .byte   0x2     /* .uleb128 0x2 */
+-
+-      .byte   0x4     /* DW_CFA_advance_loc4 */
+-      .long   .LCFI3-.LCFI2
+-      .byte   0xd     /* DW_CFA_def_cfa_register CFA = r5 = %ebp */
+-      .byte   0x5     /* .uleb128 0x5 */
+-
+-      /* End of DW_CFA_xxx CFI instructions.  */
+-      .align 4
+-.LEFDE2:
+-
+-
+ .LSFDE3:
+       .long   .LEFDE3-.LASFDE3        /* FDE Length */
+ .LASFDE3:
+@@ -875,3 +1050,6 @@ _ffi_closure_STDCALL:
+       /* End of DW_CFA_xxx CFI instructions.  */
+       .align 4
+ .LEFDE5:
++
++#endif /* !_MSC_VER */
++
index 060b184..829923e 100644 (file)
@@ -1,3 +1,4 @@
 stand-alone
 closure-api-example-doc
 sunstudio
+msvc
index eb78f9a..c1c3b9a 100644 (file)
@@ -109,16 +109,13 @@ ffi_status ffi_prep_cif(ffi_cif *cif, ffi_abi abi, unsigned int nargs,
   /* Perform a sanity check on the return type */
   FFI_ASSERT_VALID_TYPE(cif->rtype);
 
-  /* x86-64 and s390 stack space allocation is handled in prep_machdep.  */
-#if !defined M68K && !defined __x86_64__ && !defined S390 && !defined PA
+  /* x86, x86-64 and s390 stack space allocation is handled in prep_machdep. */
+#if !defined M68K && !defined __i386__ && !defined __x86_64__ && !defined S390 && !defined PA
   /* Make space for the return structure pointer */
   if (cif->rtype->type == FFI_TYPE_STRUCT
 #ifdef SPARC
       && (cif->abi != FFI_V9 || cif->rtype->size > 32)
 #endif
-#ifdef X86_DARWIN
-      && (cif->rtype->size > 8)
-#endif
      )
     bytes = STACK_ARG_SIZE(sizeof(void*));
 #endif
@@ -134,7 +131,7 @@ ffi_status ffi_prep_cif(ffi_cif *cif, ffi_abi abi, unsigned int nargs,
         check after the initialization.  */
       FFI_ASSERT_VALID_TYPE(*ptr);
 
-#if !defined __x86_64__ && !defined S390 && !defined PA
+#if !defined __i386__ && !defined __x86_64__ && !defined S390 && !defined PA
 #ifdef SPARC
       if (((*ptr)->type == FFI_TYPE_STRUCT
           && ((*ptr)->size > 16 || cif->abi != FFI_V9))
index 594235a..d99528b 100644 (file)
@@ -148,13 +148,13 @@ void ffi_prep_args(char *stack, extended_cif *ecif)
 /* Perform machine dependent cif processing */
 ffi_status ffi_prep_cif_machdep(ffi_cif *cif)
 {
+  unsigned int i;
+  ffi_type **ptr;
+
   /* Set the return type flag */
   switch (cif->rtype->type)
     {
     case FFI_TYPE_VOID:
-#ifdef X86
-    case FFI_TYPE_STRUCT:
-#endif
 #if defined(X86) || defined (X86_WIN32) || defined(X86_FREEBSD) || defined(X86_DARWIN) || defined(X86_WIN64)
     case FFI_TYPE_UINT8:
     case FFI_TYPE_UINT16:
@@ -165,7 +165,6 @@ ffi_status ffi_prep_cif_machdep(ffi_cif *cif)
     case FFI_TYPE_UINT32:
     case FFI_TYPE_SINT32:
 #endif
-
     case FFI_TYPE_SINT64:
     case FFI_TYPE_FLOAT:
     case FFI_TYPE_DOUBLE:
@@ -184,8 +183,8 @@ ffi_status ffi_prep_cif_machdep(ffi_cif *cif)
       cif->flags = FFI_TYPE_SINT64;
       break;
 
-#ifndef X86
     case FFI_TYPE_STRUCT:
+#ifndef X86
       if (cif->rtype->size == 1)
         {
           cif->flags = FFI_TYPE_SMALL_STRUCT_1B; /* same as char size */
@@ -207,15 +206,13 @@ ffi_status ffi_prep_cif_machdep(ffi_cif *cif)
           cif->flags = FFI_TYPE_SINT64; /* same as int64 type */
         }
       else
+#endif
         {
           cif->flags = FFI_TYPE_STRUCT;
-#ifdef X86_WIN64
           // allocate space for return value pointer
           cif->bytes += ALIGN(sizeof(void*), FFI_SIZEOF_ARG);
-#endif
         }
       break;
-#endif
 
     default:
 #ifdef X86_WIN64
@@ -229,41 +226,36 @@ ffi_status ffi_prep_cif_machdep(ffi_cif *cif)
       break;
     }
 
-#ifdef X86_DARWIN
-  cif->bytes = (cif->bytes + 15) & ~0xF;
-#endif
+  for (ptr = cif->arg_types, i = cif->nargs; i > 0; i--, ptr++)
+    {
+      if (((*ptr)->alignment - 1) & cif->bytes)
+        cif->bytes = ALIGN(cif->bytes, (*ptr)->alignment);
+      cif->bytes += ALIGN((*ptr)->size, FFI_SIZEOF_ARG);
+    }
 
 #ifdef X86_WIN64
-  {
-    unsigned int i;
-    ffi_type **ptr;
-
-    for (ptr = cif->arg_types, i = cif->nargs; i > 0; i--, ptr++)
-      {
-        if (((*ptr)->alignment - 1) & cif->bytes)
-          cif->bytes = ALIGN(cif->bytes, (*ptr)->alignment);
-        cif->bytes += ALIGN((*ptr)->size, FFI_SIZEOF_ARG);
-      }
-  }
   // ensure space for storing four registers
   cif->bytes += 4 * sizeof(ffi_arg);
 #endif
 
+#ifdef X86_DARWIN
+  cif->bytes = (cif->bytes + 15) & ~0xF;
+#endif
+
   return FFI_OK;
 }
 
-extern void ffi_call_SYSV(void (*)(char *, extended_cif *), extended_cif *,
-                          unsigned, unsigned, unsigned *, void (*fn)(void));
-
-#ifdef X86_WIN32
-extern void ffi_call_STDCALL(void (*)(char *, extended_cif *), extended_cif *,
-                          unsigned, unsigned, unsigned *, void (*fn)(void));
-
-#endif /* X86_WIN32 */
 #ifdef X86_WIN64
 extern int
 ffi_call_win64(void (*)(char *, extended_cif *), extended_cif *,
                unsigned, unsigned, unsigned *, void (*fn)(void));
+#elif defined(X86_WIN32)
+extern void
+ffi_call_win32(void (*)(char *, extended_cif *), extended_cif *,
+               unsigned, unsigned, unsigned *, void (*fn)(void));
+#else
+extern void ffi_call_SYSV(void (*)(char *, extended_cif *), extended_cif *,
+                          unsigned, unsigned, unsigned *, void (*fn)(void));
 #endif
 
 void ffi_call(ffi_cif *cif, void (*fn)(void), void *rvalue, void **avalue)
@@ -321,18 +313,18 @@ void ffi_call(ffi_cif *cif, void (*fn)(void), void *rvalue, void **avalue)
                        cif->flags, ecif.rvalue, fn);
       }
       break;
+#elif defined(X86_WIN32)
+    case FFI_SYSV:
+    case FFI_STDCALL:
+      ffi_call_win32(ffi_prep_args, &ecif, cif->bytes, cif->flags,
+                     ecif.rvalue, fn);
+      break;
 #else
     case FFI_SYSV:
       ffi_call_SYSV(ffi_prep_args, &ecif, cif->bytes, cif->flags, ecif.rvalue,
                     fn);
       break;
-#ifdef X86_WIN32
-    case FFI_STDCALL:
-      ffi_call_STDCALL(ffi_prep_args, &ecif, cif->bytes, cif->flags,
-                       ecif.rvalue, fn);
-      break;
-#endif /* X86_WIN32 */
-#endif /* X86_WIN64 */
+#endif
     default:
       FFI_ASSERT(0);
       break;
@@ -342,6 +334,8 @@ void ffi_call(ffi_cif *cif, void (*fn)(void), void *rvalue, void **avalue)
 
 /** private members **/
 
+/* The following __attribute__((regparm(1))) decorations will have no effect
+   on MSVC - standard cdecl convention applies. */
 static void ffi_prep_incoming_args_SYSV (char *stack, void **ret,
                                          void** args, ffi_cif* cif);
 void FFI_HIDDEN ffi_closure_SYSV (ffi_closure *)
@@ -390,11 +384,8 @@ ffi_closure_win64_inner (ffi_closure *closure, void *args) {
 }
 
 #else
-unsigned int FFI_HIDDEN
-ffi_closure_SYSV_inner (closure, respp, args)
-     ffi_closure *closure;
-     void **respp;
-     void *args;
+unsigned int FFI_HIDDEN __attribute__ ((regparm(1)))
+ffi_closure_SYSV_inner (ffi_closure *closure, void **respp, void *args)
 {
   /* our various things...  */
   ffi_cif       *cif;
@@ -505,7 +496,7 @@ ffi_prep_incoming_args_SYSV(char *stack, void **rvalue, void **avalue,
 /* How to make a trampoline.  Derived from gcc/config/i386/i386.c. */
 
 #define FFI_INIT_TRAMPOLINE(TRAMP,FUN,CTX) \
-({ unsigned char *__tramp = (unsigned char*)(TRAMP); \
+{ unsigned char *__tramp = (unsigned char*)(TRAMP); \
    unsigned int  __fun = (unsigned int)(FUN); \
    unsigned int  __ctx = (unsigned int)(CTX); \
    unsigned int  __dis = __fun - (__ctx + 10);  \
@@ -513,10 +504,10 @@ ffi_prep_incoming_args_SYSV(char *stack, void **rvalue, void **avalue,
    *(unsigned int*)  &__tramp[1] = __ctx; /* movl __ctx, %eax */ \
    *(unsigned char *)  &__tramp[5] = 0xe9; \
    *(unsigned int*)  &__tramp[6] = __dis; /* jmp __fun  */ \
- })
+ }
 
 #define FFI_INIT_TRAMPOLINE_STDCALL(TRAMP,FUN,CTX,SIZE)  \
-({ unsigned char *__tramp = (unsigned char*)(TRAMP); \
+{ unsigned char *__tramp = (unsigned char*)(TRAMP); \
    unsigned int  __fun = (unsigned int)(FUN); \
    unsigned int  __ctx = (unsigned int)(CTX); \
    unsigned int  __dis = __fun - (__ctx + 10); \
@@ -527,7 +518,7 @@ ffi_prep_incoming_args_SYSV(char *stack, void **rvalue, void **avalue,
    *(unsigned int*)  &__tramp[6] = __dis; /* call __fun  */ \
    *(unsigned char *)  &__tramp[10] = 0xc2; \
    *(unsigned short*)  &__tramp[11] = __size; /* ret __size  */ \
- })
+ }
 
 /* the cif must already be prep'ed */
 
@@ -627,16 +618,6 @@ ffi_prep_args_raw(char *stack, extended_cif *ecif)
  * libffi-1.20, this is not the case.)
  */
 
-extern void
-ffi_call_SYSV(void (*)(char *, extended_cif *), extended_cif *, unsigned, 
-              unsigned, unsigned *, void (*fn)(void));
-
-#ifdef X86_WIN32
-extern void
-ffi_call_STDCALL(void (*)(char *, extended_cif *), extended_cif *, unsigned,
-                 unsigned, unsigned *, void (*fn)(void));
-#endif /* X86_WIN32 */
-
 void
 ffi_raw_call(ffi_cif *cif, void (*fn)(void), void *rvalue, ffi_raw *fake_avalue)
 {
@@ -660,16 +641,18 @@ ffi_raw_call(ffi_cif *cif, void (*fn)(void), void *rvalue, ffi_raw *fake_avalue)
   
   switch (cif->abi) 
     {
+#ifdef X86_WIN32
+    case FFI_SYSV:
+    case FFI_STDCALL:
+      ffi_call_win32(ffi_prep_args_raw, &ecif, cif->bytes, cif->flags,
+                     ecif.rvalue, fn);
+      break;
+#else
     case FFI_SYSV:
       ffi_call_SYSV(ffi_prep_args_raw, &ecif, cif->bytes, cif->flags,
                     ecif.rvalue, fn);
       break;
-#ifdef X86_WIN32
-    case FFI_STDCALL:
-      ffi_call_STDCALL(ffi_prep_args_raw, &ecif, cif->bytes, cif->flags,
-                       ecif.rvalue, fn);
-      break;
-#endif /* X86_WIN32 */
+#endif
     default:
       FFI_ASSERT(0);
       break;
index 7489dab..ac1ed6f 100644 (file)
@@ -2,6 +2,7 @@
    win32.S - Copyright (c) 1996, 1998, 2001, 2002, 2009  Red Hat, Inc.
             Copyright (c) 2001  John Beniton
             Copyright (c) 2002  Ranjit Mathew
+            Copyright (c) 2009  Daniel Witte
                        
  
    X86 Foreign Function Interface
 #define LIBFFI_ASM
 #include <fficonfig.h>
 #include <ffi.h>
+
+#ifdef _MSC_VER
+
+.386
+.MODEL FLAT, C
+
+EXTRN ffi_closure_SYSV_inner:NEAR
+
+_TEXT SEGMENT
+
+ffi_call_win32 PROC NEAR,
+    ffi_prep_args : NEAR PTR DWORD,
+    ecif          : NEAR PTR DWORD,
+    cif_bytes     : DWORD,
+    cif_flags     : DWORD,
+    rvalue        : NEAR PTR DWORD,
+    fn            : NEAR PTR DWORD
+
+        ;; Make room for all of the new args.
+        mov  ecx, cif_bytes
+        sub  esp, ecx
+
+        mov  eax, esp
+
+        ;; Place all of the ffi_prep_args in position
+        push ecif
+        push eax
+        call ffi_prep_args
+
+        ;; Return stack to previous state and call the function
+        add  esp, 8
+
+        call fn
+
+        ;; cdecl:   we restore esp in the epilogue, so there's no need to
+        ;;          remove the space we pushed for the args.
+        ;; stdcall: the callee has already cleaned the stack.
+
+        ;; Load ecx with the return type code
+        mov  ecx, cif_flags
+
+        ;; If the return value pointer is NULL, assume no return value.
+        cmp  rvalue, 0
+        jne  ca_jumptable
+
+        ;; Even if there is no space for the return value, we are
+        ;; obliged to handle floating-point values.
+        cmp  ecx, FFI_TYPE_FLOAT
+        jne  ca_epilogue
+        fstp st(0)
+
+        jmp  ca_epilogue
+
+ca_jumptable:
+        jmp  [ca_jumpdata + 4 * ecx]
+ca_jumpdata:
+        ;; Do not insert anything here between label and jump table.
+        dd offset ca_epilogue       ;; FFI_TYPE_VOID
+        dd offset ca_retint         ;; FFI_TYPE_INT
+        dd offset ca_retfloat       ;; FFI_TYPE_FLOAT
+        dd offset ca_retdouble      ;; FFI_TYPE_DOUBLE
+        dd offset ca_retlongdouble  ;; FFI_TYPE_LONGDOUBLE
+        dd offset ca_retint8        ;; FFI_TYPE_UINT8
+        dd offset ca_retint8        ;; FFI_TYPE_SINT8
+        dd offset ca_retint16       ;; FFI_TYPE_UINT16
+        dd offset ca_retint16       ;; FFI_TYPE_SINT16
+        dd offset ca_retint         ;; FFI_TYPE_UINT32
+        dd offset ca_retint         ;; FFI_TYPE_SINT32
+        dd offset ca_retint64       ;; FFI_TYPE_UINT64
+        dd offset ca_retint64       ;; FFI_TYPE_SINT64
+        dd offset ca_epilogue       ;; FFI_TYPE_STRUCT
+        dd offset ca_retint         ;; FFI_TYPE_POINTER
+        dd offset ca_retint8        ;; FFI_TYPE_SMALL_STRUCT_1B
+        dd offset ca_retint16       ;; FFI_TYPE_SMALL_STRUCT_2B
+        dd offset ca_retint         ;; FFI_TYPE_SMALL_STRUCT_4B
+
+ca_retint8:
+        ;; Load %ecx with the pointer to storage for the return value
+        mov   ecx, rvalue
+        mov   [ecx + 0], al
+        jmp   ca_epilogue
+
+ca_retint16:
+        ;; Load %ecx with the pointer to storage for the return value
+        mov   ecx, rvalue
+        mov   [ecx + 0], ax
+        jmp   ca_epilogue
+
+ca_retint:
+        ;; Load %ecx with the pointer to storage for the return value
+        mov   ecx, rvalue
+        mov   [ecx + 0], eax
+        jmp   ca_epilogue
+
+ca_retint64:
+        ;; Load %ecx with the pointer to storage for the return value
+        mov   ecx, rvalue
+        mov   [ecx + 0], eax
+        mov   [ecx + 4], edx
+        jmp   ca_epilogue
+
+ca_retfloat:
+        ;; Load %ecx with the pointer to storage for the return value
+        mov   ecx, rvalue
+        fstp  DWORD PTR [ecx]
+        jmp   ca_epilogue
+
+ca_retdouble:
+        ;; Load %ecx with the pointer to storage for the return value
+        mov   ecx, rvalue
+        fstp  QWORD PTR [ecx]
+        jmp   ca_epilogue
+
+ca_retlongdouble:
+        ;; Load %ecx with the pointer to storage for the return value
+        mov   ecx, rvalue
+        fstp  TBYTE PTR [ecx]
+        jmp   ca_epilogue
+
+ca_epilogue:
+        ;; Epilogue code is autogenerated.
+        ret
+ffi_call_win32 ENDP
+
+ffi_closure_SYSV PROC NEAR FORCEFRAME
+    ;; the ffi_closure ctx is passed in eax by the trampoline.
+
+        sub  esp, 40
+        lea  edx, [ebp - 24]
+        mov  [ebp - 12], edx         ;; resp
+        lea  edx, [ebp + 8]
+        mov  [esp + 8], edx          ;; args
+        lea  edx, [ebp - 12]
+        mov  [esp + 4], edx          ;; &resp
+        mov  [esp], eax              ;; closure
+        call ffi_closure_SYSV_inner
+        mov  ecx, [ebp - 12]
+
+cs_jumptable:
+        jmp  [cs_jumpdata + 4 * eax]
+cs_jumpdata:
+        ;; Do not insert anything here between the label and jump table.
+        dd offset cs_epilogue       ;; FFI_TYPE_VOID
+        dd offset cs_retint         ;; FFI_TYPE_INT
+        dd offset cs_retfloat       ;; FFI_TYPE_FLOAT
+        dd offset cs_retdouble      ;; FFI_TYPE_DOUBLE
+        dd offset cs_retlongdouble  ;; FFI_TYPE_LONGDOUBLE
+        dd offset cs_retint8        ;; FFI_TYPE_UINT8
+        dd offset cs_retint8        ;; FFI_TYPE_SINT8
+        dd offset cs_retint16       ;; FFI_TYPE_UINT16
+        dd offset cs_retint16       ;; FFI_TYPE_SINT16
+        dd offset cs_retint         ;; FFI_TYPE_UINT32
+        dd offset cs_retint         ;; FFI_TYPE_SINT32
+        dd offset cs_retint64       ;; FFI_TYPE_UINT64
+        dd offset cs_retint64       ;; FFI_TYPE_SINT64
+        dd offset cs_retstruct      ;; FFI_TYPE_STRUCT
+        dd offset cs_retint         ;; FFI_TYPE_POINTER
+        dd offset cs_retint8        ;; FFI_TYPE_SMALL_STRUCT_1B
+        dd offset cs_retint16       ;; FFI_TYPE_SMALL_STRUCT_2B
+        dd offset cs_retint         ;; FFI_TYPE_SMALL_STRUCT_4B
+
+cs_retint8:
+        mov   al, [ecx]
+        jmp   cs_epilogue
+
+cs_retint16:
+        mov   ax, [ecx]
+        jmp   cs_epilogue
+
+cs_retint:
+        mov   eax, [ecx]
+        jmp   cs_epilogue
+
+cs_retint64:
+        mov   eax, [ecx + 0]
+        mov   edx, [ecx + 4]
+        jmp   cs_epilogue
+
+cs_retfloat:
+        fld   DWORD PTR [ecx]
+        jmp   cs_epilogue
+
+cs_retdouble:
+        fld   QWORD PTR [ecx]
+        jmp   cs_epilogue
+
+cs_retlongdouble:
+        fld   TBYTE PTR [ecx]
+        jmp   cs_epilogue
+
+cs_retstruct:
+        ;; Caller expects us to pop struct return value pointer hidden arg.
+        ;; Epilogue code is autogenerated.
+        ret    4
+
+cs_epilogue:
+        ;; Epilogue code is autogenerated.
+        ret
+ffi_closure_SYSV ENDP
+
+#if !FFI_NO_RAW_API
+
+#define RAW_CLOSURE_CIF_OFFSET ((FFI_TRAMPOLINE_SIZE + 3) AND NOT 3)
+#define RAW_CLOSURE_FUN_OFFSET (RAW_CLOSURE_CIF_OFFSET + 4)
+#define RAW_CLOSURE_USER_DATA_OFFSET (RAW_CLOSURE_FUN_OFFSET + 4)
+#define CIF_FLAGS_OFFSET 20
+
+ffi_closure_raw_SYSV PROC NEAR USES esi
+    ;; the ffi_closure ctx is passed in eax by the trampoline.
+
+        sub  esp, 40
+        mov  esi, [eax + RAW_CLOSURE_CIF_OFFSET]        ;; closure->cif
+        mov  edx, [eax + RAW_CLOSURE_USER_DATA_OFFSET]  ;; closure->user_data
+        mov  [esp + 12], edx                            ;; user_data
+        lea  edx, [ebp + 8]
+        mov  [esp + 8], edx                             ;; raw_args
+        lea  edx, [ebp - 24]
+        mov  [esp + 4], edx                             ;; &res
+        mov  [esp], esi                                 ;; cif
+        call DWORD PTR [eax + RAW_CLOSURE_FUN_OFFSET]   ;; closure->fun
+        mov  eax, [esi + CIF_FLAGS_OFFSET]              ;; cif->flags
+        lea  ecx, [ebp - 24]
+
+cr_jumptable:
+        jmp  [cr_jumpdata + 4 * eax]
+cr_jumpdata:
+        ;; Do not insert anything here between the label and jump table.
+        dd offset cr_epilogue       ;; FFI_TYPE_VOID
+        dd offset cr_retint         ;; FFI_TYPE_INT
+        dd offset cr_retfloat       ;; FFI_TYPE_FLOAT
+        dd offset cr_retdouble      ;; FFI_TYPE_DOUBLE
+        dd offset cr_retlongdouble  ;; FFI_TYPE_LONGDOUBLE
+        dd offset cr_retint8        ;; FFI_TYPE_UINT8
+        dd offset cr_retint8        ;; FFI_TYPE_SINT8
+        dd offset cr_retint16       ;; FFI_TYPE_UINT16
+        dd offset cr_retint16       ;; FFI_TYPE_SINT16
+        dd offset cr_retint         ;; FFI_TYPE_UINT32
+        dd offset cr_retint         ;; FFI_TYPE_SINT32
+        dd offset cr_retint64       ;; FFI_TYPE_UINT64
+        dd offset cr_retint64       ;; FFI_TYPE_SINT64
+        dd offset cr_epilogue       ;; FFI_TYPE_STRUCT
+        dd offset cr_retint         ;; FFI_TYPE_POINTER
+        dd offset cr_retint8        ;; FFI_TYPE_SMALL_STRUCT_1B
+        dd offset cr_retint16       ;; FFI_TYPE_SMALL_STRUCT_2B
+        dd offset cr_retint         ;; FFI_TYPE_SMALL_STRUCT_4B
+
+cr_retint8:
+        mov   al, [ecx]
+        jmp   cr_epilogue
+
+cr_retint16:
+        mov   ax, [ecx]
+        jmp   cr_epilogue
+
+cr_retint:
+        mov   eax, [ecx]
+        jmp   cr_epilogue
+
+cr_retint64:
+        mov   eax, [ecx + 0]
+        mov   edx, [ecx + 4]
+        jmp   cr_epilogue
+
+cr_retfloat:
+        fld   DWORD PTR [ecx]
+        jmp   cr_epilogue
+
+cr_retdouble:
+        fld   QWORD PTR [ecx]
+        jmp   cr_epilogue
+
+cr_retlongdouble:
+        fld   TBYTE PTR [ecx]
+        jmp   cr_epilogue
+
+cr_epilogue:
+        ;; Epilogue code is autogenerated.
+        ret
+ffi_closure_raw_SYSV ENDP
+
+#endif /* !FFI_NO_RAW_API */
+
+ffi_closure_STDCALL PROC NEAR FORCEFRAME
+    ;; the ffi_closure ctx is passed in eax by the trampoline.
+
+        sub  esp, 40
+        lea  edx, [ebp - 24]
+        mov  [ebp - 12], edx         ;; resp
+        lea  edx, [ebp + 12]         ;; account for stub return address on stack
+        mov  [esp + 8], edx          ;; args
+        lea  edx, [ebp - 12]
+        mov  [esp + 4], edx          ;; &resp
+        mov  [esp], eax              ;; closure
+        call ffi_closure_SYSV_inner
+        mov  ecx, [ebp - 12]
+
+cd_jumptable:
+        jmp  [cd_jumpdata + 4 * eax]
+cd_jumpdata:
+        ;; Do not insert anything here between the label and jump table.
+        dd offset cd_epilogue       ;; FFI_TYPE_VOID
+        dd offset cd_retint         ;; FFI_TYPE_INT
+        dd offset cd_retfloat       ;; FFI_TYPE_FLOAT
+        dd offset cd_retdouble      ;; FFI_TYPE_DOUBLE
+        dd offset cd_retlongdouble  ;; FFI_TYPE_LONGDOUBLE
+        dd offset cd_retint8        ;; FFI_TYPE_UINT8
+        dd offset cd_retint8        ;; FFI_TYPE_SINT8
+        dd offset cd_retint16       ;; FFI_TYPE_UINT16
+        dd offset cd_retint16       ;; FFI_TYPE_SINT16
+        dd offset cd_retint         ;; FFI_TYPE_UINT32
+        dd offset cd_retint         ;; FFI_TYPE_SINT32
+        dd offset cd_retint64       ;; FFI_TYPE_UINT64
+        dd offset cd_retint64       ;; FFI_TYPE_SINT64
+        dd offset cd_epilogue       ;; FFI_TYPE_STRUCT
+        dd offset cd_retint         ;; FFI_TYPE_POINTER
+        dd offset cd_retint8        ;; FFI_TYPE_SMALL_STRUCT_1B
+        dd offset cd_retint16       ;; FFI_TYPE_SMALL_STRUCT_2B
+        dd offset cd_retint         ;; FFI_TYPE_SMALL_STRUCT_4B
+
+cd_retint8:
+        mov   al, [ecx]
+        jmp   cd_epilogue
+
+cd_retint16:
+        mov   ax, [ecx]
+        jmp   cd_epilogue
+
+cd_retint:
+        mov   eax, [ecx]
+        jmp   cd_epilogue
+
+cd_retint64:
+        mov   eax, [ecx + 0]
+        mov   edx, [ecx + 4]
+        jmp   cd_epilogue
+
+cd_retfloat:
+        fld   DWORD PTR [ecx]
+        jmp   cd_epilogue
+
+cd_retdouble:
+        fld   QWORD PTR [ecx]
+        jmp   cd_epilogue
+
+cd_retlongdouble:
+        fld   TBYTE PTR [ecx]
+        jmp   cd_epilogue
+
+cd_epilogue:
+        ;; Epilogue code is autogenerated.
+        ret
+ffi_closure_STDCALL ENDP
+
+_TEXT ENDS
+END
+
+#else
+
        .text
  
         # This assumes we are using gas.
         .balign 16
-       .globl  _ffi_call_SYSV
-       .def    _ffi_call_SYSV; .scl    2;      .type   32;     .endef
-_ffi_call_SYSV:
+       .globl  _ffi_call_win32
+       .def    _ffi_call_win32;        .scl    2;      .type   32;     .endef
+_ffi_call_win32:
 .LFB1:
         pushl %ebp
 .LCFI0:
@@ -61,8 +419,10 @@ _ffi_call_SYSV:
         # FIXME: Align the stack to a 128-bit boundary to avoid
         # potential performance hits.
 
-       call  *28(%ebp)
+        call  *28(%ebp)
  
+        # stdcall functions pop arguments off the stack themselves
+
         # Load %ecx with the return type code
         movl  20(%ebp),%ecx
  
@@ -181,164 +541,11 @@ _ffi_call_SYSV:
         movl %ebp,%esp
         popl %ebp
         ret
-.ffi_call_SYSV_end:
+.ffi_call_win32_end:
 .LFE1:
 
         # This assumes we are using gas.
         .balign 16
-       .globl  _ffi_call_STDCALL
-       .def    _ffi_call_STDCALL;      .scl    2;      .type   32;     .endef
-_ffi_call_STDCALL:
-.LFB2:
-        pushl %ebp
-.LCFI2:
-        movl  %esp,%ebp
-.LCFI3:
-        # Make room for all of the new args.
-        movl  16(%ebp),%ecx 
-        subl  %ecx,%esp
-
-        movl  %esp,%eax
-
-        # Place all of the ffi_prep_args in position
-        pushl 12(%ebp)
-        pushl %eax
-        call  *8(%ebp)
-
-        # Return stack to previous state and call the function
-        addl  $8,%esp
-
-        # FIXME: Align the stack to a 128-bit boundary to avoid
-        # potential performance hits.
-
-        call  *28(%ebp)
-
-        # stdcall functions pop arguments off the stack themselves
-
-        # Load %ecx with the return type code
-        movl  20(%ebp),%ecx
-
-        # If the return value pointer is NULL, assume no return value.
-        cmpl  $0,24(%ebp)
-        jne   0f
-
-        # Even if there is no space for the return value, we are
-        # obliged to handle floating-point values.
-        cmpl  $FFI_TYPE_FLOAT,%ecx
-        jne   .Lsc_noretval
-        fstp  %st(0)
-
-        jmp   .Lsc_epilogue
-
-0:
-       call    1f
-       # Do not insert anything here between the call and the jump table.
-.Lsc_store_table:
-       .long   .Lsc_noretval           /* FFI_TYPE_VOID */
-       .long   .Lsc_retint             /* FFI_TYPE_INT */
-       .long   .Lsc_retfloat           /* FFI_TYPE_FLOAT */
-       .long   .Lsc_retdouble          /* FFI_TYPE_DOUBLE */
-       .long   .Lsc_retlongdouble      /* FFI_TYPE_LONGDOUBLE */
-       .long   .Lsc_retuint8           /* FFI_TYPE_UINT8 */
-       .long   .Lsc_retsint8           /* FFI_TYPE_SINT8 */
-       .long   .Lsc_retuint16          /* FFI_TYPE_UINT16 */
-       .long   .Lsc_retsint16          /* FFI_TYPE_SINT16 */
-       .long   .Lsc_retint             /* FFI_TYPE_UINT32 */
-       .long   .Lsc_retint             /* FFI_TYPE_SINT32 */
-       .long   .Lsc_retint64           /* FFI_TYPE_UINT64 */
-       .long   .Lsc_retint64           /* FFI_TYPE_SINT64 */
-       .long   .Lsc_retstruct          /* FFI_TYPE_STRUCT */
-       .long   .Lsc_retint             /* FFI_TYPE_POINTER */
-       .long   .Lsc_retstruct1b        /* FFI_TYPE_SMALL_STRUCT_1B */
-       .long   .Lsc_retstruct2b        /* FFI_TYPE_SMALL_STRUCT_2B */
-       .long   .Lsc_retstruct4b        /* FFI_TYPE_SMALL_STRUCT_4B */
-
-1:
-       add     %ecx, %ecx
-       add     %ecx, %ecx
-       add     (%esp),%ecx
-       add     $4, %esp
-       jmp     *(%ecx)
-
-       /* Sign/zero extend as appropriate.  */
-.Lsc_retsint8:
-       movsbl  %al, %eax
-       jmp     .Lsc_retint
-
-.Lsc_retsint16:
-       movswl  %ax, %eax
-       jmp     .Lsc_retint
-
-.Lsc_retuint8:
-       movzbl  %al, %eax
-       jmp     .Lsc_retint
-
-.Lsc_retuint16:
-       movzwl  %ax, %eax
-       jmp     .Lsc_retint
-
-.Lsc_retint:
-        # Load %ecx with the pointer to storage for the return value
-        movl  24(%ebp),%ecx
-        movl  %eax,0(%ecx)
-        jmp   .Lsc_epilogue
-
-.Lsc_retfloat:
-         # Load %ecx with the pointer to storage for the return value
-        movl  24(%ebp),%ecx
-        fstps (%ecx)
-        jmp   .Lsc_epilogue
-
-.Lsc_retdouble:
-        # Load %ecx with the pointer to storage for the return value
-        movl  24(%ebp),%ecx
-        fstpl (%ecx)
-        jmp   .Lsc_epilogue
-
-.Lsc_retlongdouble:
-        # Load %ecx with the pointer to storage for the return value
-        movl  24(%ebp),%ecx
-        fstpt (%ecx)
-        jmp   .Lsc_epilogue
-
-.Lsc_retint64:
-        # Load %ecx with the pointer to storage for the return value
-        movl  24(%ebp),%ecx
-        movl  %eax,0(%ecx)
-        movl  %edx,4(%ecx)
-       jmp   .Lsc_epilogue
-
-.Lsc_retstruct1b:
-        # Load %ecx with the pointer to storage for the return value
-        movl  24(%ebp),%ecx
-        movb  %al,0(%ecx)
-        jmp   .Lsc_epilogue
-
-.Lsc_retstruct2b:
-        # Load %ecx with the pointer to storage for the return value
-        movl  24(%ebp),%ecx
-        movw  %ax,0(%ecx)
-        jmp   .Lsc_epilogue
-
-.Lsc_retstruct4b:
-        # Load %ecx with the pointer to storage for the return value
-        movl  24(%ebp),%ecx
-        movl  %eax,0(%ecx)
-        jmp   .Lsc_epilogue
-
-.Lsc_retstruct:
-        # Nothing to do!
-
-.Lsc_noretval:
-.Lsc_epilogue:
-        movl %ebp,%esp
-        popl %ebp
-        ret
-.ffi_call_STDCALL_end:
-.LFE2:
-
-        # This assumes we are using gas.
-        .balign 16
        .globl  _ffi_closure_SYSV
        .def    _ffi_closure_SYSV;      .scl    2;      .type   32;     .endef
 _ffi_closure_SYSV:
@@ -742,38 +949,6 @@ _ffi_closure_STDCALL:
 .LEFDE1:
 
 
-.LSFDE2:
-       .long   .LEFDE2-.LASFDE2        /* FDE Length */
-.LASFDE2:
-       .long   .LASFDE2-.Lframe1       /* FDE CIE offset */
-#if defined __PIC__ && defined HAVE_AS_X86_PCREL
-       .long   .LFB2-. /* FDE initial location */
-#else
-       .long   .LFB2
-#endif
-       .long   .LFE2-.LFB2     /* FDE address range */
-#ifdef __PIC__
-       .byte   0x0     /* .uleb128 0x0; Augmentation size */
-#endif
-       /* DW_CFA_xxx CFI instructions go here.  */
-
-       .byte   0x4     /* DW_CFA_advance_loc4 */
-       .long   .LCFI2-.LFB2
-       .byte   0xe     /* DW_CFA_def_cfa_offset CFA = r4 + 8 = 8(%esp) */
-       .byte   0x8     /* .uleb128 0x8 */
-       .byte   0x85    /* DW_CFA_offset, column 0x5 %ebp at CFA + 2 * -4 */
-       .byte   0x2     /* .uleb128 0x2 */
-
-       .byte   0x4     /* DW_CFA_advance_loc4 */
-       .long   .LCFI3-.LCFI2
-       .byte   0xd     /* DW_CFA_def_cfa_register CFA = r5 = %ebp */
-       .byte   0x5     /* .uleb128 0x5 */
-
-       /* End of DW_CFA_xxx CFI instructions.  */
-       .align 4
-.LEFDE2:
-
-
 .LSFDE3:
        .long   .LEFDE3-.LASFDE3        /* FDE Length */
 .LASFDE3:
@@ -875,3 +1050,6 @@ _ffi_closure_STDCALL:
        /* End of DW_CFA_xxx CFI instructions.  */
        .align 4
 .LEFDE5:
+
+#endif /* !_MSC_VER */
+