ChatGPT解决这个技术问题 Extra ChatGPT

如何获取 Makefile 的当前相对目录?

我在应用程序特定目录中有几个 Makefile,如下所示:

/project1/apps/app_typeA/Makefile
/project1/apps/app_typeB/Makefile
/project1/apps/app_typeC/Makefile

每个 Makefile 在此路径中都包含一个 .inc 文件,向上一级:

/project1/apps/app_rules.inc

在 app_rules.inc 中,我正在设置构建时希望放置二进制文件的位置。我希望所有二进制文件都在各自的 app_type 路径中:

/project1/bin/app_typeA/

我尝试使用 $(CURDIR),如下所示:

OUTPUT_PATH = /project1/bin/$(CURDIR)

但相反,我将二进制文件埋在整个路径名中,如下所示:(注意冗余)

/project1/bin/projects/users/bob/project1/apps/app_typeA

我可以做些什么来获取执行的“当前目录”,以便我可以只知道 app_typeX 以便将二进制文件放入它们各自的类型文件夹中?


B
Bernardo Ramos

外壳函数。

您可以使用 shell 函数:current_dir = $(shell pwd)。或 shellnotdir 组合,如果您不需要绝对路径:current_dir = $(notdir $(shell pwd))

更新。

仅当您从 Makefile 的当前目录运行 make 时,给定的解决方案才有效。
正如@Flimm 所说:

请注意,这将返回当前工作目录,而不是 Makefile 的父目录。例如,如果您运行 cd /; make -f /home/username/project/Makefile,current_dir 变量将是 /,而不是 /home/username/project/。

下面的代码适用于从任何目录调用的 Makefile:

mkfile_path := $(abspath $(lastword $(MAKEFILE_LIST)))
current_dir := $(notdir $(patsubst %/,%,$(dir $(mkfile_path))))

那是行不通的。这是同样的问题。 pwd 打印完整路径名,我只想要我所在的实际当前文件夹的名称。
它需要立即扩展值,因此应使用 := 运算符。与 mkfile_path := 和 current_dir = 一样(否则,当包含其他 Makefile 后 MAKEFILE_LIST 发生更改时,可以稍后评估右侧值
我知道这是一个老问题,但我刚刚遇到并认为这可能很有用。当 makefile 的相对路径中有空格时,这也不起作用。具体来说,路径 "Sub Directory/Makefile" 处的 makefile 的 $(lastword "$(MAKEFILE_LIST)") 将为您提供 "Directory/Makefile"。我认为没有办法解决这个问题,因为带有两个 makefile 的 $(MAKEFILE_LIST) 给了你一个字符串 "Makefile One Makefile Two,它不能处理空格
那个 current_dir 对我不起作用。 $(PWD) 可以,而且要简单得多。
"solution only works when you are running make from the Makefile's current directory" 是表示“没有真正工作”的温和方式。我会将最新的文章放在答案的顶部。
L
Lesmana

取自here

ROOT_DIR:=$(shell dirname $(realpath $(firstword $(MAKEFILE_LIST))))

显示为;

$ cd /home/user/

$ make -f test/Makefile 
/home/user/test

$ cd test; make Makefile 
/home/user/test

希望这可以帮助


恕我直言,最好使用内部 make-function dir 而不是 shell dirname 尽管它获取带有尾随 / 字符的路径名。
因为减少了对外部工具的依赖。即,如果在某些系统上会有 dirname 命令的别名怎么办?
这几乎对我有用,但 DIR 有一个前导空格,所以一个小的改动就可以完美地工作:DIR:=$(strip $(shell dirname $(realpath $(lastword $(MAKEFILE_LIST)))))
要引用 current 生成文件,必须在任何 include 操作 (docs) 之前使用 $MAKEFILE_LIST
应该使用引号 - 至少在 dirname 的参数周围 - 以防路径中有空格。
A
Alastair Irvine

如果您使用 GNU make,$(CURDIR) 实际上是一个内置变量。它是 Makefile 驻留在当前工作目录的位置,这可能是 Makefile 所在的位置,但并非总是如此。

OUTPUT_PATH = /project1/bin/$(notdir $(CURDIR))

请参阅 http://www.gnu.org/software/make/manual/make.html 中的附录 A 快速参考


来自 GNU Make 手册(第 51 页):“当 GNU make 启动时(在它处理了任何 -C 选项之后),它会将变量 CURDIR 设置为当前工作目录的路径名。”不是 Makefile 所在的位置——尽管它们可能是相同的。
正如 Simon Peverett 在之前的评论中所指出的那样,因为答案在技术上不正确,所以被否决了。
@SimonPeverett:括号中的表达式表示“应用 -C 选择后的当前工作目录”。即 $(CURDIR) 为“make -C xxx”和“cd xxx; make”给出完全相同的结果。它还删除了尾随的 /。我尝试与 gmake v3.81 绑定。但是如果 make 被称为 make -f xxx/Makefile,你是对的。
CURDIR 对我有用,而 PWD 没有。我做了 mkdir -p a/s/d,然后在每个文件夹中都有一个在 +$(MAKE) <subdir> 之前打印 PWDCURDIR 的生成文件。
这应该是正确的答案。无需重新发明轮子。 Make 为你做 ootb。
L
Lesmana
THIS_DIR := $(dir $(abspath $(firstword $(MAKEFILE_LIST))))

...除非您在任何路径中有空格。
...除非您在上一行中包含了其他一些 makefile
@robal 是的。这就是我刚刚遇到的。如果您只有两个 Makefile(主要的一个加上一个包含的),这将非常有用。
这对我来说效果很好(没有测试目录中的空间,但我不希望遇到这种情况)。我对这段代码的主要问题是 $(dir) 保留了最后一个路径分隔符。在解决该问题之前添加 $(realpath)
B
Bruno Bronosky

我喜欢选择的答案,但我认为实际展示它的工作比解释它更有帮助。

/tmp/makefile_path_test.sh

#!/bin/bash -eu

# Create a testing dir
temp_dir=/tmp/makefile_path_test
proj_dir=$temp_dir/dir1/dir2/dir3
mkdir -p $proj_dir

# Create the Makefile in $proj_dir
# (Because of this, $proj_dir is what $(path) should evaluate to.)
cat > $proj_dir/Makefile <<'EOF'
path := $(patsubst %/,%,$(dir $(abspath $(lastword $(MAKEFILE_LIST)))))
cwd  := $(shell pwd)

all:
    @echo "MAKEFILE_LIST: $(MAKEFILE_LIST)"
    @echo "         path: $(path)"
    @echo "          cwd: $(cwd)"
    @echo ""
EOF

# See/debug each command
set -x

# Test using the Makefile in the current directory
cd $proj_dir
make

# Test passing a Makefile
cd $temp_dir
make -f $proj_dir/Makefile

# Cleanup
rm -rf $temp_dir

输出:

+ cd /tmp/makefile_path_test/dir1/dir2/dir3
+ make
MAKEFILE_LIST:  Makefile
         path: /private/tmp/makefile_path_test/dir1/dir2/dir3
          cwd: /tmp/makefile_path_test/dir1/dir2/dir3

+ cd /tmp/makefile_path_test
+ make -f /tmp/makefile_path_test/dir1/dir2/dir3/Makefile
MAKEFILE_LIST:  /tmp/makefile_path_test/dir1/dir2/dir3/Makefile
         path: /tmp/makefile_path_test/dir1/dir2/dir3
          cwd: /tmp/makefile_path_test

+ rm -rf /tmp/makefile_path_test

注意:函数 $(patsubst %/,%,[path/goes/here/]) 用于去除尾部斜杠。


J
Jesse Chisholm

我尝试了许多这样的答案,但是在我的 AIX 系统上使用 gnu make 3.80 我需要做一些老派的事情。

原来 lastwordabspathrealpath 直到 3.81 才添加。 :(

mkfile_path := $(word $(words $(MAKEFILE_LIST)),$(MAKEFILE_LIST))
mkfile_dir:=$(shell cd $(shell dirname $(mkfile_path)); pwd)
current_dir:=$(notdir $(mkfile_dir))

正如其他人所说,它不是最优雅的,因为它两次调用 shell,并且仍然存在空格问题。

但由于我的路径中没有任何空格,因此无论我如何开始 make,它都适用于我:

make -f ../无论在哪里/makefile

make -C ../无论在哪里

make -C ~/无论在哪里

cd ../任何地方;制作

全部给我 wherevercurrent_dir 和到 wherever 的绝对路径为 mkfile_dir


一方面,我在 aix (5-6-7) 上编译 gnu-make-3.82 没有问题。
是的,在 3.81 之前,实现了早期解决方案所需的功能。我的答案是针对 3.80 或更早版本的旧系统。遗憾的是,在工作中,我不允许向 AIX 系统添加或升级工具。 :(
k
kenorb

这是使用 shell 语法获取 Makefile 文件的绝对路径的单行代码:

SHELL := /bin/bash
CWD := $(shell cd -P -- '$(shell dirname -- "$0")' && pwd -P)

这是基于 @0xff answer 的不带外壳的版本:

CWD := $(abspath $(patsubst %/,%,$(dir $(abspath $(lastword $(MAKEFILE_LIST))))))

通过打印来测试它,例如:

cwd:
        @echo $(CWD)

至关重要!上面的第二个示例需要 := 赋值运算符,否则复杂的 makefile 可能会发生一些非常奇怪和愤怒的事情(相信我)
第一个是重复错误,不适用于树外构建。
p
parasrish

示例供您参考,如下:

文件夹结构可能如下:

https://i.stack.imgur.com/1vdmj.png

其中有两个 Makefile,每个如下;

sample/Makefile
test/Makefile

现在,让我们看看 Makefile 的内容。

样本/Makefile

export ROOT_DIR=${PWD}

all:
    echo ${ROOT_DIR}
    $(MAKE) -C test

测试/生成文件

all:
    echo ${ROOT_DIR}
    echo "make test ends here !"

现在,执行示例/Makefile,如下所示;

cd sample
make

输出:

echo /home/symphony/sample
/home/symphony/sample
make -C test
make[1]: Entering directory `/home/symphony/sample/test'
echo /home/symphony/sample
/home/symphony/sample
echo "make test ends here !"
make test ends here !
make[1]: Leaving directory `/home/symphony/sample/test'

解释是,父/主目录可以存储在环境标志中,并且可以导出,以便可以在所有子目录makefile中使用。


这将获取当前工作目录,而不是 makefile 的主目录。
这些是 Makefile 中的行。现在,随着 Makefile 命令的执行,这将获得 Makefile 所在的路径。我理解“makefile 主目录”指的是相同的,即 Makefile 正在执行的目录路径。
@Jesse Chisholm 请重新访问。我已经解释了用法。
一个重复的错误:它不适用于树外构建。此外,您的 gnome 终端提供了一种复制文本的简单方法:只需选择它。有传言说 Imgur 破产了。
S
Soma Bini

在此处找到解决方案:https://sourceforge.net/p/ipt-netflow/bugs-requests-patches/53/

解决方案是:$(CURDIR)

你可以这样使用它:

CUR_DIR = $(CURDIR)

## Start :
start:
    cd $(CUR_DIR)/path_to_folder

欢迎来到堆栈溢出。您可以在答案中添加更多细节吗?该问题指出 $(CURDIR) 已尝试失败。
警告,快速谷歌用户:这不是 makefile 所在的目录,而是当前working 目录(启动 make 的位置)。这确实是 OP 真正想要的,但问题标题是虚假的,所以问题本身是不一致的。因此,这是对不正确问题的正确答案。 ;)
这是完全错误的,而且是多余的错误答案
P
Peter Frost

据我所知,这是这里唯一适用于空格的答案:

space:= 
space+=

CURRENT_PATH := $(subst $(lastword $(notdir $(MAKEFILE_LIST))),,$(subst $(space),\$(space),$(shell realpath '$(strip $(MAKEFILE_LIST))')))

它本质上是通过将 ' ' 替换为 '\ ' 来转义空格字符来工作的,这允许 Make 正确解析它,然后它通过执行另一个替换来删除 MAKEFILE_LIST 中的 makefile 的文件名,这样你就剩下那个目录了makefile 在里面。这不是世界上最紧凑的东西,但它确实有效。

你最终会得到这样的东西,其中所有的空间都被转义:

$(info CURRENT_PATH = $(CURRENT_PATH))

CURRENT_PATH = /mnt/c/Users/foobar/gDrive/P\ roje\ cts/we\ b/sitecompiler/

z
zhukunqian

更新 2018/03/05 finnaly 我用这个:

shellPath=`echo $PWD/``echo ${0%/*}`

# process absolute path
shellPath1=`echo $PWD/`
shellPath2=`echo ${0%/*}`
if [ ${shellPath2:0:1} == '/' ] ; then
    shellPath=${shellPath2}
fi

它可以在相对路径或绝对路径中正确执行。由 crontab 调用执行正确。在其他 shell 中执行正确。

显示示例,a.sh 打印自身路径。

[root@izbp1a7wyzv7b5hitowq2yz /]# more /root/test/a.sh
shellPath=`echo $PWD/``echo ${0%/*}`

# process absolute path
shellPath1=`echo $PWD/`
shellPath2=`echo ${0%/*}`
if [ ${shellPath2:0:1} == '/' ] ; then
    shellPath=${shellPath2}
fi

echo $shellPath
[root@izbp1a7wyzv7b5hitowq2yz /]# more /root/b.sh
shellPath=`echo $PWD/``echo ${0%/*}`

# process absolute path
shellPath1=`echo $PWD/`
shellPath2=`echo ${0%/*}`
if [ ${shellPath2:0:1} == '/' ] ; then
    shellPath=${shellPath2}
fi

$shellPath/test/a.sh
[root@izbp1a7wyzv7b5hitowq2yz /]# ~/b.sh
/root/test
[root@izbp1a7wyzv7b5hitowq2yz /]# /root/b.sh
/root/test
[root@izbp1a7wyzv7b5hitowq2yz /]# cd ~
[root@izbp1a7wyzv7b5hitowq2yz ~]# ./b.sh
/root/./test
[root@izbp1a7wyzv7b5hitowq2yz ~]# test/a.sh
/root/test
[root@izbp1a7wyzv7b5hitowq2yz ~]# cd test
[root@izbp1a7wyzv7b5hitowq2yz test]# ./a.sh
/root/test/.
[root@izbp1a7wyzv7b5hitowq2yz test]# cd /
[root@izbp1a7wyzv7b5hitowq2yz /]# /root/test/a.sh
/root/test
[root@izbp1a7wyzv7b5hitowq2yz /]# 

老:我用这个:

MAKEFILE_PATH := $(PWD)/$({0%/*})

如果在其他 shell 和其他目录中执行,它可以显示正确。


“如果执行其他 shell 和其他目录,它可以显示正确”在英语中没有意义。你能再解释一下吗?
对不起我糟糕的英语
感谢您的编辑,我知道您现在的意思是 IN。在我尝试之前你能解释一下这是做什么的吗?我不确定如何使用它。就像你所说的“其他外壳”是什么意思
这个答案有很多错误的东西,不可能修复。我建议一个简单的删除。
T
Thach Van

Makefile 中的一行就足够了:

DIR := $(notdir $(CURDIR))


关注公众号,不定期副业成功案例分享
关注公众号

不定期副业成功案例分享

领先一步获取最新的外包任务吗?

立即订阅