新聞中心

EEPW首頁(yè) > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > KEIL51調(diào)試時(shí)一些錯(cuò)誤總結(jié)

KEIL51調(diào)試時(shí)一些錯(cuò)誤總結(jié)

作者: 時(shí)間:2016-09-26 來(lái)源:網(wǎng)絡(luò) 收藏

  (1)提示無(wú)M51文件

本文引用地址:http://m.butianyuan.cn/article/201609/310318.htm

  編譯時(shí)候提示:

  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)有一個(gè)函數(shù)可能會(huì)被主函數(shù)和一個(gè)中斷服務(wù)程序(或者調(diào)用中斷服務(wù)程序的函數(shù))同時(shí)調(diào)用,或者同時(shí)被多個(gè)中斷服務(wù)程序調(diào)用。

  出現(xiàn)這種問(wèn)題的原因之一是這個(gè)函數(shù)是不可重入性函數(shù),當(dāng)該函數(shù)運(yùn)行時(shí)它可能會(huì)被一個(gè)中斷打斷,從而使得結(jié)果發(fā)生變化并可能會(huì)引起一些變量形式的沖突(即引起函數(shù)內(nèi)一些數(shù)據(jù)的丟失,可重入性函數(shù)在任何時(shí)候都可以被ISR打斷,一段時(shí)間后又可以

  運(yùn)行,但是相應(yīng)數(shù)據(jù)不會(huì)丟失)。

  原因之二是用于局部變量和變量(暫且這樣翻譯,arguments,[自變量,變?cè)粩?shù)值,用于確定程序或子程序的值])的內(nèi)存區(qū)被其他函數(shù)的內(nèi)存區(qū)所覆蓋,如果該函數(shù)被中斷,則它的內(nèi)存區(qū)就會(huì)被使用,這將導(dǎo)致其他函數(shù)的內(nèi)存沖突。

  例如,第一個(gè)警告中函數(shù)WRITE_GMVLX1_REG 在D_GMVLX1.C 或者D_GMVLX1.A51被定義,它被一個(gè)中斷服務(wù)程序或者一個(gè)調(diào)用了中斷服務(wù)程序的函數(shù)調(diào)用了,調(diào)用它的函數(shù)是VSYNC_INTERRUPT,在MAIN.C中。

  解決方法:

  如果你確定兩個(gè)函數(shù)決不會(huì)在同一時(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)用,你可能會(huì)需要也將這些函數(shù)排除在覆蓋分析(overlay analysis)之外。這種OVERLAY指令能使編譯器除去上述警告信息。

  如果函數(shù)可以在其執(zhí)行時(shí)被調(diào)用,則情況會(huì)變得更復(fù)雜一些。這時(shí)可以采用以下幾種方法:

  1.主程序調(diào)用該函數(shù)時(shí)禁止中斷,可以在該函數(shù)被調(diào)用時(shí)用#pragma disable語(yǔ)句來(lái)實(shí)現(xiàn)禁止中斷的目的。必須使用OVERLAY指令將該函數(shù)從覆蓋分析中除去。

  2.復(fù)制兩份該函數(shù)的代碼,一份到主程序中,另一份復(fù)制到中斷服務(wù)程序中。

  3.將該函數(shù)設(shè)為重入型。例如:

  void myfunc(void) reentrant {

  ...

  }

  這種設(shè)置將會(huì)產(chǎn)生一個(gè)可重入堆棧,該堆棧被被用于存儲(chǔ)函數(shù)值和局部變量,用這種方法時(shí)重入堆棧必須在STARTUP.A51文件中配置。這種方法消耗更多的RAM并會(huì)降低重入函數(shù)的執(zhí)行速度。

  (3)L16無(wú)調(diào)用

  *** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS

  SEGMENT: ?PR?_COMPARE?TESTLCD

  說(shuō)明:程序中有些函數(shù)例如COMPARE(或片段)以前(調(diào)試過(guò)程中)從未被調(diào)用過(guò),或者根本沒(méi)有調(diào)用它的語(yǔ)句。

  這條警告信息前應(yīng)該還有一條信息指示出是哪個(gè)函數(shù)導(dǎo)致了這一問(wèn)題。只要做點(diǎn)簡(jiǎn)單的調(diào)整就可以。不理它也沒(méi)什么大不了的。

  解決方法:去掉COMPARE()函數(shù)或利用條件編譯#if …..#endif,可保留該函數(shù)并不編譯。

  (4)L10和L16"主程序名字寫(xiě)錯(cuò)(或無(wú)主程序)"

  程序中:

  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主程序沒(méi)用到前面定義的函數(shù)

  主程序里沒(méi)用到前面定義的函數(shù),編譯時(shí)顯示:

  *** WARNING L16: UNCALLED SEGMENT, IGNORED FOR OVERLAY PROCESS

  SEGMENT: ?PR?DELAY?MAIN

  (6)L210程序前生成SRC語(yǔ)句

  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è)置上的問(wèn)題,在程序里屏蔽掉#pragma src即可

  (7)C206函數(shù)未定義

  該函數(shù)沒(méi)定義

  MAIN.C(15): warning C206: 'delay1': missing function-prototype

  (8)C141少分號(hào)

  部分程序:

  {

  pval = P1 /* Read P1 into pval */

  P3 = pval; /* Write pval to P3 */

  }

  編譯提示出錯(cuò):

  MAIN.C(22): error C141: syntax error near 'P3'

  改正: P1后加";"

  (9)C129匯編與C后綴問(wèn)題

  例如寫(xiě)這么一段小程序,保存為c0.c,編譯時(shí)出現(xiàn)error c129,miss ; before 0000;

  如果保存為:c0.asm就不會(huì)出現(xiàn)這個(gè)錯(cuò)誤,保存為c的話,先調(diào)用c51編譯器,按c語(yǔ)言的要求編譯,所以出現(xiàn)錯(cuò)誤;可以參考一些書(shū),專(zhuān)門(mén)介紹keilc這個(gè)編譯器的;

  (10)C101和C141關(guān)于數(shù)組引號(hào)問(wèn)題

  定義了如下的數(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'};

  可是編譯的時(shí)候總通不過(guò),錯(cuò)誤提示如下:

  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

  解決方法:去掉'...'引號(hào)

  (11)C100和C141和C129程序有中文標(biāo)點(diǎn)

  用keil編譯時(shí)出現(xiàn)錯(cuò)誤,如下:D:KEILC51INCREG52.H(1): error C100: unprintable character 0xA1 skipped

  同上錯(cuò)誤有很多個(gè),還有D:KEILC51INCREG52.H(2): error C141: syntax error near '#'

  D:KEILC51INCREG52.H(2): error C129: missing ';' before '<'

  但是reg52.h頭文件是keil 自帶的(見(jiàn)下),為何會(huì)報(bào)錯(cuò)呀。

  ----

  回答:程序里有帶中文標(biāo)點(diǎn),用英文重新寫(xiě)一遍即可

  (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”。主要錯(cuò)在這里。

  應(yīng)該輸入數(shù)字 “0”,而你輸入字母“O”了。

  (13)C141錯(cuò)誤

  提示 001.C(23): error C141: syntax error near 'unsigned'

  這行之前的語(yǔ)句"bit flag_Key_Service_song=0"少分號(hào)了

  (14)C129錯(cuò)誤

  提示 001.C(22): error C129: missing ';' before 'flag_Key_Service_song'

  定義里 "bi flag_Key_Service_song=0;"

  改為bit



關(guān)鍵詞: KEIL51

評(píng)論


技術(shù)專(zhuān)區(qū)

關(guān)閉