keilC51編譯常見錯誤和警告說明
***WARNING L15: MULTIPLE CALL TO SEGMENT
SEGMENT: ?PR?SPI_RECEIVE_WORD?D_SPI
CALLER1: ?PR?VSYNC_INTERRUPT?MAIN
CALLER2: ?C_C51STARTUP
該警告表示連接器發(fā)現(xiàn)有一個函數(shù)可能會被主函數(shù)和一個中斷服務(wù)程序 ( 或者調(diào)用中斷服務(wù)程序的函數(shù) ) 同時調(diào)用 , 或者同時被多個中斷服務(wù)程序調(diào)用。
出現(xiàn)這種問題的原因之一是這個函數(shù)是不可重入性函數(shù) , 當(dāng)該函數(shù)運行時它可能會被一個中斷打斷 , 從
而使得結(jié)果發(fā)生變化并可能會引起一些變量形式的沖突 ( 即引起函數(shù)內(nèi)一些數(shù)據(jù)的丟失 , 可重入性函數(shù)在任何時候都可以被 ISR 打斷 , 一段時間后又可以運行 , 但是相應(yīng)數(shù)據(jù)不會丟失 ) 。
原因之二是用于局部變量和變量 ( 暫且這樣翻譯 ,arguments,[ 自變量 , 變元一數(shù)值 , 用于確定程序或子程序的值 ]) 的內(nèi)存區(qū)被其他函數(shù)的內(nèi)存區(qū)所覆蓋 , 如果該函數(shù)被中斷 , 則它的內(nèi)存區(qū)就會被使用 , 這將導(dǎo)致其他函數(shù)
的內(nèi)存沖突。
例如 , 第一個警告中函數(shù) WRITE_GMVLX1_REG 在 D_GMVLX1.C 或者 D_GMVLX1.A51 被定義 , 它被一個中斷
服務(wù)程序或者一個調(diào)用了中斷服務(wù)程序的函數(shù)調(diào)用了 , 調(diào)用它的函數(shù)是 VSYNC_INTERRUPT, 在 MAIN.C 中。
解決方法:
如果你確定兩個函數(shù)決不會在同一時間執(zhí)行 ( 該函數(shù)被主程序調(diào)用并且中斷被禁止 ), 并且該函數(shù)不占用內(nèi)存 ( 假設(shè)只使用寄存器 ), 則你可以完全忽略這種警告。
如果該函數(shù)占用了內(nèi)存 , 則應(yīng)該使用連接器 (linker)OVERLAY 指令將函數(shù)從覆蓋分析 (overlay
analysis) 中除去 , 例如:
OVERLAY (?PR?_WRITE_GMVLX1_REG?D_GMVLX1 ! *)
上面的指令防止了該函數(shù)使用的內(nèi)存區(qū)被其他函數(shù)覆蓋 。 如果該函數(shù)中調(diào)用了其他函數(shù) , 而這些被調(diào)用在
程序中其他地方也被調(diào)用 , 你可能會需要也將這些函數(shù)排除在覆蓋分析 (overlay analysis) 之外。這種 OVERLAY 指
令能使編譯器除去上述警告信息。
如果函數(shù)可以在其執(zhí)行時被調(diào)用 , 則情況會變得更復(fù)雜一些。這時可以采用以下幾種方法:
1. 主程序調(diào)用該函數(shù)時禁止中斷 , 可以在該函數(shù)被調(diào)用時用 #pragma disable 語句來實現(xiàn)禁止中斷的目的。必
須使用 OVERLAY 指令將該函數(shù)從覆蓋分析中除去。
2. 復(fù)制兩份該函數(shù)的代碼 , 一份到主程序中 , 另一份復(fù)制到中斷服務(wù)程序中。
3. 將該函數(shù)設(shè)為重入型。例如:void myfunc(void) reentrant {
...
}
這種設(shè)置將會產(chǎn)生一個可重入堆棧 , 該堆棧被被用于存儲函數(shù)值和局部變量 , 用這種方法時重入堆棧必須在 STARTUP.A51 文件中配置。這種方法消耗更多的 RAM 并會降低重入函數(shù)的執(zhí)行速度。
( 2 ) 提示無 M51 文件
編譯時候提示:
F:...XX.M51
File has been changed outside the editor, reload ?
------
解決方法:
重新生成項目,產(chǎn)生 STARTUP.A51 即可。
(3)L16 無調(diào)用
*** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS
SEGMENT: ?PR?_COMPARE?TESTLCD
說明 : 程序中有些函數(shù)例如 COMPARE ( 或片段 ) 以前 ( 調(diào)試過程中 ) 從未被調(diào)用過 , 或者根本沒
有調(diào)用它的語句。
這條警告信息前應(yīng)該還有一條信息指示出是哪個函數(shù)導(dǎo)致了這一問題。只要做點簡單的調(diào)整就
可以。不理它也沒什么大不了的。
解決方法 : 去掉 COMPARE() 函數(shù)或利用條件編譯 #if … ..#endif, 可保留該函數(shù)并不編譯。
(4)L10 和 L16" 主程序名字寫錯 ( 或無主程序 )"
程序中 :
void mian (void)
編譯提示 :
*** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS
SEGMENT: ?PR?MIAN?MAIN
*** WARNING L10: CANNOT DETERMINE ROOT SEGMENTProgram Size: data=8.0 xdata=0 code=9
---
修改 :
缺少主程序 ( 其實是筆誤 ), 將 mian 改為 main
(5)L16 主程序沒用到前面定義的函數(shù)
主程序里沒用到前面定義的函數(shù),編譯時顯示:
*** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS
SEGMENT: ?PR?DELAY?MAIN
(6)L210 程序前生成 SRC 語句
Build target Target 1
assembling STARTUP.A51...
compiling test.C...
linking...
BL51 BANKED LINKER/LOCATER V6.00 - SN: K1JXC-94Z4V9
COPYRIGHT KEIL ELEKTRONIK GmbH 1987 - 2005
"STARTUP.obj",
"test.obj"
TO "test"
*** FATAL ERROR L210: I/O ERROR ON INPUT FILE:
EXCEPTION 0021H: PATH OR FILE NOT FOUND
FILE: test.obj
Target not created
---------
設(shè)置上的問題,在程序里屏蔽掉 #pragma src 即可
(7)C206 函數(shù)未定義
該函數(shù)沒定義
MAIN.C(15): warning C206: delay1: missing function-prototype(8)C141 少分號
部分程序 :
{
pval = P1
P3 = pval;
}
編譯提示出錯 :
MAIN.C(22): error C141: syntax error near P3
改正 : P1 后加 ";"
(9)C129 匯編與 C 后綴問題
例如寫這么一段小程序,保存為 c0.c ,編譯時出現(xiàn) error c129 , miss ; before 0000 ;
如果保存為: c0.asm 就不會出現(xiàn)這個錯誤,保存為 c 的話,先調(diào)用 c51 編譯器,按 c 語言的要求編譯,所以
出現(xiàn)錯誤;可以參考一些書,專門介紹 keilc 這個編譯器的;
(10)C101 和 C141 關(guān)于數(shù)組引號問題
定義了如下的數(shù)組:
unsigned char
a[36]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f,0x7e,0x7d,0x7b,0x77,0x6f,
0x5f,0x3f,0x3e,0x3d,0x3b,0x37,0x2f,0x1f,0x1e,0x1d,0x1b,0x17,0x0f,0x0e,0
x0d,0x0b,0x07,0x06,0x05,0x03,0x02,0x01,0x00};
可是編譯的時候總通不過,錯誤提示如下:
Build target Target 1
compiling shaomiao.c...
SHAOMIAO.C(3): error C101: 0: invalid character constant
SHAOMIAO.C(3): error C141: syntax error near xfe
SHAOMIAO.C(3): error C101: }: invalid character constant
Target not created
解決方法:去掉 ... 引號
(11)C100 和 C141 和 C129 程序有中文標點
用 keil 編譯時出現(xiàn)錯誤,如下: D:KEILC51INCREG52.H(1): error C100: unprintable character 0xA1
skipped同上錯誤有很多個,還有 D:KEILC51INCREG52.H(2): error C141: syntax error near #
D:KEILC51INCREG52.H(2): error C129: missing ; before<
但是 reg52.h 頭文件是 keil 自帶的(見下),為何會報錯呀。
----
回答: 程序里有帶中文標點,用英文重新寫一遍即可
(12)A45 匯編出現(xiàn)數(shù)字、字母混淆
MOV PO,A ;put on next 11
...
MOV RO,#0FFH ; 14
MOV R1,#OFFH ; 15
...
DJNZ RO,DLY_LP ;19
MOV R0,#OFFH ; 20
...
編譯后:
ledtest.asm(11): error A45: UNDEFINED SYMBOL (PASS-2)
ledtest.asm(14): error A45: UNDEFINED SYMBOL (PASS-2)
ledtest.asm(15): error A45: UNDEFINED SYMBOL (PASS-2)
ledtest.asm(19): error A45: UNDEFINED SYMBOL (PASS-2)
ledtest.asm(20): error A45: UNDEFINED SYMBOL (PASS-2)
Target not created
---------
注意:
字母 “ O ” 和 數(shù)字 “ 0 ” 。主要錯在這里。
應(yīng)該輸入數(shù)字 “ 0 ” ,而你輸入字母 “ O ” 了。
(13)
說明局部變量 i 在函數(shù)中未作任何的存取操作解決方法消除函數(shù)中 i 變量的宣告2 Warning 206: ’ Music3 ’ :missing function-prototype
說明 Music3( ) 函數(shù)未作宣告或未作外部宣告所以無法給其他函數(shù)調(diào)用
解決方法將敘述 void Music3(void) 寫在程序的最前端作宣告如果是其他文件的函數(shù)則要寫
成 extern void Music3(void), 即作外部宣告
3 Compling :C:8051MANN.C
Error:318:can ’ t open file ‘ beep.h ’
說明在編譯 C:8051MANN.C 程序過程中由于 main.c 用了指令# i nclude “ beep.h ” , 但
卻找不到所致解決方法編寫一個 beep.h 的包含檔并存入到 c:8051 的工作目錄中
(14)Compling:C:8051LED.C
Error 237: ’ LedOn ’ :function already has a body
說明 LedOn( ) 函數(shù)名稱重復(fù)定義即有兩個以上一樣的函數(shù)名稱
解決方法修正其中的一個函數(shù)名稱使得函數(shù)名稱都是獨立的
15)***WARNING 16:UNCALLED SEGMENT,IGNORED FOR OVERLAY PROCESS
SEGMENT: ?PR?_DELAYX1MS?DELAY
說明 DelayX1ms( ) 函數(shù)未被其它函數(shù)調(diào)用也會占用程序記憶體空間解決方法去掉DelayX1ms( ) 函數(shù)或利用條件編譯 #if … ..#endif, 可保留該函數(shù)并不編譯
(16) ***WARNING 6 :XDATA SPACE MEMORY OVERLAP
FROM : 0025H
TO: 0025H
說明外部資料 ROM 的 0025H 重復(fù)定義地址
解決方法外部資料 ROM 的定義如下 Pdata unsigned char XFR_ADC _at_0x25 其中 XFR_ADC
變量的名稱為 0x25, 請檢查是否有其它的變量名稱也是定義在 0x25 處并修正它
(17) WARNING 206: ’ DelayX1ms ’ : missing function-prototype
C:8051INPUT.C
Error 267 : ’ DelayX1ms ‘ :requires ANSI-style prototype C:8051INPUT.C
說明程序中有調(diào)用 DelayX1ms 函數(shù)但該函數(shù)沒定義即未編寫程序內(nèi)容或函數(shù)已定義但未作宣告
解決方法編寫 DelayX1ms 的內(nèi)容編寫完后也要作宣告或作外部宣告可在 delay.h 的包含檔宣告成外部以便其它函數(shù)調(diào)用
(18) ***WARNING 1:UNRESOLVED EXTERNAL SYMBOL
SYMBOL:MUSIC3
MODULE:C:8051MUSIC.OBJ(MUSIC)
***WARNING 2:REFERENCE MADE TO UNRESOLVED EXTERNAL
SYMBOL:MUSIC3
MODULE:C:8051MUSIC.OBJ(MUSIC)
ADDRESS:0018H
說明程序中有調(diào)用 MUSIC 函數(shù)但未將該函數(shù)的含擴檔 C 加入到工程檔 Prj 作編譯和連接
解決方法設(shè) MUSIC3 函數(shù)在 MUSIC C 里將 MUSIC C 添加到工程文件中去
(19) ***ERROR 107:ADDESS SPACE OVERFLOW
SPACE: DATA
SEGMENT: _DATA_GOUP_
LENGTH: 0018H
***ERROR 118: REFERENCE MADE TO ERRONEOUS EXTERNAL
SYMBOL: VOLUME
MODULE: C:8051OSDM.OBJ (OSDM)
ADDRESS: 4036H
說明 data 存儲空間的地址范圍為 0~0x7f, 當(dāng)公用變量數(shù)目和函數(shù)里的局部變量如果存儲模式設(shè)為 SMALL 則局部變量先使用工作寄存器 R2~R7 作暫存當(dāng)存儲器不夠用時則會
以 data 型別的空間作暫存的個數(shù)超過 0x7f 時就會出現(xiàn)地址不夠的現(xiàn)象
解決方法將以 data 型別定義的公共變量修改為 idata 型別的定義
(20).***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
該警告表示連接器發(fā)現(xiàn)有一個函數(shù)可能會被主函數(shù)和一個中斷服務(wù)程序 ( 或者調(diào)用中斷服務(wù)程序的函數(shù) ) 同時調(diào)用 ,或者同時被多個中斷服務(wù)程序調(diào)用。
出現(xiàn)這種問題的原因之一是這個函數(shù)是不可重入性函數(shù) , 當(dāng)該函數(shù)運行時它可能會被一
個中斷打斷 , 從而使得結(jié)果發(fā)生變化并可能會引起一些變量形式的沖突 ( 即引起函數(shù)內(nèi)一些數(shù)據(jù)的丟失 , 可重入性函數(shù)在任何時候都可以被 ISR 打斷 , 一段時間后又可以運行 , 但是相應(yīng)數(shù)據(jù)不會丟失 ) 。
原因之二是用于局部變量和變量 ( 暫且這樣翻譯 ,arguments,[ 自變量 , 變元一數(shù)值 , 用于確定程序或子程序的值 ]) 的內(nèi)存區(qū)被其他函數(shù)的內(nèi)存區(qū)所覆蓋 , 如果該函數(shù)被中斷 , 則它的內(nèi)存區(qū)就會被使用 , 這將導(dǎo)致其他函數(shù)的內(nèi)存沖突。
例如 , 第一個警告中函數(shù) WRITE_GMVLX1_REG 在 D_GMVLX1.C 或者 D_GMVLX1.A51 被定義 ,
它被一個中斷服務(wù)程序或者一個調(diào)用了中斷
服務(wù)程序的函數(shù)調(diào)用了 , 調(diào)用它的函數(shù)是 VSYNC_INTERRUPT, 在 MAIN.C 中。
解決方法:
如果你確定兩個函數(shù)決不會在同一時間執(zhí)行 ( 該函數(shù)被主程序調(diào)用并且中斷被禁止 ), 并
且該函數(shù)不占用內(nèi)存 ( 假設(shè)只使用寄存器 ),
則你可以完全忽略這種警告。
如果該函數(shù)占用了內(nèi)存 , 則應(yīng)該使用連接器 (linker)OVERLAY 指令將函數(shù)從覆蓋分析
(overlay analysis) 中除去 , 例如:
OVERLAY (?PR?_WRITE_GMVLX1_REG?D_GMVLX1 ! *)
上面的指令防止了該函數(shù)使用的內(nèi)存區(qū)被其他函數(shù)覆蓋。如果該函數(shù)中調(diào)用了其他函數(shù) ,
而這些被調(diào)用在程序中其他地方也被調(diào)用 ,
你可能會需要也將這些函數(shù)排除在覆蓋分析 (overlay analysis) 之外。這種 OVERLAY 指令能
使編譯器除去上述警告信息。
如果函數(shù)可以在其執(zhí)行時被調(diào)用 , 則情況會變得更復(fù)雜一些。這時可以采用以下幾種方
法:
1. 主程序調(diào)用該函數(shù)時禁止中斷 , 可以在該函數(shù)被調(diào)用時用 #pragma disable 語句來實現(xiàn)禁
止中斷的目的。必須使用 OVERLAY 指令將該函數(shù)
從覆蓋分析中除去。
2. 復(fù)制兩份該函數(shù)的代碼 , 一份到主程序中 , 另一份復(fù)制到中斷服務(wù)程序中。3. 將該函數(shù)設(shè)為重入型。例如:
void myfunc(void) reentrant {
...
}
這種設(shè)置將會產(chǎn)生一個可重入堆棧 , 該堆棧被被用于存儲函數(shù)值和局部變量 , 用這種
方法時重入堆棧必須在 STARTUP.A51 文件中配置。
這種方法消耗更多的 RAM 并會降低重入函數(shù)的執(zhí)行速度。
(21). *** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS
SEGMENT: ?PR?_COMPARE?TESTLCD
說明:程序中有些函數(shù)(或片段)以前(調(diào)試過程中)從未被調(diào)用過 , 或者根本沒有
調(diào)用它的語句。
這條警告信息前應(yīng)該還有一條信息指示出是哪個函數(shù)導(dǎo)致了這一問題。只要做點簡
單的調(diào)整就可以。不理它也沒什么大不了的。
解決方法 : 去掉 COMPARE() 函數(shù)或利用條件編譯 #if … ..#endif, 可保留該函數(shù)并不
編譯。
評論