我经常在几台不同的电脑和不同的操作系统上工作,比如Mac OS X、Linux或Solaris。对于我正在进行的项目,我从远程git存储库中提取代码。

不管我在哪个终端,我都喜欢能够在我的项目上工作。到目前为止,我已经找到了绕过操作系统更改的方法,每次换电脑时都要更改makefile。然而,这很乏味,而且会引起很多头疼。

我如何修改我的makefile,以便它检测到我正在使用的操作系统并相应地修改语法?

下面是makefile文件:

cc = gcc -g
CC = g++ -g
yacc=$(YACC)
lex=$(FLEX)

all: assembler

assembler: y.tab.o lex.yy.o
        $(CC) -o assembler y.tab.o lex.yy.o -ll -l y

assembler.o: assembler.c
        $(cc) -o assembler.o assembler.c

y.tab.o: assem.y
        $(yacc) -d assem.y
        $(CC) -c y.tab.c

lex.yy.o: assem.l
        $(lex) assem.l
        $(cc) -c lex.yy.c

clean:
        rm -f lex.yy.c y.tab.c y.tab.h assembler *.o *.tmp *.debug *.acts

当前回答

不带参数的uname命令(http://developer.apple.com/documentation/Darwin/Reference/ManPages/man1/uname.1.html)应该告诉您操作系统的名称。我会使用它,然后根据返回值创建条件。

例子

UNAME := $(shell uname)

ifeq ($(UNAME), Linux)
# do something Linux-y
endif
ifeq ($(UNAME), Solaris)
# do something Solaris-y
endif

其他回答

git makefile包含了许多如何在没有autoconf/automake的情况下进行管理的示例,但仍然可以在许多unix平台上工作。

为了回答这个我一直在问自己的问题,我最近在做实验。以下是我的结论:

因为在Windows中,不能确定uname命令是否可用,所以可以使用gcc -dumpmachine。这将显示编译器目标。

如果您想进行交叉编译,在使用uname时也可能会出现问题。

下面是gcc -dumpmachine可能输出的示例列表:

mingw32 i686-pc-cygwin x86_64-redhat-linux

你可以像这样在makefile中检查结果:

SYS := $(shell gcc -dumpmachine)
ifneq (, $(findstring linux, $(SYS)))
 # Do Linux things
else ifneq(, $(findstring mingw, $(SYS)))
 # Do MinGW things
else ifneq(, $(findstring cygwin, $(SYS)))
 # Do Cygwin things
else
 # Do things for others
endif

它对我来说工作得很好,但我不确定这是获得系统类型的可靠方法。至少MinGW是可靠的,这就是我所需要的,因为它不需要在Windows中有uname命令或MSYS包。

总而言之,uname提供了正在进行编译的系统,而gcc -dumpmachine提供了正在进行编译的系统。

不带参数的uname命令(http://developer.apple.com/documentation/Darwin/Reference/ManPages/man1/uname.1.html)应该告诉您操作系统的名称。我会使用它,然后根据返回值创建条件。

例子

UNAME := $(shell uname)

ifeq ($(UNAME), Linux)
# do something Linux-y
endif
ifeq ($(UNAME), Solaris)
# do something Solaris-y
endif

我有一个案例,我必须检测两个版本的Fedora之间的差异,以调整inkscape的命令行选项: 在Fedora 31中,默认的inkscape是1.0beta,使用——export-file 在Fedora < 31中,默认inkscape是0.92,使用——export-pdf

我的Makefile包含以下内容

# set VERSION_ID from /etc/os-release

$(eval $(shell grep VERSION_ID /etc/os-release))

# select the inkscape export syntax

ifeq ($(VERSION_ID),31)
EXPORT = export-file
else
EXPORT = export-pdf
endif

# rule to convert inkscape SVG (drawing) to PDF

%.pdf : %.svg
    inkscape --export-area-drawing $< --$(EXPORT)=$@

这是因为/etc/os-release包含一行

VERSION_ID=<value>

所以shell命令在Makefile中返回字符串VERSION_ID=<值>,然后eval命令在此基础上设置Makefile变量VERSION_ID。 这显然可以针对其他操作系统进行调整,具体取决于元数据的存储方式。注意,在Fedora中没有给出操作系统版本的默认环境变量,否则我会使用它!

我今天遇到了这个问题,我需要在Solaris上使用它,所以这里有一种POSIX标准方法(非常接近)。

#Detect OS
UNAME = `uname`

# Build based on OS name
DetectOS:
    -@make $(UNAME)


# OS is Linux, use GCC
Linux: program.c
    @SHELL_VARIABLE="-D_LINUX_STUFF_HERE_"
    rm -f program
    gcc $(SHELL_VARIABLE) -o program program.c

# OS is Solaris, use c99
SunOS: program.c
    @SHELL_VARIABLE="-D_SOLARIS_STUFF_HERE_"
    rm -f program
    c99 $(SHELL_VARIABLE) -o program program.c