mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	s390/vfio-ap: add s390dbf logging to the vfio_ap_irq_enable function
This patch adds s390dbf logging to the function that executes the PQAP(AQIC) instruction on behalf of the guest to which the queue for which interrupts are being enabled or disabled is attached. Currently, the vfio_ap_irq_enable function sets status response code 06 (notification indicator byte address (nib) invalid) in the status word when the vfio_pin_pages function - called to pin the page containing the nib - returns an error or a different number of pages pinned than requested. Setting the response code returned to userspace without also logging a message in the kernel makes it impossible to determine whether the response was due to an error detected by the vfio_ap device driver or because the response code was returned by the firmware in response to the PQAP(AQIC) instruction. In addition to logging a warning for the situation above, this patch adds the following: * A function to validate the nib address invoked prior to calling the vfio_pin_pages function. This allows for logging a message informing the reader of the reason the page containing the nib can not be pinned if the nib address is not valid. Response code 06 (invalid nib address) will be set in the status word returned to the guest from the instruction. * Checks the return value from the kvm_s390_gisc_register and logs a message informing the reader of the failure. Status response code 08 (invalid gisa) will be set in the status word returned to the guest from the PQAP(AQIC) instruction. * Checks the status response code returned from execution of the PQAP(AQIC) instruction and if it indicates an error, logs a message informing the reader. Signed-off-by: Tony Krowiak <akrowiak@linux.ibm.com> Reviewed-by: Matthew Rosato <mjrosato@linux.ibm.com> Acked-by: Heiko Carstens <hca@linux.ibm.com> Signed-off-by: Vasily Gorbik <gor@linux.ibm.com>
This commit is contained in:
		
							parent
							
								
									68f554b7d2
								
							
						
					
					
						commit
						783f0a3ccd
					
				| @ -47,6 +47,7 @@ static inline int ap_test_bit(unsigned int *ptr, unsigned int nr) | |||||||
| #define AP_RESPONSE_BUSY		0x05 | #define AP_RESPONSE_BUSY		0x05 | ||||||
| #define AP_RESPONSE_INVALID_ADDRESS	0x06 | #define AP_RESPONSE_INVALID_ADDRESS	0x06 | ||||||
| #define AP_RESPONSE_OTHERWISE_CHANGED	0x07 | #define AP_RESPONSE_OTHERWISE_CHANGED	0x07 | ||||||
|  | #define AP_RESPONSE_INVALID_GISA	0x08 | ||||||
| #define AP_RESPONSE_Q_FULL		0x10 | #define AP_RESPONSE_Q_FULL		0x10 | ||||||
| #define AP_RESPONSE_NO_PENDING_REPLY	0x10 | #define AP_RESPONSE_NO_PENDING_REPLY	0x10 | ||||||
| #define AP_RESPONSE_INDEX_TOO_BIG	0x11 | #define AP_RESPONSE_INDEX_TOO_BIG	0x11 | ||||||
|  | |||||||
| @ -185,12 +185,44 @@ end_free: | |||||||
| 	return status; | 	return status; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /**
 | ||||||
|  |  * vfio_ap_validate_nib - validate a notification indicator byte (nib) address. | ||||||
|  |  * | ||||||
|  |  * @vcpu: the object representing the vcpu executing the PQAP(AQIC) instruction. | ||||||
|  |  * @nib: the location for storing the nib address. | ||||||
|  |  * @g_pfn: the location for storing the page frame number of the page containing | ||||||
|  |  *	   the nib. | ||||||
|  |  * | ||||||
|  |  * When the PQAP(AQIC) instruction is executed, general register 2 contains the | ||||||
|  |  * address of the notification indicator byte (nib) used for IRQ notification. | ||||||
|  |  * This function parses the nib from gr2 and calculates the page frame | ||||||
|  |  * number for the guest of the page containing the nib. The values are | ||||||
|  |  * stored in @nib and @g_pfn respectively. | ||||||
|  |  * | ||||||
|  |  * The g_pfn of the nib is then validated to ensure the nib address is valid. | ||||||
|  |  * | ||||||
|  |  * Return: returns zero if the nib address is a valid; otherwise, returns | ||||||
|  |  *	   -EINVAL. | ||||||
|  |  */ | ||||||
|  | static int vfio_ap_validate_nib(struct kvm_vcpu *vcpu, unsigned long *nib, | ||||||
|  | 				unsigned long *g_pfn) | ||||||
|  | { | ||||||
|  | 	*nib = vcpu->run->s.regs.gprs[2]; | ||||||
|  | 	*g_pfn = *nib >> PAGE_SHIFT; | ||||||
|  | 
 | ||||||
|  | 	if (kvm_is_error_hva(gfn_to_hva(vcpu->kvm, *g_pfn))) | ||||||
|  | 		return -EINVAL; | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * vfio_ap_irq_enable - Enable Interruption for a APQN |  * vfio_ap_irq_enable - Enable Interruption for a APQN | ||||||
|  * |  * | ||||||
|  * @q:	 the vfio_ap_queue holding AQIC parameters |  * @q:	 the vfio_ap_queue holding AQIC parameters | ||||||
|  * @isc: the guest ISC to register with the GIB interface |  * @isc: the guest ISC to register with the GIB interface | ||||||
|  * @nib: the notification indicator byte to pin. |  * @vcpu: the vcpu object containing the registers specifying the parameters | ||||||
|  |  *	  passed to the PQAP(AQIC) instruction. | ||||||
|  * |  * | ||||||
|  * Pin the NIB saved in *q |  * Pin the NIB saved in *q | ||||||
|  * Register the guest ISC to GIB interface and retrieve the |  * Register the guest ISC to GIB interface and retrieve the | ||||||
| @ -206,22 +238,36 @@ end_free: | |||||||
|  */ |  */ | ||||||
| static struct ap_queue_status vfio_ap_irq_enable(struct vfio_ap_queue *q, | static struct ap_queue_status vfio_ap_irq_enable(struct vfio_ap_queue *q, | ||||||
| 						 int isc, | 						 int isc, | ||||||
| 						 unsigned long nib) | 						 struct kvm_vcpu *vcpu) | ||||||
| { | { | ||||||
|  | 	unsigned long nib; | ||||||
| 	struct ap_qirq_ctrl aqic_gisa = {}; | 	struct ap_qirq_ctrl aqic_gisa = {}; | ||||||
| 	struct ap_queue_status status = {}; | 	struct ap_queue_status status = {}; | ||||||
| 	struct kvm_s390_gisa *gisa; | 	struct kvm_s390_gisa *gisa; | ||||||
|  | 	int nisc; | ||||||
| 	struct kvm *kvm; | 	struct kvm *kvm; | ||||||
| 	unsigned long h_nib, g_pfn, h_pfn; | 	unsigned long h_nib, g_pfn, h_pfn; | ||||||
| 	int ret; | 	int ret; | ||||||
| 
 | 
 | ||||||
| 	g_pfn = nib >> PAGE_SHIFT; | 	/* Verify that the notification indicator byte address is valid */ | ||||||
|  | 	if (vfio_ap_validate_nib(vcpu, &nib, &g_pfn)) { | ||||||
|  | 		VFIO_AP_DBF_WARN("%s: invalid NIB address: nib=%#lx, g_pfn=%#lx, apqn=%#04x\n", | ||||||
|  | 				 __func__, nib, g_pfn, q->apqn); | ||||||
|  | 
 | ||||||
|  | 		status.response_code = AP_RESPONSE_INVALID_ADDRESS; | ||||||
|  | 		return status; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	ret = vfio_pin_pages(mdev_dev(q->matrix_mdev->mdev), &g_pfn, 1, | 	ret = vfio_pin_pages(mdev_dev(q->matrix_mdev->mdev), &g_pfn, 1, | ||||||
| 			     IOMMU_READ | IOMMU_WRITE, &h_pfn); | 			     IOMMU_READ | IOMMU_WRITE, &h_pfn); | ||||||
| 	switch (ret) { | 	switch (ret) { | ||||||
| 	case 1: | 	case 1: | ||||||
| 		break; | 		break; | ||||||
| 	default: | 	default: | ||||||
|  | 		VFIO_AP_DBF_WARN("%s: vfio_pin_pages failed: rc=%d," | ||||||
|  | 				 "nib=%#lx, g_pfn=%#lx, apqn=%#04x\n", | ||||||
|  | 				 __func__, ret, nib, g_pfn, q->apqn); | ||||||
|  | 
 | ||||||
| 		status.response_code = AP_RESPONSE_INVALID_ADDRESS; | 		status.response_code = AP_RESPONSE_INVALID_ADDRESS; | ||||||
| 		return status; | 		return status; | ||||||
| 	} | 	} | ||||||
| @ -231,7 +277,17 @@ static struct ap_queue_status vfio_ap_irq_enable(struct vfio_ap_queue *q, | |||||||
| 
 | 
 | ||||||
| 	h_nib = (h_pfn << PAGE_SHIFT) | (nib & ~PAGE_MASK); | 	h_nib = (h_pfn << PAGE_SHIFT) | (nib & ~PAGE_MASK); | ||||||
| 	aqic_gisa.gisc = isc; | 	aqic_gisa.gisc = isc; | ||||||
| 	aqic_gisa.isc = kvm_s390_gisc_register(kvm, isc); | 
 | ||||||
|  | 	nisc = kvm_s390_gisc_register(kvm, isc); | ||||||
|  | 	if (nisc < 0) { | ||||||
|  | 		VFIO_AP_DBF_WARN("%s: gisc registration failed: nisc=%d, isc=%d, apqn=%#04x\n", | ||||||
|  | 				 __func__, nisc, isc, q->apqn); | ||||||
|  | 
 | ||||||
|  | 		status.response_code = AP_RESPONSE_INVALID_GISA; | ||||||
|  | 		return status; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	aqic_gisa.isc = nisc; | ||||||
| 	aqic_gisa.ir = 1; | 	aqic_gisa.ir = 1; | ||||||
| 	aqic_gisa.gisa = (uint64_t)gisa >> 4; | 	aqic_gisa.gisa = (uint64_t)gisa >> 4; | ||||||
| 
 | 
 | ||||||
| @ -255,6 +311,16 @@ static struct ap_queue_status vfio_ap_irq_enable(struct vfio_ap_queue *q, | |||||||
| 		break; | 		break; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if (status.response_code != AP_RESPONSE_NORMAL) { | ||||||
|  | 		VFIO_AP_DBF_WARN("%s: PQAP(AQIC) failed with status=%#02x: " | ||||||
|  | 				 "zone=%#x, ir=%#x, gisc=%#x, f=%#x," | ||||||
|  | 				 "gisa=%#x, isc=%#x, apqn=%#04x\n", | ||||||
|  | 				 __func__, status.response_code, | ||||||
|  | 				 aqic_gisa.zone, aqic_gisa.ir, aqic_gisa.gisc, | ||||||
|  | 				 aqic_gisa.gf, aqic_gisa.gisa, aqic_gisa.isc, | ||||||
|  | 				 q->apqn); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	return status; | 	return status; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -372,8 +438,7 @@ static int handle_pqap(struct kvm_vcpu *vcpu) | |||||||
| 
 | 
 | ||||||
| 	/* If IR bit(16) is set we enable the interrupt */ | 	/* If IR bit(16) is set we enable the interrupt */ | ||||||
| 	if ((status >> (63 - 16)) & 0x01) | 	if ((status >> (63 - 16)) & 0x01) | ||||||
| 		qstatus = vfio_ap_irq_enable(q, status & 0x07, | 		qstatus = vfio_ap_irq_enable(q, status & 0x07, vcpu); | ||||||
| 					     vcpu->run->s.regs.gprs[2]); |  | ||||||
| 	else | 	else | ||||||
| 		qstatus = vfio_ap_irq_disable(q); | 		qstatus = vfio_ap_irq_disable(q); | ||||||
| 
 | 
 | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Tony Krowiak
						Tony Krowiak