Fw: [問題] makefile的寫法問題
※ [本文轉錄自 Linux 看板 #1I5b1JOM ]
作者: SeamusBerloz (軒摩斯) 看板: Linux
標題: Re: [問題] makefile的寫法問題
時間: Fri Aug 23 01:54:25 2013
※ 引述《Zoxge (Zoxge)》之銘言:
: 如果有很多個.cpp檔
: 就必須要有每個.cpp對應的target
: 但問題來了,當.cpp檔有幾百幾千個,又想要能自行判斷每個檔案是否被改過
: 這樣寫makefile不就xxx.o這樣的target也得要寫幾百幾千個 = =
: 請問有比較輕鬆的寫法嗎?
: 謝謝大家
給你我的 makefile 架構原形範例,
拿去對照查書理解一下內容,也許你會找到比我更好的寫法。
使用方法是,將你的 .cpp 檔案列表,照這個格式放進原始碼串列中,
如果 .cpp 有引入 .h 檔案,這個 makefile 也會一併幫你檢查 .h。
# 檔案開始
# .cpp 原始碼檔案名稱串列
SOURCE_CPP_FILES =\
./Tools/tools.cpp\
./Modules/modules.cpp\
./main.cpp
# .cpp 目的碼檔案名稱串列
OUTPUT_OBJECT_OF_CPP_FILES = $(SOURCE_CPP_FILES:.cpp=.o)
# .cpp 相依性規則輸出檔案名稱串列
OUTPUT_DEPENDENCY_RULE_OF_CPP_FILES = $(SOURCE_CPP_FILES:.cpp=.dep)
# 總輸出目的碼資源檔案名稱串列
ALL_OBJECT_FILES = $(OUTPUT_OBJECT_OF_CPP_FILES)
# ---------- 目標檔案巨集定義 ----------
# 目標檔檔案名稱
ALL_TARGET = ./MyTarget
# ---------- 主要 make 規則 ----------
all: $(ALL_TARGET)
# ---------- 目的碼檔案 make 規則 ----------
objects: $(ALL_OBJECT_FILES)
# ---------- 目的碼檔案以及目標檔案 clean 規則 ----------
clean: cleantarget cleanobjects cleandep
# ---------- 目標檔案 clean 規則 ----------
cleantarget:
rm -f $(ALL_TARGET)
# ---------- 目的碼檔案 clean 規則 ----------
cleanobjects:
rm -f $(ALL_OBJECT_FILES)
# ---------- 相依性規則檔案 clean 規則 ----------
cleandep:
rm -f $(OUTPUT_DEPENDENCY_RULE_OF_CPP_FILES)
# ---------- 連結目的碼檔案產生目標 ----------
$(ALL_TARGET): $(ALL_OBJECT_FILES)
g++ -o $@ $(ALL_OBJECT_FILES)
# ---------- .cpp 原始碼檔案相依性規則之建造規則 ----------
%.dep: %.cpp
g++ -MM $< -MT $*.o > $@
# ---------- 編譯 .cpp 原始碼檔案產生目的碼檔 ----------
%.o: %.cpp
g++ -c -o $@ $<
# ---------- 載入建造好的相依性規則 ----------
-include $(OUTPUT_DEPENDENCY_RULE_OF_CPP_FILES)
# 檔案結束
沒錯,很空洞,這只是個架構,將你自己相關的編譯參數,代入修改一下吧。
重點是你一定要去好好理解一下使用萬用字元與隱含目標的好處。
當然,如果真的原始碼列表真的數量驚人,那就得靠外部方式處理,這再舉個例子,
去改寫一下 makefile,將開頭的 SOURCE_CPP_FILES 巨集定義清除,換成:
-include MySource.lst
然後,下指令:
echo "SOURCE_CPP_FILES =\\" > MySource.lst;\
find ./ -name "*.cpp" | awk '{ print $1"\\" }' >> MySource.lst
(指令只是拋磚引玉,列表最後一個檔案名稱結尾會多個 \ 符號,
留給你自己研究怎樣方便清除嘍)
--
※ 發信站: 批踢踢實業坊(ptt.cc)
◆ From: 123.240.167.99
※ 發信站: 批踢踢實業坊(ptt.cc)
※ 轉錄者: SeamusBerloz (123.240.167.99), 時間: 08/23/2013 13:59:46
推
08/24 08:24, , 1F
08/24 08:24, 1F
→
08/24 08:25, , 2F
08/24 08:25, 2F
→
08/24 08:26, , 3F
08/24 08:26, 3F
這一項就是用來建造你所謂數量龐大的 .cpp target 用的。
他會為每一個 .cpp 都建立一個 .dep 檔案,
你可以打開每個 .dep 檔案看看內容,裡面就是 .cpp 生成 .o 的 target rule。
→
08/24 08:27, , 4F
08/24 08:27, 4F
會的,誠如上述,每一個 .dep 中都紀錄了該 .cpp 的需求相依檔案,
相依檔案包含 .h 也列在內,只要任何一個檔案被改過,就會從新建造目標。
當然,也會重新建造 .dep,因為會確認你是否增減或修改 .cpp 中的 #include 敘述。
(08/26 補充)
但是若增減修改的是 *.h 中的 #include 敘述,
然而這很可能會超出 .dep 中的為你設立的相依性檢查規則範圍,而導致不重新編譯,
你就得在重新編譯前,先下達 make cleandep 了。
推
08/24 10:14, , 5F
08/24 10:14, 5F
→
08/24 10:22, , 6F
08/24 10:22, 6F
我承認,這個可能是這個 Makefile 的最大疏失問題,
雖說沒有什麼太大的問題,但是的確會在 .cpp 數量龐大的時候滿浪費時間的 ^^;。
或許經過這次研究你會有更好的寫法去精進這個 Makefile。
→
08/24 10:22, , 7F
08/24 10:22, 7F
事實上,重跑建造任何你修改過的 .cpp 去建造新 .o,
就是從重新建造 .dep 開始的,為的就是重新確認 .cpp 對所有檔案的相依性關係。
當然,整個個去除掉 .dep 那一段,可能一樣會跑,
但是就檢查不到相依的檔案是否有更動修改了。
※ 編輯: SeamusBerloz 來自: 123.240.167.99 (08/24 15:04)
推
08/26 00:42, , 8F
08/26 00:42, 8F
→
08/26 00:43, , 9F
08/26 00:43, 9F
→
08/26 00:43, , 10F
08/26 00:43, 10F
*[1;31m→ *[33mZoxge*[m*[33m:有點爛 XD 太複雜的用法我還是不太會啊 囧 *[m 08/26 00:43
(^^^^^ 這一行貼文被我不小心手賤弄壞了,不好意思)
不會阿,怎會爛?只要適合自己的,且邏輯沒問題,運作起來也正確,
語法又是自己所能理解與掌握的,就是好寫法!
而且,我還很謝謝你呢!
Why?因提到 *.dep 的清除,回想起當初我把 cleandep 放在 clean rule 中的目的,
是為了的確有可能因為原始碼當 #include 敘述被更動的並非是 *.cpp 中所有時,
且若 *.dep 不重建,進而有可能使 *.cpp 不重新編譯 (如同你前面貼文提到的疑慮)。
以下情節就有可能發生,假設原始碼有以下 #include 關係:
<< main.cpp >>
...
#include "global.h"
...
printf("MY_CONST_VALUE=%d", (int)MY_CONST_VALUE);
...
<< global.h >>
...
#include "my_test.h" // <-- 新加入這一行
...
#ifndef MY_CONST_VALUE
MY_CONST_VALUE 0
#endif
...
建立新檔案:
<< my_test.h >>
#define MY_CONST_VALUE 1
這時若又忘了去重建 main.dep,嘿嘿...
不管 my_test.h 中的 MY_CONST_VALUE 的值怎樣設定,MY_CONST_VALUE 都只會印出零。
當然這其實是程式設計師的錯啦,但是若執行編譯者又另有他人,這恐怕是場災難。
於是從此,每次我大擴展程式,都習慣先下 make clean cleandep 通通輕乾淨,
再來才是 make,為了省打 cleandep 就順手把 cleandep 給放入了 clean 規則裡。
但自食惡果,小修改若不含原始碼檔案相依變動,搞的怕先下 make clean,
頂多下個 make cleanobjects,或僅僅只有直接下 make,
因為我也會被建立 *.dep 給煩死...,當然誤人的地方,也深感抱歉 -___-"。
然而,這次見識到你的 ifneq 敘述,我終於可以將 cleandep 放回他的原位了!
既然要清除 *.dep,就不要再次白痴的去 include *.dep,我怎沒想到呢,呵呵~
我這裡加入除了你的 ifneq 敘述,還將 clean: 那行中,把 cleandep 去掉了,
這表示當下達執行:
~$ make clean
工作目標將不再包含清除 *.dep,當若需要完全清除 *.dep 得靠下達執行:
~$ make cleandep
於是搭配你的 ifneq 包覆 -include 敘述的寫法,改成:
ifneq ($(MAKECMDGOALS), cleandep)
-include $(OUTPUT_DEPENDENCY_RULE_OF_CPP_FILES)
endif
真是教學相長,再次感謝!
※ 編輯: SeamusBerloz 來自: 123.240.167.99 (08/26 06:33)
推
08/27 00:11, , 11F
08/27 00:11, 11F
→
08/27 00:12, , 12F
08/27 00:12, 12F
→
08/27 00:12, , 13F
08/27 00:12, 13F
→
08/27 00:14, , 14F
08/27 00:14, 14F
→
08/27 00:15, , 15F
08/27 00:15, 15F
推
08/27 00:18, , 16F
08/27 00:18, 16F
→
08/27 00:19, , 17F
08/27 00:19, 17F
→
08/27 00:21, , 18F
08/27 00:21, 18F
→
08/27 00:21, , 19F
08/27 00:21, 19F
→
08/27 00:21, , 20F
08/27 00:21, 20F
→
08/27 00:23, , 21F
08/27 00:23, 21F
討論串 (同標題文章)