Linux-2.6.32.2內(nèi)核在mini2440上的移植(九)---觸摸屏驅(qū)動移植
移植環(huán)境
1,主機環(huán)境:VMare下CentOS 5.5 ,1G內(nèi)存。
2,集成開發(fā)環(huán)境:Elipse IDE
3,編譯編譯環(huán)境:arm-linux-gcc v4.4.3,arm-none-linux-gnueabi-gcc v4.5.1。
4,開發(fā)板:mini2440,2M nor flash,128M nand flash。
5,u-boot版本:u-boot-2009.08
6,linux 版本:linux-2.6.32.2
7,參考文章:
嵌入式linux應用開發(fā)完全手冊,韋東山,編著。
Mini2440 之Linux 移植開發(fā)實戰(zhàn)指南
【1】在內(nèi)核中添加觸摸屏驅(qū)動程序
Linux-2.6.32.2 內(nèi)核也沒有包含支持S3C2440 的觸摸屏驅(qū)動,因此友善官方自行設計了一個s3c2410_ts.c,它位于linux-src/drivers/input/touchscreen 目錄下,你可以自己增加一個s3c2410_ts.c 文件,并復制如下內(nèi)容:
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
/* For ts.dev.id.version */
#define S3C2410TSVERSION0x0101
/*定義一個WAIT4INT宏,該宏將對ADC觸摸屏控制寄存器進行操作
S3C2410_ADCTSC_YM_SEN這些宏都定義在regs-adc.h中*/
#define WAIT4INT(x) (((x)<<8) |
S3C2410_ADCTSC_YM_SEN | S3C2410_ADCTSC_YP_SEN | S3C2410_ADCTSC_XP_SEN |
S3C2410_ADCTSC_XY_PST(3))
#define AUTOPST (S3C2410_ADCTSC_YM_SEN | S3C2410_ADCTSC_YP_SEN | S3C2410_ADCTSC_XP_SEN |
S3C2410_ADCTSC_AUTO_PST | S3C2410_ADCTSC_XY_PST(0))
//static char *s3c2410ts_name = "s3c2410 TouchScreen";
#define DEVICE_NAME "mini2440_TouchScreen"/*設備名稱*/
staticstruct input_dev *ts_dev;/*定義一個輸入設備來表示我們的觸摸屏設備*/
staticlong xp;
staticlong yp;
staticint count;
/*定義一個外部的信號量ADC_LOCK,因為ADC_LOCK在ADC驅(qū)動程序中已申明
這樣就能保證ADC資源在ADC驅(qū)動和觸摸屏驅(qū)動中進行互斥訪問*/
extern struct semaphore ADC_LOCK;
static int OwnADC = 0;
static void __iomem *base_addr;/*定義了一個用來保存經(jīng)過虛擬映射后的內(nèi)存地址*/
static inline void s3c2410_ts_connect(void)
{
s3c2410_gpio_cfgpin(S3C2410_GPG(12), S3C2410_GPG12_XMON);
s3c2410_gpio_cfgpin(S3C2410_GPG(13), S3C2410_GPG13_nXPON);
s3c2410_gpio_cfgpin(S3C2410_GPG(14), S3C2410_GPG14_YMON);
s3c2410_gpio_cfgpin(S3C2410_GPG(15), S3C2410_GPG15_nYPON);
}
static void touch_timer_fire(unsigned long data)
{
/*用于記錄這一次AD轉(zhuǎn)換后的值*/
unsigned long data0;
unsigned long data1;
int updown;/*用于記錄觸摸屏操作狀態(tài)是按下還是抬起*/
data0 = ioread32(base_addr+S3C2410_ADCDAT0);
data1 = ioread32(base_addr+S3C2410_ADCDAT1);
/*記錄這一次對觸摸屏是壓下還是抬起,該狀態(tài)保存在數(shù)據(jù)寄存器的第15位,所以需要邏輯與上S3C2410_ADCDAT0_UPDOWN*/
updown = (!(data0 & S3C2410_ADCDAT0_UPDOWN)) && (!(data1 & S3C2410_ADCDAT0_UPDOWN));
if (updown)/*判斷觸摸屏的操作狀態(tài)*/
{
/*如果狀態(tài)是按下,并且ADC已經(jīng)轉(zhuǎn)換了就報告事件和數(shù)據(jù)*/
if (count != 0)//轉(zhuǎn)換四次后進行事件匯報
{
long tmp;
tmp = xp;
xp = yp;
yp = tmp;
//這里進行轉(zhuǎn)換是因為我們的屏幕使用時采用的是240*320,相當于把原來的屏幕的X,Y 軸變換。
//個人理解,不知是否正確
//設備X,Y 值
xp >>= 2;
yp >>= 2;
#ifdef CONFIG_TOUCHSCREEN_MINI2440_DEBUG
/*觸摸屏調(diào)試信息,編譯內(nèi)核時選上此項后,點擊觸摸屏會在終端上打印出坐標信息*/
struct timeval tv;
do_gettimeofday(&tv);
printk(KERN_DEBUG "T: %06d, X: %03ld, Y: %03ldn", (int)tv.tv_usec, xp, yp);
#endif
input_report_abs(ts_dev, ABS_X, xp);
input_report_abs(ts_dev, ABS_Y, yp);
/*報告按鍵事件,鍵值為1(代表觸摸屏對應的按鍵被按下)*/
input_report_key(ts_dev, BTN_TOUCH, 1);
/*報告觸摸屏的狀態(tài),1表明觸摸屏被按下*/
input_report_abs(ts_dev, ABS_PRESSURE, 1);
/*等待接收方受到數(shù)據(jù)后回復確認,用于同步*/
input_sync(ts_dev);
//這個表明我們上報了一次完整的觸摸屏事件,用來間隔下一次的報告
}
/*如果狀態(tài)是按下,并且ADC還沒有開始轉(zhuǎn)換就啟動ADC進行轉(zhuǎn)換*/
xp = 0;
yp = 0;
count = 0;
/*設置觸摸屏的模式為自動轉(zhuǎn)換模式*/
iowrite32(S3C2410_ADCTSC_PULL_UP_DISABLE | AUTOPST, base_addr+S3C2410_ADCTSC);
/*啟動ADC轉(zhuǎn)換*/
iowrite32(ioread32(base_addr+S3C2410_ADCCON) | S3C2410_ADCCON_ENABLE_START, base_addr+S3C2410_ADCCON);
//如果還沒有啟動ADC 或者ACD 轉(zhuǎn)換四次完畢后則啟動ADC
}
else/*否則是抬起狀態(tài)*/
{
//如果是up 狀態(tài),則提出報告并讓觸摸屏處在等待觸摸的階段
count = 0;
input_report_key(ts_dev, BTN_TOUCH, 0);/*報告按鍵事件,鍵值為0(代表觸摸屏對應的按鍵被釋放)*/
input_report_abs(ts_dev, ABS_PRESSURE, 0);/*報告觸摸屏的狀態(tài),0表明觸摸屏沒被按下*/
input_sync(ts_dev);/*等待接收方受到數(shù)據(jù)后回復確認,用于同步*/
iowrite32(WAIT4INT(0), base_addr+S3C2410_ADCTSC);
if (OwnADC)
{
OwnADC = 0;
up(&ADC_LOCK);
}
}
}
/*定義并初始化了一個定時器touch_timer,定時器服務程序為touch_timer_fire*/
static struct timer_list touch_timer =TIMER_INITIALIZER(touch_timer_fire, 0, 0);
/*ADC中斷服務程序,AD轉(zhuǎn)換完成后觸發(fā)執(zhí)行*/
static irqreturn_t stylus_updown(int irq, void *dev_id)
{
unsigned long data0;
unsigned long data1;
int updown;
//注意在觸摸屏驅(qū)動模塊中,這個ADC_LOCK 的作用是保證任何時候都只有一個驅(qū)動程序使用ADC 的
//中斷線,因為在mini2440adc 模塊中也會使用到ADC,這樣只有擁有了這個鎖,才能進入到啟動ADC
if (down_trylock(&ADC_LOCK) == 0)
{
OwnADC = 1;
data0 = ioread32(base_addr+S3C2410_ADCDAT0);
data1 = ioread32(base_addr+S3C2410_ADCDAT1);
/*記錄這一次對觸摸屏是壓下還是抬起,該狀態(tài)保存在數(shù)據(jù)寄存器的第15位,所以需要邏輯與上S3C2410_ADCDAT0_UPDOWN*/
updown = (!(data0 & S3C2410_ADCDAT0_UPDOWN)) && (!(data1 & S3C2410_ADCDAT0_UPDOWN));
if (updown)
{
touch_timer_fire(0);//這是一個定時器函數(shù),當然在這里是作為普通函數(shù)調(diào)用,用來啟動ADC
}
else
{
OwnADC = 0;
up(&ADC_LOCK);//注意這部分是基本不會執(zhí)行的,除非你觸摸后以飛快的速度是否,還來
//不及啟動ADC,當然這種飛快的速度一般是達不到的,筆者調(diào)試程序時發(fā)現(xiàn)這里是進入不了的
}
}
return IRQ_HANDLED;
}
static irqreturn_t stylus_action(int irq, void *dev_id)
{
unsigned long data0;
unsigned long data1;
if (OwnADC) {//讀取數(shù)據(jù)
data0 = ioread32(base_addr+S3C2410_ADCDAT0);
data1 = ioread32(base_addr+S3C2410_ADCDAT1);
xp += data0 & S3C2410_ADCDAT0_XPDATA_MASK;
yp += data1 & S3C2410_ADCDAT1_YPDATA_MASK;
count++;
if (count < (1<<2)) {//如果小如四次重新啟動轉(zhuǎn)換
iowrite32(S3C2410_ADCTSC_PULL_UP_DISABLE | AUTOPST, base_addr+S3C2410_ADCTSC);
iowrite32(ioread32(base_addr+S3C2410_ADCCON) | S3C2410_ADCCON_ENABLE_START, base_addr+S3C2410_ADCCON);
} else {//如果超過四次,則等待1ms 后進行數(shù)據(jù)上報
mod_timer(&touch_timer, jiffies+1);
iowrite32(WAIT4INT(1), base_addr+S3C2410_ADCTSC);
}
}