| /* |
| * All test cases of nested virtualization should be in this file |
| * |
| * Author : Arthur Chunqi Li <yzt356@gmail.com> |
| */ |
| #include "vmx.h" |
| #include "msr.h" |
| #include "processor.h" |
| #include "vm.h" |
| #include "fwcfg.h" |
| #include "isr.h" |
| #include "desc.h" |
| #include "apic.h" |
| #include "types.h" |
| |
| u64 ia32_pat; |
| u64 ia32_efer; |
| void *io_bitmap_a, *io_bitmap_b; |
| u16 ioport; |
| |
| unsigned long *pml4; |
| u64 eptp; |
| void *data_page1, *data_page2; |
| |
| static inline void vmcall() |
| { |
| asm volatile("vmcall"); |
| } |
| |
| void basic_guest_main() |
| { |
| } |
| |
| int basic_exit_handler() |
| { |
| report("Basic VMX test", 0); |
| print_vmexit_info(); |
| return VMX_TEST_EXIT; |
| } |
| |
| void vmenter_main() |
| { |
| u64 rax; |
| u64 rsp, resume_rsp; |
| |
| report("test vmlaunch", 1); |
| |
| asm volatile( |
| "mov %%rsp, %0\n\t" |
| "mov %3, %%rax\n\t" |
| "vmcall\n\t" |
| "mov %%rax, %1\n\t" |
| "mov %%rsp, %2\n\t" |
| : "=r"(rsp), "=r"(rax), "=r"(resume_rsp) |
| : "g"(0xABCD)); |
| report("test vmresume", (rax == 0xFFFF) && (rsp == resume_rsp)); |
| } |
| |
| int vmenter_exit_handler() |
| { |
| u64 guest_rip; |
| ulong reason; |
| |
| guest_rip = vmcs_read(GUEST_RIP); |
| reason = vmcs_read(EXI_REASON) & 0xff; |
| switch (reason) { |
| case VMX_VMCALL: |
| if (regs.rax != 0xABCD) { |
| report("test vmresume", 0); |
| return VMX_TEST_VMEXIT; |
| } |
| regs.rax = 0xFFFF; |
| vmcs_write(GUEST_RIP, guest_rip + 3); |
| return VMX_TEST_RESUME; |
| default: |
| report("test vmresume", 0); |
| print_vmexit_info(); |
| } |
| return VMX_TEST_VMEXIT; |
| } |
| |
| u32 preempt_scale; |
| volatile unsigned long long tsc_val; |
| volatile u32 preempt_val; |
| u64 saved_rip; |
| |
| int preemption_timer_init() |
| { |
| if (!(ctrl_pin_rev.clr & PIN_PREEMPT)) { |
| printf("\tPreemption timer is not supported\n"); |
| return VMX_TEST_EXIT; |
| } |
| vmcs_write(PIN_CONTROLS, vmcs_read(PIN_CONTROLS) | PIN_PREEMPT); |
| preempt_val = 10000000; |
| vmcs_write(PREEMPT_TIMER_VALUE, preempt_val); |
| preempt_scale = rdmsr(MSR_IA32_VMX_MISC) & 0x1F; |
| |
| if (!(ctrl_exit_rev.clr & EXI_SAVE_PREEMPT)) |
| printf("\tSave preemption value is not supported\n"); |
| |
| return VMX_TEST_START; |
| } |
| |
| void preemption_timer_main() |
| { |
| tsc_val = rdtsc(); |
| if (ctrl_exit_rev.clr & EXI_SAVE_PREEMPT) { |
| vmx_set_test_stage(0); |
| vmcall(); |
| if (vmx_get_test_stage() == 1) |
| vmcall(); |
| } |
| vmx_set_test_stage(1); |
| while (vmx_get_test_stage() == 1) { |
| if (((rdtsc() - tsc_val) >> preempt_scale) |
| > 10 * preempt_val) { |
| vmx_set_test_stage(2); |
| vmcall(); |
| } |
| } |
| tsc_val = rdtsc(); |
| asm volatile ("hlt"); |
| vmcall(); |
| vmx_set_test_stage(5); |
| vmcall(); |
| } |
| |
| int preemption_timer_exit_handler() |
| { |
| bool guest_halted; |
| u64 guest_rip; |
| ulong reason; |
| u32 insn_len; |
| u32 ctrl_exit; |
| |
| guest_rip = vmcs_read(GUEST_RIP); |
| reason = vmcs_read(EXI_REASON) & 0xff; |
| insn_len = vmcs_read(EXI_INST_LEN); |
| switch (reason) { |
| case VMX_PREEMPT: |
| switch (vmx_get_test_stage()) { |
| case 1: |
| case 2: |
| report("busy-wait for preemption timer", |
| ((rdtsc() - tsc_val) >> preempt_scale) >= |
| preempt_val); |
| vmx_set_test_stage(3); |
| vmcs_write(PREEMPT_TIMER_VALUE, preempt_val); |
| return VMX_TEST_RESUME; |
| case 3: |
| guest_halted = |
| (vmcs_read(GUEST_ACTV_STATE) == ACTV_HLT); |
| report("preemption timer during hlt", |
| ((rdtsc() - tsc_val) >> preempt_scale) >= |
| preempt_val && guest_halted); |
| vmx_set_test_stage(4); |
| vmcs_write(PIN_CONTROLS, |
| vmcs_read(PIN_CONTROLS) & ~PIN_PREEMPT); |
| vmcs_write(GUEST_ACTV_STATE, ACTV_ACTIVE); |
| return VMX_TEST_RESUME; |
| case 4: |
| report("preemption timer with 0 value", |
| saved_rip == guest_rip); |
| break; |
| default: |
| printf("Invalid stage.\n"); |
| print_vmexit_info(); |
| break; |
| } |
| break; |
| case VMX_VMCALL: |
| vmcs_write(GUEST_RIP, guest_rip + insn_len); |
| switch (vmx_get_test_stage()) { |
| case 0: |
| report("Keep preemption value", |
| vmcs_read(PREEMPT_TIMER_VALUE) == preempt_val); |
| vmx_set_test_stage(1); |
| vmcs_write(PREEMPT_TIMER_VALUE, preempt_val); |
| ctrl_exit = (vmcs_read(EXI_CONTROLS) | |
| EXI_SAVE_PREEMPT) & ctrl_exit_rev.clr; |
| vmcs_write(EXI_CONTROLS, ctrl_exit); |
| return VMX_TEST_RESUME; |
| case 1: |
| report("Save preemption value", |
| vmcs_read(PREEMPT_TIMER_VALUE) < preempt_val); |
| return VMX_TEST_RESUME; |
| case 2: |
| report("busy-wait for preemption timer", 0); |
| vmx_set_test_stage(3); |
| vmcs_write(PREEMPT_TIMER_VALUE, preempt_val); |
| return VMX_TEST_RESUME; |
| case 3: |
| report("preemption timer during hlt", 0); |
| vmx_set_test_stage(4); |
| /* fall through */ |
| case 4: |
| vmcs_write(PIN_CONTROLS, |
| vmcs_read(PIN_CONTROLS) | PIN_PREEMPT); |
| vmcs_write(PREEMPT_TIMER_VALUE, 0); |
| saved_rip = guest_rip + insn_len; |
| return VMX_TEST_RESUME; |
| case 5: |
| report("preemption timer with 0 value (vmcall stage 5)", 0); |
| break; |
| default: |
| // Should not reach here |
| printf("ERROR : unexpected stage, %d\n", |
| vmx_get_test_stage()); |
| print_vmexit_info(); |
| return VMX_TEST_VMEXIT; |
| } |
| break; |
| default: |
| printf("Unknown exit reason, %ld\n", reason); |
| print_vmexit_info(); |
| } |
| vmcs_write(PIN_CONTROLS, vmcs_read(PIN_CONTROLS) & ~PIN_PREEMPT); |
| return VMX_TEST_VMEXIT; |
| } |
| |
| void msr_bmp_init() |
| { |
| void *msr_bitmap; |
| u32 ctrl_cpu0; |
| |
| msr_bitmap = alloc_page(); |
| memset(msr_bitmap, 0x0, PAGE_SIZE); |
| ctrl_cpu0 = vmcs_read(CPU_EXEC_CTRL0); |
| ctrl_cpu0 |= CPU_MSR_BITMAP; |
| vmcs_write(CPU_EXEC_CTRL0, ctrl_cpu0); |
| vmcs_write(MSR_BITMAP, (u64)msr_bitmap); |
| } |
| |
| static int test_ctrl_pat_init() |
| { |
| u64 ctrl_ent; |
| u64 ctrl_exi; |
| |
| msr_bmp_init(); |
| if (!(ctrl_exit_rev.clr & EXI_SAVE_PAT) && |
| !(ctrl_exit_rev.clr & EXI_LOAD_PAT) && |
| !(ctrl_enter_rev.clr & ENT_LOAD_PAT)) { |
| printf("\tSave/load PAT is not supported\n"); |
| return 1; |
| } |
| |
| ctrl_ent = vmcs_read(ENT_CONTROLS); |
| ctrl_exi = vmcs_read(EXI_CONTROLS); |
| ctrl_ent |= ctrl_enter_rev.clr & ENT_LOAD_PAT; |
| ctrl_exi |= ctrl_exit_rev.clr & (EXI_SAVE_PAT | EXI_LOAD_PAT); |
| vmcs_write(ENT_CONTROLS, ctrl_ent); |
| vmcs_write(EXI_CONTROLS, ctrl_exi); |
| ia32_pat = rdmsr(MSR_IA32_CR_PAT); |
| vmcs_write(GUEST_PAT, 0x0); |
| vmcs_write(HOST_PAT, ia32_pat); |
| return VMX_TEST_START; |
| } |
| |
| static void test_ctrl_pat_main() |
| { |
| u64 guest_ia32_pat; |
| |
| guest_ia32_pat = rdmsr(MSR_IA32_CR_PAT); |
| if (!(ctrl_enter_rev.clr & ENT_LOAD_PAT)) |
| printf("\tENT_LOAD_PAT is not supported.\n"); |
| else { |
| if (guest_ia32_pat != 0) { |
| report("Entry load PAT", 0); |
| return; |
| } |
| } |
| wrmsr(MSR_IA32_CR_PAT, 0x6); |
| vmcall(); |
| guest_ia32_pat = rdmsr(MSR_IA32_CR_PAT); |
| if (ctrl_enter_rev.clr & ENT_LOAD_PAT) |
| report("Entry load PAT", guest_ia32_pat == ia32_pat); |
| } |
| |
| static int test_ctrl_pat_exit_handler() |
| { |
| u64 guest_rip; |
| ulong reason; |
| u64 guest_pat; |
| |
| guest_rip = vmcs_read(GUEST_RIP); |
| reason = vmcs_read(EXI_REASON) & 0xff; |
| switch (reason) { |
| case VMX_VMCALL: |
| guest_pat = vmcs_read(GUEST_PAT); |
| if (!(ctrl_exit_rev.clr & EXI_SAVE_PAT)) { |
| printf("\tEXI_SAVE_PAT is not supported\n"); |
| vmcs_write(GUEST_PAT, 0x6); |
| } else { |
| report("Exit save PAT", guest_pat == 0x6); |
| } |
| if (!(ctrl_exit_rev.clr & EXI_LOAD_PAT)) |
| printf("\tEXI_LOAD_PAT is not supported\n"); |
| else |
| report("Exit load PAT", rdmsr(MSR_IA32_CR_PAT) == ia32_pat); |
| vmcs_write(GUEST_PAT, ia32_pat); |
| vmcs_write(GUEST_RIP, guest_rip + 3); |
| return VMX_TEST_RESUME; |
| default: |
| printf("ERROR : Undefined exit reason, reason = %ld.\n", reason); |
| break; |
| } |
| return VMX_TEST_VMEXIT; |
| } |
| |
| static int test_ctrl_efer_init() |
| { |
| u64 ctrl_ent; |
| u64 ctrl_exi; |
| |
| msr_bmp_init(); |
| ctrl_ent = vmcs_read(ENT_CONTROLS) | ENT_LOAD_EFER; |
| ctrl_exi = vmcs_read(EXI_CONTROLS) | EXI_SAVE_EFER | EXI_LOAD_EFER; |
| vmcs_write(ENT_CONTROLS, ctrl_ent & ctrl_enter_rev.clr); |
| vmcs_write(EXI_CONTROLS, ctrl_exi & ctrl_exit_rev.clr); |
| ia32_efer = rdmsr(MSR_EFER); |
| vmcs_write(GUEST_EFER, ia32_efer ^ EFER_NX); |
| vmcs_write(HOST_EFER, ia32_efer ^ EFER_NX); |
| return VMX_TEST_START; |
| } |
| |
| static void test_ctrl_efer_main() |
| { |
| u64 guest_ia32_efer; |
| |
| guest_ia32_efer = rdmsr(MSR_EFER); |
| if (!(ctrl_enter_rev.clr & ENT_LOAD_EFER)) |
| printf("\tENT_LOAD_EFER is not supported.\n"); |
| else { |
| if (guest_ia32_efer != (ia32_efer ^ EFER_NX)) { |
| report("Entry load EFER", 0); |
| return; |
| } |
| } |
| wrmsr(MSR_EFER, ia32_efer); |
| vmcall(); |
| guest_ia32_efer = rdmsr(MSR_EFER); |
| if (ctrl_enter_rev.clr & ENT_LOAD_EFER) |
| report("Entry load EFER", guest_ia32_efer == ia32_efer); |
| } |
| |
| static int test_ctrl_efer_exit_handler() |
| { |
| u64 guest_rip; |
| ulong reason; |
| u64 guest_efer; |
| |
| guest_rip = vmcs_read(GUEST_RIP); |
| reason = vmcs_read(EXI_REASON) & 0xff; |
| switch (reason) { |
| case VMX_VMCALL: |
| guest_efer = vmcs_read(GUEST_EFER); |
| if (!(ctrl_exit_rev.clr & EXI_SAVE_EFER)) { |
| printf("\tEXI_SAVE_EFER is not supported\n"); |
| vmcs_write(GUEST_EFER, ia32_efer); |
| } else { |
| report("Exit save EFER", guest_efer == ia32_efer); |
| } |
| if (!(ctrl_exit_rev.clr & EXI_LOAD_EFER)) { |
| printf("\tEXI_LOAD_EFER is not supported\n"); |
| wrmsr(MSR_EFER, ia32_efer ^ EFER_NX); |
| } else { |
| report("Exit load EFER", rdmsr(MSR_EFER) == (ia32_efer ^ EFER_NX)); |
| } |
| vmcs_write(GUEST_PAT, ia32_efer); |
| vmcs_write(GUEST_RIP, guest_rip + 3); |
| return VMX_TEST_RESUME; |
| default: |
| printf("ERROR : Undefined exit reason, reason = %ld.\n", reason); |
| break; |
| } |
| return VMX_TEST_VMEXIT; |
| } |
| |
| u32 guest_cr0, guest_cr4; |
| |
| static void cr_shadowing_main() |
| { |
| u32 cr0, cr4, tmp; |
| |
| // Test read through |
| vmx_set_test_stage(0); |
| guest_cr0 = read_cr0(); |
| if (vmx_get_test_stage() == 1) |
| report("Read through CR0", 0); |
| else |
| vmcall(); |
| vmx_set_test_stage(1); |
| guest_cr4 = read_cr4(); |
| if (vmx_get_test_stage() == 2) |
| report("Read through CR4", 0); |
| else |
| vmcall(); |
| // Test write through |
| guest_cr0 = guest_cr0 ^ (X86_CR0_TS | X86_CR0_MP); |
| guest_cr4 = guest_cr4 ^ (X86_CR4_TSD | X86_CR4_DE); |
| vmx_set_test_stage(2); |
| write_cr0(guest_cr0); |
| if (vmx_get_test_stage() == 3) |
| report("Write throuth CR0", 0); |
| else |
| vmcall(); |
| vmx_set_test_stage(3); |
| write_cr4(guest_cr4); |
| if (vmx_get_test_stage() == 4) |
| report("Write through CR4", 0); |
| else |
| vmcall(); |
| // Test read shadow |
| vmx_set_test_stage(4); |
| vmcall(); |
| cr0 = read_cr0(); |
| if (vmx_get_test_stage() != 5) |
| report("Read shadowing CR0", cr0 == guest_cr0); |
| vmx_set_test_stage(5); |
| cr4 = read_cr4(); |
| if (vmx_get_test_stage() != 6) |
| report("Read shadowing CR4", cr4 == guest_cr4); |
| // Test write shadow (same value with shadow) |
| vmx_set_test_stage(6); |
| write_cr0(guest_cr0); |
| if (vmx_get_test_stage() == 7) |
| report("Write shadowing CR0 (same value with shadow)", 0); |
| else |
| vmcall(); |
| vmx_set_test_stage(7); |
| write_cr4(guest_cr4); |
| if (vmx_get_test_stage() == 8) |
| report("Write shadowing CR4 (same value with shadow)", 0); |
| else |
| vmcall(); |
| // Test write shadow (different value) |
| vmx_set_test_stage(8); |
| tmp = guest_cr0 ^ X86_CR0_TS; |
| asm volatile("mov %0, %%rsi\n\t" |
| "mov %%rsi, %%cr0\n\t" |
| ::"m"(tmp) |
| :"rsi", "memory", "cc"); |
| report("Write shadowing different X86_CR0_TS", vmx_get_test_stage() == 9); |
| vmx_set_test_stage(9); |
| tmp = guest_cr0 ^ X86_CR0_MP; |
| asm volatile("mov %0, %%rsi\n\t" |
| "mov %%rsi, %%cr0\n\t" |
| ::"m"(tmp) |
| :"rsi", "memory", "cc"); |
| report("Write shadowing different X86_CR0_MP", vmx_get_test_stage() == 10); |
| vmx_set_test_stage(10); |
| tmp = guest_cr4 ^ X86_CR4_TSD; |
| asm volatile("mov %0, %%rsi\n\t" |
| "mov %%rsi, %%cr4\n\t" |
| ::"m"(tmp) |
| :"rsi", "memory", "cc"); |
| report("Write shadowing different X86_CR4_TSD", vmx_get_test_stage() == 11); |
| vmx_set_test_stage(11); |
| tmp = guest_cr4 ^ X86_CR4_DE; |
| asm volatile("mov %0, %%rsi\n\t" |
| "mov %%rsi, %%cr4\n\t" |
| ::"m"(tmp) |
| :"rsi", "memory", "cc"); |
| report("Write shadowing different X86_CR4_DE", vmx_get_test_stage() == 12); |
| } |
| |
| static int cr_shadowing_exit_handler() |
| { |
| u64 guest_rip; |
| ulong reason; |
| u32 insn_len; |
| u32 exit_qual; |
| |
| guest_rip = vmcs_read(GUEST_RIP); |
| reason = vmcs_read(EXI_REASON) & 0xff; |
| insn_len = vmcs_read(EXI_INST_LEN); |
| exit_qual = vmcs_read(EXI_QUALIFICATION); |
| switch (reason) { |
| case VMX_VMCALL: |
| switch (vmx_get_test_stage()) { |
| case 0: |
| report("Read through CR0", guest_cr0 == vmcs_read(GUEST_CR0)); |
| break; |
| case 1: |
| report("Read through CR4", guest_cr4 == vmcs_read(GUEST_CR4)); |
| break; |
| case 2: |
| report("Write through CR0", guest_cr0 == vmcs_read(GUEST_CR0)); |
| break; |
| case 3: |
| report("Write through CR4", guest_cr4 == vmcs_read(GUEST_CR4)); |
| break; |
| case 4: |
| guest_cr0 = vmcs_read(GUEST_CR0) ^ (X86_CR0_TS | X86_CR0_MP); |
| guest_cr4 = vmcs_read(GUEST_CR4) ^ (X86_CR4_TSD | X86_CR4_DE); |
| vmcs_write(CR0_MASK, X86_CR0_TS | X86_CR0_MP); |
| vmcs_write(CR0_READ_SHADOW, guest_cr0 & (X86_CR0_TS | X86_CR0_MP)); |
| vmcs_write(CR4_MASK, X86_CR4_TSD | X86_CR4_DE); |
| vmcs_write(CR4_READ_SHADOW, guest_cr4 & (X86_CR4_TSD | X86_CR4_DE)); |
| break; |
| case 6: |
| report("Write shadowing CR0 (same value)", |
| guest_cr0 == (vmcs_read(GUEST_CR0) ^ (X86_CR0_TS | X86_CR0_MP))); |
| break; |
| case 7: |
| report("Write shadowing CR4 (same value)", |
| guest_cr4 == (vmcs_read(GUEST_CR4) ^ (X86_CR4_TSD | X86_CR4_DE))); |
| break; |
| default: |
| // Should not reach here |
| printf("ERROR : unexpected stage, %d\n", |
| vmx_get_test_stage()); |
| print_vmexit_info(); |
| return VMX_TEST_VMEXIT; |
| } |
| vmcs_write(GUEST_RIP, guest_rip + insn_len); |
| return VMX_TEST_RESUME; |
| case VMX_CR: |
| switch (vmx_get_test_stage()) { |
| case 4: |
| report("Read shadowing CR0", 0); |
| vmx_inc_test_stage(); |
| break; |
| case 5: |
| report("Read shadowing CR4", 0); |
| vmx_inc_test_stage(); |
| break; |
| case 6: |
| report("Write shadowing CR0 (same value)", 0); |
| vmx_inc_test_stage(); |
| break; |
| case 7: |
| report("Write shadowing CR4 (same value)", 0); |
| vmx_inc_test_stage(); |
| break; |
| case 8: |
| case 9: |
| // 0x600 encodes "mov %esi, %cr0" |
| if (exit_qual == 0x600) |
| vmx_inc_test_stage(); |
| break; |
| case 10: |
| case 11: |
| // 0x604 encodes "mov %esi, %cr4" |
| if (exit_qual == 0x604) |
| vmx_inc_test_stage(); |
| break; |
| default: |
| // Should not reach here |
| printf("ERROR : unexpected stage, %d\n", |
| vmx_get_test_stage()); |
| print_vmexit_info(); |
| return VMX_TEST_VMEXIT; |
| } |
| vmcs_write(GUEST_RIP, guest_rip + insn_len); |
| return VMX_TEST_RESUME; |
| default: |
| printf("Unknown exit reason, %ld\n", reason); |
| print_vmexit_info(); |
| } |
| return VMX_TEST_VMEXIT; |
| } |
| |
| static int iobmp_init() |
| { |
| u32 ctrl_cpu0; |
| |
| io_bitmap_a = alloc_page(); |
| io_bitmap_b = alloc_page(); |
| memset(io_bitmap_a, 0x0, PAGE_SIZE); |
| memset(io_bitmap_b, 0x0, PAGE_SIZE); |
| ctrl_cpu0 = vmcs_read(CPU_EXEC_CTRL0); |
| ctrl_cpu0 |= CPU_IO_BITMAP; |
| ctrl_cpu0 &= (~CPU_IO); |
| vmcs_write(CPU_EXEC_CTRL0, ctrl_cpu0); |
| vmcs_write(IO_BITMAP_A, (u64)io_bitmap_a); |
| vmcs_write(IO_BITMAP_B, (u64)io_bitmap_b); |
| return VMX_TEST_START; |
| } |
| |
| static void iobmp_main() |
| { |
| // stage 0, test IO pass |
| vmx_set_test_stage(0); |
| inb(0x5000); |
| outb(0x0, 0x5000); |
| report("I/O bitmap - I/O pass", vmx_get_test_stage() == 0); |
| // test IO width, in/out |
| ((u8 *)io_bitmap_a)[0] = 0xFF; |
| vmx_set_test_stage(2); |
| inb(0x0); |
| report("I/O bitmap - trap in", vmx_get_test_stage() == 3); |
| vmx_set_test_stage(3); |
| outw(0x0, 0x0); |
| report("I/O bitmap - trap out", vmx_get_test_stage() == 4); |
| vmx_set_test_stage(4); |
| inl(0x0); |
| report("I/O bitmap - I/O width, long", vmx_get_test_stage() == 5); |
| // test low/high IO port |
| vmx_set_test_stage(5); |
| ((u8 *)io_bitmap_a)[0x5000 / 8] = (1 << (0x5000 % 8)); |
| inb(0x5000); |
| report("I/O bitmap - I/O port, low part", vmx_get_test_stage() == 6); |
| vmx_set_test_stage(6); |
| ((u8 *)io_bitmap_b)[0x1000 / 8] = (1 << (0x1000 % 8)); |
| inb(0x9000); |
| report("I/O bitmap - I/O port, high part", vmx_get_test_stage() == 7); |
| // test partial pass |
| vmx_set_test_stage(7); |
| inl(0x4FFF); |
| report("I/O bitmap - partial pass", vmx_get_test_stage() == 8); |
| // test overrun |
| vmx_set_test_stage(8); |
| memset(io_bitmap_a, 0x0, PAGE_SIZE); |
| memset(io_bitmap_b, 0x0, PAGE_SIZE); |
| inl(0xFFFF); |
| report("I/O bitmap - overrun", vmx_get_test_stage() == 9); |
| vmx_set_test_stage(9); |
| vmcall(); |
| outb(0x0, 0x0); |
| report("I/O bitmap - ignore unconditional exiting", |
| vmx_get_test_stage() == 9); |
| vmx_set_test_stage(10); |
| vmcall(); |
| outb(0x0, 0x0); |
| report("I/O bitmap - unconditional exiting", |
| vmx_get_test_stage() == 11); |
| } |
| |
| static int iobmp_exit_handler() |
| { |
| u64 guest_rip; |
| ulong reason, exit_qual; |
| u32 insn_len, ctrl_cpu0; |
| |
| guest_rip = vmcs_read(GUEST_RIP); |
| reason = vmcs_read(EXI_REASON) & 0xff; |
| exit_qual = vmcs_read(EXI_QUALIFICATION); |
| insn_len = vmcs_read(EXI_INST_LEN); |
| switch (reason) { |
| case VMX_IO: |
| switch (vmx_get_test_stage()) { |
| case 0: |
| case 1: |
| vmx_inc_test_stage(); |
| break; |
| case 2: |
| report("I/O bitmap - I/O width, byte", |
| (exit_qual & VMX_IO_SIZE_MASK) == _VMX_IO_BYTE); |
| report("I/O bitmap - I/O direction, in", exit_qual & VMX_IO_IN); |
| vmx_inc_test_stage(); |
| break; |
| case 3: |
| report("I/O bitmap - I/O width, word", |
| (exit_qual & VMX_IO_SIZE_MASK) == _VMX_IO_WORD); |
| report("I/O bitmap - I/O direction, out", |
| !(exit_qual & VMX_IO_IN)); |
| vmx_inc_test_stage(); |
| break; |
| case 4: |
| report("I/O bitmap - I/O width, long", |
| (exit_qual & VMX_IO_SIZE_MASK) == _VMX_IO_LONG); |
| vmx_inc_test_stage(); |
| break; |
| case 5: |
| if (((exit_qual & VMX_IO_PORT_MASK) >> VMX_IO_PORT_SHIFT) == 0x5000) |
| vmx_inc_test_stage(); |
| break; |
| case 6: |
| if (((exit_qual & VMX_IO_PORT_MASK) >> VMX_IO_PORT_SHIFT) == 0x9000) |
| vmx_inc_test_stage(); |
| break; |
| case 7: |
| if (((exit_qual & VMX_IO_PORT_MASK) >> VMX_IO_PORT_SHIFT) == 0x4FFF) |
| vmx_inc_test_stage(); |
| break; |
| case 8: |
| if (((exit_qual & VMX_IO_PORT_MASK) >> VMX_IO_PORT_SHIFT) == 0xFFFF) |
| vmx_inc_test_stage(); |
| break; |
| case 9: |
| case 10: |
| ctrl_cpu0 = vmcs_read(CPU_EXEC_CTRL0); |
| vmcs_write(CPU_EXEC_CTRL0, ctrl_cpu0 & ~CPU_IO); |
| vmx_inc_test_stage(); |
| break; |
| default: |
| // Should not reach here |
| printf("ERROR : unexpected stage, %d\n", |
| vmx_get_test_stage()); |
| print_vmexit_info(); |
| return VMX_TEST_VMEXIT; |
| } |
| vmcs_write(GUEST_RIP, guest_rip + insn_len); |
| return VMX_TEST_RESUME; |
| case VMX_VMCALL: |
| switch (vmx_get_test_stage()) { |
| case 9: |
| ctrl_cpu0 = vmcs_read(CPU_EXEC_CTRL0); |
| ctrl_cpu0 |= CPU_IO | CPU_IO_BITMAP; |
| vmcs_write(CPU_EXEC_CTRL0, ctrl_cpu0); |
| break; |
| case 10: |
| ctrl_cpu0 = vmcs_read(CPU_EXEC_CTRL0); |
| ctrl_cpu0 = (ctrl_cpu0 & ~CPU_IO_BITMAP) | CPU_IO; |
| vmcs_write(CPU_EXEC_CTRL0, ctrl_cpu0); |
| break; |
| default: |
| // Should not reach here |
| printf("ERROR : unexpected stage, %d\n", |
| vmx_get_test_stage()); |
| print_vmexit_info(); |
| return VMX_TEST_VMEXIT; |
| } |
| vmcs_write(GUEST_RIP, guest_rip + insn_len); |
| return VMX_TEST_RESUME; |
| default: |
| printf("guest_rip = 0x%lx\n", guest_rip); |
| printf("\tERROR : Undefined exit reason, reason = %ld.\n", reason); |
| break; |
| } |
| return VMX_TEST_VMEXIT; |
| } |
| |
| #define INSN_CPU0 0 |
| #define INSN_CPU1 1 |
| #define INSN_ALWAYS_TRAP 2 |
| |
| #define FIELD_EXIT_QUAL (1 << 0) |
| #define FIELD_INSN_INFO (1 << 1) |
| |
| asm( |
| "insn_hlt: hlt;ret\n\t" |
| "insn_invlpg: invlpg 0x12345678;ret\n\t" |
| "insn_mwait: mwait;ret\n\t" |
| "insn_rdpmc: xor %ecx, %ecx; rdpmc;ret\n\t" |
| "insn_rdtsc: rdtsc;ret\n\t" |
| "insn_cr3_load: mov cr3,%rax; mov %rax,%cr3;ret\n\t" |
| "insn_cr3_store: mov %cr3,%rax;ret\n\t" |
| #ifdef __x86_64__ |
| "insn_cr8_load: mov %rax,%cr8;ret\n\t" |
| "insn_cr8_store: mov %cr8,%rax;ret\n\t" |
| #endif |
| "insn_monitor: monitor;ret\n\t" |
| "insn_pause: pause;ret\n\t" |
| "insn_wbinvd: wbinvd;ret\n\t" |
| "insn_cpuid: mov $10, %eax; cpuid;ret\n\t" |
| "insn_invd: invd;ret\n\t" |
| "insn_sgdt: sgdt gdt64_desc;ret\n\t" |
| "insn_lgdt: lgdt gdt64_desc;ret\n\t" |
| "insn_sidt: sidt idt_descr;ret\n\t" |
| "insn_lidt: lidt idt_descr;ret\n\t" |
| "insn_sldt: sldt %ax;ret\n\t" |
| "insn_lldt: xor %eax, %eax; lldt %ax;ret\n\t" |
| "insn_str: str %ax;ret\n\t" |
| ); |
| extern void insn_hlt(); |
| extern void insn_invlpg(); |
| extern void insn_mwait(); |
| extern void insn_rdpmc(); |
| extern void insn_rdtsc(); |
| extern void insn_cr3_load(); |
| extern void insn_cr3_store(); |
| #ifdef __x86_64__ |
| extern void insn_cr8_load(); |
| extern void insn_cr8_store(); |
| #endif |
| extern void insn_monitor(); |
| extern void insn_pause(); |
| extern void insn_wbinvd(); |
| extern void insn_sgdt(); |
| extern void insn_lgdt(); |
| extern void insn_sidt(); |
| extern void insn_lidt(); |
| extern void insn_sldt(); |
| extern void insn_lldt(); |
| extern void insn_str(); |
| extern void insn_cpuid(); |
| extern void insn_invd(); |
| |
| u32 cur_insn; |
| u64 cr3; |
| |
| struct insn_table { |
| const char *name; |
| u32 flag; |
| void (*insn_func)(); |
| u32 type; |
| u32 reason; |
| ulong exit_qual; |
| u32 insn_info; |
| // Use FIELD_EXIT_QUAL and FIELD_INSN_INFO to define |
| // which field need to be tested, reason is always tested |
| u32 test_field; |
| }; |
| |
| /* |
| * Add more test cases of instruction intercept here. Elements in this |
| * table is: |
| * name/control flag/insn function/type/exit reason/exit qulification/ |
| * instruction info/field to test |
| * The last field defines which fields (exit_qual and insn_info) need to be |
| * tested in exit handler. If set to 0, only "reason" is checked. |
| */ |
| static struct insn_table insn_table[] = { |
| // Flags for Primary Processor-Based VM-Execution Controls |
| {"HLT", CPU_HLT, insn_hlt, INSN_CPU0, 12, 0, 0, 0}, |
| {"INVLPG", CPU_INVLPG, insn_invlpg, INSN_CPU0, 14, |
| 0x12345678, 0, FIELD_EXIT_QUAL}, |
| {"MWAIT", CPU_MWAIT, insn_mwait, INSN_CPU0, 36, 0, 0, 0}, |
| {"RDPMC", CPU_RDPMC, insn_rdpmc, INSN_CPU0, 15, 0, 0, 0}, |
| {"RDTSC", CPU_RDTSC, insn_rdtsc, INSN_CPU0, 16, 0, 0, 0}, |
| {"CR3 load", CPU_CR3_LOAD, insn_cr3_load, INSN_CPU0, 28, 0x3, 0, |
| FIELD_EXIT_QUAL}, |
| {"CR3 store", CPU_CR3_STORE, insn_cr3_store, INSN_CPU0, 28, 0x13, 0, |
| FIELD_EXIT_QUAL}, |
| #ifdef __x86_64__ |
| {"CR8 load", CPU_CR8_LOAD, insn_cr8_load, INSN_CPU0, 28, 0x8, 0, |
| FIELD_EXIT_QUAL}, |
| {"CR8 store", CPU_CR8_STORE, insn_cr8_store, INSN_CPU0, 28, 0x18, 0, |
| FIELD_EXIT_QUAL}, |
| #endif |
| {"MONITOR", CPU_MONITOR, insn_monitor, INSN_CPU0, 39, 0, 0, 0}, |
| {"PAUSE", CPU_PAUSE, insn_pause, INSN_CPU0, 40, 0, 0, 0}, |
| // Flags for Secondary Processor-Based VM-Execution Controls |
| {"WBINVD", CPU_WBINVD, insn_wbinvd, INSN_CPU1, 54, 0, 0, 0}, |
| {"DESC_TABLE (SGDT)", CPU_DESC_TABLE, insn_sgdt, INSN_CPU1, 46, 0, 0, 0}, |
| {"DESC_TABLE (LGDT)", CPU_DESC_TABLE, insn_lgdt, INSN_CPU1, 46, 0, 0, 0}, |
| {"DESC_TABLE (SIDT)", CPU_DESC_TABLE, insn_sidt, INSN_CPU1, 46, 0, 0, 0}, |
| {"DESC_TABLE (LIDT)", CPU_DESC_TABLE, insn_lidt, INSN_CPU1, 46, 0, 0, 0}, |
| {"DESC_TABLE (SLDT)", CPU_DESC_TABLE, insn_sldt, INSN_CPU1, 47, 0, 0, 0}, |
| {"DESC_TABLE (LLDT)", CPU_DESC_TABLE, insn_lldt, INSN_CPU1, 47, 0, 0, 0}, |
| {"DESC_TABLE (STR)", CPU_DESC_TABLE, insn_str, INSN_CPU1, 47, 0, 0, 0}, |
| /* LTR causes a #GP if done with a busy selector, so it is not tested. */ |
| // Instructions always trap |
| {"CPUID", 0, insn_cpuid, INSN_ALWAYS_TRAP, 10, 0, 0, 0}, |
| {"INVD", 0, insn_invd, INSN_ALWAYS_TRAP, 13, 0, 0, 0}, |
| // Instructions never trap |
| {NULL}, |
| }; |
| |
| static int insn_intercept_init() |
| { |
| u32 ctrl_cpu; |
| |
| ctrl_cpu = ctrl_cpu_rev[0].set | CPU_SECONDARY; |
| ctrl_cpu &= ctrl_cpu_rev[0].clr; |
| vmcs_write(CPU_EXEC_CTRL0, ctrl_cpu); |
| vmcs_write(CPU_EXEC_CTRL1, ctrl_cpu_rev[1].set); |
| cr3 = read_cr3(); |
| return VMX_TEST_START; |
| } |
| |
| static void insn_intercept_main() |
| { |
| for (cur_insn = 0; insn_table[cur_insn].name != NULL; cur_insn++) { |
| vmx_set_test_stage(cur_insn * 2); |
| if ((insn_table[cur_insn].type == INSN_CPU0 && |
| !(ctrl_cpu_rev[0].clr & insn_table[cur_insn].flag)) || |
| (insn_table[cur_insn].type == INSN_CPU1 && |
| !(ctrl_cpu_rev[1].clr & insn_table[cur_insn].flag))) { |
| printf("\tCPU_CTRL%d.CPU_%s is not supported.\n", |
| insn_table[cur_insn].type - INSN_CPU0, |
| insn_table[cur_insn].name); |
| continue; |
| } |
| |
| if ((insn_table[cur_insn].type == INSN_CPU0 && |
| !(ctrl_cpu_rev[0].set & insn_table[cur_insn].flag)) || |
| (insn_table[cur_insn].type == INSN_CPU1 && |
| !(ctrl_cpu_rev[1].set & insn_table[cur_insn].flag))) { |
| /* skip hlt, it stalls the guest and is tested below */ |
| if (insn_table[cur_insn].insn_func != insn_hlt) |
| insn_table[cur_insn].insn_func(); |
| report("execute %s", vmx_get_test_stage() == cur_insn * 2, |
| insn_table[cur_insn].name); |
| } else if (insn_table[cur_insn].type != INSN_ALWAYS_TRAP) |
| printf("\tCPU_CTRL%d.CPU_%s always traps.\n", |
| insn_table[cur_insn].type - INSN_CPU0, |
| insn_table[cur_insn].name); |
| |
| vmcall(); |
| |
| insn_table[cur_insn].insn_func(); |
| report("intercept %s", vmx_get_test_stage() == cur_insn * 2 + 1, |
| insn_table[cur_insn].name); |
| |
| vmx_set_test_stage(cur_insn * 2 + 1); |
| vmcall(); |
| } |
| } |
| |
| static int insn_intercept_exit_handler() |
| { |
| u64 guest_rip; |
| u32 reason; |
| ulong exit_qual; |
| u32 insn_len; |
| u32 insn_info; |
| bool pass; |
| |
| guest_rip = vmcs_read(GUEST_RIP); |
| reason = vmcs_read(EXI_REASON) & 0xff; |
| exit_qual = vmcs_read(EXI_QUALIFICATION); |
| insn_len = vmcs_read(EXI_INST_LEN); |
| insn_info = vmcs_read(EXI_INST_INFO); |
| |
| if (reason == VMX_VMCALL) { |
| u32 val = 0; |
| |
| if (insn_table[cur_insn].type == INSN_CPU0) |
| val = vmcs_read(CPU_EXEC_CTRL0); |
| else if (insn_table[cur_insn].type == INSN_CPU1) |
| val = vmcs_read(CPU_EXEC_CTRL1); |
| |
| if (vmx_get_test_stage() & 1) |
| val &= ~insn_table[cur_insn].flag; |
| else |
| val |= insn_table[cur_insn].flag; |
| |
| if (insn_table[cur_insn].type == INSN_CPU0) |
| vmcs_write(CPU_EXEC_CTRL0, val | ctrl_cpu_rev[0].set); |
| else if (insn_table[cur_insn].type == INSN_CPU1) |
| vmcs_write(CPU_EXEC_CTRL1, val | ctrl_cpu_rev[1].set); |
| } else { |
| pass = (cur_insn * 2 == vmx_get_test_stage()) && |
| insn_table[cur_insn].reason == reason; |
| if (insn_table[cur_insn].test_field & FIELD_EXIT_QUAL && |
| insn_table[cur_insn].exit_qual != exit_qual) |
| pass = false; |
| if (insn_table[cur_insn].test_field & FIELD_INSN_INFO && |
| insn_table[cur_insn].insn_info != insn_info) |
| pass = false; |
| if (pass) |
| vmx_inc_test_stage(); |
| } |
| vmcs_write(GUEST_RIP, guest_rip + insn_len); |
| return VMX_TEST_RESUME; |
| } |
| |
| |
| static int setup_ept() |
| { |
| int support_2m; |
| unsigned long end_of_memory; |
| |
| if (!(ept_vpid.val & EPT_CAP_UC) && |
| !(ept_vpid.val & EPT_CAP_WB)) { |
| printf("\tEPT paging-structure memory type " |
| "UC&WB are not supported\n"); |
| return 1; |
| } |
| if (ept_vpid.val & EPT_CAP_UC) |
| eptp = EPT_MEM_TYPE_UC; |
| else |
| eptp = EPT_MEM_TYPE_WB; |
| if (!(ept_vpid.val & EPT_CAP_PWL4)) { |
| printf("\tPWL4 is not supported\n"); |
| return 1; |
| } |
| eptp |= (3 << EPTP_PG_WALK_LEN_SHIFT); |
| pml4 = alloc_page(); |
| memset(pml4, 0, PAGE_SIZE); |
| eptp |= virt_to_phys(pml4); |
| vmcs_write(EPTP, eptp); |
| support_2m = !!(ept_vpid.val & EPT_CAP_2M_PAGE); |
| end_of_memory = fwcfg_get_u64(FW_CFG_RAM_SIZE); |
| if (end_of_memory < (1ul << 32)) |
| end_of_memory = (1ul << 32); |
| setup_ept_range(pml4, 0, end_of_memory, 0, support_2m, |
| EPT_WA | EPT_RA | EPT_EA); |
| return 0; |
| } |
| |
| static int apic_version; |
| |
| static int ept_init() |
| { |
| u32 ctrl_cpu[2]; |
| |
| if (!(ctrl_cpu_rev[0].clr & CPU_SECONDARY) || |
| !(ctrl_cpu_rev[1].clr & CPU_EPT)) { |
| printf("\tEPT is not supported"); |
| return VMX_TEST_EXIT; |
| } |
| |
| ctrl_cpu[0] = vmcs_read(CPU_EXEC_CTRL0); |
| ctrl_cpu[1] = vmcs_read(CPU_EXEC_CTRL1); |
| ctrl_cpu[0] = (ctrl_cpu[0] | CPU_SECONDARY) |
| & ctrl_cpu_rev[0].clr; |
| ctrl_cpu[1] = (ctrl_cpu[1] | CPU_EPT) |
| & ctrl_cpu_rev[1].clr; |
| vmcs_write(CPU_EXEC_CTRL0, ctrl_cpu[0]); |
| vmcs_write(CPU_EXEC_CTRL1, ctrl_cpu[1]); |
| if (setup_ept()) |
| return VMX_TEST_EXIT; |
| data_page1 = alloc_page(); |
| data_page2 = alloc_page(); |
| memset(data_page1, 0x0, PAGE_SIZE); |
| memset(data_page2, 0x0, PAGE_SIZE); |
| *((u32 *)data_page1) = MAGIC_VAL_1; |
| *((u32 *)data_page2) = MAGIC_VAL_2; |
| install_ept(pml4, (unsigned long)data_page1, (unsigned long)data_page2, |
| EPT_RA | EPT_WA | EPT_EA); |
| |
| apic_version = *((u32 *)0xfee00030UL); |
| return VMX_TEST_START; |
| } |
| |
| static void ept_main() |
| { |
| vmx_set_test_stage(0); |
| if (*((u32 *)data_page2) != MAGIC_VAL_1 || |
| *((u32 *)data_page1) != MAGIC_VAL_1) |
| report("EPT basic framework - read", 0); |
| else { |
| *((u32 *)data_page2) = MAGIC_VAL_3; |
| vmcall(); |
| if (vmx_get_test_stage() == 1) { |
| if (*((u32 *)data_page1) == MAGIC_VAL_3 && |
| *((u32 *)data_page2) == MAGIC_VAL_2) |
| report("EPT basic framework", 1); |
| else |
| report("EPT basic framework - remap", 1); |
| } |
| } |
| // Test EPT Misconfigurations |
| vmx_set_test_stage(1); |
| vmcall(); |
| *((u32 *)data_page1) = MAGIC_VAL_1; |
| if (vmx_get_test_stage() != 2) { |
| report("EPT misconfigurations", 0); |
| goto t1; |
| } |
| vmx_set_test_stage(2); |
| vmcall(); |
| *((u32 *)data_page1) = MAGIC_VAL_1; |
| report("EPT misconfigurations", vmx_get_test_stage() == 3); |
| t1: |
| // Test EPT violation |
| vmx_set_test_stage(3); |
| vmcall(); |
| *((u32 *)data_page1) = MAGIC_VAL_1; |
| report("EPT violation - page permission", vmx_get_test_stage() == 4); |
| // Violation caused by EPT paging structure |
| vmx_set_test_stage(4); |
| vmcall(); |
| *((u32 *)data_page1) = MAGIC_VAL_2; |
| report("EPT violation - paging structure", vmx_get_test_stage() == 5); |
| |
| // Test EPT access to L1 MMIO |
| vmx_set_test_stage(6); |
| report("EPT - MMIO access", *((u32 *)0xfee00030UL) == apic_version); |
| |
| // Test invalid operand for INVEPT |
| vmcall(); |
| report("EPT - unsupported INVEPT", vmx_get_test_stage() == 7); |
| } |
| |
| bool invept_test(int type, u64 eptp) |
| { |
| bool ret, supported; |
| |
| supported = ept_vpid.val & (EPT_CAP_INVEPT_SINGLE >> INVEPT_SINGLE << type); |
| ret = invept(type, eptp); |
| |
| if (ret == !supported) |
| return false; |
| |
| if (!supported) |
| printf("WARNING: unsupported invept passed!\n"); |
| else |
| printf("WARNING: invept failed!\n"); |
| |
| return true; |
| } |
| |
| static int ept_exit_handler() |
| { |
| u64 guest_rip; |
| ulong reason; |
| u32 insn_len; |
| u32 exit_qual; |
| static unsigned long data_page1_pte, data_page1_pte_pte; |
| |
| guest_rip = vmcs_read(GUEST_RIP); |
| reason = vmcs_read(EXI_REASON) & 0xff; |
| insn_len = vmcs_read(EXI_INST_LEN); |
| exit_qual = vmcs_read(EXI_QUALIFICATION); |
| switch (reason) { |
| case VMX_VMCALL: |
| switch (vmx_get_test_stage()) { |
| case 0: |
| if (*((u32 *)data_page1) == MAGIC_VAL_3 && |
| *((u32 *)data_page2) == MAGIC_VAL_2) { |
| vmx_inc_test_stage(); |
| install_ept(pml4, (unsigned long)data_page2, |
| (unsigned long)data_page2, |
| EPT_RA | EPT_WA | EPT_EA); |
| } else |
| report("EPT basic framework - write", 0); |
| break; |
| case 1: |
| install_ept(pml4, (unsigned long)data_page1, |
| (unsigned long)data_page1, EPT_WA); |
| ept_sync(INVEPT_SINGLE, eptp); |
| break; |
| case 2: |
| install_ept(pml4, (unsigned long)data_page1, |
| (unsigned long)data_page1, |
| EPT_RA | EPT_WA | EPT_EA | |
| (2 << EPT_MEM_TYPE_SHIFT)); |
| ept_sync(INVEPT_SINGLE, eptp); |
| break; |
| case 3: |
| data_page1_pte = get_ept_pte(pml4, |
| (unsigned long)data_page1, 1); |
| set_ept_pte(pml4, (unsigned long)data_page1, |
| 1, data_page1_pte & (~EPT_PRESENT)); |
| ept_sync(INVEPT_SINGLE, eptp); |
| break; |
| case 4: |
| data_page1_pte = get_ept_pte(pml4, |
| (unsigned long)data_page1, 2); |
| data_page1_pte &= PAGE_MASK; |
| data_page1_pte_pte = get_ept_pte(pml4, data_page1_pte, 2); |
| set_ept_pte(pml4, data_page1_pte, 2, |
| data_page1_pte_pte & (~EPT_PRESENT)); |
| ept_sync(INVEPT_SINGLE, eptp); |
| break; |
| case 6: |
| if (!invept_test(0, eptp)) |
| vmx_inc_test_stage(); |
| break; |
| // Should not reach here |
| default: |
| printf("ERROR - unexpected stage, %d.\n", |
| vmx_get_test_stage()); |
| print_vmexit_info(); |
| return VMX_TEST_VMEXIT; |
| } |
| vmcs_write(GUEST_RIP, guest_rip + insn_len); |
| return VMX_TEST_RESUME; |
| case VMX_EPT_MISCONFIG: |
| switch (vmx_get_test_stage()) { |
| case 1: |
| case 2: |
| vmx_inc_test_stage(); |
| install_ept(pml4, (unsigned long)data_page1, |
| (unsigned long)data_page1, |
| EPT_RA | EPT_WA | EPT_EA); |
| ept_sync(INVEPT_SINGLE, eptp); |
| break; |
| // Should not reach here |
| default: |
| printf("ERROR - unexpected stage, %d.\n", |
| vmx_get_test_stage()); |
| print_vmexit_info(); |
| return VMX_TEST_VMEXIT; |
| } |
| return VMX_TEST_RESUME; |
| case VMX_EPT_VIOLATION: |
| switch(vmx_get_test_stage()) { |
| case 3: |
| if (exit_qual == (EPT_VLT_WR | EPT_VLT_LADDR_VLD | |
| EPT_VLT_PADDR)) |
| vmx_inc_test_stage(); |
| set_ept_pte(pml4, (unsigned long)data_page1, |
| 1, data_page1_pte | (EPT_PRESENT)); |
| ept_sync(INVEPT_SINGLE, eptp); |
| break; |
| case 4: |
| if (exit_qual == (EPT_VLT_RD | EPT_VLT_LADDR_VLD)) |
| vmx_inc_test_stage(); |
| set_ept_pte(pml4, data_page1_pte, 2, |
| data_page1_pte_pte | (EPT_PRESENT)); |
| ept_sync(INVEPT_SINGLE, eptp); |
| break; |
| default: |
| // Should not reach here |
| printf("ERROR : unexpected stage, %d\n", |
| vmx_get_test_stage()); |
| print_vmexit_info(); |
| return VMX_TEST_VMEXIT; |
| } |
| return VMX_TEST_RESUME; |
| default: |
| printf("Unknown exit reason, %ld\n", reason); |
| print_vmexit_info(); |
| } |
| return VMX_TEST_VMEXIT; |
| } |
| |
| bool invvpid_test(int type, u16 vpid) |
| { |
| bool ret, supported; |
| |
| supported = ept_vpid.val & (VPID_CAP_INVVPID_SINGLE >> INVVPID_SINGLE << type); |
| ret = invvpid(type, vpid, 0); |
| |
| if (ret == !supported) |
| return false; |
| |
| if (!supported) |
| printf("WARNING: unsupported invvpid passed!\n"); |
| else |
| printf("WARNING: invvpid failed!\n"); |
| |
| return true; |
| } |
| |
| static int vpid_init() |
| { |
| u32 ctrl_cpu1; |
| |
| if (!(ctrl_cpu_rev[0].clr & CPU_SECONDARY) || |
| !(ctrl_cpu_rev[1].clr & CPU_VPID)) { |
| printf("\tVPID is not supported"); |
| return VMX_TEST_EXIT; |
| } |
| |
| ctrl_cpu1 = vmcs_read(CPU_EXEC_CTRL1); |
| ctrl_cpu1 |= CPU_VPID; |
| vmcs_write(CPU_EXEC_CTRL1, ctrl_cpu1); |
| return VMX_TEST_START; |
| } |
| |
| static void vpid_main() |
| { |
| vmx_set_test_stage(0); |
| vmcall(); |
| report("INVVPID SINGLE ADDRESS", vmx_get_test_stage() == 1); |
| vmx_set_test_stage(2); |
| vmcall(); |
| report("INVVPID SINGLE", vmx_get_test_stage() == 3); |
| vmx_set_test_stage(4); |
| vmcall(); |
| report("INVVPID ALL", vmx_get_test_stage() == 5); |
| } |
| |
| static int vpid_exit_handler() |
| { |
| u64 guest_rip; |
| ulong reason; |
| u32 insn_len; |
| |
| guest_rip = vmcs_read(GUEST_RIP); |
| reason = vmcs_read(EXI_REASON) & 0xff; |
| insn_len = vmcs_read(EXI_INST_LEN); |
| |
| switch (reason) { |
| case VMX_VMCALL: |
| switch(vmx_get_test_stage()) { |
| case 0: |
| if (!invvpid_test(INVVPID_SINGLE_ADDRESS, 1)) |
| vmx_inc_test_stage(); |
| break; |
| case 2: |
| if (!invvpid_test(INVVPID_SINGLE, 1)) |
| vmx_inc_test_stage(); |
| break; |
| case 4: |
| if (!invvpid_test(INVVPID_ALL, 1)) |
| vmx_inc_test_stage(); |
| break; |
| default: |
| printf("ERROR: unexpected stage, %d\n", |
| vmx_get_test_stage()); |
| print_vmexit_info(); |
| return VMX_TEST_VMEXIT; |
| } |
| vmcs_write(GUEST_RIP, guest_rip + insn_len); |
| return VMX_TEST_RESUME; |
| default: |
| printf("Unknown exit reason, %ld\n", reason); |
| print_vmexit_info(); |
| } |
| return VMX_TEST_VMEXIT; |
| } |
| |
| #define TIMER_VECTOR 222 |
| |
| static volatile bool timer_fired; |
| |
| static void timer_isr(isr_regs_t *regs) |
| { |
| timer_fired = true; |
| apic_write(APIC_EOI, 0); |
| } |
| |
| static int interrupt_init(struct vmcs *vmcs) |
| { |
| msr_bmp_init(); |
| vmcs_write(PIN_CONTROLS, vmcs_read(PIN_CONTROLS) & ~PIN_EXTINT); |
| handle_irq(TIMER_VECTOR, timer_isr); |
| return VMX_TEST_START; |
| } |
| |
| static void interrupt_main(void) |
| { |
| long long start, loops; |
| |
| vmx_set_test_stage(0); |
| |
| apic_write(APIC_LVTT, TIMER_VECTOR); |
| irq_enable(); |
| |
| apic_write(APIC_TMICT, 1); |
| for (loops = 0; loops < 10000000 && !timer_fired; loops++) |
| asm volatile ("nop"); |
| report("direct interrupt while running guest", timer_fired); |
| |
| apic_write(APIC_TMICT, 0); |
| irq_disable(); |
| vmcall(); |
| timer_fired = false; |
| apic_write(APIC_TMICT, 1); |
| for (loops = 0; loops < 10000000 && !timer_fired; loops++) |
| asm volatile ("nop"); |
| report("intercepted interrupt while running guest", timer_fired); |
| |
| irq_enable(); |
| apic_write(APIC_TMICT, 0); |
| irq_disable(); |
| vmcall(); |
| timer_fired = false; |
| start = rdtsc(); |
| apic_write(APIC_TMICT, 1000000); |
| |
| asm volatile ("sti; hlt"); |
| |
| report("direct interrupt + hlt", |
| rdtsc() - start > 1000000 && timer_fired); |
| |
| apic_write(APIC_TMICT, 0); |
| irq_disable(); |
| vmcall(); |
| timer_fired = false; |
| start = rdtsc(); |
| apic_write(APIC_TMICT, 1000000); |
| |
| asm volatile ("sti; hlt"); |
| |
| report("intercepted interrupt + hlt", |
| rdtsc() - start > 10000 && timer_fired); |
| |
| apic_write(APIC_TMICT, 0); |
| irq_disable(); |
| vmcall(); |
| timer_fired = false; |
| start = rdtsc(); |
| apic_write(APIC_TMICT, 1000000); |
| |
| irq_enable(); |
| asm volatile ("nop"); |
| vmcall(); |
| |
| report("direct interrupt + activity state hlt", |
| rdtsc() - start > 10000 && timer_fired); |
| |
| apic_write(APIC_TMICT, 0); |
| irq_disable(); |
| vmcall(); |
| timer_fired = false; |
| start = rdtsc(); |
| apic_write(APIC_TMICT, 1000000); |
| |
| irq_enable(); |
| asm volatile ("nop"); |
| vmcall(); |
| |
| report("intercepted interrupt + activity state hlt", |
| rdtsc() - start > 10000 && timer_fired); |
| |
| apic_write(APIC_TMICT, 0); |
| irq_disable(); |
| vmx_set_test_stage(7); |
| vmcall(); |
| timer_fired = false; |
| apic_write(APIC_TMICT, 1); |
| for (loops = 0; loops < 10000000 && !timer_fired; loops++) |
| asm volatile ("nop"); |
| report("running a guest with interrupt acknowledgement set", timer_fired); |
| } |
| |
| static int interrupt_exit_handler(void) |
| { |
| u64 guest_rip = vmcs_read(GUEST_RIP); |
| ulong reason = vmcs_read(EXI_REASON) & 0xff; |
| u32 insn_len = vmcs_read(EXI_INST_LEN); |
| |
| switch (reason) { |
| case VMX_VMCALL: |
| switch (vmx_get_test_stage()) { |
| case 0: |
| case 2: |
| case 5: |
| vmcs_write(PIN_CONTROLS, |
| vmcs_read(PIN_CONTROLS) | PIN_EXTINT); |
| break; |
| case 7: |
| vmcs_write(EXI_CONTROLS, vmcs_read(EXI_CONTROLS) | EXI_INTA); |
| vmcs_write(PIN_CONTROLS, |
| vmcs_read(PIN_CONTROLS) | PIN_EXTINT); |
| break; |
| case 1: |
| case 3: |
| vmcs_write(PIN_CONTROLS, |
| vmcs_read(PIN_CONTROLS) & ~PIN_EXTINT); |
| break; |
| case 4: |
| case 6: |
| vmcs_write(GUEST_ACTV_STATE, ACTV_HLT); |
| break; |
| } |
| vmx_inc_test_stage(); |
| vmcs_write(GUEST_RIP, guest_rip + insn_len); |
| return VMX_TEST_RESUME; |
| case VMX_EXTINT: |
| if (vmcs_read(EXI_CONTROLS) & EXI_INTA) { |
| int vector = vmcs_read(EXI_INTR_INFO) & 0xff; |
| handle_external_interrupt(vector); |
| } else { |
| irq_enable(); |
| asm volatile ("nop"); |
| irq_disable(); |
| } |
| if (vmx_get_test_stage() >= 2) |
| vmcs_write(GUEST_ACTV_STATE, ACTV_ACTIVE); |
| return VMX_TEST_RESUME; |
| default: |
| printf("Unknown exit reason, %ld\n", reason); |
| print_vmexit_info(); |
| } |
| |
| return VMX_TEST_VMEXIT; |
| } |
| |
| static int dbgctls_init(struct vmcs *vmcs) |
| { |
| u64 dr7 = 0x402; |
| u64 zero = 0; |
| |
| msr_bmp_init(); |
| asm volatile( |
| "mov %0,%%dr0\n\t" |
| "mov %0,%%dr1\n\t" |
| "mov %0,%%dr2\n\t" |
| "mov %1,%%dr7\n\t" |
| : : "r" (zero), "r" (dr7)); |
| wrmsr(MSR_IA32_DEBUGCTLMSR, 0x1); |
| vmcs_write(GUEST_DR7, 0x404); |
| vmcs_write(GUEST_DEBUGCTL, 0x2); |
| |
| vmcs_write(ENT_CONTROLS, vmcs_read(ENT_CONTROLS) | ENT_LOAD_DBGCTLS); |
| vmcs_write(EXI_CONTROLS, vmcs_read(EXI_CONTROLS) | EXI_SAVE_DBGCTLS); |
| |
| return VMX_TEST_START; |
| } |
| |
| static void dbgctls_main(void) |
| { |
| u64 dr7, debugctl; |
| |
| asm volatile("mov %%dr7,%0" : "=r" (dr7)); |
| debugctl = rdmsr(MSR_IA32_DEBUGCTLMSR); |
| /* Commented out: KVM does not support DEBUGCTL so far */ |
| (void)debugctl; |
| report("Load debug controls", dr7 == 0x404 /* && debugctl == 0x2 */); |
| |
| dr7 = 0x408; |
| asm volatile("mov %0,%%dr7" : : "r" (dr7)); |
| wrmsr(MSR_IA32_DEBUGCTLMSR, 0x3); |
| |
| vmx_set_test_stage(0); |
| vmcall(); |
| report("Save debug controls", vmx_get_test_stage() == 1); |
| |
| if (ctrl_enter_rev.set & ENT_LOAD_DBGCTLS || |
| ctrl_exit_rev.set & EXI_SAVE_DBGCTLS) { |
| printf("\tDebug controls are always loaded/saved\n"); |
| return; |
| } |
| vmx_set_test_stage(2); |
| vmcall(); |
| |
| asm volatile("mov %%dr7,%0" : "=r" (dr7)); |
| debugctl = rdmsr(MSR_IA32_DEBUGCTLMSR); |
| /* Commented out: KVM does not support DEBUGCTL so far */ |
| (void)debugctl; |
| report("Guest=host debug controls", dr7 == 0x402 /* && debugctl == 0x1 */); |
| |
| dr7 = 0x408; |
| asm volatile("mov %0,%%dr7" : : "r" (dr7)); |
| wrmsr(MSR_IA32_DEBUGCTLMSR, 0x3); |
| |
| vmx_set_test_stage(3); |
| vmcall(); |
| report("Don't save debug controls", vmx_get_test_stage() == 4); |
| } |
| |
| static int dbgctls_exit_handler(void) |
| { |
| unsigned int reason = vmcs_read(EXI_REASON) & 0xff; |
| u32 insn_len = vmcs_read(EXI_INST_LEN); |
| u64 guest_rip = vmcs_read(GUEST_RIP); |
| u64 dr7, debugctl; |
| |
| asm volatile("mov %%dr7,%0" : "=r" (dr7)); |
| debugctl = rdmsr(MSR_IA32_DEBUGCTLMSR); |
| |
| switch (reason) { |
| case VMX_VMCALL: |
| switch (vmx_get_test_stage()) { |
| case 0: |
| if (dr7 == 0x400 && debugctl == 0 && |
| vmcs_read(GUEST_DR7) == 0x408 /* && |
| Commented out: KVM does not support DEBUGCTL so far |
| vmcs_read(GUEST_DEBUGCTL) == 0x3 */) |
| vmx_inc_test_stage(); |
| break; |
| case 2: |
| dr7 = 0x402; |
| asm volatile("mov %0,%%dr7" : : "r" (dr7)); |
| wrmsr(MSR_IA32_DEBUGCTLMSR, 0x1); |
| vmcs_write(GUEST_DR7, 0x404); |
| vmcs_write(GUEST_DEBUGCTL, 0x2); |
| |
| vmcs_write(ENT_CONTROLS, |
| vmcs_read(ENT_CONTROLS) & ~ENT_LOAD_DBGCTLS); |
| vmcs_write(EXI_CONTROLS, |
| vmcs_read(EXI_CONTROLS) & ~EXI_SAVE_DBGCTLS); |
| break; |
| case 3: |
| if (dr7 == 0x400 && debugctl == 0 && |
| vmcs_read(GUEST_DR7) == 0x404 /* && |
| Commented out: KVM does not support DEBUGCTL so far |
| vmcs_read(GUEST_DEBUGCTL) == 0x2 */) |
| vmx_inc_test_stage(); |
| break; |
| } |
| vmcs_write(GUEST_RIP, guest_rip + insn_len); |
| return VMX_TEST_RESUME; |
| default: |
| printf("Unknown exit reason, %d\n", reason); |
| print_vmexit_info(); |
| } |
| return VMX_TEST_VMEXIT; |
| } |
| |
| struct vmx_msr_entry { |
| u32 index; |
| u32 reserved; |
| u64 value; |
| } __attribute__((packed)); |
| |
| #define MSR_MAGIC 0x31415926 |
| struct vmx_msr_entry *exit_msr_store, *entry_msr_load, *exit_msr_load; |
| |
| static int msr_switch_init(struct vmcs *vmcs) |
| { |
| msr_bmp_init(); |
| exit_msr_store = alloc_page(); |
| exit_msr_load = alloc_page(); |
| entry_msr_load = alloc_page(); |
| memset(exit_msr_store, 0, PAGE_SIZE); |
| memset(exit_msr_load, 0, PAGE_SIZE); |
| memset(entry_msr_load, 0, PAGE_SIZE); |
| entry_msr_load[0].index = MSR_KERNEL_GS_BASE; |
| entry_msr_load[0].value = MSR_MAGIC; |
| |
| vmx_set_test_stage(1); |
| vmcs_write(ENT_MSR_LD_CNT, 1); |
| vmcs_write(ENTER_MSR_LD_ADDR, (u64)entry_msr_load); |
| vmcs_write(EXI_MSR_ST_CNT, 1); |
| vmcs_write(EXIT_MSR_ST_ADDR, (u64)exit_msr_store); |
| vmcs_write(EXI_MSR_LD_CNT, 1); |
| vmcs_write(EXIT_MSR_LD_ADDR, (u64)exit_msr_load); |
| return VMX_TEST_START; |
| } |
| |
| static void msr_switch_main() |
| { |
| if (vmx_get_test_stage() == 1) { |
| report("VM entry MSR load", |
| rdmsr(MSR_KERNEL_GS_BASE) == MSR_MAGIC); |
| vmx_set_test_stage(2); |
| wrmsr(MSR_KERNEL_GS_BASE, MSR_MAGIC + 1); |
| exit_msr_store[0].index = MSR_KERNEL_GS_BASE; |
| exit_msr_load[0].index = MSR_KERNEL_GS_BASE; |
| exit_msr_load[0].value = MSR_MAGIC + 2; |
| } |
| vmcall(); |
| } |
| |
| static int msr_switch_exit_handler() |
| { |
| ulong reason; |
| |
| reason = vmcs_read(EXI_REASON); |
| if (reason == VMX_VMCALL && vmx_get_test_stage() == 2) { |
| report("VM exit MSR store", |
| exit_msr_store[0].value == MSR_MAGIC + 1); |
| report("VM exit MSR load", |
| rdmsr(MSR_KERNEL_GS_BASE) == MSR_MAGIC + 2); |
| vmx_set_test_stage(3); |
| entry_msr_load[0].index = MSR_FS_BASE; |
| return VMX_TEST_RESUME; |
| } |
| printf("ERROR %s: unexpected stage=%u or reason=%lu\n", |
| __func__, vmx_get_test_stage(), reason); |
| return VMX_TEST_EXIT; |
| } |
| |
| static int msr_switch_entry_failure(struct vmentry_failure *failure) |
| { |
| ulong reason; |
| |
| if (failure->early) { |
| printf("ERROR %s: early exit\n", __func__); |
| return VMX_TEST_EXIT; |
| } |
| |
| reason = vmcs_read(EXI_REASON); |
| if (reason == (VMX_ENTRY_FAILURE | VMX_FAIL_MSR) && |
| vmx_get_test_stage() == 3) { |
| report("VM entry MSR load: try to load FS_BASE", |
| vmcs_read(EXI_QUALIFICATION) == 1); |
| return VMX_TEST_VMEXIT; |
| } |
| printf("ERROR %s: unexpected stage=%u or reason=%lu\n", |
| __func__, vmx_get_test_stage(), reason); |
| return VMX_TEST_EXIT; |
| } |
| |
| static int vmmcall_init(struct vmcs *vmcs ) |
| { |
| vmcs_write(EXC_BITMAP, 1 << UD_VECTOR); |
| return VMX_TEST_START; |
| } |
| |
| static void vmmcall_main(void) |
| { |
| asm volatile( |
| "mov $0xABCD, %%rax\n\t" |
| "vmmcall\n\t" |
| ::: "rax"); |
| |
| report("VMMCALL", 0); |
| } |
| |
| static int vmmcall_exit_handler() |
| { |
| ulong reason; |
| |
| reason = vmcs_read(EXI_REASON); |
| switch (reason) { |
| case VMX_VMCALL: |
| printf("here\n"); |
| report("VMMCALL triggers #UD", 0); |
| break; |
| case VMX_EXC_NMI: |
| report("VMMCALL triggers #UD", |
| (vmcs_read(EXI_INTR_INFO) & 0xff) == UD_VECTOR); |
| break; |
| default: |
| printf("Unknown exit reason, %ld\n", reason); |
| print_vmexit_info(); |
| } |
| |
| return VMX_TEST_VMEXIT; |
| } |
| |
| static int disable_rdtscp_init(struct vmcs *vmcs) |
| { |
| u32 ctrl_cpu1; |
| |
| if (ctrl_cpu_rev[0].clr & CPU_SECONDARY) { |
| ctrl_cpu1 = vmcs_read(CPU_EXEC_CTRL1); |
| ctrl_cpu1 &= ~CPU_RDTSCP; |
| vmcs_write(CPU_EXEC_CTRL1, ctrl_cpu1); |
| } |
| |
| return VMX_TEST_START; |
| } |
| |
| static void disable_rdtscp_ud_handler(struct ex_regs *regs) |
| { |
| switch (vmx_get_test_stage()) { |
| case 0: |
| report("RDTSCP triggers #UD", true); |
| vmx_inc_test_stage(); |
| regs->rip += 3; |
| break; |
| case 2: |
| report("RDPID triggers #UD", true); |
| vmx_inc_test_stage(); |
| regs->rip += 4; |
| break; |
| } |
| return; |
| |
| } |
| |
| static void disable_rdtscp_main(void) |
| { |
| /* Test that #UD is properly injected in L2. */ |
| handle_exception(UD_VECTOR, disable_rdtscp_ud_handler); |
| |
| vmx_set_test_stage(0); |
| asm volatile("rdtscp" : : : "eax", "ecx", "edx"); |
| vmcall(); |
| asm volatile(".byte 0xf3, 0x0f, 0xc7, 0xf8" : : : "eax"); |
| vmcall(); |
| } |
| |
| static int disable_rdtscp_exit_handler(void) |
| { |
| unsigned int reason = vmcs_read(EXI_REASON) & 0xff; |
| |
| switch (reason) { |
| case VMX_VMCALL: |
| switch (vmx_get_test_stage()) { |
| case 0: |
| report("RDTSCP triggers #UD", false); |
| vmx_inc_test_stage(); |
| /* fallthrough */ |
| case 1: |
| vmx_inc_test_stage(); |
| vmcs_write(GUEST_RIP, vmcs_read(GUEST_RIP) + 3); |
| return VMX_TEST_RESUME; |
| case 2: |
| report("RDPID triggers #UD", false); |
| break; |
| } |
| break; |
| |
| default: |
| printf("Unknown exit reason, %d\n", reason); |
| print_vmexit_info(); |
| } |
| return VMX_TEST_VMEXIT; |
| } |
| |
| int int3_init() |
| { |
| vmcs_write(EXC_BITMAP, ~0u); |
| return VMX_TEST_START; |
| } |
| |
| void int3_guest_main() |
| { |
| asm volatile ("int3"); |
| } |
| |
| int int3_exit_handler() |
| { |
| u32 reason = vmcs_read(EXI_REASON); |
| u32 intr_info = vmcs_read(EXI_INTR_INFO); |
| |
| report("L1 intercepts #BP", reason == VMX_EXC_NMI && |
| (intr_info & INTR_INFO_VALID_MASK) && |
| (intr_info & INTR_INFO_VECTOR_MASK) == BP_VECTOR && |
| ((intr_info & INTR_INFO_INTR_TYPE_MASK) >> |
| INTR_INFO_INTR_TYPE_SHIFT) == VMX_INTR_TYPE_SOFT_EXCEPTION); |
| |
| return VMX_TEST_VMEXIT; |
| } |
| |
| int into_init() |
| { |
| vmcs_write(EXC_BITMAP, ~0u); |
| return VMX_TEST_START; |
| } |
| |
| void into_guest_main() |
| { |
| struct far_pointer32 fp = { |
| .offset = (uintptr_t)&&into, |
| .selector = KERNEL_CS32, |
| }; |
| register uintptr_t rsp asm("rsp"); |
| |
| if (fp.offset != (uintptr_t)&&into) { |
| printf("Code address too high.\n"); |
| return; |
| } |
| if ((u32)rsp != rsp) { |
| printf("Stack address too high.\n"); |
| return; |
| } |
| |
| asm goto ("lcall *%0" : : "m" (fp) : "rax" : into); |
| return; |
| into: |
| asm volatile (".code32;" |
| "movl $0x7fffffff, %eax;" |
| "addl %eax, %eax;" |
| "into;" |
| "lret;" |
| ".code64"); |
| __builtin_unreachable(); |
| } |
| |
| int into_exit_handler() |
| { |
| u32 reason = vmcs_read(EXI_REASON); |
| u32 intr_info = vmcs_read(EXI_INTR_INFO); |
| |
| report("L1 intercepts #OF", reason == VMX_EXC_NMI && |
| (intr_info & INTR_INFO_VALID_MASK) && |
| (intr_info & INTR_INFO_VECTOR_MASK) == OF_VECTOR && |
| ((intr_info & INTR_INFO_INTR_TYPE_MASK) >> |
| INTR_INFO_INTR_TYPE_SHIFT) == VMX_INTR_TYPE_SOFT_EXCEPTION); |
| |
| return VMX_TEST_VMEXIT; |
| } |
| |
| /* name/init/guest_main/exit_handler/syscall_handler/guest_regs */ |
| struct vmx_test vmx_tests[] = { |
| { "null", NULL, basic_guest_main, basic_exit_handler, NULL, {0} }, |
| { "vmenter", NULL, vmenter_main, vmenter_exit_handler, NULL, {0} }, |
| { "preemption timer", preemption_timer_init, preemption_timer_main, |
| preemption_timer_exit_handler, NULL, {0} }, |
| { "control field PAT", test_ctrl_pat_init, test_ctrl_pat_main, |
| test_ctrl_pat_exit_handler, NULL, {0} }, |
| { "control field EFER", test_ctrl_efer_init, test_ctrl_efer_main, |
| test_ctrl_efer_exit_handler, NULL, {0} }, |
| { "CR shadowing", NULL, cr_shadowing_main, |
| cr_shadowing_exit_handler, NULL, {0} }, |
| { "I/O bitmap", iobmp_init, iobmp_main, iobmp_exit_handler, |
| NULL, {0} }, |
| { "instruction intercept", insn_intercept_init, insn_intercept_main, |
| insn_intercept_exit_handler, NULL, {0} }, |
| { "EPT framework", ept_init, ept_main, ept_exit_handler, NULL, {0} }, |
| { "VPID", vpid_init, vpid_main, vpid_exit_handler, NULL, {0} }, |
| { "interrupt", interrupt_init, interrupt_main, |
| interrupt_exit_handler, NULL, {0} }, |
| { "debug controls", dbgctls_init, dbgctls_main, dbgctls_exit_handler, |
| NULL, {0} }, |
| { "MSR switch", msr_switch_init, msr_switch_main, |
| msr_switch_exit_handler, NULL, {0}, msr_switch_entry_failure }, |
| { "vmmcall", vmmcall_init, vmmcall_main, vmmcall_exit_handler, NULL, {0} }, |
| { "disable RDTSCP", disable_rdtscp_init, disable_rdtscp_main, |
| disable_rdtscp_exit_handler, NULL, {0} }, |
| { "int3", int3_init, int3_guest_main, int3_exit_handler, NULL, {0} }, |
| { "into", into_init, into_guest_main, into_exit_handler, NULL, {0} }, |
| { NULL, NULL, NULL, NULL, NULL, {0} }, |
| }; |