我目前是初学者,我正在学习如何使用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多行,所以无论谁读到这个,我都会有空。从本质上讲,我只是在寻找关于如何正常修复产生上述错误的命名冲突的一般答案。任何见解将不胜感激。
答案 0 :(得分:2)
我的预感是(我刚刚使用g++ -S
和gcc -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
开头的名称。