Received: by 2002:ac0:a5a7:0:0:0:0:0 with SMTP id m36-v6csp3340186imm; Tue, 17 Jul 2018 03:11:38 -0700 (PDT) X-Google-Smtp-Source: AAOMgpdlbtILK8karbwarEEkXT/v0CpmO5x/am1WV1oDFAYdorqPghGK4FOZf7KC0tjRi77HTPBZ X-Received: by 2002:a17:902:59da:: with SMTP id d26-v6mr1040653plj.42.1531822298624; Tue, 17 Jul 2018 03:11:38 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1531822298; cv=none; d=google.com; s=arc-20160816; b=o5gparuDJ4nKEzXVMbjGIZDi31opyc5gEeUNSOpH1dz0oURS545xQy/qRi66blmjwU 2BZm+9S7+8TWRRgTTUmcr0zCdvJGosr8muq5Tg4g+jKOuF6VUtXp8JjaPxLnRBDc4BFS iIWkxUd9Dg1HMIhpc7/INFguNzXUuLrh22SZbaAq4je5qU8xZBRS+n00XsEJBwi1Mo5T YiPNJv1NYOmidSzKuhaRWaKcswQIuEDR0CzFqTk8q3vceOHIciUzR0jNKfgYIHH0USEe GMLz4QQHZ8Fz4h9jwCJInPXjyEJbc46mZVoEQDJiBzfia7MnufDDS0aU/XYXVpXaffT2 B6fg== 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:cc:to:subject:arc-authentication-results; bh=zV+Q85sfM/6VXc0oJHveRG0NB2jxId2+BupC0M23pz0=; b=r91mLsosubrKY7ikWclIOTw46oDUzBbTA4wwFxD2UbXak6n9JxWkHkqScTXycNX22f 9nCOqethj+8O6Buwt0yyHgCPxCJHzbQ9PLlD/xTSTPlISTcNtOpW1tI9Ugu4mLqRbFgJ kVdrEzK2YkT31pHuJRXjTGGVg/3yAreSF/I+8PBjrl/UTKpDqnG+AEA+9viqv7GyLRY1 idFk+Jhe99Dje1r7zcY7VtzD+IYCNQ+oyiL9C447uL8SPXyZoa0UIfdwhC/SBDnr237G YVmYWsYFzqGlYUA7LVOiIFj03PNxx0z2X4pWq446iN0z92DmtDs0npTh+xvRIU5PZzFD pZnQ== 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=nvidia.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id g15-v6si571038pgh.418.2018.07.17.03.11.23; Tue, 17 Jul 2018 03:11:38 -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; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=nvidia.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1729976AbeGQKml (ORCPT + 99 others); Tue, 17 Jul 2018 06:42:41 -0400 Received: from hqemgate16.nvidia.com ([216.228.121.65]:16403 "EHLO hqemgate16.nvidia.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1729655AbeGQKmk (ORCPT ); Tue, 17 Jul 2018 06:42:40 -0400 Received: from hqpgpgate101.nvidia.com (Not Verified[216.228.121.13]) by hqemgate16.nvidia.com (using TLS: TLSv1, AES128-SHA) id ; Tue, 17 Jul 2018 03:10:40 -0700 Received: from HQMAIL101.nvidia.com ([172.20.161.6]) by hqpgpgate101.nvidia.com (PGP Universal service); Tue, 17 Jul 2018 03:10:48 -0700 X-PGP-Universal: processed; by hqpgpgate101.nvidia.com on Tue, 17 Jul 2018 03:10:48 -0700 Received: from [10.21.132.122] (172.20.13.39) by HQMAIL101.nvidia.com (172.20.187.10) with Microsoft SMTP Server (TLS) id 15.0.1347.2; Tue, 17 Jul 2018 10:10:45 +0000 Subject: Re: [PATCH] mmc: tegra: Force correct divider calculation on DDR50/52 To: Aapo Vienamo CC: Adrian Hunter , Ulf Hansson , Thierry Reding , Marcel Ziswiler , Stefan Agner , , , References: <1531751669-26584-1-git-send-email-avienamo@nvidia.com> <16caa6d0-4368-8931-a77c-23fb76bee200@nvidia.com> <20180717120850.774c4e9d@dhcp-10-21-25-168> From: Jon Hunter Message-ID: Date: Tue, 17 Jul 2018 11:10:43 +0100 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:52.0) Gecko/20100101 Thunderbird/52.8.0 MIME-Version: 1.0 In-Reply-To: <20180717120850.774c4e9d@dhcp-10-21-25-168> X-Originating-IP: [172.20.13.39] X-ClientProxiedBy: HQMAIL107.nvidia.com (172.20.187.13) To HQMAIL101.nvidia.com (172.20.187.10) Content-Type: text/plain; charset="utf-8" Content-Language: en-US Content-Transfer-Encoding: 7bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 17/07/18 10:08, Aapo Vienamo wrote: > On Mon, 16 Jul 2018 21:03:08 +0100 > Jon Hunter wrote: > >> On 16/07/18 15:34, Aapo Vienamo wrote: >>> Tegra SDHCI controllers require the SDHCI clock divider to be configured >>> to divide the clock by two in DDR50/52 modes. Incorrectly configured >>> clock divider results in corrupted data. >>> >>> Prevent the possibility of incorrectly calculating the divider value due >>> to clock rate rounding or low parent clock frequency by not assigning >>> host->max_clk to clk_get_rate() on tegra_sdhci_set_clock(). >>> >>> See the comments for further details. >>> >>> Fixes: a8e326a ("mmc: tegra: implement module external clock change") >>> Signed-off-by: Aapo Vienamo >>> --- >>> drivers/mmc/host/sdhci-tegra.c | 17 ++++++++++++++++- >>> 1 file changed, 16 insertions(+), 1 deletion(-) >>> >>> diff --git a/drivers/mmc/host/sdhci-tegra.c b/drivers/mmc/host/sdhci-tegra.c >>> index ddf00166..908b23e 100644 >>> --- a/drivers/mmc/host/sdhci-tegra.c >>> +++ b/drivers/mmc/host/sdhci-tegra.c >>> @@ -210,9 +210,24 @@ static void tegra_sdhci_set_clock(struct sdhci_host *host, unsigned int clock) >>> if (!clock) >>> return sdhci_set_clock(host, clock); >>> >>> + /* >>> + * In DDR50/52 modes the Tegra SDHCI controllers require the SDHCI >>> + * divider to be configured to divided the host clock by two. The SDHC >>> + * clock divider is calculated as part of sdhci_set_clock() by >>> + * sdhci_calc_clk(). The divider is calculated from host->max_clk and >>> + * the requested clock rate. >>> + * >>> + * By setting the host->max_clk to clock * 2 the divider calculation >>> + * will always result in the correct value for DDR50/52 modes, >>> + * regardless of clock rate rounding, which may happen if the value >>> + * from clk_get_rate() is used. >>> + */ >>> host_clk = tegra_host->ddr_signaling ? clock * 2 : clock; >>> clk_set_rate(pltfm_host->clk, host_clk); >>> - host->max_clk = clk_get_rate(pltfm_host->clk); >>> + if (tegra_host->ddr_signaling) >>> + host->max_clk = host_clk; >>> + else >>> + host->max_clk = clk_get_rate(pltfm_host->clk); >>> >>> sdhci_set_clock(host, clock); >>> >> >> I see what you are saying, but should we be concerned that we are not >> getting the rate we are requesting in the first place? > > The rates themselves aren't as critical as the divider on DDR50/52. It's > true that in most sane configurations we should not hit the cases where > the divider will not get configured correctly if the value from > clk_get_rate() is used. > >> Maybe it would help if you could provide a specific example showing a >> case where we request rate X and get Y, and then this leads to a bad >> rate Z later. > > There are two possible cases where the divider will get configured > incorrectly: either to divide by one or anything greater than two. I > verified that at least on t124 greater dividers also fail in practice. So looking at a few Tegra TRMs, I see comments to the effect ... 'In DDR50 mode, SD clock divisor should always be 2' ... which aligns with the above statement and your comment. > One option is that the parent clock is unable to supply a rate low > enough. Lets consider DDR50 mode: we request 100 MHz from the parent > clock and let's say it gets rounded to 104 MHz. In this case we end up > with a divider of four because 104 MHz / 2 <= 50 MHz is false, and the > divider is incremented to the next step. This happens because the > divider is calculated the following manner in sdhci_calc_clk(), where in > this case host->max_clk would be 104 MHz and clock 50 MHz: > > for (div = 2; div < SDHCI_MAX_DIV_SPEC_300; > div += 2) { > if ((host->max_clk / div) <= clock) > break; > } > > With the patch we would get DDR50 mode runinng at 52 MHz and without > the patch all IO to the device would fail. > > The less likely option is that divider of one is calculated if > host->max_clk is less than or equal to clock. This would happen if the > parent clock is unable to supply a high enough clock rate. So let's say > we are setting up the clocks for DDR50 and request the parent clock to > supply 100 MHz and we get say 50 MHz instead. In this case originally we > would get I/O errors, but with the patch we end up with at DDR50 mode > where the bus is actually run at 25 MHz. Yes I guess my concern is that we end up running at a rate much less than expected. I am not sure if it is worth warning against this or not. > While at least the later case would most likely be a bug or > misconfiguration, it would be still nice to have a mechanism for > graceful dergadation instead of complete failure. Agree. > Another option I considered was verifying the parent clock behaviour > during probe and masking DDR50/52 out from the host capability bits > depending on the results. The problem with that is that the exact rate > requested is determined based on CSD register values read from the MMC > device itself. > > It's really unfortunate that we have this quirk in the hardware as > dealing with it in a robust manner results in a fair bit of complexity. > Oh well. Thanks for the explanation. Looks good to me ... Acked-by: Jon Hunter Cheers Jon -- nvpublic