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