KVM: MIPS: Pass type of fault down to kvm_mips_map_page()
authorJames Hogan <james.hogan@imgtec.com>
Fri, 1 May 2015 13:56:31 +0000 (14:56 +0100)
committerJames Hogan <james.hogan@imgtec.com>
Fri, 3 Feb 2017 15:21:19 +0000 (15:21 +0000)
kvm_mips_map_page() will need to know whether the fault was due to a
read or a write in order to support dirty page tracking,
KVM_CAP_SYNC_MMU, and read only memory regions, so get that information
passed down to it via new bool write_fault arguments to various
functions.

Signed-off-by: James Hogan <james.hogan@imgtec.com>
Cc: Paolo Bonzini <pbonzini@redhat.com>
Cc: "Radim Krčmář" <rkrcmar@redhat.com>
Cc: Ralf Baechle <ralf@linux-mips.org>
Cc: linux-mips@linux-mips.org
Cc: kvm@vger.kernel.org
arch/mips/include/asm/kvm_host.h
arch/mips/kvm/emulate.c
arch/mips/kvm/mmu.c
arch/mips/kvm/trap_emul.c

index ea1b495c042c9be0b1d153b3850d8c353af32951..17db9d6c7c93fdf2d0dde1cea2404d45084c1951 100644 (file)
@@ -597,19 +597,22 @@ u32 kvm_get_user_asid(struct kvm_vcpu *vcpu);
 u32 kvm_get_commpage_asid (struct kvm_vcpu *vcpu);
 
 extern int kvm_mips_handle_kseg0_tlb_fault(unsigned long badbaddr,
-                                          struct kvm_vcpu *vcpu);
+                                          struct kvm_vcpu *vcpu,
+                                          bool write_fault);
 
 extern int kvm_mips_handle_commpage_tlb_fault(unsigned long badvaddr,
                                              struct kvm_vcpu *vcpu);
 
 extern int kvm_mips_handle_mapped_seg_tlb_fault(struct kvm_vcpu *vcpu,
                                                struct kvm_mips_tlb *tlb,
-                                               unsigned long gva);
+                                               unsigned long gva,
+                                               bool write_fault);
 
 extern enum emulation_result kvm_mips_handle_tlbmiss(u32 cause,
                                                     u32 *opc,
                                                     struct kvm_run *run,
-                                                    struct kvm_vcpu *vcpu);
+                                                    struct kvm_vcpu *vcpu,
+                                                    bool write_fault);
 
 extern enum emulation_result kvm_mips_handle_tlbmod(u32 cause,
                                                    u32 *opc,
index 40159cf5166b932537adc153550e3341aee38964..4875afca3f268ab689441ced09123d5b0ea08fda 100644 (file)
@@ -2704,7 +2704,8 @@ enum emulation_result kvm_mips_check_privilege(u32 cause,
 enum emulation_result kvm_mips_handle_tlbmiss(u32 cause,
                                              u32 *opc,
                                              struct kvm_run *run,
-                                             struct kvm_vcpu *vcpu)
+                                             struct kvm_vcpu *vcpu,
+                                             bool write_fault)
 {
        enum emulation_result er = EMULATE_DONE;
        u32 exccode = (cause >> CAUSEB_EXCCODE) & 0x1f;
@@ -2760,8 +2761,8 @@ enum emulation_result kvm_mips_handle_tlbmiss(u32 cause,
                         * OK we have a Guest TLB entry, now inject it into the
                         * shadow host TLB
                         */
-                       if (kvm_mips_handle_mapped_seg_tlb_fault(vcpu, tlb,
-                                                                va)) {
+                       if (kvm_mips_handle_mapped_seg_tlb_fault(vcpu, tlb, va,
+                                                                write_fault)) {
                                kvm_err("%s: handling mapped seg tlb fault for %lx, index: %u, vcpu: %p, ASID: %#lx\n",
                                        __func__, va, index, vcpu,
                                        read_c0_entryhi());
index b3da473e15693318f7bfb5bb1ca35eed35cb73b7..1af65f2e6bb7ffd21319b191f2c27dc7e9194815 100644 (file)
@@ -308,6 +308,7 @@ bool kvm_mips_flush_gpa_pt(struct kvm *kvm, gfn_t start_gfn, gfn_t end_gfn)
  * kvm_mips_map_page() - Map a guest physical page.
  * @vcpu:              VCPU pointer.
  * @gpa:               Guest physical address of fault.
+ * @write_fault:       Whether the fault was due to a write.
  * @out_entry:         New PTE for @gpa (written on success unless NULL).
  * @out_buddy:         New PTE for @gpa's buddy (written on success unless
  *                     NULL).
@@ -327,6 +328,7 @@ bool kvm_mips_flush_gpa_pt(struct kvm *kvm, gfn_t start_gfn, gfn_t end_gfn)
  *             as an MMIO access.
  */
 static int kvm_mips_map_page(struct kvm_vcpu *vcpu, unsigned long gpa,
+                            bool write_fault,
                             pte_t *out_entry, pte_t *out_buddy)
 {
        struct kvm *kvm = vcpu->kvm;
@@ -558,7 +560,8 @@ void kvm_mips_flush_gva_pt(pgd_t *pgd, enum kvm_mips_flush flags)
 
 /* XXXKYMA: Must be called with interrupts disabled */
 int kvm_mips_handle_kseg0_tlb_fault(unsigned long badvaddr,
-                                   struct kvm_vcpu *vcpu)
+                                   struct kvm_vcpu *vcpu,
+                                   bool write_fault)
 {
        unsigned long gpa;
        kvm_pfn_t pfn0, pfn1;
@@ -576,10 +579,11 @@ int kvm_mips_handle_kseg0_tlb_fault(unsigned long badvaddr,
        gpa = KVM_GUEST_CPHYSADDR(badvaddr & (PAGE_MASK << 1));
        vaddr = badvaddr & (PAGE_MASK << 1);
 
-       if (kvm_mips_map_page(vcpu, gpa, &pte_gpa[0], NULL) < 0)
+       if (kvm_mips_map_page(vcpu, gpa, write_fault, &pte_gpa[0], NULL) < 0)
                return -1;
 
-       if (kvm_mips_map_page(vcpu, gpa | PAGE_SIZE, &pte_gpa[1], NULL) < 0)
+       if (kvm_mips_map_page(vcpu, gpa | PAGE_SIZE, write_fault, &pte_gpa[1],
+                             NULL) < 0)
                return -1;
 
        pfn0 = pte_pfn(pte_gpa[0]);
@@ -604,7 +608,8 @@ int kvm_mips_handle_kseg0_tlb_fault(unsigned long badvaddr,
 
 int kvm_mips_handle_mapped_seg_tlb_fault(struct kvm_vcpu *vcpu,
                                         struct kvm_mips_tlb *tlb,
-                                        unsigned long gva)
+                                        unsigned long gva,
+                                        bool write_fault)
 {
        kvm_pfn_t pfn;
        long tlb_lo = 0;
@@ -621,8 +626,8 @@ int kvm_mips_handle_mapped_seg_tlb_fault(struct kvm_vcpu *vcpu,
                tlb_lo = tlb->tlb_lo[idx];
 
        /* Find host PFN */
-       if (kvm_mips_map_page(vcpu, mips3_tlbpfn_to_paddr(tlb_lo), &pte_gpa,
-                             NULL) < 0)
+       if (kvm_mips_map_page(vcpu, mips3_tlbpfn_to_paddr(tlb_lo), write_fault,
+                             &pte_gpa, NULL) < 0)
                return -1;
        pfn = pte_pfn(pte_gpa);
 
@@ -757,7 +762,7 @@ enum kvm_mips_fault_result kvm_trap_emul_gva_fault(struct kvm_vcpu *vcpu,
        int index;
 
        if (KVM_GUEST_KSEGX(gva) == KVM_GUEST_KSEG0) {
-               if (kvm_mips_handle_kseg0_tlb_fault(gva, vcpu) < 0)
+               if (kvm_mips_handle_kseg0_tlb_fault(gva, vcpu, write) < 0)
                        return KVM_MIPS_GPA;
        } else if ((KVM_GUEST_KSEGX(gva) < KVM_GUEST_KSEG0) ||
                   KVM_GUEST_KSEGX(gva) == KVM_GUEST_KSEG23) {
@@ -774,7 +779,7 @@ enum kvm_mips_fault_result kvm_trap_emul_gva_fault(struct kvm_vcpu *vcpu,
                if (write && !TLB_IS_DIRTY(*tlb, gva))
                        return KVM_MIPS_TLBMOD;
 
-               if (kvm_mips_handle_mapped_seg_tlb_fault(vcpu, tlb, gva))
+               if (kvm_mips_handle_mapped_seg_tlb_fault(vcpu, tlb, gva, write))
                        return KVM_MIPS_GPA;
        } else {
                return KVM_MIPS_GVA;
index d0b6409cb4a328756cb7b994df9fc3b5fe323616..070d1ddbc7ee6c344084b2faaf2828f0c89d6c53 100644 (file)
@@ -159,7 +159,7 @@ static int kvm_trap_emul_handle_tlb_miss(struct kvm_vcpu *vcpu, bool store)
                 *     into the shadow host TLB
                 */
 
-               er = kvm_mips_handle_tlbmiss(cause, opc, run, vcpu);
+               er = kvm_mips_handle_tlbmiss(cause, opc, run, vcpu, store);
                if (er == EMULATE_DONE)
                        ret = RESUME_GUEST;
                else {
@@ -172,7 +172,7 @@ static int kvm_trap_emul_handle_tlb_miss(struct kvm_vcpu *vcpu, bool store)
                 * not expect to ever get them
                 */
                if (kvm_mips_handle_kseg0_tlb_fault
-                   (vcpu->arch.host_cp0_badvaddr, vcpu) < 0) {
+                   (vcpu->arch.host_cp0_badvaddr, vcpu, store) < 0) {
                        run->exit_reason = KVM_EXIT_INTERNAL_ERROR;
                        ret = RESUME_HOST;
                }