八位微控制器的代碼優(yōu)化技巧
全局變量更好用
將參數(shù)傳遞給函數(shù)是一個(gè)很好的代碼經(jīng)驗(yàn)。在 C 程序中,編譯器可絕對(duì)確保調(diào)用的子程序不會(huì)修改參數(shù)。編譯器可處理存儲(chǔ)器管理的問(wèn)題。不過(guò),這將占用難以承受的大量時(shí)間和空間。試考慮下面這段代碼:
Main()
{
Int effectiveGlobal;
Foo(effectiveGlobal)
}
由于變量在 main() 中已經(jīng)聲明,因此該變量與真正的全局變量之間的真正差別是命名空間 (namespace)。但是,每次調(diào)用 foo() 時(shí),編譯器都必須在新的位置存儲(chǔ) effectiveGlobal。聲明真正的全局變量有助于降低因調(diào)用而造成的代碼和數(shù)據(jù)開銷。
向編譯器提供盡可能多的信息
8051 可提供 64K 的地址空間 XDATA、256 字節(jié)的堆棧與間接尋址空間 IDATA 以及 256 字節(jié)的直接尋址空間 DATA 等多個(gè)存儲(chǔ)器空間。在大多數(shù)情況下,代碼編寫人員都知道指針指向了哪個(gè)存儲(chǔ)器空間。如果用戶指定了存儲(chǔ)器空間,編譯器就無(wú)需包含對(duì)例程中的所有三類存儲(chǔ)器進(jìn)行尋址的代碼,只需使用一個(gè)即可。由于指針無(wú)需包含數(shù)據(jù)空間信息,因此有助于節(jié)約數(shù)據(jù)空間。
在我的 8051 編譯器中,上述變量可通過(guò)包含 OPTR 字符串的庫(kù)例程進(jìn)行存取。在列表和庫(kù)文件中搜索對(duì)OPTR的引用可以發(fā)現(xiàn)長(zhǎng)變量被多次使用,而且由于在代碼中假定了指針的大小,其中某些長(zhǎng)變量還會(huì)導(dǎo)致一些問(wèn)題。
在變量聲明中使用 const 關(guān)鍵詞可以實(shí)現(xiàn)兩方面的優(yōu)化:第一,編譯器不必再存儲(chǔ)變量的初始值;第二,編譯器能在編譯時(shí)間而非執(zhí)行時(shí)間執(zhí)行一些數(shù) 學(xué) 運(yùn) 算。查看示例程序的編譯輸出,以確定對(duì) const與 #define 的處理是否真的一樣。以下是我對(duì)代碼的測(cè)試:
經(jīng)過(guò)測(cè)試,得到以下輸出,表明它并不清楚 const 變量的值。
匯編語(yǔ)言
不少嵌入式固件工程師信誓旦旦的表示他們始終能比編譯器做得更好,不僅如此,他們還認(rèn)為應(yīng)該使用匯編語(yǔ)言重新編寫所有代碼。然而事實(shí)上,現(xiàn)代編譯器提供的許多特性已經(jīng)能趕上人腦的水平了。
變量共享:一些 8 位處理器尚無(wú)有效的機(jī)制來(lái)存取堆棧上的變量。一般的解決方案是創(chuàng)建調(diào)用樹,并在相互不進(jìn)行調(diào)用的函數(shù)間共享變量。在匯編程序中要想保持這種結(jié)構(gòu)相當(dāng)困難,且容易出錯(cuò)。
可靠性:任何從事專業(yè)軟件或固件開發(fā)工作的人員都能讀懂 C 語(yǔ)言程序。如果您需要將代碼交給其它開發(fā)人員處理,他們無(wú)需掌握那些為發(fā)揮匯編語(yǔ)言的最大效率而需要的所有技巧便可立即開始修改代碼。
可移植性:C 語(yǔ)言最初的開發(fā)目的之一就是要提供一種非常抽象,以便可以在多種處理器上應(yīng)用的語(yǔ)言。這一目標(biāo)至今仍然非常重要。
代碼共享:許多 8 位編譯器都能在鏈接時(shí)間之后進(jìn)行優(yōu)化,這使得編譯器不僅能執(zhí)行許多人工能完成的優(yōu)化,而且還能完成一些人工所不能完成的優(yōu)化。例如,現(xiàn)在許多編譯器都能搜索不同函數(shù) 中 共 有的代碼字符串,并將其合并為一個(gè)新的函數(shù)。而人類是不可能記住每個(gè)編譯周期中執(zhí)行此函數(shù)所需要的全部細(xì)節(jié)的。
匯編語(yǔ)言現(xiàn)在仍占有一席之地。不過(guò),在使用匯編語(yǔ)言之前應(yīng)首先考慮上述所有因素。
結(jié)論
在撰寫本文的過(guò)程中,我將成熟程序的大小從 0x6000 多字節(jié)縮減到了 0x5f2b 字節(jié),節(jié)約了 200 多字節(jié)。該程序過(guò)去曾是多次試圖優(yōu)化程序大小的目標(biāo)。
評(píng)論