Linux汇编----变址寻址与间接寻址

    xiaoxiao2025-04-03  10

    经常会把AT&T汇编语法中的变址寻址(indexed memory mode)与寄存器间接寻址搞混。

    1. 变址寻址模式

    变址寻址特别适合于访问内存中的一系列连续数据,即数组。表达式格式为

    base_address(offset_address,index,size)

    其中base_address是内存地址(例如数组首地址),而offset_address 和 index的值必须是寄存器保存的值,size可以为数字值。

    最终的地址为base_address + offset_address + index * size。下面用一个实例代码演示一下。

    <pre name="code" class="objc">#test1.s .section .data output: .asciz "The value is %d\n" values: .int 10,15,12,18,5 .section .text .global _start _start: movl $0,%edi loop: movl values(,%edi,4),%eax pushl %eax pushl $output call printf addl $8,%esp inc %edi cmpl $5,%edi jne loop movl $0,%ebx movl $1,%eax int $0x80

    然后编译

    gcc test1.s -nostdlib -lc这里的-nostdlib是告诉连接器别把默认的标准库文件(crt1.o)链接进去,-lc的意思是链接libc,因为我们遇到了printf。然后运行,得到如下结果

    The value is 10 The value is 15 The value is 12 The value is 18 The value is 5

    2. 间接寻址模式

    当寄存器中保存的是内存地址时,它就被称为指针(pointer)。表达式格式为

    size(%register)其中size为数字,可为负值。

    最终的内存地址为%register保存的值 + size

    我们把上例稍作修改

    #test2.s .section .data output: .asciz "The value is %d\n" values: .int 10,15,12,18,5 .section .text .global _start _start: movl $values,%edi movl 4(%edi),%eax pushl %eax pushl $output call printf addl $8,%esp movl $values,%edi movl 12(%edi),%eax pushl %eax pushl $output call printf addl $8,%esp movl $0,%ebx movl $1,%eax int $0x80

    编译运行后,输出

    The value is 15 The value is 18

    转载请注明原文地址: https://ju.6miu.com/read-1297672.html
    最新回复(0)