【问题标题】:Why this code generates an exception?为什么这段代码会产生异常?
【发布时间】:2011-06-29 09:55:51
【问题描述】:

我今天写了一些代码,它将列出 PE 文件中的所有部分...该代码有效,但最后它给出了一个异常:无效的指针操作...我不知道为什么...可以有人请找出错误

这里是代码

procedure TForm1.Button1Click(Sender: TObject);
var
  IDH:PImageDosHeader;
  buf:Pointer;
  INH:PImageNtHeaders;
  ISH:array of TImageSectionHeader;
  FS:TFileStream;
  i,total:Word;
begin
  if OpenDialog1.Execute then
    begin
        Self.Caption:=OpenDialog1.FileName;
        FS:=TFileStream.Create(OpenDialog1.FileName,fmOpenRead or fmShareDenyNone);
        GetMem(buf,FS.Size);
        FS.Read(buf^,FS.Size);
        FS.Free;
        IDH:=buf;
        INH:=Ptr(Cardinal(buf)+Cardinal(IDH^._lfanew));
        ISH:=Ptr(Cardinal(buf)+Cardinal(IDH^._lfanew) + Sizeof(TImageNtHeaders));
        total:=INH^.FileHeader.NumberOfSections - 1 ;
        for i:=0 to total  do
        begin
              ListBox1.Items.Add(PAnsichar(@ISH[i].Name));
              Application.ProcessMessages;
        end;

    end;
end;

【问题讨论】:

  • @CodeInChaos 这不是问题,如果你发现你对指针一无所知的问题
  • 我只是混淆了变量名,它们太相似了。 ISH 的分配对我来说看起来很奇怪。我认为ISH 应该是指向普通数组而不是动态数组的指针。而且令我惊讶的是,将无类型指针分配给动态数组可以在没有显式转换的情况下编译。
  • @CodeInChaos,我回滚了标题编辑,因为我不认为这个问题是专门由解析 PE 标头引起的;即使是那些我无法发现实际错误的人。
  • @CodeInChaos 代码工作的问题是循环结束时会产生异常

标签: delphi pointers dynamic-arrays


【解决方案1】:
ISH:array of TImageSectionHeader;

这声明了一个动态数组。虽然动态数组是指针,但它们需要在它们指向的数据前面添加额外的数据,包括长度和引用计数。

因此让它指向 PE 头中的一些数据是没有意义的:

ISH:=Ptr(Cardinal(buf)+Cardinal(IDH^._lfanew) + Sizeof(TImageNtHeaders));

虽然这部分似乎出于某种原因编译,但访问数组可能会出现错误:

ISH[i]//This might not work correctly since ISH does not point to a valid dynamic array.

或者,如果代码在该部分仍然存在(也许您禁用了数组边界检查或长度信息恰好足够大),那么一旦数组超出范围,delphi 就会尝试减少引用计数并可能释放数组。并且该部分访问数组指向的数据前面的引用计数信息,这在您的情况下无效。

如果我没记错的话,动态数组的内存布局是这样的:

--------------------------
|refcount|length|data....|
--------------------------
                ^ pointer goes here

这意味着您会遇到问题,因为 refcount/length 字段包含垃圾。


我想你想把它声明为:

type TImageSectionHeaderArray=array[0..70000]TImageSectionHeader;//No idea what the limit on section headers is
     PImageSectionHeaderArray=^TImageSectionHeaderArray;
...
var ISH:PImageSectionHeaderArray;

(我的delphi有点生疏,可能有一些小的语法错误)

【讨论】:

  • +1,是否生锈的delphi,就是这样;这就是代码完成后“弹出”错误的原因:它在 Delphi 尝试完成动态数组时弹出。
  • 好朋友不知道动态数组还有额外的数据值得了解!
  • 动态数组是非常通用的字符串。
  • 我认为编译器至少应该在这里给出警告。 (ISH := Ptr(...))。
猜你喜欢
  • 1970-01-01
  • 2016-07-23
  • 1970-01-01
  • 2016-02-08
  • 1970-01-01
  • 2021-12-10
  • 2011-01-17
  • 1970-01-01
相关资源
最近更新 更多