从Delphi DLL导出全局符号

前端之家收集整理的这篇文章主要介绍了从Delphi DLL导出全局符号前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我试图在Delphi中创建一个Gecko 2.0兼容的DLL.

以前(前Gecko 2.0)导出NSGetModule()函数所需的DLL.这完美无瑕.

从Firefox 4开始,我的DLL被加载(我已经通过我的初始化部分中的断点验证了这一点),但是我的NSGetModule()函数不再被调用了.这是设计的行为,因为从Gecko 2.0(Firefox 4)开始,二进制组件不应该导出NSGetModule()函数

https://developer.mozilla.org/en/XPCOM/XPCOM_changes_in_Gecko_2.0#Binary_components

根据这些文档,我的DLL需要导出指向一个结构体的NSModule数据符号.在Delphi术语中,我认为这是一个全局变量,它指向Delphi记录.

在C中,这是如何导出(全局)数据符号:

define NSMODULE_DEFN(_name) extern "C" NS_EXPORT mozilla::Module const *const NSModule

我的问题:如何在Delphi中完成这个?如何导出全局变量

感谢您的反馈.

解决方法

Delphi以类似于导出函数的方式从DLL导出全局变量
library exp;
var
  global: Integer;
exports global;
end.

Delphi可以从DLL导入全局变量,但它有点黑客:声明与导入全局相同名称的DLL导入过程,然后获取过程的地址并进行适当调整.从Delphi的角度看,DLL导入的程序是通过DLL导入表进行间接跳转的存根.导出的变量由OS加载器链接,将导出的全局地址放在导入表中,几乎完全像导出过程的地址类似于修补.

例如:

{$apptype console}

procedure global; external 'exp.dll';

function GetGlobalAddr: PInteger;
type
  PPPointer = ^PPointer;
var
  p: PByte;
begin
  p := @global;
  Assert(p^ = $FF); // $FF $25 => indirect jump m32
  Inc(p);
  Assert(p^ = $25);
  Inc(p);
  Result := PPPointer(p)^^
end;

begin
  Writeln(GetGlobalAddr^);
end.

当然,后面的细节是实现和平台依赖的.可能一个更安全的方法是使用LoadLibrary与GetProcAddress,它会在传递其名称时返回全局变量的地址.当然这也是平台依赖的.

64位更新:

在Windows 64位中,代码略有不同.操作码是相同的,但是相同指令序列的寻址模式是不同的;而不是一个32位的绝对偏移量,它是一个32位的相对偏移量.

function GetGlobalAddr: PInteger;
type
  PPPointer = ^PPointer;
var
  p: PByte;
  ofs: Integer;
begin
  p := @global;
  Assert(p^ = $FF); // $FF $25 => indirect jump m32
  Inc(p);
  Assert(p^ = $25);
  Inc(p);
  // 32-bit offset follows
  ofs := PInteger(p)^;
  // offset is relative to next instruction
  Inc(p,SizeOf(ofs) + ofs);
  Result := PPPointer(p)^^
end;
原文链接:/delphi/102435.html

猜你在找的Delphi相关文章