微机原理与接口技术软件实验一、实验目的1.掌握汇编程序的编辑,编译,连接和执行的全过程;2.学习和掌握用DEBUG调试程序的方法.二、源程序DAT SEGMENTA DB 4 ;(自定)B DB 5 ;(自定)Y DB 3 DUP(0)Z DB 0,0DAT ENDSSTA SEGMENT STACKDW 50 DUP(?)STA ENDSCOD SEGMENTASSUME CS:COD,DS:DATSTAR PROC FARPUSH DSXOR AX,AXPUSH AXMOV AX,DATMOV DS,AXMOV AX,STAMOV SS,AXMOV AL,AMOV Z,ALMOV Z+1,ALCALL SUB1 ;A*AMOV AL,BMOV Z,ALMOV Z+1,ALCALL SUB1 ;B*BMOV AL,AMOV Z,ALMOV AL,BMOV Z+1,AL ;Z stores the two OPRD temporallyCALL SUB1 ;A*BADD WORD PTR Y,AXADC BYTE PTR[Y+2],0 ;A*B*2RETSTAR ENDPSUB1 PROCMOV AL,ZMOV AH,Z+1MUL AHADD WORD PTR Y,AX ;Y stores the resultADC BYTE PTR[Y+2],0 ;Y+2 stores the signal flagRETSUB1 ENDPCOD ENDSEND STAR三、实验文档中的要求(1)用D 命令观察数据区在内存中的具体内容,记录单元A 和B 的具体地址.(图一:反汇编结果以及数据区具体内容)由图一,可知A的地址为:141CH:0000H;B的地址为:141CH:0001H。
(2)用U 命令对目标代码反汇编,观察反汇编后的结果.注意发现源程序的起始位置, 并记录这个起始地址.由图一,可知:DATA段段基址:141CHA的偏移地址:0000HB的偏移地址:0001HY的偏移地址:0002HZ的偏移地址:0005HSTACK段段基址:141DHCODE段段基址:1424H第一条指令的偏移地址:0000H子程序SUB1入口的偏移地址:003FH(3)用T 命令作单步跟踪调试.比较每条指令执行后的结果和原来的理解是否一致.得出程序运行的结果:它们是写在什么单元,具体内容是什么; 并判断结果是否正确.现仅将关键步骤的单步调试结果列出,配合r命令查看结果。
(图二:单步调试结果一)这是第一次调用SUB1子程序退出后的结果,从图中可以看出Y存储的数据是+16(A已经改为4;B已经改为5),即是A*A的结果,符号位为0表示是正数。
(图三:单步调试结果二)这是第二次调用SUB1子程序退出后的结果,从图中可以看出执行完“MUL AH”指令后,AX 存储的数据是25,即是B*B的结果,子程序退出后Y存储的数据是+41,即是16+25的结果。
(图四:单步调试结果三)这是第三次调用SUB1子程序的结果,从图中可以看出执行完“MUL AH”指令后,AX存储的数据是20,即是A*B的结果,子程序退出后Y存储的数据是+61,即是41+20的结果。
(图五:单步调试结果四)这是程序退出时的最终结果,从图中可以看出,Y存储的数据最终为+81 ,即为(A+B)*(A+B)的结果。
结果正确。
(4)在子程序SUB1 的入口处设一断点,用G 命令执行程序. 在断点处观察堆栈的内容,比较堆栈的内容和程序返回地址是否一致.(图六:调用子程序的堆栈内容改变结果)从图中可以看出,IP指针的变化说明调用了子程序SUB1,而堆栈段段基址SS和栈顶指针SP的值都没有改变。
程序返回地址一致。
(5)用E 命令修改单元A,B 的内容,重新执行程序,并记录结果.(图七:修改A,B数值后的结果)从图中可见,分别将A,B的值改为1,1后,程序执行完毕,Y存储的内容是+4,结果正确。
(6)用M 命令和A 命令对程序进行修改: 将主程序中最后两条指令(ADD 和ADC) 修改为一条CALL SUB1 指令,重新执行程序.(图八:a命令练习使用的结果)反汇编发现程序最后的ADD指令在CODE段的偏移地址为0035,将此处改为调用SUB1子程序的指令,然后将0038处改为RETF。
重新执行后,查看Y的内容,结果正确。
(7) 重新使用编辑软件,把源程序最后一句中的STAR 去掉.再次生成可执行文件,并用DEBUG 调入内存.当使用U 命令时,显示的结果与前一次(未加STAR) 的结果有何不同?(图九:修改源程序后执行的结果)观察到两点不同(虽然不明了其原因),其一:不能立即进入程序,需执行许多冗余指令;其二:不能正确退出程序。
预习题:若SS=2000H,SP=FFFFH, 向堆栈中压入 4 字节数据后,如何用 D 命令显示压入堆栈的内容?使用debug指令“d 2000:0000”即可显示压入堆栈的内容,这是因为SP+1会溢出,变成0000。
四、心得体会汇编语言编程时要求程序员有严谨、缜密的思维,在某些程度上,我感觉使用汇编语言的编程过程就是把高级语言的结构过程拉平成线性过程来考虑问题。
经过此次试验,我基本掌握了汇编程序的编辑、编译、连接、执行以及调试的主要步骤。
应注意以下几点:在debug过程中,任何存储器单元都无法用标号来识别,要使用它们,只能使用物理地址调用。
使用u指令修改asm代码时要对照反汇编的结果,若要删去某条指令,可以找到那条指令在代码段的偏移地址,再用u指令将其修改为NOP(从该指令开始到下一条指令开始的地址-1处)。
可以结合r指令、t指令、已经用g指令设置断点来诊断问题。
微机原理与接口技术软件实验实验报告二姓名:杨帆学院:信息与通信工程学号:070547班级:07118实验二分支,循环程序设计一、实验目的:1.开始独立进行汇编语言程序设计;2.掌握基本分支,循环程序设计;3.掌握最简单的DOS 功能调用。
实验内容:1.安排一个数据区,内存有若干个正数,负数和零。
每类数的个数都不超过9。
2.编写一个程序统计数据区中正数,负数和零的个数。
3.将统计结果在屏幕上显示。
二、预习题:1.十进制数0 -- 9 所对应的ASCII 码是什么?如何将十进制数0 -- 9 在屏幕上显示出来?答:0的ASCII码是30h,以后依次递增。
要屏显0-9的数码,只需将AH置成02H(DOS功能调用),然后将要显示的数码的ASCII 码存进DL里,然后执行INT 21H就可以打印字符。
或者,若这些数码是以字符串的格式存储,则可以将AH置成09H,(最后以’$’字符结束,)然后将串首地址传给DS,然后执行INT 21H就可以打印字符串。
2.如何检验一个数为正,为负或为零?你能举出多少种不同的方法?答:可以将待检验数与0比较(使用CMP指令后判断CF、OF或者以表达式【实用EQ、GE 等操作符】的返回值来判断),也可以将待检验数与80H相与,判断ZF的值。
三、源程序DATA SEGMENTBUF0 DB 2BH,3AH ;'+',':'POSI DB 0BUF1 DB 0AH,0DH,30H,3AH ;LF,CR,'0',':'ZERO DB 0BUF2 DB 0AH,0DH,2DH,3AH ;LF,CR,'-',':'NEGA DB 0NUM DB 72H,88H,91H,01H,00H,00H,23H,77H,21H;you can add other numbersCNT EQU $-NUM ;LENTHDATA ENDSSTACK SEGMENT STACK 'STACK'DB 100 DUP (?)STACK ENDSCODE SEGMENTASSUME CS:CODE,DS:DATA,SS:STACKSTART PROC FARPUSH DSXOR AX,AXPUSH AXMOV AX,DATAMOV DS,AXMOV CX,0MOV BX,0LOOP1: MOV AL,NUM[BX]CALL FAR PTR CMPRINC BXINC CXCMP CX,CNT ;LOOP CNT TIMESJNZ LOOP1ADD POSI,30HADD ZERO,30HADD NEGA,30H ;TRANSFER TO ASC2 CODE MOV AH,02H ;TO PRINT CHARSMOV CX,0MOV BX,0LOOP2: MOV DL,BUF0[BX]INT 21HINC BXINC CXCMP CX,0DH ;LOOP 13 TIMESJNZ LOOP2MOV AX,4C00HINT 21HRETSTART ENDP CMPR PROC FAR CMP AL,0JGE BIGGERINC NEGAJMP EXIT BIGGER: JE EQUAL INC POSIJMP EXIT EQUAL: INC ZERO EXIT: RET CMPR ENDP CODE ENDS END START四、程序流程图(图表一:主程序流程图)(图表二:cmpr子程序流程图)五、实验结果与调试过程(图一:程序运行结果)分析:设置的数据区待分析数据为“72H,88H,91H,01H,00H,00H,23H,77H,21H”,其中正数为72H,01H,23H,77H,21H;负数为88H,91H;还有两个零。
可见程序得出的结果是正确的。
(图二:反汇编结果)分析:DATA段地址:1466;POSI偏移地址:0002;ZERO偏移地址:0007;NEGA偏移地址:000C;CODE段地址:1468;子程序CMPR偏移地址:0049;CNT的值+09D。
六、选作题:统计出正奇数,正偶数,负奇数,负偶数以及零的个数。
源代码如下:DATA SEGMENTBUF DB 2BH,4FH,3AH ;'+','O',':'POSOD DB 0DB 0AH,0DH,2BH,45H,3AH ;LF,CR,'+','E',':' POSEV DB 0DB 0AH,0DH,30H,3AH ;LF,CR,'0',':'ZERO DB 0DB 0AH,0DH,2DH,4FH,3AH ;LF,CR,'-','O',':' NEGAOD DB 0DB 0AH,0DH,2DH,45H,3AH ;LF,CR,'-','E',':' NEGAEV DB 0NUM DB 72H,88H,91H,01H,00H,00H,23H,77H,21H ;you can add other numbersCNT EQU $-NUM ;LENTHDATA ENDSSTACK SEGMENT STACK 'STACK'DB 100 DUP (?)STACK ENDSCODE SEGMENTASSUME CS:CODE,DS:DATA,SS:STACKSTART PROC FARPUSH DSXOR AX,AXPUSH AXMOV AX,DATAMOV DS,AXMOV CX,0MOV BX,0LOOP1: MOV AL,NUM[BX]CALL FAR PTR CMPRINC BXINC CXCMP CX,CNT ;LOOP CNT TIMESJNZ LOOP1ADD POSOD,30HADD POSEV,30HADD ZERO,30HADD NEGAOD,30HADD NEGAEV,30H ;TRANSFER TO ASC2 CODE MOV AH,02H ;TO PRINT CHARSMOV CX,0MOV BX,0LOOP2: MOV DL,BUF[BX]INT 21HINC BXINC CXCMP CX,1BH ;LOOP 26 TIMESJNZ LOOP2MOV AX,4C00HINT 21HRETSTART ENDPCMPR PROC FARCMP AL,0JGE BIGGERTEST AL,01HJNZ NODDINC NEGAEVJMP EXITNODD: INC NEGAODJMP EXITBIGGER: JE EQUALTEST AL,01HJNZ PODDINC POSEVJMP EXITPODD: INC POSODJMP EXITEQUAL: INC ZEROEXIT: RETCMPR ENDPCODE ENDSEND START运行结果:(图三:选做题程序运行结果)根据NUM中的数据,这个结果是正确的。