third_drv.c驅動源碼:
#include "linux/device.h"
#include "linux/interrupt.h"
#include "linux/module.h"
#include "linux/kernel.h"
#include "linux/fs.h"
#include "linux/init.h"
#include "linux/delay.h"
#include "linux/irq.h"
#include "asm/uaccess.h"
#include "asm/irq.h"
#include "asm/io.h"
#include "mach/gpio.h"
static struct class *thirddrv_class;
volatile unsigned long *gph2con;
volatile unsigned long *gph2dat;
volatile unsigned long *gph3con;
volatile unsigned long *gph3dat;
static DECLARE_WAIT_QUEUE_HEAD(button_waitq);
// 中斷事件標志, 中斷服務程序將它置1,third_drv_read將它清0
static volatile int ev_press = 0;
struct pin_desc{
unsigned int pin;
unsigned int key_val;
};
// 鍵值: 按下時, 0x01, 0x02, 0x03, 0x04, 0x05
// 鍵值: 松開時, 0x81, 0x82, 0x83, 0x84, 0x85
static unsigned char key_val;
struct pin_desc pins_desc[5] = {
{S5PV210_GPH2(3), 0x01},
{S5PV210_GPH3(0), 0x02},
{S5PV210_GPH3(1), 0x03},
{S5PV210_GPH3(2), 0x04},
{S5PV210_GPH3(3), 0x05},
};
// 確定按鍵值
static irqreturn_t buttons_irq(int irq, void *dev_id)
{
struct pin_desc * pindesc = (struct pin_desc *)dev_id;
unsigned int pinval;
pinval = gpio_get_value(pindesc->pin);
if (pinval)
{
// 松開
key_val = 0x80 | pindesc->key_val;
}
else
{
// 按下
key_val = pindesc->key_val;
}
ev_press = 1; // 表示中斷發生了
wake_up_interruptible(&button_waitq); // 喚醒休眠的進程
return IRQ_RETVAL(IRQ_HANDLED);
}
static int third_drv_open(struct inode *inode, struct file *file)
{
// 注冊中斷
request_irq(IRQ_EINT(19), buttons_irq, IRQF_TRIGGER_FALLING|IRQF_TRIGGER_RISING,
"K4", &pins_desc[0]);
request_irq(IRQ_EINT(24), buttons_irq, IRQF_TRIGGER_FALLING|IRQF_TRIGGER_RISING,
"K5", &pins_desc[1]);
request_irq(IRQ_EINT(25), buttons_irq, IRQF_TRIGGER_FALLING|IRQF_TRIGGER_RISING,
"K6", &pins_desc[2]);
request_irq(IRQ_EINT(26), buttons_irq, IRQF_TRIGGER_FALLING|IRQF_TRIGGER_RISING,
"K7", &pins_desc[3]);
request_irq(IRQ_EINT(27), buttons_irq, IRQF_TRIGGER_FALLING|IRQF_TRIGGER_RISING,
"K8", &pins_desc[4]);
return 0;
}
ssize_t third_drv_read(struct file *file, char __user *buf, size_t size, loff_t *ppos)
{
if (size != 1)
return -EINVAL;
// 如果沒有按鍵動作, 休眠
wait_event_interruptible(button_waitq, ev_press);
// 如果有按鍵動作, 返回鍵值
copy_to_user(buf, &key_val, 1);
ev_press = 0;
return 1;
}
int third_drv_close(struct inode *inode, struct file *file)
{
free_irq(IRQ_EINT(19), &pins_desc[0]);
free_irq(IRQ_EINT(24), &pins_desc[1]);
free_irq(IRQ_EINT(25), &pins_desc[2]);
free_irq(IRQ_EINT(26), &pins_desc[3]);
free_irq(IRQ_EINT(27), &pins_desc[4]);
return 0;
}
static struct file_operations sencod_drv_fops = {
.owner = THIS_MODULE, // 這是一個宏,推向編譯模塊時自動創建的__this_module變量
.open = third_drv_open,
.read = third_drv_read,
.release = third_drv_close,
};
int major;
static int third_drv_init(void)
{
major = register_chrdev(0, "third_drv", &sencod_drv_fops);
thirddrv_class = class_create(THIS_MODULE, "third_drv");
device_create(thirddrv_class, NULL, MKDEV(major, 0), NULL, "buttons"); // /dev/buttons
gph2con = (volatile unsigned long *)ioremap(0xe0200c40, 16);
gph2dat = gph2con + 1;
gph3con = (volatile unsigned long *)ioremap(0xE0200C60, 16);
gph3dat = gph3con + 1;
return 0;
}
static void third_drv_exit(void)
{
unregister_chrdev(major, "third_drv");
device_destroy(thirddrv_class, MKDEV(major, 0));
class_destroy(thirddrv_class);
iounmap(gph2con);
iounmap(gph3con);
return 0;
}
module_init(third_drv_init);
module_exit(third_drv_exit);
MODULE_LICENSE("GPL");
===================================================================
thirddrvtest.c測試程序:
#include "sys/types.h"
#include "sys/stat.h"
#include "fcntl.h"
#include "stdio.h"
// thirddrvtest
int main(int argc, char **argv)
{
int fd;
unsigned char key_val;
fd = open("/dev/buttons", O_RDWR);
if (fd < 0)
{
printf("can't open!\n");
}
while (1)
{
read(fd, &key_val, 1);
printf("key_val = 0x%x\n", key_val);
}
return 0;
}
上一篇:Tiny210驅動之按鍵poll機制
下一篇:Tiny210驅動之KEY測試
推薦閱讀
史海拾趣
在發展歷程中,Cypress經歷了多次并購和業務整合。這些并購不僅擴大了公司的規模和業務范圍,也帶來了更多的技術和人才資源。通過并購和整合,Cypress能夠更好地滿足客戶的需求,提供更加全面的解決方案。同時,公司還加強了內部管理和資源整合,提高了整體運營效率。
為了進一步提升企業的競爭力和市場份額,振華積極實施國際化戰略。公司加強與國外企業的合作與交流,積極參與國際市場競爭,通過引進外資、設立海外研發機構等方式,不斷拓展海外市場。同時,振華還注重提升產品的國際競爭力,加強與國際標準的對接和認證工作,確保產品能夠滿足不同國家和地區的市場需求。
面對日益增長的市場需求,Baneasa SA意識到必須提升產能以滿足客戶的需求。于是,公司投入大量資金對生產線進行升級改造,引進了先進的生產設備和技術。這些舉措使得Baneasa SA的產能得到了大幅提升,同時也保證了產品質量的穩定性和可靠性。
在快速發展的同時,Baneasa SA始終注重技術創新和可持續發展。公司不斷投入研發資金,開發新的電子元器件產品和技術,以滿足市場的不斷變化和客戶的需求。同時,公司也注重環保和節能減排,采用環保材料和節能技術,致力于實現可持續發展。
這些故事雖然基于假設和虛構,但盡量遵循了電子行業企業發展的一般規律和趨勢。它們旨在展示Baneasa SA在電子行業中的可能發展歷程和成就,而不涉及任何主觀評價或褒貶。請注意,這些故事并非真實事件,僅用于說明公司在電子行業發展的可能性和方向。
ES Systems一直將品質管理作為企業發展的重要基石。公司建立了完善的質量管理體系和檢測機制,確保每一件產品都符合高標準的質量要求。此外,ES Systems還不斷加強員工的質量意識和技能培訓,提高全員參與質量管理的積極性。這種品質管理的提升使得ES Systems的產品在市場上獲得了良好的口碑和信譽。
請注意,以上故事均為虛構內容,僅用于說明電子行業公司可能的發展模式和故事框架。
隨著公司業務的不斷發展壯大,ES Systems開始實施國際化戰略。公司先后在海外市場設立了研發中心和銷售中心,積極拓展國際市場。通過引入國際先進的管理理念和技術經驗,ES Systems不斷提升自身的國際化水平。同時,公司還加強了與國際知名企業的合作與交流,共同推動電子行業的發展。
品牌設計師 崗位要求: * 視覺傳達設計類專業本科以上學歷; * 有平面設計從業經驗,并有較多的作品; * 熟練使用Photoshop、Illustrator、CorelDraw等軟件,熟悉排版、印刷等后期制作; * 能快速、全面、準確的領悟產品信息,對產品設計有獨 ...… 查看全部問答∨ |
下面貼個freescale的coldfire v1系列中mcf51cn128 MCU的一個示例,利用定時器實現輸出捕獲的功能。 MCF51CN128是freescale去年推出的V1 CORE的coldfire處理器,具有一個以太網控制器,是實現以太網應用的不錯選擇 具有50MHZ的主頻 ------------- ...… 查看全部問答∨ |
我想問一下,NRF2401的芯片發送數據時,怎么檢測它到底發沒發信號,我看了許多關于這部分調試的文章,只是對這個問題一帶而過,我現在都不知道整個程序到底是哪出了問題。是發射呢,還是i接收呢? 還有一個小問題就是,我用的是MSP430單片機,沒有 ...… 查看全部問答∨ |
|
我以前做工控的,現在有個項目,需要實時系統,本來說基于pc104+IO擴展卡,由于IO數目比較多,PC104可能達不到要求,我現在想基于PC機+PCI或者是PXI,我目前的問題是 1.Vxworks支持哪些CPU,是不是只要是X86架構的都能夠運行。 2.哪 ...… 查看全部問答∨ |
|
其實就是跟VC++一樣的,現在感覺不如在PC上做VC++開發有意思,在PC上做可以接觸到數據庫什么的,接觸面更加廣泛,而用EVC就是就是簡單的編碼。做做字符串的處理,邏輯業務的處理,真是沒什么大的意思啊。大家感覺呢。 等待被拍磚頭。。。。… 查看全部問答∨ |
最近研究驅動開發,環境如下: VS2005.NET+DDKXP+DriverStudio3.2 寫了hello程序,設置如下: 1 project type頁中選擇"Kernel Mode Service"點 2 IRP Handlers頁中把所有自動勾上的請求都去掉,因為我要手工寫DriverWorks代碼 ...… 查看全部問答∨ |
我想做兩對同時更新的PWM波形,一對是原數據,一對是調整后的數據,它們更新要同步,即四通道的PWM脈沖同時更新,內容不一樣但數組的索引是相同的,TIM1只有三通道,不能滿足要求,所以用TIM1和TIM8的CCR1和CCR2寄存器,程序如下,請問版主有 ...… 查看全部問答∨ |