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