Received: by 2002:a25:ad19:0:0:0:0:0 with SMTP id y25csp10513455ybi; Thu, 11 Jul 2019 06:32:35 -0700 (PDT) X-Google-Smtp-Source: APXvYqzSLSzFuk+HDFpEkSZQWAcsxkVR51HchseuAJqNzVBZTAPolPw5gx18bkeXdBpk1IzSMap+ X-Received: by 2002:a17:90a:8b98:: with SMTP id z24mr4976855pjn.77.1562851955146; Thu, 11 Jul 2019 06:32:35 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1562851955; cv=none; d=google.com; s=arc-20160816; b=haMc2ZjtRT41vXn5PuQQVneB04lChHsEgF4FKCZ1+R5TZ55zTyyIhdUO9Zy3vLVqdY BhKQekViOGqner2OYywkAaD/qR/oTUno63IS4licXpK2uo/80HjYWDdqiE3A6eCqOGjI ESRAE1IsSFxfMOs/X5TTmP0QOfcJETmTfLAFFVvCyEyT8VN1A5qyThz1JWS/IRPX343m i2MFzRoDdowrLmsmfGbaIOol2AZzhaDpYpM6dvwnMcz2XaVWt7ppotEghCEc9Lm8f3/c Qc8SsRp6hOYtwfe/yGVyskc0NmM8XLNgO+Vh6QHt/a0PvarHkCPOPCUF5KQLSN0o8ejU ApVQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:message-id:date:subject:cc:to:from; bh=jXd1PbBcbFmJPpLusB5QZR7vy3ySVy5lpbhn+zITWyk=; b=mGhiILjJj6+nYwyARWQAeQk8kdcT1CBHYvHrmbKFGmAMQi4YAyMoW0Qm2X+fCo7ZJW tdnhY9Xucu5H9Ydb2LVInrgpsLKaHT1F17GGlbKrkgY8n3GDYrAklvCo/7RlGzlHo1ay YuD9Fm9yq8Fuzo+TzDB9APt6yDnxPsvJ5akhpT1vWvS5FqAfqhCFvRwlAQvHZTFMv3LR dJiRgpJGAYiQc4DWp5lE3qLin0heUyyZW+fsj0+4z+yQgVUN/+P+dAi9VYorgWwF19HC a70f3WNedPejY5PqkLGxSreJqf9KWdFa2g7KErcd+CvV+MBh5oi6aRQ/MdLfQvTIVTSU V9Gw== 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 e36si3313272pgm.17.2019.07.11.06.32.18; Thu, 11 Jul 2019 06:32:35 -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 S1728667AbfGKNEF (ORCPT + 99 others); Thu, 11 Jul 2019 09:04:05 -0400 Received: from xavier.telenet-ops.be ([195.130.132.52]:56606 "EHLO xavier.telenet-ops.be" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1728294AbfGKNEF (ORCPT ); Thu, 11 Jul 2019 09:04:05 -0400 Received: from ramsan ([84.194.98.4]) by xavier.telenet-ops.be with bizsmtp id bR422000r05gfCL01R42Ad; Thu, 11 Jul 2019 15:04:03 +0200 Received: from rox.of.borg ([192.168.97.57]) by ramsan with esmtp (Exim 4.90_1) (envelope-from ) id 1hlYjy-00085u-IG; Thu, 11 Jul 2019 15:04:02 +0200 Received: from geert by rox.of.borg with local (Exim 4.90_1) (envelope-from ) id 1hlYjy-0000cu-GL; Thu, 11 Jul 2019 15:04:02 +0200 From: Geert Uytterhoeven To: Michael Turquette , Stephen Boyd Cc: Yao Lihua , Yoshihiro Shimoda , Wolfram Sang , linux-clk@vger.kernel.org, linux-renesas-soc@vger.kernel.org, linux-kernel@vger.kernel.org, Geert Uytterhoeven Subject: [PATCH] clk: renesas: cpg-mssr: Fix reset control race condition Date: Thu, 11 Jul 2019 15:03:59 +0200 Message-Id: <20190711130359.1060-1-geert+renesas@glider.be> X-Mailer: git-send-email 2.17.1 Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org The module reset code in the Renesas CPG/MSSR driver uses read-modify-write (RMW) operations to write to a Software Reset Register (SRCRn), and simple writes to write to a Software Reset Clearing Register (SRSTCLRn), as was mandated by the R-Car Gen2 and Gen3 Hardware User's Manuals. However, this may cause a race condition when two devices are reset in parallel: if the reset for device A completes in the middle of the RMW operation for device B, device A may be reset again, causing subtle failures (e.g. i2c timeouts): thread A thread B -------- -------- val = SRCRn val |= bit A SRCRn = val delay val = SRCRn (bit A is set) SRSTCLRn = bit A (bit A in SRCRn is cleared) val |= bit B SRCRn = val (bit A and B are set) This can be reproduced on e.g. Salvator-XS using: $ while true; do i2cdump -f -y 4 0x6A b > /dev/null; done & $ while true; do i2cdump -f -y 2 0x10 b > /dev/null; done & i2c-rcar e6510000.i2c: error -110 : 40000002 i2c-rcar e66d8000.i2c: error -110 : 40000002 According to the R-Car Gen3 Hardware Manual Errata for Rev. 0.80 of Feb 28, 2018, reflected in Rev. 1.00 of the R-Car Gen3 Hardware User's Manual, writes to SRCRn do not require read-modify-write cycles. Note that the R-Car Gen2 Hardware User's Manual has not been updated yet, and still says a read-modify-write sequence is required. According to the hardware team, the reset hardware block is the same on both R-Car Gen2 and Gen3, though. Hence fix the issue by replacing the read-modify-write operations on SRCRn by simple writes. Reported-by: Yao Lihua Fixes: 6197aa65c4905532 ("clk: renesas: cpg-mssr: Add support for reset control") Signed-off-by: Geert Uytterhoeven --- So far I haven't been able to reproduce the issue on R-Car Gen2 (after forcing i2c reset on Gen2, too). Perhaps my Koelsch doesn't have enough CPU cores. What about Lager? Hi Mike, Stephen, As this is a bugfix, can you please take this directly, if accepted? Thanks! --- drivers/clk/renesas/renesas-cpg-mssr.c | 16 ++-------------- 1 file changed, 2 insertions(+), 14 deletions(-) diff --git a/drivers/clk/renesas/renesas-cpg-mssr.c b/drivers/clk/renesas/renesas-cpg-mssr.c index 52bbb9ce3807db31..d4075b13067429cd 100644 --- a/drivers/clk/renesas/renesas-cpg-mssr.c +++ b/drivers/clk/renesas/renesas-cpg-mssr.c @@ -572,17 +572,11 @@ static int cpg_mssr_reset(struct reset_controller_dev *rcdev, unsigned int reg = id / 32; unsigned int bit = id % 32; u32 bitmask = BIT(bit); - unsigned long flags; - u32 value; dev_dbg(priv->dev, "reset %u%02u\n", reg, bit); /* Reset module */ - spin_lock_irqsave(&priv->rmw_lock, flags); - value = readl(priv->base + SRCR(reg)); - value |= bitmask; - writel(value, priv->base + SRCR(reg)); - spin_unlock_irqrestore(&priv->rmw_lock, flags); + writel(bitmask, priv->base + SRCR(reg)); /* Wait for at least one cycle of the RCLK clock (@ ca. 32 kHz) */ udelay(35); @@ -599,16 +593,10 @@ static int cpg_mssr_assert(struct reset_controller_dev *rcdev, unsigned long id) unsigned int reg = id / 32; unsigned int bit = id % 32; u32 bitmask = BIT(bit); - unsigned long flags; - u32 value; dev_dbg(priv->dev, "assert %u%02u\n", reg, bit); - spin_lock_irqsave(&priv->rmw_lock, flags); - value = readl(priv->base + SRCR(reg)); - value |= bitmask; - writel(value, priv->base + SRCR(reg)); - spin_unlock_irqrestore(&priv->rmw_lock, flags); + writel(bitmask, priv->base + SRCR(reg)); return 0; } -- 2.17.1