我已经在Linux上完成了一些汇编编程,现在我正尝试在Windows上使用MASM进行编程。我遇到了几个问题。
(这里我试图实现strlen()函数。我知道函数逻辑/指令并不是最优的,但我只是想把一些东西弄脏,这样我才能继续实现其他C库函数。)
.386
.model flat, stdcall
option casemap:none
include \masm32\include\kernel32.inc
includelib \masm32\lib\kernel32.lib
include \masm32\include\masm32.inc
includelib \masm32\lib\masm32.lib
.data
testString db "test string", 0 ; 0 -> terminator
.code
my_strlen proc
mov ebp, esp ; function prologue
push esp
push ebx
push ecx
xor ecx, ecx ; set count to 0
mov bl, byte ptr [ebp + 8] ; set low of b to 1st character
repeat:
cmp bl, 0 ; if null terminator, return
jz done
inc ecx ; increase count
mov bl, byte ptr [ebp + 8 + ecx] ; increase *ebx
jmp repeat ; repeat
done:
mov eax, ecx ; return count
pop ecx ; function epilogue
pop ebx
pop esp
ret
my_strlen endp
main:
push offset testString ; result = my_strlen(testString)
call my_strlen
push eax ; StdOut(result)
call StdOut
push 0 ; ExitProcess(0)
call ExitProcess
end main
当我尝试编译时,它看起来不像我的jmp标签,抛出无与伦比的宏嵌套等等。这是正确的方法吗?附:我试图避免在可能的情况下使用MASM宏,我更喜欢自己编写指令。
有人可以让这个程序编译吗?一旦我看到它是如何正确完成的,我将以愉快的方式,我将能够。是的,我寻找资源,但仍然是这个问题。
答案 0 :(得分:2)
有人可以让这个程序编译吗?
Voilà(评论中的解释):
.386
.model flat, stdcall
option casemap:none
include \masm32\include\kernel32.inc
includelib \masm32\lib\kernel32.lib
include \masm32\include\masm32.inc
includelib \masm32\lib\masm32.lib
.data
testString db "test string", 0
.code
my_strlen proc
; mov ebp, esp ; function prologue
; push esp
push ebp ; this is a correct prologue
mov ebp, esp
push ebx
push ecx
push esi
mov esi, [ebp+8] ; in [EBP+8] is a pointer
xor ecx, ecx
mov bl, byte ptr [esi]
repea: ; "repeat" is a keyword
cmp bl, 0
jz done
inc ecx
mov bl, byte ptr [esi + ecx] ; increase *esi
jmp repea ; repeat
done:
mov eax, ecx
pop esi
pop ecx
pop ebx
leave ; this is a correct epilogue
ret
my_strlen endp
main proc ; this is better
push offset testString
call my_strlen
; push eax ; StdOut(result)
push offset testString ; The MASM32-function StdOut needs only an ASCIZ-string
call StdOut
push 0
call ExitProcess
main endp
end main
我怀疑你想输出my_strlen的结果。 StdOut
无法做到这一点,因为StdOut
需要指向字符串的指针。您必须创建一个将EAX
转换为字符串的函数。
答案 1 :(得分:0)
这是一个包含宏的解决方案。字符串的长度最终存储在EAX中。
第一个宏从内存中检查整个DWORD,并且比逐字节读取和检查更快。 适合较大的琴弦。
;//-----------------------------------------------------------------------
strlenFast MACRO stringPtr:REQ
;//-----------------------------------------------------------------------
local STRLEN_LOOP
mov eax, stringPtr
mov esi, 0
STRLEN_LOOP:
mov ebx, [eax+esi]
mov ecx, ebx
inc esi
and ebx, 0FFh
jz EOS
mov ebx, ecx
inc esi
and ebx, 0FF00h
jz EOS
mov ebx, ecx
inc esi
and ebx, 0FF0000h
jz EOS
mov ebx, ecx
inc esi
and ebx, 0FF000000h
jnz short STRLEN_LOOP
EOS:
dec esi
mov eax, esi
ENDM
带字节检查的简短解决方案:
;//-----------------------------------------------------------------------
strlenShort MACRO stringPtr:REQ
;//-----------------------------------------------------------------------
local STRLEN_LOOP
mov eax, stringPtr
mov esi, -1
STRLEN_LOOP:
inc esi
mov ebx, [eax+esi]
and ebx, 0FFh
jnz STRLEN_LOOP
mov eax, esi
ENDM
在主要过程中:
.data
anyString byte "Hello World", 0
.code
main PROC
strlenShort OFFSET anyString ;// => result in EAX
strlenFast OFFSET anyString ;// => result in EAX