Received: by 2002:a25:8b91:0:0:0:0:0 with SMTP id j17csp1709213ybl; Wed, 8 Jan 2020 23:50:47 -0800 (PST) X-Google-Smtp-Source: APXvYqxiLWczR0o3Ts0LsXgnUGQdRtx/tUx7z+z5FnXmmUBnWExQN+SwQ+98uEj4jhhNxrYKyRmw X-Received: by 2002:a9d:3bc4:: with SMTP id k62mr7767479otc.186.1578556246982; Wed, 08 Jan 2020 23:50:46 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1578556246; cv=none; d=google.com; s=arc-20160816; b=fjr+aIRxbhIJLmjmNyrTcD8X1HC5Xf+ctB9yIR5LxwTuOfa4Jb8DvGelM6IFwlBJCI z23bBH5MlwrPX4TNfOijT8pg72ldJFv0LdU8jzF28zhMMC+2CNmqI43li/zqOZkroagI gnMnMLDL8mzJGpI2GMuJkTORql0idWZChR64tzjNUuIIDSp5Mhbnk9Ep83JFF1RaGvKO ugENnuNFyUUUkHy55DWXaMEAzhGhmDjQVBTmwR7RiFB7u9CrDxGl6K4FjuBDZsnjS7us +WPdolIK/qV7x4yVlNGa1QgIGTzzhlZq4WPVnM+XZFc/j4+y8qu4j+aRFQG3Do5QUx5a /kTg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:in-reply-to:content-disposition :mime-version:references:mail-followup-to:message-id:subject:cc:to :from:date:ironport-sdr; bh=V9f7YSclU4cEUkRX4wA43rOzGWQPPAyn0KBEA1tZG38=; b=iaG+KxFcO6MIKtreN62hVbP1nbuMfyxghgNp9tlv7De6pP7UxLljaJqoua3cmnvgoo 2+yvjIXk3oHogSmtzmpNXSyAfje6fsjo0CCbY95o8ED3JjXki5g6XgQ+2irrATyDdluv PZN4ChOrUCJoq8JndUwM+hBAT9wrWyIRXhtRVL4RTm7n/ARUbuSPe1+9QRHBj+asglcx mODMyAmtF0LUIT0pbGsK8pcN97MvWHqeho5ul29sdE8WZNLgUMbx2ZEQ0dztdhlJxxQJ 44dfTqF4qR1QnWUCFMoJ7jtVDDQH7gtnHaHsjRQ1KhJ6ox7rX/jiCY9RSSDPi7DBSA3L r4xA== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=microchip.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id i6si3302324otp.5.2020.01.08.23.50.34; Wed, 08 Jan 2020 23:50:46 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=microchip.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1728347AbgAIHsw (ORCPT + 99 others); Thu, 9 Jan 2020 02:48:52 -0500 Received: from esa3.microchip.iphmx.com ([68.232.153.233]:28384 "EHLO esa3.microchip.iphmx.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1728184AbgAIHsv (ORCPT ); Thu, 9 Jan 2020 02:48:51 -0500 Received-SPF: Pass (esa3.microchip.iphmx.com: domain of Ludovic.Desroches@microchip.com designates 198.175.253.82 as permitted sender) identity=mailfrom; client-ip=198.175.253.82; receiver=esa3.microchip.iphmx.com; envelope-from="Ludovic.Desroches@microchip.com"; x-sender="Ludovic.Desroches@microchip.com"; x-conformance=spf_only; x-record-type="v=spf1"; x-record-text="v=spf1 mx a:ushub1.microchip.com a:smtpout.microchip.com -exists:%{i}.spf.microchip.iphmx.com include:servers.mcsv.net include:mktomail.com include:spf.protection.outlook.com ~all" Received-SPF: None (esa3.microchip.iphmx.com: no sender authenticity information available from domain of postmaster@email.microchip.com) identity=helo; client-ip=198.175.253.82; receiver=esa3.microchip.iphmx.com; envelope-from="Ludovic.Desroches@microchip.com"; x-sender="postmaster@email.microchip.com"; x-conformance=spf_only Authentication-Results: esa3.microchip.iphmx.com; dkim=none (message not signed) header.i=none; spf=Pass smtp.mailfrom=Ludovic.Desroches@microchip.com; spf=None smtp.helo=postmaster@email.microchip.com; dmarc=pass (p=none dis=none) d=microchip.com IronPort-SDR: rG1Whl6gGJ0wvj4joH9EGcUs6mAaYZ9l6hAj/qsmuIp8l+Z8h4ITGtskG3vkjGQxC+an+0UsR/ TCsdTvsoFVGNsVs4nii1bsrwvqlODakTvaFWRt2TuUuqLOuCrnnm5I4EiGGloSMU1yWc56bQXN RYqLAjkDs6uzNqERXeEN9LqV6ybHvnqHoHhnDnHxKY9lXxaBU4mb3yp41hwTEEGl+z7R0d+y/z Hc/H6EMyw2GxJ4FvlyXI2n4X4bQKyU0OB1XfHfHMVD7KX9ddw0peK6Qj+MXOPEqzQKp3/MeCxG REI= X-IronPort-AV: E=Sophos;i="5.69,413,1571727600"; d="scan'208";a="62680662" Received: from smtpout.microchip.com (HELO email.microchip.com) ([198.175.253.82]) by esa3.microchip.iphmx.com with ESMTP/TLS/AES256-SHA256; 09 Jan 2020 00:48:50 -0700 Received: from chn-vm-ex03.mchp-main.com (10.10.85.151) by chn-vm-ex03.mchp-main.com (10.10.85.151) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256) id 15.1.1713.5; Thu, 9 Jan 2020 00:48:48 -0700 Received: from localhost (10.10.85.251) by chn-vm-ex03.mchp-main.com (10.10.85.151) with Microsoft SMTP Server id 15.1.1713.5 via Frontend Transport; Thu, 9 Jan 2020 00:48:48 -0700 Date: Thu, 9 Jan 2020 08:48:19 +0100 From: Ludovic Desroches To: Codrin Ciubotariu - M19940 CC: "linux-i2c@vger.kernel.org" , "devicetree@vger.kernel.org" , "linux-arm-kernel@lists.infradead.org" , "linux-kernel@vger.kernel.org" , "kamel.bouhara@bootlin.com" , "wsa@the-dreams.de" , "Nicolas Ferre - M43238" , "alexandre.belloni@bootlin.com" , "robh@kernel.org" Subject: Re: [PATCH v2 2/6] i2c: at91: implement i2c bus recovery Message-ID: <20200109074819.rhlaxg3sgwlng5xm@M43218.corp.atmel.com> Mail-Followup-To: Codrin Ciubotariu - M19940 , "linux-i2c@vger.kernel.org" , "devicetree@vger.kernel.org" , "linux-arm-kernel@lists.infradead.org" , "linux-kernel@vger.kernel.org" , "kamel.bouhara@bootlin.com" , "wsa@the-dreams.de" , Nicolas Ferre - M43238 , "alexandre.belloni@bootlin.com" , "robh@kernel.org" References: <20200103094821.13185-1-codrin.ciubotariu@microchip.com> <20200103094821.13185-3-codrin.ciubotariu@microchip.com> MIME-Version: 1.0 Content-Type: text/plain; charset="us-ascii" Content-Disposition: inline In-Reply-To: <20200103094821.13185-3-codrin.ciubotariu@microchip.com> Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Fri, Jan 03, 2020 at 09:49:07AM +0000, Codrin Ciubotariu - M19940 wrote: > From: Kamel Bouhara > > Implement i2c bus recovery when slaves devices might hold SDA low. > In this case re-assign SCL/SDA to gpios and issue 9 dummy clock pulses > until the slave release SDA. > > Signed-off-by: Kamel Bouhara > [codrin.ciubotariu@microchip.com: release gpios on error; move i2c_recover_bus] > Signed-off-by: Codrin Ciubotariu > --- > > Changes in v2: > - called i2c_recover_bus() after an error occurs, if SDA is down; > - release gpios if recovery information is incomplete; > > drivers/i2c/busses/i2c-at91-master.c | 79 ++++++++++++++++++++++++++++ > drivers/i2c/busses/i2c-at91.h | 8 +++ > 2 files changed, 87 insertions(+) > > diff --git a/drivers/i2c/busses/i2c-at91-master.c b/drivers/i2c/busses/i2c-at91-master.c > index 7a862e00b475..c55e8ff35201 100644 > --- a/drivers/i2c/busses/i2c-at91-master.c > +++ b/drivers/i2c/busses/i2c-at91-master.c > @@ -18,11 +18,13 @@ > #include > #include > #include > +#include > #include > #include > #include > #include > #include > +#include > #include > #include > #include > @@ -478,6 +480,7 @@ static int at91_do_twi_transfer(struct at91_twi_dev *dev) > unsigned long time_left; > bool has_unre_flag = dev->pdata->has_unre_flag; > bool has_alt_cmd = dev->pdata->has_alt_cmd; > + struct i2c_bus_recovery_info *rinfo = &dev->rinfo; > > /* > * WARNING: the TXCOMP bit in the Status Register is NOT a clear on > @@ -637,6 +640,13 @@ static int at91_do_twi_transfer(struct at91_twi_dev *dev) > at91_twi_write(dev, AT91_TWI_CR, > AT91_TWI_THRCLR | AT91_TWI_LOCKCLR); > } > + > + if (rinfo->get_sda && !(rinfo->get_sda(&dev->adapter))) { > + dev_dbg(dev->dev, > + "SDA is down; clear bus using gpio\n"); > + i2c_recover_bus(&dev->adapter); > + } > + > return ret; > } > > @@ -806,6 +816,71 @@ static int at91_twi_configure_dma(struct at91_twi_dev *dev, u32 phy_addr) > return ret; > } > > +static void at91_prepare_twi_recovery(struct i2c_adapter *adap) > +{ > + struct at91_twi_dev *dev = i2c_get_adapdata(adap); > + > + pinctrl_select_state(dev->pinctrl, dev->pinctrl_pins_gpio); > +} > + > +static void at91_unprepare_twi_recovery(struct i2c_adapter *adap) > +{ > + struct at91_twi_dev *dev = i2c_get_adapdata(adap); > + > + pinctrl_select_state(dev->pinctrl, dev->pinctrl_pins_default); > +} > + > +static int at91_init_twi_recovery_info(struct platform_device *pdev, > + struct at91_twi_dev *dev) > +{ > + struct i2c_bus_recovery_info *rinfo = &dev->rinfo; > + > + dev->pinctrl = devm_pinctrl_get(&pdev->dev); > + if (!dev->pinctrl || IS_ERR(dev->pinctrl)) { > + dev_info(dev->dev, "can't get pinctrl, bus recovery not supported\n"); > + return PTR_ERR(dev->pinctrl); > + } > + > + dev->pinctrl_pins_default = pinctrl_lookup_state(dev->pinctrl, > + PINCTRL_STATE_DEFAULT); > + dev->pinctrl_pins_gpio = pinctrl_lookup_state(dev->pinctrl, > + "gpio"); > + rinfo->sda_gpiod = devm_gpiod_get(&pdev->dev, "sda", GPIOD_IN); > + if (PTR_ERR(rinfo->sda_gpiod) == -EPROBE_DEFER) > + return -EPROBE_DEFER; > + > + rinfo->scl_gpiod = devm_gpiod_get(&pdev->dev, "scl", > + GPIOD_OUT_HIGH_OPEN_DRAIN); > + if (PTR_ERR(rinfo->scl_gpiod) == -EPROBE_DEFER) > + return -EPROBE_DEFER; > + > + if (IS_ERR(rinfo->sda_gpiod) || > + IS_ERR(rinfo->scl_gpiod) || > + IS_ERR(dev->pinctrl_pins_default) || > + IS_ERR(dev->pinctrl_pins_gpio)) { > + dev_info(&pdev->dev, "recovery information incomplete\n"); > + if (!IS_ERR(rinfo->sda_gpiod)) { > + gpiod_put(rinfo->sda_gpiod); > + rinfo->sda_gpiod = NULL; > + } > + if (!IS_ERR(rinfo->scl_gpiod)) { > + gpiod_put(rinfo->scl_gpiod); > + rinfo->scl_gpiod = NULL; > + } > + return -EINVAL; > + } > + > + dev_info(&pdev->dev, "using scl%s for recovery\n", > + rinfo->sda_gpiod ? ",sda" : ""); Is it possible to have scl without sda? From the condition before, if there is an error for sda or scl, an error is returned. > + > + rinfo->prepare_recovery = at91_prepare_twi_recovery; > + rinfo->unprepare_recovery = at91_unprepare_twi_recovery; > + rinfo->recover_bus = i2c_generic_scl_recovery; > + dev->adapter.bus_recovery_info = rinfo; > + > + return 0; > +} > + > int at91_twi_probe_master(struct platform_device *pdev, > u32 phy_addr, struct at91_twi_dev *dev) > { > @@ -838,6 +913,10 @@ int at91_twi_probe_master(struct platform_device *pdev, > "i2c-analog-filter"); > at91_calc_twi_clock(dev); > > + rc = at91_init_twi_recovery_info(pdev, dev); > + if (rc == -EPROBE_DEFER) > + return rc; > + > dev->adapter.algo = &at91_twi_algorithm; > dev->adapter.quirks = &at91_twi_quirks; > > diff --git a/drivers/i2c/busses/i2c-at91.h b/drivers/i2c/busses/i2c-at91.h > index 977a67bc0f88..eb335b71e775 100644 > --- a/drivers/i2c/busses/i2c-at91.h > +++ b/drivers/i2c/busses/i2c-at91.h > @@ -151,6 +151,10 @@ struct at91_twi_dev { > u32 fifo_size; > struct at91_twi_dma dma; > bool slave_detected; > + struct i2c_bus_recovery_info rinfo; > + struct pinctrl *pinctrl; > + struct pinctrl_state *pinctrl_pins_default; > + struct pinctrl_state *pinctrl_pins_gpio; > #ifdef CONFIG_I2C_AT91_SLAVE_EXPERIMENTAL > unsigned smr; > struct i2c_client *slave; > @@ -171,6 +175,10 @@ void at91_init_twi_bus_master(struct at91_twi_dev *dev); > int at91_twi_probe_master(struct platform_device *pdev, u32 phy_addr, > struct at91_twi_dev *dev); > Wrong names here: > +void at91_twi_prepare_recovery(struct i2c_adapter *adap); at91_prepare_twi_recovery > +void at91_twi_unprepare_recovery(struct i2c_adapter *adap); at91_unprepare_twi_recovery > +void at91_twi_init_recovery_info(struct at91_twi_dev *dev); at91_init_twi_recovery_info Are these declarations still needed? Ludovic > + > #ifdef CONFIG_I2C_AT91_SLAVE_EXPERIMENTAL > void at91_init_twi_bus_slave(struct at91_twi_dev *dev); > int at91_twi_probe_slave(struct platform_device *pdev, u32 phy_addr, > -- > 2.20.1