0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

用實(shí)例引起大家在嵌入式中做項(xiàng)目時(shí)對(duì)一些問題的關(guān)注

m3eY_edn_china ? 2018-01-05 11:31 ? 次閱讀

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

能從PC機(jī)器編程去看嵌入式問題,那是第一步;學(xué)會(huì)用嵌入式編程思想,那是第二步;用PC的思想和嵌入式的思想結(jié)合在一起,應(yīng)用于實(shí)際的項(xiàng)目,那是第三步。

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

第一個(gè)問題:

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

bExit = FALSE;

do {

if (GetRxBuffCharNum() >= 30)

bExit = ReadRxBuff(buff, GetRxBuffCharNum());

} while (!bExit);

這段代碼判斷當(dāng)前緩沖區(qū)中超過30個(gè)字符,就將緩沖區(qū)中全部字符讀到緩沖區(qū)中,直到讀取成功為止。邏輯清楚,思路也清楚。但這段代碼是不能正常工作。如果是在PC機(jī)上,定然是沒有任何問題,工作的異常正常。但在嵌入式里真的是不得而知了。同事很郁悶,不知道為什么。來請(qǐng)我解決問題,當(dāng)時(shí)我看到代碼,就問了他,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()之間是個(gè)全局臨界區(qū)域,保證gRxBufCharNum的完整性。但是,由于在外層的do { } while() 循環(huán)中,CPU頻繁的關(guān)閉中斷,打開中斷,這個(gè)時(shí)間非常的短。實(shí)際上CPU可能不能正常的響應(yīng)UART的中斷。當(dāng)然這和uart的波特率、硬件緩沖區(qū)的大小還有CPU的速度都有關(guān)系。我們使用的波特率非常高,大約有3Mbps。uart起始信號(hào)和停止信號(hào)占一個(gè)比特位。一個(gè)字節(jié)需要消耗10個(gè)周期。3Mbps的波特率大約需要3.3us傳輸一個(gè)字節(jié)。3.3us能執(zhí)行多少個(gè)CPU指令呢?100MHz的ARM,大約能執(zhí)行150條指令左右。結(jié)果關(guān)閉中斷的時(shí)間是多長呢?一般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); //延時(shí) 20us,一般采用空循環(huán)指令實(shí)現(xiàn)

num = GetRxBuffCharNum();

if (num >= 30)

bExit = ReadRxBuff(buff, num);

} while (!bExit);

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

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

第二個(gè)例子:

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

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

{

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

SetClockHigh();

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

SetClockLow();

}

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

修改也比較簡單:

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

{

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

SetClockHigh();

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

SetClockLow();

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

}

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

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

__volatile__ __asm__("nop;\n");

從這個(gè)例子中可以清楚的看到,寫好一段好代碼,是需要很多知識(shí)支撐的。你說呢?

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 嵌入式
    +關(guān)注

    關(guān)注

    5052

    文章

    18913

    瀏覽量

    300832
  • PC
    PC
    +關(guān)注

    關(guān)注

    9

    文章

    2049

    瀏覽量

    153816

原文標(biāo)題:高手閑談嵌入式編程和PC編程的區(qū)別,實(shí)例分析一波...

文章出處:【微信號(hào):edn-china,微信公眾號(hào):EDN電子技術(shù)設(shè)計(jì)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    關(guān)于嵌入式培訓(xùn)的一些看法

    標(biāo)榜自己是真正的嵌入式培訓(xùn)學(xué)校。有各種的認(rèn)證體系。對(duì)于這個(gè)問題和學(xué)校的老師做了一些討論?,F(xiàn)給大家一些建議:首先,要看看他的課程體系。因?yàn)?b class='flag-5'>嵌入式
    發(fā)表于 02-02 15:08

    求教嵌入式和ARM的一些問題

    搞定了51單片機(jī),學(xué)校的其他專業(yè)課程又沒什么興趣,嵌入式系統(tǒng)下學(xué)期才有課,本來想深入學(xué)習(xí)一些別的單片機(jī)比如AVR什么的,但是后來想想還是想學(xué)嵌入式先。本人對(duì)編程還是很有信心的,但是嵌入式
    發(fā)表于 11-08 21:20

    對(duì)于嵌入式一些認(rèn)識(shí)

    本來學(xué)嵌入式就不簡單,所以大家都會(huì)去找一些學(xué)習(xí)嵌入式的方法或步驟什么的,如果是你去學(xué)嵌入式的話,肯定也不會(huì)例外,下面可以
    發(fā)表于 05-25 17:40

    嵌入式開發(fā)一些硬件相關(guān)的概念有哪些

    嵌入式系統(tǒng)開發(fā),經(jīng)常要接觸硬件。嵌入式開發(fā)對(duì)數(shù)字電路和模擬電路要有定的了解。這樣才能深入的研究下去。下面我們簡單的介紹
    發(fā)表于 02-24 08:56

    時(shí)鐘設(shè)計(jì)應(yīng)該注意的一些問題

    1. 前言時(shí)鐘是嵌入式系統(tǒng)中非常重要,但又常常被忽視的地方。它的成本只占整個(gè)系統(tǒng)的個(gè)零頭,但由于時(shí)鐘的停振,或其它異常最終導(dǎo)致廠商付出高昂代價(jià)的案例卻并不少見。下面我們看一下在時(shí)鐘設(shè)計(jì)
    發(fā)表于 07-21 09:01

    學(xué)習(xí)嵌入式linux上的一些問題分享

    不知不覺,博客似乎也成為了自己記錄學(xué)習(xí)道路上的一些迷茫的個(gè)小日記??赡芤彩亲约阂咔槠陂g在家呆了4個(gè)月的時(shí)間,也可能是最近課程壓力比較大,或者是最近和導(dǎo)師聊了下之后的想法,還有最近自己學(xué)習(xí)嵌入式
    發(fā)表于 11-04 06:50

    梳理嵌入式Linux的一些知識(shí)

    前言這篇文章簡單我們來起梳理嵌入式Linux的一些知識(shí),方便于一些想跟我樣想要由單片機(jī)進(jìn)階到嵌入式
    發(fā)表于 11-04 07:33

    如何解決Linux嵌入式開發(fā)中斷按鍵實(shí)驗(yàn)碰到的一些問題?

    如何解決Linux嵌入式開發(fā)中斷按鍵實(shí)驗(yàn)碰到的一些問題
    發(fā)表于 11-30 06:51

    嵌入式項(xiàng)目中一些常用知識(shí)及技巧是什么

    大家好,我是ZhengN。本次給大家分享一些C語言、嵌入式項(xiàng)目中的常用知識(shí)。1、使用宏給結(jié)構(gòu)體初始化 如果頻繁使用
    發(fā)表于 12-15 07:10

    嵌入式一些日常小技巧分享

    雜記–算法學(xué)習(xí)日志+關(guān)于實(shí)驗(yàn)室部門架構(gòu)的思考+嵌入式一些日常小技巧分享算法學(xué)習(xí)日志因?yàn)閺某踔械浆F(xiàn)在直是學(xué)習(xí)嵌入式程序的開發(fā)和研究,并沒有
    發(fā)表于 12-21 08:07

    cadence畫板遇到的一些問題

    cadence畫板遇到的一些問題,在這里分享給大家,多多指教
    發(fā)表于 02-25 16:01 ?0次下載

    解析嵌入式開發(fā)一些硬件相關(guān)概念

    嵌入式系統(tǒng)開發(fā),經(jīng)常要接觸硬件。嵌入式開發(fā)對(duì)數(shù)字電路和模擬電路要有定的了解。這樣才能深入的研究下去。下面我們簡單的介紹
    發(fā)表于 06-06 16:35 ?880次閱讀
    解析<b class='flag-5'>嵌入式</b>開發(fā)<b class='flag-5'>中</b>的<b class='flag-5'>一些</b>硬件相關(guān)概念

    嵌入式系統(tǒng)的一些基礎(chǔ)知識(shí)

    本文主要介紹嵌入式系統(tǒng)的一些基礎(chǔ)知識(shí),希望對(duì)各位有幫助。
    的頭像 發(fā)表于 02-18 10:44 ?2227次閱讀

    DSP嵌入式系統(tǒng)的應(yīng)用實(shí)例

    DSP嵌入式系統(tǒng)的應(yīng)用實(shí)例(嵌入式開發(fā)工具軟件)-該文檔為DSP
    發(fā)表于 07-30 13:09 ?22次下載
    DSP<b class='flag-5'>在</b><b class='flag-5'>嵌入式</b>系統(tǒng)<b class='flag-5'>中</b>的應(yīng)用<b class='flag-5'>實(shí)例</b>

    嵌入式一些概念

    嵌入式一些概念嵌入式系統(tǒng)的概念什么是MCU、CPU、MPU、DSPMCUMPUDSP幾大MCU公司瑞薩電子(Renesas)恩智浦(NXP)+飛思卡爾(Freescale)(后者被
    發(fā)表于 10-19 18:33 ?8次下載
    <b class='flag-5'>嵌入式</b><b class='flag-5'>中</b>的<b class='flag-5'>一些</b>概念