c - 为什么 "movl $1, %edx"而不是 "movl $0, %edx"

标签 c gcc for-loop assembly x86

最近我在看汇编 IA32,我做了一个简单的玩具示例:

#include <stdio.h>

int array[10];
int i = 0;
int sum = 0;

int main(void)
{
    for (i = 0; i < 10; i++)
    {
        array[i] = i;
        sum += array[i];
    }

    printf("SUM = %d\n",sum);
    return 0;
}

是的,我知道不太推荐使用全局变量。我在没有优化的情况下编译了上面的代码并使用了标志 -s,我得到了这个程序集:

  main:
        ...
        movl    $0, %eax
        subl    %eax, %esp
        movl    $0, i
    .L2:
        cmpl    $9, i
        jle .L5
        jmp .L3
    .L5:
        movl    i, %edx
        movl    i, %eax
        movl    %eax, array(,%edx,4)
        movl    i, %eax
        movl    array(,%eax,4), %eax
        addl    %eax, sum
        incl    i
        jmp .L2

没什么花哨的,也很容易理解,这是一个普通的while循环。然后我用 -O2 编译了相同的代码并得到了以下程序集:

main:
    ...
    xorl    %eax, %eax
    movl    $0, i
    movl    $1, %edx
    .p2align 2,,3
.L6:
    movl    sum, %ecx
    addl    %eax, %ecx
    movl    %eax, array-4(,%edx,4)
    movl    %edx, %eax
    incl    %edx
    cmpl    $9, %eax
    movl    %ecx, sum
    movl    %eax, i
    jle .L6
    subl    $8, %esp
    pushl   %ecx
    pushl   $.LC0
    call    printf
    xorl    %eax, %eax
    leave
    ret

在这种情况下,它转换为 do while 类型的循环。从上面的程序集中我不明白的是为什么 "movl $1, %edx" 然后是 "movl %eax, array-4(,%edx,4)" .

%edx 以 1 而不是 0 开始,然后在访问数组时它从初始位置开始 -4(4 字节 = 整数)。为什么不简单呢?

movl    $0, %edx
...
array (,%edx,4)

如果您需要一直执行 -4,则不要从 1 开始。

出于教育原因,我正在使用“GCC: (GNU) 3.2.3 20030502 (Red Hat Linux 3.2.3-24)”来生成易于理解的程序集。

最佳答案

我想我终于明白了,我测试了:

...

int main(void)
{
        for (i = 0; i < 10; i+=2)
        {
         ...
        }
}

得到:

movl    $2, %edx

和 for (i = 0; i < 10; i +=3) 并得到:

movl    $3, %edx

最后用 (i = 1; i < 10; i +=3) 得到:

movl    $4, %edx

因此,编译器正在初始化 %edx = i (i 的初始值) + incrementStep;

关于c - 为什么 "movl $1, %edx"而不是 "movl $0, %edx",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29780215/

相关文章:

c - 使用带有数组的 `addres of` 运算符作为指针的初始值设定项时收到警告

c++ - 关于 gcc 中类型双关的编译器警告

c++ - 在 Cygwin 上使用 CMake 编译 OpenCV 项目,为 Windows 安装 OpenCV

java - 从 int 获取 01 而不是 for 循环中的 1

javascript - JS从没有高阶函数的数组中删除重复项

c - 如何以相反的顺序打印二维字符串数组

c - 在理解 char->ASCII 和一些基础知识方面需要一些帮助

c++ - 涉及逻辑与 (&&) 的复杂表达式

gcc - 如何修复此编译器错误 'format not a string literal and no format arguments'

java - 如何在给定的双变量集中找到第二小的数字