这个x86汇编程序是如何工作的

时间:2016-01-20 21:12:15

标签: assembly x86 tasm

这是一个x86 asm程序,它在屏幕中间从000到999打印一个“计数器”,工作正常(我没有写)。问题是我试图了解它是如何工作的。使用DIV和STACK在 THIS 程序中的用途。

该计划将如何知道如何去从009到010.什么是内循环?

我知道它如何清除屏幕并打印到屏幕中间,我知道时钟延迟功能。

DIVPUSHPOP在此计划中的用途是什么?

基本上我正试图追踪它并理解所使用的功能。我还是个初学者,请帮忙。谢谢。
这是代码:

.MODEL SMALL
.STACK 64
.DATA
CNT DW 0h
.CODE
MOV AX,@data
MOV DS,AX

MOV AX, 0600H
MOV BH,07
MOV CX, 0000
MOV DH,12
MOV DL,39       

INT 10H

MOV AX,0600H
MOV BH,07 
MOV CX,0000 
MOV DX,184FH
INT 10H         

MOV CNT,0

L1: MOV AH,02h
MOV BH,0
MOV DH,12
MOV DL,39
INT 10h         

MOV AX,CNT
MOV BL,10
DIV BL              
PUSH AX

MOV AH,00
MOV BL,10
DIV BL                
PUSH AX               
MOV DL,AL
ADD DL,48
MOV AH,02h
INT 21h

POP AX
MOV DL,AH
ADD DL,48       
MOV AH,02h
INT 21h

POP AX
MOV DL,AH
ADD DL,48
MOV AH,02h      
INT 21h             

MOV CX, 0007h
MOV DX, 2120h 
MOV AH, 86H         
INT 15H             

INC CNT
CMP CNT,999
JBE L1  

MOV AH,4ch          
INT 21h
END

2 个答案:

答案 0 :(得分:2)

  

该计划将如何知道如何去从009到010,什么是内循环?

内部循环由L1标签到JBE L1指令形成。内存CNT保存正在显示的计数器。它从可执行文件初始化为零,并且递增(INC CNT),与999相比,并且当它低于或等于时循环。

  

DIV,PUSH和POP在此计划中的用途是什么?

目的是将CNT值拆分为单独的数字并显示它们。 DIV BLAX除以BL,并将商数存储在AL中,将余数存储在AH中。

或许对这是做什么的最佳解释是通过一个例子。假设,CNT的值为123.要计算每个可打印的数字,我们将除以10并得到12的商,余数为3.现在,3是我们需要打印的最后一个所以我们需要存储在堆栈中。然后我们将重复此过程获得2然后1.现在,我们可以打印出1,弹出堆栈中的2并打印出来然后弹出3并打印出来,这样我们就打印出来了。

代码段

MOV DL,AH
ADD DL,48       
MOV AH,02h
INT 21h

负责打印每个数字。它首先获取存储在AH中的余数,加上48(030h)得到ASCII数字('0'= 30h ..'9'= 39h)然后调用函数02h / int21打印单个屏幕上的角色。

答案 1 :(得分:0)

该指令:

.STACK 64               // reserve a 64 bytes "stack space" for temp data

这条指令:

DIV BL                  // divide the contents of register ax by register bl

因为它们除以10,我认为这是将每个数字分开来显示它。

循环在这里:

INC CNT                 // increment (add 1 to) variable CNT (in .DATA)
CMP CNT,999             // compare CNT and 999
JBE L1                  // Jump Below or Equal (to label 'L1')
顺便说一句,您没有问过的一点代码是:

MOV CX, 0007h           // high bytes of wait time \__ microseconds
MOV DX, 2120h           // low bytes of wait time  /   (this is ~0.47 secs)
MOV AH, 86H             //
INT 15H                 // INT 15H/86H is wait ("sleep")