[PATCH V3 1/2] pseries/eeh: Handle RTAS delay requests in configure_bridge

Gavin Shan gwshan at linux.vnet.ibm.com
Fri Apr 8 09:45:18 AEST 2016


On Thu, Apr 07, 2016 at 04:28:26PM +1000, Russell Currey wrote:
>In the "ibm,configure-pe" and "ibm,configure-bridge" RTAS calls, the
>spec states that values of 9900-9905 can be returned, indicating that
>software should delay for 10^x (where x is the last digit, i.e. 990x)
>milliseconds and attempt the call again. Currently, the kernel doesn't
>know about this, and respecting it fixes some PCI failures when the
>hypervisor is busy.
>
>The delay is capped at 0.2 seconds.
>
>Cc: <stable at vger.kernel.org> # 3.10+
>Signed-off-by: Russell Currey <ruscur at russell.cc>

Acked-by: Gavin Shan <gwshan at linux.vnet.ibm.com>

>---
>V3 changelog:
> - Refactorings and rewordings thanks to Gavin
> - Treat return values >9902 as 9902 thanks to Tyrel
>---
> arch/powerpc/platforms/pseries/eeh_pseries.c | 51 ++++++++++++++++++++--------
> 1 file changed, 36 insertions(+), 15 deletions(-)
>
>diff --git a/arch/powerpc/platforms/pseries/eeh_pseries.c b/arch/powerpc/platforms/pseries/eeh_pseries.c
>index ac3ffd9..405baaf 100644
>--- a/arch/powerpc/platforms/pseries/eeh_pseries.c
>+++ b/arch/powerpc/platforms/pseries/eeh_pseries.c
>@@ -615,29 +615,50 @@ static int pseries_eeh_configure_bridge(struct eeh_pe *pe)
> {
> 	int config_addr;
> 	int ret;
>+	/* Waiting 0.2s maximum before skipping configuration */
>+	int max_wait = 200;
> 
> 	/* Figure out the PE address */
> 	config_addr = pe->config_addr;
> 	if (pe->addr)
> 		config_addr = pe->addr;
> 
>-	/* Use new configure-pe function, if supported */
>-	if (ibm_configure_pe != RTAS_UNKNOWN_SERVICE) {
>-		ret = rtas_call(ibm_configure_pe, 3, 1, NULL,
>-				config_addr, BUID_HI(pe->phb->buid),
>-				BUID_LO(pe->phb->buid));
>-	} else if (ibm_configure_bridge != RTAS_UNKNOWN_SERVICE) {
>-		ret = rtas_call(ibm_configure_bridge, 3, 1, NULL,
>-				config_addr, BUID_HI(pe->phb->buid),
>-				BUID_LO(pe->phb->buid));
>-	} else {
>-		return -EFAULT;
>-	}
>+	while (max_wait > 0) {
>+		/* Use new configure-pe function, if supported */
>+		if (ibm_configure_pe != RTAS_UNKNOWN_SERVICE) {
>+			ret = rtas_call(ibm_configure_pe, 3, 1, NULL,
>+					config_addr, BUID_HI(pe->phb->buid),
>+					BUID_LO(pe->phb->buid));
>+		} else if (ibm_configure_bridge != RTAS_UNKNOWN_SERVICE) {
>+			ret = rtas_call(ibm_configure_bridge, 3, 1, NULL,
>+					config_addr, BUID_HI(pe->phb->buid),
>+					BUID_LO(pe->phb->buid));
>+		} else {
>+			return -EFAULT;
>+		}
> 
>-	if (ret)
>-		pr_warn("%s: Unable to configure bridge PHB#%d-PE#%x (%d)\n",
>-			__func__, pe->phb->global_number, pe->addr, ret);
>+		if (!ret)
>+			return ret;
>+
>+		/*
>+		 * If RTAS returns a delay value that's above 100ms, cut it
>+		 * down to 100ms in case firmware made a mistake.  For more
>+		 * on how these delay values work see rtas_busy_delay_time
>+		 */
>+		if (ret > RTAS_EXTENDED_DELAY_MIN+2 &&
>+		    ret <= RTAS_EXTENDED_DELAY_MAX)
>+			ret = RTAS_EXTENDED_DELAY_MIN+2;
>+
>+		max_wait -= rtas_busy_delay_time(ret);
>+
>+		if (max_wait < 0)
>+			break;
>+
>+		rtas_busy_delay(ret);
>+	}
> 
>+	pr_warn("%s: Unable to configure bridge PHB#%d-PE#%x (%d)\n",
>+		__func__, pe->phb->global_number, pe->addr, ret);
> 	return ret;
> }
> 
>-- 
>2.8.0
>
>_______________________________________________
>Linuxppc-dev mailing list
>Linuxppc-dev at lists.ozlabs.org
>https://lists.ozlabs.org/listinfo/linuxppc-dev



More information about the Linuxppc-dev mailing list