Hi3519v101gpio字符驱动-按键中断

25 篇文章 0 订阅
10 篇文章 0 订阅

之前测试按键中断存在触发中断之后,读取中断寄存器为0的情况,后来测试发现是挂载海思文件系统之后,有些文件由于权限问题不能初始化,更改为自己配置的文件系统测试就不会出现此类情况。

1、key_irq_drv.c

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/delay.h>
#include <linux/poll.h>
#include <linux/irq.h>
#include <asm/irq.h>
#include <linux/interrupt.h>
#include <asm/uaccess.h>
#include <linux/platform_device.h>
#include <linux/cdev.h>
#include <linux/miscdevice.h>
#include <asm/io.h>
#include <linux/sched.h>

//主设备号
int major;

//udev自动创建节点
static struct class *keyirqdrv_class;
static struct class_device *keyirqdrv_class_dev;

// 操作管脚为gpio14_2
// 该管脚采用boot_sel[1:0]硬件配置复用功能,现单板已经配置为gpio,默认高电平,按键按下为低电平
static void __iomem *key_base; //寄存器基地址

#define gpio_phy_base 0x1214E000   //key管脚物理地址
#define gpio_data key_base + 0x010 //GPIO 数据寄存器 ,0b00_0001_0000
#define gpio_dir key_base + 0x400  //GPIO 方向控制寄存器
#define gpio_is key_base + 0x404   //GPIO 中断触发寄存器
#define gpio_ibe key_base + 0x408  //GPIO 双沿触发中断寄存器
#define gpio_iev key_base + 0x40C  //GPIO 触发中断条件寄存器
#define gpio_ie key_base + 0x410   //GPIO 中断屏蔽寄存器
#define gpio_ris key_base + 0x414  //GPIO 原始中断状态寄存器
#define gpio_mis key_base + 0x418  //GPIO 屏蔽状态中断寄存器
#define gpio_ic key_base + 0x41C   //GPIO 中断清除寄存器

#define gpio_in 0x00 //配置为输入,1输出 0输入def
#define gpio_out_h 0xFF
#define gpio_out_l 0x00

static DECLARE_WAIT_QUEUE_HEAD(button_waitq);
/* 中断事件标志, 中断服务程序置1 不进休眠,drv_read将它清0 进休眠*/
static volatile int ev_press = 0;

unsigned int val; //中断值

//引脚描述结构体
struct pin_irq_desc
{
    unsigned int irq;    //中断号
    unsigned int pin;    //中断标志寄存器,有中断产生时为1,无中断时为0
    unsigned int number; //编号
    char *name;          //名称
};

struct pin_irq_desc pins_desc[] =
    {
        {76, 0x04, 0x01, "key_irq"},
};

static irqreturn_t irq_interrupt(int irq, void *dev_id)
{
    static unsigned int cnt = 0;
    unsigned int key;
    // printk("cnt = %d\n", cnt++);

    key = ioread8(gpio_ris); //读取中断标志位
    // printk("key = %d\n", key);
    val = key;
    ev_press = 1;                         //不进去休眠
    wake_up_interruptible(&button_waitq); //唤醒休眠的进程

    iowrite8(0xFF, gpio_ic); //清除中断

    return IRQ_HANDLED;
}

static int key_drv_open(struct inode *inode, struct file *filp)
{
    iowrite8(gpio_in, gpio_dir); //gpio14_2 配置为输入
    iowrite8(0x04, gpio_is);     //配置电平触发
    iowrite8(0x00, gpio_iev);    //配置下降沿触发或者是低电平触发
    iowrite8(0x00, gpio_ibe);    //配置单边沿触发
    iowrite8(0xFF, gpio_ic);     //清除中断
    iowrite8(0x04, gpio_ie);     //使能gpio14_2中断

    //Hi3519v101好几组GPIO只有一个中断号,所以这里得是共享中断IRQF_SHARED,作为共享中断,dev用来区分不同的中断
    request_irq(pins_desc->irq, irq_interrupt, IRQF_SHARED, pins_desc->name, (void *)&pins_desc);
    printk("set key low irq mode\n");
    return 0;
}

static int key_drv_read(struct file *file, char __user *buf, size_t size, loff_t *ppos)
{
    //如果没有按键动作, 休眠
    wait_event_interruptible(button_waitq, ev_press);

    __copy_to_user(buf, &val, 1);
    ev_press = 0; //数据拷贝给运用之后,一次中断完成,进去休眠

    return 0;
}

int key_drv_close(struct inode *inode, struct file *file)
{
    free_irq(pins_desc->irq, (void *)&pins_desc);
    return 0;
}

static struct file_operations key_drv_fops =
    {
        .owner = THIS_MODULE,
        .open = key_drv_open,
        .read = key_drv_read,
        .release = key_drv_close,
};

int key_drv_int(void)
{
    major = register_chrdev(0, "key_drv", &key_drv_fops);
    keyirqdrv_class = class_create(THIS_MODULE, "key_drv");
    keyirqdrv_class_dev = device_create(keyirqdrv_class, NULL, MKDEV(major, 0), NULL, "key"); //dev/key

    key_base = ioremap(gpio_phy_base, 0x1000);

    printk("key_data = 0x%x\n", (unsigned int)gpio_data);
    printk("key_dir = 0x%x\n", (unsigned int)gpio_dir);
    return 0;
}

void key_drv_exit(void)
{
    unregister_chrdev(major, "key_drv");
    device_unregister(keyirqdrv_class_dev);
    class_destroy(keyirqdrv_class);
}

module_init(key_drv_int);
module_exit(key_drv_exit);

MODULE_LICENSE("GPL");

2、key_irq_drv_test.c

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <unistd.h>

int main(int argc, char **argv)
{
    int fd;
    unsigned int val = 0;

    fd = open("/dev/key", O_RDWR);
    if(fd < 0)
        printf("can`t open!\n");
    while (1)
    {
        read(fd, &val, 1);
        printf("key down!get val = %d\n", val);
    }    

    return 0;
}

3、Makefile

#内核路径
KERN_DIR = /home/osrc/Hi3519V101_SDK_V1.0.4.0/osdrv/opensource/kernel/linux-3.18.y
 
all:
	make -C $(KERN_DIR) M=`pwd` modules 
 
clean:
	make -C $(KERN_DIR) M=`pwd` modules clean
	rm -rf modules.order
 
obj-m += key_irq_drv.o

4、测试

/ # insmod key_irq_drv.ko
key_data = 0xfea4e010
key_dir = 0xfea4e400
/ # ./key_irq_drv_test &
/ # set key rising irq mode

/ #
/ # key down!get val = 4
key down!get val = 4

查看cpu占用率 0%

/ # top
Mem: 12968K used, 61796K free, 0K shrd, 0K buff, 1972K cached
CPU:  0.0% usr  0.0% sys  0.0% nic  100% idle  0.0% io  0.0% irq  0.0% sirq
Load average: 0.10 0.05 0.02 1/46 155
  PID  PPID USER     STAT   VSZ %VSZ CPU %CPU COMMAND
  133     1 root     S     1848  2.4   0  0.0 -sh
  155   133 root     R     1828  2.4   0  0.0 top
    1     0 root     S     1824  2.4   0  0.0 {linuxrc} init
   71     1 root     S <   1276  1.7   0  0.0 udevd --daemon
  124    71 root     S <   1268  1.6   1  0.0 udevd --daemon
  123    71 root     S <   1268  1.6   0  0.0 udevd --daemon
  154   133 root     S      868  1.1   0  0.0 ./key_irq_drv_test
   10     2 root     SW       0  0.0   1  0.0 [migration/1]
   56     2 root     SW       0  0.0   0  0.0 [kworker/0:1]
   21     2 root     SW       0  0.0   0  0.0 [kworker/u4:1]
   36     2 root     SW       0  0.0   1  0.0 [kworker/1:1]
    7     2 root     SW       0  0.0   0  0.0 [rcu_sched]
    3     2 root     SW       0  0.0   0  0.0 [ksoftirqd/0]
   48     2 root     SW       0  0.0   1  0.0 [cfinteractive]
   12     2 root     SW       0  0.0   1  0.0 [kworker/1:0]
   11     2 root     SW       0  0.0   1  0.0 [ksoftirqd/1]
   16     2 root     SW<      0  0.0   1  0.0 [netns]
    9     2 root     SW       0  0.0   0  0.0 [migration/0]
   19     2 root     SW<      0  0.0   1  0.0 [kblockd]
   20     2 root     SW       0  0.0   1  0.0 [spi0]

 

 

 

 

 

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值