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