GIC Interrupt Map --- Linux Kernel 实现赏析

1 : ID 分配
gic控制器最多支持 1020 个中断源, 支持多少,是soc 芯片设计时确定下来的
如 Altera cyclone 平台支持 256个中断源

SPI 外设中断 (Shared Peripheral Interrupt) 32 ~ 1019

这里写图片描述

先看一下中断控制器的初始化

main.c/* 源头 : uboot最终会跳入这个坑里,唯有源头活水来,uboot现在都die了 */
void start_kernel(void)
{
    ❷:irq.c 
    init_IRQ();
    {
        ❸:irqchip.c
        /* 如果支持Device-Tree ,machine_desc 下面会详细说明*/
        if (IS_ENABLED(CONFIG_OF) && !machine_desc->init_irq)
            irqchip_init();
            {
                ❹ : irq.c   
                of_irq_init();
                {
                    ❺irq-gic.c
                    /* 中断控制器开始初始化,到这里才init,坑太深!*/
                    gic_of_init()  
                }
            }
        else
            machine_desc->init_irq(); 
    }
}

这里写图片描述

好,下面先看一下这个函数

/*
    它是怎么被调用的,下面会说,这里先看它的实现
*/
static void __exception_irq_entry gic_handle_irq(struct pt_regs * regs)
{
        /*
            irqnr   : GIC控制器支持中断源的数目.最大为1020.
                      SOC在芯片设计阶段,是怎么确定下来的呢?
            irqstat : GIC中断控制器中 中断确认寄存器的值,会取出它的低10位的值.          
        */
        u32 irqstat, irqnr;
        /*
            GIC描述结构
            struct gic_chip_data {
                union gic_base dist_base;    中断分发器的地址空间.
                union gic_base cpu_base;     cpu interface的基地址. 
                struct irq_domain * domain;  对应的irq_domain数据结构.
                unsigned int gic_irqs;       GIC支持的IRQ的数目,在gic_init_base中读取对应的寄存器后会为他赋值.    
            };
        */
        struct gic_chip_data *gic = &gic_data[0];
        /*
            在初始化中断控制器的时候已经赋值了,
            资源在初始化阶段被初始化并赋值给对应的结构体成员,到用的时候直接从该结构体中取出,常见且经典的做法.
            gic_of_init(....)
            {
                void __iomem * cpu_base;
                ....
                cpu_base = of_iomap(node,0);  底层使用 ioremap()来映射实现
                ....
                gic_init_bases(gic_cnt,-1,dist_base,cpu_base,percpu_offset,node);
                {
                    ....
                    gic->cpu_base.common_base = dist_base;  //在这里赋值了.
                }
            }
        */
        void __iomem * cpu_base = gic_data_cpu_base(gic);
        /*
            下面是一个 do while()循环, 164 mo ISR会一直中断风暴来了,哈哈! 
        */
        do {
                /*
                    上面说过了,这里读取寄存器赋值
                    Interrupt Acknowledge Register
                    GICC_IAR RO 0x0000003FF 只读 中断ID (硬件中断id)
                */
                irqstat = readl_relaxed(cpu_base + GIC_CPU_INTACK);
                /*
                    bit[9:0] : 发生中断的硬件中断号,

                    Qualcomm 的linux kernel 维护者,姐,看到了吧,是从中断控制器中获取的硬件中断ID
                */
                irqnr = irqstat & GICC_IAR_INT_ID_MASK;
                /*
                    硬件中断号 15 ~ 1020,它是PPI 或者 SPI类型的中断.
                */
                if (likely(irqnr > 15 && irqnr < 1021)) {
                    /*
                        硬件中断号传入进去了,会找到对应的软件中断号,
                        现在 还没有进入中断上下文

                        颠覆OS的革命技术 Interrupt 现在开始 闪亮登场了 . 哈哈!
                    */
                    handle_domain_irq(gic->domain, irqnr, regs);
                    /*
                        神来之笔!
                    */
                    continue;
                }
                /*
                    SGI中断,用于多核之间通讯.暂不分析.
                */  
                if (irqnr < 16) {
                    ....
                }
                break;
    } while (1);
}
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
`interrupt-map`是设备树中用来描述设备与中断控制器之间的中断映射关系的属性之一。它的值是一个中断映射列表,每个中断映射包含以下字段: - `controller`:中断控制器的设备节点路径。 - `data`:表示设备和中断控制器之间的中断映射信息。它可以是一个中断号,也可以是一个中断号数组,表示设备可以使用多个中断号。 - `input-regs`:中断控制器中用来接收中断号的寄存器地址。 下面举一个例子,介绍如何在设备树中使用`interrupt-map`描述设备与中断控制器之间的中断映射关系。假设我们有一个设备节点`/soc/ethernet@10001000`,它使用GIC中断控制器,并使用中断号`68`和`69`: ``` /soc { ethernet@10001000 { compatible = "ethernet"; reg = <0x10001000 0x1000>; interrupt-parent = <&gic>; interrupts = <68 0x4>, <69 0x4>; interrupt-map = <&gic 0 68 0>, <&gic 0 69 0>; interrupt-map-mask = <0 0 0 0xff>; }; }; ``` 在上面的设备节点中,`interrupt-parent`属性指向GIC中断控制器的设备节点路径,并且`interrupts`属性指定了设备使用的中断号。`interrupt-map`属性则用来描述设备和中断控制器之间的中断映射关系。在这个例子中,我们使用了两个中断映射,分别将中断号`68`和`69`映射到了GIC中断控制器的中断号`0`上。`interrupt-map-mask`属性用来指定中断映射的掩码,这里设置为`0xff`,表示只使用了中断控制器的第一个中断号。这个掩码可以用来指定中断映射中需要使用哪些中断号,如果设置为`0`,则表示不使用中断控制器的中断号,而是使用数据中指定的中断号。 总之,使用`interrupt-map`可以让设备树中的设备和中断控制器正确地建立中断映射关系,让内核能够正确地为设备分配中断,并正确地处理中断请求。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值