return PAGE_SIZE;
}
+static bool kvm_riscv_mmu_dirty_log_write_fault_fast(struct kvm *kvm,
+ struct kvm_memory_slot *memslot,
+ gpa_t gpa,
+ struct kvm_gstage_mapping *out_map)
+{
+ struct kvm_gstage gstage;
+ unsigned long mmu_seq;
+ pte_t old_pte, new_pte;
+ pte_t *ptep;
+ gfn_t gfn = gpa >> PAGE_SHIFT;
+ u32 ptep_level;
+ bool dirty_marked = false;
+ bool ret;
+
+ kvm_riscv_gstage_init(&gstage, kvm);
+ mmu_seq = kvm->mmu_invalidate_seq;
+
+ read_lock(&kvm->mmu_lock);
+
+ if (mmu_invalidate_retry_gfn(kvm, mmu_seq, gfn)) {
+ ret = false;
+ goto out_unlock;
+ }
+
+ if (!kvm_riscv_gstage_get_leaf(&gstage, gpa, &ptep, &ptep_level) ||
+ ptep_level) {
+ ret = false;
+ goto out_unlock;
+ }
+
+ for (;;) {
+ old_pte = ptep_get(ptep);
+ if (!(pte_val(old_pte) & _PAGE_LEAF)) {
+ ret = false;
+ break;
+ }
+
+ if (!dirty_marked) {
+ mark_page_dirty_in_slot(kvm, memslot, gfn);
+ dirty_marked = true;
+ }
+
+ if ((pte_val(old_pte) & (_PAGE_WRITE | _PAGE_DIRTY)) ==
+ (_PAGE_WRITE | _PAGE_DIRTY)) {
+ new_pte = old_pte;
+ ret = true;
+ break;
+ }
+
+ new_pte = pte_mkdirty(pte_mkwrite_novma(old_pte));
+
+ if (kvm_riscv_gstage_try_update_pte(&gstage, ptep_level, gpa,
+ ptep, old_pte, new_pte)) {
+ ret = true;
+ break;
+ }
+ cpu_relax();
+ }
+
+out_unlock:
+ read_unlock(&kvm->mmu_lock);
+
+ if (ret) {
+ out_map->addr = gpa & PAGE_MASK;
+ out_map->level = 0;
+ out_map->pte = new_pte;
+ }
+
+ return ret;
+}
+
int kvm_riscv_mmu_map(struct kvm_vcpu *vcpu, struct kvm_memory_slot *memslot,
gpa_t gpa, unsigned long hva, bool is_write,
struct kvm_gstage_mapping *out_map)
/* Setup initial state of output mapping */
memset(out_map, 0, sizeof(*out_map));
+ if (is_write && logging &&
+ kvm_riscv_mmu_dirty_log_write_fault_fast(kvm, memslot, gpa, out_map))
+ return 0;
+
/* We need minimum second+third level pages */
ret = kvm_mmu_topup_memory_cache(pcache, kvm->arch.pgd_levels);
if (ret) {