makefile不同的代碼進(jìn)行不同的編譯,比如cuda代碼中想將cpp代碼和cu代碼分開(kāi)分別編譯成.o文件,再將.o文件編譯成一個(gè)哭文件或者可執(zhí)行文件,這時(shí)可能cpp和cuda的編譯器不一樣,那么就不能夠使用自動(dòng)推導(dǎo),當(dāng)然,也可以將兩個(gè)編譯器組合成一個(gè)編譯器,讓系統(tǒng)根據(jù)代碼自己去選擇相應(yīng)的編譯器,這里將的是如果想分開(kāi)的話,應(yīng)該怎么做?
首先,假設(shè)這里有3個(gè)cpp文件,2個(gè)cu文件:
a.cpp ,b.cpp,d.cppkernel1.cu,kernel2.cu首先獲取相應(yīng)的文件名稱以及去掉后綴后的文件名:
SRC :=$(wildcard *.cpp)OBJ :=$(SRC:%.cpp=%.o)NAM :=$(foreach src,$(SRC),$(eval NAM += $$(shell basename $(src) .cpp));)然后將不同的代碼進(jìn)行編譯
gen_cpp: $(foreach n, $(NAM), $(EXEC) $(HOST_COMPLIER) $(INCLUDES) $(ALL_CCFLAGS) -o ${n}.o -c $(n).cpp;)上面的代碼就會(huì)自動(dòng)生成相應(yīng)的.o文件 同樣,對(duì)cu代碼:
SRC_CU :=$(wildcard *.cu)OBJ_CU :=$(SRC_CU:%.cu=%.o)NAM_CU :=$(foreach src,$(SRC_CU),$(eval NAM_CU += $$(shell basename $(src) .cu));)#下面這句和上面這句一樣,不同的是,使用的是makefile自己的basename變量:#shell的basename是去掉路徑的前綴,一般結(jié)合pwd可以用來(lái)獲取目錄名,在后面加上.cu可以再去掉后綴#而makefile中的basename是用來(lái)去掉后綴的$(foreach src,$(SRC_CU),$(eval NAM_CU += $$(basename $(src)));)然后將不同的代碼進(jìn)行編譯
gen_cu: $(foreach n, $(NAM_CU), $(NVCC) $(INCLUDES) $(ALL_CCFLAGS) -o ${n}.o -c $(n).cu;)最后,完整的代碼:
.PHONY : gen_cpp gen_cu#注意,這里gen_cpp,gen_cu一定要先放在build的前面,因?yàn)閎uild會(huì)依賴這兩項(xiàng)生成的中間文件all: gen_cpp gen_cu buildSRC :=$(wildcard *.cpp)OBJ :=$(SRC:%.cpp=%.o)NAM :=$(foreach src,$(SRC),$(eval NAM += $$(shell basename $(src) .cpp));)gen_cpp: $(foreach n, $(NAM), $(EXEC) $(HOST_COMPLIER) $(INCLUDES) $(ALL_CCFLAGS) -o ${n}.o -c $(n).cpp;)SRC_CU :=$(wildcard *.cu)OBJ_CU :=$(SRC_CU:%.cu=%.o)NAM_CU :=$(foreach src,$(SRC_CU),$(eval NAM_CU += $$(shell basename $(src) .cu));)gen_cu: $(foreach n, $(NAM_CU), $(NVCC) $(INCLUDES) $(ALL_CCFLAGS) -o ${n}.o -c $(n).cu;)build : target.sotarget:$(gen_cpp) $(gen_cu) $(ALL_COMPLIER) $(INCLUDES) $(ALL_CCFLAGS) -o $@ -shared $^還有一種更簡(jiǎn)單的模式規(guī)則可以使用,相比上面,下面代碼的好處是,由于需要生成的目標(biāo)是實(shí)際存在的,因此,如果編譯時(shí)間較長(zhǎng)時(shí),上面這種方法由于是偽目標(biāo),每次都需要重新進(jìn)行編譯,會(huì)很耗時(shí),而下面這種,只會(huì)編譯更改后的源文件:
#注意,這里gen_cpp,gen_cu一定要先放在build的前面,因?yàn)閎uild會(huì)依賴這兩項(xiàng)生成的中間文件SRC :=$(wildcard *.cpp)OBJ :=$(SRC:%.cpp=%.o)SRC_CU :=$(wildcard *.cu)OBJ_CU :=$(SRC_CU:%.cu=%.o)all: $(OBJ_CU) $(OBJ) target.so$(OBJ_CU):%.o:%.cu $(NVCC_COMPLIER) $(INCLUDES) $(ALL_CCFLAGS) -o $@ $^$(OBJ):%.o:%.cpp $(HOST_COMPLIER) $(INCLUDES) $(ALL_CCFLAGS) -o $@ $^target.so:$(OBJ_CU) $(OBJ) $(ALL_COMPLIER) $(INCLUDES) $(ALL_CCFLAGS) -o $@ $^注意如果將上面的
$(OBJ_CU):%.o:%.cu $(NVCC_COMPLIER) $(INCLUDES) $(ALL_CCFLAGS) -o $@ $^$(OBJ):%.o:%.cpp $(HOST_COMPLIER) $(INCLUDES) $(ALL_CCFLAGS) -o $@ $^更改成
$(OBJ_CU):$(SRC_CU) $(NVCC_COMPLIER) $(INCLUDES) $(ALL_CCFLAGS) -o $@ $^$(OBJ):$(SRC) $(HOST_COMPLIER) $(INCLUDES) $(ALL_CCFLAGS) -o $@ $^看似是正確的,實(shí)際上會(huì)報(bào)錯(cuò)
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注