Keil C51編譯錯誤總結
1.第一種錯誤信息
***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
該警告表示連接器發現有一個函數可能會被主函數和一個中斷服務程序(或者調用中斷服務程序的函數)同時調用,
或者同時被多個中斷服務程序調用。
出現這種問題的原因之一是這個函數是不可重入性函數,當該函數運行時它可能會被一個中斷打斷,從而使得結果發生變化
并可能會引起一些變量形式的沖突(即引起函數內一些數據的丟失,可重入性函數在任何時候都可以被ISR打斷,一段時間后又可以
運行,但是相應數據不會丟失)。
原因之二是用于局部變量和變量(暫且這樣翻譯,arguments,[自變量,變元一數值,用于確定程序或子程序的值])的內存區被其他函數的內存區所覆蓋,如果該函數被中斷,則它的內存區就會被使用,這將導致其他函數的內存沖突。
例如,第一個警告中函數WRITE_GMVLX1_REG 在D_GMVLX1.C 或者D_GMVLX1.A51被定義,它被一個中斷服務程序或者一個調用了中斷服務程序的函數調用了,調用它的函數是VSYNC_INTERRUPT,在MAIN.C中。
解決方法:
如果你確定兩個函數決不會在同一時間執行(該函數被主程序調用并且中斷被禁止),并且該函數不占用內存(假設只使用寄存器),
則你可以完全忽略這種警告。
如果該函數占用了內存,則應該使用連接器(linker)OVERLAY指令將函數從覆蓋分析(overlay analysis)中除去,例如:
OVERLAY (?PR?_WRITE_GMVLX1_REG?D_GMVLX1 ! *)
上面的指令防止了該函數使用的內存區被其他函數覆蓋。如果該函數中調用了其他函數,而這些被調用在程序中其他地方也被調用,
你可能會需要也將這些函數排除在覆蓋分析(overlay analysis)之外。這種OVERLAY指令能使編譯器除去上述警告信息。
如果函數可以在其執行時被調用,則情況會變得更復雜一些。這時可以采用以下幾種方法:
1.主程序調用該函數時禁止中斷,可以在該函數被調用時用#pragma disable語句來實現禁止中斷的目的。必須使用OVERLAY指令將該函數
從覆蓋分析中除去。
2.復制兩份該函數的代碼,一份到主程序中,另一份復制到中斷服務程序中。
3.將該函數設為重入型。例如:
void myfunc(void) reentrant {
...
}
這種設置將會產生一個可重入堆棧,該堆棧被被用于存儲函數值和局部變量,用這種方法時重入堆棧必須在STARTUP.A51文件中配置。
這種方法消耗更多的RAM并會降低重入函數的執行速度。
2.第二種錯誤信息
*** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS
SEGMENT: ?PR?_COMPARE?TESTLCD
說明:程序中有些函數(或片段)以前(調試過程中)從未被調用過,或者根本沒有調用它的語句。
這條警告信息前應該還有一條信息指示出是哪個函數導致了這一問題。只要做點簡單的調整就可以。不理它也沒什么大不了的。
解決方法:去掉COMPARE()函數或利用條件編譯#if …..#endif,可保留該函數并不編譯。
評論