我经常在几台不同的电脑和不同的操作系统上工作,比如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

当前回答

我今天遇到了这个问题,我需要在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

其他回答

这就是GNU的automake/autoconf设计用来解决的问题。你可能想调查一下。

或者,您可以在不同的平台上设置环境变量,并根据它们设置Makefile条件。

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

这里已经有很多很好的答案,但我想分享一个更完整的例子:

不会假设uname在Windows上存在 还可以检测处理器

这里定义的CCFLAGS不一定是推荐的或理想的;它们只是我添加OS/CPU自动检测的项目碰巧正在使用的。

ifeq ($(OS),Windows_NT)
    CCFLAGS += -D WIN32
    ifeq ($(PROCESSOR_ARCHITEW6432),AMD64)
        CCFLAGS += -D AMD64
    else
        ifeq ($(PROCESSOR_ARCHITECTURE),AMD64)
            CCFLAGS += -D AMD64
        endif
        ifeq ($(PROCESSOR_ARCHITECTURE),x86)
            CCFLAGS += -D IA32
        endif
    endif
else
    UNAME_S := $(shell uname -s)
    ifeq ($(UNAME_S),Linux)
        CCFLAGS += -D LINUX
    endif
    ifeq ($(UNAME_S),Darwin)
        CCFLAGS += -D OSX
    endif
    UNAME_P := $(shell uname -p)
    ifeq ($(UNAME_P),x86_64)
        CCFLAGS += -D AMD64
    endif
    ifneq ($(filter %86,$(UNAME_P)),)
        CCFLAGS += -D IA32
    endif
    ifneq ($(filter arm%,$(UNAME_P)),)
        CCFLAGS += -D ARM
    endif
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中没有给出操作系统版本的默认环境变量,否则我会使用它!

另一种方法是使用“configure”脚本。如果您已经在makefile中使用了一个,那么您可以使用uname和sed的组合来解决问题。首先,在你的脚本中,做到:

UNAME=uname

然后,为了将其放到Makefile中,请从Makefile开始。里面应该有这样的东西

UNAME=@@UNAME@@

在里面。

在配置脚本中,在UNAME= UNAME位后面使用下面的sed命令。

sed -e "s|@@UNAME@@|$UNAME|" < Makefile.in > Makefile

现在,您的makefile应该按照需要定义了UNAME。剩下的就只有If/elif/else语句了!