摘 要: 首先介紹了MAX1202的基本性能和特點,詳細闡述了MAX1202在嵌入式Linux中驅動程序的實現方法,最后介紹了編寫應用程序測試驅動程序的基本方法,為嵌入式Linux系統下的驅動設計提供一個模板。
關鍵詞: MAX1202;驅動程序;A/D轉換;嵌入式Linux
0 引言
科技的進步使智能化設備越來越多地應用到工業生產、農業種植、醫療衛生、航天設備甚至是居民的日常生活中,智能化設備要處理一些環境中的物理量就需要使用相關傳感器將其轉化成電量,如電壓、電流等。但是這些量要送給處理器處理,則必須要通過模/數轉換器進行轉換。
本文中采用MAX1202模/數轉換器和友善之臂的Micro2440開發板,開發板使用三星公司的s3c2440的ARM9微處理器。在Linux系統中開發MAX1202驅動程序并編寫應用程序進行驅動程序的測試。
1 芯片介紹
MAX1202是一款8通道12位串行A/D轉換器。串行接口工作頻率最高可以達到2 MHz[1]。工作采用單端+5 V供電或雙端±5 V供電。內部有一個8通道的多路轉換器、高帶寬的跟蹤/保持電路以及高轉換速度和低功耗的串行接口,芯片提供了符合SPI通信標準的SPI接口,以便于編程實現數據的轉換。
其典型應用電路如圖1所示。
2 驅動程序設計
Linux驅動程序使用module_init宏中所定義的初始化函數注冊該驅動及初始化硬件設備;使用module_exit宏中定義的注銷函數注銷設備,釋放相關資源。
結構struct file_operations列出了設備驅動程序可供應用程序調用的所有函數。其中的成員都是函數指針,指向該函數的入口執行位置,當應用程序調用open、read、write等函數時,驅動程序會通過該結構找到對應應該執行的M1202函數,進而根據傳入的參數執行,以響應應用程序的調用。驅動程序的編寫主要就在于file_operations結構體中各驅動函數的實現,并不是每一個函數都要實現[2],對于不需要實現的函數可以賦值為空,也可以在函數實現中直接返回0,或直接調用系統默認的實現函數。MAX1202的file_operations結構體定義如下:
static const struct file_operations M1202_fops=
{
.owner=THIS_MODULE,
.open=M1202_open,
.read=M1202_read,
.write=M1202_write,
.ioctl=NULL,
};
2.1 模塊初始化函數M1202_init
在驅動加載時,系統進程會通過module_init(M1202_init)宏找到所定義的驅動初始化函數M1202_init(),進行驅動程序的加載。主設備號可以自已定義,向函數MKDEV(M1202_major,0)傳遞主次設備號產生dev_t類型的devno結構[3];也可以由alloc_chrdev_region(&devno,0,1,M1202_name)函數自動分配主設備號;本例中自己定義主設備號,然后由register_chrdev_region(devno,1,M1202_name)函數在Linux中為驅動程序獲取設備編號;接著就是向Linux內核注冊字符設備,指出該驅動可提供給應用程序的接口結構。實現代碼如下:
cdev_init(&M1202Cdev,&M1202_fops);
M1202Cdev.owner=THIS_MODULE;
M1202Cdev.ops=& M1202_fops;
cdev_add(&M1202Cdev,devno,1);
代碼中省去了錯誤調試信息。
2.2 設備打開函數M1202_open
s3c2440芯片配備了2組SPI接口[4],Micro2440開發板引出了一組SPI1,對應是PA7、PA8、PA9、PA10引腳,這是一組可以復用的端口,既可以用中斷、普通端口,也可以用于SPI通信。MAX1202在外部時鐘模式下的工作時序如圖2所示。
查看MAX1202在外部時鐘模式下的工作時序可以發現,在控制字送入的8個時鐘后,芯片即開始了邊輸出邊轉換的過程,相比較內部時鐘模式的轉換完成后再讀取輸出結果而言,有著更快的轉換效率,故采用外部時鐘模式。在控制字輸出后,轉換結果緊跟著輸出,為了減少頻繁對寄存器的操作,降低編程的難度,采用普通輸出端口輪流輸出高低電平的方法模擬主SPI設備的時鐘輸出。使用s3c2410_gpio_cfgpin(S3C2410_GPG11,S3C2410_GPG11_OUTP);將GPG11端配置成輸出端口用于輸出MAX1202工作時所需的片選信號;類似地用s3c2410_gpio_cfgpin()函數配置GPG6、GPG7端為輸出端,分別用于產生MAX1202的控制字輸入與時鐘輸入;使用s3c2410_gpio_cfgpin(S3C2410_GPG5,S3C2410_GPG5_INP);將GPG5端口配置成輸入端口,用于讀取MAX1202的Dout端的輸出結果。在MAX1202正式工作前需將片選端CS端置為1,相應地有s3c2410_gpio_setpin(S3C2410_GPG3,1);函數;另外三個端口s3c2410_gpio_setpin函數配置為0。這些就是open函數所需要完成的端口配置。
2.3 寫函數M1202_write
設備打開后,每次轉換之前,需要向MAX1202寫控制字,以設置MAX1202選通的工作通道、時鐘模式、信號輸入模式等。MAX1202控制字格式如表1所示。
Bit7:起始位一般選擇1,用于標識控制字的開始;
Bit6~Bit4:是8路通道的選擇,在單端模式下,依據8421編碼,000對應第0通道CH0,001對應第1通道CH1,010對應第2通道CH2,以此類推;
Bit3:為信號的單雙極性選擇位,這里選擇1,單極性;
Bit2:為信號輸入方式選擇位,這里選擇1,單端輸入;
Bit1~Bit0:時鐘和斷電模式選擇位,這里選擇11,外部時鐘。
驅動屬于內核的一部分,而應用程序屬于用戶空間,內核空間和用戶空間的數據不能共享,數據的傳輸需要使用特定的函數copy_from_user()和copy_to_user()。因此,由應用程序傳入的控制字信息需要使用copy_from_user()傳入內核空間。使用循環移位和與操作,輪流讀取8位控制字的每一位,由s3c2410_gpio_setpin(S3C2410_GPG6,X)函數將各位值輸出到MAX1202得DIN端,在時鐘的下降沿由MAX1202讀取,X代表了要傳輸的一位值。
2.4 讀函數M1202_read
在外部時鐘模式下,控制字輸入完后即可在緊接著的16個時鐘周期內讀取轉換結果,這16位轉換結果的最低4位為無效位,均為0,雖然無效,但必須在時鐘的作用下讀取出來,不然會影響到下一次的轉換結果,故函數讀取16位后左移4位用于消除低4位的無效位。while函數的執行條件控制讀取次數為16次,s3c2410_gpio_getpin(S3C2410_GPG5)用于從MAX1202的DOUT端讀入當前時鐘下的輸出值。讀取結束后,要使用copy_to_user()函數將得到的結果傳遞到用戶空間,以供顯示或處理。
2.5 模塊卸載函數M1202_exit
模塊卸載函數首先要刪除字符設備,然后釋放占用的驅動設備號,以供其他的設備使用。
cdev_del(&M1202Cdev);
unregister_chrdev_region(MKDEV(M1202_major,0),1);
3 MAX1202在嵌入式系統中的應用
3.1 設備驅動的加載
Linux下的驅動有靜態加載和動態加載兩種方式[5]。靜態加載將驅動程序編譯到內核里,系統啟動后直接可以由應用程序調用,但每次修改驅動程序都必須重新編譯內核,較麻煩。動態加載是在系統啟動后使用insmod命令,把編譯好的M1202.ko文件加載到系統中,不需要時可以使用rmmod命令卸載。但是在重新開機之后,該驅動就沒了,需要重新加載。故靜態加載適合于驅動開發完成后的產品量化;而動態加載適合于驅動開發過程中頻繁的修改。
在開發板中驅動的動態加載方法:
#cd/lib/modules/2.6.32.3-FriendlyARM
#insmod M1202.ko
#mknod/dev/M1202 c 55 0
在Linux中,設備被當做文件一樣處理,任何可用設備在/dev/目錄下都會有一個對應的設備文件,使用mknod命令創建設備節點,在應用程序中即可像操作文件一樣操作該設備。
3.2 數據采集
驅動加載并且創建了設備節點后就可以編寫應用程序進行電壓數據的采集。
使用open函數以只讀的方式打開已經創建了的M1202節點即打開了MAX1202硬件。傳入控制字0x8F給write函數,開啟通道0的轉換,使用單極性單端輸入外部時鐘模式,read函數將轉換結果保存到指針num所指向的地址中,然后就可以打印或者處理轉換結果,最后像關閉文件一樣使用close()函數關閉設備。使用交叉編譯指令#arm-linux-gcc╞o M1202test M1202test.c編譯后生成應用程序文件M1202test,通過串口導入Micro2440中或將應用程序文件放到網絡根文件系統中,#chmod+x M1202test增加文件的可執行權限,#./M1202test執行可執行文件,即可對通道0引腳的數據進行采集轉換。
部分應用程序如下:
int fd=open(“/dev/M1202”,0_RDWR);
int ctlword=0x8F;
write(fd,ctlword,1);
read(fd,&num,2);
printf(“num=%d\n”,num);
close(fd);
在實際使用中可以使用for(int i=0,i<8,i++)和ctlword=ctlword|(i<<4);改變控制字的Bit6-Bit4位,循環采集引腳0-7端輸入的電壓,或者根據需要使用其中部分引腳。經測試,A/D轉換結果較好,在轉換誤差范圍內。
4 結論
隨著智能化設備的發展,嵌入式系統將涉及生活中的方方面面。本文詳細介紹了MAX1202在嵌入式Linux系統中的驅動程序的開發方法,對于相關驅動程序的開發有一定的參考價值,且該MAX1202的A/D實現方法也可以應用到一些工程實際中去。
參考文獻
[1] 蔣雙梅,高敦堂,都思丹.8通道12位串行A/D轉換器MAX1202及其應用[J].微電子學,2000,30(6):437-440.
[2] 韋東山.嵌入式Linux應用開發完全手冊(第1版)[M].北京:人民郵電出版社,2008.
[3] 付興武,張軍,王洋.基于SPI總線協議的字符設備驅動程序[J].計算機系統應用,2013,22(2):146-150.
[4] 楊小容,陳建政.串口AD嵌入式Linux驅動實現[J].中國測試,2010,36(2):84-87.
[5] 黃智偉,鄧月明,王彥.ARM9嵌入式系統設計基礎教程(第1版)[M].北京:北京航空航天大學出版社,2008.