《汇编语言 基于x86处理器》第七章整数运算部分的代码
▶ 书中第七章的程序,使用各种位移运算,加深了对内存、寄存器中整数类型变量存储的认识
● 代码,双字数组右移 4 位
INCLUDE Irvine32.inc COUNT = ; 右移位数 .data
array DWORD 148B2165h, 8C943A29h, 6DFA4B86h, 91F76C04h, 8BAF9857h .code
main PROC
mov esi, OFFSET array ; 移之前的情况
mov ecx, LENGTHOF array
mov ebx, TYPE array
call DumpMem mov bl, COUNT
call ShiftDoublewords mov esi, OFFSET array ; 移之后的情况
mov ecx, LENGTHOF array
mov ebx, TYPE array
call DumpMem call WaitMsg
exit
main ENDP ShiftDoublewords PROC
mov esi, OFFSET array
mov ecx, (LENGTHOF array) - ; 循环次数等于数组长度减一,最后一个数单独处理 L1:
push ecx ; 需要使用 cl 移动数
mov cl, COUNT
mov eax, [esi + TYPE DWORD] ; 下一个数字 DWORD 移入 eax
shrd [esi], eax, cl ; eax 逐渐移入当前数字 push esi ; 移了 4 位,显示一下
push ecx
push ebx
mov esi, OFFSET array
mov ecx, LENGTHOF array
mov ebx, TYPE array
call DumpMem
pop ebx
pop ecx
pop esi add esi, TYPE DWORD ; 移完了,esi 指向下一个数字
pop ecx ; 恢复 ecx
loop L1 shr DWORD PTR [esi], COUNT ; 最后一个数,直接右移,偷偷使用 cf 中的值 ret
ShiftDoublewords ENDP END main
● 输出结果
Dump of offset // 移之前
-------------------------------
148B2165 8C943A29 6DFA4B86 91F76C04 8BAF9857 Dump of offset // 移第 1 个 DWORD,"9" 来自第二个数的最低 4 位
-------------------------------
9148B216 8C943A29 6DFA4B86 91F76C04 8BAF9857 Dump of offset // 移第 2 个 DWORD
-------------------------------
9148B216 68C943A2 6DFA4B86 91F76C04 8BAF9857 Dump of offset // 移第 3 个 DWORD
-------------------------------
9148B216 68C943A2 46DFA4B8 91F76C04 8BAF9857 Dump of offset // 移第 4 个 DWORD
-------------------------------
9148B216 68C943A2 46DFA4B8 791F76C0 8BAF9857 Dump of offset // 移第 5 个 DWORD,完成移动
-------------------------------
9148B216 68C943A2 46DFA4B8 791F76C0 08BAF985
● 代码,内存右移 1 位
INCLUDE Irvine32.inc .data
array BYTE 45h, 67h, 89h .code
main PROC
call DisplayArray mov esi,
mov eax, OFFSET array
shr array[esi+], ; 最高字节(第一个元素/字节)右移
rcr array[esi+], ; 中间元素循环右移,偷偷使用 CF
rcr array[esi], ; 最低字节循环右移 call DisplayArray call WaitMsg
exit
main ENDP DisplayArray PROC
pushad mov esi, OFFSET array ; 先用 DumpMem 显示内存情况(注意元素/字节是倒着存的)
mov ecx, LENGTHOF array
mov ebx, TYPE array
call DumpMem mov esi, LENGTHOF array ; esi 循环变量
dec esi
L1:
mov al, array[esi]
mov ebx,
call WriteBinB ; display binary bits
mov al, ' '
call WriteChar
dec esi
Loop L1 call Crlf
popad
ret
DisplayArray ENDP END main
● 输出结果
Dump of offset 00F66000 // 右移前,注意元素/字节是顺着存的
------------------------------- // 用 DumpMem 显示反而会翻转过来 Dump of offset 00F66000 // 右移后
-------------------------------
A2 B3
● 代码,二进制数转 ASCII 显示(逐字节打印),结果输出 00010010001101001010101111001101
INCLUDE Irvine32.inc .data
binVal DWORD 1234ABCDh ; 需要输出的数字
buffer BYTE dup(), .code
main PROC
mov eax, binVal
mov esi, OFFSET buffer
call BinToAsc mov edx,OFFSET buffer
call WriteString call Crlf
call WaitMsg
exit
main ENDP BinToAsc PROC uses ecx esi mov ecx, L1:
shl eax, ; eax 中的二进制串左移一位,最高位放入 cf 用于下面的跳转
; binVal 在内存中存成了 cd ab 34 12(4 字节整数模式下可以看到 1234abcd)
; 但对寄存器进行左移的时候还是按照 1234ABCDh 来左移
; 例如第一次左移后得到 2469579Ah
mov BYTE PTR [esi],'' ; 内存先写上 '0' 再说
jnc L2 ; 如果 cf 中是 1,把内存改写成 '1',否则跳过改写
mov BYTE PTR [esi],'' L2:
inc esi
loop L1 ret
BinToAsc ENDP END main
● 代码,用 adc 指令实现长整数加法
INCLUDE Irvine32.inc .data
op1 BYTE 34h,12h,98h,74h,06h,0A4h,0B2h,0A2h
op2 BYTE 02h,45h,23h,00h,00h, 87h,10h, 80h sum BYTE dup() ; = 0122C32B0674BB5736h .code
main PROC mov esi, OFFSET op1
mov edi, OFFSET op2
mov ebx, OFFSET sum
mov ecx, LENGTHOF op1 call Extended_Add mov esi, OFFSET sum
mov ecx, LENGTHOF sum add esi, ecx ; 找到数组结尾,从高位逐字节输出
sub esi, TYPE BYTE
mov ebx, TYPE BYTE L1:
mov al,[esi]
call WriteHexB
sub esi, TYPE BYTE
loop L1 call Crlf
call WaitMsg
exit
main ENDP Extended_Add PROC
pushad
clc ; 清除 CF mov edx,
L1:
mov al, [esi]
adc al, [edi]
pushfd ; CF(进位标志)压栈
mov [ebx], al
add esi, ; 更新指针
add edi,
add ebx,
popfd ; CF 出栈,准备进行下一次加法
loop L1 mov byte ptr [ebx], ; 最后一次加法,最高字节加上剩余的进位标志
adc byte ptr [ebx], popad
ret
Extended_Add ENDP END main
● ASCII 长整数加法,结果输出 1000525533291780
INCLUDE Irvine32.inc DECIMAL_OFFSET = ; 小数点位于位 5 的右边(没用到)
.data
decimal_one BYTE "" ; == 1001234567.89765
decimal_two BYTE "" ; == 9004020765.02015
sum BYTE (SIZEOF decimal_one + ) DUP(), ; 预留进位 .code
main PROC
mov esi, SIZEOF decimal_one - ; 从最低位开始
mov edi, SIZEOF decimal_one
mov ecx, SIZEOF decimal_one
mov bh, ; 旧进位标志,初始化为 0 L1:
mov ah,
mov al, decimal_one[esi]
add al, bh ; 加旧进位
aaa ; 调整和
mov bh, ah ; ah 存储了进位,将其放进 bh
or bh, 30h ; 转成 ASCII
add al, decimal_two[esi] ; 加一个数字,过程同上
aaa
or bh, ah ; 9 + 1 + 9 = 19 < 20,加了两次但最多进位 1
or bh, 30h ; 进位和结果都转成 ASCII
or al, 30h
mov sum[edi], al ; 当前位结果放入内存
dec esi ; 指向下一对计算的数字
dec edi
loop L1
mov sum[edi], bh ; 最后的进位放在预留的最高位上 mov edx, OFFSET sum
call WriteString
call Crlf
call WaitMsg
exit
main ENDP
END main
● 压缩十进制加法,输出结果 00011743。逻辑简单,但是不能理解 daa 的实现方法
INCLUDE Irvine32.inc .data
packed_1 WORD 4536h
packed_2 WORD 7207h
sum DWORD ? .code
main PROC
mov sum,
mov esi,
mov eax, mov al,BYTE PTR packed_1[esi] ; 低字节,daa 将 al 中的和转化为十进制和的低两位,放入内存
add al,BYTE PTR packed_2[esi]
daa
mov BYTE PTR sum[esi],al inc esi ; 中字节,包含进位
mov al,BYTE PTR packed_1[esi]
adc al,BYTE PTR packed_2[esi]
daa
mov BYTE PTR sum[esi],al inc esi ; 高字节,只考虑进位
mov al,
adc al,
mov BYTE PTR sum[esi],al mov eax,sum
call WriteHex
call Crlf
call WaitMsg
exit
main ENDP
END main
《汇编语言 基于x86处理器》第七章整数运算部分的代码的更多相关文章
- 《汇编语言 基于x86处理器》第九章字符串与数组部分的代码
▶ 书中第九章的程序,主要讲了字符串相关的输入.输出,以及冒泡排序.二分搜索 ● 代码,Irvine32 中的字符串库函数代码范例 INCLUDE Irvine32.inc .data str1 BY ...
- 《汇编语言 基于x86处理器》第八章高级过程部分的代码 - 两种规范计算数组元素的和
▶ 输入 Count 个 32 位有符号整数,计算他们的和 ● 代码,使用堆栈传参 ;// Sum_main.asm,主过程 INCLUDE Irvine32.inc EXTERN PromptFor ...
- 《汇编语言 基于x86处理器》第十三章高级语言接口部分的代码 part 2
▶ 书中第十三章的程序,主要讲了汇编语言和 C/++ 相互调用的方法 ● 代码,汇编中调用 C++ 函数 ; subr.asm INCLUDE Irvine32.inc askForInteger P ...
- 《汇编语言 基于x86处理器》前五章的小程序
▶ 书中前五章的几个小程序,基本的运算操作,使用了作者的库 Irvine32 和 Irvine64(一开始以为作者网站过期了,各网站上找到的文件大小都不一样,最后发现是要搭梯子 Orz,顺利下载).注 ...
- 《汇编语言 基于x86处理器》第十一章 MS-DOS 编程部分的代码 part 2
▶ 书中第十一章的程序,主要讲了 Windows 接口,在小黑框中进行程序交互 ● 在屏幕指定位置输出带自定义属性的文字 INCLUDE Irvine32.inc .data outHandle HA ...
- 《汇编语言 基于x86处理器》第十二章浮点数部分的代码
▶ 书中第十二章的程序,主要讲了 FPU 的指令和浮点数计算的过程 ● 代码,简单的 32 为浮点数测试 INCLUDE Irvine32.inc INCLUDE macros.inc .data f ...
- 《汇编语言 基于x86处理器》第六章条件处理部分的代码
▶ 书中第六章的程序,使用了条件判断和跳转来实现一些功能 ● 代码,查找数组首个非零值 INCLUDE Irvine32.inc .data intArray SWORD , , , , , , , ...
- 《汇编语言 基于x86处理器》第十章 - 运行一个 16位实地址汇编程序
▶ 书上第 10 章,主要讲了宏,引用了一个 16 位实地址的程序,从代码开始到运行 ● 代码 ; main.asm INCLUDE Macros.inc IF IsDefined( RealMode ...
- 《汇编语言 基于x86处理器》第十章结构和宏部分的代码
▶ 书中第十章的程序,主要讲了结构与宏的使用 ● 代码,使用结构,对比是否对齐的性能差距 INCLUDE Irvine32.inc INCLUDE macros.inc structN STRUCT ...
随机推荐
- 虚拟机中安装centOS及破解nuke的方法
跟the foundry的工程师邮件交流,我的一个Nuke脚本在对方机器上执行有问题,确认对方是centOS之后我决定在自己机器上安装一个centOS来找一下问题所在.安装重点如下: 一: 一定要下载 ...
- 解决 VUE 微信登录验证 【感谢原文:https://segmentfault.com/a/1190000009493199】
[感谢原文:https://segmentfault.com/a/1190000009493199] SPA单页应用中微信授权登录的一点思路 单页应用应该如何解决微信授权登录的尴尬跳转?后退无法返回? ...
- 两招解决异常_Cannot find any information on property 'XXX' in a bean of type 'XXX'的问题
第一招 在进行Java Web项目开发的时候,我碰到过下面这个异常: Cannot find any information on property 'XXX' in a bean of type ' ...
- 4:WPF中查看PDF文件
引用连接:https://www.cnblogs.com/yang-fei/p/4885570.html 在Github上看到一个非常好的WPF中承载PDF文件的类库. https://github. ...
- Reg2Bat_By Slore(生成同名bat文件,支持XP WIN7 WIN7X64).vbs
原文http://slore.blogbus.com/logs/52627038.htmlSlore编写的这个reg文件转换为bat文件,是逐句转换的,不是通过批处理生成临时reg文件然后导入的方法, ...
- 享元(FlyWeight)模式
享元模式(Flyweight Pattern)主要用于减少创建对象的数量,以减少内存占用和提高性能.这种类型的设计模式属于结构型模式,它提供了减少对象数量从而改善应用所需的对象结构的方式.享元模式尝试 ...
- [转]一图读懂JVM架构解析
每个Java开发人员都知道字节码经由JRE(Java运行时环境)执行.但他们或许不知道JRE其实是由Java虚拟机(JVM)实现,JVM分析字节码,解释并执行它.作为开发人员,了解JVM的架构是非常重 ...
- 关于在Intellij IDEA工具中配置热加载问题
第一步,创建一个maven项目,然后在pom.xml文件中添加依赖(上图内容). 第二步:来到intellij idea主页面,点击File->Settings->Build->co ...
- sso CAS
sso:single sign on,在多个应用系统中,用户只需要登陆一次就可以访问所有相互信任的应用系统 CAS框架:Central Authentication Service是实现sso单点登录 ...
- 用微信小程序连接WordPress网站
随着微信小程序的功能越来越强,特别对个人开发者的开放,让个人开发者有机会尝试微信小程序.如果你有自己的个人网站,就可以把个人网站搬到微信小程序里,通过小程序直接访问网站的内容. 要想微信小程序可以获取 ...