gate.h 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. /**
  2. * @file gate.h
  3. * @author longjin
  4. * @brief 门定义
  5. * @date 2022-01-24
  6. *
  7. */
  8. #pragma once
  9. //描述符表的结构体
  10. struct desc_struct
  11. {
  12. unsigned char x[8];
  13. };
  14. //门的结构体
  15. struct gate_struct
  16. {
  17. unsigned char x[16];
  18. };
  19. extern struct desc_struct GDT_Table[]; //GDT_Table是head.S中的GDT_Table
  20. extern struct gate_struct IDT_Table[]; //IDT_Table是head.S中的IDT_Table
  21. extern unsigned int TSS64_Table[26];
  22. /**
  23. * @brief 初始化中段描述符表内的门描述符(每个16B)
  24. * @param gate_selector_addr IDT表项的地址
  25. * @param attr P、DPL、TYPE的属性
  26. * @param ist 中断栈表号
  27. * @param code_addr 中断服务程序的地址
  28. */
  29. // todo:在系统异常处理主功能完成后,将这段代码用C来写一遍。这段汇编实在是太晦涩难懂了,我看了半个钟才看明白。
  30. /*
  31. #define _set_gate(gate_selector_addr, attr, ist, code_addr) \
  32. do{ \
  33. unsigned long __d0, __d1; \
  34. __asm__ __volatile__ ( "movw %%dx, %%ax \n\t" \
  35. "andq $0x7, %%rcx \n\t" // 清空rcx中除了2:0以外的所有位 此前ist的值已经被赋给了rcx \
  36. "addq %4, %%rcx \n\t" // 将P,DPL, Type的值加到rcx中 \
  37. "shlq $32, %%rcx \n\t" \
  38. "addq %%rcx, %%rax \n\t" // 设置ist \
  39. "xorq %%rcx, %%rcx \n\t" // 清空rcx \
  40. "movl %%edx, %%ecx \n\t" \
  41. "shrq $16, %%ecx \n\t" \
  42. "shlq $48, %%rcx \n\t" // 左移到低8B中表示段内偏移的[31:16]处 \
  43. "addq %%rcx, %%rax \n\t" // 设置段内偏移[31:16] \
  44. "movq %%rax, %0 \n\t" // 输出到门选择子的低8B \
  45. "shrq $32, %%rdx \n\t" \
  46. "movq %%rdx, %1 \n\t" // 输出到门选择子的高8B \
  47. :"=m"(*((unsigned long *)(gate_selector_addr))) , \
  48. "=m"(*(1 + (unsigned long *)(gate_selector_addr))),"=&a"(__d0),"=&d"(__d1) \
  49. :"i"(attr << 8), \
  50. "3"((unsigned long *)(code_addr)),"2"(0x8 << 16),"c"(ist) \
  51. :"memory" \
  52. ); \
  53. }while(0)
  54. */
  55. //由于带上注释就编译不过,因此复制一份到这里
  56. #define _set_gate(gate_selector_addr, attr, ist, code_addr) \
  57. do \
  58. { \
  59. unsigned long __d0, __d1; \
  60. __asm__ __volatile__("movw %%dx, %%ax \n\t" \
  61. "andq $0x7, %%rcx \n\t" \
  62. "addq %4, %%rcx \n\t" \
  63. "shlq $32, %%rcx \n\t" \
  64. "addq %%rcx, %%rax \n\t" \
  65. "xorq %%rcx, %%rcx \n\t" \
  66. "movl %%edx, %%ecx \n\t" \
  67. "shrq $16, %%rcx \n\t" \
  68. "shlq $48, %%rcx \n\t" \
  69. "addq %%rcx, %%rax \n\t" \
  70. "movq %%rax, %0 \n\t" \
  71. "shrq $32, %%rdx \n\t" \
  72. "movq %%rdx, %1 \n\t" \
  73. : "=m"(*((unsigned long *)(gate_selector_addr))), \
  74. "=m"(*(1 + (unsigned long *)(gate_selector_addr))), "=&a"(__d0), "=&d"(__d1) \
  75. : "i"(attr << 8), \
  76. "3"((unsigned long *)(code_addr)), "2"(0x8 << 16), "c"(ist) \
  77. : "memory"); \
  78. } while (0)
  79. /**
  80. * @brief 加载任务状态段寄存器
  81. * @param n TSS基地址在GDT中的第几项
  82. * 左移3位的原因是GDT每项占8字节
  83. */
  84. #define load_TR(n) \
  85. do \
  86. { \
  87. __asm__ __volatile__("ltr %%ax" ::"a"(n << 3)); \
  88. } while (0)
  89. /**
  90. * @brief 设置中断门
  91. *
  92. * @param n 中断号
  93. * @param ist ist
  94. * @param addr 服务程序的地址
  95. */
  96. void set_intr_gate(unsigned int n, unsigned char ist, void *addr)
  97. {
  98. _set_gate(IDT_Table + n, 0x8E, ist, addr); // p=1,DPL=0, type=E
  99. }
  100. /**
  101. * @brief 设置64位,DPL=0的陷阱门
  102. *
  103. * @param n 中断号
  104. * @param ist ist
  105. * @param addr 服务程序的地址
  106. */
  107. void set_trap_gate(unsigned int n, unsigned char ist, void *addr)
  108. {
  109. _set_gate(IDT_Table + n, 0x8F, ist, addr); // p=1,DPL=0, type=F
  110. }
  111. /**
  112. * @brief 设置64位,DPL=3的陷阱门
  113. *
  114. * @param n 中断号
  115. * @param ist ist
  116. * @param addr 服务程序的地址
  117. */
  118. void set_system_trap_gate(unsigned int n, unsigned char ist, void *addr)
  119. {
  120. _set_gate(IDT_Table + n, 0xEF, ist, addr); // p=1,DPL=3, type=F
  121. }
  122. /**
  123. * @brief 初始化TSS表的内容
  124. *
  125. */
  126. void set_TSS64(ul rsp0, ul rsp1, ul rsp2, ul ist1, ul ist2, ul ist3, ul ist4, ul ist5, ul ist6, ul ist7)
  127. {
  128. *(ul *)(TSS64_Table + 1) = rsp0;
  129. *(ul *)(TSS64_Table + 3) = rsp1;
  130. *(ul *)(TSS64_Table + 5) = rsp2;
  131. *(ul *)(TSS64_Table + 9) = ist1;
  132. *(ul *)(TSS64_Table + 11) = ist2;
  133. *(ul *)(TSS64_Table + 13) = ist3;
  134. *(ul *)(TSS64_Table + 15) = ist4;
  135. *(ul *)(TSS64_Table + 17) = ist5;
  136. *(ul *)(TSS64_Table + 19) = ist6;
  137. *(ul *)(TSS64_Table + 21) = ist7;
  138. }