data segment
var1 db 82h, 05h, 23h, 11h, 08h, 27h, 02h, 0c2h ;变量var1保存长度为8个字节的有符号数1
var2 db 8ch, 4ch, 0c2h, 0abh,7ah, 35h, 0a5h, 08h ;变量var2保存长度为8个字节的有符号数2
len equ $-var2 ;len计算每个加数所占字节数,len不占用内存
sum db len dup(0) ;用于保存运算结果
data ends
stack segment
dw 128 dup(0)
ends
code segment
assume cs:code,ds:data,ss:stack
start:
; set segment registers:
mov ax, data
mov ds, ax
mov cx,len
mov si,0
clc ;将cf置0
lop:
mov al,sum[si]
push ax
mov al,var1[si]
push ax
mov al,var2[si]
push ax
call addlong ;调用相加函数
add si,1
loop lop
mov ax, 4c00h ; exit to operating system.
int 21h
addlong proc
push ax
push cx
mov bp,sp
mov al,[bp+6] ;var1 必须放在低八位
mov bl,[bp+8] ;var2
adc al,bl
mov sum[si],al
pop cx
pop ax
ret 6
addlong endp
ends
end start ; set entry point and stop the assembler.
一、实验目的
掌握汇编语言中的子程序设计方法、子程序参数传递的不同方式以及子程序中的现场保护。
二、实验内容
试设计一汇编程序,完成两个长补码的加法(有符号数加法),其中两个长补码的长度是相同的,并且采用低地址存放低位,高地址存放高位的方式存储。对于加法功能,使用子程序实现,两个长补码的起始地址、长补码的长度(以字节为单位)、运算结果的起始地址均由主程序以入口参数的形式传递给子程序。注意,参数传递方式采用堆栈传递方式。
以下是定义好的数据段和堆栈段:
data segment
var1 db 82h, 05h, 23h, 11h, 08h, 27h, 02h, 0c2h ;变量var1保存长度为8个字节的有符号数1
var2 db 8ch, 4ch, 0c2h, 0abh,7ah, 35h, 0a5h, 08h ;变量var2保存长度为8个字节的有符号数2
len equ $-var2 ;len计算每个加数所占字节数,len不占用内存
sum db len dup(0) ;用于保存运算结果
data ends
stack1 segment stack ;子程序设计必须定义堆栈段
dw 40 dup(0)
stack1 ends
注:可考虑使用循环结构来实现以上长补码的加法,每次循环中数1和数2的一个字节相加,且使用adc指令来实现加法的运算。(提示:adc指令是带进位的加法指令,它需要利用CF标志位上记录的进位值。第一次使用adc指令之前,需要用clc指令将CF位置0
1.在第一次写这个程序的时候的想法是只调用一次子程序,所以把每个的偏移量传了进去,但是不能成功,继续偏移的话是继续在栈中偏移。
所以把寻址放在了外面 在循环内部进行子程序的调用。
2.在栈内寻址的时候计算错误,所以导致cx中存放的值及其大,导致分配的栈的空间不够,出现unknown opcode skipped 这个错误
3.在之后改成了现在这个程序,在向栈中传值的时候需要传的是16位,但是在向往寄存器ax存放的时候只能放8位,只能放在低8位,不能放在高8位。否则低八位是被占用了的。
自己学到了什么:这么一问感觉很惆怅啊,似乎什么也没有学到,对于这些东西搞得不清楚。