C内联汇编标签问题

时间:2012-04-09 23:00:26

标签: c assembly inline-assembly

我目前是初学者,我正在学习如何使用C语言进行内联汇编。话虽这么说,我在编译文件时遇到这个特殊错误很难:

/tmp/cckHnU89.s: Assembler messages:
/tmp/cckHnU89.s:550: Error: symbol `.L16' is already defined
/tmp/cckHnU89.s:571: Error: symbol `.L18' is already defined
/tmp/cckHnU89.s:576: Error: symbol `.L17' is already defined

我尝试用其他名称替换标签的名称,因为我从.s文件中注意到标签.L16,.L17和.L18在我的main方法和我的一个函数中使用。然而,当我这样做时,我最终因运行程序而出现了分段错误。有没有办法更改标签的名称或其他东西来修复哪些显然似乎是命名冲突?

就我的CPU而言,我正在运行英特尔奔腾T4500处理器,而我正在使用gcc 4.4.3版进行编译。对于内联汇编部分,我的代码是300多行,所以无论谁读到这个,我都会有空。从本质上讲,我只是在寻找关于如何正常修复产生上述错误的命名冲突的一般答案。任何见解将不胜感激。

1 个答案:

答案 0 :(得分:2)

我的预感是(我刚刚使用g++ -Sgcc -S进行验证)您自己的标签正在模仿自动分配给汇编程序代码的标签的命名方案(.L<num>)在GCC。

执行以下操作:

# for C:
gcc -S source.c
# for C++
g++ -S source.cpp

...然后cat(或less)生成的.s文件(相同的基本名称,.s后缀,例如source.s)。您会发现该方案的许多标签(.L<num>)。现在,如果您自己创建包含与已自动创建的标签(从C代码中)相同名称的内联汇编,那么这显然会导致冲突。

所以要点:不要使用.L<num>作为标签的命名方案,因为它会发生冲突。

通常.L开头的名称似乎在这里遇到麻烦。


示例(test.cpp),使用g++ -S test.cpp编译:

#include <cstdio>

int main(int argc, char**argv)
{
        switch(argc)
        {
                case 0:
                        printf("test 0\n");
                        break;
                case 1:
                        printf("test %d\n", argc);
                        break;
                case 2:
                        printf("test %d -> %s\n", argc, argv[0]);
                        break;
                default:
                        printf("Default\n");
                        break;
        }
        return 0;
}

在x64上编译(test.s的内容):

        .file   "test.cpp"
        .section        .rodata
.LC0:
        .string "test 0"
.LC1:
        .string "test %d\n"
.LC2:
        .string "test %d -> %s\n"
.LC3:
        .string "Default"
        .text
.globl main
        .type   main, @function
main:
.LFB0:
        .cfi_startproc
        .cfi_personality 0x3,__gxx_personality_v0
        pushq   %rbp
        .cfi_def_cfa_offset 16
        movq    %rsp, %rbp
        .cfi_offset 6, -16
        .cfi_def_cfa_register 6
        subq    $16, %rsp
        movl    %edi, -4(%rbp)
        movq    %rsi, -16(%rbp)
        movl    -4(%rbp), %eax
        cmpl    $1, %eax
        je      .L4
        cmpl    $2, %eax
        je      .L5
        testl   %eax, %eax
        jne     .L8
.L3:
        movl    $.LC0, %edi
        call    puts
        jmp     .L6
.L4:
        movl    -4(%rbp), %eax
        movl    %eax, %esi
        movl    $.LC1, %edi
        movl    $0, %eax
        call    printf
        jmp     .L6
.L5:
        movq    -16(%rbp), %rax
        movq    (%rax), %rdx
        movl    -4(%rbp), %eax
        movl    %eax, %esi
        movl    $.LC2, %edi
        movl    $0, %eax
        call    printf
        jmp     .L6
.L8:
        movl    $.LC3, %edi
        call    puts
.L6:
        movl    $0, %eax
        leave
        ret
        .cfi_endproc
.LFE0:
        .size   main, .-main
        .ident  "GCC: (Debian 4.4.5-8) 4.4.5"
        .section        .note.GNU-stack,"",@progbits

在生成的汇编程序文件中观察以.L开头的名称。