【发布时间】:2021-09-08 00:56:51
【问题描述】:
我有一个程序 (main.c):
#include <stdio.h>
#include <math.h>
int main() {
int result = sqrt(9);
printf("result: %d\n" ,result);
return 0;
}
double sqrt(double blah) {
return 0;
}
当我运行它时,我的结果是
result: 3
这将告诉我链接器正在选择 libm 库的 sqrt 函数而不是我的函数来调用我的 main 函数。
在启用所有警告的情况下编译此程序时,我没有收到任何错误或警告:
gcc main.c -Wall
我的问题:
- 为什么链接器没有选择我定义的
sqrt来调用?- 这是确定性的吗?
- 为什么我没有收到任何错误或警告?似乎具有相同签名的函数的多个定义是一个陷阱,应该以某种方式指出。
- 有没有办法输出链接到哪里的函数?那么如果遇到引用了意外定义的情况,我可以调试吗?
我唯一能想到的是当我运行gcc --precompile时,我看到了这个函数声明:
extern double sqrt(double);
这是否告诉链接器 sqrt 是在此文件之外定义的?而且既然这已经满足sqrt的定义了,那么链接的时候我自己的定义就被忽略了?
gcc 信息(我知道它真的很响,因为我在 mac 上,不确定这是否对这个问题有影响)
gcc --version
Configured with: --prefix=/Library/Developer/CommandLineTools/usr --with-gxx-include-dir=/Library/Developer/CommandLineTools/SDKs/MacOSX.sdk/usr/include/c++/4.2.1
Apple clang version 11.0.0 (clang-1100.0.33.17)
Target: x86_64-apple-darwin19.6.0
Thread model: posix
InstalledDir: /Library/Developer/CommandLineTools/usr/bin
编辑:
汇编输出:
.section __TEXT,__text,regular,pure_instructions
.build_version macos, 10, 15 sdk_version 10, 15
.section __TEXT,__literal8,8byte_literals
.p2align 3 ## -- Begin function main
LCPI0_0:
.quad 4621256167635550208 ## double 9
.section __TEXT,__text,regular,pure_instructions
.globl _main
.p2align 4, 0x90
_main: ## @main
.cfi_startproc
## %bb.0:
pushq %rbp
.cfi_def_cfa_offset 16
.cfi_offset %rbp, -16
movq %rsp, %rbp
.cfi_def_cfa_register %rbp
subq $16, %rsp
movsd LCPI0_0(%rip), %xmm0 ## xmm0 = mem[0],zero
movl $0, -4(%rbp)
sqrtsd %xmm0, %xmm0
cvttsd2si %xmm0, %eax
movl %eax, -8(%rbp)
movl -8(%rbp), %esi
leaq L_.str(%rip), %rdi
movb $0, %al
callq _printf
xorl %esi, %esi
movl %eax, -12(%rbp) ## 4-byte Spill
movl %esi, %eax
addq $16, %rsp
popq %rbp
retq
.cfi_endproc
## -- End function
.globl _sqrt ## -- Begin function sqrt
.p2align 4, 0x90
_sqrt: ## @sqrt
.cfi_startproc
## %bb.0:
pushq %rbp
.cfi_def_cfa_offset 16
.cfi_offset %rbp, -16
movq %rsp, %rbp
.cfi_def_cfa_register %rbp
movsd %xmm0, -8(%rbp)
xorps %xmm0, %xmm0
popq %rbp
retq
.cfi_endproc
## -- End function
.section __TEXT,__cstring,cstring_literals
L_.str: ## @.str
.asciz "result: %d\n"
.subsections_via_symbols
【问题讨论】:
-
如果将函数调用替换为在编译时计算的常量值,您不会感到惊讶。检查汇编输出。
-
我不是这个编译到的程序集家族的专家,但看起来好像 sqrt 是在 .s 程序集文件中定义的。我可以看到我认为的定义和我认为的调用。
-
@Shawn 我将程序集输出添加到原始问题中
-
sqrtsd %xmm0, %xmm0。那不是函数调用。所以看来编译器已经将其转换为直接汇编指令(这是有道理的)。 -
感谢两位的帮助。这间接回答了我的大部分问题。我现在看到函数调用类似于“callq _printf”。将来,当我不确定正在链接什么函数时,我会检查程序集输出