學(xué)會用嵌入式編程思想,那是第二步;
用PC的思想和嵌入式的思想結(jié)合在一起,應(yīng)用于實際的項目,那是第三步。
很多朋友都是從PC編程轉(zhuǎn)向嵌入式編程的。
在中國,嵌入式編程的朋友很少是正兒八經(jīng)從計算機專業(yè)畢業(yè)的,都是從自動控制,電子相關(guān)的專業(yè)畢業(yè)的。
這些童鞋們,實踐經(jīng)驗雄厚,但是理論知識缺乏;計算機專業(yè)畢業(yè)的童鞋很大一部分去弄網(wǎng)游、網(wǎng)頁這些獨立于操作系統(tǒng)的更高層的應(yīng)用了。
也不太愿意從事嵌入式行業(yè),畢竟這條路不好走。他們理論知識雄厚,但缺乏電路等相關(guān)的知識,在嵌入式里學(xué)習(xí)需要再學(xué)習(xí)一些具體的知識,比較難走。
雖然沒有做過產(chǎn)業(yè)調(diào)查,但從我所見和所招聘人員,從事嵌入式行業(yè)的工程師,要么缺乏理論知識,要么缺乏實踐經(jīng)驗。
很少兩者兼?zhèn)涞?。究其原因,還是中國的大學(xué)教育的問題。這里不探討這個問題,避免口水戰(zhàn)。我想列出我實踐中的幾個例子。引起大家在嵌入式中做項目時對一些問題的關(guān)注。
第一個例子:
同事在uC/OS-II下開發(fā)一個串口的驅(qū)動程序,驅(qū)動和接口在測試中均未發(fā)現(xiàn)問題。應(yīng)用中開發(fā)了個通訊程序,串口驅(qū)動提供了一個查詢驅(qū)動緩沖區(qū)字符的函數(shù):GetRxBuffCharNum()。
應(yīng)用層需要接收一定數(shù)量的字符以后才能對包做解析。一個同事撰寫的代碼,用偽代碼表示如下:

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

很明顯,由于在循環(huán)中,interruput_disable() 和 interrupt_enable() 之間是個全局臨界區(qū)域,保證 gRxBufCharNum 的完整性。
但是,由于在外層的do { } while() 循環(huán)中,CPU頻繁地關(guān)閉中斷,打開中斷,這個時間非常的短。
實際上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條指令左右。關(guān)閉中斷的時間是多長呢?一般ARM關(guān)閉中斷都需要 4 條以上的指令,打開又有 4 條以上的指令。
接收 uart 中斷的代碼實際上是不止 20 條指令的。所以,這樣下來,就有可能出現(xiàn)丟失通信數(shù)據(jù)的Bug,體現(xiàn)在系統(tǒng)層面上,就是通信不穩(wěn)定。
修改這段代碼其實很簡單,最簡單的辦法是從高層修改。即:

這樣,讓CPU有時間去執(zhí)行中斷的代碼,從而避免了頻繁關(guān)閉中斷造成的中斷代碼執(zhí)行不及時,產(chǎn)生的信息丟失。
在嵌入式系統(tǒng)里,大部分的RTOS應(yīng)用都是不帶串口驅(qū)動。自己設(shè)計代碼時,沒有充分考慮代碼與內(nèi)核的結(jié)合。會造成代碼深層次的問題。
RTOS之所以稱為RTOS,就是因為對事件的快速響應(yīng);事件快速的響應(yīng)依賴于CPU對中斷的響應(yīng)速度。
驅(qū)動在Linux這種系統(tǒng)中都是與內(nèi)核高度整合,一起運行在內(nèi)核態(tài)。RTOS雖然不能抄襲linux這種結(jié)構(gòu),但有一定的借鑒意義。
從上面的例子可以看清楚,嵌入式需要開發(fā)人員對代碼的各個環(huán)節(jié)需要了解清楚。
第二個例子:
同事驅(qū)動一個14094串轉(zhuǎn)并的芯片。串行信號是采用IO模擬的,因為沒有專用的硬件。同事就隨手寫了個驅(qū)動,結(jié)果調(diào)試了3、4天,仍舊是有問題。
我實在看不下去了,就去看了看,控制的并行信號有時候正常有時候不正常。我看了看代碼,用偽代碼大概是:

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

這樣就完全正常了。但是這個還是不能很好移植的代碼,因為編譯器一優(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");
從這個例子中可以清楚的看到,寫好一段好代碼,是需要很多知識支撐的。你說呢?
-
PC
+關(guān)注
關(guān)注
9文章
2167瀏覽量
159312 -
編程
+關(guān)注
關(guān)注
90文章
3716瀏覽量
97178 -
嵌入式編程
+關(guān)注
關(guān)注
0文章
27瀏覽量
10696
發(fā)布評論請先 登錄
嵌入式編程和PC編程,不同點有多少?
【我是電子發(fā)燒友】嵌入式編程與PC編程有何區(qū)別?
嵌入式學(xué)習(xí)由淺入深的必經(jīng)之路
從項目出發(fā),分析嵌入式編程和PC機器編程的不同
嵌入式編程和PC編程的區(qū)別在哪
什么是嵌入式編程
嵌入式編程和PC編程究竟有何區(qū)別?
嵌入式C++編程
怎么從PC編程轉(zhuǎn)向嵌入式編程
評論