我正在尝试为我的新编程语言提供 FFI 支持,该语言是使用 MinGW 工具链用 QT Creator 用 C++ 编写的。
为此,我使用了在这里找到的定制版本的 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);
}
}