对象pascal上的C回调函数崩溃

问题描述 投票:6回答:1

我正在创建一个像下面的C dll。

#ifdef TEST_EXPORTS
#define TEST_API __declspec(dllexport)
#else
#define TEST_API __declspec(dllimport)
#endif

#ifdef __cplusplus
extern "C" {
#endif
    /* fun accept an argument status_callback which is a function pointer*/
    TEST_API void fun(void(*status_callback)(int));
#ifdef __cplusplus
}
#endif

和乐趣实现为:

void fun(void(*status_callback)(int)) {
   status_callback(0);
}

在对象pascal上使用这个dll与delphi xe如下所示我在回调函数的地址遇到了访问冲突的崩溃

type
 TStatusCallBack = procedure( Value: Integer ); stdcall;
 T_fun = procedure(cb : TStatusCallback );

procedure BCallBack( Value: Integer ); stdcall;
begin
   BStatus := value;
end;

begin
 _fun(BCallBack);
end;

实例化代码:

DLLHandle := LoadLibrary( ‘mylib.dll’ );
if (DLLHandle > HDLL_ERROR) then
begin
  _fun := T_fun(GetProcAddress( DLLHandle, 'fun');
end;

有什么问题?既然我是C的noobie任何评论?

c++ delphi
1个回答
6
投票

我们缺少重要的细节。我们无法在C ++方面看到fun的实现,我们无法看到你如何在Delphi方面声明_fun

但是我们可以看到一个错误。回调的调用约定不匹配。 C ++代码将回调定义为__cdecl。您需要使用cdecl来匹配Delphi端,而不是stdcall

同样地,我希望你的Delphi代码获得_fun的调用约定错误。再次它应该是cdecl但错误消息表明你的声明实际上是register

总之,对Delphi代码进行以下更改:

  • 将回调函数类型声明为cdecl
  • 将实现回调的函数声明为cdecl
  • 宣布fun的进口为cdecl

查看编辑,检查LoadLibrary返回值的错误是错误的。返回值为零表示错误。比较等于零以测试失败。

调用GetProcAddress时也无法检查错误。您需要处理返回nil的情况。同样,这可能会导致运行时出现错误,并且代码已写入。

编辑中的代码显示我的所有猜测都是正确的。解决方案详见上述要点。

© www.soinside.com 2019 - 2024. All rights reserved.