函數指針在C語言中應用較為靈活。在單片機系統(tǒng)中,嵌入式操作系統(tǒng)、文件系統(tǒng)和網絡協議棧等一些較為復雜的應用都大量地使用了函數指針。Keil公司推出的C51編譯器是事實上80C51 C編程的工業(yè)標準,它針對8051系列CPU硬件在標準ANSI C的基礎上進行了擴展;但由于編譯器及8051體系結構的限制,造成了在使用函數指針時有很多與ANSI C不同的地方。下面舉例說明在不同的情形下函數指針的使用。以下代碼均在Keil μVision3、v8.08 C51、默認優(yōu)化等級的開發(fā)環(huán)境下驗證通過。
1、指向固定地址的指針
在程序設計中,常需要跳轉到某一特定的地址上執(zhí)行,如引導程序的設計。可通過如下C語言實現:
intmain(void){((void(code*)(void))0x2000)();return0;} 此代碼使得主函數執(zhí)行位于0x2000地址的程序代碼。其中( (void (code* )(void) )是一種數據類型,表示一指向代碼段函數的指針,該函數無參數和無返回值。它對數據0x2000進行了強制類型轉換,使函數指針指向地址為0x2000的代碼段地址。關于復雜類型的聲明詳見參考文獻[1]。
通過反匯編窗口可看到編譯器生成了如下匯編代碼:
C:0x000F122000LCALLC:2000
由上可以看出, Keil C51是非常高效的編譯器,產生了非常簡潔的輸出。這正是我們所期望的。
2、無參數的函數指針
Keil C51中不帶參數的函數指針的使用方法與ANSI C基本相同。示例如下:
voidfoo(void){return;}intmain(void){void(*pfoo)(void);//申明函數指針pfoopfoo=foo;//對該指針賦值,指針指向foo函數代碼段(*pfoo)();//通過指針調用其指向的函數,就是運行foo函數return0;} 3、帶參數的函數指針
一般來說,函數參數是通過堆棧來傳遞,用PUSH和POP匯編指令來實現的;但由于8051體系及其編譯器的一些限制,使得其函數參數的傳遞需要一些特殊的方法。
通過函數指針調用函數屬于函數的間接調用,根據C51的規(guī)定,所有的函數參數都需要通過寄存器傳遞。由于8051的寄存器數目的限制,函數指針最多只能傳遞3個參數(具體傳遞規(guī)則詳見參考文獻[2])。其聲明與調用方式如下:
void(*pfun)(char,short,int);//申明函數指針(*pfun)('c',0x1234,0x5678);//調用改函數 如果需要傳遞3個以上函數的參數,可以把參數存放到結構體[1]里面,再用一個指針指向該結構體作為參數傳遞給函數指針。也可以使用reentrant關鍵字將函數聲明為可重入函數[2]。
4、分析調用樹正確使用指針函數
Keil C51編譯器與ANSC C編譯器的區(qū)別之一是,它并不把函數參數壓入堆棧中,而是把函數參數放在寄存器(register)或固定的內存位置(fixed memory location)[2]中。
調用樹(call tree)是由Keil鏈接器自動生成的,用于描述函數的調用關系。鏈接器通過分析調用樹來確定哪些寄存器或內存位置是可安全覆蓋的。這樣兩個不同時調用的函數就可以共享同一塊memory作為傳遞參數使用。但對于函數指針來說,編譯器并不知道函數指針將指向哪個函數。這導致了調用樹構造出錯的可能,函數的參數也可能被錯誤覆蓋。示例如下:
voidfoo_caller(int(code*fptr)(unsignedint)){unsignedchari;for(i=0;i<5;++i) (*fptr)(i);}intfoo(unsignedintcount){ longj,k; k=0; for(j=0;j 對工程“Build target”之后,打開該工程目錄下的M51文件查看代碼覆蓋及函數調用情況,如下:
OVERLAYMAPOFMODULE:test(?C_STARTUP)SEGMENTDATA_GROUP +﹥CALLEDSEGMENTSTARTLENGTH?C_C51STARTUP +﹥?PR?MAIN?MAIN?PR?MAIN?MAIN +﹥?PR?_FOO?MAIN +﹥?PR?_FOO_CALLER?MAIN?PR?_FOO?MAIN 0008H0008H?PR?_FOO_CALLER?MAIN0008H0003H
從該M51文件可以看出,Keil C51編譯器認為main函數依次調用了foo與foo_caller函數。這顯然違反了上面C代碼的初衷,而且foo函數占用了0008H~0010H,foo_caller函數占用了0008H~000BH DATA區(qū),二者傳遞參數的區(qū)域相互覆蓋。通過Keil調試器可知,由于參數fptr被錯誤覆蓋,在第2次調用(*ftpr)()時,程序已經不能正確跳轉至foo函數執(zhí)行了。
顯然,造成上述結果的原因是生成的調用樹出錯了。Keil提供了鏈接器OVERLAY偽指令,可讓用戶自行修改調用樹,調整函數的調用關系??稍阪溄用钚休斎胍韵旅睿∣VERLAY指令的用法詳見參考文獻[2]):
OVERLAY(?PR?MAIN?MAIN~?PR?_FOO?MAIN,?PR?_FOO_CALLER?MAIN!?PR?_FOO?MAIN)
或在Keil集成開發(fā)環(huán)境中,在“BL51 Misc”-“Overlay”中填入:
?PR?MAIN?MAIN~?PR?_FOO?MAIN,?PR?_FOO_CALLER?MAIN!?PR?_FOO?MAIN
再次對工程“Build target”之后,M51文件片段如下所示:
OVERLAYMAPOFMODULE:test(?C_STARTUP)SEGMENTDATA_GROUP +﹥CALLEDSEGMENTSTARTLENGTH?C_C51STARTUP +﹥?PR?MAIN?MAIN?PR?MAIN?MAIN +﹥?PR?_FOO_CALLER?MAIN?PR?_FOO_CALLER?MAIN0008H0003H +﹥?PR?_FOO?MAIN?PR?_FOO?MAIN000BH0008H
對比之前生成的M51文件可看出,foo與foo_caller函數的DATA區(qū)不再重疊,調用樹正確地構造了。調試結果也顯示,輸出正如所期望的一樣。
結語:采用C51設計較為復雜的單片機軟件系統(tǒng)是一種較為理想的方法。如何在C51下編寫高效而正確的代碼對軟件開發(fā)人員是一個挑戰(zhàn)。本文介紹的幾種函數指針使用方法為嵌入式軟件開發(fā)人員提供了有益的參考。
參考文獻
[1] Kernighan Brian W, Rithie Dennis M.The C Programming Language 2nd Ed[M]. 北京:機械工業(yè)出版社,2004.
[2] Keil Software Inc. C51.pdf,2001.
[3] 馬忠梅,劉濱,等. 單片機C語言Windows環(huán)境編程寶典[M]. 北京:北京航空航天大學出版社,2004.
朱博(碩士研究生),主要研究方向為智能交通信息采集;許論輝(博士、教授),主要研究方向為智能交通控制。





