汇编语言Irivne64字符串过程详解[附带实例]
下面将一些比较重要的字符串处理过程从 Irvine32 链接库转换为 64 位模式。变化非常简单,删除堆栈参数,并将所有的 32 位寄存器都替换为 64 位寄存器。
下表列出了这些字符串过程、过程说明及其输入输出。
Str_compare |
比较两个字符串 输入参数:RSI 为源串指针,RDI 为目的串指针 返回值:若源串 < 目的串,则进位标志位 CF=1;若源串 = 目的串,则零标志位 ZF=1;若源串 > 目的串,则 CF=0 且 ZF=0 |
Str_copy |
将源串复制到目的指针指向的位置 输入参数:RSI 为源串指针,RDI 指向被复制串将要存储的位置 |
Str_length |
返回空字节结束字符串的长度 输入参数:RCX 为字符串指针 返回值:RAX 为该字符串的长度 |
Str_compare 过程中,RSI 和 RDI 是输入参数的合理选择,因为字符串比较循环会用到它们。使用这两个寄存器参数能在过程开始时避免将输入参数复制到 RSI 和 RDI 寄存器中:
;------------------------------------ ;Str_compare ;比较两个字符串 ;接收:RSI 为源串指针 ; RDT 为目的串指针 ;返回:若字符串相等,ZF 置 1 ; 若源串 < 目的串,CF 置 1 ;------------------------------------ Str_compare PROC USES rax rdx rsi rdi L1: mov al,[rsi] mov dl,[rdi] cmp al, 0 ; string1 结束? jne L2 ; 否 cmp dl, 0 ;是:string2 结束? jne L2 ;否 jmp L3 ;是:退出且 ZF=1 L2: inc rsi ;指向下一个字符 inc rdi cmp al,dl ;字符相等? je L1 ;是:继续循环 ;否:退出并设置标志位 L3: ret Str_compare ENDP
注意,PROC 伪指令用 USES 关键字列出了所有需要在过程开始时入栈、在过程时返回出栈的寄存器。
Str_copy 过程用 RSI 和 RDI 接收字符串指针:
;------------------------------------- ;Str_copy ;复制字符串 ;接收:RSI 为源串指针 ; RDI 为目的串指针 ;返回:无 ;------------------------------------- Str_copy PROC USES rax rex rsi rdi mov rex,rsi ;获得源串长度 call Str_length ;RAX 返回长度 mov rex,rax ;循环计数器 inc rex ;有空字节,加 1 cld ;方向为正向 rep movsb ;复制字符串 ret Str_copy ENDP
Str_length 过程用 RCX 接收字符串指针,然后循环扫描该字符串直到发现空字节。字符串长度用 RAX 返回:
;------------------------------------- ;Str_length ;计算辜符串长度 ;接收:RCX 指向字符串 ;返回:RAX 为字符串长度 ;------------------------------------- Str_length PROC USES rdi mov rdi,rex ;获得指针 mov eax,0 ;字符计数 L1: cmp BYTE PTR [rdi],0 ;字符串结束? je L2 ;是:退出 inc rdi ;否:指向下一个字符 inc rax ;计数器加 1 jmp L1 L2: ret ;RAX 返回计数值 Str_length ENDP
一个简单的测试程序
下面的测试程序调用了 64 位的 Str_length、Str_copy 和Str_compare 过程。虽然程序中没有显示字符串的语句,但是建议在 Visual Studio 凋试器中运行,这样就可以查看内存窗口、寄存器和标志位。
; 测试 Irvine64 字符串程序 Str_compare proto Str_length proto Str_copy proto ExitProcess proto .data source byte "AABCDEFGAABCDFG",0 ; 大小为 15 target byte 20 dup(0) .code main proc mov rax,offset source call Str_length ; 用 RAX 返回长度 mov rsi,offset source mov rdi,offset target call str_copy ; 由于刚刚才复制了字符串,因此它们应该相等 call str_compare ; ZF = 1, 字符串相等 ; 修改目的串的第一个字符,再比较两个字符串 ; compare them again: mov target,'B' call str_compare ; CF = 1, 源串 < 目的串 mov ecx,0 call ExitProcess main ENDP
发表评论