采用Linux/Qtopia的車載溫度網(wǎng)絡(luò)采集
3 驅(qū)動實現(xiàn)
本節(jié)將實現(xiàn)一線制溫度傳感器網(wǎng)絡(luò)的驅(qū)動模塊。驅(qū)動從總體上看分為兩部分:驅(qū)動與內(nèi)核接口層、硬件設(shè)備接口層。
3.1 驅(qū)動與內(nèi)核接口層
驅(qū)動與內(nèi)核接口層主要完成驅(qū)動模塊在Linux內(nèi)核的注冊加載、卸載清除工作。這部分工作分別由初始化和退出函數(shù)完成。
?、?初始化函數(shù)完成驅(qū)動模塊加載:
static int __init DS18B20_init(void){
……
register_chrdev(DS18B20_MAJOR,DEVICE_NAME, DS18B20_fops);//完成設(shè)備注冊
#ifdefCONFIG_DEVFS_FS//創(chuàng)建設(shè)備文件系統(tǒng)
devfs_mk_cdev(MKDEV(DS18B20_MAJOR,0),S_IFCHR | S_IRUSR | S_IWUSR | S_IRGRP,DEVICE_NAME);
#endif
……
}
?、?退出函數(shù)完成驅(qū)動模塊卸載:
static void __exit DS18B20_exit(void) {
#ifdef CONFIG_DEVFS_FS
devfs_remove(DEVICE_NAME);//移除設(shè)備文件
#endif
unregister_chrdev(DS18B20_MAJOR,DEVICE_NAME); //完成設(shè)備注銷
……
}
3.2 硬件設(shè)備接口層
硬件設(shè)備接口層用來描述驅(qū)動程序與設(shè)備的交互。這些工作通過虛擬文件系統(tǒng)與設(shè)備驅(qū)動程序的接口實現(xiàn)。這個接口由file_operation結(jié)構(gòu)定義,其結(jié)構(gòu)如下:
static struct file_operations DS18B20_fops ={
.owner=THIS_MODULE, //指向擁有該結(jié)構(gòu)的模塊,內(nèi)核使用該結(jié)構(gòu)維護模塊使用計數(shù)
.open=DS18B20_open, //打開設(shè)備函數(shù)
.read=DS18B20_read, //讀接口函數(shù)
.write=DS18B20_write,//寫接口函數(shù)
.fasync=DS18B20_fasync, //異步通知函數(shù)
.poll=DS18B20_poll,//poll函數(shù)
.release=DS18B20_release, //釋放設(shè)備函數(shù)
};
3.2.1 打開設(shè)備函數(shù)
打開設(shè)備函數(shù)主要完成設(shè)備的初始化。
DS18B20_open(struct inode *inode,struct file *filp) {
Initial_Timer( );//初始化定時器,使內(nèi)核模塊按一定周期讀溫度
Initial_Device_DS18B20();//初始化硬件
readtemperature();//開始讀取……
}
void readtemperature(void) {
……Temperature=DS18B20read();//讀取2個8位數(shù)據(jù),此函數(shù)完成的硬件操作時序,由當前讀通道號變量指定當前通道
DS_SLOT_NO();//將本次讀通道號放入緩沖區(qū)
DS18B20Event();//數(shù)據(jù)放入緩沖區(qū),喚醒等待隊列并啟動異步通知
if(ReleaseFlag)
CycleTimer_Delay_Soft(hdelay);//如果沒有讀停止信號,通過內(nèi)核定時器延時,進行下一次讀,在中斷服務(wù)程序中再次啟動讀
……
}
在使用內(nèi)核定時器之前需定義一個定時器結(jié)構(gòu)體 static struct timer_list CycleTimer。下面是定時器的具體操作:
static void Initial_Timer(void) {
init_timer(CycleTimer); );//初始化定時器結(jié)構(gòu)
CycleTimer.function=DS18B20_timer; //掛接定時中斷服務(wù)程序
}
3.2.2 讀接口函數(shù)
用戶程序執(zhí)行讀操作的時候可能沒有可以讀取的數(shù)據(jù),此時需要讓read操作等待直到有數(shù)據(jù)可以讀取。在此采用等待隊列使進程在無數(shù)據(jù)讀取時進入等待,數(shù)據(jù)到達時喚醒。等待隊列設(shè)置成一個循環(huán)緩沖區(qū),每放入一個新數(shù)據(jù)作為緩沖區(qū)的頭,存放時間最久還未被取走的數(shù)據(jù)為緩沖區(qū)的尾。
DS18B20_read( ) {
DECLARE_WAITQUEUE(wait,current);//聲明等待隊列……
Next_try:
if(DS18B20dev.head != DS18B20dev.tail) {//等待隊列不為空,即有數(shù)據(jù)
DS18B20_ret=Read_Buffer_DS18B20(); //取走緩沖區(qū)的尾
copy_to_user( ); //讀取的數(shù)據(jù)送到用戶空間
}
else { ……//等待隊列為空,即沒有數(shù)據(jù)
add_wait_queue(queue,wait);
current>state=TASK_INTERRUPTIBLE;//添加等待隊列,聲明狀態(tài)為任務(wù)可中斷
while((DS18B20dev.head==DS18B20dev.tail)!signal_pending(current) {//進入等待
schedule();
current>state=TASK_INTERRUPTIBLE;
}//如果緩沖區(qū)為空,Linux內(nèi)核調(diào)度,等待通知
current>state = TASK_RUNNING;//得到有數(shù)據(jù)的通知,聲明任務(wù)狀態(tài)為運行
remove_wait_queue(queue,wait);//刪除等待隊列
goto Next_try;//返回到讀取數(shù)據(jù)
}
}
評論