當(dāng)前位置:首頁 > 單片機(jī) > 單片機(jī)
[導(dǎo)讀]能從PC機(jī)器編程去看嵌入式問題,那是第一步;學(xué)會用嵌入式編程思想,那是第二步;用PC的思想和嵌入式的思想結(jié)合在一起,應(yīng)用于實(shí)際的項(xiàng)目,那是第三步。很多朋友都是從PC編程轉(zhuǎn)向嵌入式編程的。在中國,嵌入式編程的朋

能從PC機(jī)器編程去看嵌入式問題,那是第一步;學(xué)會用嵌入式編程思想,那是第二步;用PC的思想和嵌入式的思想結(jié)合在一起,應(yīng)用于實(shí)際的項(xiàng)目,那是第三步。很多朋友都是從PC編程轉(zhuǎn)向嵌入式編程的。在中國,嵌入式編程的朋友很少是正兒八經(jīng)從計算機(jī)專業(yè)畢業(yè)的,都是從自動控制啊,電子相關(guān)的專業(yè)畢業(yè)的。這些童鞋們,實(shí)踐經(jīng)驗(yàn)雄厚,但是理論知識缺乏;計算機(jī)專業(yè)畢業(yè)的童鞋很大一部分去弄網(wǎng)游、網(wǎng)頁這些獨(dú)立于操作系統(tǒng)的更高層的應(yīng)用了。也不太愿意從事嵌入式行業(yè),畢竟這條路不好走。他們理論知識雄厚,但缺乏電路等相關(guān)的知識,在嵌入式里學(xué)習(xí)需要再學(xué)習(xí)一些具體的知識,比較難走。

雖然沒有做過產(chǎn)業(yè)調(diào)查,但從我所見和所招聘人員,從事嵌入式行業(yè)的工程師,要么缺乏理論知識,要么缺乏實(shí)踐經(jīng)驗(yàn)。很少兩者兼?zhèn)涞?。究其原因,還是中國的大學(xué)教育的問題。這里不探討這個問題,避免口水戰(zhàn)。我想列出我實(shí)踐中的幾個例子。引起大家在嵌入式中做項(xiàng)目時對一些問題的關(guān)注。

第一個問題:

同事在uC/OS-II下開發(fā)一個串口的驅(qū)動程序,驅(qū)動和接口在測試中均為發(fā)現(xiàn)問題。應(yīng)用中開發(fā)了個通訊程序,串口驅(qū)動提供了一個查詢驅(qū)動緩沖區(qū)字符的函數(shù):GetRxBuffCharNum()。 高層需要接受一定數(shù)量的字符以后才能對包做解析。一個同事撰寫的代碼,用偽代碼表示如下:

bExit = FALSE;

do {

if (GetRxBuffCharNum() >= 30)

bExit = ReadRxBuff(buff, GetRxBuffCharNum());

} while (!bExit);

這段代碼判斷當(dāng)前緩沖區(qū)中超過30個字符,就將緩沖區(qū)中全部字符讀到緩沖區(qū)中,直到讀取成功為止。邏輯清楚,思路也清楚。但這段代碼是不能正常工作。如果是在PC機(jī)上,定然是沒有任何問題,工作的異常正常。但在嵌入式里真的是不得而知了。同事很郁悶,不知道為什么。來請我解決問題,當(dāng)時我看到代碼,就問了他,GetRxBuffCharNum()是怎么實(shí)現(xiàn)的?打開一看:

unsigned GetRxBuffCharNum(void)

{

cpu_register reg;

unsigned num;

reg = interrupt_disable();

num = gRxBuffCharNum;

interrupt_enable(reg);

return (num);

}

很明顯,由于在循環(huán)中,interruput_disable()和interrupt_enable()之間是個全局臨界區(qū)域,保證gRxBufCharNum的完整性。但是,由于在外層的do { } while() 循環(huán)中,CPU頻繁的關(guān)閉中斷,打開中斷,這個時間非常的短。實(shí)際上CPU可能不能正常的響應(yīng)UART的中斷。當(dāng)然這和uart的波特率、硬件緩沖區(qū)的大小還有CPU的速度都有關(guān)系。我們使用的波特率非常高,大約有3Mbps。uart起始信號和停止信號占一個比特位。一個字節(jié)需要消耗10個周期。3Mbps的波特率大約需要3.3us傳輸一個字節(jié)。3.3us能執(zhí)行多少個CPU指令呢?100MHz的ARM,大約能執(zhí)行150條指令左右。結(jié)果關(guān)閉中斷的時間是多長呢?一般ARM關(guān)閉中斷都需要4條以上的指令,打開又有4條以上的指令。接收uart中斷的代碼實(shí)際上是不止20條指令的。所以,這樣下來,就有可能出現(xiàn)丟失通信數(shù)據(jù)的Bug,體現(xiàn)在系統(tǒng)層面上,就是通信不穩(wěn)定。

修改這段代碼其實(shí)很簡單,最簡單的辦法是從高層修改。即:

bExit = FALSE;

do {

DelayUs(20); //延時 20us,一般采用空循環(huán)指令實(shí)現(xiàn)

num = GetRxBuffCharNum();

if (num >= 30)

bExit = ReadRxBuff(buff, num);

} while (!bExit);

這樣,讓CPU有時間去執(zhí)行中斷的代碼,從而避免了頻繁關(guān)閉中斷造成的中斷代碼執(zhí)行不及時,產(chǎn)生的信息丟失。在嵌入式系統(tǒng)里,大部分的RTOS應(yīng)用都是不帶串口驅(qū)動。自己設(shè)計代碼時,沒有充分考慮代碼與內(nèi)核的結(jié)合。造成代碼深層次的問題。RTOS之所以稱為RTOS,就是因?yàn)閷κ录目焖夙憫?yīng);事件快速的響應(yīng)依賴于CPU對中斷的響應(yīng)速度。驅(qū)動在Linux這種系統(tǒng)中都是與內(nèi)核高度整合,一起運(yùn)行在內(nèi)核態(tài)。RTOS雖然不能抄襲linux這種結(jié)構(gòu),但有一定的借鑒意義。

從上面的例子可以看清楚,嵌入式需要開發(fā)人員對代碼的各個環(huán)節(jié)需要了解清楚。

第二個例子:

同事驅(qū)動一個14094串轉(zhuǎn)并的芯片。串行信號是采用IO模擬的,因?yàn)闆]有專用的硬件。同事就隨手寫了個驅(qū)動,結(jié)果調(diào)試了3、4天,仍舊是有問題。我實(shí)在看不下去了,就去看了看,控制的并行信號有時候正常有時候不正常。我看了看代碼,用偽代碼大概是:

for (i = 0; i < 8; i++)

{

SetData((data >> i) & 0x1);

SetClockHigh();

for (j = 0; j < 5; j++);

SetClockLow();

}

將數(shù)據(jù)的8個bit在每個高電平從bit0到bit7依次發(fā)送出去。應(yīng)該是正常的啊??床怀鰡栴}在哪啊?我仔細(xì)想了想,有看了14094的datasheet,明白了。原來,14094要求clock的高電平持續(xù)10個ns,低電平也要持續(xù)10個ns。這段代碼之做了高電平時間的延時,沒有做低電平的延時。如果中斷插在低電平之間工作,那么這段代碼是可以的。但是如果CPU沒有中斷插在低電平時執(zhí)行,則是不能正常工作的。所以就時好時壞。

修改也比較簡單:

for (i = 0; i < 8; i++)

{

SetData((data >> i) & 0x1);

SetClockHigh();

for (j = 0; j < 5; j++);

SetClockLow();

for (j = 0; j < 5; j++);

}

這樣就完全正常了。但是這個還是不能很好移植的一個代碼,因?yàn)榫幾g器一優(yōu)化,就有可能造成這兩個延時循環(huán)的丟失。丟失了,就不能保證高電平低電平持續(xù)10ns的要求,也就不能正常工作了。所以,真正的可以移植的代碼,應(yīng)該把這個循環(huán)做成一個納秒級的DelayNs(10);

像Linux一樣,上電時,先測量一下,nop指令執(zhí)行需要多長時間執(zhí)行,多少個nop指令執(zhí)行10ns。執(zhí)行一定的nop指令就可以了。利用編譯器防止優(yōu)化的編譯指令或者特殊的關(guān)鍵字,防止延時循環(huán)被編譯器優(yōu)化掉。如GCC中的

__volatile__ __asm__("nop;\n");

從這個例子中可以清楚的看到,寫好一段好代碼,是需要很多知識支撐的。
 

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運(yùn)營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉