国产亚洲精品AA片在线观看,丰满熟妇HD,亚洲成色www成人网站妖精,丁香五月天婷婷五月天男人天堂

新聞中心

EEPW首頁 > 嵌入式系統(tǒng) > 設計應用 > KELl警告: MULTIPLE CALL TO SEGMENT

KELl警告: MULTIPLE CALL TO SEGMENT

作者: 時間:2016-11-09 來源:網絡 收藏
今天來說說*** WARNING L15: MULTIPLE CALL TO SEGMENT這個問題!

其實這個問題實際上就是函數重入問題,在操作系統(tǒng)的多線程很常見。應該是引起注意的,有可能引起程序沖突,但是一般時候程序運行不會有問題,但是如果出來問題,那將會是很討厭的問題.

本文引用地址:http://www.ljygm.com/article/201611/317704.htm

分析一下產生這一警告的一個根源是:例如在主循環(huán)里調用了一個函數,而在中斷服務中,你又一次調用了同樣的函數。這樣當主循環(huán)運行到該函數中時,一旦產生中斷,則在中斷里又再次調用該函數!而使得該子函數發(fā)生了重入,這時,盡管概率很低,但是很可能出錯!這樣,編譯器就給出了警告!告訴你*** WARNING L15: MULTIPLE CALL TO SEGMENT ,表達的意思是發(fā)生了重入!字面意思自己理解去吧~~~

想要避免這種情況的方法

一.用reentrant使函數重入

關于reentrant的說明:

1,重入函數不能傳遞bit類型的參數和變量;

2,重入函數建立的是模擬堆棧區(qū),所以不使用一般函數位于存儲模式默認空間的可覆蓋式堆棧,而是在同一空間從頂端另行分配一個非覆蓋式的重入堆棧。

small 默認空間是 data;

compact 默認空間是 pdata;

largr 默認空間是 xdata;

3,由于要保存參數和局部變量,所以會消耗很大的棧空間;盡量少用這種模式;

4、 在同一程序中可以定義和使用不同存儲器模式的重入函數,任意模式的重入函數不能調用不同存儲器模式的重入函數,但可以調用普通函數。
5、 實際參數可以傳遞給間接調用的重入函數。無重入屬性的間接調用函數不能包含調用參數。

二.如果空間多的話,可以定義兩個同功能的函數,分別在中斷和中斷外調用

別的方法沒研究出來,嘿嘿~~~對了 我建議用第二種方法好點,第一種有些限制,不爽~~

 在其它環(huán)境下(比如PC,比如ARM),函數重入的問題一般不是要特別注意的問題.只要你沒有使用static變量,或者指向static變量的指針,一般情況下,函數自然而然地就是可重入的.

  但C51不一樣,如果你不特別設計你的函數,它就是不可重入的.

  引起這個差別的原因在于:一般的C編譯器(或者更確切點地說:基于一般的處理器上的C編譯器),其函數的局部變量是存放于堆棧中的,而C51是存放于一個可覆蓋的(數據)段中的.

WARNING L15: MULTIPLE CALL TO SEGMENT(轉--感覺講的不錯)

這個問題必須注意,可能引起程序沖突,假設你用于自動化領域,則可能導致信號產生尖峰。 產生這一警告的一個根源是:你在主循環(huán)里調用了一個函數(如aaa),而在中斷服務函數里,你用調用了這個函數(如aaa)。這樣當主循環(huán)運行到該函數中是,一旦產生中斷,則在中斷里又再次調用該函數!這時,很可能出錯! 避免這種情況的方法很多:如,在進中斷的時候置需調用該函數的標志,而在主循環(huán)中調用該函數
Keil C -WARNING L15: MULTIPLE CALL TO SEGMENT

1.第一種錯誤信息

***WARNING L15: MULTIPLE CALL TO SEGMENT
SEGMENT:?PR?_WRITE_GMVLX1_REG?D_GMVLX1
CALLER1:?PR?VSYNC_INTERRUPT?MAIN
CALLER2:?C_C51STARTUP

該警告表示連接器發(fā)現有一個函數可能會被主函數和一個中斷服務程序(或者調用中斷服務程序的函數)同時調用,或者同時被多個中斷服務程序調用。
出現這種問題的原因之一是這個函數是不可重入性函數,當該函數運行時它可能會被一個中斷打斷,從而使得結果發(fā)生變化并可能會引起一些變量形式的沖突(即引起函數內一些數據的丟失,可重入性函數在任何時候都可以被ISR打斷,一段時間后又可以
運行,但是相應數據不會丟失)。
原因之二是用于局部變量和變量(暫且這樣翻譯,arguments,[自變量,變元一數值,用于確定程序或子程序的值])的內存區(qū)被其他函數的內存區(qū)所覆蓋,如果該函數被中斷,則它的內存區(qū)就會被使用,這將導致其他函數的內存沖突。
例如,第一個警告中函數WRITE_GMVLX1_REG 在D_GMVLX1.C 或者D_GMVLX1.A51被定義,它被一個中斷服務程序或者一個調用了中斷服務程序的函數調用了,調用它的函數是VSYNC_INTERRUPT,在MAIN.C中。
解決方法:
如果你確定兩個函數決不會在同一時間執(zhí)行(該函數被主程序調用并且中斷被禁止),并且該函數不占用內存(假設只使用寄存器),則你可以完全忽略這種警告。
如果該函數占用了內存,則應該使用連接器(linker)OVERLAY指令將函數從覆蓋分析(overlay analysis)中除去,例如:
OVERLAY (?PR?_WRITE_GMVLX1_REG?D_GMVLX1 ! *)
上面的指令防止了該函數使用的內存區(qū)被其他函數覆蓋。如果該函數中調用了其他函數,而這些被調用在程序中其他地方也被調用,你可能會需要也將這些函數排除在覆蓋分析(overlay analysis)之外。這種OVERLAY指令能使編譯器除去上述警告信息。
如果函數可以在其執(zhí)行時被調用,則情況會變得更復雜一些。這時可以采用以下幾種方法:
1.主程序調用該函數時禁止中斷,可以在該函數被調用時用#pragma disable語句來實現禁止中斷的目的。必須使用OVERLAY指令將該函數從覆蓋分析中除去。
2.復制兩份該函數的代碼,一份到主程序中,另一份復制到中斷服務程序中。
3.將該函數設為重入型。例如:
void myfunc(void) reentrant
{
...
}

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



評論


技術專區(qū)

關閉