基于華邦W90P710處理器的Linux內核應用及串口驅動(dòng)的實(shí)現
嵌入式Linux是一種很受歡迎的操作系統,具有開(kāi)放源碼、不存在黑箱技術(shù)、內核小、功能強大、運行穩定、效率高、易于定制裁減等特點(diǎn)[1],廣泛應用于工控產(chǎn)品。很多工控產(chǎn)品需要和外部設備進(jìn)行信息交換,而串口通信是最簡(jiǎn)單快捷的實(shí)現方法。在不同的工控產(chǎn)品中,由于對所選用的串口元件或者串口通信的數據格式、波特率等有不同的需求,需要對串口驅動(dòng)進(jìn)行開(kāi)發(fā)。華邦W90P710采用ARM的ARM7TDMI微處理器核心,采用?滋CLinux-2.4.20內核,支持4組通用異步接收發(fā)送口(UART),下面基于華邦W90P710的串口驅動(dòng)詳細分析串口驅動(dòng)的實(shí)現方法,實(shí)現嵌入式設備通過(guò)串口對外通信。
1 華邦W90P710 UART介紹
華邦W90P710支持4組UART,串口的控制主要通過(guò)以下寄存器實(shí)現[2]:
(1)行寄存器(UART_LCR):設置數據位長(cháng)度、奇偶校驗、停止位數。
(2)波特率除數寄存器(UART_DLL、UART_DLM):波特率發(fā)生器的公式為:BaudOut=crystal clock/16×[Divisor +2],Divisor為當前波特率。
(3)Modem控制寄存器(UART_MCR):控制RTS、CTS等信號。
(4)FIFO控制寄存器(UART_FCR):設置FIFO的長(cháng)度,復位FIFO等控制。
(5)接收超時(shí)寄存器(UART_TOR):收到首個(gè)字節后接收器啟動(dòng)本超時(shí),之后每收到一個(gè)字節后都會(huì )重置該值,在此超時(shí)時(shí)間內不再收到數據時(shí),接收器會(huì )產(chǎn)生一個(gè)接收中斷。
(6)中斷控制器(UART_IER):設置接收、發(fā)送、行中斷等。
在使用RXDn、TXDn前必須對GPIO進(jìn)行配置,使能RXDn、TXDn,串口才可正常運行。GPIO配置對應表如表1所示。
2 Linux系統驅動(dòng)介紹
設備驅動(dòng)程序是操作系統內核和機器硬件之間的接口。設備驅動(dòng)程序為應用程序屏蔽了硬件的細節,這樣在應用程序看來(lái),硬件設備只是一個(gè)設備文件,應用程序可以像操作普通文件一樣對硬件設備進(jìn)行操作。同時(shí),設備驅動(dòng)程序是內核的一部分[3]。圖1所示為設備驅動(dòng)程序接口流程圖。
Linux系統的設備分為字符設備、塊設備和網(wǎng)絡(luò )設備三種。字符設備是指存取時(shí)沒(méi)有緩存的設備,只能順序讀寫(xiě)。典型的字符設備包括鼠標、鍵盤(pán)、串行口等;塊設備一般都有緩存來(lái)支持,并且塊設備必須能夠支持隨機存取。塊設備主要包括硬盤(pán)設備、CD-ROM等;網(wǎng)絡(luò )設備在Linux系統中用做專(zhuān)門(mén)的處理,Linux的網(wǎng)絡(luò )系統主要是基于BSD Unix的socket機制[4]。
3 串口驅動(dòng)程序詳細介紹
一般來(lái)說(shuō),Linux的設備驅動(dòng)程序包括驅動(dòng)程序的注冊和注銷(xiāo)、設備的打開(kāi)和釋放、設備的讀寫(xiě)操作、設備的控制操作、設備的中斷和輪詢(xún)處理等功能。下面就這些功能對串口驅動(dòng)進(jìn)行詳細說(shuō)明。
(1)串口設備的數據結構包括串口參數接收發(fā)送緩沖區等。串口參數包括波特率、數據位、數據起始位、奇偶校驗、串口類(lèi)型、發(fā)送緩沖區、接收緩沖區等,每個(gè)串口對應一個(gè)如下的數據結構:
typedef struct{
int bps;
int databits;
int stopbits;
int parity;
int siotype; //串口參數
int openflag;
int recvTrigTimeout;
SIO_D_SEND_BUFFER *pSendBuf;//發(fā)送緩沖區
SIO_D_RECV_BUFFER *pRecvBuf;//接收緩沖區
struct fasync_struct *fasync_queue;
wait_queue_head_t read_wait;
}serial_dev;
static serial_dev serial_device;
(2)文件系統操作
入口函數對應文件操作函數read ()、write()、ioctl()、open()、close()。
struct file_operations serial_fops = {
owner: THIS_MODULE,
poll: serial_poll,
read: serial_read,
write: serial_write,
ioctl: serial_ioctl,
open: serial_open,
release: serial_release,
};
(3)驅動(dòng)程序注冊和注銷(xiāo)。
驅動(dòng)程序在應用前,需要在模塊初始化時(shí)將設備注冊到系統設備表中;不再使用時(shí),將設備從系統中卸除。注冊包括初始化定時(shí)器、初始化串口數據結構serial_device和字符設備注冊。注銷(xiāo)時(shí)直接調用設備注銷(xiāo)函數[5]。
int __init topbandserial1_init(void)
{
init_timer(timer);//初始化定時(shí)器結構
memset(serial_device, 0, sizeof(serial_device));
result=register_chrdev(SERIAL1_MAJOR, serial1,
serial_fops);
…
}
(4)串口設備打開(kāi)包括分配串口的接收發(fā)送緩沖區及中斷注冊[5]。
static int serial_open(struct inode *inode, struct file *filp)
{
dev->pRecvBuf = kmalloc(sizeof(SIO_D_RECV_BUFFER), GFP_KERNEL);
request_irq(INT_UART1,serial_interrupt,SA_SHIRQ,
TopbandSerial1,serial_device);
…
}
(5)串口設備釋放包括釋放內存空間、注銷(xiāo)中斷和刪除定時(shí)器[5]。
static int serial_release(struct inode *inode, struct file *flip)
{
serial_dev *dev = flip->private_data;//釋放內存空間
kfree(dev->fasync_queue);
CSR_WRITE(COM_IER_1, 0x00); /* 中斷禁止 */
free_irq(INT_UART1, dev); //注銷(xiāo)中斷
del_timer(timer);//刪除定時(shí)器
MOD_DEC_USE_COUNT;
dev->openflag = 0;
…
}
(6)串口讀數據是指返回接收緩沖區中已收到的數據。讀取數據有兩種方式,阻塞方式和非阻塞方式。阻塞方式[6]中用戶(hù)程序執行讀操作時(shí)如果沒(méi)有數據可讀,即讓read()操作等待直到數據可讀;非阻塞方式中當用戶(hù)執行讀操作時(shí),不論串口是否接收到數據,設備驅動(dòng)xxx_read()函數會(huì )立刻返回,read()函數系統調用也隨即返回。
static int serial_read(struct file *filp, char *buf, size_t
count, loff_t *f_pos)
{
if(filp->f_flags O_NONBLOCK)/非阻塞方式讀取
retsts = serial_nonblock_read(dev,buf,count);
else /*阻塞方式讀取*/
retsts = serial_block_read(dev,buf,count);
…
}
(7)串口寫(xiě)數據包括把數據存放在發(fā)送緩沖區、啟動(dòng)硬件發(fā)送及發(fā)送中斷。當發(fā)送第一個(gè)字節后,硬件會(huì )產(chǎn)生發(fā)送中斷,剩下的數據將在中斷處理程序中發(fā)送。
static int serial_write(struct file *filp, const char *buf,
size_t count, loff_t *f_pos)
{
copy_from_user(pSendBuf->frameData[pSendBuf->
bufWritex].data[0],buf, count);
CSR_WRITE(CMBOARD_GPIO_DATAOUT1,status1);
enable_tx_interrupt_1();
…
}
(8)串口控制包括設置串口波特率、奇偶校、停止位等,還可以定義其他特殊的控制。應用程序通過(guò)ioctl()調用把串口的參數傳遞給驅動(dòng)程序,驅動(dòng)程序再通過(guò)對硬件串口控制寄存器進(jìn)行設置,來(lái)滿(mǎn)足應用層用戶(hù)要求。
static int serial_ioctl(struct inode *inode, struct file *flip,
unsigned int cmd, unsigned long arg)
{
switch(cmd){
case SERIAL_IOC_BPS:
…
break;
case SERIAL_IOC_SENDBUF:
…
break;
}
}
(9)中斷處理包括對接收中斷、發(fā)送中斷、異常中斷的處理。讀取中斷寄存器的狀態(tài),根據不同的中斷類(lèi)型分別處理。當收到數據時(shí),硬件會(huì )產(chǎn)生接收中斷,驅動(dòng)程序把串口的數據讀取出來(lái),放在接收緩沖區中,直到所有數據讀取完成;當發(fā)送數據時(shí),硬件會(huì )產(chǎn)生發(fā)送中斷,驅動(dòng)程序把發(fā)送緩沖區的數據發(fā)送出去,直到所有數據發(fā)送完成;當串口接收或發(fā)送發(fā)生異常時(shí),會(huì )產(chǎn)生異常中斷,驅動(dòng)程序根據情況把串口重新初始化,以便串口恢復正常。
static void serial_interrupt(int irq, void * dev_id,
struct pt_regs *regs)
{
status = CSR_READ(COM_IIR_1);
while(status UART_IIR_STATUS_NO) == 0)
{
switch(status)
{
case UART_IIR_STATUS_RDA:
case UART_IIR_STATUS_TOUT:
receive_chars(dev,status);
break;
case UART_IIR_THRE:
transmit_chars(dev);
break;
}
status = CSR_READ(COM_IIR_1);
}
}
(10)定時(shí)器處理。中斷接收程序只負責把數據讀取到緩沖區,并沒(méi)有指示緩沖區的數據可被用戶(hù)使用,這時(shí)需要在超時(shí)程序中把可用標志置上,當用戶(hù)調用read()函數時(shí)就可把接收緩沖區的數據返回。
static void serial_timer(unsigned long dummy)
{
…
serial_device.pRecvBuf->frameData
[serial_device.pRecvBuf->bufWritex].finished = 1;
mod_timer(timer,jiffies+2);/* 20 ms 進(jìn)一次 */
}
通過(guò)以上幾個(gè)函數的處理,實(shí)現了串口的驅動(dòng)。
4 驅動(dòng)程序編譯進(jìn)Linux內核
以下以UART1為例,介紹驅動(dòng)程序編譯進(jìn)Linux內核的過(guò)程,步驟如下:
(1)添加主次設備號。
主次設備號用來(lái)標識一個(gè)具體設備。主設備號用于標識設備類(lèi)型,每種類(lèi)型的設備需要一個(gè)對應的設備驅動(dòng)程序。一個(gè)主設備可以有多個(gè)具體的設備與之對應。次設備號用于區分使用同種驅動(dòng)程序的同類(lèi)設備中多個(gè)不同的設備實(shí)例[7]。
在W90P710-?滋Clinux/?滋Clinux-distlinux-2.4.x/include/
linux目錄下的major.h中定義主設備號,添加如下代碼:
#define SERIAL1_MAJOR 230
在W90P710-?滋Clinux/?滋Clinux-dist/vendors/Winbond/W90P710目錄下的makefile中建立設備主次設備號(主設備號為230,次設備號為1),添加如下代碼:
serial1,c,230,1
(2)在W90P710-?滋Clinux/?滋Clinux-dist/linux-2.4.x/drivers/char目錄下的makefile中添加如下代碼:
obj-$(CONFIG_TOPBAND_SERIAL1)+=w90p710_serial_1.o
(3)在W90P710-?滋Clinux/?滋Clinux-dist/linux-2.4.x/drivers/char目錄下的config.in字符設備段中添加如下代碼:
#if [ $CONFIG_TOPBAND_SERIAL1 = y ]; then
bool 'Topband serial1 support' CONFIG_TOPBAND_
SERIAL1
#fi
(4)在W90P710-?滋Clinux/?滋Clinux-dist目錄下運行make menuconfig,在menuconfig的字符設備選項中可以看見(jiàn)剛剛添加的“CONFIG_TOPBAND_SERIAL1”選項,選上該項。使用make dep、 make clean、make三個(gè)命令編譯Linux內核,生成內核文件linux.bin[8]。
(5)在W90P710-?滋Clinux/romdisk/dev目錄下創(chuàng )建設備文件, 輸入命令:
mknod serial1 c 230 1
生成設備文件“serial1”,應用程序通過(guò)使用“/dev/ serial1”這個(gè)設備文件名就可對串口進(jìn)行操作。
最后編寫(xiě)簡(jiǎn)單的串口測試程序,編譯生成鏡像文件;再把鏡像文件romfs.img和內核文件linux.bin下載到開(kāi)發(fā)板,把開(kāi)發(fā)板的串口和PC機相連,PC機端使用串口調試工具發(fā)送測試數據,開(kāi)發(fā)板能正確收發(fā)數據。
本文按驅動(dòng)程序的功能詳細介紹了W90P710微處理器實(shí)現串口驅動(dòng)的方法,串口驅動(dòng)程序是很典型的字符設備驅動(dòng)程序,其他字符設備驅動(dòng)和串口的實(shí)現方法是相同的,這對開(kāi)發(fā)其他字符設備驅動(dòng)程序有一定的借鑒作用。
linux操作系統文章專(zhuān)題:linux操作系統詳解(linux不再難懂)
linux相關(guān)文章:linux教程
評論