5

我正在寻求帮助,让我的 Makefile 做我想做的事。

我已经想出了如何将预处理代码添加到我的 C 源代码中,只有在我为调试而编译时才会编译:

#if DEBUG
  printf("main()\n");

  {
    /* Pauses execution so gdb can attach. */
    int i=9;
    pid_t PID;
    char hostname[256];
    gethostname(hostname, sizeof(hostname));
    printf("PID %d on %s ready for attach.\n", PID=getpid(), hostname);
    fflush(stdout);
    while (i>0) {
      sleep(5);
      i--;
    }
  }
#endif

而且我发现如果我添加-DDEBUG=1到我的编译语句中,上面的代码将被编译(否则它不会被编译)。

接下来,我想向我的 Makefile 传递一个标志,该标志将包含或不包含该-D选项。目前,我有两个单独的编译行,我酌情评论和取消评论。这是我的 Makefile(我从某人那里继承的,我很难理解)。看到说的行CFLAGS吗?:

SHELL = /bin/sh

prefix       = /home/schwarz/sundials/instdir
exec_prefix  = ${prefix}
includedir   = ${prefix}/include
libdir       = ${exec_prefix}/lib

CPP         = cc -E
CPPFLAGS    =
CC          = cc
# CFLAGS      = -Wall -g
CFLAGS      = -Wall -g -DDEBUG=1
# CFLAGS      = -g -O2
LDFLAGS     =
LIBS        = -lm
MPICC       = /usr/local/mpi/bin/mpicc
MPI_INC_DIR = /usr/local/mpi/bin/../include
MPI_LIB_DIR = /usr/local/mpi/bin/../lib
MPI_LIBS    =
MPI_FLAGS   =

INCLUDES = -I${includedir} -I${MPI_INC_DIR}
LIBRARIES = -lsundials_cvode -lsundials_nvecparallel ${LIBS}
LIBRARIES_BL =

EXAMPLES = FPU          # cvAdvDiff_non_p cvDiurnal_kry_bbd_p cvDiurnal_kry_p


OBJECTS = ${EXAMPLES:=.o}

# -----------------------------------------------------------------------------------------

.SUFFIXES : .o .c

.c.o :
        ${MPICC} ${CPPFLAGS} ${CFLAGS} ${MPI_FLAGS} ${INCLUDES} -c $<

# -----------------------------------------------------------------------------------------

all: ${OBJECTS}
        @for i in ${EXAMPLES} ; do \
          echo "${MPICC} -o $${i} $${i}.o ${MPI_FLAGS} ${CFLAGS} ${LDFLAGS} -L${libdir} ${LIBRARIES} -L${MPI_LIB_DIR} ${MPI_LIBS} ${LIBRARIES_BL}" ; \
          ${MPICC} -o $${i} $${i}.o ${MPI_FLAGS} ${CFLAGS} ${LDFLAGS} -L${libdir} ${LIBRARIES} -L${MPI_LIB_DIR} ${MPI_LIBS} ${LIBRARIES_BL}; \
        done

clean:
        rm -f ${OBJECTS}
        rm -f ${EXAMPLES}

我在网上搜索过——我发誓!- 但无法弄清楚如何将参数传递到makefile。如何设置我的 Makefile 以便它针对不同的编译需求对相同代码进行不同类型的编译?如果我可以将特定值传递给 Makefile 会更好,它会将特定值传递给编译器,我可以使用它i在代码中进行初始化。(IOW,我编译为make 8,C 编译为int i=8;)。在 Make 中是否有可能做到这一点?

顺便说一句,我通过列出带有标签(冒号)的文件以及它们的依赖关系来了解 Makefile 以及它们如何工作。但我不明白这个 Makefile 是如何工作的,或者为什么它会发出两个编译语句。因此,简短的解释也会有所帮助。

4

5 回答 5

10

要在调试版本和发布版本之间进行选择:

ifeq($(debug),1)
  CFLAGS      = -Wall -g -DDEBUG=1
else
  CFLAGS      = -Wall -g
endif

然后调用make:

make debug=1

用 DEBUG 编译,或者:

make

无需调试即可编译。

对于初始化 i:

#include <stdio.h>

#ifndef INIT_I
# define INIT_I 9
#endif
int main() {
  int i = INIT_I;
  return 0;
}

和生成文件:

ifneq ($(init_i),)
CFLAGS = "-DINIT_I=$(init_i)"
endif

all:
  gcc d.c $(CFLAGS) -E

设置初始 i:

make init_i=10

或使用默认值:

make
于 2012-08-27T09:34:51.007 回答
3

示例生成文件:

...
CFLAGS = whatever
....
ifeq ("$(DEBUG)","yes")
CFLAGS += -DDEBUG=1
endif

有了这个,你可以运行 make :

make "DEBUG=yes"

基本上,您可以在 make 命令行上设置 makefile 变量。您可以像使用 makefile 中的任何其他变量一样使用它们,因此您也可以直接使用它们。

编辑:不过,你必须小心。除非您注意适当地命名您的目标文件,或使用不同的构建目录,否则使用不同的“调试”设置运行 make 不会自动重新编译。

于 2012-08-27T09:35:13.503 回答
2

是的。

您可以使用 gcc 的-D选项来声明一个宏,在您的代码中使用它:

> gcc -DMYINT=8 foo.c

与 foo.c

int foo(void)
{
  int my = MYINT;
}

并在您的makefile中使用:

CFLAGS    =    -DMYINT=$(MYINT)

并调用 make :

> make MYINT=8
于 2012-08-27T09:36:39.277 回答
1

实现您想要的一种方法是将值CFLAGS从您的 shell 传递给您的 makefile。

CFLAGS='-Wall -g -DDEBUG=1' make -e 

为您调试版本,或

CFLAGS='-Wall -g' make -e

对于您的非调试

于 2012-08-27T09:36:21.837 回答
1

你可以像这样在make中分配你的CFLAGS变量:CFLAGS ?= -Wall -g -DDEBUG=1

如果你运行不带参数的 make 命令,CFLAGS 将有一个默认值 '-Wall -g -DDEBUG=1'

而如果你像这样运行make:make CFLAGS='-DDEBUG=0'

此命令将使用值“-DDEBUG=0”分配 CFLAGS 值,分配的值将覆盖 makefile 中分配的默认值

于 2012-08-27T09:38:25.430 回答