• <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牛人

            搜索

            積分與排名

            最新評論

            閱讀排行榜

            評論排行榜

            久久久久久精品久久久久| 韩国无遮挡三级久久| 嫩草影院久久99| 精品九九久久国内精品| 婷婷综合久久中文字幕蜜桃三电影| 久久精品国产精品亚洲下载| 99久久国产热无码精品免费久久久久| 国产精品一区二区久久国产| 精品久久久久久无码中文字幕一区| 久久综合久久自在自线精品自 | 大美女久久久久久j久久| 久久亚洲国产午夜精品理论片 | 亚洲国产美女精品久久久久∴| 无码人妻久久一区二区三区免费 | www.久久热.com| 伊人丁香狠狠色综合久久| 国产精品激情综合久久| 日日噜噜夜夜狠狠久久丁香五月| 久久99国产精品一区二区| 中文精品99久久国产 | 久久精品国产精品亚洲精品| 国产∨亚洲V天堂无码久久久| 亚洲欧美日韩精品久久| 久久性生大片免费观看性| 日韩欧美亚洲综合久久| 国产精品久久永久免费| 久久综合色之久久综合| 激情伊人五月天久久综合| 日韩亚洲欧美久久久www综合网 | 亚洲精品无码久久久| 亚洲AV成人无码久久精品老人| 久久综合综合久久狠狠狠97色88| 久久久不卡国产精品一区二区| 久久精品国产亚洲AV蜜臀色欲 | 色天使久久综合网天天| 久久电影网一区| 精品久久久无码21p发布| 品成人欧美大片久久国产欧美| 中文字幕乱码人妻无码久久| 久久香蕉一级毛片| 麻豆亚洲AV永久无码精品久久|