ARM和WinCE6.0下nRF24L01的驅(qū)動設(shè)計
摘要:隨著嵌入式WinCE系統(tǒng)的發(fā)展和近距離無線通信的廣泛使用,有必要為近距離無線通信芯片編寫驅(qū)動。由于nRF24L01在WinCE系統(tǒng)下驅(qū)動的缺乏,使得WinCE與nRF24L01芯片之間的直接通信成為困難。本文的驅(qū)動設(shè)計正好解決了這一難題。
關(guān)鍵詞:WinCE;nRF24L01;驅(qū)動程序
引言
nRF24L01是一款工作在2.4~2.5 GHz世界通用ISM頻段的單片無線收發(fā)器芯片。它在無線數(shù)據(jù)通訊、無線門禁、遙感勘測、工業(yè)傳感器和玩具中都有應(yīng)用。
隨著測控技術(shù)的發(fā)展,nRF24L01與單片機組成的系統(tǒng)進行無線測控的實例已經(jīng)有很多,如基于nRF24L01的無線溫度監(jiān)測系統(tǒng)、基于nRF24 L01的近距離無線數(shù)據(jù)傳輸系統(tǒng)等等。近年來,隨著ARM和嵌入式WinCE系統(tǒng)的迅速發(fā)展,由于在WinCE系統(tǒng)下nRF24L01和ARM的通信缺乏驅(qū)動,致使兩者之間不能直接通信,一般的解決方法是借助于第三方單片機的串行口來進行兩者的通信。這種方式的不足之處在于,由于要借助于第三方單片機,使得成本增加且通信速度下降。nRF24L01驅(qū)動的設(shè)計與實現(xiàn)正好添補了這一空白,使WinCE系統(tǒng)和nRF24L01可以直接通信進行無線數(shù)據(jù)收發(fā)。
1 WinCE流式驅(qū)動模型
在WinCE系統(tǒng)中有4種驅(qū)動模型,其中流式接口設(shè)備驅(qū)動模型和本地設(shè)備驅(qū)動模型是可以由用戶來開發(fā)的。nRF24L01可以看作是一種流式接口設(shè)備,所以選用流式接口設(shè)備驅(qū)動模型來編寫驅(qū)動較為合適。
1.1 流式接口驅(qū)動程序的體系結(jié)構(gòu)
如圖1所示,流式接口驅(qū)動程序的體系結(jié)構(gòu)由5部分組成,分別是應(yīng)用程序、FileSys.exe、設(shè)備管理器、流式接口驅(qū)動程序和硬件。
圖1中陰影部分是由用戶編寫程序來完成,白色部分是由WinCE系統(tǒng)自身提供,用戶不能修改。在應(yīng)用程序使用文件API對設(shè)備進行訪問時,文件API調(diào)用被操作系統(tǒng)轉(zhuǎn)發(fā)到FileSys.exe進程中;若發(fā)現(xiàn)是對設(shè)備的訪問,則FileSys.exe就會把操作交給設(shè)備管理器;然后,設(shè)備管理器根據(jù)具體操作調(diào)用不同流式接口驅(qū)動中提供的接口;最后,由流式接口驅(qū)動程序負責(zé)與硬件交互。
從體系結(jié)構(gòu)中可知,流式接口驅(qū)動體系是把設(shè)備抽象作為文件進行操作的。所以編寫流式接口驅(qū)動就是將文件操作應(yīng)有的打開、關(guān)閉、讀取、寫入和移動文件指針的操作在驅(qū)動程序中以函數(shù)的形式實現(xiàn),這些函數(shù)有其標(biāo)準的定義形式,稱之為流接口函數(shù)。但對于驅(qū)動程序而言除了和文件一樣的標(biāo)準操作外還需要驅(qū)動程序的加載、卸載等流接口函數(shù)。下面將介紹nRF24L01驅(qū)動中主要用到的流接口函數(shù):
①XXX_Open()函數(shù),功能是打開設(shè)備準備讀寫,對應(yīng)的文件操作函數(shù)是CreatFile()。
②XXX_Close()函數(shù),功能是關(guān)閉設(shè)備,對應(yīng)文件操作函數(shù)是CloseHandle()。
③XXX_Init()函數(shù),功能是初始化設(shè)備,在系統(tǒng)啟動時由設(shè)備管理器淵用。
④XXX_Read()函數(shù),功能是讀取設(shè)備中的數(shù)據(jù),對應(yīng)文件操作函數(shù)為ReadFile()。
⑤XXX_Write()函數(shù),功能是寫入數(shù)據(jù)到設(shè)備中,對應(yīng)文件操作函數(shù)為WriteFile()。
⑥XXX_IOControl()函數(shù),功能是對設(shè)備發(fā)送控制命令,對應(yīng)文件操作DeviceIOControl()。
其中XXX是驅(qū)動程序在注冊表中注冊的設(shè)備名稱。
2 硬件設(shè)計
ARM芯片選擇三星公司生產(chǎn)的S3C2440,該芯片工作頻率為400 MHz,最高可達533 MHz,可穩(wěn)定運行WinCE 6.0操作系統(tǒng)。nRF24L01是通過SPI總線協(xié)議和單片機進行通信的,在WinCE 6.0中SPI通信的實現(xiàn)可以通過ARM的SPI接口,也可以通過GPIO端口軟件模擬的方式實現(xiàn)。在本次設(shè)計中選用了GPIO端口軟件模擬的方式。硬件連接如圖2所示。
3 nRF24L01驅(qū)動的實現(xiàn)
3.1 主要實現(xiàn)的函數(shù)
(1)NRF_Init()設(shè)備初始化函數(shù)
在該函數(shù)中主要進行GPF端口的初始化,接收完成同步事件和接收線程的創(chuàng)建主要的核心代碼如下:
[!--empirenews.page--]
為了方便映射虛擬地址,在本驅(qū)動中將所用到的特殊功能寄存器的虛擬地址由自定義類CS3c2440SFR來進行組織。其中“sfr”是類CS3c2440SFR的實例,D()是一個自定義宏,其源碼為“#define D(v_add)(*(volatile unsigned long*)v_add)”。
(2)NRF_Open()函數(shù)
NRF Open()函數(shù)主要實現(xiàn)了對初始化是否成功的判斷及對nRF24L01進行初始配制。核心代碼如下:
SPI_Write_Bur(WRITE REG+TX_ADDR,TX_ADDRESS,TX_ADR_WIDTH); //寫本地默認地址
SPI_Write_Buf(WRITE_REG+RX_ADDR_P0,RX_ADDRESS,RX_ADR_WIDTH); //寫接收端默認地址
SPI_RW_Reg(WRITE_REG+EN_AA,0x01);
SPI_RW_Reg(WRITE_REG+EN_RXADDR,0x01);
SPI_RW_Reg(WRITE_REG+RF_CH,0);//設(shè)置信道工作為2.4 GHz,收發(fā)必須一致
SPI_RW_Reg(WRITE_REG+RX_PW_P0,RX_PLOADWIDTH); //設(shè)置接收數(shù)據(jù)長度
SPI_RW_Reg(WRITE_REG+RF_SETUP,0x07);//設(shè)置發(fā)射速率為1 MHz,發(fā)射功率為最大值0 dB
(3)NRF_Read()函數(shù)
NRE_Read()函數(shù)實現(xiàn)了對接收緩沖區(qū)的讀取并傳送至應(yīng)用程序。由于nRF24L01一次最多能接收32個字節(jié)的信息,所以該函數(shù)一次最多能傳送32字節(jié)的信息。主要核心代碼如下:
(4)NRF_Write()函數(shù)
NRE_Write()函數(shù)主要實現(xiàn)發(fā)送數(shù)據(jù)寫入發(fā)送緩沖區(qū)并發(fā)送的功能。主要核心代碼如下:
[!--empirenews.page--]
(5)NRF_IOControl()函數(shù)
NRF_IOControl()函數(shù)主要實現(xiàn)了對本機地址的配置、接收方地址的配置、發(fā)送模式和接收模式的配置等。
3.2 驅(qū)動設(shè)計重點
(1)如何實現(xiàn)特殊功能寄存器的虛擬地址映射
由于WinCE 6.0對虛擬地址的映射住安全性方面提高了要求,所以WinCE 6.0中地址映射不能再使用以前版本中常用的VirtualAlloc()和VirtualFree()函數(shù)來分配和釋放虛擬地址,它們的使用會產(chǎn)生編譯錯誤。解決方式是使用CEDDK庫中的MmMapIoSpace()函數(shù)和MmUnmapIo Space()函數(shù)來實現(xiàn)虛擬地址的分配和釋放。
(2)如何實現(xiàn)數(shù)據(jù)的接收
在數(shù)據(jù)接收中采用多線程編程,在設(shè)備初始化時調(diào)用CreateThread()函數(shù)創(chuàng)建一新線程對應(yīng)線程函數(shù)DWORD AcceptThread()。
(3)AcceptThread()函數(shù)的實現(xiàn)
在該函數(shù)中主要進行中斷配置、中斷同步事件的創(chuàng)建、中斷的撤除等。使用的API函數(shù)主要有:KernclIoControl()用來獲取邏輯巾斷號;InterruptInitialize()邏輯中斷和同步事件關(guān)聯(lián);WaitForSingleObject()阻塞線程并等待中斷同步事件發(fā)生,中斷同步事件發(fā)生即接收完一次數(shù)據(jù),則使接收完成同步事件有效,可通知應(yīng)用程序數(shù)據(jù)已接收完成;(InterruptDone)函數(shù)中斷結(jié)束,并撤除相應(yīng)的中斷標(biāo)志位準備響應(yīng)下次中斷。
4 驅(qū)動測試程序
4.1 數(shù)據(jù)發(fā)送
數(shù)據(jù)發(fā)送步驟如下:
①調(diào)用CreateFile()函數(shù)打開設(shè)備,核心代碼如下:
NRFdriver=CreateFilc(L“NRF1:”,GENERIC_READ|GENERIC_WRITE,0,NULL,OPEN EXISTING,0,NULL);
②通過DeviceIoControl()函數(shù)設(shè)置為發(fā)送模式并使用默認地址(默認地址在驅(qū)動程序中配置完成),核心代碼如下:
DevlccIoControl(NRFdriver.IS SEND STATUS,NULL,0,NULL,0.NULL,NULL);
③調(diào)用WriteFile函數(shù)完成發(fā)送,核心代碼如下:
WriteFile(NRFdriver,(LPCVOID)p,strL,NULL,NULL);
4.2 數(shù)據(jù)接收
數(shù)據(jù)接收步驟如下:
①打開設(shè)備,方法同4.1節(jié)步驟①。
②通過DeviceIocontrol()函數(shù)設(shè)置為接收模式并使用默認地址,核心代碼如下:
DeviceIoControl(NRFdrivcr,IS_ACCEPT_STATUS,NULL,0,NULL,0,NULL,NULL);
③創(chuàng)建接收線程及與驅(qū)動中同名的接收完成同步事件,在接收線程中調(diào)用WaitForSingleObject()函數(shù)等待有效。如有效,則調(diào)用ReadFi le()函數(shù)讀取數(shù)據(jù)并顯示。核心代碼如下:
[!--empirenews.page--]
以上驅(qū)動及測試程序均是在VC2008環(huán)境中完成。在WinCE6.0系統(tǒng)中運行測試程序還應(yīng)該在注冊表中導(dǎo)入如下注冊信息:
[HKEY_LOCAL_MACHINE\Drivers\BuiltIn\NRF24L01]
“Prefix”=“NRF”
“Dll”=“NRFDriver.DLL”
以上內(nèi)容中,驅(qū)動程序注冊的設(shè)備名稱為“NRF”,驅(qū)動程序名為“NRFDriver.dll”。
5 測試效果
5.1 接收測試
采用MSP430F149開發(fā)板和nRF24L01模塊搭建發(fā)射端,然后利用該發(fā)射端每秒發(fā)送32個字節(jié)的數(shù)據(jù),數(shù)據(jù)內(nèi)容為1~32。ARM接收端接收到數(shù)據(jù)并將它顯示到LCD上。
5.2 發(fā)送測試
用MSP430F149開發(fā)板和nRF24L01模塊搭建接收端,然后利用MSP430F149的串行口將接收到的數(shù)據(jù)發(fā)送到計算機,通過串口調(diào)試助手就可以觀察到ARM端發(fā)送來的數(shù)據(jù)。
結(jié)語
驅(qū)動程序的編寫是嵌入式系統(tǒng)開發(fā)的重要核心步驟。文中詳細闡述了nRF24L01驅(qū)動程序和測試程序的開發(fā),通過測試該驅(qū)動程序能夠較好地完成無線通信的功能,滿足實際需求。