WARNING 15 (MULTIPLE CALL TO SEGMENT)
此處的信息適用于:
C51所有版本
癥狀
我添加了一個中斷處理程序(ISR)到我的項目中,然而我卻得到了如下的警告:
***WARNING L15: MULTIPLE CALL TO SEGMENT
SEGMENT: ?PR?_WRITE_GMVLX1_REG?D_GMVLX1
CALLER1: ?PR?VSYNC_INTERRUPT?MAIN
CALLER2: ?C_C51STARTUP
***WARNING L15: MULTIPLE CALL TO SEGMENT
SEGMENT: ?PR?_SPI_SEND_WORD?D_SPI
CALLER1: ?PR?VSYNC_INTERRUPT?MAIN
CALLER2: ?C_C51STARTUP
***WARNING L15: MULTIPLE CALL TO SEGMENT
SEGMENT: ?PR?SPI_RECEIVE_WORD?D_SPI
CALLER1: ?PR?VSYNC_INTERRUPT?MAIN
CALLER2: ?C_C51STARTUP
上面這些都是什么?我該如何解決這個問題呢?
原因
Warning 15向我們表明了linker發(fā)現(xiàn)了一個函數(shù),這個函數(shù)不僅在main code里被調用了,而且在ISR(或者被ISR調用的函數(shù)中)被調用了?;蛘呤潜煌瑫r被多個ISR同時調用了。
這樣會產生一個問題,就是在此函數(shù)不是一個可重入函數(shù),而當此函數(shù)已經在執(zhí)行時它可能被另一個ISR所調用。這樣就會導致結果是可變的而且很可能會導致一些參數(shù)的錯誤。
另一個問題就是本地變量和參數(shù)所使用的內存可能被其他函數(shù)的內存覆蓋。如果函數(shù)是由中斷所調用的,則此函數(shù)的內存就會被使用。這會引起其它函數(shù)的內存錯誤。
舉例來說,對于你的第一個警告,WRITE_GMVLX1_REG是會被多個root所調用。其被定義在D_GMVLX1.C或者D_GMVLX1.A51中。他不僅會被ISR(或者被ISR調用的函數(shù))而且也會被MAIN.C中的VSYNC_INTERRUPT函數(shù)所調用。
解決方法
有幾種方法去解決這個問題
如果你100%確認這個函數(shù)的兩個副本都不會同時執(zhí)行(如果此函數(shù)是被main調用并且中斷是未被使能的)并且此函數(shù)沒有使用內存(只使用的寄存器),那么你就可以忽略此警告。
如果此函數(shù)使用了內存,你就要使用OVERLAY directive來將此函數(shù)從覆蓋分析(overlay anaysis)中移除。舉例如下:
OVERLAY (?PR?_WRITE_GMVLX1_REG?D_GMVLX1 ! *)
如上語句能阻止被此函數(shù)使用的內存遭到覆蓋。如果這個函數(shù)調用了你程序中其他的在別處的函數(shù),那么你可能需要將這些函數(shù)也排除在覆蓋分析之外。
如果當此函數(shù)在執(zhí)行時可以被調用,那么事情就會變得比較的復雜。你可能需要:
無論何時當從main中調用此函數(shù)時,需要關閉中斷。你可能需要對被調用的函數(shù)使用#pragma disable。你也必須使用OVERLAY directive將此函數(shù)從overlay analysis中移除。
為此函數(shù)創(chuàng)建兩個副本。一個給main,一個給ISR。
使此函數(shù)可重入。舉例如下: void myfunc(void) reentrant {
...
}
上面的定義會產生一個用來存儲參數(shù)和本地變量的可重入的棧。如果使用了這種方法那么這個可重入的棧必須在STARTUP.A51中配置。這樣會花費更多的RAM并且會減緩可沖入函數(shù)的執(zhí)行。





