泰晓科技 -- 聚焦 Linux - 追本溯源,见微知著!
网站地址:https://tinylab.org

泰晓Linux实验盘,即刻上手内核与嵌入式开发
请稍侯

RISC-V 缺页异常处理程序分析(1):do_page_fault() 和 handle_mm_fault()

Jinyu Tang 创作于 2023/03/28

Corrector: TinyCorrect v0.1-rc3 - [pangu] Author: tjytimi tjytimi@163.com Date: 2022/07/25 Revisor: lzufalcon falcon@tinylab.org Project: RISC-V Linux 内核剖析 Sponsor: PLCT Lab, ISCAS

1 前言

本系列将分析缺页异常处理,其中,与处理器架构相关的部分采用 RICS-V 架构下对应的代码。

本文为缺页异常的第一篇,此篇分析 do_page_fault() 函数在 RISC-V 架构下的实现以及用户态合法地址缺页异常处理中 handle_mm_fault() 函数流程。

内核版本为 Linux 5.17。

2 缺页异常函数流程分析(do_page_fault())

缺页异常函数由处理器架构相关函数 do_page_fault 实现。实际上不同架构处理逻辑类似。内核为每个处理器架构分别实现此函数是因为内核需要从处理器的寄存器中获取缺页异常产生的地址及原因等信息,不同架构的处理器对应的寄存器和机制不尽相同。

尽管不同处理器实现方式不同,但任何架构处理器均需有寄存器和相关机制为内核提供这些信息,由此可见操作系统和处理器间互相依存的关系。

do_page_fault 函数的讲解,很多教材采取了画流程图的方式,不过我在看书时感觉这个函数因为分支较多,图看起来也会让人混乱。开发代码阶段画好图有帮助,但现在有代码,就直接在代码上放上注释,读者根据代码注释看下来,并参考代码下的总结文字就能理清此函数的流程。

  1. asmlinkage void do_page_fault(struct pt_regs *regs)
  2. {
  3. unsigned long addr, cause;
  4. ...
  5. /*
  6. * 异常处理开始前,会将寄存器值压入内核栈,regs 即为栈内存有寄存器值的结构体
  7. * cause 寄存器存有页异常原因有关信息,badaddr 寄存器存有异常对应的虚拟地址
  8. */
  9. cause = regs->cause;
  10. addr = regs->badaddr;
  11. /* 获取获取当前进程的内存上下文 mm */
  12. tsk = current;
  13. mm = tsk->mm;
  14. /*
  15. * Fault-in kernel-space virtual memory on-demand.
  16. * The 'reference' page table is init_mm.pgd.
  17. *
  18. * NOTE! We MUST NOT take any locks for this case. We may
  19. * be in an interrupt or a critical region, and should
  20. * only copy the information from the master page table,
  21. * nothing more.
  22. */
  23. /* 在 vmalloc 区域的错误,由 vmalloc_fault() 处理 */
  24. if (unlikely((addr >= VMALLOC_START) && (addr < VMALLOC_END))) {
  25. vmalloc_fault(regs, code, addr);
  26. return;
  27. }
  28. ...
  29. /*
  30. * If we're in an interrupt, have no user context, or are running
  31. * in an atomic region, then we must not take the fault.
  32. */
  33. /* 内核线程、中断上下文、原子上下文或缺页处理句柄关闭,交由 no_context() 处理 */
  34. if (unlikely(faulthandler_disabled() || !mm)) {
  35. tsk->thread.bad_cause = cause;
  36. no_context(regs, addr);
  37. return;
  38. }
  39. /*
  40. * 同时满足缺页发生在内核态,地址在进程地址范围内,且缺页不是内核访问用户态页表里存在的地址(status 中 SUM 位未置位,在处理器处于
  41. * 内核态时访问用户空间地址会由处理器对该位置位),那就是一个 bug
  42. */
  43. if (!user_mode(regs) && addr < TASK_SIZE &&
  44. unlikely(!(regs->status & SR_SUM)))
  45. die_kernel_fault("access to user memory without uaccess routines",
  46. addr, regs);
  47. retry:
  48. mmap_read_lock(mm);
  49. /* 找到第一个末尾大于等于 addr 的虚拟地址空间 */
  50. vma = find_vma(mm, addr);
  51. /* 没有找到,说明该地址一定没有被包含在任何地址空间内(注:栈向下生长,也不可能由于栈没有分配),为用户态编程错误,由 bad_area() 处理 */
  52. if (unlikely(!vma)) {
  53. tsk->thread.bad_cause = cause;
  54. bad_area(regs, mm, code, addr);
  55. return;
  56. }
  57. /* 地址被该虚拟地址空间覆盖,跳到 good_area 处理,此标号表明可以抢救一下 */
  58. if (likely(vma->vm_start <= addr))
  59. goto good_area;
  60. /* 如果不是向下增长的虚拟地址空间(栈对应的),走到这说明错误地址不在进程地址空间,也去 bad_area() 里处理 */
  61. if (unlikely(!(vma->vm_flags & VM_GROWSDOWN))) {
  62. tsk->thread.bad_cause = cause;
  63. bad_area(regs, mm, code, addr);
  64. return;
  65. }
  66. /*
  67. * 如果是栈区,有可能是压栈操作栈不够导致(给栈分配虚拟地址空间机制是用完了继续扩),进一步判断后扩大一下栈对应的虚拟地址空间,
  68. * 如果判断地址离得的太远等则不是压栈操作,也去 bad_area() 处理
  69. */
  70. if (unlikely(expand_stack(vma, addr))) {
  71. tsk->thread.bad_cause = cause;
  72. bad_area(regs, mm, code, addr);
  73. return;
  74. }
  75. /*
  76. * Ok, we have a good vm_area for this memory access, so
  77. * we can handle it.
  78. */
  79. /* 到这里说明缺页的地址是属于该进程 */
  80. good_area:
  81. code = SEGV_ACCERR;
  82. /* access_error() 为真说明权限和访问操作不对应,访问不合法,也到 bad_area() 处理 */
  83. if (unlikely(access_error(cause, vma))) {
  84. tsk->thread.bad_cause = cause;
  85. bad_area(regs, mm, code, addr);
  86. return;
  87. }
  88. /*
  89. * If for any reason at all we could not handle the fault,
  90. * make sure we exit gracefully rather than endlessly redo
  91. * the fault.
  92. */
  93. /* 走到这里说明页异常发生在处理器处于用户态时且地址处于合法地址空间,将交由 handle_mm_fault() 进一步处理 */
  94. fault = handle_mm_fault(vma, addr, flags, regs);
  95. ...
  96. }

上面的代码用文字描述如下:

  • 函数首先获取 cause 寄存器存放的缺页异常原因有关信息,并从 badaddr 寄存器得到产生异常的虚拟地址,接着获取当进程的内存上下文 mm

  • vmalloc 区域的错误,由 vmalloc_fault() 处理。

  • 内核线程(mm 为空),中断上下文、原子上下文或缺页处理句柄关闭(faulthandler_disabled() 函数为真),交由 no_context() 处理。

  • 缺页发生在内核态,缺页地址在进程地址范围内(addr < TASK_SIZE),且缺页不是发生在内核态访问用户态页表里存在的地址时(statusSUM 位未置位,该位在处理器内核态访问用户空间地址时由处理器置位),那就是一个 bug,直接 Oops 并且杀死此进程。

  • 找到第一个末尾大于等于 addr 的虚拟地址空间 vma,若没有找到,说明该地址一定没有被包含在任何地址空间内(注:由于栈是向下生长,所以此处也不可能由于栈没有分配导致的异常),用户态编程错误,由 bad_area() 处理。

  • 找到且确定地址被该虚拟地址空间覆盖,跳到 good_area 标号处处理。

  • 没有被虚拟地址覆盖,继续判断是不是向下增长的虚拟地址空间(栈为情况),若不是,则为错误,也去 bad_area() 里处理。

  • 若是栈区,有可能是压栈操作栈不够导致(给栈分配虚拟地址空间是用完了才继续扩),进一步判断并扩大一下栈对应的虚拟地址空间,如果判断地址离得的太远等则不是压栈操作导致,也去 bad_area() 处理。

  • 执行到了 good_area 标号处,继续调用 access_error() 判断权限,access_error() 为真说明权限和访问操作不对应,访问非法,也到 bad_area() 处理。

  • 运行至此,终于可以确定为合法的缺页异常了,调用 handle_mm_fault() 处理,该异常是缺页异常最常见的原因。

3 缺页异常发生在处理器处于用户态时且地址处于合法地址空间的进一步处理

下面介绍 handle_mm_fault() 相关流程。do_page_fault()bad_area()vmalloc_fault() 等代码分支流程,将在后续连载中介绍。

3.1 __handle_mm_fault() 函数

handle_mm_fault() 函数负责处理用户态缺页且地址合法的情形。该函数主要功能由 __handle_mm_fault() 实现,代码如下,其中精简了与巨页有关的代码,同样采用注释和文字的形式分析。

  1. // mm/memory.c:4619
  2. static vm_fault_t __handle_mm_fault(struct vm_area_struct *vma,
  3. unsigned long address, unsigned int flags)
  4. {
  5. struct vm_fault vmf = {
  6. .vma = vma,
  7. .address = address & PAGE_MASK,
  8. .flags = flags,
  9. .pgoff = linear_page_index(vma, address),
  10. .gfp_mask = __get_fault_gfp_mask(vma),
  11. };
  12. /* 获取虚拟地址空间对应的内存描述符 */
  13. struct mm_struct *mm = vma->vm_mm;
  14. pgd_t *pgd;
  15. p4d_t *p4d;
  16. vm_fault_t ret;
  17. /* 通过内存描述符和虚拟地址,得到该虚拟地址在页全局目录(pgd)中对应的目录项指针 */
  18. pgd = pgd_offset(mm, address);
  19. /* 通过内存描述符、pgd 项和虚拟地址,得到 p4d 项指针 */
  20. p4d = p4d_alloc(mm, pgd, address);
  21. if (!p4d)
  22. return VM_FAULT_OOM;
  23. ...
  24. /* 通过内存描述符、p4d 项和虚拟地址,得到 pud 项指针 */
  25. vmf.pud = pud_alloc(mm, p4d, address);
  26. if (!vmf.pud)
  27. return VM_FAULT_OOM;
  28. ...
  29. /* 通过内存描述符、pud 项和虚拟地址,得到 pmd 项指针 */
  30. vmf.pmd = pmd_alloc(mm, vmf.pud, address);
  31. if (!vmf.pmd)
  32. return VM_FAULT_OOM;
  33. ...
  34. /* 将 vmf 的地址传给 handle_pte_fault() 进行页表项的进一步处理 */
  35. return handle_pte_fault(&vmf);
  36. }

该函数首先初始化结构体变量 vmf,用于整个流程的参数传递,让参数不至于那么多。随后通过以下流程一路下来得到缺页地址对应的页中间目录项的指针(vmf.pmd),该指针对应地址的值将会是页表的物理地址:

  • 获取虚拟地址空间对应的内存描述符 mm

  • pgd_offset() 通过内存描述符和虚拟地址,得到该虚拟地址在页全局目录(pgd)中对应的目录项指针。

  • p4d_alloc() 通过内存描述符、pgd 项和虚拟地址,得到 p4d 项指针,具体获取方法下文叙述。

  • pud_alloc() 通过内存描述符、p4d 项和虚拟地址,得到 pud 项指针,具体获取方法下文叙述。

  • pmd_alloc() 通过内存描述符、pud 项和虚拟地址,得到 pmd 项指针,具体获取方法下文叙述。

  • 将存有上述结果的 vmf 的地址传给 handle_pte_fault() 进行页表项的处理。

需要说明的是,不同处理器架构,以及同处理器架构选择不同的内存方案,会有不同的分页层次。如 RISC-V 中 Rv39 分页方案就没有 p4dpud,内核会将这两个目录设置成只有一项,该项直接指向下一级的目录,做到一套代码适应不同的分页层次。

3.2 获取各级目录项的处理方式

上文中获取各级目录项的指针代码实现如下:

  1. // include/linux/mm.h :2233
  2. static inline p4d_t *p4d_alloc(struct mm_struct *mm, pgd_t *pgd,
  3. unsigned long address)
  4. {
  5. return (unlikely(pgd_none(*pgd)) && __p4d_alloc(mm, pgd, address)) ?
  6. NULL : p4d_offset(pgd, address);
  7. }
  8. static inline pud_t *pud_alloc(struct mm_struct *mm, p4d_t *p4d,
  9. unsigned long address)
  10. {
  11. return (unlikely(p4d_none(*p4d)) && __pud_alloc(mm, p4d, address)) ?
  12. NULL : pud_offset(p4d, address);
  13. }
  14. // mm/memory.c:4675
  15. static inline pmd_t *pmd_alloc(struct mm_struct *mm, pud_t *pud, unsigned long address)
  16. {
  17. return (unlikely(pud_none(*pud)) && __pmd_alloc(mm, pud, address))?
  18. NULL: pmd_offset(pud, address);
  19. }

比较上述代码可见不同层获取目录项的实现完全类似,故仅对 pmd_alloc() 进行说明,读者可以自行看其它源码举一反三。

pmd_alloc() 函数通过内存描述符 mmpud 项和虚拟地址,得到 pmd 项指针:

  • 检查 pud 项的值是否为 0,不为 0pud_none 宏的值为 0,那么 ? 之前的结果即确定为假,编译器编出的程序会直接跳过下一步,执行第三步 pmd_offset()

  • pud 项为 0,说明还没有对应的 pmd 目录,则执行 __pmd_alloc() 函数(下面会详细说明此函数),为 address 申请一块物理地址,作为 pmd 目录。

  • 调用 pmd_offset() 函数,该函数比较简单,直接根据虚拟地址 address 对应在在目录中的偏移,从 pmd 目录中得到该偏移对应的指针。

__pmd_alloc() 函数代码如下:

  1. // mm/memory.c:4874
  2. int __pmd_alloc(struct mm_struct *mm, pud_t *pud, unsigned long address)
  3. {
  4. spinlock_t *ptl;
  5. pmd_t *new = pmd_alloc_one(mm, address);
  6. if (!new)
  7. return -ENOMEM;
  8. ptl = pud_lock(mm, pud);
  9. if (!pud_present(*pud)) {
  10. mm_inc_nr_pmds(mm);
  11. smp_wmb(); /* See comment in pmd_install() */
  12. pud_populate(mm, pud, new);
  13. } else { /* Another has populated it */
  14. pmd_free(mm, new);
  15. }
  16. spin_unlock(ptl);
  17. return 0;
  18. }

该函数首先调用 pmd_alloc_one() 申请一块物理地址作为 pmd 目录,再调用 pud_populate() 将刚申请的 pmd 目录的物理地址写到 pud 项中。 此函数中 !pud_present(*pud) 的判断,后续会专门一节写内核内存管理中类似的处理,目前可不关注。

pmd_alloc_one() 函数代码如下,其通过 alloc_pages() 接口获取一页,并最终返回该页对应的内核态虚拟地址,该虚拟地址会传给 pud_populate() 函数:

  1. // include/asm-generic: 119
  2. static inline pmd_t *pmd_alloc_one(struct mm_struct *mm, unsigned long addr)
  3. {
  4. struct page *page;
  5. gfp_t gfp = GFP_PGTABLE_USER;
  6. if (mm == &init_mm)
  7. gfp = GFP_PGTABLE_KERNEL;
  8. page = alloc_pages(gfp, 0);
  9. if (!page)
  10. return NULL;
  11. if (!pgtable_pmd_page_ctor(page)) {
  12. __free_pages(page, 0);
  13. return NULL;
  14. }
  15. return (pmd_t *)page_address(page);
  16. }

pud_populate() 函数在 RISC-V 架构下定义如下,直接获取刚刚申请的 pmd 虚拟地址对应的物理地址,然后调用 __pud 宏,再将物理地址填入 pud 项中。

  1. // arch/riscv/include/asm/pgalloc.h:35
  2. static inline void pud_populate(struct mm_struct *mm, pud_t *pud, pmd_t *pmd)
  3. {
  4. unsigned long pfn = virt_to_pfn(pmd);
  5. set_pud(pud, __pud((pfn << _PAGE_PFN_SHIFT) | _PAGE_TABLE));
  6. }

__handle_mm_fault() 函数的最后,会调用 handle_pte_fault() 进行 Legal 地址缺页异常的页表项处理流程,根据页表项的情况会有不同的分支,这包括经常听到的写时复制机制,请求调页机制等。在下一节将进行详细分析。

至此,完成 __handle_mm_fault() 函数流程的分析。

4 总结

本文详细分析了 do_page_fault() 函数在 RISC-V 架构下的实现以及用户态合法地址缺页异常处理中 handle_mm_fault() 函数的实现细节。涉及分支较多,可对照源码仔细阅读。

5 参考资料

  • [1] DANILE.PBOVET、MARCO CESATI 著,陈莉君、张琼声、张宏伟 译。深入理解 Linux 内核 [M].北京:中国电力出版社,2007
  • [2] 陈华才。用”芯”探核 基于龙芯的 Linux 内核探索解析 [M].北京:中国工信出版社/人民邮电出版社,2020.


Read Album:

Read Related:

Read Latest: