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

时间:2013-03-06 11:04:21

标签: c linux module linux-kernel interrupt

我正在编写一个简单的内核模块,它可以注册一个中断并处理它。 但是,当我尝试通过调用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);

2 个答案:

答案 0 :(得分:12)

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

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

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

答案 1 :(得分:-2)

irqflags的类型为unsigned int,但最初的类型为long

尝试以下语句,肯定会有效:

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