使用shell函数时是否缺少Separator?

时间:2017-08-24 16:58:40

标签: shell makefile solaris gnu-make sunstudio

我正在使用Solaris 11,完全打补丁。我试图通过在ISA下转储预处理器宏来确定编译器是否支持ISA。

Missing Separator 而死亡。在与GNU make shell function一起使用时,我无法找到有关 Missing Separator 错误的信息。

这是减少的案例。没有空格,所以它不是像Make error: missing separator和朋友那样的空格/标签问题。

$ cat -n GNUmakefile-test
 1  EGREP ?= egrep
 2  SUN_COMPILER := $(shell $(CXX) -V 2>&1 | $(EGREP) -i -c "CC: (Sun|Studio)")
 3
 4  # Begin SunCC
 5  ifeq ($(SUN_COMPILER),1)
 6  $(info "Sun compiler")
 7  $(shell $(CXX) $(CXXFLAGS) -E -xarch=ssse3 -xdumpmacros /dev/null 2>/dev/null)
 8  ifeq ($(.SHELLSTATUS),0)
 9  $(info "SSSE3")
10  SSSE3_FLAG = -xarch=ssse3 -D__SSSE3__=1
11  endif
12  endif
13  # End SunCC
14
15  all:
16          $(info "Do nothing")

上述想法是,SunCC不为ISA提供宏,如__AES____SHA__。但是,如果不支持ISA,SunCC将会出错,例如SunCC 12.4上的-xarch=sha。如果我没有收到错误,那么我知道编译器支持ISA,例如SunCC 12.4上的-xarch=aes。如果出现错误,我可以从.SHELLSTATUS获取错误。 (SunCC与Clang,GCC,Intel ICC或MSVC不同)。

结果如下:

$ CXX=/opt/solarisstudio12.4/bin/CC gmake -f GNUmakefile-test
"Sun compiler"
GNUmakefile-test:7: *** missing separator.  Stop.

缺少的分隔符在哪里?或者,报告的真正错误是什么?也许别的什么?

我很抱歉问过这个问题的次数。

我添加了标签以试图安抚make。它产生了同样的错误。

$ cat -n GNUmakefile-test
 1  EGREP ?= egrep
 2  SUN_COMPILER := $(shell $(CXX) -V 2>&1 | $(EGREP) -i -c "CC: (Sun|Studio)")
 3
 4  # Begin SunCC
 5  ifeq ($(SUN_COMPILER),1)
 6       $(info "Sun compiler")
 7       $(shell $(CXX) $(CXXFLAGS) -E -xarch=ssse3 -xdumpmacros /dev/null 2>/dev/null)
 8       ifeq ($(.SHELLSTATUS),0)
 9            $(info "SSSE3")
10            SSSE3_FLAG = -xarch=ssse3 -D__SSSE3__=1
11       endif
12  endif
13  # End SunCC
14
15  all:
16       $(info "Do nothing")

2 个答案:

答案 0 :(得分:4)

shell函数的工作方式(如手册中所述)是它运行命令,然后扩展为命令的输出。这就是为什么,当你看到:

SUN_COMPILER := $(shell $(CXX) -V 2>&1 | $(EGREP) -i -c "CC: (Sun|Studio)")

变量SUN_COMPILER设置为该shell命令的输出。

所以当你这样写:

$(shell $(CXX) $(CXXFLAGS) -E -xarch=ssse3 -xdumpmacros /dev/null 2>/dev/null)

运行命令,然后替换输出。之后,make尝试将结果解析为make语法。但是该命令的输出显然不是语法,所以你得到这个错误。

如果您不关心输出并且只关心退出代码,则需要丢弃输出:

$(shell $(CXX) $(CXXFLAGS) -E -xarch=ssse3 -xdumpmacros /dev/null >/dev/null 2>&1)

或者将其分配给虚拟变量:

_x := $(shell $(CXX) $(CXXFLAGS) -E -xarch=ssse3 -xdumpmacros /dev/null 2>/dev/null)

这样make就会将结果赋给变量,而不是认为它是make语法。

答案 1 :(得分:1)

我们决定避免.SHELLSTATUS,因为它似乎有一些问题。我们回到了grep和字符串"illegal value ignored"

ifeq ($(SUN_COMPILER),1)
  COUNT := $(shell $(CXX) $(CXXFLAGS) -E -xarch=ssse3 -xdumpmacros /dev/null 2>&1 | $(EGREP) -i -c "illegal value ignored")
  ifeq ($(COUNT),0)
    SSSE3_FLAG = -xarch=ssse3 -D__SSSE3__=1
    ARIA_FLAG = -xarch=ssse3 -D__SSSE3__=1
  endif
  COUNT := $(shell $(CXX) $(CXXFLAGS) -E -xarch=sse4_2 -xdumpmacros /dev/null 2>&1 | $(EGREP) -i -c "illegal value ignored")
  ifeq ($(COUNT),0)
    BLAKE2_FLAG = -xarch=sse4_2 -D__SSE4_2__=1
    CRC_FLAG = -xarch=sse4_2 -D__SSE4_2__=1
  endif
  COUNT := $(shell $(CXX) $(CXXFLAGS) -E -xarch=aes -xdumpmacros /dev/null 2>&1 | $(EGREP) -i -c "illegal value ignored")
  ifeq ($(COUNT),0)
    GCM_FLAG = -xarch=aes -D__PCLMUL__=1
    AES_FLAG = -xarch=aes -D__AES__=1
  endif
  COUNT := $(shell $(CXX) $(CXXFLAGS) -E -xarch=sha -xdumpmacros /dev/null 2>&1 | $(EGREP) -i -c "illegal value ignored")
  ifeq ($(COUNT),0)
    SHA_FLAG = -xarch=sha -D__SHA__=1
  endif
endif
# End SunCC

我们不确定-xarch=sha。目前我们正在努力弄清楚需要什么。