Received: by 10.213.65.68 with SMTP id h4csp1384230imn; Mon, 19 Mar 2018 02:38:24 -0700 (PDT) X-Google-Smtp-Source: AG47ELuL3W1bWS03Y2rJ/1FrcSXA8VDT3nh1YrncTHw1QmCVJF/GrASHXFKXZxqbR/TlAXilxvjv X-Received: by 2002:a17:902:464:: with SMTP id 91-v6mr11655174ple.126.1521452303933; Mon, 19 Mar 2018 02:38:23 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1521452303; cv=none; d=google.com; s=arc-20160816; b=crBSlzssmo4o5nNUQF8RU+zfODWxybUrsaTalTnhHLWEtL+SXzWwswe89dfxFuJj2k mXjE+0141KLAY66gLyJHeV0e+C39MB5RvBJKg+7Mh+9z1JZPlkCOzEjz/TGYCApTj1jD j5iyAtxeRj++5pR6kbtQnEPUC2keV8X7ib8a0KkTCT4KmBbpy6M3caHYZ8VkXyIPN/76 pFW95Jk2/jUXVmjEGgYaSHJZ1RTiZGluHy9607tSwMNa2BSX5VB8B2qwPcAcUDrBuu+z 0EHfc72c+wjMJk54lVc9ylCUesoRjLPS8NtrldOcfI3AHbzfUcw1Ba62yXyNCoqT3P3V syUA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding :content-language:in-reply-to:mime-version:user-agent:date :message-id:from:references:to:subject:arc-authentication-results; bh=NuQ30uRJEvQ/SSxtHBJyaqMnfRmHBKEcAdmdTvJvTiM=; b=wEiTmJf9p+x1mCVxov4eW/br4LUB1NT8UrCFKdj4/4EYu0JDWpMTljWJPdI7uoSXt3 ZguIGxw/WSzeDuD4VjH4PBgxjgNvxJ3iVWhbMi8sHGbv9ah9f0kYccvVSdQH4ehrBM5g 4RqelBNhD85IOLNyc2tZEupc0SdF+hIVuCBFBRrczNcCksfKBb9NYzHlSd36tpB7uTry qgokkVezIHASMQ0u48ZhkD/d3CIiFCpCIvu+zioH+4Xk9Ju1EURplM4t4n5Ki8ZmMRAG lyQi2r3CrdIROpcVHDv1D+e1BuH6MJkNc+WcAy058xXtEZ8earDT82m2sn0mraRa8Uuf 9q/g== 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 Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id d16-v6si11645357pli.557.2018.03.19.02.38.10; Mon, 19 Mar 2018 02:38:23 -0700 (PDT) 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 Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S932541AbeCSJgq (ORCPT + 99 others); Mon, 19 Mar 2018 05:36:46 -0400 Received: from anchovy3.45ru.net.au ([203.30.46.155]:44679 "EHLO anchovy3.45ru.net.au" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S932328AbeCSJgn (ORCPT ); Mon, 19 Mar 2018 05:36:43 -0400 Received: (qmail 4144 invoked by uid 5089); 19 Mar 2018 09:36:41 -0000 Received: by simscan 1.2.0 ppid: 4071, pid: 4072, t: 0.0444s scanners: regex: 1.2.0 attach: 1.2.0 clamav: 0.88.3/m:40/d:1950 Received: from unknown (HELO ?192.168.0.122?) (preid@electromag.com.au@203.59.235.95) by anchovy2.45ru.net.au with ESMTPA; 19 Mar 2018 09:36:41 -0000 Subject: Re: [RESEND PATCH v1 6/6] i2c: i2c-stm32f7: Implement I2C recovery mechanism To: Pierre-Yves MORDRET , Wolfram Sang , Maxime Coquelin , Alexandre Torgue , linux-i2c@vger.kernel.org, linux-arm-kernel@lists.infradead.org, linux-kernel@vger.kernel.org References: <1520852023-27083-1-git-send-email-pierre-yves.mordret@st.com> <1520852023-27083-7-git-send-email-pierre-yves.mordret@st.com> From: Phil Reid Message-ID: Date: Mon, 19 Mar 2018 17:36:30 +0800 User-Agent: Mozilla/5.0 (Windows NT 10.0; WOW64; rv:52.0) Gecko/20100101 Thunderbird/52.6.0 MIME-Version: 1.0 In-Reply-To: <1520852023-27083-7-git-send-email-pierre-yves.mordret@st.com> Content-Type: text/plain; charset=utf-8; format=flowed Content-Language: en-AU Content-Transfer-Encoding: 7bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 12/03/2018 18:53, Pierre-Yves MORDRET wrote: > Feature prevents I2C lock-ups. Mechanism resets I2C state machine > and releases SCL/SDA signals but preserves I2C registers. > > Signed-off-by: Pierre-Yves MORDRET > --- > Version history: > v1: > * Initial > --- > --- > drivers/i2c/busses/i2c-stm32f7.c | 32 +++++++++++++++++++++++++++++--- > 1 file changed, 29 insertions(+), 3 deletions(-) > > diff --git a/drivers/i2c/busses/i2c-stm32f7.c b/drivers/i2c/busses/i2c-stm32f7.c > index 69a2e5e..3808bc9 100644 > --- a/drivers/i2c/busses/i2c-stm32f7.c > +++ b/drivers/i2c/busses/i2c-stm32f7.c > @@ -718,6 +718,20 @@ static void stm32f7_i2c_smbus_reload(struct stm32f7_i2c_dev *i2c_dev) > writel_relaxed(cr2, i2c_dev->base + STM32F7_I2C_CR2); > } > > +static int stm32f7_i2c_recover_bus(struct i2c_adapter *i2c_adap) > +{ > + struct stm32f7_i2c_dev *i2c_dev = i2c_get_adapdata(i2c_adap); > + > + dev_info(i2c_dev->dev, "Trying to recover bus\n"); > + > + stm32f7_i2c_clr_bits(i2c_dev->base + STM32F7_I2C_CR1, > + STM32F7_I2C_CR1_PE); This only "releases" the scl / sda on the stm32f7 end (outputs are hiz I guess). It doesn't trigger the scl clocking needed to recover a stuck device via i2c recovery from what I can see in the data sheet. > + > + stm32f7_i2c_hw_config(i2c_dev); Nothing in here either I think. > + > + return 0; > +} > + > static int stm32f7_i2c_wait_free_bus(struct stm32f7_i2c_dev *i2c_dev) > { > u32 status; > @@ -727,12 +741,18 @@ static int stm32f7_i2c_wait_free_bus(struct stm32f7_i2c_dev *i2c_dev) > status, > !(status & STM32F7_I2C_ISR_BUSY), > 10, 1000); > + if (!ret) > + return 0; > + > + dev_info(i2c_dev->dev, "bus busy\n"); > + > + ret = i2c_recover_bus(&i2c_dev->adap); > if (ret) { > - dev_dbg(i2c_dev->dev, "bus busy\n"); > - ret = -EBUSY; > + dev_err(i2c_dev->dev, "Failed to recover the bus (%d)\n", ret); > + return ret; > } > > - return ret; > + return -EBUSY; > } > > static void stm32f7_i2c_xfer_msg(struct stm32f7_i2c_dev *i2c_dev, > @@ -1476,6 +1496,7 @@ static irqreturn_t stm32f7_i2c_isr_error(int irq, void *data) > if (status & STM32F7_I2C_ISR_BERR) { > dev_err(dev, "<%s>: Bus error\n", __func__); > writel_relaxed(STM32F7_I2C_ICR_BERRCF, base + STM32F7_I2C_ICR); > + i2c_recover_bus(&i2c_dev->adap); > f7_msg->result = -EIO; > } > > @@ -1760,6 +1781,10 @@ static struct i2c_algorithm stm32f7_i2c_algo = { > .unreg_slave = stm32f7_i2c_unreg_slave, > }; > > +static struct i2c_bus_recovery_info stm32f7_i2c_recovery_info = { > + .recover_bus = stm32f7_i2c_recover_bus, > +}; > + > static int stm32f7_i2c_probe(struct platform_device *pdev) > { > struct device_node *np = pdev->dev.of_node; > @@ -1879,6 +1904,7 @@ static int stm32f7_i2c_probe(struct platform_device *pdev) > adap->algo = &stm32f7_i2c_algo; > adap->dev.parent = &pdev->dev; > adap->dev.of_node = pdev->dev.of_node; > + adap->bus_recovery_info = &stm32f7_i2c_recovery_info; > > init_completion(&i2c_dev->complete); > > -- Regards Phil