delphi – TTypeInfo之前的“身份指针”是什么?

如果你在Delphi内部进行足够的讨论,你会发现一些奇怪的东西,并且显然没有记录编译器生成的TTypeInfo记录.如果PTypeInfo指向地址X处的TTypeInfo记录,则在X-4处,您将发现接下来的4个字节描述了指向X的指针.例如:
procedure test(info: PTypeInfo);
var
  addr: cardinal;
  ptr: PPointer;
begin
  addr := cardinal(info);
  writeln('addr: ',addr);
  dec(addr,4);
  ptr := PPointer(addr);
  addr := cardinal(ptr^);
  writeln('addr: ',addr);
end;

将编译器生成的任何合法的PTypeInfo传递给此例程,并且它将输出相同的地址两次.我在TypInfo.pas中稍微捅了一下,但是我没有看到任何提到这个“身份指针”的东西或它的用途.有谁知道为什么会这样?对于从至少D3到D2010的每个版本的Delphi,这似乎都是正确的.

解决方法

它非常简单:包和动态链接.

BPL是DLL. DLL通过修补的表进行链接,而不是EXE或DLL中的所有代码链接到正在修补的DLL(这会对在多个进程之间共享只读内存造成很大的危害).为了防止在代码中的某处或者EXE或DLL的typeinfo中引用TypeInfo(SomeType),在链接到BPL时进行修改,而是通过导入表进行间接寻址.

在此程序中,静态链接与BPL链接时很容易看出差异:

{$apptype console}
uses TypInfo,SysUtils;
type
  TFoo = class(TObject);
var
  x: PPTypeInfo;
begin
  x := GetTypeData(TypeInfo(TFoo))^.ParentInfo;
  Writeln(x^^.Name);
  Writeln(Format('x  %p',[x]));
  Writeln(Format('x^ %p',[x^]));
end.

在我的本地机器上,使用dcc32 test.pas编译,它输出

TObject
x  00401B64
x^ 00401B68

但是当使用带有dcc32 -LUrtl test.pas的RTL包编译时,它会输出

TObject
x  004051F0
x^ 40001DA4

希望这可以解决它.

相关文章

 从网上看到《Delphi API HOOK完全说明》这篇文章,基本上都...
  从网上看到《Delphi API HOOK完全说明》这篇文章,基本上...
ffmpeg 是一套强大的开源的多媒体库 一般都是用 c/c+&#x...
32位CPU所含有的寄存器有:4个数据寄存器(EAX、EBX、ECX和ED...
1 mov dst, src dst是目的操作数,src是源操作数,指令实现的...