diff options
Diffstat (limited to 'arch/s390/kvm/kvm-s390.c')
| -rw-r--r-- | arch/s390/kvm/kvm-s390.c | 39 | 
1 files changed, 24 insertions, 15 deletions
| diff --git a/arch/s390/kvm/kvm-s390.c b/arch/s390/kvm/kvm-s390.c index 98ad8b9e0360..2598cf243b86 100644 --- a/arch/s390/kvm/kvm-s390.c +++ b/arch/s390/kvm/kvm-s390.c @@ -1,11 +1,8 @@ +// SPDX-License-Identifier: GPL-2.0  /* - * hosting zSeries kernel virtual machines + * hosting IBM Z kernel virtual machines (s390x)   * - * Copyright IBM Corp. 2008, 2009 - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License (version 2 only) - * as published by the Free Software Foundation. + * Copyright IBM Corp. 2008, 2017   *   *    Author(s): Carsten Otte <[email protected]>   *               Christian Borntraeger <[email protected]> @@ -424,6 +421,9 @@ int kvm_vm_ioctl_check_extension(struct kvm *kvm, long ext)  	case KVM_CAP_S390_GS:  		r = test_facility(133);  		break; +	case KVM_CAP_S390_BPB: +		r = test_facility(82); +		break;  	default:  		r = 0;  	} @@ -795,11 +795,12 @@ static int kvm_s390_vm_start_migration(struct kvm *kvm)  	if (kvm->arch.use_cmma) {  		/* -		 * Get the last slot. They should be sorted by base_gfn, so the -		 * last slot is also the one at the end of the address space. -		 * We have verified above that at least one slot is present. +		 * Get the first slot. They are reverse sorted by base_gfn, so +		 * the first slot is also the one at the end of the address +		 * space. We have verified above that at least one slot is +		 * present.  		 */ -		ms = slots->memslots + slots->used_slots - 1; +		ms = slots->memslots;  		/* round up so we only use full longs */  		ram_pages = roundup(ms->base_gfn + ms->npages, BITS_PER_LONG);  		/* allocate enough bytes to store all the bits */ @@ -2200,6 +2201,8 @@ int kvm_arch_vcpu_init(struct kvm_vcpu *vcpu)  	kvm_s390_set_prefix(vcpu, 0);  	if (test_kvm_facility(vcpu->kvm, 64))  		vcpu->run->kvm_valid_regs |= KVM_SYNC_RICCB; +	if (test_kvm_facility(vcpu->kvm, 82)) +		vcpu->run->kvm_valid_regs |= KVM_SYNC_BPBC;  	if (test_kvm_facility(vcpu->kvm, 133))  		vcpu->run->kvm_valid_regs |= KVM_SYNC_GSCB;  	/* fprs can be synchronized via vrs, even if the guest has no vx. With @@ -2341,6 +2344,7 @@ static void kvm_s390_vcpu_initial_reset(struct kvm_vcpu *vcpu)  	current->thread.fpu.fpc = 0;  	vcpu->arch.sie_block->gbea = 1;  	vcpu->arch.sie_block->pp = 0; +	vcpu->arch.sie_block->fpf &= ~FPF_BPBC;  	vcpu->arch.pfault_token = KVM_S390_PFAULT_TOKEN_INVALID;  	kvm_clear_async_pf_completion_queue(vcpu);  	if (!kvm_s390_user_cpu_state_ctrl(vcpu->kvm)) @@ -3300,6 +3304,11 @@ static void sync_regs(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)  		vcpu->arch.sie_block->ecd |= ECD_HOSTREGMGMT;  		vcpu->arch.gs_enabled = 1;  	} +	if ((kvm_run->kvm_dirty_regs & KVM_SYNC_BPBC) && +	    test_kvm_facility(vcpu->kvm, 82)) { +		vcpu->arch.sie_block->fpf &= ~FPF_BPBC; +		vcpu->arch.sie_block->fpf |= kvm_run->s.regs.bpbc ? FPF_BPBC : 0; +	}  	save_access_regs(vcpu->arch.host_acrs);  	restore_access_regs(vcpu->run->s.regs.acrs);  	/* save host (userspace) fprs/vrs */ @@ -3346,6 +3355,7 @@ static void store_regs(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)  	kvm_run->s.regs.pft = vcpu->arch.pfault_token;  	kvm_run->s.regs.pfs = vcpu->arch.pfault_select;  	kvm_run->s.regs.pfc = vcpu->arch.pfault_compare; +	kvm_run->s.regs.bpbc = (vcpu->arch.sie_block->fpf & FPF_BPBC) == FPF_BPBC;  	save_access_regs(vcpu->run->s.regs.acrs);  	restore_access_regs(vcpu->arch.host_acrs);  	/* Save guest register state */ @@ -3372,7 +3382,6 @@ static void store_regs(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)  int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)  {  	int rc; -	sigset_t sigsaved;  	if (kvm_run->immediate_exit)  		return -EINTR; @@ -3382,8 +3391,7 @@ int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)  		return 0;  	} -	if (vcpu->sigset_active) -		sigprocmask(SIG_SETMASK, &vcpu->sigset, &sigsaved); +	kvm_sigset_activate(vcpu);  	if (!kvm_s390_user_cpu_state_ctrl(vcpu->kvm)) {  		kvm_s390_vcpu_start(vcpu); @@ -3417,8 +3425,7 @@ int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)  	disable_cpu_timer_accounting(vcpu);  	store_regs(vcpu, kvm_run); -	if (vcpu->sigset_active) -		sigprocmask(SIG_SETMASK, &sigsaved, NULL); +	kvm_sigset_deactivate(vcpu);  	vcpu->stat.exit_userspace++;  	return rc; @@ -3811,6 +3818,7 @@ long kvm_arch_vcpu_ioctl(struct file *filp,  			r = -EINVAL;  			break;  		} +		/* do not use irq_state.flags, it will break old QEMUs */  		r = kvm_s390_set_irq_state(vcpu,  					   (void __user *) irq_state.buf,  					   irq_state.len); @@ -3826,6 +3834,7 @@ long kvm_arch_vcpu_ioctl(struct file *filp,  			r = -EINVAL;  			break;  		} +		/* do not use irq_state.flags, it will break old QEMUs */  		r = kvm_s390_get_irq_state(vcpu,  					   (__u8 __user *)  irq_state.buf,  					   irq_state.len); |