keil MDK開發(fā)STM32使用內(nèi)聯(lián)函數(shù)
掃描二維碼
隨時(shí)隨地手機(jī)看文章
筆者在做移植時(shí),將Embest IDE環(huán)境下的例程移到REALVIEWMDK的過(guò)程中,曾經(jīng)遇到這樣一個(gè)問題:在生成工程時(shí),編譯全部通過(guò),但在鏈接時(shí)提示許多符號(hào)未定義!如果讀者也遇到過(guò)這個(gè)問題,請(qǐng)繼續(xù)看下去,如果鏈接時(shí)提示未定義的變量是一些內(nèi)聯(lián)函數(shù)(即使用了關(guān)鍵字__inline)。那么就是筆者遇到的問題了。希望這篇文章對(duì)您有所幫助。
在MDK中使用關(guān)鍵字__inline時(shí),除了執(zhí)行速度和占用空間和普通函數(shù)不一樣之外,還有以下與普通函數(shù)不一樣的地方:表現(xiàn)為,不能將其申明為外部函數(shù),即不能將__inline函數(shù)定義在一個(gè)外部文件中然后在引用的文件中申明為extern類型。
原因是MDK中使用的__inline函數(shù)時(shí)和標(biāo)準(zhǔn)C++中的inline函數(shù)有相同的語(yǔ)義。
在C++標(biāo)準(zhǔn)中,一個(gè)內(nèi)聯(lián)函數(shù)在用到它的每個(gè)解釋單元需要相同的定義,這樣看來(lái)只有內(nèi)部鏈接的才可能內(nèi)聯(lián)。將外部文件中的函數(shù)鏈接并內(nèi)聯(lián)似乎不可能,特別是外部函數(shù)是非內(nèi)聯(lián)的。為了在不同文件之間的函數(shù)可以內(nèi)聯(lián),必須:
將這些函數(shù)放在一個(gè)通用的頭文件中,例如foo.h;
將這些函數(shù)標(biāo)記為extern __inline;
在需要該內(nèi)聯(lián)函數(shù)時(shí),#include內(nèi)聯(lián)函數(shù)所在的頭文件。
當(dāng)編譯器決定不內(nèi)聯(lián)某個(gè)函數(shù)時(shí),在編譯鏈接之后該函數(shù)只有一個(gè)拷貝。注:根據(jù)筆者的實(shí)驗(yàn)發(fā)現(xiàn)也可以將內(nèi)聯(lián)函數(shù)直接定義為__inline。
在MDK中如果將__inline函數(shù)申明為extern的話,在編譯時(shí)會(huì)通過(guò),而在鏈接時(shí)會(huì)說(shuō)這些函數(shù)沒定義。這點(diǎn)與Embest IDE不同。Embest IDE允許將內(nèi)聯(lián)函數(shù)在外部文件中定義,在引用的文件中,只須申明為extern類型即可通過(guò)編譯鏈接。
解決辦法,將內(nèi)聯(lián)函數(shù)定義在.h文件中,在需要引用的文件中包含該頭文件。這樣就可以調(diào)用頭文件中的內(nèi)聯(lián)函數(shù)了。并且能正確通過(guò)編譯、鏈接。
其實(shí)在MDK中AT91RM9200的標(biāo)準(zhǔn)庫(kù)中也是這么做的,AT91RM9200的標(biāo)準(zhǔn)庫(kù)有兩個(gè).h文件――AT91RM9200.h和lib_AT91RM9200.h,其中AT91RM9200.h中定義一些常量和外圍接口結(jié)構(gòu)體。其中l(wèi)ib_AT91RM9200.h中存放的就是庫(kù)函數(shù),其庫(kù)函數(shù)均為內(nèi)聯(lián)函數(shù)。
注:以上兩個(gè)頭文件存放的路徑為C:KeilARMINCAtmelRM9200。(其中MDK安裝路徑為C:Keil)
你用 __inline就行了,要注意,是小寫的,這是標(biāo)準(zhǔn)的C語(yǔ)言寫法,而你看到的 __INLINE,是STM32的庫(kù)進(jìn)行的宏定義,不是標(biāo)準(zhǔn)的C語(yǔ)言。
那個(gè)宏定義在 core_m3.c文件里,只有在那個(gè)文件,__INLINE才起作用。
#elif defined ( __TASKING__ )
#define __ASM __asm /*!< asm keyword for TASKING Compiler */
#define __INLINE inline /*!< inline keyword for TASKING Compiler */
#endif
這就是那個(gè)C文件里的宏定義。
在IAR中的寫法如下:
#pragma inline
int GetTwoByte(const char *cmd,int pos)
{
return ((((int)*(cmd+pos))<<8)|(((int)*(cmd+pos+1))<<0));
}
在Keil中的寫法如下:
__inline int GetTwoByte(const char *cmd,int pos)
{
return ((((int)*(cmd+pos))<<8)|(((int)*(cmd+pos+1))<<0));
}