靈活使用ARM匯編的WEAK關(guān)鍵字
//=====================================================================
//TITLE:
// 靈活使用ARM匯編的WEAK關(guān)鍵字
//AUTHOR:
// norains
//DATE:
// Tuesday 20-October-2010
//Environment:
// KEIL MDK 4.0
//=====================================================================
ARM匯編中的WEAK關(guān)鍵字是一個(gè)很有意思的功能,如果能夠靈活使用,能減輕不少繁瑣。一般來說,這個(gè)關(guān)鍵字使用在IMPORT和EXPORT這兩個(gè)聲明段。
如果我們有一個(gè)名為ARM_Vectors的量表,向量表的第一個(gè)數(shù)值指向一個(gè)StackTop函數(shù)的地址。因?yàn)槲覀兛赡芏x了該函數(shù),也可能沒有,為了代碼的簡(jiǎn)便,我們可以使用WEAK關(guān)鍵字,如:
////////////////////////////////////////////////////////////////////////////
//VectorsTrampolines.s
////////////////////////////////////////////////////////////////////////////
IMPORT StackTop [WEAK]
AREA |.text|, CODE, READONLY
;Vector list
ARM_Vectors
DCD StackTop
當(dāng)我們定義了StackTop函數(shù)后,那么ARM_Vectors里第一個(gè)向量值就是StackTop函數(shù)的地址。如果沒有定義StackTop,那么編譯器不會(huì)報(bào)錯(cuò),而這時(shí)候第一個(gè)向量值就直接賦予0。
那么對(duì)于EXPORT,WEAK又有什么樣的功能呢?如果你EXPORT的函數(shù)帶有WEAK標(biāo)志的話,并且別的源代碼沒有定義同名函數(shù),那么連接時(shí)就是該函數(shù);否則,就是另外的一個(gè)同名函數(shù)。這個(gè)機(jī)制,和類的繼承有點(diǎn)相像,都是一個(gè)函數(shù)將另一個(gè)函數(shù)給掩蓋了;所不同的是,WEAK里的這個(gè)掩蓋,是徹徹底底讓另外一個(gè)函數(shù)消失。
可能這樣說還是有點(diǎn)不太明白,我們以實(shí)例來說明:
////////////////////////////////////////////////////////////////////////////
//VectorsTrampolines.s
////////////////////////////////////////////////////////////////////////////
IMPORT StackTop
AREA |.text|, CODE, READONLY
;Vector list
ARM_Vectors
DCD StackTop
////////////////////////////////////////////////////////////////////////////
//VectorsHandlers.s
////////////////////////////////////////////////////////////////////////////
EXPORT StackTop [WEAK]
AREA |i.DefaultHandler|, CODE, READONLY
StackTop PROC
B .
ENDP
雖然這時(shí)候StackTop在通過EXPORT導(dǎo)出時(shí)帶有WEAK關(guān)鍵字,但因?yàn)檎麄€(gè)源代碼文件中只有這里有StackTop,所以VectorsTrampolines.s文件中連接的StackTop是VectorsHandlers.s定義的同名函數(shù)。
如果另外的源代碼也定義了同名的函數(shù),如:
////////////////////////////////////////////////////////////////////////////
//VectorsTrampolines.s
////////////////////////////////////////////////////////////////////////////
IMPORT StackTop
AREA |.text|, CODE, READONLY
;Vector list
ARM_Vectors
DCD StackTop
////////////////////////////////////////////////////////////////////////////
//VectorsHandlers.s
////////////////////////////////////////////////////////////////////////////
EXPORT StackTop [WEAK]
AREA |i.DefaultHandler|, CODE, READONLY
StackTop PROC
B .
ENDP
////////////////////////////////////////////////////////////////////////////
//Func.c
////////////////////////////////////////////////////////////////////////////
Extern “C” StackTop()
{}
因?yàn)閂ectorsHandlers.s的StackTop是用WEAK導(dǎo)出的,而Func.c又有同名的函數(shù),那么這時(shí)候VectorsTrampolines.s里的StackTop是連接Func.c里定義的StackTop。
這里有一個(gè)很有意思的問題,如果EXPORT和IMPORT都用WEAK聲明,如:
////////////////////////////////////////////////////////////////////////////
//VectorsTrampolines.s
////////////////////////////////////////////////////////////////////////////
IMPORT StackTop [WEAK]
AREA |.text|, CODE, READONLY
;Vector list
ARM_Vectors
DCD StackTop
////////////////////////////////////////////////////////////////////////////
//VectorsHandlers.s
////////////////////////////////////////////////////////////////////////////
EXPORT StackTop [WEAK]
AREA |i.DefaultHandler|, CODE, READONLY
StackTop PROC
B .
ENDP
這時(shí)候VectorsTrampolines.s里的ARM_Vectors的第一個(gè)向量值是什么呢?還是VectorsHandlers.s里的StackTop函數(shù)地址么?很遺憾,不是,而直接是0!所以這點(diǎn)就注意了,最好不要對(duì)同名的函數(shù)在IMPORT和EXPORT時(shí)都使用WEAK,否則結(jié)果很可能讓你抓狂!
最后,以一個(gè)列表作為總結(jié):
VectorsTrampolines.s
VectorsHandlers.s
Func.c
ARM_Vectors第一個(gè)向量值
IMPORT StackTop [WEAK]
無
無
0
IMPORT StackTop [WEAK]
EXPORT StackTop
無
VectorsHandlers.s的StackTop地址
IMPORT StackTop
EXPORT StackTop [WEAK]
有同名的StackTop函數(shù)
Func.c的StackTop地址
IMPORT StackTop
EXPORT StackTop
有同名的StackTop函數(shù)
存在兩個(gè)StackTop,編譯出錯(cuò)
IMPORT StackTop [WEAK]
EXPORT StackTop [WEAK]
無
0