移植嵌入式Linux到ARM處理器S3C2410:設(shè)備驅(qū)動
設(shè)備驅(qū)動程序是操作系統(tǒng)內(nèi)核和機器硬件之間的接口,它為應(yīng)用程序屏蔽硬件的細(xì)節(jié),一般來說,Linux的設(shè)備驅(qū)動程序需要完成如下功能:
·設(shè)備初始化、釋放;
·提供各類設(shè)備服務(wù);
·負(fù)責(zé)內(nèi)核和設(shè)備之間的數(shù)據(jù)交換;
·檢測和處理設(shè)備工作過程中出現(xiàn)的錯誤。
Linux下的設(shè)備驅(qū)動程序被組織為一組完成不同任務(wù)的函數(shù)的集合,通過這些函數(shù)使得Windows的設(shè)備操作猶如文件一般。在應(yīng)用程序看來,硬件設(shè)備只是一個設(shè)備文件,應(yīng)用程序可以象操作普通文件一樣對硬件設(shè)備進(jìn)行操作,如open ()、close ()、read ()、write () 等。
Linux主要將設(shè)備分為二類:字符設(shè)備和塊設(shè)備。字符設(shè)備是指設(shè)備發(fā)送和接收數(shù)據(jù)以字符的形式進(jìn)行;而塊設(shè)備則以整個數(shù)據(jù)緩沖區(qū)的形式進(jìn)行。在對字符設(shè)備發(fā)出讀/寫請求時,實際的硬件I/O一般就緊接著發(fā)生了;而塊設(shè)備則不然,它利用一塊系統(tǒng)內(nèi)存作緩沖區(qū),當(dāng)用戶進(jìn)程對設(shè)備請求能滿足用戶的要求,就返回請求的數(shù)據(jù),如果不能,就調(diào)用請求函數(shù)來進(jìn)行實際的I/O操作。塊設(shè)備主要針對磁盤等慢速設(shè)備。
1.內(nèi)存分配
由于Linux驅(qū)動程序在內(nèi)核中運行,因此在設(shè)備驅(qū)動程序需要申請/釋放內(nèi)存時,不能使用用戶級的malloc/free函數(shù),而需由內(nèi)核級的函數(shù)kmalloc/kfree () 來實現(xiàn),kmalloc()函數(shù)的原型為:
void kmalloc (size_t size ,int priority);
參數(shù)size為申請分配內(nèi)存的字節(jié)數(shù),kmalloc最多只能開辟128k的內(nèi)存;參數(shù)priority說明若kmalloc()不能馬上分配內(nèi)存時用戶進(jìn)程要采用的動作:GFP_KERNEL 表示等待,即等kmalloc()函數(shù)將一些內(nèi)存安排到交換區(qū)來滿足你的內(nèi)存需要,GFP_ATOMIC 表示不等待,如不能立即分配到內(nèi)存則返回0 值;函數(shù)的返回值指向已分配內(nèi)存的起始地址,出錯時,返回0。
kmalloc ()分配的內(nèi)存需用kfree()函數(shù)來釋放,kfree ()被定義為:
# define kfree (n) kfree_s( (n) ,0)
其中kfree_s () 函數(shù)原型為:
void kfree_s (void * ptr ,int size);
參數(shù)ptr為kmalloc()返回的已分配內(nèi)存的指針,size是要釋放內(nèi)存的字節(jié)數(shù),若為0 時,由內(nèi)核自動確定內(nèi)存的大小。
2.中斷
許多設(shè)備涉及到中斷操作,因此,在這樣的設(shè)備的驅(qū)動程序中需要對硬件產(chǎn)生的中斷請求提供中斷服務(wù)程序。與注冊基本入口點一樣,驅(qū)動程序也要請求內(nèi)核將特定的中斷請求和中斷服務(wù)程序聯(lián)系在一起。在Linux中,用request_irq()函數(shù)來實現(xiàn)請求:
int request_irq (unsigned int irq ,void( * handler) int ,unsigned long type ,char * name);
參數(shù)irq為要中斷請求號,參數(shù)handler為指向中斷服務(wù)程序的指針,參數(shù)type 用來確定是正常中斷還是快速中斷(正常中斷指中斷服務(wù)子程序返回后,內(nèi)核可以執(zhí)行調(diào)度程序來確定將運行哪一個進(jìn)程;而快速中斷是指中斷服務(wù)子程序返回后,立即執(zhí)行被中斷程序,正常中斷type 取值為0 ,快速中斷type 取值為SA_INTERRUPT),參數(shù)name是設(shè)備驅(qū)動程序的名稱。
3.字符設(shè)備驅(qū)動
我們必須為字符設(shè)備提供一個初始化函數(shù),該函數(shù)用來完成對所控設(shè)備的初始化工作,并調(diào)用register_chrdev() 函數(shù)注冊字符設(shè)備。假設(shè)有一字符設(shè)備"exampledev",則其init 函數(shù)為:
void exampledev_init(void)
{
if (register_chrdev(MAJOR_NUM, " exampledev ", &exampledev_fops))
TRACE_TXT("Device exampledev driver registered error");
else
TRACE_TXT("Device exampledev driver registered successfully");
…//設(shè)備初始化
}
其中,register_chrdev函數(shù)中的參數(shù)MAJOR_NUM為主設(shè)備號,"exampledev"為設(shè)備名,exampledev_fops 為包含基本函數(shù)入口點的結(jié)構(gòu)體,類型為file_operations。當(dāng)執(zhí)行exampledev_init時,它將調(diào)用內(nèi)核函數(shù) register_chrdev,把驅(qū)動程序的基本入口點指針存放在內(nèi)核的字符設(shè)備地址表中,在用戶進(jìn)程對該設(shè)備執(zhí)行系統(tǒng)調(diào)用時提供入口地址。
較早版本內(nèi)核的file_operations結(jié)構(gòu)體定義為(代碼及圖示):
struct file_operations
{
int (*lseek)();
int (*read)();
int (*write)();
int (*readdir)();
int (*select)();
int (*ioctl)();
int (*mmap)();
int (*open)();
void(*release)();
int (*fsync)();
int (*fasync)();
int (*check_media_change)();
void(*revalidate)();
};
隨著內(nèi)核功能的加強,file_operations結(jié)構(gòu)體也變得更加龐大。但是大多數(shù)的驅(qū)動程序只是利用了其中的一部分,對于驅(qū)動程序中無需提供的功能,只需要把相應(yīng)位置的值設(shè)為NULL。對于字符設(shè)備來說,要提供的主要入口有:open ()、release ()、read ()、write ()、ioctl ()等。
open()函數(shù) 對設(shè)備特殊文件進(jìn)行open()系統(tǒng)調(diào)用時,將調(diào)用驅(qū)動程序的open () 函數(shù):
int (*open)(struct inode * inode,struct file *filp);
其中參數(shù)inode為設(shè)備特殊文件的inode (索引結(jié)點) 結(jié)構(gòu)的指針,參數(shù)filp是指向這一設(shè)備的文件結(jié)構(gòu)的指針。open()的主要任務(wù)是確定硬件處在就緒狀態(tài)、驗證次設(shè)備號的合法性(次設(shè)備號可以用 MINOR(inode-> i_rdev) 取得)、控制使用設(shè)備的進(jìn)程數(shù)、根據(jù)執(zhí)行情況返回狀態(tài)碼(0表示成功,負(fù)數(shù)表示存在錯誤) 等;
release()函數(shù) 當(dāng)最后一個打開設(shè)備的用戶進(jìn)程執(zhí)行close ()系統(tǒng)調(diào)用時,內(nèi)核將調(diào)用驅(qū)動程序的release () 函數(shù):
void (*release) (struct inode * inode,struct file *filp) ;
release 函數(shù)的主要任務(wù)是清理未結(jié)束的輸入/輸出操作、釋放資源、用戶自定義排他標(biāo)志的復(fù)位等。
read()函數(shù) 當(dāng)對設(shè)備特殊文件進(jìn)行read() 系統(tǒng)調(diào)用時,將調(diào)用驅(qū)動程序read() 函數(shù):
ssize_t (*read) (struct file * filp, char * buf, size_t count, loff_t * offp);
參數(shù)buf是指向用戶空間緩沖區(qū)的指針,由用戶進(jìn)程給出,count 為用戶進(jìn)程要求讀取的字節(jié)數(shù),也由用戶給出。
read() 函數(shù)的功能就是從硬設(shè)備或內(nèi)核內(nèi)存中讀取或復(fù)制count個字節(jié)到buf 指定的緩沖區(qū)中。在復(fù)制數(shù)據(jù)時要注意,驅(qū)動程序運行在內(nèi)核中,而buf指定的緩沖區(qū)在用戶內(nèi)存區(qū)中,是不能直接在內(nèi)核中訪問使用的,因此,必須使用特殊的復(fù)制函數(shù)來完成復(fù)制工作,這些函數(shù)在include/asm/uaccess.h中被聲明:
unsigned long copy_to_user (void * to, void * from, unsigned long len);
此外,put_user()函數(shù)用于內(nèi)核空間和用戶空間的單值交互(如char、int、long)。
write( ) 函數(shù) 當(dāng)設(shè)備特殊文件進(jìn)行write () 系統(tǒng)調(diào)用時,將調(diào)用驅(qū)動程序的write () 函數(shù):
ssize_t (*write) (struct file *, const char *, size_t, loff_t *);
write ()的功能是將參數(shù)buf 指定的緩沖區(qū)中的count 個字節(jié)內(nèi)容復(fù)制到硬件或內(nèi)核內(nèi)存中,和read() 一樣,復(fù)制工作也需要由特殊函數(shù)來完成:
unsigned long copy_from_user(void *to, const void *from, unsigned long n);
此外,get_user()函數(shù)用于內(nèi)核空間和用戶空間的單值交互(如char、int、long)。
ioctl() 函數(shù) 該函數(shù)是特殊的控制函數(shù),可以通過它向設(shè)備傳遞控制信息或從設(shè)備取得狀態(tài)信息,函數(shù)原型為:
int (*ioctl) (struct inode * inode,struct file * filp,unsigned int cmd,unsigned long arg);
參數(shù)cmd為設(shè)備驅(qū)動程序要執(zhí)行的命令的代碼,由用戶自定義,參數(shù)arg 為相應(yīng)的命令提供參數(shù),類型可以是整型、指針等。
同樣,在驅(qū)動程序中,這些函數(shù)的定義也必須符合命名規(guī)則,按照本文約定,設(shè)備"exampledev"的驅(qū)動程序的這些函數(shù)應(yīng)分別命名為 exampledev_open、exampledev_ release、exampledev_read、exampledev_write、exampledev_ioctl,因此設(shè)備 "exampledev"的基本入口點結(jié)構(gòu)變量exampledev_fops 賦值如下(對較早版本的內(nèi)核):
struct file_operations exampledev_fops {
NULL ,
exampledev_read ,
exampledev_write ,
NULL ,
NULL ,
exampledev_ioctl ,
NULL ,
exampledev_open ,
exampledev_release ,
NULL ,
NULL ,
NULL ,
NULL
} ;
就目前而言,由于file_operations結(jié)構(gòu)體已經(jīng)很龐大,我們更適合用GNU擴展的C語法來初始化exampledev_fops:
struct file_operations exampledev_fops = {
read: exampledev _read,
write: exampledev _write,
ioctl: exampledev_ioctl ,
open: exampledev_open ,
release : exampledev_release ,
};
看看第一章電路板硬件原理圖,板上包含四個用戶可編程的發(fā)光二極管(LED),這些LED連接在ARM處理器的可編程I/O口(GPIO)上,現(xiàn)在來編寫這些LED的驅(qū)動:
#include
#include
#include
#include
#include
#include
#include
#define DEVICE_NAME "leds" /*定義led 設(shè)備的名字*/
#define LED_MAJOR 231 /*定義led 設(shè)備的主設(shè)備號*/
static unsigned long led_table[] =
{
/*I/O 方式led 設(shè)備對應(yīng)的硬件資源*/
GPIO_B10, GPIO_B8, GPIO_B5, GPIO_B6,
};
/*使用ioctl 控制led*/
static int leds_ioctl(struct inode *inode, struct file *file, unsigned int cmd,
unsigned long arg)
{
switch (cmd)
{
case 0:
case 1:
if (arg > 4)
{
return -EINVAL;
}
write_gpio_bit(led_table[arg], !cmd);
default:
return -EINVAL;
}
}
static struct file_operations leds_fops =
{
owner: THIS_MODULE, ioctl: leds_ioctl,
};
static devfs_handle_t devfs_handle;
static int __init leds_init(void)
{
int ret;
int i;
/*在內(nèi)核中注冊設(shè)備*/
ret = register_chrdev(LED_MAJOR, DEVICE_NAME, &leds_fops);
if (ret < 0)
{
printk(DEVICE_NAME " can't register major numbern");
return ret;
}
devfs_handle = devfs_register(NULL, DEVICE_NAME, DEVFS_FL_DEFAULT, LED_MAJOR,
0, S_IFCHR | S_IRUSR | S_IWUSR, &leds_fops, NULL);
/*使用宏進(jìn)行端口初始化,set_gpio_ctrl 和write_gpio_bit 均為宏定義*/
for (i = 0; i < 8; i++)
{
set_gpio_ctrl(led_table[i] | GPIO_PULLUP_EN | GPIO_MODE_OUT);
write_gpio_bit(led_table[i], 1);
}
printk(DEVICE_NAME " initializedn");
return 0;
}