/* SPDX-License-Identifier: GPL-2.0 */ /* * tools/testing/selftests/kvm/include/vmx.h * * Copyright (C) 2018, Red Hat, Inc. * */ #ifndef SELFTEST_KVM_EVMCS_H #define SELFTEST_KVM_EVMCS_H #include <stdint.h> #include "vmx.h" #define u16 uint16_t #define u32 uint32_t #define u64 uint64_t extern bool enable_evmcs; struct hv_vp_assist_page { __u32 apic_assist; __u32 reserved; __u64 vtl_control[2]; __u64 nested_enlightenments_control[2]; __u32 enlighten_vmentry; __u64 current_nested_vmcs; }; struct hv_enlightened_vmcs { u32 revision_id; u32 abort; u16 host_es_selector; u16 host_cs_selector; u16 host_ss_selector; u16 host_ds_selector; u16 host_fs_selector; u16 host_gs_selector; u16 host_tr_selector; u64 host_ia32_pat; u64 host_ia32_efer; u64 host_cr0; u64 host_cr3; u64 host_cr4; u64 host_ia32_sysenter_esp; u64 host_ia32_sysenter_eip; u64 host_rip; u32 host_ia32_sysenter_cs; u32 pin_based_vm_exec_control; u32 vm_exit_controls; u32 secondary_vm_exec_control; u64 io_bitmap_a; u64 io_bitmap_b; u64 msr_bitmap; u16 guest_es_selector; u16 guest_cs_selector; u16 guest_ss_selector; u16 guest_ds_selector; u16 guest_fs_selector; u16 guest_gs_selector; u16 guest_ldtr_selector; u16 guest_tr_selector; u32 guest_es_limit; u32 guest_cs_limit; u32 guest_ss_limit; u32 guest_ds_limit; u32 guest_fs_limit; u32 guest_gs_limit; u32 guest_ldtr_limit; u32 guest_tr_limit; u32 guest_gdtr_limit; u32 guest_idtr_limit; u32 guest_es_ar_bytes; u32 guest_cs_ar_bytes; u32 guest_ss_ar_bytes; u32 guest_ds_ar_bytes; u32 guest_fs_ar_bytes; u32 guest_gs_ar_bytes; u32 guest_ldtr_ar_bytes; u32 guest_tr_ar_bytes; u64 guest_es_base; u64 guest_cs_base; u64 guest_ss_base; u64 guest_ds_base; u64 guest_fs_base; u64 guest_gs_base; u64 guest_ldtr_base; u64 guest_tr_base; u64 guest_gdtr_base; u64 guest_idtr_base; u64 padding64_1[3]; u64 vm_exit_msr_store_addr; u64 vm_exit_msr_load_addr; u64 vm_entry_msr_load_addr; u64 cr3_target_value0; u64 cr3_target_value1; u64 cr3_target_value2; u64 cr3_target_value3; u32 page_fault_error_code_mask; u32 page_fault_error_code_match; u32 cr3_target_count; u32 vm_exit_msr_store_count; u32 vm_exit_msr_load_count; u32 vm_entry_msr_load_count; u64 tsc_offset; u64 virtual_apic_page_addr; u64 vmcs_link_pointer; u64 guest_ia32_debugctl; u64 guest_ia32_pat; u64 guest_ia32_efer; u64 guest_pdptr0; u64 guest_pdptr1; u64 guest_pdptr2; u64 guest_pdptr3; u64 guest_pending_dbg_exceptions; u64 guest_sysenter_esp; u64 guest_sysenter_eip; u32 guest_activity_state; u32 guest_sysenter_cs; u64 cr0_guest_host_mask; u64 cr4_guest_host_mask; u64 cr0_read_shadow; u64 cr4_read_shadow; u64 guest_cr0; u64 guest_cr3; u64 guest_cr4; u64 guest_dr7; u64 host_fs_base; u64 host_gs_base; u64 host_tr_base; u64 host_gdtr_base; u64 host_idtr_base; u64 host_rsp; u64 ept_pointer; u16 virtual_processor_id; u16 padding16[3]; u64 padding64_2[5]; u64 guest_physical_address; u32 vm_instruction_error; u32 vm_exit_reason; u32 vm_exit_intr_info; u32 vm_exit_intr_error_code; u32 idt_vectoring_info_field; u32 idt_vectoring_error_code; u32 vm_exit_instruction_len; u32 vmx_instruction_info; u64 exit_qualification; u64 exit_io_instruction_ecx; u64 exit_io_instruction_esi; u64 exit_io_instruction_edi; u64 exit_io_instruction_eip; u64 guest_linear_address; u64 guest_rsp; u64 guest_rflags; u32 guest_interruptibility_info; u32 cpu_based_vm_exec_control; u32 exception_bitmap; u32 vm_entry_controls; u32 vm_entry_intr_info_field; u32 vm_entry_exception_error_code; u32 vm_entry_instruction_len; u32 tpr_threshold; u64 guest_rip; u32 hv_clean_fields; u32 hv_padding_32; u32 hv_synthetic_controls; struct { u32 nested_flush_hypercall:1; u32 msr_bitmap:1; u32 reserved:30; } hv_enlightenments_control; u32 hv_vp_id; u64 hv_vm_id; u64 partition_assist_page; u64 padding64_4[4]; u64 guest_bndcfgs; u64 padding64_5[7]; u64 xss_exit_bitmap; u64 padding64_6[7]; }; #define HV_X64_MSR_VP_ASSIST_PAGE 0x40000073 #define HV_X64_MSR_VP_ASSIST_PAGE_ENABLE 0x00000001 #define HV_X64_MSR_VP_ASSIST_PAGE_ADDRESS_SHIFT 12 #define HV_X64_MSR_VP_ASSIST_PAGE_ADDRESS_MASK \ (~((1ull << HV_X64_MSR_VP_ASSIST_PAGE_ADDRESS_SHIFT) - 1)) struct hv_enlightened_vmcs *current_evmcs; struct hv_vp_assist_page *current_vp_assist; static inline int enable_vp_assist(uint64_t vp_assist_pa, void *vp_assist) { u64 val = (vp_assist_pa & HV_X64_MSR_VP_ASSIST_PAGE_ADDRESS_MASK) | HV_X64_MSR_VP_ASSIST_PAGE_ENABLE; wrmsr(HV_X64_MSR_VP_ASSIST_PAGE, val); current_vp_assist = vp_assist; enable_evmcs = true; return 0; } static inline int evmcs_vmptrld(uint64_t vmcs_pa, void *vmcs) { current_vp_assist->current_nested_vmcs = vmcs_pa; current_vp_assist->enlighten_vmentry = 1; current_evmcs = vmcs; return 0; } static inline int evmcs_vmptrst(uint64_t *value) { *value = current_vp_assist->current_nested_vmcs & ~HV_X64_MSR_VP_ASSIST_PAGE_ENABLE; return 0; } static inline int evmcs_vmread(uint64_t encoding, uint64_t *value) { switch (encoding) { case GUEST_RIP: *value = current_evmcs->guest_rip; break; case GUEST_RSP: *value = current_evmcs->guest_rsp; break; case GUEST_RFLAGS: *value = current_evmcs->guest_rflags; break; case HOST_IA32_PAT: *value = current_evmcs->host_ia32_pat; break; case HOST_IA32_EFER: *value = current_evmcs->host_ia32_efer; break; case HOST_CR0: *value = current_evmcs->host_cr0; break; case HOST_CR3: *value = current_evmcs->host_cr3; break; case HOST_CR4: *value = current_evmcs->host_cr4; break; case HOST_IA32_SYSENTER_ESP: *value = current_evmcs->host_ia32_sysenter_esp; break; case HOST_IA32_SYSENTER_EIP: *value = current_evmcs->host_ia32_sysenter_eip; break; case HOST_RIP: *value = current_evmcs->host_rip; break; case IO_BITMAP_A: *value = current_evmcs->io_bitmap_a; break; case IO_BITMAP_B: *value = current_evmcs->io_bitmap_b; break; case MSR_BITMAP: *value = current_evmcs->msr_bitmap; break; case GUEST_ES_BASE: *value = current_evmcs->guest_es_base; break; case GUEST_CS_BASE: *value = current_evmcs->guest_cs_base; break; case GUEST_SS_BASE: *value = current_evmcs->guest_ss_base; break; case GUEST_DS_BASE: *value = current_evmcs->guest_ds_base; break; case GUEST_FS_BASE: *value = current_evmcs->guest_fs_base; break; case GUEST_GS_BASE: *value = current_evmcs->guest_gs_base; break; case GUEST_LDTR_BASE: *value = current_evmcs->guest_ldtr_base; break; case GUEST_TR_BASE: *value = current_evmcs->guest_tr_base; break; case GUEST_GDTR_BASE: *value = current_evmcs->guest_gdtr_base; break; case GUEST_IDTR_BASE: *value = current_evmcs->guest_idtr_base; break; case TSC_OFFSET: *value = current_evmcs->tsc_offset; break; case VIRTUAL_APIC_PAGE_ADDR: *value = current_evmcs->virtual_apic_page_addr; break; case VMCS_LINK_POINTER: *value = current_evmcs->vmcs_link_pointer; break; case GUEST_IA32_DEBUGCTL: *value = current_evmcs->guest_ia32_debugctl; break; case GUEST_IA32_PAT: *value = current_evmcs->guest_ia32_pat; break; case GUEST_IA32_EFER: *value = current_evmcs->guest_ia32_efer; break; case GUEST_PDPTR0: *value = current_evmcs->guest_pdptr0; break; case GUEST_PDPTR1: *value = current_evmcs->guest_pdptr1; break; case GUEST_PDPTR2: *value = current_evmcs->guest_pdptr2; break; case GUEST_PDPTR3: *value = current_evmcs->guest_pdptr3; break; case GUEST_PENDING_DBG_EXCEPTIONS: *value = current_evmcs->guest_pending_dbg_exceptions; break; case GUEST_SYSENTER_ESP: *value = current_evmcs->guest_sysenter_esp; break; case GUEST_SYSENTER_EIP: *value = current_evmcs->guest_sysenter_eip; break; case CR0_GUEST_HOST_MASK: *value = current_evmcs->cr0_guest_host_mask; break; case CR4_GUEST_HOST_MASK: *value = current_evmcs->cr4_guest_host_mask; break; case CR0_READ_SHADOW: *value = current_evmcs->cr0_read_shadow; break; case CR4_READ_SHADOW: *value = current_evmcs->cr4_read_shadow; break; case GUEST_CR0: *value = current_evmcs->guest_cr0; break; case GUEST_CR3: *value = current_evmcs->guest_cr3; break; case GUEST_CR4: *value = current_evmcs->guest_cr4; break; case GUEST_DR7: *value = current_evmcs->guest_dr7; break; case HOST_FS_BASE: *value = current_evmcs->host_fs_base; break; case HOST_GS_BASE: *value = current_evmcs->host_gs_base; break; case HOST_TR_BASE: *value = current_evmcs->host_tr_base; break; case HOST_GDTR_BASE: *value = current_evmcs->host_gdtr_base; break; case HOST_IDTR_BASE: *value = current_evmcs->host_idtr_base; break; case HOST_RSP: *value = current_evmcs->host_rsp; break; case EPT_POINTER: *value = current_evmcs->ept_pointer; break; case GUEST_BNDCFGS: *value = current_evmcs->guest_bndcfgs; break; case XSS_EXIT_BITMAP: *value = current_evmcs->xss_exit_bitmap; break; case GUEST_PHYSICAL_ADDRESS: *value = current_evmcs->guest_physical_address; break; case EXIT_QUALIFICATION: *value = current_evmcs->exit_qualification; break; case GUEST_LINEAR_ADDRESS: *value = current_evmcs->guest_linear_address; break; case VM_EXIT_MSR_STORE_ADDR: *value = current_evmcs->vm_exit_msr_store_addr; break; case VM_EXIT_MSR_LOAD_ADDR: *value = current_evmcs->vm_exit_msr_load_addr; break; case VM_ENTRY_MSR_LOAD_ADDR: *value = current_evmcs->vm_entry_msr_load_addr; break; case CR3_TARGET_VALUE0: *value = current_evmcs->cr3_target_value0; break; case CR3_TARGET_VALUE1: *value = current_evmcs->cr3_target_value1; break; case CR3_TARGET_VALUE2: *value = current_evmcs->cr3_target_value2; break; case CR3_TARGET_VALUE3: *value = current_evmcs->cr3_target_value3; break; case TPR_THRESHOLD: *value = current_evmcs->tpr_threshold; break; case GUEST_INTERRUPTIBILITY_INFO: *value = current_evmcs->guest_interruptibility_info; break; case CPU_BASED_VM_EXEC_CONTROL: *value = current_evmcs->cpu_based_vm_exec_control; break; case EXCEPTION_BITMAP: *value = current_evmcs->exception_bitmap; break; case VM_ENTRY_CONTROLS: *value = current_evmcs->vm_entry_controls; break; case VM_ENTRY_INTR_INFO_FIELD: *value = current_evmcs->vm_entry_intr_info_field; break; case VM_ENTRY_EXCEPTION_ERROR_CODE: *value = current_evmcs->vm_entry_exception_error_code; break; case VM_ENTRY_INSTRUCTION_LEN: *value = current_evmcs->vm_entry_instruction_len; break; case HOST_IA32_SYSENTER_CS: *value = current_evmcs->host_ia32_sysenter_cs; break; case PIN_BASED_VM_EXEC_CONTROL: *value = current_evmcs->pin_based_vm_exec_control; break; case VM_EXIT_CONTROLS: *value = current_evmcs->vm_exit_controls; break; case SECONDARY_VM_EXEC_CONTROL: *value = current_evmcs->secondary_vm_exec_control; break; case GUEST_ES_LIMIT: *value = current_evmcs->guest_es_limit; break; case GUEST_CS_LIMIT: *value = current_evmcs->guest_cs_limit; break; case GUEST_SS_LIMIT: *value = current_evmcs->guest_ss_limit; break; case GUEST_DS_LIMIT: *value = current_evmcs->guest_ds_limit; break; case GUEST_FS_LIMIT: *value = current_evmcs->guest_fs_limit; break; case GUEST_GS_LIMIT: *value = current_evmcs->guest_gs_limit; break; case GUEST_LDTR_LIMIT: *value = current_evmcs->guest_ldtr_limit; break; case GUEST_TR_LIMIT: *value = current_evmcs->guest_tr_limit; break; case GUEST_GDTR_LIMIT: *value = current_evmcs->guest_gdtr_limit; break; case GUEST_IDTR_LIMIT: *value = current_evmcs->guest_idtr_limit; break; case GUEST_ES_AR_BYTES: *value = current_evmcs->guest_es_ar_bytes; break; case GUEST_CS_AR_BYTES: *value = current_evmcs->guest_cs_ar_bytes; break; case GUEST_SS_AR_BYTES: *value = current_evmcs->guest_ss_ar_bytes; break; case GUEST_DS_AR_BYTES: *value = current_evmcs->guest_ds_ar_bytes; break; case GUEST_FS_AR_BYTES: *value = current_evmcs->guest_fs_ar_bytes; break; case GUEST_GS_AR_BYTES: *value = current_evmcs->guest_gs_ar_bytes; break; case GUEST_LDTR_AR_BYTES: *value = current_evmcs->guest_ldtr_ar_bytes; break; case GUEST_TR_AR_BYTES: *value = current_evmcs->guest_tr_ar_bytes; break; case GUEST_ACTIVITY_STATE: *value = current_evmcs->guest_activity_state; break; case GUEST_SYSENTER_CS: *value = current_evmcs->guest_sysenter_cs; break; case VM_INSTRUCTION_ERROR: *value = current_evmcs->vm_instruction_error; break; case VM_EXIT_REASON: *value = current_evmcs->vm_exit_reason; break; case VM_EXIT_INTR_INFO: *value = current_evmcs->vm_exit_intr_info; break; case VM_EXIT_INTR_ERROR_CODE: *value = current_evmcs->vm_exit_intr_error_code; break; case IDT_VECTORING_INFO_FIELD: *value = current_evmcs->idt_vectoring_info_field; break; case IDT_VECTORING_ERROR_CODE: *value = current_evmcs->idt_vectoring_error_code; break; case VM_EXIT_INSTRUCTION_LEN: *value = current_evmcs->vm_exit_instruction_len; break; case VMX_INSTRUCTION_INFO: *value = current_evmcs->vmx_instruction_info; break; case PAGE_FAULT_ERROR_CODE_MASK: *value = current_evmcs->page_fault_error_code_mask; break; case PAGE_FAULT_ERROR_CODE_MATCH: *value = current_evmcs->page_fault_error_code_match; break; case CR3_TARGET_COUNT: *value = current_evmcs->cr3_target_count; break; case VM_EXIT_MSR_STORE_COUNT: *value = current_evmcs->vm_exit_msr_store_count; break; case VM_EXIT_MSR_LOAD_COUNT: *value = current_evmcs->vm_exit_msr_load_count; break; case VM_ENTRY_MSR_LOAD_COUNT: *value = current_evmcs->vm_entry_msr_load_count; break; case HOST_ES_SELECTOR: *value = current_evmcs->host_es_selector; break; case HOST_CS_SELECTOR: *value = current_evmcs->host_cs_selector; break; case HOST_SS_SELECTOR: *value = current_evmcs->host_ss_selector; break; case HOST_DS_SELECTOR: *value = current_evmcs->host_ds_selector; break; case HOST_FS_SELECTOR: *value = current_evmcs->host_fs_selector; break; case HOST_GS_SELECTOR: *value = current_evmcs->host_gs_selector; break; case HOST_TR_SELECTOR: *value = current_evmcs->host_tr_selector; break; case GUEST_ES_SELECTOR: *value = current_evmcs->guest_es_selector; break; case GUEST_CS_SELECTOR: *value = current_evmcs->guest_cs_selector; break; case GUEST_SS_SELECTOR: *value = current_evmcs->guest_ss_selector; break; case GUEST_DS_SELECTOR: *value = current_evmcs->guest_ds_selector; break; case GUEST_FS_SELECTOR: *value = current_evmcs->guest_fs_selector; break; case GUEST_GS_SELECTOR: *value = current_evmcs->guest_gs_selector; break; case GUEST_LDTR_SELECTOR: *value = current_evmcs->guest_ldtr_selector; break; case GUEST_TR_SELECTOR: *value = current_evmcs->guest_tr_selector; break; case VIRTUAL_PROCESSOR_ID: *value = current_evmcs->virtual_processor_id; break; default: return 1; } return 0; } static inline int evmcs_vmwrite(uint64_t encoding, uint64_t value) { switch (encoding) { case GUEST_RIP: current_evmcs->guest_rip = value; break; case GUEST_RSP: current_evmcs->guest_rsp = value; break; case GUEST_RFLAGS: current_evmcs->guest_rflags = value; break; case HOST_IA32_PAT: current_evmcs->host_ia32_pat = value; break; case HOST_IA32_EFER: current_evmcs->host_ia32_efer = value; break; case HOST_CR0: current_evmcs->host_cr0 = value; break; case HOST_CR3: current_evmcs->host_cr3 = value; break; case HOST_CR4: current_evmcs->host_cr4 = value; break; case HOST_IA32_SYSENTER_ESP: current_evmcs->host_ia32_sysenter_esp = value; break; case HOST_IA32_SYSENTER_EIP: current_evmcs->host_ia32_sysenter_eip = value; break; case HOST_RIP: current_evmcs->host_rip = value; break; case IO_BITMAP_A: current_evmcs->io_bitmap_a = value; break; case IO_BITMAP_B: current_evmcs->io_bitmap_b = value; break; case MSR_BITMAP: current_evmcs->msr_bitmap = value; break; case GUEST_ES_BASE: current_evmcs->guest_es_base = value; break; case GUEST_CS_BASE: current_evmcs->guest_cs_base = value; break; case GUEST_SS_BASE: current_evmcs->guest_ss_base = value; break; case GUEST_DS_BASE: current_evmcs->guest_ds_base = value; break; case GUEST_FS_BASE: current_evmcs->guest_fs_base = value; break; case GUEST_GS_BASE: current_evmcs->guest_gs_base = value; break; case GUEST_LDTR_BASE: current_evmcs->guest_ldtr_base = value; break; case GUEST_TR_BASE: current_evmcs->guest_tr_base = value; break; case GUEST_GDTR_BASE: current_evmcs->guest_gdtr_base = value; break; case GUEST_IDTR_BASE: current_evmcs->guest_idtr_base = value; break; case TSC_OFFSET: current_evmcs->tsc_offset = value; break; case VIRTUAL_APIC_PAGE_ADDR: current_evmcs->virtual_apic_page_addr = value; break; case VMCS_LINK_POINTER: current_evmcs->vmcs_link_pointer = value; break; case GUEST_IA32_DEBUGCTL: current_evmcs->guest_ia32_debugctl = value; break; case GUEST_IA32_PAT: current_evmcs->guest_ia32_pat = value; break; case GUEST_IA32_EFER: current_evmcs->guest_ia32_efer = value; break; case GUEST_PDPTR0: current_evmcs->guest_pdptr0 = value; break; case GUEST_PDPTR1: current_evmcs->guest_pdptr1 = value; break; case GUEST_PDPTR2: current_evmcs->guest_pdptr2 = value; break; case GUEST_PDPTR3: current_evmcs->guest_pdptr3 = value; break; case GUEST_PENDING_DBG_EXCEPTIONS: current_evmcs->guest_pending_dbg_exceptions = value; break; case GUEST_SYSENTER_ESP: current_evmcs->guest_sysenter_esp = value; break; case GUEST_SYSENTER_EIP: current_evmcs->guest_sysenter_eip = value; break; case CR0_GUEST_HOST_MASK: current_evmcs->cr0_guest_host_mask = value; break; case CR4_GUEST_HOST_MASK: current_evmcs->cr4_guest_host_mask = value; break; case CR0_READ_SHADOW: current_evmcs->cr0_read_shadow = value; break; case CR4_READ_SHADOW: current_evmcs->cr4_read_shadow = value; break; case GUEST_CR0: current_evmcs->guest_cr0 = value; break; case GUEST_CR3: current_evmcs->guest_cr3 = value; break; case GUEST_CR4: current_evmcs->guest_cr4 = value; break; case GUEST_DR7: current_evmcs->guest_dr7 = value; break; case HOST_FS_BASE: current_evmcs->host_fs_base = value; break; case HOST_GS_BASE: current_evmcs->host_gs_base = value; break; case HOST_TR_BASE: current_evmcs->host_tr_base = value; break; case HOST_GDTR_BASE: current_evmcs->host_gdtr_base = value; break; case HOST_IDTR_BASE: current_evmcs->host_idtr_base = value; break; case HOST_RSP: current_evmcs->host_rsp = value; break; case EPT_POINTER: current_evmcs->ept_pointer = value; break; case GUEST_BNDCFGS: current_evmcs->guest_bndcfgs = value; break; case XSS_EXIT_BITMAP: current_evmcs->xss_exit_bitmap = value; break; case GUEST_PHYSICAL_ADDRESS: current_evmcs->guest_physical_address = value; break; case EXIT_QUALIFICATION: current_evmcs->exit_qualification = value; break; case GUEST_LINEAR_ADDRESS: current_evmcs->guest_linear_address = value; break; case VM_EXIT_MSR_STORE_ADDR: current_evmcs->vm_exit_msr_store_addr = value; break; case VM_EXIT_MSR_LOAD_ADDR: current_evmcs->vm_exit_msr_load_addr = value; break; case VM_ENTRY_MSR_LOAD_ADDR: current_evmcs->vm_entry_msr_load_addr = value; break; case CR3_TARGET_VALUE0: current_evmcs->cr3_target_value0 = value; break; case CR3_TARGET_VALUE1: current_evmcs->cr3_target_value1 = value; break; case CR3_TARGET_VALUE2: current_evmcs->cr3_target_value2 = value; break; case CR3_TARGET_VALUE3: current_evmcs->cr3_target_value3 = value; break; case TPR_THRESHOLD: current_evmcs->tpr_threshold = value; break; case GUEST_INTERRUPTIBILITY_INFO: current_evmcs->guest_interruptibility_info = value; break; case CPU_BASED_VM_EXEC_CONTROL: current_evmcs->cpu_based_vm_exec_control = value; break; case EXCEPTION_BITMAP: current_evmcs->exception_bitmap = value; break; case VM_ENTRY_CONTROLS: current_evmcs->vm_entry_controls = value; break; case VM_ENTRY_INTR_INFO_FIELD: current_evmcs->vm_entry_intr_info_field = value; break; case VM_ENTRY_EXCEPTION_ERROR_CODE: current_evmcs->vm_entry_exception_error_code = value; break; case VM_ENTRY_INSTRUCTION_LEN: current_evmcs->vm_entry_instruction_len = value; break; case HOST_IA32_SYSENTER_CS: current_evmcs->host_ia32_sysenter_cs = value; break; case PIN_BASED_VM_EXEC_CONTROL: current_evmcs->pin_based_vm_exec_control = value; break; case VM_EXIT_CONTROLS: current_evmcs->vm_exit_controls = value; break; case SECONDARY_VM_EXEC_CONTROL: current_evmcs->secondary_vm_exec_control = value; break; case GUEST_ES_LIMIT: current_evmcs->guest_es_limit = value; break; case GUEST_CS_LIMIT: current_evmcs->guest_cs_limit = value; break; case GUEST_SS_LIMIT: current_evmcs->guest_ss_limit = value; break; case GUEST_DS_LIMIT: current_evmcs->guest_ds_limit = value; break; case GUEST_FS_LIMIT: current_evmcs->guest_fs_limit = value; break; case GUEST_GS_LIMIT: current_evmcs->guest_gs_limit = value; break; case GUEST_LDTR_LIMIT: current_evmcs->guest_ldtr_limit = value; break; case GUEST_TR_LIMIT: current_evmcs->guest_tr_limit = value; break; case GUEST_GDTR_LIMIT: current_evmcs->guest_gdtr_limit = value; break; case GUEST_IDTR_LIMIT: current_evmcs->guest_idtr_limit = value; break; case GUEST_ES_AR_BYTES: current_evmcs->guest_es_ar_bytes = value; break; case GUEST_CS_AR_BYTES: current_evmcs->guest_cs_ar_bytes = value; break; case GUEST_SS_AR_BYTES: current_evmcs->guest_ss_ar_bytes = value; break; case GUEST_DS_AR_BYTES: current_evmcs->guest_ds_ar_bytes = value; break; case GUEST_FS_AR_BYTES: current_evmcs->guest_fs_ar_bytes = value; break; case GUEST_GS_AR_BYTES: current_evmcs->guest_gs_ar_bytes = value; break; case GUEST_LDTR_AR_BYTES: current_evmcs->guest_ldtr_ar_bytes = value; break; case GUEST_TR_AR_BYTES: current_evmcs->guest_tr_ar_bytes = value; break; case GUEST_ACTIVITY_STATE: current_evmcs->guest_activity_state = value; break; case GUEST_SYSENTER_CS: current_evmcs->guest_sysenter_cs = value; break; case VM_INSTRUCTION_ERROR: current_evmcs->vm_instruction_error = value; break; case VM_EXIT_REASON: current_evmcs->vm_exit_reason = value; break; case VM_EXIT_INTR_INFO: current_evmcs->vm_exit_intr_info = value; break; case VM_EXIT_INTR_ERROR_CODE: current_evmcs->vm_exit_intr_error_code = value; break; case IDT_VECTORING_INFO_FIELD: current_evmcs->idt_vectoring_info_field = value; break; case IDT_VECTORING_ERROR_CODE: current_evmcs->idt_vectoring_error_code = value; break; case VM_EXIT_INSTRUCTION_LEN: current_evmcs->vm_exit_instruction_len = value; break; case VMX_INSTRUCTION_INFO: current_evmcs->vmx_instruction_info = value; break; case PAGE_FAULT_ERROR_CODE_MASK: current_evmcs->page_fault_error_code_mask = value; break; case PAGE_FAULT_ERROR_CODE_MATCH: current_evmcs->page_fault_error_code_match = value; break; case CR3_TARGET_COUNT: current_evmcs->cr3_target_count = value; break; case VM_EXIT_MSR_STORE_COUNT: current_evmcs->vm_exit_msr_store_count = value; break; case VM_EXIT_MSR_LOAD_COUNT: current_evmcs->vm_exit_msr_load_count = value; break; case VM_ENTRY_MSR_LOAD_COUNT: current_evmcs->vm_entry_msr_load_count = value; break; case HOST_ES_SELECTOR: current_evmcs->host_es_selector = value; break; case HOST_CS_SELECTOR: current_evmcs->host_cs_selector = value; break; case HOST_SS_SELECTOR: current_evmcs->host_ss_selector = value; break; case HOST_DS_SELECTOR: current_evmcs->host_ds_selector = value; break; case HOST_FS_SELECTOR: current_evmcs->host_fs_selector = value; break; case HOST_GS_SELECTOR: current_evmcs->host_gs_selector = value; break; case HOST_TR_SELECTOR: current_evmcs->host_tr_selector = value; break; case GUEST_ES_SELECTOR: current_evmcs->guest_es_selector = value; break; case GUEST_CS_SELECTOR: current_evmcs->guest_cs_selector = value; break; case GUEST_SS_SELECTOR: current_evmcs->guest_ss_selector = value; break; case GUEST_DS_SELECTOR: current_evmcs->guest_ds_selector = value; break; case GUEST_FS_SELECTOR: current_evmcs->guest_fs_selector = value; break; case GUEST_GS_SELECTOR: current_evmcs->guest_gs_selector = value; break; case GUEST_LDTR_SELECTOR: current_evmcs->guest_ldtr_selector = value; break; case GUEST_TR_SELECTOR: current_evmcs->guest_tr_selector = value; break; case VIRTUAL_PROCESSOR_ID: current_evmcs->virtual_processor_id = value; break; default: return 1; } return 0; } static inline int evmcs_vmlaunch(void) { int ret; current_evmcs->hv_clean_fields = 0; __asm__ __volatile__("push %%rbp;" "push %%rcx;" "push %%rdx;" "push %%rsi;" "push %%rdi;" "push $0;" "mov %%rsp, (%[host_rsp]);" "lea 1f(%%rip), %%rax;" "mov %%rax, (%[host_rip]);" "vmlaunch;" "incq (%%rsp);" "1: pop %%rax;" "pop %%rdi;" "pop %%rsi;" "pop %%rdx;" "pop %%rcx;" "pop %%rbp;" : [ret]"=&a"(ret) : [host_rsp]"r" ((uint64_t)¤t_evmcs->host_rsp), [host_rip]"r" ((uint64_t)¤t_evmcs->host_rip) : "memory", "cc", "rbx", "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15"); return ret; } /* * No guest state (e.g. GPRs) is established by this vmresume. */ static inline int evmcs_vmresume(void) { int ret; current_evmcs->hv_clean_fields = 0; __asm__ __volatile__("push %%rbp;" "push %%rcx;" "push %%rdx;" "push %%rsi;" "push %%rdi;" "push $0;" "mov %%rsp, (%[host_rsp]);" "lea 1f(%%rip), %%rax;" "mov %%rax, (%[host_rip]);" "vmresume;" "incq (%%rsp);" "1: pop %%rax;" "pop %%rdi;" "pop %%rsi;" "pop %%rdx;" "pop %%rcx;" "pop %%rbp;" : [ret]"=&a"(ret) : [host_rsp]"r" ((uint64_t)¤t_evmcs->host_rsp), [host_rip]"r" ((uint64_t)¤t_evmcs->host_rip) : "memory", "cc", "rbx", "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15"); return ret; } #endif /* !SELFTEST_KVM_EVMCS_H */