GPIO IRQ для встроенного Linux на базе ARM
Я пытаюсь запрограммировать GPIO IRQ на оценочной плате AT91SAM9M10-EKES. Я успешно зарегистрировал IRQ, и IRQ работает. Однако некоторые прерывания пропущены. Я отправляю 26, а получаю только 22.
Код:
static irqreturn_t wiegand_interrupt(int irq, void *dev_id){
atomic_inc(&counter);
printk(KERN_WARNING "IRQ recieved, counting... %d\n",atomic_read(&counter));
return 0;
}
irq1 = gpio_to_irq(AT91_PIN_PA21);
if (irq1 < 0) {
err = irq1;
printk("Unable to get irq number for GPIO %d, error %d\n",AT91_PIN_PA21, err);
goto fail;
}
err = request_irq(irq1,wiegand_interrupt,0 ,"wiegand",NULL);
irq2 = gpio_to_irq(AT91_PIN_PA20);
if (irq2 < 0) {
err = irq2;
printk("Unable to get irq number for GPIO %d, error %d\n",AT91_PIN_PA21, err);
goto fail;
}
err = request_irq(irq2,wiegand_interrupt,0 ,"wiegand",NULL);
Это не весь драйвер, но это фактическая часть, которая имеет дело с IRQ. Если кто-то видит проблему в коде или может предложить способ узнать, почему я теряю 4 прерывания, ответьте. Я застрял на этом в течение нескольких часов...:(
Благодарю. Рамон.
2 ответа
Я предполагаю, что вы запускаете свои прерывания с помощью внешней системы (может быть, микроконтроллер или что-то, что может переключать GPIOS). Поскольку я не вижу реального подтверждения прерывания, я предполагаю, что внешняя система не ожидает обработки прерывания, чтобы, возможно, вызвать новое.
printk - очень медленная функция, и поэтому вы можете пропустить некоторые прерывания: новое может быть запущено, пока вы еще обрабатываете предыдущее.
Поэтому я бы посоветовал не использовать printk в обработчике. Если вы хотите достичь чего-то подобного, было бы лучше использовать тасклет или рабочую очередь в качестве нижней половины обработчика прерываний.
Я могу только порекомендовать прочитать главу 10 драйверов устройств Linux.
Да, и, кстати, ваш обработчик IRQ должен возвращать не 0, а IRQ_HANDLED.
Хорошо, на самом деле, проблема в том, что я использовал контакты GPIO, в то время как контакты GPIO не поддерживают флаг IRQF_TRIGGER_FALLING, а это именно то, что мне нужно. так что, вероятно, обработчик прерываний не распознает сигнал правильно. Я обнаружил, что мне нужно использовать внешние контакты для IRQF_TRIGGER_FALLING включает IRQ.