X86 Assemble指令--MOVS指令簇

时间:2022-06-13
本文章向大家介绍X86 Assemble指令--MOVS指令簇,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

介绍

MOVS系列的指令簇主要用来复制字符串(move string)。

作用

主要将DS:SI地址的数据复制到ES:DI地址中。

该指令簇包括: MOVS mem16,mem16:显式指定源操作数与目的操作数 MOVSB:每次移动一个字节 MOVSW:每次移动一个字 MOVSD:每次移动两个字

描述

从第二个操作数(源操作数)地址开始移动byte,word或者double word到第一个操作数(目的操作数)所指定的位置。源操作数与目的操作数都处于内存中。

源操作数的地址从DS:ESI寄存器中读取,或者从DS:SI寄存器中读取。(这取决于该指令集的地址宽度是32位还是16位)。

目的操作数的地址则从ES:EDI寄存器或者ES:DI寄存器中获取(同样取决于该指令集的地址宽度)。DS段可以被段前缀重写,而ES段则不允许被重写。

在Assemble代码级别,该指令有两种形式:显示操作数隐式操作数

显式操作数的形式通过使用MOVS符号,允许源操作数与目的操作数被显式指定。这种方式源操作数与目的操作数需要标明移动大小以及源操作数与目的操作数的地址。源操作数与目的操作数必须指定正确的操作数大小(bytes,words,doublewords),但是不用指定正确的地址。

源操作数与目的操作数总是在DS:(E)SI以及ES:(E)DI寄存器中指定,而这寄存器中的地址必须在move string指令执行前被正确加载。

而无操作数的move string指令同样也是使用DS:(E)SI与ES:(E)DI寄存器来指定源操作数与目的操作数的地址。并且移动的大小在指令中体现:MOVSB(byte move),MOVSW(word move),MOVSD(double word move)。

在Move操作结束后,(E)SI与(E)DI寄存器会自动的增加或者减少,而依据则是根据EFLAGS寄存器中的DF标志位来判断增加还是减少。

  • 如果DF标志位为0,则(E)SI与(E)DI寄存器则自增
  • 如果DF标志位为1,则(E)SI与(E)DI寄存器则自减

CLD(clear direction flag)该指令使DF=0,使地址自动增量 STD(set direction flag)该指令使DF=1,使地址自动减量

而每次自增/自减的长度根据指令而定:

  • MOVSB则增加/减少一个字节
  • MOVSW则增加/减少一个字
  • MOVSD则增加/减少两个字

MOVS,MOVSB,MOVSW,MOVSD指令会先于REP前缀的指令集执行bytes,words,doublewords的内存块移动。

异常

当目的操作数的地址在不能写的段中时,会出现#GP(0)异常 当目的操作数的地址不在内存中时,会发生缺页中断,出现#PF异常

实例

MOV CX ,100
LEA SI,FIRST
LEA DI,SECOND
REP MOVSB
  1. 先将100加载到CX寄存器中
  2. FIRST标号的地址加载到DS:SI寄存器中,将SECOND标号的地址加载到ES:DI寄存器中
  3. 执行MOVSB,将FIRST标号作为基址大小为一个字节的数据复制到SECOND标号位置
  4. 执行完后,将CX-1,重复执行MOVSB
  5. 直到CX寄存器为0时,再往下执行

该段代码主要完成了从FIRST标号地址开始复制100个字节到SECOND标号地址位置

参考资料

MOVS/MOVSB/MOVSW/MOVSD