3

ecxの値が0になった後にプログラムを終了させたいのに、なぜプログラムが無限ループに入るのか理解できませんか?助けてください?

section .data
;get external functions
extern printf
global main
main:

;set up stack frame
push rbp
mov rbp, rsp

;if(x<y)
;print x is less
;else
;print y is larger than x

;mov values into register to compare them
mov rax,[x]
mov rbx,[y]
cmp rax,rbx ;cmp x,y
jg .x_is_greater
lea rdi,[y_less]
xor eax,eax ;must clear eax when using printf
call printf
jmp .done

.x_is_greater:
;print "X is greater to the screen"

;mov r11,[count]
;lea rdi,[x_greater]
;xor eax,eax
;call printf
;mov r12,[zero]
;cmp r11,r12
;jg .myloop ;jump to myloop if greater than zero
;jmp .done ;return if equal to 0
mov ecx, 3; [count]
;mov r12, [zero]
jmp .myloop
.myloop:
;;dec r11
;dec rcx
lea rdi,[fmt]
lea rsi,[ecx]   
;mov rdx,[r12]
xor eax,eax ;must clear eax when using printf
call printf

cmp ecx, 0
jz .done
lea rdi,[x_greater]
xor eax,eax ;must clear eax when using printf
call printf
lea rdi,[fmt]
lea rsi,[ecx]   
;mov rdx,[r12]
xor eax,eax ;must clear eax when using printf
call printf
dec ecx

;sub rcx,[one]
jmp .myloop
;jmp .done
.done:
leave
;xor eax, eax
ret;exit program

;leave ;destroy stack frame

section .bss

section .data
prompt db "This is a practice program to test what I know!",0x0a,0
y_less db "Y < X",0x0a,0
x_greater db "X > Y ",0x0a,0
x db 10
y db 1
count dq 3
zero db 0
one dq 1
fmt db "R11 %d ",0x0a,0
4

3 に答える 3

5

関数(たとえば)を呼び出すときはprintf、次の値を保持する必要があります。ecx

http://www.x86-64.org/documentation/abi.pdf

レジスタ%rbp、%rbx、および%r12から%r15は、呼び出し元の関数に「属し」、呼び出された関数はそれらの値を保持する必要があります。つまり、呼び出された関数は、呼び出し元のためにこれらのレジスタの値を保持する必要があります。残りのレジスタは、呼び出された関数に「属します」。呼び出し元の関数が関数呼び出し全体でそのようなレジスタ値を保持したい場合は、その値をローカルスタックフレームに保存する必要があります。

于 2012-10-27T05:18:39.080 に答える
1

呼び出しにより、レジスタのprintf値が変更される場合があります。ecxしたがって、次のいずれかを実行する必要があります。

  • printf呼び出しの前とpopprintfの呼び出しの後にスタックにプッシュします。
  • 呼び出し先が保存したレジスタをループカウンターとして使用します。また
  • 呼び出し先が保存したレジスタに保存し、復元します。

最初のオプションの例:

.myloop:
lea rdi,[fmt]
lea rsi,[ecx]
xor eax,eax ;must clear eax when using printf
push ecx ; saved
call printf
pop ecx ; restored

cmp ecx, 0
jz .done
lea rdi,[x_greater]

push ecx ; saved
xor eax,eax ;must clear eax when using printf
call printf
pop ecx ; restored

lea rdi,[fmt]
lea rsi,[ecx]

push ecx ; saved
xor eax,eax ;must clear eax when using printf
call printf
push ecx ; restored

dec ecx
jmp .myloop
于 2012-10-27T05:20:51.203 に答える
1

Printfを使用してWHILE-DOというループを作成し、Microsoft Visual StudioでRAX、RCX、RDX、R9、R8がCallPrintfで使用されていることを確認しました。そこで、EBXに番号を追加しました。ポップ、プッシュ、スタッキングなしで動作します:)。それが何人かの人々に役立つことを願っています。

extrn ExitProcess: PROC
extrn printf: PROC

.data
fmt db '%s',10,0 ; The printf format, "\n", '0'
check db 'HALO',0

.code
Start PROC

xor eax,eax
mov rbx,1

start_while:
    cmp rbx,10
    jge end_while

    lea rcx,check
    lea rdx,fmt
    call printf


    add rbx,1
    jmp start_while
end_while:
xor eax,eax
xor rbx,rbx
call ExitProcess
Start ENDP
End
于 2016-06-02T23:23:16.103 に答える