arm 外部中断

main.c:

#include"key_inc.h"
//封装延时函数
void delay(int ms)
{
    int i,j;
    for(i=0;i<ms;i++)
    {
        for(j=0;j<2000;j++)
        {}

    }
}
int main()
{
    //按键中断的初始化
    key1_it_config();
    key2_it_config();
    key3_it_config();
    while(1)
    {
        printf("in main pro\n");
        delay(1000);
    }
    return 0;
}

key_inc.c:

#include "key_inc.h"
void key1_it_config()
{
    // 使能GPIOF外设时钟
    RCC->MP_AHB4ENSETR |= (0x1 << 5);
    // 将PF9设置为输入模式
    GPIOF->MODER &= (~(0x3 << 18));
    // 设置由PF9管脚产生EXTI9事件
    EXTI->EXTICR3 &= (~(0XFF << 8));
    EXTI->EXTICR3 |= (0X05 << 8);
    // 通过下降沿检测EXTI9事件是否产生
    EXTI->FTSR1 |= (0x1 << 9);
    // 取消EXTI9中断屏蔽
    EXTI->C1IMR1 |= (0x1 << 9);
    // 设置EXTI9中断全局使能
    GICD->ISENABLER[3] |= (0x1 << 3);
    // 设置中断优先级
    GICD->IPRIORITYR[24] &= (~(0x1f << 27));
    // 设置CPU0处理当前中断
    GICD->ITARGETSR[24] |= (0x1 << 24);
    // 设置组0中断可以被转发到GICC层
    GICD->CTRL |= 0X1;
    // 设置中断优先级掩码
    GICC->PMR |= (0x1f << 3);
    // 设置组0中断交给指定处理器处理
    GICC->CTRL |= 0x1;
}

void key2_it_config()
{
    // 使能GPIOF外设时钟
    RCC->MP_AHB4ENSETR |= (0x1 << 5);
    // 将PF8设置为输入模式
    GPIOF->MODER &= (~(0x3 << 14));
    // 设置由PF7管脚产生EXTI7事件
    EXTI->EXTICR2 &= (~(0XFF << 24));
    EXTI->EXTICR2 |= (0X05 << 24);
    // 通过下降沿检测EXTI7事件是否产生
    EXTI->FTSR1 |= (0x1 << 7);
    // 取消EXTI7中断屏蔽
    EXTI->C1IMR1 |= (0x1 << 7);
    // 设置EXTI7中断全局使能
    GICD->ISENABLER[3] |= (0x1 << 1);
    // 设置中断优先级
    GICD->IPRIORITYR[24] &= (~(0x1f << 11));
    // 设置CPU0处理当前中断
    GICD->ITARGETSR[24] |= (0x1 << 8);
    // 设置组0中断可以被转发到GICC层
    GICD->CTRL |= 0X1;
    // 设置中断优先级掩码
    GICC->PMR |= (0x1f << 3);
    // 设置组0中断交给指定处理器处理
    GICC->CTRL |= 0x1;
}

void key3_it_config()
{
    // 使能GPIOF外设时钟
    RCC->MP_AHB4ENSETR |= (0x1 << 5);
    // 将PF8设置为输入模式
    GPIOF->MODER &= (~(0x3 << 16));
    // 设置由PF8管脚产生EXTI9事件
    EXTI->EXTICR3 &= (~(0XFF << 0));
    EXTI->EXTICR3 |= (0X05 << 0);
    // 通过下降沿检测EXTI8事件是否产生
    EXTI->FTSR1 |= (0x1 << 8);
    // 取消EXTI8中断屏蔽
    EXTI->C1IMR1 |= (0x1 << 8);
    // 设置EXTI8中断全局使能
    GICD->ISENABLER[3] |= (0x1 << 2);
    // 设置中断优先级
    GICD->IPRIORITYR[24] &= (~(0x1f << 19));
    // 设置CPU0处理当前中断
    GICD->ITARGETSR[24] |= (0x1 << 16);
    // 设置组0中断可以被转发到GICC层
    GICD->CTRL |= 0X1;
    // 设置中断优先级掩码
    GICC->PMR |= (0x1f << 3);
    // 设置组0中断交给指定处理器处理
    GICC->CTRL |= 0x1;
}

do_irq.c:

#include "stm32mp1xx_gic.h"
#include "stm32mp1xx_exti.h"
extern void printf(const char *fmt, ...);
unsigned int i = 0;
void do_irq(void)
{
    // 获取要处理的中断的中断号
    unsigned int irqno = GICC->IAR & 0x3ff;
    switch (irqno)
    {
    case 99:
        printf("key1_intc\n");
        // 清除GICD中断排队标志
        GICD->ICPENDR[3] |= (0X1 << 3);
        // 清除EXTI中断触发标志
        EXTI->FPR1 |= (0X1 << 9);
        break;

    case 98:
        printf("key3_intc\n");
        // 清除GICD中断排队标志
        GICD->ICPENDR[3] |= (0X1 << 2);
        // 清除EXTI中断触发标志
        EXTI->FPR1 |= (0X1 << 8);
        break;

    case 97:
        printf("key2_intc\n");
        // 清除GICD中断排队标志
        GICD->ICPENDR[3] |= (0X1 << 1);
        // 清除EXTI中断触发标志
        EXTI->FPR1 |= (0X1 << 7);
        break;

    default:
        break;
    }
    // 清除IAR寄存器中的中断号
    GICC->EOIR |= irqno;
}

  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: "Linux ARM GIC 中断流程" 是指在 ARM 架构的 Linux 操作系统中,使用 Generic Interrupt Controller(GIC)处理器来管理系统中各种中断的流程。它负责将来自不同设备和外部事件的中断请求,分派给适当的处理程序来处理,并确保系统按顺序执行这些处理程序。中断处理是系统中一个重要的组成部分,它允许设备和软件互相通信,并提高了系统的稳定性和可靠性。 ### 回答2: Linux是一个非常流行的开源操作系统,可以在不同架构的计算机上运行。ARM架构是一种广泛使用的嵌入式系统架构,因此,许多嵌入式设备都使用Linux作为其操作系统。在ARM架构上,通常会使用GIC(通用中断控制器)来管理中断,这个过程可以分为中断触发、CPU响应和中断处理三个部分。 中断触发是指中断信号从设备到达GIC的过程。当一个设备需要发送一个中断时,它会向GIC发送一个中断请求信号,并指定中断号,这个中断号是唯一的,用于区分不同的中断。GIC会根据中断号去查找到这个中断对应的中断控制器,进而把这个请求传递给指定的CPU。 CPU响应是指CPU接收到中断请求信号后的响应过程。当GIC把中断请求传递给CPU时,CPU需要检查是否允许这个中断请求,也就是检查中断屏蔽寄存器(Interrupt Mask Register)。如果这个中断请求已被屏蔽,则CPU不会响应,否则,它会设置自己的中断挂起寄存器(Interrupt Pending Register),告诉GIC它已经准备好去处理这个中断。 中断处理是指CPU执行中断处理程序,处理具体的中断。当CPU设置了它的中断挂起寄存器后,GIC会向CPU发送一个中断信号。CPU会暂停当前的进程,并把当前的上下文信息(比如,寄存器值)保存到内存中。之后,CPU会跳转到中断处理程序(Interrupt Service Routine),开始执行具体的中断处理代码。中断处理程序完成后,CPU会从内存中恢复之前保存的上下文信息,并恢复之前进程的执行。 总的来说,ARM架构上的Linux操作系统通常使用GIC来管理中断,其中包括中断触发、CPU响应和中断处理三个方面。这个流程对于保证系统的稳定性和快速响应非常重要。 ### 回答3: 在ARM架构的Linux系统中,GIC(Generic Interrupt Controller)被用来管理中断。当发生中断时,GIC会将中断信号发送到CPU,然后CPU会停止当前的进程并处理该中断。 GIC的中断流程如下: 1. 报告中断:设备或其他外部事件引发中断信号,设备向GIC发送中断信号,GIC会产生一个中断源标识符,然后将其发给CPU。 2. 响应中断:CPU根据中断源标识符查询GIC,查看中断请求的优先级和处理器状态,如果中断请求的优先级高于当前中断处理器,那么CPU会中止当前进程,执行中断处理程序。 3. 中断处理:中断处理程序会读取设备状态,进行相应的操作,处理完成后会发出一个中断信号,通知GIC中断已被处理。 4. 中断结束:GIC收到来自设备的中断完成信号后,将中断源标识符置为未激活状态。 在这个流程中,GIC起到了一种路由的作用,将中断信号从设备传输到CPU,同时根据中断请求的优先级来优先处理高优先级的中断。这样就可以保证系统的稳定性和可靠性。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值