嵌入式uClinux下的CAN總線設備驅動程序設計
1、uClinux操作系統概述
uClinux是Linux2.0的一個分支,它被設計用于沒有MMU的微控制器領域,即被廣泛應用于嵌入式Linux領域。uClinux的最大特征就是沒有MMU(內存管理單元模塊)。它很適合那些沒有MMU的處理器,如ARM7TDMI,m68ez328等。
uClinux具有完全的TCP/IP協議棧,同時對其他許多的網絡協議都提供支持。這些網絡協議都在uClinux上得到了很好的實現。uClinux可以稱作是一個針對嵌入式系統的優秀網絡操作系統。
2、Linux驅動程序設計概述
Linux系統內核通過設備驅動程序與外圍設備交互,設備驅動程序是Linux內核的一部分,它是一組數據結構和函數,這些數據結構和函數通過定義的接口控制一個或多個設備。
和UNIX一樣,Linux中所有的設備均作為文件來對待,這些文件一般稱為特殊文件,這樣做的一個好處是使用戶或應用程序可按操縱普通文件的方式進行訪問控制硬件設備。
Linux內核有三種類型的設備驅動程序:字符設備驅動程序、塊設備驅動程序和網絡設備驅動程序。Linux的設備由一個主設備號(major)和一個次設備號(minor)標識。主設備號唯一標識了設備類型,它是塊設備表或字符設備表中設備表的索引。次設備號僅由設備驅動程序解釋,用于識別同類設備中,I/O請求所涉及到的那個設備。設備驅動程序可以分為3個主要組成部分:
(l)自動配置和初始化子程序,負責檢測所要驅動的硬件設備是否能正常工作。
(2)服務于I/O請求的子程序,又稱為驅動程序的上半部分。
(3)中斷服務子程序,又稱為驅動程序的下半部分。
3、uClinux下CAN設備的驅動程序編寫
根據上文對LINUX下設備驅動程序的描述,以及參考相關的實例分析,下面對CAN總線設備SJA1000的驅動程序進行編寫。
CAN設備驅動程序實際上是linux內核直接對sja1000器件的初始化與讀寫操作。經分析,sja1000 CAN驅動程序構成包括如下幾個部分:
1)定義sja1000芯片內所有寄存器的訪問地址,用于完成對其內部寄存器以及緩沖區的讀寫訪問。例如:
#define IO_PMOD (*(volatile unsigned *)0x3ff5000)
#define IO_PDATA (*(volatile unsigned *)0x3ff5008)
#define IO_PCON (*(volatile unsigned *)0x3ff5004)
#define SJA_MOD (0x2700000) #define SJA_CMR (0x2700004)
…………………
#define SJA_CANRXB7 (0x270006c) #define SJA_CANRXB8 (0x2700070)
因為在我們的系統中,對sja1000的讀寫是采用的部分模擬時序的方式,所以用到了S3C4510的IO端口。下面對sja1000地址的定義進行分析。因為uClinux運行的時候,采用的是32位方式,即兩個相鄰地址間相隔4個字節,而在sja1000內部的地址間的間隔只有1個字節。雖然可以對S3C4510的內部寄存器定義為在訪問sja1000的時候,將位寬度定義為8位,但這樣會與linux系統運行不匹配,經測試發現讀寫不正常。所以將sja1000的地址定義為32位寬度。于是各個寄存器地址為(基址+sja1000內部地址×4)。這里將sja1000的基址定義為0x2700000。
2)編寫對SJA1000內部寄存器訪問的讀寫函數
因為S3C4510B處理器的地址和數據總線是分開的,而SJA1000的地址與數據總線是8位分時復用的。所以我們只有采用先向sja1000的8位地址數據總線上送出地址,然后再送數據或者讀數據的方式。片選信號/CS,讀信號/RD,寫信號/WR仍由S3C4510B自己產生。需要模擬的是鎖存信號ALE、地址數據總線AD0-AD7。參照sja1000時序圖,具體的操作步驟見下面程序和注釋。
寫子程序如下:
void sja_write(unsigned int data, unsigned int addr)
{ unsigned char tmp;
tmp=(addr)>>2;//將32位地址右移2位,tmp的低8位即為sja1000實際地址。
outl(tmp,addr);//將地址信息作為數據送往SJA1000數據總線
IO_PDATA=0x32;//ALE=0,讓SJA1000將該地址鎖存
outl(data,addr);//將數據信息送往SJA1000數據總線
O_PDATA=0x33; } //將ALE置高電平,74HC245的/OE置高位
讀子程序如下:
unsigned char sja_read(unsigned int addr)
{ unsigned char data;
volatile unsigned int data1;
unsigned char tmp;
tmp=(addr)>>2; //將32位地址右移2位,tmp的低8位即為sja1000實際地址S3C2410
outl(tmp,addr); //將地址信息作為數據送往SJA1000數據總線
IO_PDATA=0x32; //p0-ALE=0,鎖存地址信息
IO_PDATA=0x12; //p5-245dir=0,將74HC245的方向置為CPU輸入方向
data1=inl(addr); //讀出所需的數據
IO_PDATA=0x33; //ALE置高,74HC245置為不工作狀態
data=data1; return(data); }//返回數據
后面對sja1000的初始化、CAN發送與CAN接收函數中需要對寄存器操作均調用sja_write()和sja_read()函數進行。
評論