22

在makefile中,我得到了所有目标及其依赖全递归。我搜索整个文件,但我无法定义全递归。我认为全递归也必须是一个目标,或者下一步该怎么做?所以有人可以告诉我如何处理这个问题,我将非常感谢你的帮助。

all: $(BUILT_SOURCES)
    $(MAKE) $(AM_MAKEFLAGS) all-recursive

我无法得到“全递归”的定义。如果我删除它,make 程序将继续处理所有目标。那是“全递归”内置的吗?

4

2 回答 2

29

我也花了几个小时找到它们。不,这不是 ; 的内置功能make。原来这是 Autotools 生成的 Makefile 的一个特征。

这些THING-recursive目标实际上是在那个 Makefile 中定义的,但是以一种复杂的方式,您无法使用 simplegrep来找到它们。

它从RECURSIVE_TARGETSMakefile 中的变量定义开始,如下所示:

RECURSIVE_TARGETS = all-recursive check-recursive dvi-recursive \
    html-recursive info-recursive install-data-recursive \
    install-dvi-recursive install-exec-recursive \
    install-html-recursive install-info-recursive \
    install-pdf-recursive install-ps-recursive install-recursive \
    installcheck-recursive installdirs-recursive pdf-recursive \
    ps-recursive uninstall-recursive

然后是下面某处对这些目标的真实定义:

$(RECURSIVE_TARGETS):
    @fail= failcom='exit 1'; \
    for f in x $$MAKEFLAGS; do \
      case $$f in \
        *=* | --[!k]*);; \
        *k*) failcom='fail=yes';; \
      esac; \
    done; \
    dot_seen=no; \
    target=`echo $@ | sed s/-recursive//`; \
    list='$(SUBDIRS)'; for subdir in $$list; do \
      echo "Making $$target in $$subdir"; \
      if test "$$subdir" = "."; then \
        dot_seen=yes; \
        local_target="$$target-am"; \
      else \
        local_target="$$target"; \
      fi; \
      ($(am__cd) $$subdir && $(MAKE) $(AM_MAKEFLAGS) $$local_target) \
      || eval $$failcom; \
    done; \
    if test "$$dot_seen" = "no"; then \
      $(MAKE) $(AM_MAKEFLAGS) "$$target-am" || exit 1; \
    fi; test -z "$$fail"

这解决了这个问题:

all-recursive check-recursive dvi-recursive \
    html-recursive info-recursive install-data-recursive \
    install-dvi-recursive install-exec-recursive \
    install-html-recursive install-info-recursive \
    install-pdf-recursive install-ps-recursive install-recursive \
    installcheck-recursive installdirs-recursive pdf-recursive \
    ps-recursive uninstall-recursive:
        @fail= failcom='exit 1'; \
        for f in x $$MAKEFLAGS; do \
          case $$f in \
            *=* | --[!k]*);; \
            *k*) failcom='fail=yes';; \
          esac; \
        done; \
        dot_seen=no; \
        target=`echo $@ | sed s/-recursive//`; \
        list='$(SUBDIRS)'; for subdir in $$list; do \
          echo "Making $$target in $$subdir"; \
          if test "$$subdir" = "."; then \
            dot_seen=yes; \
            local_target="$$target-am"; \
          else \
            local_target="$$target"; \
          fi; \
          ($(am__cd) $$subdir && $(MAKE) $(AM_MAKEFLAGS) $$local_target) \
          || eval $$failcom; \
        done; \
        if test "$$dot_seen" = "no"; then \
          $(MAKE) $(AM_MAKEFLAGS) "$$target-am" || exit 1; \
        fi; test -z "$$fail"

这个目标的配方本质上是一个样板代码,用于循环在当前文件夹中找到的每个子目录并make在其中开始,使用相同的目标名称但去掉“-recursive”。

请注意,这些THING-recursive目标并不意味着用户可以直接调用;它将作为正常THING目标的一部分自动运行(没有“-recursive”),作为触发子项目树中相同目标构建的机制。

附录:示例代码取自GNU Flash Player(版本e9eb84e)的配置根 Makefile。

于 2017-02-20T10:54:10.363 回答
2

查看它是否包含在目标 $(RECURSIVE_TARGETS) 中定义

于 2016-04-13T04:29:25.303 回答