Как я могу автоматически создать (и удалить) временный каталог в Makefile?

Возможно ли иметь make создать временный каталог до того, как он выполнит первую цель? Может быть, использовать какой-нибудь хак, дополнительную цель и т. Д.

Все команды в Makefile смогут ссылаться на автоматически созданный каталог как $TMPDIRи каталог будет автоматически удален, когда make Команда заканчивается.

4 ответа

Решение

Кажется, я помню рекурсивную возможность вызова make, что-то вроде:

all:
    -mkdir $(TEMPDIR)
    $(MAKE) $(MLAGS) old_all
    -rm -rf $(TEMPDIR)

old_all: ... rest of stuff.

Я сделал аналогичные трюки для создания в подкаталогах:

all:
    @for i in $(SUBDIRS); do \
        echo "make all in $$i..."; \
        (cd $$i; $(MAKE) $(MLAGS) all); \
    done

Только что проверил, и это прекрасно работает:

$ cat Makefile
all:
    -mkdir tempdir
    -echo hello >tempdir/hello
    -echo goodbye >tempdir/goodbye
    $(MAKE) $(MFLAGS) old_all
    -rm -rf tempdir

old_all:
    ls -al tempdir

$ make all
mkdir tempdir
echo hello >tempdir/hello
echo goodbye >tempdir/goodbye
make  old_all
make[1]: Entering directory '/home/pax'
ls -al tempdir
total 2
drwxr-xr-x+ 2 allachan None 0 Feb 26 15:00 .
drwxrwxrwx+ 4 allachan None 0 Feb 26 15:00 ..
-rw-r--r--  1 allachan None 8 Feb 26 15:00 goodbye
-rw-r--r--  1 allachan None 6 Feb 26 15:00 hello
make[1]: Leaving directory '/home/pax'
rm -rf tempdir

$ ls -al tempdir
ls: cannot access tempdir: No such file or directory

Эти предыдущие ответы либо не работали, либо казались слишком сложными. Вот гораздо более простой пример, который мне удалось выяснить:

PACKAGE := "audit"
all:
    $(eval TMP := $(shell mktemp -d))
    @mkdir $(TMP)/$(PACKAGE)
    rm -rf $(TMP)

С GNU сделай, как минимум,

TMPDIR := $(shell mktemp -d)

получите ваш временный каталог. Я не могу придумать хороший способ очистить его в конце, кроме очевидного rmdir "$(TMPDIR)" как часть all цель.

См. Получение имени make-файла из make-файла для $(self) трюк

ifeq ($(tmpdir),)

location = $(CURDIR)/$(word $(words $(MAKEFILE_LIST)),$(MAKEFILE_LIST))
self := $(location)

%:
    @tmpdir=`mktemp --tmpdir -d`; \
    trap 'rm -rf "$$tmpdir"' EXIT; \
    $(MAKE) -f $(self) --no-print-directory tmpdir=$$tmpdir $@

else
# [your real Makefile]
%:
    @echo Running target $@ with $(tmpdir)
endif
Другие вопросы по тегам