<output id="qn6qe"></output>

    1. <output id="qn6qe"><tt id="qn6qe"></tt></output>
    2. <strike id="qn6qe"></strike>

      亚洲 日本 欧洲 欧美 视频,日韩中文字幕有码av,一本一道av中文字幕无码,国产线播放免费人成视频播放,人妻少妇偷人无码视频,日夜啪啪一区二区三区,国产尤物精品自在拍视频首页,久热这里只有精品12

      GICV3中斷控制器調用流程

      GICV3中斷控制器初始化調用鏈

      image

      /kernel/irq/handle.c:
      #ifdef CONFIG_GENERIC_IRQ_MULTI_HANDLER
      int __init set_handle_irq(void (*handle_irq)(struct pt_regs *))
      {
          if (handle_arch_irq)
              return -EBUSY;
      
          handle_arch_irq = handle_irq;
          return 0;
      }
      #endif
      
      /drivers/irqchip/irq-gic-v3.c:
      static int __init gic_init_bases(void __iomem *dist_base,
                       struct redist_region *rdist_regs,
                       u32 nr_redist_regions,
                       u64 redist_stride,
                       struct fwnode_handle *handle)
      {
      ... ...
          set_handle_irq(gic_handle_irq);
      ... ...
      }
      
      /drivers/irqchip/irq-gic-v3.c:
      
      IRQCHIP_DECLARE(gic_v3, "arm,gic-v3", gic_of_init);
      
      static int __init gic_of_init(struct device_node *node, struct device_node *parent)
      {
      ... ...
          err = gic_init_bases(dist_base, rdist_regs, nr_redist_regions,
                       redist_stride, &node->fwnode);
      ... ....
      }
      
      /include/linux/irqchip.h:
      #define IRQCHIP_DECLARE(name, compat, fn) OF_DECLARE_2(irqchip, name, compat, fn)
      
      /include/linux/of.h:
      #define OF_DECLARE_2(table, name, compat, fn) \
              _OF_DECLARE(table, name, compat, fn, of_init_fn_2)
              
      #define _OF_DECLARE(table, name, compat, fn, fn_type)            \
          static const struct of_device_id __of_table_##name        \
              __used __section(__##table##_of_table)            \
               = { .compatible = compat,                \
                   .data = (fn == (fn_type)NULL) ? fn : fn  }
      
      /arch/arm64/kernel/vmlinux.lds:
       .init.data : {
       ... ...
       . = ALIGN(8); __irqchip_of_table = .; KEEP(*(__irqchip_of_table)) KEEP(*(__irqchip_of_table_end)) . = ALIGN(8);
       ... ...
       }
      
      /drivers/irqchip/irqchip.c:
      
      void __init irqchip_init(void)
      {
          of_irq_init(__irqchip_of_table);
          acpi_probe_device_table(irqchip);
      }
      
      /arch/arm64/kernel/irq.c:
      void __init init_IRQ(void)
      {
          init_irq_stacks();
          irqchip_init();
      ... ...
      }
      
      asmlinkage __visible void __init start_kernel(void)
      {
      ... ...
      init_IRQ();
      ... ...
      }
      

      中斷的處理過程

      中斷調用流程

      異常向量表vectors

      /arch/arm64/kernel/entry.S:
      ENTRY(vectors)
          kernel_ventry    1, sync_invalid            // Synchronous EL1t
          kernel_ventry    1, irq_invalid            // IRQ EL1t
          kernel_ventry    1, fiq_invalid            // FIQ EL1t
          kernel_ventry    1, error_invalid        // Error EL1t
      
          kernel_ventry    1, sync                // Synchronous EL1h
          kernel_ventry    1, irq                // IRQ EL1h
          kernel_ventry    1, fiq_invalid            // FIQ EL1h
          kernel_ventry    1, error            // Error EL1h
      
          kernel_ventry    0, sync                // Synchronous 64-bit EL0
          kernel_ventry    0, irq                // IRQ 64-bit EL0
          kernel_ventry    0, fiq_invalid            // FIQ 64-bit EL0
          kernel_ventry    0, error            // Error 64-bit EL0
      
      #ifdef CONFIG_COMPAT
          kernel_ventry    0, sync_compat, 32        // Synchronous 32-bit EL0
          kernel_ventry    0, irq_compat, 32        // IRQ 32-bit EL0
          kernel_ventry    0, fiq_invalid_compat, 32    // FIQ 32-bit EL0
          kernel_ventry    0, error_compat, 32        // Error 32-bit EL0
      #else
          kernel_ventry    0, sync_invalid, 32        // Synchronous 32-bit EL0
          kernel_ventry    0, irq_invalid, 32        // IRQ 32-bit EL0
          kernel_ventry    0, fiq_invalid, 32        // FIQ 32-bit EL0
          kernel_ventry    0, error_invalid, 32        // Error 32-bit EL0
      #endif
      END(vectors)
      

      el1_irq

      /arch/arm64/kernel/entry.S:
          .align    6
      el1_irq:
          kernel_entry 1
          gic_prio_irq_setup pmr=x20, tmp=x1
          enable_da_f
      
      #ifdef CONFIG_ARM64_PSEUDO_NMI
          test_irqs_unmasked    res=x0, pmr=x20
          cbz    x0, 1f
          bl    asm_nmi_enter
      1:
      #endif
      
      #ifdef CONFIG_TRACE_IRQFLAGS
          bl    trace_hardirqs_off
      #endif
      
          irq_handler
      
      #ifdef CONFIG_PREEMPT
          ldr    x24, [tsk, #TSK_TI_PREEMPT]    // get preempt count
      alternative_if ARM64_HAS_IRQ_PRIO_MASKING
          /*
           * DA_F were cleared at start of handling. If anything is set in DAIF,
           * we come back from an NMI, so skip preemption
           */
          mrs    x0, daif
          orr    x24, x24, x0
      alternative_else_nop_endif
          cbnz    x24, 1f                // preempt count != 0 || NMI return path
          bl    arm64_preempt_schedule_irq    // irq en/disable is done inside
      1:
      #endif
      
      #ifdef CONFIG_ARM64_PSEUDO_NMI
          /*
           * When using IRQ priority masking, we can get spurious interrupts while
           * PMR is set to GIC_PRIO_IRQOFF. An NMI might also have occurred in a
           * section with interrupts disabled. Skip tracing in those cases.
           */
          test_irqs_unmasked    res=x0, pmr=x20
          cbz    x0, 1f
          bl    asm_nmi_exit
      1:
      #endif
      
      #ifdef CONFIG_TRACE_IRQFLAGS
      #ifdef CONFIG_ARM64_PSEUDO_NMI
          test_irqs_unmasked    res=x0, pmr=x20
          cbnz    x0, 1f
      #endif
          bl    trace_hardirqs_on
      1:
      #endif
      
          kernel_exit 1
      ENDPROC(el1_irq)
      

      el0_irq

          .align    6
      el0_irq:
          kernel_entry 0
      el0_irq_naked:
          gic_prio_irq_setup pmr=x20, tmp=x0
          ct_user_exit_irqoff
          enable_da_f
      
      #ifdef CONFIG_TRACE_IRQFLAGS
          bl    trace_hardirqs_off
      #endif
      
      #ifdef CONFIG_HARDEN_BRANCH_PREDICTOR
          tbz    x22, #55, 1f
          bl    do_el0_irq_bp_hardening
      1:
      #endif
          irq_handler
      
      #ifdef CONFIG_TRACE_IRQFLAGS
          bl    trace_hardirqs_on
      #endif
          b    ret_to_user
      ENDPROC(el0_irq)
      
      

      irq_handler

      /arch/arm64/kernel/entry.S:
      /*
       * Interrupt handling.
       */
          .macro    irq_handler
          ldr_l    x1, handle_arch_irq
          mov    x0, sp
          irq_stack_entry
          blr    x1
          irq_stack_exit
          .endm
      

      gic_handle_irq

      /drivers/irqchip/irq-gic-v3.c:
      static asmlinkage void __exception_irq_entry gic_handle_irq(struct pt_regs *regs)
      {
          u32 irqnr;
      
          irqnr = gic_read_iar();
      ... ...
      
          /* Check for special IDs first */
          if ((irqnr >= 1020 && irqnr <= 1023))
              return;
      
          /* Treat anything but SGIs in a uniform way */
          if (likely(irqnr > 15)) {
      ... ...
              err = handle_domain_irq(gic_data.domain, irqnr, regs);
      ... ...
          }
          if (irqnr < 16) {
              gic_write_eoir(irqnr);
              if (static_branch_likely(&supports_deactivate_key))
                  gic_write_dir(irqnr);
      #ifdef CONFIG_SMP
              /*
               * Unlike GICv2, we don't need an smp_rmb() here.
               * The control dependency from gic_read_iar to
               * the ISB in gic_write_eoir is enough to ensure
               * that any shared data read by handle_IPI will
               * be read after the ACK.
               */
              handle_IPI(irqnr, regs);
      #else
              WARN_ONCE(true, "Unexpected SGI received!\n");
      #endif
          }
      }
      
      

      中斷狀態

      中斷狀態切換

      1. 硬件觸發中斷信號,中斷assert,GIC標記中斷為PENDING狀態。
      2. GIC中distributor選擇優先級最高的PENDING中斷,發送給CPU interface, CPU interface對優先級進行判定,然后GIC發送中斷請求信號給CPU, CPU進入中斷異常后,通過GICC_IAR讀取硬中斷號,中斷進入ACTIVE狀態。
      3. 硬件觸發新的中斷信號,中斷assert, GIC標記中斷為ACTIVE_AND_PENDING狀態
      4. CPU完成中斷處理,發送EIO信號到GIC(寫EOIR寄存器)

      需要注意的是,INACTIVE PENDING ACTIVE ACTIVE_AND_PENDING,在此例中均屬于GIC硬件所標記的中斷狀態,另外當GIC中斷信號處于PENDING狀態時,硬件外設無法發送新的中斷信號給GIC中斷控制器。事實上中斷屬異步通知并且沒有排隊的概念。

      posted @ 2024-07-18 17:22  StepForwards  閱讀(266)  評論(0)    收藏  舉報
      主站蜘蛛池模板: 东京热人妻无码一区二区AV| 国产亚洲亚洲国产一二区| 久久国产成人av蜜臀| 国内自拍av在线免费| 中文字幕日本一区二区在线观看| 国产精品亚洲综合色区丝瓜 | 偷拍精品一区二区三区| 国产婷婷精品av在线| 日韩有码中文字幕国产| 国产一区二区三区小说| 性欧美老人牲交xxxxx视频 | www免费视频com| 亚洲成在人线在线播放无码| 国99久9在线 | 免费| 国产精品久久久天天影视香蕉 | 天干天干夜啦天干天干国产| 亚洲中文字幕成人综合网| 亚洲中文字幕日韩精品| 91中文字幕一区二区| 国产精品人成视频免| 中文字幕理伦午夜福利片| 亚洲日本中文字幕乱码中文| av无码精品一区二区乱子| 亚洲码国产精品高潮在线| 神木县| 亚洲一区二区av高清| 久久99精品国产99久久6尤物| 激情五月开心综合亚洲| 亚洲色无码专区一区| 日本中文一区二区三区亚洲| 久久精品这里热有精品| 苏尼特左旗| ww污污污网站在线看com| 亚洲国产欧美在线人成AAAA| 午夜精品久久久久久久爽 | 国产精品成人中文字幕| 欧美成人一区二区三区不卡| 国产欧美日韩另类在线专区| av亚洲在线一区二区| 国产午夜精品福利视频| 亚洲熟女精品一区二区|