链接多个C源文件

mac*_*ium 4 c linker gentoo gcc4

我在安装了GCC 4.4.5的gentoo linux上.我可以使用gcc main.c -o main编译和链接这样的程序而没有任何错误,并且命令./main正确返回结果.

[main.c] 
#include <math.h>
#include <stdio.h>
int main(void)
{
    double c = ceil(2.5);
    printf("The ceil of 2.5 is %f\n", c);
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

但是,当我将ceil的调用放入另一个源文件时,就会出现问题.

[calc.h]
#ifndef _CALC_H_
#define _CALC_H_
double myceil(double n);
#endif

[calc.c]
#include <math.h>
#include "calc.h"
double myceil(double n)
{
    return ceil(n);
}

[main1.c]
#include <stdio.h>
#include "calc.h"
int main(void)
{
    double c = myceil(2.5);
    printf("The ceil of 2.5 is %f\n", c);
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

使用命令gcc calc.c main1.c -o main1,会发生此类错误:

/tmp/cc6GhJvZ.o: In function `myceil':
calc.c:(.text+0x19): undefined reference to `ceil'
collect2: ld returned 1 exit status
Run Code Online (Sandbox Code Playgroud)

那么为什么在后一种情况下会发生恼人的错误"未定义的引用"呢?我知道可以通过添加库-lm来消除错误,但是,我只是想知道为什么gcc会在后一种情况下抛出错误.

asv*_*kau 5

我的猜测是GCC优化ceil(2.5)为常量,而ceil(n)不是常数,因为n在编译时不知道calc.c,并且它需要引用该函数.您可以通过查看程序集输出(gcc -S)来确认这一点.

更新:这是x86上的gcc 4.2.1给了我类似于你的第一个例子:

.LC1:
    .string "%f\n"
    // [snip]
main:
    // [snip]
    fldl    .LC0
    fstpl   4(%esp)
    movl    $.LC1, (%esp)
    call    printf
    // [snip]
.LC0:
    .long   0
    .long   1074266112
Run Code Online (Sandbox Code Playgroud)

在这里,我们看到printf被一个double常数调用.

现在,如果我做了类似于你的第二个例子:

myceil:
    // [snip]
    fldl    -8(%ebp)
    fstpl   (%esp)
    call    ceil
    // [snip]
Run Code Online (Sandbox Code Playgroud)

在这里,我们看到ceil被引用.

是的.我会说你的电话被优化为一个不用的电话-lm.