异常向量表的安装与调用

Posted 韦东山

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了异常向量表的安装与调用相关的知识,希望对你有一定的参考价值。

1. 回顾中断的发生、处理过程

  • 中断发生的硬件过程

  • 中断处理的软件处理流程
    • CPU执行完当前指令,检查到发生了中断,跳到向量表
    • 保存现场、执行GIC提供的处理函数、恢复现场

2. 异常向量表的安装

2.1 复制向量表

  • 汇编代码
// arch\\arm\\kernel\\head.S
1. bl	__lookup_processor_type
   ...... 
2. bl	__create_page_tables
3. ldr	r13, =__mmap_switched
4. b	__enable_mmu
   b	__turn_mmu_on
   mov	r3, r13
   ret	r3
5. __mmap_switched: // arch\\arm\\kernel\\head-common.S
6. b	start_kernel
  • 复制向量表
start_kernel // init\\main.c
    setup_arch(&command_line); // arch\\arm\\kernel\\setup.c
		paging_init(mdesc);    // arch\\arm\\mm\\mmu.c
			devicemaps_init(mdesc); // arch\\arm\\mm\\mmu.c
				vectors = early_alloc(PAGE_SIZE * 2); // 1.分配新向量表
				early_trap_init(vectors);             // 2.从代码把向量表复制到新向量表

				// 3. 映射新向量表到虚拟地址0xffff0000
                /*
                 * Create a mapping for the machine vectors at the high-vectors
                 * location (0xffff0000).  If we aren't using high-vectors, also
                 * create a mapping at the low-vectors virtual address.
                 */
                map.pfn = __phys_to_pfn(virt_to_phys(vectors));
                map.virtual = 0xffff0000;
                map.length = PAGE_SIZE;
            #ifdef CONFIG_KUSER_HELPERS
                map.type = MT_HIGH_VECTORS;
            #else
                map.type = MT_LOW_VECTORS;
            #endif
                create_mapping(&map);

2.2 向量表在哪

上面代码中可以看到代码中向量表位于__vectors_start,它在arch/arm/kernel/vmlinux.lds中定义:

 __vectors_start = .;
 .vectors 0xffff0000 : AT(__vectors_start) {
  *(.vectors)
 }
 . = __vectors_start + SIZEOF(.vectors);
 __vectors_end = .;
 __stubs_start = .;
 .stubs ADDR(.vectors) + 0x1000 : AT(__stubs_start) {
  *(.stubs)
 }

在代码里搜.vectors,可以找到向量表:

3. 中断向量

发生中断时,CPU跳到向量表去执行b vector_irq

vector_irq函数使用宏来定义:

4. 处理流程

5. 处理函数

以上是关于异常向量表的安装与调用的主要内容,如果未能解决你的问题,请参考以下文章

ARM64的启动过程之:异常向量表的设定

从内核模块中查找异常向量表的物理地址

s5pv210——中断

Linux软中断与系统调用

Linux异常处理体系结构

从活动调用片段方法时出现空指针异常