On 08/03/2018 01:37 PM, David Hildenbrand wrote: > On 03.08.2018 13:29, Christian Borntraeger wrote: >> >> >> On 08/02/2018 11:01 AM, David Hildenbrand wrote: >>> On 31.07.2018 10:44, Christian Borntraeger wrote: >>>> From: Janosch Frank <frankja@xxxxxxxxxxxxx> >>>> >>>> When doing skey emulation for huge guests, we now need to fault in >>>> pmds, as we don't have PGSTES anymore to store them when we do not >>>> have valid table entries. >>>> >>>> Signed-off-by: Janosch Frank <frankja@xxxxxxxxxxxxx> >>>> --- >>>> arch/s390/kvm/kvm-s390.c | 15 ++++-- >>>> arch/s390/kvm/priv.c | 105 ++++++++++++++++++++++++++------------- >>>> 2 files changed, 83 insertions(+), 37 deletions(-) >>>> >>>> diff --git a/arch/s390/kvm/kvm-s390.c b/arch/s390/kvm/kvm-s390.c >>>> index 4cff5e31ca36..662f4d8046db 100644 >>>> --- a/arch/s390/kvm/kvm-s390.c >>>> +++ b/arch/s390/kvm/kvm-s390.c >>>> @@ -1551,6 +1551,7 @@ static long kvm_s390_set_skeys(struct kvm *kvm, struct kvm_s390_skeys *args) >>>> uint8_t *keys; >>>> uint64_t hva; >>>> int srcu_idx, i, r = 0; >>>> + bool unlocked; >>>> >>>> if (args->flags != 0) >>>> return -EINVAL; >>>> @@ -1575,9 +1576,11 @@ static long kvm_s390_set_skeys(struct kvm *kvm, struct kvm_s390_skeys *args) >>>> if (r) >>>> goto out; >>>> >>>> + i = 0; >>>> down_read(¤t->mm->mmap_sem); >>>> srcu_idx = srcu_read_lock(&kvm->srcu); >>>> - for (i = 0; i < args->count; i++) { >>>> + while (i < args->count) { >>>> + unlocked = false; >>>> hva = gfn_to_hva(kvm, args->start_gfn + i); >>>> if (kvm_is_error_hva(hva)) { >>>> r = -EFAULT; >>>> @@ -1591,8 +1594,14 @@ static long kvm_s390_set_skeys(struct kvm *kvm, struct kvm_s390_skeys *args) >>>> } >>>> >>>> r = set_guest_storage_key(current->mm, hva, keys[i], 0); >>>> - if (r) >>>> - break; >>>> + if (r) { >>>> + r = fixup_user_fault(current, current->mm, hva, >>>> + FAULT_FLAG_WRITE, &unlocked); >>>> + if (r) >>>> + break; >>>> + } >>>> + if (!r) >>>> + i++; >>>> } >>>> srcu_read_unlock(&kvm->srcu, srcu_idx); >>>> up_read(¤t->mm->mmap_sem); >>>> diff --git a/arch/s390/kvm/priv.c b/arch/s390/kvm/priv.c >>>> index eb0eb60c7be6..cfc5a62329f6 100644 >>>> --- a/arch/s390/kvm/priv.c >>>> +++ b/arch/s390/kvm/priv.c >>>> @@ -246,9 +246,10 @@ static int try_handle_skey(struct kvm_vcpu *vcpu) >>>> >>>> static int handle_iske(struct kvm_vcpu *vcpu) >>>> { >>>> - unsigned long addr; >>>> + unsigned long gaddr, vmaddr; >>>> unsigned char key; >>>> int reg1, reg2; >>>> + bool unlocked; >>>> int rc; >>>> >>>> vcpu->stat.instruction_iske++; >>>> @@ -262,18 +263,28 @@ static int handle_iske(struct kvm_vcpu *vcpu) >>>> >>>> kvm_s390_get_regs_rre(vcpu, ®1, ®2); >>>> >>>> - addr = vcpu->run->s.regs.gprs[reg2] & PAGE_MASK; >>>> - addr = kvm_s390_logical_to_effective(vcpu, addr); >>>> - addr = kvm_s390_real_to_abs(vcpu, addr); >>>> - addr = gfn_to_hva(vcpu->kvm, gpa_to_gfn(addr)); >>>> - if (kvm_is_error_hva(addr)) >>>> + gaddr = vcpu->run->s.regs.gprs[reg2] & PAGE_MASK; >>>> + gaddr = kvm_s390_logical_to_effective(vcpu, gaddr); >>>> + gaddr = kvm_s390_real_to_abs(vcpu, gaddr); >>>> + vmaddr = gfn_to_hva(vcpu->kvm, gpa_to_gfn(gaddr)); >>>> + if (kvm_is_error_hva(vmaddr)) >>>> return kvm_s390_inject_program_int(vcpu, PGM_ADDRESSING); >>>> - >>>> +retry: >>>> + unlocked = false; >>>> down_read(¤t->mm->mmap_sem); >>>> - rc = get_guest_storage_key(current->mm, addr, &key); >>>> - up_read(¤t->mm->mmap_sem); >>>> + rc = get_guest_storage_key(current->mm, vmaddr, &key); >>>> + >>>> + if (rc) { >>>> + rc = fixup_user_fault(current, current->mm, vmaddr, >>>> + FAULT_FLAG_WRITE, &unlocked); >>>> + if (!rc) { >>>> + up_read(¤t->mm->mmap_sem); >>>> + goto retry; >>>> + } >>>> + } >>>> if (rc) >>>> return kvm_s390_inject_program_int(vcpu, PGM_ADDRESSING); >>>> + up_read(¤t->mm->mmap_sem); >>>> vcpu->run->s.regs.gprs[reg1] &= ~0xff; >>>> vcpu->run->s.regs.gprs[reg1] |= key; >>>> return 0; >>>> @@ -281,8 +292,9 @@ static int handle_iske(struct kvm_vcpu *vcpu) >>>> >>>> static int handle_rrbe(struct kvm_vcpu *vcpu) >>>> { >>>> - unsigned long addr; >>>> + unsigned long vmaddr, gaddr; >>>> int reg1, reg2; >>>> + bool unlocked; >>>> int rc; >>>> >>>> vcpu->stat.instruction_rrbe++; >>>> @@ -296,19 +308,27 @@ static int handle_rrbe(struct kvm_vcpu *vcpu) >>>> >>>> kvm_s390_get_regs_rre(vcpu, ®1, ®2); >>>> >>>> - addr = vcpu->run->s.regs.gprs[reg2] & PAGE_MASK; >>>> - addr = kvm_s390_logical_to_effective(vcpu, addr); >>>> - addr = kvm_s390_real_to_abs(vcpu, addr); >>>> - addr = gfn_to_hva(vcpu->kvm, gpa_to_gfn(addr)); >>>> - if (kvm_is_error_hva(addr)) >>>> + gaddr = vcpu->run->s.regs.gprs[reg2] & PAGE_MASK; >>>> + gaddr = kvm_s390_logical_to_effective(vcpu, gaddr); >>>> + gaddr = kvm_s390_real_to_abs(vcpu, gaddr); >>>> + vmaddr = gfn_to_hva(vcpu->kvm, gpa_to_gfn(gaddr)); >>>> + if (kvm_is_error_hva(vmaddr)) >>>> return kvm_s390_inject_program_int(vcpu, PGM_ADDRESSING); >>>> - >>>> +retry: >>>> + unlocked = false; >>>> down_read(¤t->mm->mmap_sem); >>>> - rc = reset_guest_reference_bit(current->mm, addr); >>>> - up_read(¤t->mm->mmap_sem); >>>> + rc = reset_guest_reference_bit(current->mm, vmaddr); >>>> + if (rc < 0) { >>>> + rc = fixup_user_fault(current, current->mm, vmaddr, >>>> + FAULT_FLAG_WRITE, &unlocked); >>>> + if (!rc) { >>>> + up_read(¤t->mm->mmap_sem); >>>> + goto retry; >>>> + } >>>> + } >>>> if (rc < 0) >>>> return kvm_s390_inject_program_int(vcpu, PGM_ADDRESSING); >>>> - >>>> + up_read(¤t->mm->mmap_sem); >>>> kvm_s390_set_psw_cc(vcpu, rc); >>>> return 0; >>>> } >>>> @@ -323,6 +343,7 @@ static int handle_sske(struct kvm_vcpu *vcpu) >>>> unsigned long start, end; >>>> unsigned char key, oldkey; >>>> int reg1, reg2; >>>> + bool unlocked; >>>> int rc; >>>> >>>> vcpu->stat.instruction_sske++; >>>> @@ -355,19 +376,28 @@ static int handle_sske(struct kvm_vcpu *vcpu) >>>> } >>>> >>>> while (start != end) { >>>> - unsigned long addr = gfn_to_hva(vcpu->kvm, gpa_to_gfn(start)); >>>> + unsigned long vmaddr = gfn_to_hva(vcpu->kvm, gpa_to_gfn(start)); >>>> + unlocked = false; >>>> >>>> - if (kvm_is_error_hva(addr)) >>>> + if (kvm_is_error_hva(vmaddr)) >>>> return kvm_s390_inject_program_int(vcpu, PGM_ADDRESSING); >>>> >>>> down_read(¤t->mm->mmap_sem); >>>> - rc = cond_set_guest_storage_key(current->mm, addr, key, &oldkey, >>>> + rc = cond_set_guest_storage_key(current->mm, vmaddr, key, &oldkey, >>>> m3 & SSKE_NQ, m3 & SSKE_MR, >>>> m3 & SSKE_MC); >>>> - up_read(¤t->mm->mmap_sem); >>>> - if (rc < 0) >>>> + >>>> + if (rc < 0) { >>>> + rc = fixup_user_fault(current, current->mm, vmaddr, >>>> + FAULT_FLAG_WRITE, &unlocked); >>>> + rc = !rc ? -EAGAIN : rc; >>>> + } >>>> + if (rc == -EFAULT) >>>> return kvm_s390_inject_program_int(vcpu, PGM_ADDRESSING); >>>> - start += PAGE_SIZE; >>>> + >>>> + up_read(¤t->mm->mmap_sem); >>>> + if (rc >= 0) >>>> + start += PAGE_SIZE; >>>> } >>>> >>>> if (m3 & (SSKE_MC | SSKE_MR)) { >>>> @@ -948,15 +978,16 @@ static int handle_pfmf(struct kvm_vcpu *vcpu) >>>> } >>>> >>>> while (start != end) { >>>> - unsigned long useraddr; >>>> + unsigned long vmaddr; >>>> + bool unlocked = false; >>>> >>>> /* Translate guest address to host address */ >>>> - useraddr = gfn_to_hva(vcpu->kvm, gpa_to_gfn(start)); >>>> - if (kvm_is_error_hva(useraddr)) >>>> + vmaddr = gfn_to_hva(vcpu->kvm, gpa_to_gfn(start)); >>>> + if (kvm_is_error_hva(vmaddr)) >>>> return kvm_s390_inject_program_int(vcpu, PGM_ADDRESSING); >>>> >>>> if (vcpu->run->s.regs.gprs[reg1] & PFMF_CF) { >>>> - if (clear_user((void __user *)useraddr, PAGE_SIZE)) >>>> + if (clear_user((void __user *)vmaddr, PAGE_SIZE)) >>>> return kvm_s390_inject_program_int(vcpu, PGM_ADDRESSING); >>>> } >>> >>> What happened to the patch >>> >>> >>> commit 0230cae75df62de5813a4ca39a425ba439d036da >>> Author: Janosch Frank <frankja@xxxxxxxxxxxxx> >>> Date: Wed Jul 4 15:50:01 2018 +0100 >>> >>> KVM: s390: Replace clear_user with kvm_clear_guest >>> >>> kvm_clear_guest also does the dirty tracking for us, which we want to >>> have. >> >> Patch 1/19 >> > > Yes, but this patch does not respect 1/19. Cherry picking/applying the > patches one by one results and a conflict at this very spot. > > As said, the final result (on the branch and merged) looks fine, but > looking at this series (including the branch) I have no clue where this > merge problem is actually resolved. I can neither spot it in this series > nor on the branch. > > It's git magic. Its resolved in the merge Merge tag 'hlp_stage1' of git://git.kernel.org/.../kvms390/linux into kvms390/next