0

这是代码:

section .data
v dw 4, 6, 8, 12
len equ 4
section .text
    global main
main:
    mov eax, 0 ;this is i
    mov ebx, 0 ;this is j
cycle:
    cmp eax, 2 ;i < len/2
    jge exit
    mov ebx, 0
    jmp inner_cycle
continue:
    inc eax
    jmp cycle
inner_cycle:
    cmp ebx, 2
    jge continue
    mov di, [v + eax * 2 * 2 + ebx * 2]
    inc ebx
    jmp inner_cycle
exit:
    push dword 0
    mov eax, 0
    sub esp, 4
    int 0x80

我正在使用一个数组并将其扫描为矩阵,这是上述代码的 C 翻译

int m[4] = {1,2,3,4};
for(i = 0; i < 2; i++){
    for(j = 0; j < 2; j++){
        printf("%d\n", m[i*2 + j]);
    }
}

当我尝试编译汇编代码时,出现此错误:

DoubleForMatrix.asm:20: error: beroset-p-592-invalid effective address

指的是这条线

mov di, [v + eax * 2 * 2 + ebx * 2]

有人可以解释一下这条线有什么问题吗?我认为这是因为寄存器尺寸,我尝试过

mov edi, [v + eax * 2 * 2 + ebx * 2]

但我有同样的错误。

这是 Mac OS X 的程序集,要使其在另一个 SO 上工作,您必须更改退出系统调用。

4

2 回答 2

1

SIB ( Scale Immediate Base) 寻址模式只需要一个 Scale 参数(1,2,4 或 8)被应用到一个寄存器。

建议的解决方案是将 eax 预乘 4(也必须修改比较)。然后inc eax可以用add eax,4和 非法指令替换为mov di,[v+eax+ebx*2]

更高级别的优化将只是for (i=0;i<4;i++) printf("%d\n",m[i]);

于 2012-11-27T10:15:06.113 回答
1

您不能在汇编程序中使用任意表达式。只允许少数寻址模式。

基本上最复杂的形式是 register/imm+register*scale,比例为 1,2,4,8

当然常数(如 2*2)可能会被折叠为 4,因此算作 4 的单个比例(而不是两次乘法)

你的例子试图一次做两个乘法。

解决方法:插入额外的LEA指令计算v+ebx*2,并在mov中使用结果。

     lea regx , [v+ebx*2]
     mov edi, [eax*2*2+regx]

其中 regx 是一个免费寄存器。

于 2012-11-27T10:20:09.107 回答