• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            milkyway的窩

            最初想法的誕生地

             

            WinCE中OEM適配層編程點滴之創(chuàng)建OAL

            付林林:

              2001年計算機(jī)專業(yè)畢業(yè)。從畢業(yè)起一直從事軟件開發(fā)工作。目前從事 Windows CE 下操作系統(tǒng)內(nèi)核定制和應(yīng)用程序開發(fā)。在實際工作中積累了CE下開發(fā)的一些經(jīng)驗。希望和 CE 下開發(fā)者交流、探討,更希望你們能不吝賜教。我的EMail:windowsce@tom.com

              如果您有技術(shù)問題向我咨詢,請登錄天極網(wǎng)嵌入式開發(fā)論壇,本人將在此論壇回復(fù)您的問題。在論壇上交流會更方便些,其它網(wǎng)友也可以回答參與,彌補(bǔ)了我的不足。

              進(jìn)入作者專欄

              正文

              正如CE的幫助文檔所言,創(chuàng)建OAL是一個非常復(fù)雜的任務(wù),而通常的辦法是復(fù)制原有的相同平臺的OAL代碼,然后修改來適應(yīng)平臺的特殊要求。也就是說對于沒有特殊要求的平臺,復(fù)制原有相同平臺的OAL代碼就足夠了。由于OAL的復(fù)雜性在這篇文章中我只講解常用的部分。

              一、實現(xiàn)ISR

              1. ISR的概念

              ISR(interrupt service routine)是處理IRQs(interrupt request line)的程序。Windows CE用一個ISR來處理所有的IRQ請求。當(dāng)一個中斷發(fā)生時,內(nèi)核的異常處理程序先調(diào)用內(nèi)核ISR,內(nèi)核ISR禁用所有具有相同優(yōu)先級和較低優(yōu)先級的中斷,然后調(diào)用已經(jīng)注冊的OAL ISR程序,一般ISR有下列特征:

              1) 執(zhí)行最小的中斷處理,最小的中斷處理指能夠檢驗、答復(fù)產(chǎn)生中斷的硬件,而把更多的處理工作留給IST(interrupt service thread)。

              2) 當(dāng)ISR完成時返回中斷ID(中斷ID大部分是預(yù)定義的)。

              2. X86平臺的ISR結(jié)構(gòu)

              X86平臺的ISR保存在%_WINCEROOT%\PUBLIC\COMMON\OAK\CSP\I486\OAL\fwpc.c中,函數(shù)名為PeRPISR。下面分析一下此函數(shù)的主要代碼:

            ULONG PeRPISR(void)
            {
             ULONG ulRet = SYSINTR_NOP; ///返回值,既中斷ID(以SYSINTR_為前綴)
             UCHAR ucCurrentInterrupt; ///當(dāng)前中斷號
             if (fIntrTime) ////// fIntrTime 用于測試SR和IST的延時時間,測試工具為ILTiming.exe。
              ......
              ucCurrentInterrupt = PICGetCurrentInterrupt(); ////返回當(dāng)前中斷IRQ
             if (ucCurrentInterrupt == INTR_TIMER0) ///IRQ0,IRQ0為系統(tǒng)時鐘(system tick)中斷,具體見“二、實現(xiàn)系統(tǒng)時鐘”
             ......
             if (dwRebootAddress) ////是否需要重啟動
              RebootHandler();
              ......
             if(ucCurrentInterrupt == INTR_RTC) ////IRQ8,real-time clock的中斷
              ......
             else if (ucCurrentInterrupt <= INTR_MAXIMUM) ///如果中斷小于 INTR_MAXIMUM
             {
              ulRet = NKCallIntChain(ucCurrentInterrupt); ////調(diào)用中斷鏈
              if (ulRet == SYSINTR_CHAIN) ///如果中斷鏈未包含中斷
               ulRet = OEMTranslateIrq(ucCurrentInterrupt); ////在IRQ 和SYSINTR之間轉(zhuǎn)換,此函數(shù)返回IRQ對應(yīng)的SYSINTR
               ......
               PICEnableInterrupt(ucCurrentInterrupt, FALSE); ///啟用除當(dāng)前中斷以外的所有中斷
             } ///else if
             OEMIndicateIntSource(ulRet); ///通知內(nèi)核已經(jīng)發(fā)生SYSINTR中斷
            }

              從以上代碼不難看出ISR的任務(wù)就是返回以“SYSINTR_”為前綴的中斷ID,如果不需要進(jìn)一步執(zhí)行IST,那么就返回SYSINTR_NOP。

              3. 中斷注冊步驟

              參考X86平臺的代碼,中斷注冊步驟如下:

              1) 用SETUP_INTERRUPT_MAP宏關(guān)聯(lián)SYSINTR和IRQ。以“SYSINTR_”為前綴的常量由內(nèi)核使用,用于唯一標(biāo)識發(fā)生中斷的硬件。在Nkintr.h文件中預(yù)定義了一些SYSINTR,OEM可以在Oalintr.h文件中自定義SYSINTR。

              2) 用HookInterrupt函數(shù)關(guān)聯(lián)硬件中斷號和ISR。這里的硬件中斷號為物理中斷號,而非邏輯中斷號IRQ。在InitPICs函數(shù)(和上述ISR位于同一文件)的最后調(diào)用了HookInterrupt函數(shù),如下:

            for (i = 64; i < 80; i++)
             HookInterrupt(i, (void *)PeRPISR); ///用ISR關(guān)聯(lián)16個中斷號

              4. 中斷處理步驟

              1) 調(diào)用InterruptInitialize函數(shù)關(guān)聯(lián)SYSINTR和IST,具體是關(guān)聯(lián)IST等待的事件。一般在驅(qū)動程序中按如下編寫:

            hEvent = CreateEvent(...) ///創(chuàng)建一個事件對象
            InterruptInitialize(SYSINTR_SERIAL, hEvent, ...) ///關(guān)聯(lián)一個串口中斷ID和這個事件
            hThd = CreateThread(..., MyISTRoutine, hEvent, ...) ///創(chuàng)建一個線程(IST)
            CeSetThreadPriority(hThd, 152); ///提高此線程的優(yōu)先級

              2) IST執(zhí)行I/O操作,一般IST按如下編寫:

            for(;;) ///驅(qū)動程序一直處于服務(wù)狀態(tài)
            {
             WaitForSingleObject(hEvent, INFINITE); ////無限等待事件
             ...... //// I/O操作
             InterruptDone(InterruptId); ///結(jié)束當(dāng)前中斷處理
            }

              3) ISR和IST之間數(shù)據(jù)傳輸

              假如我們要從一個設(shè)備頻繁的讀取數(shù)據(jù)而每次讀取量非常少,那么每次讀取都要調(diào)用IST會降低性能。作為解決方案,ISR可以做讀取工作(存放到緩沖區(qū)),并在緩沖區(qū)存放滿后由IST到緩沖區(qū)讀取。因為ISR運行在內(nèi)核模式而IST運行在用戶模式,IST不能輕易地訪問ISR的緩沖區(qū),為此CE提供了一個辦法(參見標(biāo)題為“Passing Data between an ISR and an IST”的幫助文檔),您也可以到天極網(wǎng)嵌入式開發(fā)論壇詢問。

              二、實現(xiàn)系統(tǒng)時鐘

              1. 系統(tǒng)時鐘(system tick)概念

              系統(tǒng)時鐘是內(nèi)核需要的唯一中斷(IRQ0),系統(tǒng)時鐘每毫秒產(chǎn)生一個中斷,當(dāng)發(fā)生中斷時內(nèi)核在ISR中累計,到1000的倍數(shù)就是過了一秒鐘。在處理系統(tǒng)時鐘的ISR中不僅要累計計數(shù),還要決定是否通知內(nèi)核開始重新調(diào)度當(dāng)前所有的線程。要實現(xiàn)一個OAL,系統(tǒng)時鐘是第一個必須做的事。

              2. X86平臺系統(tǒng)時鐘中斷的處理工作 系統(tǒng)時鐘由InitClock函數(shù)負(fù)責(zé)初始化工作,一般是在OEMInit函數(shù)中調(diào)用。當(dāng)發(fā)生中斷時,ISR首先用下列語句累計計數(shù):

            CurMSec += SYSTEM_TICK_MS; /////SYSTEM_TICK_MS = 1

              然后根據(jù)下列語句判斷應(yīng)該返回什么值:

            if ((int) (dwReschedTime – CurMSec) >= 0)
             return SYSINTR_RESCHED; ///重新調(diào)度
            else
             return SYSINTR_NOP; ///不再執(zhí)行任何操作

              上述代碼中全局變量dwReschedTime在schedule.c中定義,也就是由內(nèi)核的調(diào)度模塊決定在何時開始重新調(diào)度線程。CurMSec累計了從WindowsCE啟動到當(dāng)前總共產(chǎn)生了多少個system tick。實現(xiàn)系統(tǒng)時鐘后還要實現(xiàn)OEMIdle函數(shù),當(dāng)沒有線程準(zhǔn)備運行時OEMIdle被調(diào)用,OEMIdle函數(shù)將CPU置于空閑模式,但在空閑模式下仍然要累計系統(tǒng)時鐘。

              三、I/O控制代碼

              1. I/O控制代碼作用

              應(yīng)用軟件或者驅(qū)動程序可以調(diào)用KernelIoControl函數(shù)與OAL層通信,而KernelIoControl在內(nèi)部調(diào)用OEMIoControl函數(shù)。OEMIoControl是一個OAL API,OEM可以在OEMIoControl中編寫自己的I/O控制代碼實現(xiàn)一些功能,或者說與應(yīng)用軟件通信。I/O控制代碼常用的例子如重啟計算機(jī)、得到系統(tǒng)信息、設(shè)置RTC、得到設(shè)備ID等。還有一些系統(tǒng)程序使用的特殊的I/O控制代碼。在這里說明一下,我經(jīng)過實驗證實CE提供的得到設(shè)備ID方法并非有效。

              2. 編寫自己的I/O控制代碼步驟

              1) 在pkfuncs.h或者新編寫一個.h文件中按如下格式定義:

            #define IOCTL_MY_CONTROL CTL_CODE(FILE_DEVICE_HAL, 3000, METHOD_NEITHER, FILE_ANY_ACCESS)

              2) 在oemioctl.c中修改OEMIoControl函數(shù),添加如下代碼:

            case IOCTL_MY_CONTROL:

            ......

              3) 在應(yīng)用程序中調(diào)用KernelIoControl函數(shù),具體參數(shù)參見幫助文檔

            posted on 2007-02-02 09:23 milkyway 閱讀(1176) 評論(0)  編輯 收藏 引用 所屬分類: wince(別人的文章技巧總結(jié))

            導(dǎo)航

            統(tǒng)計

            公告

            隨筆皆原創(chuàng),文章乃轉(zhuǎn)載. 歡迎留言!

            常用鏈接

            留言簿(37)

            隨筆分類(104)

            隨筆檔案(101)

            文章分類(51)

            文章檔案(53)

            wince牛人

            搜索

            積分與排名

            最新評論

            閱讀排行榜

            評論排行榜

            7国产欧美日韩综合天堂中文久久久久| 亚洲国产精品久久久久婷婷软件| 亚洲国产成人乱码精品女人久久久不卡 | 国产精品久久久久蜜芽| 伊人热热久久原色播放www| 亚洲国产精品一区二区久久hs| 99久久香蕉国产线看观香| 午夜人妻久久久久久久久| 99精品伊人久久久大香线蕉| 久久精品桃花综合| 日本久久久久久中文字幕| 中文成人无码精品久久久不卡| 久久久久亚洲av无码专区导航| 久久播电影网| 国产亚洲精品美女久久久| 亚洲国产精品综合久久一线| 97精品伊人久久大香线蕉app| 日韩电影久久久被窝网| 2021久久国自产拍精品| 伊人久久大香线蕉AV一区二区| 日韩欧美亚洲综合久久影院d3| 日本WV一本一道久久香蕉| 久久99亚洲综合精品首页| 狠狠色丁香久久综合婷婷| 日韩精品无码久久久久久| 四虎久久影院| 亚洲国产成人精品无码久久久久久综合 | 性高湖久久久久久久久AAAAA| 国产精品久久午夜夜伦鲁鲁| 久久这里都是精品| 亚洲国产成人精品女人久久久| 99久久精品国产一区二区蜜芽 | 日本精品久久久久中文字幕8| 亚洲AV日韩精品久久久久久久| 亚洲精品无码久久毛片| 久久久久女教师免费一区| 精品乱码久久久久久夜夜嗨 | 色老头网站久久网| 亚洲欧美精品一区久久中文字幕 | 久久99热狠狠色精品一区| 久久er国产精品免费观看2|