aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorArd Biesheuvel <ard.biesheuvel@linaro.org>2014-09-17 17:56:18 -0400
committerChristoffer Dall <christoffer.dall@linaro.org>2014-10-10 07:07:37 -0400
commitc40f2f8ff833eddc02cb599ef6e5a162223449ba (patch)
treea2ec51729aae4cc741dca4c264a723362b32db14
parent37b544087ef3f65ca68465ba39291a07195dac26 (diff)
arm/arm64: KVM: add 'writable' parameter to kvm_phys_addr_ioremap
Add support for read-only MMIO passthrough mappings by adding a 'writable' parameter to kvm_phys_addr_ioremap. For the moment, mappings will be read-write even if 'writable' is false, but once the definition of PAGE_S2_DEVICE gets changed, those mappings will be created read-only. Acked-by: Marc Zyngier <marc.zyngier@arm.com> Signed-off-by: Ard Biesheuvel <ard.biesheuvel@linaro.org> Signed-off-by: Christoffer Dall <christoffer.dall@linaro.org>
-rw-r--r--arch/arm/include/asm/kvm_mmu.h2
-rw-r--r--arch/arm/kvm/mmu.c5
-rw-r--r--arch/arm64/include/asm/kvm_mmu.h2
-rw-r--r--virt/kvm/arm/vgic.c3
4 files changed, 8 insertions, 4 deletions
diff --git a/arch/arm/include/asm/kvm_mmu.h b/arch/arm/include/asm/kvm_mmu.h
index 3f688b458143..eaa6deac97b2 100644
--- a/arch/arm/include/asm/kvm_mmu.h
+++ b/arch/arm/include/asm/kvm_mmu.h
@@ -50,7 +50,7 @@ void free_hyp_pgds(void);
50int kvm_alloc_stage2_pgd(struct kvm *kvm); 50int kvm_alloc_stage2_pgd(struct kvm *kvm);
51void kvm_free_stage2_pgd(struct kvm *kvm); 51void kvm_free_stage2_pgd(struct kvm *kvm);
52int kvm_phys_addr_ioremap(struct kvm *kvm, phys_addr_t guest_ipa, 52int kvm_phys_addr_ioremap(struct kvm *kvm, phys_addr_t guest_ipa,
53 phys_addr_t pa, unsigned long size); 53 phys_addr_t pa, unsigned long size, bool writable);
54 54
55int kvm_handle_guest_abort(struct kvm_vcpu *vcpu, struct kvm_run *run); 55int kvm_handle_guest_abort(struct kvm_vcpu *vcpu, struct kvm_run *run);
56 56
diff --git a/arch/arm/kvm/mmu.c b/arch/arm/kvm/mmu.c
index bb2e110e77c0..a7eabd1c4287 100644
--- a/arch/arm/kvm/mmu.c
+++ b/arch/arm/kvm/mmu.c
@@ -674,7 +674,7 @@ static int stage2_set_pte(struct kvm *kvm, struct kvm_mmu_memory_cache *cache,
674 * @size: The size of the mapping 674 * @size: The size of the mapping
675 */ 675 */
676int kvm_phys_addr_ioremap(struct kvm *kvm, phys_addr_t guest_ipa, 676int kvm_phys_addr_ioremap(struct kvm *kvm, phys_addr_t guest_ipa,
677 phys_addr_t pa, unsigned long size) 677 phys_addr_t pa, unsigned long size, bool writable)
678{ 678{
679 phys_addr_t addr, end; 679 phys_addr_t addr, end;
680 int ret = 0; 680 int ret = 0;
@@ -687,6 +687,9 @@ int kvm_phys_addr_ioremap(struct kvm *kvm, phys_addr_t guest_ipa,
687 for (addr = guest_ipa; addr < end; addr += PAGE_SIZE) { 687 for (addr = guest_ipa; addr < end; addr += PAGE_SIZE) {
688 pte_t pte = pfn_pte(pfn, PAGE_S2_DEVICE); 688 pte_t pte = pfn_pte(pfn, PAGE_S2_DEVICE);
689 689
690 if (writable)
691 kvm_set_s2pte_writable(&pte);
692
690 ret = mmu_topup_memory_cache(&cache, 2, 2); 693 ret = mmu_topup_memory_cache(&cache, 2, 2);
691 if (ret) 694 if (ret)
692 goto out; 695 goto out;
diff --git a/arch/arm64/include/asm/kvm_mmu.h b/arch/arm64/include/asm/kvm_mmu.h
index a030d163840b..e36171974d6a 100644
--- a/arch/arm64/include/asm/kvm_mmu.h
+++ b/arch/arm64/include/asm/kvm_mmu.h
@@ -77,7 +77,7 @@ void free_hyp_pgds(void);
77int kvm_alloc_stage2_pgd(struct kvm *kvm); 77int kvm_alloc_stage2_pgd(struct kvm *kvm);
78void kvm_free_stage2_pgd(struct kvm *kvm); 78void kvm_free_stage2_pgd(struct kvm *kvm);
79int kvm_phys_addr_ioremap(struct kvm *kvm, phys_addr_t guest_ipa, 79int kvm_phys_addr_ioremap(struct kvm *kvm, phys_addr_t guest_ipa,
80 phys_addr_t pa, unsigned long size); 80 phys_addr_t pa, unsigned long size, bool writable);
81 81
82int kvm_handle_guest_abort(struct kvm_vcpu *vcpu, struct kvm_run *run); 82int kvm_handle_guest_abort(struct kvm_vcpu *vcpu, struct kvm_run *run);
83 83
diff --git a/virt/kvm/arm/vgic.c b/virt/kvm/arm/vgic.c
index 862967852d5a..382fb5a88b9c 100644
--- a/virt/kvm/arm/vgic.c
+++ b/virt/kvm/arm/vgic.c
@@ -1899,7 +1899,8 @@ int kvm_vgic_init(struct kvm *kvm)
1899 } 1899 }
1900 1900
1901 ret = kvm_phys_addr_ioremap(kvm, kvm->arch.vgic.vgic_cpu_base, 1901 ret = kvm_phys_addr_ioremap(kvm, kvm->arch.vgic.vgic_cpu_base,
1902 vgic->vcpu_base, KVM_VGIC_V2_CPU_SIZE); 1902 vgic->vcpu_base, KVM_VGIC_V2_CPU_SIZE,
1903 true);
1903 if (ret) { 1904 if (ret) {
1904 kvm_err("Unable to remap VGIC CPU to VCPU\n"); 1905 kvm_err("Unable to remap VGIC CPU to VCPU\n");
1905 goto out; 1906 goto out;