gpt4 book ai didi

makefile - 使用Gnu Make编译ada项目需要gnatprep和gnatchop

转载 作者:行者123 更新时间:2023-12-02 16:18:04 25 4
gpt4 key购买 nike

我一直在尝试将多个大型 ada 编译从基于脚本的方法转换为使用 gnu make 3.82 makefile 并且可以使用一些经验丰富的知识。

一些背景:

  • Red Hat Enterprise Linux 7.9 版上的 GNAT 4.8.5
  • 每次编译都以约 1000 个 .ada 文件的源列表开始
  • 不同的版本使用一些相同的文件,在每个源列表中的大约 14k 文件总数中只有大约 7000 个独特的文件
  • 每个文件都需要准备、切碎,然后编译 - 绑定(bind)和链接在别处完成

我的方法:

  • 方法 1:对于每个文件,将文件准备到 src/foo/prepped,然后切入 src/foo/chopped
  • 方法 2:将方法 1 中创建的每个切碎文件复制到模块 SRC/文件夹中
  • 方法 3:在 SRC/文件夹中编译方法 2 中的每个切碎文件,并将其放入 OBJ/文件夹中

问题:

  • 切碎阶段会导致问题,因为文件名已更改,并且在某些情况下会创建更多文件。为了解决这个问题,我尝试为每个 src/foo/prepped 文件夹的内容设置通配符,并将它们复制到 SRC/文件夹中进行编译。由于此列表在准备/印章阶段之前是未知的,因此需要再次调用 make 以获取这些文件名。
  • 在编译阶段,我盲目运行src/文件夹中的每个文件使用双冒号规则进行编译,因为我不确定编译阶段输出的文件(.ali或.o)
  • 如果一个文件已经从以前的编译中准备/切碎,它不会做双重准备/切碎工作,但不会将文件复制到 SRC/

问题:

  • 如何在事先不知道的情况下解释使用 gnatchop 输出的文件?
  • 在编译阶段,我如何编写一条规则来从 .adb 或 .ads 文件创建 .o 或 .ali 文件?
FILE:= 'input'  #read input from file
OBJECTS:=$(shell cat ${FILE}) #need to do this, no file function in gnu make 3.82
FPOBJECTS:= $(addprefix ../../src/, $(OBJECTS)) # path + source name from pwd
OBJFILE:=$(notdir $(OBJECTS)) #just get out source file name
PDIR:=$(addprefix ../../src/, $(addsuffix prepped/ , $(dir $(OBJECTS)))) #just get out directory part of objects, add prepped/ suffix and ../../src to beginning
CDIR:= $(addprefix ../../src/, $(addsuffix chopped/ , $(dir $(OBJECTS))))# same as PDIR but with chopped/ instead of prepped/
CDIR2:= $(sort $(CDIR)) # unique directories in CDIR for folder creation
PDIR2:= $(sort $(PDIR)) # unique directories in PDIR for folder creation
DIRS:= OBJ TMP SRC #directories inside lib/version*
POBJECTS:= $(basename $(join $(PDIR), $(OBJFILE))) # join prepped directory with objects
COBJECTS = $(addprefix OBJ/, $(notdir $(wildcard SRC/*))) # this is an incorrect way to generate the objects needed, but does allow me to pass each source file for compilation
GNATOPTIONS:= -c -O1 -w -fPIC -gnatE -gnatv -LSRC/ # gcc flags
CDIRwld:=$(shell echo $(addsuffix *, $(CDIR2))) #append * to each unique chopped directory, to try and get names of each chopped file
.RECIPEPREFIX = >

#compile creates the COBJECTS in the pwd and moves them where they belong
compile : $(COBJECTS)
>mv *.ali OBJ/
>mv *.o OBJ/

$(COBJECTS)::
>gcc $(GNATOPTIONS) SRC/$(@F)

copy : $(CDIRwld)
>cp $? SRC/

prep : $(POBJECTS)

$(POBJECTS):$(FPOBJECTS) | $(DIRS) $(PDIR2) $(CDIR2)
>gnatprep -cru -Dlil $(addsuffix .ada, $(subst prepped/,,$@)) $(basename $@)
>gnatchop -rw $@ $(subst prepped/,chopped/,$(dir $@))

$(DIRS):
>mkdir -p $@

$(PDIR2):
>mkdir -p $@

$(CDIR2):
>mkdir -p $@

.PHONY: clean move

clean :
>rm -f -r $(DIRS)
>rm -f -r $(PDIR2)
>rm -f -r $(CDIR2)

move :
>mv *.ali OBJ/
>mv *.o OBJ/

最佳答案

准备好的文件和切碎的文件之间存在一对一的对应关系。

如果您可以让 GNAT 知道它何时需要编译规范,并了解依赖关系,尤其是内联和通用主体等不明显的依赖关系,那将是个好主意。

一个非常简单的 makefile(适用于 Make 3.81)可能是:

SOURCES = foo.ada bar.ada
DEFS = defs.def

CHOPPED_STAMPS = $(addsuffix .chopped, $(SOURCES))

compile: $(CHOPPED_STAMPS)
cd obj; gnatmake -c ../chopped/*

%.prepped: %
gnatprep $< prepped/ $(DEFS)
touch $@

%.chopped: %.prepped
gnatchop -w prepped/$(basename $<) chopped/
touch $@

它的缺点是在当前目录中乱放 .prepped.chopped 标记文件,但可以将它们放入子目录中。

一个更糟糕的问题是 gnatprep 阶段依赖于定义文件,我还没有弄清楚如何将这种依赖性引入隐式规则。

我是用gnatmake编译的,总的来说还是用gprbuild比较好,如果你有的话,还有一个库项目。 GPRBuild RM有点挑战,而且该链接指向当前版本,因此可能会缺少一些功能。

关于makefile - 使用Gnu Make编译ada项目需要gnatprep和gnatchop,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/66143942/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com