简单的中断处理程序:request_irq返回错误代码-22

我正在写一个简单的内核模块,它可以注册一个中断并处理它。 但是,当我尝试通过调用request_irq函数来注册中断时,它将返回错误代码-22:

错误:无法请求IRQ 30 – 代码-22,EIO 5,EINVAL 22

我相信,这个错误代码等于EINVAL(无效的参数)

请告诉我,我做错了什么。 这里是一个模块:

#include <linux/init.h> #include <linux/module.h> #include <linux/irq.h> #include <linux/io.h> #include <linux/irqdomain.h> #include <linux/interrupt.h> #include <linux/of.h> #include <linux/of_address.h> #include <asm/exception.h> #include <asm/mach/irq.h> void int068_interrupt(int irq, void *dev_id, struct pt_regs *regs) { printk("Interrupt should be handled there\n"); } static int __init clcdint_init(void) { unsigned int irq; unsigned int irqflags; int ret; irq=68; irqflags=IRQF_SHARED | IRQF_NO_SUSPEND; ret = request_irq(irq, int068_interrupt, irqflags, "clcdint-int068", NULL); if (ret!=0) { printk("ERROR: Cannot request IRQ %d", irq); printk(" - code %d , EIO %d , EINVAL %d\n", ret, EIO, EINVAL); } printk("CLCDINT_INIT\n"); return 0; } module_init(clcdint_init); static void __exit clcdint_exit(void) { unsigned int irq; irq=68; free_irq(irq, NULL); printk("CLCDINT_EXIT\n"); } module_exit(clcdint_exit); 

在处理共享中断线(IRQF_SHARED标志打开)时,不能传递NULL上下文(request_irq()调用的最后一个参数)。

要理解为什么考虑以下情况:您有两个相同的网卡共享相同的IRQ。 相同的驱动程序将传递相同的中断处理函数,相同的irq编号和相同的描述。 除了通过上下文参数之外,没有办法区分注册的两个实例。

因此,作为预防措施,如果您传递IRQF_SHARED标志,则不能传递NULL上下文参数。

irqflags有一个unsigned int类型,但是最初它的类型是long

尝试下面的说法,它肯定会工作:

request_irq(irq, int068_interrupt,IRQF_SHARED | IRQF_NO_SUSPEND, "clcdint-int068", NULL);