svm.h 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353
  1. #ifndef __SVM_H
  2. #define __SVM_H
  3. enum {
  4. INTERCEPT_INTR,
  5. INTERCEPT_NMI,
  6. INTERCEPT_SMI,
  7. INTERCEPT_INIT,
  8. INTERCEPT_VINTR,
  9. INTERCEPT_SELECTIVE_CR0,
  10. INTERCEPT_STORE_IDTR,
  11. INTERCEPT_STORE_GDTR,
  12. INTERCEPT_STORE_LDTR,
  13. INTERCEPT_STORE_TR,
  14. INTERCEPT_LOAD_IDTR,
  15. INTERCEPT_LOAD_GDTR,
  16. INTERCEPT_LOAD_LDTR,
  17. INTERCEPT_LOAD_TR,
  18. INTERCEPT_RDTSC,
  19. INTERCEPT_RDPMC,
  20. INTERCEPT_PUSHF,
  21. INTERCEPT_POPF,
  22. INTERCEPT_CPUID,
  23. INTERCEPT_RSM,
  24. INTERCEPT_IRET,
  25. INTERCEPT_INTn,
  26. INTERCEPT_INVD,
  27. INTERCEPT_PAUSE,
  28. INTERCEPT_HLT,
  29. INTERCEPT_INVLPG,
  30. INTERCEPT_INVLPGA,
  31. INTERCEPT_IOIO_PROT,
  32. INTERCEPT_MSR_PROT,
  33. INTERCEPT_TASK_SWITCH,
  34. INTERCEPT_FERR_FREEZE,
  35. INTERCEPT_SHUTDOWN,
  36. INTERCEPT_VMRUN,
  37. INTERCEPT_VMMCALL,
  38. INTERCEPT_VMLOAD,
  39. INTERCEPT_VMSAVE,
  40. INTERCEPT_STGI,
  41. INTERCEPT_CLGI,
  42. INTERCEPT_SKINIT,
  43. INTERCEPT_RDTSCP,
  44. INTERCEPT_ICEBP,
  45. INTERCEPT_WBINVD,
  46. INTERCEPT_MONITOR,
  47. INTERCEPT_MWAIT,
  48. INTERCEPT_MWAIT_COND,
  49. INTERCEPT_XSETBV,
  50. };
  51. struct __attribute__ ((__packed__)) vmcb_control_area {
  52. u32 intercept_cr;
  53. u32 intercept_dr;
  54. u32 intercept_exceptions;
  55. u64 intercept;
  56. u8 reserved_1[42];
  57. u16 pause_filter_count;
  58. u64 iopm_base_pa;
  59. u64 msrpm_base_pa;
  60. u64 tsc_offset;
  61. u32 asid;
  62. u8 tlb_ctl;
  63. u8 reserved_2[3];
  64. u32 int_ctl;
  65. u32 int_vector;
  66. u32 int_state;
  67. u8 reserved_3[4];
  68. u32 exit_code;
  69. u32 exit_code_hi;
  70. u64 exit_info_1;
  71. u64 exit_info_2;
  72. u32 exit_int_info;
  73. u32 exit_int_info_err;
  74. u64 nested_ctl;
  75. u8 reserved_4[16];
  76. u32 event_inj;
  77. u32 event_inj_err;
  78. u64 nested_cr3;
  79. u64 lbr_ctl;
  80. u32 clean;
  81. u32 reserved_5;
  82. u64 next_rip;
  83. u8 insn_len;
  84. u8 insn_bytes[15];
  85. u8 reserved_6[800];
  86. };
  87. #define TLB_CONTROL_DO_NOTHING 0
  88. #define TLB_CONTROL_FLUSH_ALL_ASID 1
  89. #define TLB_CONTROL_FLUSH_ASID 3
  90. #define TLB_CONTROL_FLUSH_ASID_LOCAL 7
  91. #define V_TPR_MASK 0x0f
  92. #define V_IRQ_SHIFT 8
  93. #define V_IRQ_MASK (1 << V_IRQ_SHIFT)
  94. #define V_INTR_PRIO_SHIFT 16
  95. #define V_INTR_PRIO_MASK (0x0f << V_INTR_PRIO_SHIFT)
  96. #define V_IGN_TPR_SHIFT 20
  97. #define V_IGN_TPR_MASK (1 << V_IGN_TPR_SHIFT)
  98. #define V_INTR_MASKING_SHIFT 24
  99. #define V_INTR_MASKING_MASK (1 << V_INTR_MASKING_SHIFT)
  100. #define SVM_INTERRUPT_SHADOW_MASK 1
  101. #define SVM_IOIO_STR_SHIFT 2
  102. #define SVM_IOIO_REP_SHIFT 3
  103. #define SVM_IOIO_SIZE_SHIFT 4
  104. #define SVM_IOIO_ASIZE_SHIFT 7
  105. #define SVM_IOIO_TYPE_MASK 1
  106. #define SVM_IOIO_STR_MASK (1 << SVM_IOIO_STR_SHIFT)
  107. #define SVM_IOIO_REP_MASK (1 << SVM_IOIO_REP_SHIFT)
  108. #define SVM_IOIO_SIZE_MASK (7 << SVM_IOIO_SIZE_SHIFT)
  109. #define SVM_IOIO_ASIZE_MASK (7 << SVM_IOIO_ASIZE_SHIFT)
  110. #define SVM_VM_CR_VALID_MASK 0x001fULL
  111. #define SVM_VM_CR_SVM_LOCK_MASK 0x0008ULL
  112. #define SVM_VM_CR_SVM_DIS_MASK 0x0010ULL
  113. struct __attribute__ ((__packed__)) vmcb_seg {
  114. u16 selector;
  115. u16 attrib;
  116. u32 limit;
  117. u64 base;
  118. };
  119. struct __attribute__ ((__packed__)) vmcb_save_area {
  120. struct vmcb_seg es;
  121. struct vmcb_seg cs;
  122. struct vmcb_seg ss;
  123. struct vmcb_seg ds;
  124. struct vmcb_seg fs;
  125. struct vmcb_seg gs;
  126. struct vmcb_seg gdtr;
  127. struct vmcb_seg ldtr;
  128. struct vmcb_seg idtr;
  129. struct vmcb_seg tr;
  130. u8 reserved_1[43];
  131. u8 cpl;
  132. u8 reserved_2[4];
  133. u64 efer;
  134. u8 reserved_3[112];
  135. u64 cr4;
  136. u64 cr3;
  137. u64 cr0;
  138. u64 dr7;
  139. u64 dr6;
  140. u64 rflags;
  141. u64 rip;
  142. u8 reserved_4[88];
  143. u64 rsp;
  144. u8 reserved_5[24];
  145. u64 rax;
  146. u64 star;
  147. u64 lstar;
  148. u64 cstar;
  149. u64 sfmask;
  150. u64 kernel_gs_base;
  151. u64 sysenter_cs;
  152. u64 sysenter_esp;
  153. u64 sysenter_eip;
  154. u64 cr2;
  155. u8 reserved_6[32];
  156. u64 g_pat;
  157. u64 dbgctl;
  158. u64 br_from;
  159. u64 br_to;
  160. u64 last_excp_from;
  161. u64 last_excp_to;
  162. };
  163. struct __attribute__ ((__packed__)) vmcb {
  164. struct vmcb_control_area control;
  165. struct vmcb_save_area save;
  166. };
  167. #define SVM_CPUID_FEATURE_SHIFT 2
  168. #define SVM_CPUID_FUNC 0x8000000a
  169. #define SVM_VM_CR_SVM_DISABLE 4
  170. #define SVM_SELECTOR_S_SHIFT 4
  171. #define SVM_SELECTOR_DPL_SHIFT 5
  172. #define SVM_SELECTOR_P_SHIFT 7
  173. #define SVM_SELECTOR_AVL_SHIFT 8
  174. #define SVM_SELECTOR_L_SHIFT 9
  175. #define SVM_SELECTOR_DB_SHIFT 10
  176. #define SVM_SELECTOR_G_SHIFT 11
  177. #define SVM_SELECTOR_TYPE_MASK (0xf)
  178. #define SVM_SELECTOR_S_MASK (1 << SVM_SELECTOR_S_SHIFT)
  179. #define SVM_SELECTOR_DPL_MASK (3 << SVM_SELECTOR_DPL_SHIFT)
  180. #define SVM_SELECTOR_P_MASK (1 << SVM_SELECTOR_P_SHIFT)
  181. #define SVM_SELECTOR_AVL_MASK (1 << SVM_SELECTOR_AVL_SHIFT)
  182. #define SVM_SELECTOR_L_MASK (1 << SVM_SELECTOR_L_SHIFT)
  183. #define SVM_SELECTOR_DB_MASK (1 << SVM_SELECTOR_DB_SHIFT)
  184. #define SVM_SELECTOR_G_MASK (1 << SVM_SELECTOR_G_SHIFT)
  185. #define SVM_SELECTOR_WRITE_MASK (1 << 1)
  186. #define SVM_SELECTOR_READ_MASK SVM_SELECTOR_WRITE_MASK
  187. #define SVM_SELECTOR_CODE_MASK (1 << 3)
  188. #define INTERCEPT_CR0_READ 0
  189. #define INTERCEPT_CR3_READ 3
  190. #define INTERCEPT_CR4_READ 4
  191. #define INTERCEPT_CR8_READ 8
  192. #define INTERCEPT_CR0_WRITE (16 + 0)
  193. #define INTERCEPT_CR3_WRITE (16 + 3)
  194. #define INTERCEPT_CR4_WRITE (16 + 4)
  195. #define INTERCEPT_CR8_WRITE (16 + 8)
  196. #define INTERCEPT_DR0_READ 0
  197. #define INTERCEPT_DR1_READ 1
  198. #define INTERCEPT_DR2_READ 2
  199. #define INTERCEPT_DR3_READ 3
  200. #define INTERCEPT_DR4_READ 4
  201. #define INTERCEPT_DR5_READ 5
  202. #define INTERCEPT_DR6_READ 6
  203. #define INTERCEPT_DR7_READ 7
  204. #define INTERCEPT_DR0_WRITE (16 + 0)
  205. #define INTERCEPT_DR1_WRITE (16 + 1)
  206. #define INTERCEPT_DR2_WRITE (16 + 2)
  207. #define INTERCEPT_DR3_WRITE (16 + 3)
  208. #define INTERCEPT_DR4_WRITE (16 + 4)
  209. #define INTERCEPT_DR5_WRITE (16 + 5)
  210. #define INTERCEPT_DR6_WRITE (16 + 6)
  211. #define INTERCEPT_DR7_WRITE (16 + 7)
  212. #define SVM_EVTINJ_VEC_MASK 0xff
  213. #define SVM_EVTINJ_TYPE_SHIFT 8
  214. #define SVM_EVTINJ_TYPE_MASK (7 << SVM_EVTINJ_TYPE_SHIFT)
  215. #define SVM_EVTINJ_TYPE_INTR (0 << SVM_EVTINJ_TYPE_SHIFT)
  216. #define SVM_EVTINJ_TYPE_NMI (2 << SVM_EVTINJ_TYPE_SHIFT)
  217. #define SVM_EVTINJ_TYPE_EXEPT (3 << SVM_EVTINJ_TYPE_SHIFT)
  218. #define SVM_EVTINJ_TYPE_SOFT (4 << SVM_EVTINJ_TYPE_SHIFT)
  219. #define SVM_EVTINJ_VALID (1 << 31)
  220. #define SVM_EVTINJ_VALID_ERR (1 << 11)
  221. #define SVM_EXITINTINFO_VEC_MASK SVM_EVTINJ_VEC_MASK
  222. #define SVM_EXITINTINFO_TYPE_MASK SVM_EVTINJ_TYPE_MASK
  223. #define SVM_EXITINTINFO_TYPE_INTR SVM_EVTINJ_TYPE_INTR
  224. #define SVM_EXITINTINFO_TYPE_NMI SVM_EVTINJ_TYPE_NMI
  225. #define SVM_EXITINTINFO_TYPE_EXEPT SVM_EVTINJ_TYPE_EXEPT
  226. #define SVM_EXITINTINFO_TYPE_SOFT SVM_EVTINJ_TYPE_SOFT
  227. #define SVM_EXITINTINFO_VALID SVM_EVTINJ_VALID
  228. #define SVM_EXITINTINFO_VALID_ERR SVM_EVTINJ_VALID_ERR
  229. #define SVM_EXITINFOSHIFT_TS_REASON_IRET 36
  230. #define SVM_EXITINFOSHIFT_TS_REASON_JMP 38
  231. #define SVM_EXITINFOSHIFT_TS_HAS_ERROR_CODE 44
  232. #define SVM_EXITINFO_REG_MASK 0x0F
  233. #define SVM_EXIT_READ_CR0 0x000
  234. #define SVM_EXIT_READ_CR3 0x003
  235. #define SVM_EXIT_READ_CR4 0x004
  236. #define SVM_EXIT_READ_CR8 0x008
  237. #define SVM_EXIT_WRITE_CR0 0x010
  238. #define SVM_EXIT_WRITE_CR3 0x013
  239. #define SVM_EXIT_WRITE_CR4 0x014
  240. #define SVM_EXIT_WRITE_CR8 0x018
  241. #define SVM_EXIT_READ_DR0 0x020
  242. #define SVM_EXIT_READ_DR1 0x021
  243. #define SVM_EXIT_READ_DR2 0x022
  244. #define SVM_EXIT_READ_DR3 0x023
  245. #define SVM_EXIT_READ_DR4 0x024
  246. #define SVM_EXIT_READ_DR5 0x025
  247. #define SVM_EXIT_READ_DR6 0x026
  248. #define SVM_EXIT_READ_DR7 0x027
  249. #define SVM_EXIT_WRITE_DR0 0x030
  250. #define SVM_EXIT_WRITE_DR1 0x031
  251. #define SVM_EXIT_WRITE_DR2 0x032
  252. #define SVM_EXIT_WRITE_DR3 0x033
  253. #define SVM_EXIT_WRITE_DR4 0x034
  254. #define SVM_EXIT_WRITE_DR5 0x035
  255. #define SVM_EXIT_WRITE_DR6 0x036
  256. #define SVM_EXIT_WRITE_DR7 0x037
  257. #define SVM_EXIT_EXCP_BASE 0x040
  258. #define SVM_EXIT_INTR 0x060
  259. #define SVM_EXIT_NMI 0x061
  260. #define SVM_EXIT_SMI 0x062
  261. #define SVM_EXIT_INIT 0x063
  262. #define SVM_EXIT_VINTR 0x064
  263. #define SVM_EXIT_CR0_SEL_WRITE 0x065
  264. #define SVM_EXIT_IDTR_READ 0x066
  265. #define SVM_EXIT_GDTR_READ 0x067
  266. #define SVM_EXIT_LDTR_READ 0x068
  267. #define SVM_EXIT_TR_READ 0x069
  268. #define SVM_EXIT_IDTR_WRITE 0x06a
  269. #define SVM_EXIT_GDTR_WRITE 0x06b
  270. #define SVM_EXIT_LDTR_WRITE 0x06c
  271. #define SVM_EXIT_TR_WRITE 0x06d
  272. #define SVM_EXIT_RDTSC 0x06e
  273. #define SVM_EXIT_RDPMC 0x06f
  274. #define SVM_EXIT_PUSHF 0x070
  275. #define SVM_EXIT_POPF 0x071
  276. #define SVM_EXIT_CPUID 0x072
  277. #define SVM_EXIT_RSM 0x073
  278. #define SVM_EXIT_IRET 0x074
  279. #define SVM_EXIT_SWINT 0x075
  280. #define SVM_EXIT_INVD 0x076
  281. #define SVM_EXIT_PAUSE 0x077
  282. #define SVM_EXIT_HLT 0x078
  283. #define SVM_EXIT_INVLPG 0x079
  284. #define SVM_EXIT_INVLPGA 0x07a
  285. #define SVM_EXIT_IOIO 0x07b
  286. #define SVM_EXIT_MSR 0x07c
  287. #define SVM_EXIT_TASK_SWITCH 0x07d
  288. #define SVM_EXIT_FERR_FREEZE 0x07e
  289. #define SVM_EXIT_SHUTDOWN 0x07f
  290. #define SVM_EXIT_VMRUN 0x080
  291. #define SVM_EXIT_VMMCALL 0x081
  292. #define SVM_EXIT_VMLOAD 0x082
  293. #define SVM_EXIT_VMSAVE 0x083
  294. #define SVM_EXIT_STGI 0x084
  295. #define SVM_EXIT_CLGI 0x085
  296. #define SVM_EXIT_SKINIT 0x086
  297. #define SVM_EXIT_RDTSCP 0x087
  298. #define SVM_EXIT_ICEBP 0x088
  299. #define SVM_EXIT_WBINVD 0x089
  300. #define SVM_EXIT_MONITOR 0x08a
  301. #define SVM_EXIT_MWAIT 0x08b
  302. #define SVM_EXIT_MWAIT_COND 0x08c
  303. #define SVM_EXIT_XSETBV 0x08d
  304. #define SVM_EXIT_NPF 0x400
  305. #define SVM_EXIT_ERR -1
  306. #define SVM_CR0_SELECTIVE_MASK (X86_CR0_TS | X86_CR0_MP)
  307. #define SVM_VMLOAD ".byte 0x0f, 0x01, 0xda"
  308. #define SVM_VMRUN ".byte 0x0f, 0x01, 0xd8"
  309. #define SVM_VMSAVE ".byte 0x0f, 0x01, 0xdb"
  310. #define SVM_CLGI ".byte 0x0f, 0x01, 0xdd"
  311. #define SVM_STGI ".byte 0x0f, 0x01, 0xdc"
  312. #define SVM_INVLPGA ".byte 0x0f, 0x01, 0xdf"
  313. #endif