utils.c 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. #include "internal.h"
  2. extern uint64_t mm_total_2M_pages;
  3. /**
  4. * @brief 获取指定虚拟地址处映射的物理地址
  5. *
  6. * @param mm 内存空间分布结构体
  7. * @param vaddr 虚拟地址
  8. * @return uint64_t 已映射的物理地址
  9. */
  10. uint64_t __mm_get_paddr(struct mm_struct *mm, uint64_t vaddr)
  11. {
  12. ul *tmp;
  13. tmp = phys_2_virt((ul *)(((ul)mm->pgd) & (~0xfffUL)) + ((vaddr >> PAGE_GDT_SHIFT) & 0x1ff));
  14. // pml4页表项为0
  15. if (*tmp == 0)
  16. return 0;
  17. tmp = phys_2_virt((ul *)(*tmp & (~0xfffUL)) + ((vaddr >> PAGE_1G_SHIFT) & 0x1ff));
  18. // pdpt页表项为0
  19. if (*tmp == 0)
  20. return 0;
  21. // 读取pdt页表项
  22. tmp = phys_2_virt(((ul *)(*tmp & (~0xfffUL)) + (((ul)(vaddr) >> PAGE_2M_SHIFT) & 0x1ff)));
  23. // pde页表项为0
  24. if (*tmp == 0)
  25. return 0;
  26. if (*tmp & (1 << 7))
  27. {
  28. // 当前为2M物理页
  29. return (*tmp) & (~0x1fffUL);
  30. }
  31. else
  32. {
  33. // 存在4级页表
  34. tmp = phys_2_virt(((ul *)(*tmp & (~0xfffUL)) + (((ul)(vaddr) >> PAGE_4K_SHIFT) & 0x1ff)));
  35. return (*tmp) & (~0x1ffUL);
  36. }
  37. }
  38. /**
  39. * @brief 检测指定地址是否已经被映射
  40. *
  41. * @param page_table_phys_addr 页表的物理地址
  42. * @param virt_addr 要检测的地址
  43. * @return true 已经被映射
  44. * @return false
  45. */
  46. bool mm_check_mapped(ul page_table_phys_addr, uint64_t virt_addr)
  47. {
  48. ul *tmp;
  49. tmp = phys_2_virt((ul *)((ul)page_table_phys_addr & (~0xfffUL)) + ((virt_addr >> PAGE_GDT_SHIFT) & 0x1ff));
  50. // pml4页表项为0
  51. if (*tmp == 0)
  52. return 0;
  53. tmp = phys_2_virt((ul *)(*tmp & (~0xfffUL)) + ((virt_addr >> PAGE_1G_SHIFT) & 0x1ff));
  54. // pdpt页表项为0
  55. if (*tmp == 0)
  56. return 0;
  57. // 读取pdt页表项
  58. tmp = phys_2_virt(((ul *)(*tmp & (~0xfffUL)) + (((ul)(virt_addr) >> PAGE_2M_SHIFT) & 0x1ff)));
  59. // pde页表项为0
  60. if (*tmp == 0)
  61. return 0;
  62. if (*tmp & (1 << 7))
  63. {
  64. // 当前为2M物理页
  65. return true;
  66. }
  67. else
  68. {
  69. // 存在4级页表
  70. tmp = phys_2_virt(((ul *)(*tmp & (~0xfffUL)) + (((ul)(virt_addr) >> PAGE_4K_SHIFT) & 0x1ff)));
  71. if (*tmp != 0)
  72. return true;
  73. else
  74. return false;
  75. }
  76. }
  77. /**
  78. * @brief 检测是否为有效的2M页(物理内存页)
  79. *
  80. * @param paddr 物理地址
  81. * @return int8_t 是 -> 1
  82. * 否 -> 0
  83. */
  84. int8_t mm_is_2M_page(uint64_t paddr)
  85. {
  86. if (likely((paddr >> PAGE_2M_SHIFT) < mm_total_2M_pages))
  87. return 1;
  88. else
  89. return 0;
  90. }