我正在尝试在 Delphi 中学习内联汇编编程,为此我发现这篇文章非常有帮助。
现在我想写一个汇编函数返回一个长字符串,特别是一个AnsiString
(为简单起见)。我已经写了
function myfunc: AnsiString;
asm
// eax = @result
mov edx, 3
mov ecx, 1252
call System.@LStrSetLength
mov [eax + 0], ord('A')
mov [eax + 1], ord('B')
mov [eax + 2], ord('C')
end;
解释:
返回字符串的函数有一个不可见的var result: AnsiString
(在这种情况下)参数,因此,在函数的开头,eax
应该保存结果字符串的地址。然后我将edx
andecx
分别设置为 3 和 1252,然后调用System._LStrSetLength
. 实际上,我确实
_LStrSetLength(@result, 3, 1252)
其中 3 是字符串的新长度(字符 = 字节),1252 是标准windows-1252代码页。
然后,知道这eax
是字符串的第一个字符的地址,我简单地将字符串设置为“ABC”。但它不起作用 - 它给了我无意义的数据或 EAccessViolation。问题是什么?
更新
现在我们有两个看似有效的实现myfunc
,一个正在使用NewAnsiString
,一个正在使用LStrSetLength
。我不禁想知道它们是否都正确,因为它们不会破坏 Delphi 对字符串的内部处理(引用计数、自动释放等)。