分享

KEIL产生WARNINGL15: MULTIPLE CALL TO SEGMENT的原因及解决方法

 鹿港小镇1989 2014-05-16

***WARNING L15: MULTIPLE CALL TO SEGMENT
SEGMENT: ?。。。?。。。
CALLER1: ?。。。?MAIN
CALLER2: ?C_C51STARTUP


该警告表示连接器发现有一个函数可能会被主函数和一个中断服务程序(或者调用中断服务程序的函数)同时调用,或者同时被多个中断服务程序调用。


 


出现这种警告的原因一般有两种:


第一:这个函数是不可重入函数,当该函数运行时可能被打断,打断后该函数又被再次运行,从而造成函数内部数据丢失;


第二:该函数的内部变量数据所占有的内存在link时被连接器认为是可覆盖的,因此在连接时进行了数据覆盖优化,但是连接器同时发现该函数在运行时被打断后,其他函数(如中断服务子程序)的运行造成了该函数的数据被覆盖。


 


对于第二种原因,我们举个例子:


void task1()


{


     int val1;


    ......\\其他程序


}


 


void task2()


{


   int val2;


  ......\\其他程序


}


void int_ISR()//中断服务子程序


{


   int aaa;


  task2();


}


这里有两个任务函数task1和task2,它们之间没有直接的调用关系,因此连接器在连接时就会做数据覆盖优化,也就是说va11和val2这两个变量会占用相同的内存空间。但是当task1运行时发生了中断,在中断服务子程序中又调用了task2,那么task1的val1值很可能会被覆盖(数据被破坏)。这种情况的解决办法一般有两种:


 


1、使用连接器(linker)OVERLAY 指令将函数从覆盖分析(overlay analysis)中除去;即告知连接器不对task1的val1做覆盖优化,让它独享自己的内存空间;这种方法的唯一缺点是task1不能做递归调用(个人认为,没做验证),要想一个函数可递归调用,唯一的方法是声明该函数为可重入。


 


2、定义两个相同功能的函数,分别在中断和中断外调用;


 


当然,你可以把task1设为可重入函数,产生一个可重入堆栈,用于存储函数的参数和局部变量,但是这种方法会消耗更多的RAM 并会降低重入函数的执行速度。


 


上述是个人理解心得,如有问题请回帖讨论,我会加以更正。

标签:  

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多