Re: [PATCH] nvmx: Check exit qualification RD/WR permission for MMIO accesses

From: Paolo Bonzini
Date: Thu Apr 12 2018 - 09:09:03 EST


On 04/03/2018 11:16, KarimAllah Ahmed wrote:
> Validate that a write MMIO access that follows a read MMIO access would
> have the correct access captured in the exit qualification.
>
> Cc: Paolo Bonzini <pbonzini@xxxxxxxxxx>
> Cc: Radim KrÄmÃÅ <rkrcmar@xxxxxxxxxx>
> Cc: kvm@xxxxxxxxxxxxxxx
> Cc: linux-kernel@xxxxxxxxxxxxxxx
> Signed-off-by: KarimAllah Ahmed <karahmed@xxxxxxxxx>
> Message-Id: <1519841208-23349-1-git-send-email-karahmed@xxxxxxxxx>
> ---
> x86/vmx_tests.c | 52 ++++++++++++++++++++++++++++++++++++++++++++++++----
> 1 file changed, 48 insertions(+), 4 deletions(-)
>
> diff --git a/x86/vmx_tests.c b/x86/vmx_tests.c
> index 598dd88..a72af1a 100644
> --- a/x86/vmx_tests.c
> +++ b/x86/vmx_tests.c
> @@ -7,6 +7,7 @@
> #include "msr.h"
> #include "processor.h"
> #include "vm.h"
> +#include "pci.h"
> #include "fwcfg.h"
> #include "isr.h"
> #include "desc.h"
> @@ -28,6 +29,8 @@ unsigned long *pml4;
> u64 eptp;
> void *data_page1, *data_page2;
>
> +phys_addr_t pci_physaddr;
> +
> void *pml_log;
> #define PML_INDEX 512
>
> @@ -1041,6 +1044,9 @@ static int apic_version;
>
> static int ept_init_common(bool have_ad)
> {
> + int ret;
> + struct pci_dev pcidev;
> +
> if (setup_ept(have_ad))
> return VMX_TEST_EXIT;
> data_page1 = alloc_page();
> @@ -1053,6 +1059,13 @@ static int ept_init_common(bool have_ad)
> EPT_RA | EPT_WA | EPT_EA);
>
> apic_version = apic_read(APIC_LVR);
> +
> + ret = pci_find_dev(PCI_VENDOR_ID_REDHAT, PCI_DEVICE_ID_REDHAT_TEST);
> + if (ret != PCIDEVADDR_INVALID) {
> + pci_dev_init(&pcidev, ret);
> + pci_physaddr = pcidev.resource[PCI_TESTDEV_BAR_MEM];
> + }
> +
> return VMX_TEST_START;
> }
>
> @@ -1101,6 +1114,16 @@ t1:
> vmcall();
> *((u32 *)data_page1) = MAGIC_VAL_2;
> report("EPT violation - paging structure", vmx_get_test_stage() == 5);
> +
> + // MMIO Read/Write
> + vmx_set_test_stage(5);
> + vmcall();
> +
> + *(u32 volatile *)pci_physaddr;
> + report("MMIO EPT violation - read", vmx_get_test_stage() == 6);
> +
> + *(u32 volatile *)pci_physaddr = MAGIC_VAL_1;
> + report("MMIO EPT violation - write", vmx_get_test_stage() == 7);
> }
>
> static void ept_main()
> @@ -1108,12 +1131,12 @@ static void ept_main()
> ept_common();
>
> // Test EPT access to L1 MMIO
> - vmx_set_test_stage(6);
> + vmx_set_test_stage(7);
> report("EPT - MMIO access", *((u32 *)0xfee00030UL) == apic_version);
>
> // Test invalid operand for INVEPT
> vmcall();
> - report("EPT - unsupported INVEPT", vmx_get_test_stage() == 7);
> + report("EPT - unsupported INVEPT", vmx_get_test_stage() == 8);
> }
>
> bool invept_test(int type, u64 eptp)
> @@ -1187,7 +1210,7 @@ static int ept_exit_handler_common(bool have_ad)
> ulong reason;
> u32 insn_len;
> u32 exit_qual;
> - static unsigned long data_page1_pte, data_page1_pte_pte;
> + static unsigned long data_page1_pte, data_page1_pte_pte, memaddr_pte;
>
> guest_rip = vmcs_read(GUEST_RIP);
> guest_cr3 = vmcs_read(GUEST_CR3);
> @@ -1249,7 +1272,12 @@ static int ept_exit_handler_common(bool have_ad)
> data_page1_pte_pte & ~EPT_PRESENT);
> ept_sync(INVEPT_SINGLE, eptp);
> break;
> - case 6:
> + case 5:
> + install_ept(pml4, (unsigned long)pci_physaddr,
> + (unsigned long)pci_physaddr, 0);
> + ept_sync(INVEPT_SINGLE, eptp);
> + break;
> + case 7:
> if (!invept_test(0, eptp))
> vmx_inc_test_stage();
> break;
> @@ -1305,6 +1333,22 @@ static int ept_exit_handler_common(bool have_ad)
> data_page1_pte_pte | (EPT_PRESENT));
> ept_sync(INVEPT_SINGLE, eptp);
> break;
> + case 5:
> + if (exit_qual & EPT_VLT_RD)
> + vmx_inc_test_stage();
> + TEST_ASSERT(get_ept_pte(pml4, (unsigned long)pci_physaddr,
> + 1, &memaddr_pte));
> + set_ept_pte(pml4, memaddr_pte, 1, memaddr_pte | EPT_RA);
> + ept_sync(INVEPT_SINGLE, eptp);
> + break;
> + case 6:
> + if (exit_qual & EPT_VLT_WR)
> + vmx_inc_test_stage();
> + TEST_ASSERT(get_ept_pte(pml4, (unsigned long)pci_physaddr,
> + 1, &memaddr_pte));
> + set_ept_pte(pml4, memaddr_pte, 1, memaddr_pte | EPT_RA | EPT_WA);
> + ept_sync(INVEPT_SINGLE, eptp);
> + break;
> default:
> // Should not reach here
> report("ERROR : unexpected stage, %d", false,
>

Pushed, thanks!

Paolo