失眠网,内容丰富有趣,生活中的好帮手!
失眠网 > linux 应用层gpio中断_树莓派官方自带gpio中断驱动bcm2708_gpio.c原理分析 linux

linux 应用层gpio中断_树莓派官方自带gpio中断驱动bcm2708_gpio.c原理分析 linux

时间:2019-01-22 22:31:57

相关推荐

linux 应用层gpio中断_树莓派官方自带gpio中断驱动bcm2708_gpio.c原理分析 linux

上一篇记录了树莓派自带的gpio驱动(外链网址已屏蔽),在bcm2708_gpio.c实现gpio驱动的同时其实也实现了中断控制器的驱动,本文记录bcm2708_gpio.c中驱动的实现。

一·bcm2708_gpio_irq_init中断初始化函数建立gpio中断描述表

static void bcm2708_gpio_irq_init(struct bcm2708_gpio *ucb)

{

unsigned irq;

printk(KERN_ERR DRIVER_NAME": bcm2708_gpio_irq_init is not null!\n");

ucb->gc.to_irq = bcm2708_gpio_to_irq;//获取该端口对应的中断号

/*针对每一个IRQ Line建立irq_desc*/

for (irq = GPIO_IRQ_START; irq < (GPIO_IRQ_START + GPIO_IRQS); irq++) {

irq_set_chip_data(irq, ucb);//设置私有数据

irq_set_chip_and_handler(irq, &bcm2708_irqchip, handle_simple_irq);//同时设置irq_desc中的handle_irq回调和irq_chip指针

set_irq_flags(irq, IRQF_VALID);

}

bcm2708_gpio_irq.dev_id= ucb;//用于标示唯一性和传递私有结构体

setup_irq(IRQ_GPIO3, &bcm2708_gpio_irq);//IRQ_GPIO3响应bank1和bank2种所有的中断

}

这个函数是在本驱动的bcm2708_gpio_probe驱动初始化函数的最后面中调用的,作用是初始化中断。

1·首先赋值中断号转换函数

ucb->gc.to_irq = bcm2708_gpio_to_irq;//获取该端口对应的中断号

2.使用循环建立系统所有的中断描述表

irq_set_chip_data(irq, ucb);//设置私有数据

irq_set_chip_and_handler(irq, &bcm2708_irqchip, handle_simple_irq);//同时设置irq_desc中的handle_irq回调和irq_chip指针

3.注册IRQ_GPIO3这个中断。

这个中断是干嘛的?这个问题我找了好多帖子才大致有点了解。IRQ_GPIO3这个中断号的作用是,在bank1或者bank2中如果有任意一个引脚发生电平转换引发中断都会引起这个IRQ_GPIO3这个中断变量的中断。然后驱动便在这个IRQ_GPIO3中断的回调函数中(bcm2708_gpio_interrupt函数)进行扫描寄存器寻找发生中断的中断号,发生中断的即调用相应的回调函数。(下文分析)

二·IRQ_GPIO3中断的bcm2708_gpio_interrupt中断处理函数

static irqreturn_t bcm2708_gpio_interrupt(int irq, void *dev_id)

{

unsignedlongedsr;

unsigned bank;inti;

unsigned gpio;

unsigned level_bits;struct bcm2708_gpio *gpio_data =dev_id;

printk(KERN_ERR DRIVER_NAME": bcm2708_gpio_interrupt %d\n", irq);/*循环两个bank中的中断*/

for (bank = 0; bank < GPIO_BANKS; bank++) {

edsr= readl(__io_address(GPIO_BASE) + GPIOEDS(bank));//获取该bank的基地址

level_bits = gpio_data->high[bank] | gpio_data->low[bank];

for_each_set_bit(i,&edsr, 32) {//循环寻找edsr中为1的位

gpio= i + bank * 32;//找出是哪个gpio

/*ack edge triggered IRQs immediately*/

if (!(level_bits & (1<

writel(1<

generic_handle_irq(gpio_to_irq(gpio));/*ack level triggered IRQ after handling them*/

if (level_bits & (1<

writel(1<

}

}returnIRQ_HANDLED;

}

本函数是IRQ_GPIO3中断向量的回调函数,作用是:扫描寄存器寻找发生中断的中断号,发生中断的即调用相应的回调函数。

1.for_each_set_bit(i, &edsr, 32) //循环寻找edsr中为1的位 即发生中断

2.generic_handle_irq(gpio_to_irq(gpio));//调用该中断向量的中断回调函数(驱动申请irq资源时会填写)gpio_to_irq(gpio)为发生中断的中断向量号

三·中断控制器回调函数

/*中断控制器*/

static struct irq_chip bcm2708_irqchip ={

.name= "GPIO",

.irq_enable= bcm2708_gpio_irq_unmask,//使能该irq,通常是直接调用irq_unmask()

.irq_disable = bcm2708_gpio_irq_mask,//禁止该irq,通常是直接调用irq_mask

.irq_unmask = bcm2708_gpio_irq_unmask,//取消屏蔽该irq

.irq_mask = bcm2708_gpio_irq_mask,//屏蔽该irq

.irq_set_type = bcm2708_gpio_irq_set_type,//设置irq的电气触发条件

};

内核为中断进行了抽象,必要的与平台相关的函数由irq_chip来表述。在这些回调函数中通通是通过写寄存器的方式来开启或者屏蔽中断。

四·打印实验

当我将gpio4同底线短接的时候就会连续发生IRQ_GPIO3中断。在应用层可以使用poll函数来监听这个事件进行相应的处理。

generic_handle_irq(gpio_to_irq(gpio));

linux 应用层gpio中断_树莓派官方自带gpio中断驱动bcm2708_gpio.c原理分析 linux 中断架构 中断子系统...

如果觉得《linux 应用层gpio中断_树莓派官方自带gpio中断驱动bcm2708_gpio.c原理分析 linux 》对你有帮助,请点赞、收藏,并留下你的观点哦!

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。