问题描述
我试图找出 make 处理 gcc 命令选项的方式:
我正在重建 Richard dobson 的程序之一,以将声音文件转换为浮动文件,
它需要链接一个位于 $HOME/.local/lib
如果我运行,项目构建没有问题
gcc -Wl,-rpath=$HOME/.local/lib -lm -o sf2float sf2float.c -lportsf
在其他地方有一些关于堆栈交换和https://nullprogram.com/blog/2017/06/19/
的非常有用的帖子我真的很想弄清楚如何从 makefile 中运行这个命令。
我尝试了以下 makefile 的几种不同变体:
SHELL=bash
CC = gcc
CFLAGS = -Wall
LDFLAGS = -Wl,-rpath=$$HOME/.local/lib -lm -lportsf
OBJECTS = sf2float.o
BINARY = sf2float
all: sf2float
example: $(OBJECTS)
$(CC) $(CFLAGS) $(OBJECTS) -o $(BINARY) $(LDFLAGS)
clean:
rm -f $(BINARY) $(OBJECTS)
这会返回编译器选项顺序略有不同的变化,然后在找不到静态库后出现一堆错误:
gcc -Wl,-rpath=$HOME/.local/lib -lm -lportsf sf2float.o -o sf2float
/usr/bin/ld: sf2float.o: in function `main':
sf2float.c:(.text+0x64): undefined reference to `psf_init'
/usr/bin/ld: sf2float.c:(.text+0x9d): undefined reference to `psf_sndOpen'
/usr/bin/ld: sf2float.c:(.text+0xfd): undefined reference to `psf_getFormatExt'
/usr/bin/ld: sf2float.c:(.text+0x15b): undefined reference to `psf_sndCreate'
/usr/bin/ld: sf2float.c:(.text+0x210): undefined reference to `psf_sndReadFloatFrames'
/usr/bin/ld: sf2float.c:(.text+0x23b): undefined reference to `psf_sndWriteFloatFrames'
/usr/bin/ld: sf2float.c:(.text+0x268): undefined reference to `psf_sndReadFloatFrames'
/usr/bin/ld: sf2float.c:(.text+0x2c7): undefined reference to `psf_sndReadPeaks'
/usr/bin/ld: sf2float.c:(.text+0x38f): undefined reference to `psf_sndClose'
/usr/bin/ld: sf2float.c:(.text+0x39f): undefined reference to `psf_sndClose'
/usr/bin/ld: sf2float.c:(.text+0x3ca): undefined reference to `psf_finish'
collect2: error: ld returned 1 exit status
make: *** [<builtin>: sf2float] Error 1
看起来库的顺序有点混乱,老实说,我只是不完全确定这里发生了什么。正如我上面提到的,该项目正在建设中,但我想更清楚地了解这个问题。非常感谢任何帮助!
解决方法
您有两个问题导致了您的问题。
第一个是 make 问题:您要求 make 构建 sf2float
,因为您将 all: sf2float
作为第一个目标。
但是,您还没有告诉 make 如何构建 sf2float
。您告诉它如何构建名为 example
的目标,而您从未将其列为任何先决条件:
example: $(OBJECTS)
所以,make 根本没有使用你的规则。幸运的是 (?) make 有内置规则,知道如何从同名的目标文件创建二进制文件。该内置规则(基本上)如下所示:
% : %.o
$(CC) $(LDFLAGS) $^ $(LDLIBS) -o $@
这为您提供了您看到的命令行:
gcc -Wl,-rpath=$HOME/.local/lib -lm -lportsf sf2float.o -o sf2float
这会为您提供您看到的错误,因为您在目标文件之前的链接行上有库,而 GCC 与大多数链接器一样是单程链接器。
要解决此问题,您应该做以下两件事之一(或两者皆有):
按照惯例,LDFLAGS
make 变量用于链接器标志,例如 -Wl...
,LDLIBS
make 变量用于库,例如 { {1}}。因此,要遵循惯例,您可以这样写:
-lm
和/或,如果您想使用您的规则而不是内置规则,您应该将目标更改为 LDFLAGS = -Wl,-rpath=$$HOME/.local/lib
LDLIBS = -lportsf -lm
(或者,在任何地方使用 sf2float
变量)。>