在MinGW上使用libffi调用Windows API

Calling Windows API with libffi on MinGW

本文关键字:调用 Windows API libffi MinGW      更新时间:2023-10-16

我正在尝试为我的新编程语言提供FFI支持,该语言是用c++和QT Creator使用MinGW工具链编写的。

为此,我使用了libffi的自定义版本:http://ftp.gnome.org/pub/GNOME/binaries/win32/dependencies/libffi-dev_3.0.6-1_win32.zip

我还尝试了另一个版本:http://pkgs.org/fedora-14/fedora-updates-i386/mingw32-libffi-3.0.9-1.fc14.noarch.rpm.html,在Linux上下载SRPM文件,提取它,并将所需的文件复制到Windows分区。

无论如何,我包含了所需的头文件,将导入库添加到项目中,并将.dll放在应用程序的。exe旁边,它编译并运行,成功调用MessageBeep()。接下来我尝试了MessageBoxA(),但它一直崩溃。调试器似乎没有提供多少有用的信息(编辑:除了对MessageBoxA 的调用确实发生的事实之外),所以我一直在摆弄东西并重新运行,但无济于事。

为了将问题从我的语言细节中分离出来,我尝试通过自己填充所有参数来手动调用MessageBoxA,导致下面的代码仍然崩溃。

所以我的问题提炼为:我怎么能得到下面的代码片段运行下QT Creator/MinGW和实际显示一个消息框?

#include "libffi/include/ffi.h"
#include <QLibrary>
void testMessageBox()
{
    int n = 4;
    ffi_cif cif;
    ffi_type **ffi_argTypes = new ffi_type*[n];
    void **values = new void*[n];
    values[0] = new ulong(0);
    values[1] = (void *) "hello";
    values[2] = (void *) "mommy";
    values[3] = new int32_t(0);
    ffi_argTypes[0] = &ffi_type_ulong;
    ffi_argTypes[1] = &ffi_type_pointer;
    ffi_argTypes[2] = &ffi_type_pointer;
    ffi_argTypes[3] = &ffi_type_uint32;
    ffi_type *c_retType = &ffi_type_sint32;
    int32_t rc; // return value
    if (ffi_prep_cif(&cif, FFI_STDCALL, n, c_retType, ffi_argTypes) == FFI_OK)
    {
        QLibrary lib("user32.dll");
        lib.load();
        void *msgbox = lib.resolve("MessageBoxA");
        ffi_call(&cif, (void (*)()) msgbox, &rc, values);
    }
}

您应该将地址传递给values数组而不是value。mingw64下的工作代码为

   #include <stdio.h>
   #include <ffi.h>
   #include <Windows.h>
   int main()
   {
     ffi_cif cif;
            HINSTANCE dllHandle = LoadLibrary("user32.dll");
            int n = 4;    
            ffi_type *ffi_argTypes[4]; 
            void *values[4];
            UINT64 a=0;
            UINT32 b=0;
            TCHAR* s1= "hello"; 
            TCHAR* s2= "hello2";  
            values[0] = &a;
            values[1] = &s1;
            values[2] = &s2;
            values[3] = &b;    
            ffi_argTypes[0] = &ffi_type_uint64;    
            ffi_argTypes[1] = &ffi_type_pointer;    
            ffi_argTypes[2] = &ffi_type_pointer;    
            ffi_argTypes[3] = &ffi_type_uint; 
            ffi_type *c_retType = &ffi_type_sint;    
            ffi_type rc; // return value    
            if (ffi_prep_cif(&cif, FFI_DEFAULT_ABI, 4, &ffi_type_sint, ffi_argTypes) == FFI_OK)    {

                ffi_call(&cif, FFI_FN(GetProcAddress(dllHandle,"MessageBoxA")), &rc, values); 
            }

       return 0;
     }