Linux对电平触发与沿触发中断的区别

对于电平触发中断和沿触发中断,在Linux中分别用了handle_level_irqhandle_edge_irq进行处理。中断发生后,系统的中断开关会自动处于disable状态,这由CPU的硬件保证(至少arm中是这样),所以两个函数都在中断禁止的环境中执行。

 

handle_level_irq

void handle_level_irq (unsigned int irq, struct irq_desc *desc)

       spin_lock(&desc->lock);

       mask_ack_irq(desc, irq);              // 调用了maskunmask之前不应该再进入这个中断

       // 如果是嵌套调用,直接退出

       if (unlikely(desc->status & IRQ_INPROGRESS))

              goto out_unlock;

       desc->status &= ~(IRQ_REPLAY | IRQ_WAITING);

      

       action = desc->action;

       if (unlikely(!action || (desc->status & IRQ_DISABLED)))

              goto out_unlock;

 

       desc->status |= IRQ_INPROGRESS;   // 加中断处理,防止嵌套调用

       spin_unlock(&desc->lock);   // 解锁,为可能的嵌套做准备

       /* handle_IRQ_event中可能会开中断,如果unmask也被调用,可能会产生嵌套调用 */

       action_ret = handle_IRQ_event(irq, action);      

       if (!noirqdebug)

              note_interrupt(irq, desc, action_ret);

 

       spin_lock(&desc->lock);      // 重新加锁

       desc->status &= ~IRQ_INPROGRESS;             // 去除中断处理标记

       if (!(desc->status & IRQ_DISABLED) && desc->chip->unmask)

              // 不管前面handle_IRQ_event中有没有unmask,这里再unmask一次

              desc->chip->unmask(irq);    

out_unlock:

       spin_unlock(&desc->lock);

 

handle_edge_irq

void handle_edge_irq (unsigned int irq, struct irq_desc *desc)

       spin_lock(&desc->lock);

       desc->status &= ~(IRQ_REPLAY | IRQ_WAITING);

 

       if (unlikely((desc->status & (IRQ_INPROGRESS | IRQ_DISABLED)) ||

                  !desc->action)) {

              desc->status |= (IRQ_PENDING | IRQ_MASKED);   // 设置中断嵌套标记

              mask_ack_irq(desc, irq);

              goto out_unlock;

       }

      

       /* ack,这个中断的状态标记被清除 */

       desc->chip->ack(irq);

       /* Mark the IRQ currently in progress.*/

       desc->status |= IRQ_INPROGRESS;                 // 设置中断处理标记位

 

       do {

              struct irqaction *action = desc->action;

              irqreturn_t action_ret;

             

              if (unlikely((desc->status &

                            (IRQ_PENDING | IRQ_MASKED | IRQ_DISABLED)) ==

                           (IRQ_PENDING | IRQ_MASKED)))

             {      // 如果有中断嵌套发生,在这里调用unmask,重新允许嵌套

                     desc->chip->unmask(irq);

                     desc->status &= ~IRQ_MASKED;

              }

 

              desc->status &= ~IRQ_PENDING;

              spin_unlock(&desc->lock);

              /* handle_IRQ_event中可能会重新开中断,导致中断嵌套 */

              action_ret = handle_IRQ_event(irq, action);

              if (!noirqdebug)

                     note_interrupt(irq, desc, action_ret);

              spin_lock(&desc->lock);

       } while ((desc->status & (IRQ_PENDING | IRQ_DISABLED)) == IRQ_PENDING);

 

       desc->status &= ~IRQ_INPROGRESS;

out_unlock:

       spin_unlock(&desc->lock);

 

根据代码可知,这两个函数的处理大同小异,都是调用handle_IRQ_event进行实质性的中断处理工作;它们的区别在于中断嵌套的处理。

l         在电平触发中断处理中,在handle_level_irq的一开始就调用了mask_ack_irq,屏蔽此中断,所以理论上不会产生同一个中断的嵌套调用。但是,在handle_IRQ_event中,会调用驱动程序登记的中断回调函数,而这些函数内核无法控制,无法保证其不调用unmask过程。所以,在调用handle_IRQ_event时,可能仍会出现同一个中断嵌套的情况。解决这个问题的方法很简单,在刚进入handle_level_irq时,使用desc->status & IRQ_INPROGRESS判断当前是否处于嵌套中,如果是嵌套,则直接返回。

l         在沿触发中断处理中,允许同一个中断的嵌套处理,此时使用IRQ_PENDING来标记嵌套,然后在handle_edge_irq中使用do-while来循环处理嵌套中断。沿触发中断在判断当前处于嵌套时,会调用mask禁止再一次出现嵌套,防止中断处理过程被频繁打断。

 

在实际写驱动过程中,经常会出现错误地使用handle_level_irq处理沿触发中断,或者错误地使用handle_edge_irq处理电平触发中断的情况,下面分析一下出现这两种错误时的现象:

使用handle_level_irq处理沿触发中断

由于handle_level_irq中调用了mask,直到中断处理完成之后才调用unmask,在这期间不会响应新的中断,所以沿触发的中断在这个过程中会丢失。

使用handle_edge_irq处理电平触发中断

handle_edge_irq不会调用mask,所以在调用handle_IRQ_event时,如果重新打开中断,就会进入嵌套。嵌套过程中会调用mask,禁止第二次嵌套出现,同时设置IRQ_PENDING标记。等到handle_IRQ_event返回时,由于那个IRQ_PENDING的存在,会再一次调用handle_IRQ_event,导致效率降低。

 

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
下面是一个简单的示例程序,可以让GPIO引脚在高低电平间交替触发中断: ```c #include <stdio.h> #include <stdlib.h> #include <fcntl.h> #include <unistd.h> #include <sys/mman.h> #define GPIO_BASE_ADDR 0x3F200000 // 树莓派GPIO控制器的物理地址 #define BLOCK_SIZE 4096 // GPIO控制器寄存器的偏移地址 #define GPFSEL0 0x00 #define GPFSEL1 0x04 #define GPSET0 0x1c #define GPCLR0 0x28 #define GPLEV0 0x34 #define GPEDS0 0x40 #define GPREN0 0x4c #define GPFEN0 0x58 #define GPHEN0 0x64 #define GPLEN0 0x70 #define GPAREN0 0x7c #define GPAFEN0 0x88 volatile unsigned int *gpio; // 用于存储GPIO控制器寄存器的指针 // 初始化GPIO控制器 void gpio_init() { int fd = open("/dev/mem", O_RDWR | O_SYNC); if (fd < 0) { perror("open /dev/mem failed"); exit(-1); } gpio = (unsigned int *)mmap(NULL, BLOCK_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd, GPIO_BASE_ADDR); if (gpio == MAP_FAILED) { perror("mmap failed"); exit(-1); } close(fd); } // 设置GPIO引脚为输入模式 void gpio_input(unsigned int pin) { unsigned int reg = pin / 10; unsigned int shift = (pin % 10) * 3; gpio[reg] &= ~(7 << shift); } // 设置GPIO引脚为输出模式 void gpio_output(unsigned int pin) { unsigned int reg = pin / 10; unsigned int shift = (pin % 10) * 3; gpio[reg] |= (1 << shift); } // 设置GPIO引脚为高电平 void gpio_high(unsigned int pin) { gpio[GPSET0 / 4] = 1 << pin; } // 设置GPIO引脚为低电平 void gpio_low(unsigned int pin) { gpio[GPCLR0 / 4] = 1 << pin; } // 读取GPIO引脚的电平值 int gpio_read(unsigned int pin) { return (gpio[GPLEV0 / 4] >> pin) & 1; } // 设置GPIO引脚触发中断的方式 void gpio_set_edge(unsigned int pin, unsigned int edge) { unsigned int reg = pin / 32; unsigned int shift = (pin % 32); gpio[reg == 0 ? GPREN0 / 4 : GPHEN0 / 4] = edge << shift; } // 清除GPIO引脚触发中断的标志位 void gpio_clear_event(unsigned int pin) { gpio[GPEDS0 / 4] = 1 << pin; } int main() { gpio_init(); gpio_input(17); // 设置GPIO17为输入模式 gpio_output(18); // 设置GPIO18为输出模式 gpio_set_edge(17, 0b10); // 设置GPIO17为上升沿触发中断 while (1) { if (gpio_read(17)) { gpio_high(18); // 如果GPIO17的电平为高,则设置GPIO18为高电平 } else { gpio_low(18); // 如果GPIO17的电平为低,则设置GPIO18为低电平 } gpio_clear_event(17); // 清除GPIO17的中断标志位 usleep(100000); // 等待100毫秒 } return 0; } ``` 在上面的程序中,我们使用了树莓派GPIO控制器的物理地址和寄存器偏移地址来控制GPIO引脚的输入输出和中断触发方式。具体来说,我们使用了 `mmap()` 函数将GPIO控制器的物理地址映射到用户空间,通过访问对应的寄存器地址来操作GPIO引脚。在 `main()` 函数中,我们不断地读取GPIO17的电平值,如果它为高电平,则设置GPIO18为高电平;如果它为低电平,则设置GPIO18为低电平。同时,我们还设置了GPIO17为上升沿触发中断的方式,并在每次循环结束时清除GPIO17的中断标志位。这样,当GPIO17的电平从低电平变为高电平时,就会触发中断,从而执行相应的中断处理函数。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值