GNU Make gcc,选项顺序

问题描述

我试图找出 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 变量)。>