当前位置:首页 > 单片机 > 单片机
[导读]  函数指针在C语言中应用较为灵活。在单片机系统中,嵌入式操作系统、文件系统和网络协议栈等一些较为复杂的应用都大量地使用了函数指针。Keil公司推出的C51编译器是事实上80C51 C编程的工业标准,它针对8051系列C

  函数指针在C语言中应用较为灵活。在单片机系统中,嵌入式操作系统、文件系统和网络协议栈等一些较为复杂的应用都大量地使用了函数指针。Keil公司推出的C51编译器是事实上80C51 C编程的工业标准,它针对8051系列CPU硬件在标准ANSI C的基础上进行了扩展;但由于编译器及8051体系结构的限制,造成了在使用函数指针时有很多与ANSI C不同的地方。下面举例说明在不同的情形下函数指针的使用。以下代码均在Keil μVision3、v8.08 C51、默认优化等级的开发环境下验证通过。
  1、指向固定地址的指针
  在程序设计中,常需要跳转到某一特定的地址上执行,如引导程序的设计。可通过如下C语言实现:

intmain(void){((void(code*)(void))0x2000)();return0;}

  此代码使得主函数执行位于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的规定,所有的函数参数都需要通过寄存器传递。由于8051的寄存器数目的限制,函数指针最多只能传递3个参数(具体传递规则详见参考文献[2])。其声明与调用方式如下:

void(*pfun)(char,short,int);//申明函数指针(*pfun)('c',0x1234,0x5678);//调用改函数

  如果需要传递3个以上函数的参数,可以把参数存放到结构体[1]里面,再用一个指针指向该结构体作为参数传递给函数指针。也可以使用reentrant关键字将函数声明为可重入函数[2]。
  4、分析调用树正确使用指针函数
  Keil C51编译器与ANSC C编译器的区别之一是,它并不把函数参数压入堆栈中,而是把函数参数放在寄存器(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区,二者传递参数的区域相互覆盖。通过Keil调试器可知,由于参数fptr被错误覆盖,在第2次调用(*ftpr)()时,程序已经不能正确跳转至foo函数执行了。
  显然,造成上述结果的原因是生成的调用树出错了。Keil提供了链接器OVERLAY伪指令,可让用户自行修改调用树,调整函数的调用关系。可在链接命令行输入以下命令(OVERLAY指令的用法详见参考文献[2]):

OVERLAY(?PR?MAIN?MAIN~?PR?_FOO?MAIN,?PR?_FOO_CALLER?MAIN!?PR?_FOO?MAIN)

  或在Keil集成开发环境中,在“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区不再重叠,调用树正确地构造了。调试结果也显示,输出正如所期望的一样。
  结语:采用C51设计较为复杂的单片机软件系统是一种较为理想的方法。如何在C51下编写高效而正确的代码对软件开发人员是一个挑战。本文介绍的几种函数指针使用方法为嵌入式软件开发人员提供了有益的参考。
参考文献
[1] Kernighan Brian W, Rithie Dennis M.The C Programming Language 2nd Ed[M]. 北京:机械工业出版社,2004.
[2] Keil Software Inc. C51.pdf,2001.
[3] 马忠梅,刘滨,等. 单片机C语言Windows环境编程宝典[M]. 北京:北京航空航天大学出版社,2004.
朱博(硕士研究生),主要研究方向为智能交通信息采集;许论辉(博士、教授),主要研究方向为智能交通控制。


本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

9月2日消息,不造车的华为或将催生出更大的独角兽公司,随着阿维塔和赛力斯的入局,华为引望愈发显得引人瞩目。

关键字: 阿维塔 塞力斯 华为

加利福尼亚州圣克拉拉县2024年8月30日 /美通社/ -- 数字化转型技术解决方案公司Trianz今天宣布,该公司与Amazon Web Services (AWS)签订了...

关键字: AWS AN BSP 数字化

伦敦2024年8月29日 /美通社/ -- 英国汽车技术公司SODA.Auto推出其旗舰产品SODA V,这是全球首款涵盖汽车工程师从创意到认证的所有需求的工具,可用于创建软件定义汽车。 SODA V工具的开发耗时1.5...

关键字: 汽车 人工智能 智能驱动 BSP

北京2024年8月28日 /美通社/ -- 越来越多用户希望企业业务能7×24不间断运行,同时企业却面临越来越多业务中断的风险,如企业系统复杂性的增加,频繁的功能更新和发布等。如何确保业务连续性,提升韧性,成...

关键字: 亚马逊 解密 控制平面 BSP

8月30日消息,据媒体报道,腾讯和网易近期正在缩减他们对日本游戏市场的投资。

关键字: 腾讯 编码器 CPU

8月28日消息,今天上午,2024中国国际大数据产业博览会开幕式在贵阳举行,华为董事、质量流程IT总裁陶景文发表了演讲。

关键字: 华为 12nm EDA 半导体

8月28日消息,在2024中国国际大数据产业博览会上,华为常务董事、华为云CEO张平安发表演讲称,数字世界的话语权最终是由生态的繁荣决定的。

关键字: 华为 12nm 手机 卫星通信

要点: 有效应对环境变化,经营业绩稳中有升 落实提质增效举措,毛利润率延续升势 战略布局成效显著,战新业务引领增长 以科技创新为引领,提升企业核心竞争力 坚持高质量发展策略,塑强核心竞争优势...

关键字: 通信 BSP 电信运营商 数字经济

北京2024年8月27日 /美通社/ -- 8月21日,由中央广播电视总台与中国电影电视技术学会联合牵头组建的NVI技术创新联盟在BIRTV2024超高清全产业链发展研讨会上宣布正式成立。 活动现场 NVI技术创新联...

关键字: VI 传输协议 音频 BSP

北京2024年8月27日 /美通社/ -- 在8月23日举办的2024年长三角生态绿色一体化发展示范区联合招商会上,软通动力信息技术(集团)股份有限公司(以下简称"软通动力")与长三角投资(上海)有限...

关键字: BSP 信息技术
关闭
关闭