汇编学习笔记

server/2024/12/28 15:12:36/

汇编

1. debug指令

-R命令(register)

查看、改变CPU寄存器的内容

r ax 修改AX中的内容

-D命令(display)

查看内存中的内容

-E命令(enter)

改写内存中的内容

-U命令(unassenble反汇编)

将内存中的机器指令翻译成汇编指令

-T命令(trace跟踪)

执行一条机器指令

-A命令(assenble汇编)

汇编指令的格式在内存中写入一条机器指令

2. mov、add、sub指令

1.mov(传送)

格式:mov ax, bx

注意:

1. 目标操作数与源操作数不能同时为内存操作数
2. 不能直接将立即数传送给段寄存器
3. CX不能以一切形式传送
4. 两个操作数类型必须一致, 字节、字、双字
2.add

格式:add ax, bx

若两数相加结果超过寄存器位数,舍弃超出的高位

eg1溢出:cx = F06F

add cx, cx 原: F06F + F06F = 1E0DE

cx = E0DE 最高位的 1 被舍去

eg2高位: BX = F037

add bh, 37 原: F0 + 37 = 127

BX = 2737

eg3低位: CX = E0DE

add cl, B6 原:DE + B6 = 194

CX = E094 低位计算也不会进位

3.sub

格式: sub op1,op2 ;意为:op1=op1-op2

若两数相减,被减数小于减数,则被减数从最高位的上一位借1

eg1: BX = 1303

sub bx, F

BX = 12F4

eg2借位: BX = 000F

sub bx, 10 原:1000F - 10 = FFFF

BX = FFFF 从高位借1

eg3: BX = 0000

sub bl, E 原:100 - E = 00F2

BX = 00F2 从目标操作数的高位借1

3.mul、div、and、or指令

1. mul(乘法)

(1)两个相乘数:要么都是8位,要么都是16位。

  1. 若是8位,一个默认放 AL 中,另一个放在 8位reg内存单元

  2. 若是16位, 一个默认放在 AX 中, 另一个放在 16位reg内存单元

(2)结果:如果是8位乘法,结果默认放在 AX

如果是16位乘法,结果 高位 默认放在 DX , 低位 默认放在 AX

格式:mul reg

mul 内存单元

注意:溢出同加法

2. div(除法)

(1)除数:8位 或 16位,在一个reg 或内存单元中

(2)被除数:默认放在 AXDXAX

如果除数为8位,被除数则为16位,默认放在 AX

如果除数为16位,被除数则为32位, 放在在 DXAX 中,DX放高16位,AX放低16位

(3)结果: 如果除数为8位, 则商放在 AL,余数放在 AH

如果除数为16位,则商放在 AX,余数放在 DX

格式: div reg

div 内存单元

除数8位:AX / BL = AL...AH

除数16位:DXAX / BX = AX...DX

eg132位被除数:DX = F , AX = 4240, BX = 2710

mov dx, F F4240 = 1000000 2710 = 10000 64 = 100

mov ax, 4240 原:F4240 / 2710 = 64

mov bx, 2710

div bx

AX = 64 通过高16位DX, 低16位AX组成一个32位的被除数,商放在AX

eg2有余数: DX = F, AX = 4241, BX = 2710

mov dx, F F4241 = 1000001

mov ax, 4241 原:F4241 / 2710 = 64...1

mov bx, 2710 1000001 / 10000 = 100...1

div bx

AX = 64, DX = 0001

3. and(与)

逻辑与指令,按位进行与运算

mov al, 63

and al, 3B

结果:al = 23

注意:该指令可将操作对象的相应位设为0,其他位不变

将al的第0位设为0的指令为:and al, FE(11111110B)

4. or (或)

逻辑或运算,按位进行或运算

mov al, 63

or al, 3B

结果:al = 7B

注意:该指令可将操作对象的相应位设为1,其他位不变

将al的第0位设为1的指令为:or al, 1(00000001B)

4.shl、shr、inc、dec、xchg

1.shl (左移)

shift left

1.将寄存器或内存单元中的数据向左移位

2.将最后移出的一位写入CF(进位标志)中

3.最低位用0补充

格式:mov al, 48 ;(01001000)

shl al, 1 ;将al中的数据都左移一位

结果: al = 90(10010000), CF = 0

2.shr(右移)

shift right

同shl

3.inc(加一)

值加一,不影响CF

4.dec(减一)

5.nop(空)

空指令,先占位

6.xchg(交换)

交互两操作数的值

格式:xchg ax, bx

7.neg(求补)

运算法一:用零减去操作数,然后结果返回操作数

运算法二:将操作数按位取反后加1

8.退出

格式:mov ax, 4c00H

int 21H

安全退出程序

5.寄存器

AX:累加寄存器 OF:溢出标志(超过机器所能表示的有符号数范围)

BX:基址寄存器 ZF:零标志(运算结果为0,ZF=1)

CX:计数寄存器 CF:进位标志

DX:数据寄存器 AF:辅助进位标志(第3位向第4位进位时位1,否则为0)

SP:堆栈指针寄存器 PF:奇偶标志(计算结果为1个数为偶则为1,否则为0)

BP:基址指针寄存器 DF:方向标志(串处理,DF=1时,每次操作后SI和DI减小,DF=0时增大)

SI:源变址寄存器 SF:符号标志(运算结果为负时SF=1)

DI:目的变址寄存器 IF:中断标志(IF=1,允许CPU响应可屏蔽中断,否则关闭中断)

CS:代码段寄存器 TF:陷阱标志(用于调试单步操作)

DS:数据段寄存器

SS:堆栈段寄存器

ES:附加段寄存器

5.CS:ip代码段寄存器、jmp

CS:IP指示了CPU当前要读取指令的地址

CS 为代码段寄存器,IP 为指令指针寄存器

jmp

功能:用寄存器中的值修改IP

格式:jmp ax ;等同于mov IP, ax 等同但不可以这样用 修改IP

jmp ax:bx ;等同于mov CS, ax mov IP, bx 修改CS:IP

6.ss:sp寄存器、栈的push、pop指令

栈底是高地址

1.push

push指令的完成,需要以下两个步骤

  1. SP = SP - 2:因为SS:SP指向前栈顶,既然需要push 数据,自然需要获取新栈顶

  2. 将 push 后面跟的寄存器中的内容写入刚开辟出的内存中,此时的SS:SP指向新栈顶

入栈时,栈顶从高地址向低地址增长

2.pop

pop指令的完成,需要以下两个步骤

  1. 将栈顶元素送入pop后面跟的寄存器中

  2. SP = SP + 2,SS:SP指向下一个元素

出栈时,栈顶从低地址向高地址增长

ss:sp 指向栈顶元素

7.bp,si、di变址寄存器

1. SI 、DI是变址寄存器

SIDI 是功能与 bx相近的寄存器,不能分成8位寄存器来使用,但是可以直接作为偏移地址(有效地址)

[BX+SI][BX+DI] 可亦作为偏移地址,但是不可加ax、cx、dx这些寄存器

2. BP寄存器

BP默认的段寄存器是SS,bx默认的段寄存器是DS

8.cmp、adc、sbb、

1. adc(进位加法)

adc 是带进位加法指令,利用 CF 位上记录的进位值

格式:adc 操作数1, 操作数2

功能:操作数1 = 操作数1+操作数2 + CF

常用于32位、48位数的相加,可实现进位

2. sbb(借位减法)

sbb 是带借位减法指令,利用 CF 位上记录的借位值

格式:sbb 操作数1, 操作数2

功能:操作数1 = 操作数1- 操作数2 - CF

利用sbb指令可以对任意大的数据进行减法运算

3. cmp(比较)

cmp是比较指令,功能相当于减法指令,只是不保存结果,但是影响标志寄存器

格式:cmp 操作数1, 操作数2

功能:计算 (操作数1 - 操作数2) 但不保存结果,仅改变标志寄存器

9.源程序

assume cs:codesg
​
codesg segment
​mov ax, 0123Hadd bx, 0456Hadd ax,bxadd ax,axmov ax,4c00Hint 21H
codesg ends
end

10.Loop指令

格式:loop 标号

CPU执行loop指令的时候要进行两步操作:

  1. (cx) = (cx) - 1

  2. 判断 cx 中的值,不为零则转至标号处执行程序,如果为零则向下执行

用loop指令实现循环功能,cx中存放循环次数

assume cs:codesg
codesg segmentmov  ax, 2mov  cx, 11    ;循环11次,输入多少,就循环多少次s:     add  ax, axloop s
​int  21H
codesg ends
end

如果 cx = 0,loop指令会陷入死循环:

因为Loop指令会先减一,即cx = 0 - 1 = FFFF,需要在运行65535次

#include<stdio.h>
int x = 0;
int sum = 0;
for (int i = 1; i <= 100; i++) {x++;sum += x;
}
return sum;

上述C语言转换为汇编语言: ​

assume cs:codesg
codesg segmentmov  ax, 0mov  bx, 0mov  cx, 100s:     inc  axadd  bx, axloop s
​int  21H
codesg ends
end

11.ret、call指令

call ==> func()

ret ==> return

assume cs:codesg
codesg segmentmov  ax, 2mov  cx, 11call s         ;调用函数sint  21H       ;若不退出,会执行下一指令,陷入死循环s:     add  ax, axloop sret            ; 封装函数s,等于return
​
codesg ends
end
1. ret 和 retf

ret 指令用栈中的数据,修改 IP,实现近转移

执行后,进行操作:

(1) IP = SS * 16 + SP

(2) SP = SP + 2

等同于: pop IP

retf指令用栈中的数据,修改 CS 和 IP ,实现远转移

(1) IP = SS * 16 + SP

(2) SP = SP + 2

(3) CS = SS * 16 + SP

(4)SP = SP + 2

等同于: pop IP

pop CS

2.call

格式: call 标号

执行后,进行操作:

(1)将当前的 IPCS 和 IP 压入栈中

(2)转移

3.call far ptr

格式:call far ptr 标号

实现段间转移

12.代码段、数据段、栈段、dup

16进制不能以字母开头

1.在代码段使用数据
assume cs:codesg
codesg segmentdw   123H, 456H, 789H, ABCHstart: mov  ax, 3mov  cx, 11call sinc  bxint  21H
​s:     add  ax,axloop s
​
codesg ends
end start

start 相当于事先把ip跳到第一行代码的有效地址上,避免了定义的数据与执行代码之间的混乱

2.数据、代码、栈放入不同的段
assume cs:code,ds:data,ss:stack
data segmentdw   123H, 456H, 789H, ABCH
data ends
​
stack segmentdb   0, 0, 0, 0, 0, 0;等价于 6 dup(0) 即重复6个;dup 'hello world'  db 3 dup('abc', 'def')
stack ends
​
code segmentstart: mov  ax, 3mov  cx, 11call sinc  bxint  21H
​s:     add  ax,axloop s
​
code ends
end start

dup 的使用格式:

db 重复次数 dup (重复的字节型数据)

dw 重复次数 dup (重复的字型数据)

dd 重复次数 dup (重复的双字型数据)

代码段、数据段、栈段的地址空间的连续的

每个段为避免冲突,至少占用16个字节,(即段地址不同?)

13.offset、jmp

1.操作符offset

功能:取得标号的偏移地址

2.jmp

jmp 为无条件转移指令,可以只修改IP,也可以同时修改CS和IP

1.jmp short 标号

实现段内短转移,对IP修改范围为 -128~127

2.jmp far ptr 标号

实现段间转移,或远转移,修改CS:IP

3.jmp word ptr 内存单元地址(段内地址)

从内存单元地址处开始取一个字,作为转移的目的偏移地址

4.jmp dword ptr 内存单元地址(段间地址)

从内存单元地址处开始取两个字,高地址作为转移的段地址,低地址作为转移的目的偏移地址

14.数组

定义字符串: db ”字符串“

1.arr 数组
assume cs:code,ds:data,ss:stack
data segmentarr  dw 12,34            ;定义数组arr2 db "hello world"
data ends
stack segmentdb 10 dup(0)
stack ends
code segmentstart:mov ax, type arr     ;type arr  可以查看数组的类型mov ax, type arr2    ;0001是字节,0002是字
code ends
end start
2.数据标号

代码段内定义:

assume cs:code,ds:data,ss:stack
data segment
data ends
stack segmentdb 10 dup(0)
stack ends
code segmentarr  dw 12,34            ;定义数组arr2 db "hello world"start:mov ax, arr[2]       ;可以通过下标,访问数组,下标增加1,偏移地址增加1字节,不与定义同步;arr[2] <==> cs:[arr+2]   mov ax, word ptr arr[2]   ;从cs:[arr+2]处开始取一个字的内容,放进axmov si, offset arr    ;可以找到数组的有效/偏移地址
code ends
end start

其他段定义:

assume cs:code,ds:data,ss:stack
data segmentarr  db 10H,20H,30H,40H            ;定义数组arr2 db "hello world"
data ends
stack segmentdb 10 dup(0)
stack ends
code segmentstart:mov ax, datamov ds, ax           ;没有上面两行,会找不到data;设置ds指向data段;即 assume处的data与定义了数组的data不同步,导致寻址不同,而找不到定义的数组;默认访问单元的段地址在ds中,而实际要访问的段为datamov ax, arr[2]       ;等价于;mov si, offset arr;mov al, ds:[si+2]mov ax, type arr2    ;
code ends
end start

15.实战1.0

输出hello world:

assume cs:codesg,ds:data,ss:stack
data segmentstring db "hello world",10,'123','$'    ;$表示终止符,避免向内存后面继续;10是换行的ASCII的值,实现换行功能。32-->空格
data ends
stack segmentdb 10 dup(0)
stack ends
codesg segmentstart: mov ax, datamov ds, axmov dx, offset stringmov ah, 09Hint 21H
​mov ah, 4CHint 21H
codesg ends
end start

大写转小写:

ASSUME CS:CODE,DS:DATA,SS:STACK
DATA SEGMENTSTR  DB "HeLlo woRLd",'$'
DATA ENDS
STACK SEGMENTDB 10 DUP(0)
STACK ENDS
CODE SEGMENTSTART:MOV  AX, DATAMOV  DS, AXMOV  BX, 0MOV  CX, 11
​S:    MOV  AL, [BX]CMP  AL, 'A'JB   NEXTCMP  AL, 'Z'JA   NEXTOR   AL, 20HMOV  [BX], ALNEXT: INC  BXLOOP S;C语言:;for(int i=0;i<strlen(str);i++)if(大写)转小写
​MOV  DX, OFFSET STR;LEA DX, STRMOV  AH, 09H             ;9号功能参数入口为DXINT  21H
​MOV  AH, 4CHINT  21Hcode ends
end start

找最大值:

ASSUME CS:CODE, DS:DATA, SS:STACK
DATA SEGMENTSTRING DB 10, 15, 22, 80, 50, 39,'$'MAX    DB 0
DATA ENDS
STACK SEGMENTDB 10 DUP(0)
STACK ENDS
CODE SEGMENTSTART:   MOV  AX, DATAMOV  DS, AXLEA  BX, STRINGMOV  CX, 5MOV  AL, [BX]S:       MOV  AH, [BX+1]CMP  AL, AHJA   CONTINUEMOV  AL, AHCONTINUE:INC  BXLOOP SMOV  [MAX], ALMOV  AH, 4CHINT  21H
CODE ENDS
END START

参考:《汇编语言》速成指南(全程敲代码)_哔哩哔哩_bilibili


http://www.ppmy.cn/server/153940.html

相关文章

GB/T34944-2017 《Java语言源代码漏洞测试规范》解读——安全功能

GB/T34944-2017 《Java语言源代码漏洞测试规范》标准是软件测试实验室开展代码测试活动的重要依据&#xff0c;也是软件测试实验室申请代码测试CNAS/CMA实验室认证时所依据的标准方法。本系列文章一起解读GB/T34944-2017 《Java语言源代码漏洞测试规范》&#xff0c;前面的文章…

使用Python pickle模块进行序列化

使用Python pickle模块进行序列化 在Python中&#xff0c;pickle模块是一个用于实现数据序列化与反序列化的强大工具。与json模块不同的是&#xff0c;pickle支持将几乎所有的Python对象进行序列化&#xff0c;包括字典、列表、类实例&#xff0c;甚至函数。这使得它在处理复杂…

mybatis 缓存

本地缓存 Mybatis 使用到了两种缓存&#xff1a;本地缓存&#xff08;local cache&#xff09;和二级缓存&#xff08;second level cache&#xff09;。 每当一个新 session 被创建&#xff0c;MyBatis 就会创建一个与之相关联的本地缓存。任何在 session 执行过的查询结果都…

【ComfyUI + 自定义节点】图片叠加掩膜(mask)部分扣掉(变黑)

Comfyui的官方示例&#xff1a;https://github.com/comfyanonymous/ComfyUI/blob/master/custom_nodes/example_node.py.example 一、代码 &#xff08; 逻辑&#xff1a;将图片对应掩膜覆盖的区域替换为黑色&#xff09; 因为comfyui加载的图片后&#xff0c;转化为tensor i…

XDP (eXpress Data Path):在操作系统内核中实现快速、可编程包处理

大家觉得有意义和帮助记得及时关注和点赞!!! 1 引言 1.1 现有方案&#xff08;kernel bypass&#xff09;存在的问题1.2 新方案&#xff1a;给内核网络栈添加可编程能力1.3 新方案&#xff08;XDP&#xff09;的优点1.4 本文组织结构2 相关工作 2.1 用户态轮询 vs. XDP2.2 内核…

USB Hub 检测设备

系列文章目录 xHCI 简单分析 USB Root Hub 分析 USB Hub 检测设备 文章目录 系列文章目录一、引言二、hub_eventshub_port_connect_changeusb_alloc_devusb_set_device_statehub_port_initusb_new_device 一、引言 USB Hub 检测设备 一文中讲到&#xff0c;当有 USB 插入时&…

【UE5.3.2】生成vs工程并rider打开

Rider是跨平台的,UE也是,当前现在windows上测试首先安装ue5.3.2 会自动有右键的菜单: windows上,右键,生成vs工程 生成的结果 sln默认是vs打开的,我的是vs2022,可以open with 选择 rider :Rider 会弹出 RiderLink是什么插

Elasticsearch 集群

集群结构 以三台物理机为例。在这三台物理机上&#xff0c;搭建了 6 个 ES 的节点&#xff0c;三个 data 节点&#xff0c;三个 master 节点&#xff08;每台物理机分别起了一个 data 和一个 master&#xff09;&#xff0c;3 个 master 节点&#xff0c;目的是达到&#xff0…