KEIL51調(diào)試時一些錯誤總結(jié)
(1)提示無M51文件
本文引用地址:http://butianyuan.cn/article/201609/310318.htm編譯時候提示:
F:...XX.M51
File has been changed outside the editor, reload ?
------
解決方法:
重新生成項(xiàng)目,產(chǎn)生STARTUP.A51即可。
(2)L15重復(fù)調(diào)用
***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ù)運(yùn)行時它可能會被一個中斷打斷,從而使得結(jié)果發(fā)生變化并可能會引起一些變量形式的沖突(即引起函數(shù)內(nèi)一些數(shù)據(jù)的丟失,可重入性函數(shù)在任何時候都可以被ISR打斷,一段時間后又可以
運(yùn)行,但是相應(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語句來實(shí)現(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í)行速度。
(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ǎn)簡單的調(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 SEGMENT
Program Size: data=8.0 xdata=0 code=9
---
修改:
缺少主程序(其實(shí)是筆誤),將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 /* Read P1 into pval */
P3 = pval; /* Write pval to P3 */
}
編譯提示出錯:
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','0x0d','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程序有中文標(biāo)點(diǎn)
用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 自帶的(見下),為何會報錯呀。
----
回答:程序里有帶中文標(biāo)點(diǎn),用英文重新寫一遍即可
(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)C141錯誤
提示 001.C(23): error C141: syntax error near 'unsigned'
這行之前的語句"bit flag_Key_Service_song=0"少分號了
(14)C129錯誤
提示 001.C(22): error C129: missing ';' before 'flag_Key_Service_song'
定義里 "bi flag_Key_Service_song=0;"
改為bit
評論