gate.h 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. /**
  2. * @file gate.h
  3. * @author longjin
  4. * @brief 门定义
  5. * @date 2022-01-24
  6. *
  7. */
  8. #ifndef __GATE_H__
  9. #define __GATE_H__
  10. #include "../common/kprint.h"
  11. //描述符表的结构体
  12. struct desc_struct
  13. {
  14. unsigned char x[8];
  15. };
  16. //门的结构体
  17. struct gate_struct
  18. {
  19. unsigned char x[16];
  20. };
  21. extern struct desc_struct GDT_Table[]; // GDT_Table是head.S中的GDT_Table
  22. extern struct gate_struct IDT_Table[]; // IDT_Table是head.S中的IDT_Table
  23. extern unsigned int TSS64_Table[26];
  24. /**
  25. * @brief 初始化中段描述符表内的门描述符(每个16B)
  26. * @param gate_selector_addr IDT表项的地址
  27. * @param attr P、DPL、TYPE的属性
  28. * @param ist 中断栈表号
  29. * @param code_addr 指向中断服务程序的指针的地址
  30. */
  31. void set_gate(ul *gate_selector_addr, ul attr, unsigned char ist, ul *code_addr)
  32. {
  33. ul __d0 = 0, __d1 = 0;
  34. ul tmp_code_addr = *code_addr;
  35. __d0 = attr << 40; //设置P、DPL、Type
  36. __d0 |= ((ul)(ist) << 32); // 设置ist
  37. __d0 |= 8 << 16; //设置段选择子为0x1000
  38. __d0 |= (0xffff & tmp_code_addr); //设置段内偏移的[15:00]
  39. tmp_code_addr >>= 16;
  40. __d0 |= (0xffff & tmp_code_addr) << 48; // 设置段内偏移[31:16]
  41. tmp_code_addr >>= 16;
  42. __d1 = (0xffffffff & tmp_code_addr); //设置段内偏移[63:32]
  43. *gate_selector_addr = __d0;
  44. *(gate_selector_addr + 1) = __d1;
  45. }
  46. #define _set_gate(gate_selector_addr, attr, ist, code_addr) \
  47. do \
  48. { \
  49. unsigned long __d0, __d1; \
  50. __asm__ __volatile__("movw %%dx, %%ax \n\t" \
  51. "andq $0x7, %%rcx \n\t" \
  52. "addq %4, %%rcx \n\t" \
  53. "shlq $32, %%rcx \n\t" \
  54. "addq %%rcx, %%rax \n\t" \
  55. "xorq %%rcx, %%rcx \n\t" \
  56. "movl %%edx, %%ecx \n\t" \
  57. "shrq $16, %%rcx \n\t" \
  58. "shlq $48, %%rcx \n\t" \
  59. "addq %%rcx, %%rax \n\t" \
  60. "movq %%rax, %0 \n\t" \
  61. "shrq $32, %%rdx \n\t" \
  62. "movq %%rdx, %1 \n\t" \
  63. : "=m"(*((unsigned long *)(gate_selector_addr))), \
  64. "=m"(*(1 + (unsigned long *)(gate_selector_addr))), "=&a"(__d0), "=&d"(__d1) \
  65. : "i"(attr << 8), \
  66. "3"((unsigned long *)(code_addr)), "2"(0x8 << 16), "c"(ist) \
  67. : "memory"); \
  68. } while (0)
  69. void set_tss_descriptor(unsigned int n, void *addr)
  70. {
  71. unsigned long limit = 103;
  72. *(unsigned long *)(&GDT_Table[n]) = (limit & 0xffff) | (((unsigned long)addr & 0xffff) << 16) | ((((unsigned long)addr >> 16) & 0xff) << 32) | ((unsigned long)0x89 << 40) | ((limit >> 16 & 0xf) << 48) | (((unsigned long)addr >> 24 & 0xff) << 56); /////89 is attribute
  73. *(unsigned long *)(&GDT_Table[n + 1]) = (((unsigned long)addr >> 32) & 0xffffffff) | 0;
  74. }
  75. /**
  76. * @brief 加载任务状态段寄存器
  77. * @param n TSS基地址在GDT中的第几项
  78. * 左移3位的原因是GDT每项占8字节
  79. */
  80. #define load_TR(n) \
  81. do \
  82. { \
  83. __asm__ __volatile__("ltr %%ax" ::"a"((n) << 3)); \
  84. } while (0)
  85. /**
  86. * @brief 设置中断门
  87. *
  88. * @param n 中断号
  89. * @param ist ist
  90. * @param addr 服务程序的地址
  91. */
  92. void set_intr_gate(unsigned int n, unsigned char ist, void *addr)
  93. {
  94. _set_gate((IDT_Table + n), 0x8E, ist, addr); // p=1,DPL=0, type=E
  95. // set_gate((ul *)(IDT_Table + n), 0x8E, ist, (ul *)(addr)); // p=1,DPL=0, type=E
  96. }
  97. /**
  98. * @brief 设置64位,DPL=0的陷阱门
  99. *
  100. * @param n 中断号
  101. * @param ist ist
  102. * @param addr 服务程序的地址
  103. */
  104. void set_trap_gate(unsigned int n, unsigned char ist, void *addr)
  105. {
  106. // kdebug("addr=%#018lx", (ul)(addr));
  107. // set_gate((ul *)(IDT_Table + n), 0x8F, ist, (ul *)(addr)); // p=1,DPL=0, type=F
  108. _set_gate((IDT_Table + n), 0x8F, ist, addr); // p=1,DPL=0, type=F
  109. }
  110. /**
  111. * @brief 设置64位,DPL=3的陷阱门
  112. *
  113. * @param n 中断号
  114. * @param ist ist
  115. * @param addr 服务程序的地址
  116. */
  117. void set_system_trap_gate(unsigned int n, unsigned char ist, void *addr)
  118. {
  119. // kdebug("addr=%#018lx", (ul)(addr));
  120. // set_gate((ul *)(IDT_Table + n), 0xEF, ist, (ul *)(addr)); // p=1,DPL=3, type=F
  121. _set_gate((IDT_Table + n), 0xEF, ist, addr); // p=1,DPL=3, type=F
  122. }
  123. /**
  124. * @brief 初始化TSS表的内容
  125. *
  126. */
  127. void set_tss64(unsigned int * Table,unsigned long rsp0,unsigned long rsp1,unsigned long rsp2,unsigned long ist1,unsigned long ist2,unsigned long ist3,
  128. unsigned long ist4,unsigned long ist5,unsigned long ist6,unsigned long ist7)
  129. {
  130. *(unsigned long *)(Table+1) = rsp0;
  131. *(unsigned long *)(Table+3) = rsp1;
  132. *(unsigned long *)(Table+5) = rsp2;
  133. *(unsigned long *)(Table+9) = ist1;
  134. *(unsigned long *)(Table+11) = ist2;
  135. *(unsigned long *)(Table+13) = ist3;
  136. *(unsigned long *)(Table+15) = ist4;
  137. *(unsigned long *)(Table+17) = ist5;
  138. *(unsigned long *)(Table+19) = ist6;
  139. *(unsigned long *)(Table+21) = ist7;
  140. }
  141. #endif