- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我发现 Delphi 5 在特定情况下会生成无效的汇编代码。我不明白一般在什么情况下。由于发生了非常奇怪的优化,下面的示例会产生访问冲突。对于记录或数组中的字节,Delphi 生成push dword [...]、pop ebx、mov ..、bl,如果该字节后有数据,则可以正常工作(我们至少需要三个才能正确推送dword),但会失败如果数据无法访问。我使用 win32 Virtual* 函数模拟了严格的边界
具体来说,当在 FeedBytesToClass 过程内访问 block 的最后一个字节时,会发生错误。如果我尝试更改诸如使用数据数组而不是删除 actionFlag 变量的对象属性之类的内容,Delphi 会生成正确的汇编指令。
const
BlockSize = 4096;
type
TSomeClass = class
private
fBytes: PByteArray;
public
property Bytes: PByteArray read fBytes;
constructor Create;
destructor Destroy;override;
end;
constructor TSomeClass.Create;
begin
inherited Create;
GetMem(fBytes, BlockSize);
end;
destructor TSomeClass.Destroy;
begin
FreeMem(fBytes);
inherited;
end;
procedure FeedBytesToClass(SrcDataBytes: PByteArray; Count: integer);
var
j: integer;
Ofs: integer;
actionFlag: boolean;
AClass: TSomeClass;
begin
AClass:=TSomeClass.Create;
try
actionFlag:=true;
for j:=0 to Count-1 do
begin
Ofs:=j;
if actionFlag then
begin
AClass.Bytes[Ofs]:=SrcDataBytes[j];
end;
end;
finally
AClass.Free;
end;
end;
procedure TForm31.Button1Click(Sender: TObject);
var
SrcDataBytes: PByteArray;
begin
SrcDataBytes:=VirtualAlloc(Nil, BlockSize, MEM_COMMIT, PAGE_READWRITE);
try
if VirtualLock(SrcDataBytes, BlockSize) then
try
FeedBytesToClass(SrcDataBytes, BlockSize);
finally
VirtualUnLock(SrcDataBytes, BlockSize);
end;
finally
VirtualFree(SrcDataBytes, MEM_DECOMMIT, BlockSize);
end;
end;
最初,当我访问位图位的 RGB 数据时,发生了错误,但那里的代码太复杂,所以我将其范围缩小到这个片段。
所以问题是这里的具体是什么使得 Delphi 产生了 push、pop、mov 优化。我需要知道这一点才能避免此类副作用。
最佳答案
哎哟,确实是一个痛苦的问题。常量 actionFlag 的存在(与 4 的倍数的常量计数相结合)会触发处理数据的推送/弹出方式。对于那些对实际汇编程序感兴趣的人(在 cpu View 不提供复制/粘贴的日子里手动输入):
AClass.Bytes[Ofs] := SrcDataBytes[j];
mov exc,[ebp-$04]
push dword ptr [ecx+eax] <- ouch
mov ecx,[ebp-$08]
mov ecx,[exc+04]
lea esi,[exc+esi]
pop ecx
mov [esi],cl
end;
inc eax
它执行了 4096 次。我查了一下,Delphi 6 没有这种行为。我认为我们可以放心地假设它在任何更高版本中也已修复。
作为解决方法,我建议简单地将 {$O-}/{$O+} 添加到该方法中。我不会过多地深入研究确切的取证,因为触发 Delphi 执行这种错误优化的条件似乎很少见,而且 Delphi 版本确实相当古老。
常量标志通常不会成为内部循环的一部分,我怀疑您的计数通常也是动态的。然而,您在生产代码中遇到了这个问题,所以也许它并不像看起来那么罕见。我只能说,我从未遇到过,而且我们 90% 的生产代码都是在 Delphi 5 中编写的。也许这只是默认内存分配的安全性。
关于delphi - 我应该做什么或不做什么才能避免 Delphi "push dword"错误。,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1294696/
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 要求提供代码的问题必须表现出对所解决问题的最低限度理解。包括尝试过的解决方案、为什么它们不起作用,以及预
这个问题在这里已经有了答案: C++ cout hex values? (10 个答案) 关闭 6 年前。 如何在 C++ 中将十进制转换为十六进制?我在 ultoa 上取得了成功,但这给了你一个字
我想在 DWORD 数组中找到不等于 0 的最重要的 DWORD。该算法应该针对最大 128 字节的数据大小进行优化。 我做了三个不同的函数,它们都返回特定 DWORD 的索引。 unsigned l
编译C++项目时出现以下错误。 Error C2664 'BOOL CryptBinaryToStringW(const BYTE *,DWORD,DWORD,LPWSTR,DWORD *)': ca
我开始学习汇编程序,这对我来说似乎不合逻辑。 为什么我不能在一个寄存器中使用多个高字节? 我了解rax的历史原因-> eax -> ax ,所以让我们关注新的 64 位寄存器。例如,我可以使用 r8和
我遇到了一个我无法理解的编译错误。我有两个使用 WLT 8.0 的 C++ 项目。我正在使用 VS2005 来编译两者。一个在 Win XP 或 7 上运行,另一个在 CE 上运行。它们都包含 WLT
所以最近我一直想从汇编中调用一些 win32 调用,并且我一直在使用 NASM 作为我的外部汇编程序。我调用 SendMessage在我的代码中以下列方式: call __imp__SendMessa
从“DWORD (*)(void*)”到“DWORD (*)(void*)”的无效转换。蛋糕==谎言1==0 我不知道这是什么意思......我在这段代码中得到了它 HANDLE CPlugin::C
我很难理解我是否以正确的方式执行此操作以及这是否是(唯一)最佳解决方案。 我正在从事的项目正在使用 Three-Dimensional Array保存和使用大量数据。 “数据”的一部分是 DWORD
这个问题在这里已经有了答案: problems using CreateThread on a member function (1 个回答) 关闭 10 年前。 我试图在类中创建线程,但出现错误。
我在这个网站上搜索了一个答案,但最重要的答案对我不起作用,我一直收到这个错误。我最近(不确定我是否成功)将我的项目从我的台式机导入到我的新笔记本电脑,没有任何错误。每当我尝试运行时,我都会在多个文件中
我想连接 2 个非字符串类型,这样我就可以将它们用作一个。这是我的代码的主要部分: #include #include #include int main() { HANDLE hwnd =
对于这个问题,汇编语言中用一下方法处理。 (1)通过寄存器名指明要处理的数据的尺寸。 例如: 下面的指令中,寄存器指明了指令进行的是字操作: mov ax,1 mov bx,ds:[0] mov
你好,我正在尝试在 intel32 上安装程序集, 我的完整问题是,这是什么意思? mov dword ptr [esp+18h], 0AH 我特别没有得到 [esp+18h] 的部分 最佳答案 中括
如果使用汇编语言 mov eax, dword ptr[ebx] 那么这意味着复制ebx指向的值(ebx包含地址值,而不是实际值,此指令将实际值复制到地址中)? 如果我们使用 mov eax, dwo
这个问题已经有答案了: Getting hex through Cin (5 个回答) 已关闭 8 年前。 我需要从 unsigned long 进行转换(十六进制)到 DWORD . 我通常会 DW
关闭。这个问题需要details or clarity .它目前不接受答案。 想改进这个问题吗? 通过 editing this post 添加细节并澄清问题. 关闭 7 年前。 Improve t
我收到此错误且无法更正。任何帮助表示赞赏。谢谢。错误 C2440:“=”:无法从“DWORD *”转换为“unsigned int”IntelliSense:“DWORD *”类型的值不能分配给“un
我正在尝试将带符号的浮点变量转换为 DWORD...DWORD 将由另一个程序使用,因此 DWORD 变量类型很重要... 首先...可以将带符号的 DWORD 解释为未签名的 DWORD...吗?
在 C++ 中,我可以简单地将指针转换为 DWORD 吗? MyClass * thing; DWORD myPtr = (DWORD)thing; 那行得通吗? 最佳答案 毫无疑问,您可以做到。 它
我是一名优秀的程序员,十分优秀!