Received: by 2002:a05:6358:9144:b0:117:f937:c515 with SMTP id r4csp3038939rwr; Sun, 7 May 2023 03:31:26 -0700 (PDT) X-Google-Smtp-Source: ACHHUZ7R2hfE361UgTrLUv50uCJ+uHIDUTl/IHnDREBJNaRj2azbDrXq9djgwancAZL8Enrq0XNQ X-Received: by 2002:a17:903:2292:b0:1aa:ed72:5c2e with SMTP id b18-20020a170903229200b001aaed725c2emr9630920plh.61.1683455485436; Sun, 07 May 2023 03:31:25 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1683455485; cv=none; d=google.com; s=arc-20160816; b=k+wBOCODI9kGsfy+J6/I9XfHtWPyIb3HOhzvaMGLIIiVyQ6EPb6cb+8997UFE5T1U7 2WIF2UVHEWfjNBb9MOrS2SMgbT7/2wB3w8RUTslW3KludzQzlhGKK4BsyIhwSFWFI31t xjeCXrBz66asvty3vj3NrCQDjYcfX3PKeB9Yy6++jbY9Ah+DdegWwMoqtXaK7MkRBp/+ FKouniL8iUr7fnhy1WGBCNP4ehTZRP7Nkc/PHUe/68DazQgf6KVYhBgm6LtsAQHTLy81 IkjWr3quGgPq5LCn5a+Wq64cKW4qQUZGZKyquJTYjYCMsUHx1z7pydtb8d45jY6u6A3k BPwQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:mime-version :organization:references:in-reply-to:message-id:subject:cc:to:from :date; bh=ZLqORaCxkewa2JTJPq/7w8DT3sr43YnfL29/vEIhY+U=; b=x+Qiekp+4Tmlrhg8mby4Fq3iAE0ItbxTTS8lu8GktmIvELcj39cZYful5IZe9gTlGG 4YYNCe64Nkvl5+7ggNhbC8kyLWv74hGNY7io91LiGT4m07WVR5BLeFxd9665YLYpDdK9 Q1OmwE7WtCxyH2Dp73Bl+d4gXi1PMMvLFdciltvvJ3EN9k9kcsTFmSD0Wd0o+WCzflzE nUBKCIyv9uNvl/5kKvmT0jrCvwBwddU4hs0qFT2kdYLsyJSyiS+TWnmU+zF39t1nEuBA Xr2D085jr/7VCRKfaslHhjsXTBHqXO1UOTEyTEjcMAZ5HX3YxRSA8oAIjz88s83DtxVE e/zg== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=arm.com Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id k14-20020a170902760e00b001a6933b3a67si5430656pll.531.2023.05.07.03.31.10; Sun, 07 May 2023 03:31:25 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=arm.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S231423AbjEGJwN (ORCPT + 99 others); Sun, 7 May 2023 05:52:13 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:51482 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S229757AbjEGJwL (ORCPT ); Sun, 7 May 2023 05:52:11 -0400 Received: from foss.arm.com (foss.arm.com [217.140.110.172]) by lindbergh.monkeyblade.net (Postfix) with ESMTP id A88E9273A; Sun, 7 May 2023 02:52:09 -0700 (PDT) Received: from usa-sjc-imap-foss1.foss.arm.com (unknown [10.121.207.14]) by usa-sjc-mx-foss1.foss.arm.com (Postfix) with ESMTP id C6A222F4; Sun, 7 May 2023 02:52:53 -0700 (PDT) Received: from slackpad.lan (unknown [172.31.20.19]) by usa-sjc-imap-foss1.foss.arm.com (Postfix) with ESMTPSA id 547B13F64C; Sun, 7 May 2023 02:52:06 -0700 (PDT) Date: Sun, 7 May 2023 10:51:50 +0100 From: Andre Przywara To: Maksim Kiselev Cc: Icenowy Zheng , Mark Brown , Rob Herring , Krzysztof Kozlowski , Conor Dooley , Chen-Yu Tsai , Jernej Skrabec , Samuel Holland , Paul Walmsley , Palmer Dabbelt , Albert Ou , Cristian Ciocaltea , Greg Kroah-Hartman , Maxime Ripard , linux-spi@vger.kernel.org, devicetree@vger.kernel.org, linux-arm-kernel@lists.infradead.org, linux-sunxi@lists.linux.dev, linux-kernel@vger.kernel.org, linux-riscv@lists.infradead.org Subject: Re: [PATCH v3 3/5] spi: sun6i: add quirk for in-controller clock divider Message-ID: <20230507105150.31643b87@slackpad.lan> In-Reply-To: <20230506232616.1792109-4-bigunclemax@gmail.com> References: <20230506232616.1792109-1-bigunclemax@gmail.com> <20230506232616.1792109-4-bigunclemax@gmail.com> Organization: Arm Ltd. X-Mailer: Claws Mail 4.1.0 (GTK 3.24.31; x86_64-slackware-linux-gnu) MIME-Version: 1.0 Content-Type: text/plain; charset=US-ASCII Content-Transfer-Encoding: 7bit X-Spam-Status: No, score=-4.2 required=5.0 tests=BAYES_00,RCVD_IN_DNSWL_MED, SPF_HELO_NONE,SPF_NONE,T_SCC_BODY_TEXT_LINE autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Sun, 7 May 2023 02:26:06 +0300 Maksim Kiselev wrote: Hi Maksim, thanks for the quick turnaround and the changes. > Previously SPI controllers in Allwinner SoCs has a clock divider inside. > However now the clock divider is removed and to set the transfer clock > rate it's only needed to set the SPI module clock to the target value > and configure a proper work mode. > > According to the datasheet there are three work modes: > > | SPI Sample Mode | SDM(bit13) | SDC(bit11) | Run Clock | > |-------------------------|------------|------------|-----------| > | normal sample | 1 | 0 | <= 24 MHz | > | delay half cycle sample | 0 | 0 | <= 40 MHz | > | delay one cycle sample | 0 | 1 | >= 80 MHz | > > Add a quirk for this kind of SPI controllers. > > Co-developed-by: Icenowy Zheng > Signed-off-by: Maksim Kiselev > --- > drivers/spi/spi-sun6i.c | 92 +++++++++++++++++++++++++++-------------- > 1 file changed, 62 insertions(+), 30 deletions(-) > > diff --git a/drivers/spi/spi-sun6i.c b/drivers/spi/spi-sun6i.c > index 01a01cd86db5..1e9e9a8159d9 100644 > --- a/drivers/spi/spi-sun6i.c > +++ b/drivers/spi/spi-sun6i.c > @@ -42,7 +42,9 @@ > #define SUN6I_TFR_CTL_CS_MANUAL BIT(6) > #define SUN6I_TFR_CTL_CS_LEVEL BIT(7) > #define SUN6I_TFR_CTL_DHB BIT(8) > +#define SUN6I_TFR_CTL_SDC BIT(11) > #define SUN6I_TFR_CTL_FBS BIT(12) > +#define SUN6I_TFR_CTL_SDM BIT(13) > #define SUN6I_TFR_CTL_XCH BIT(31) > > #define SUN6I_INT_CTL_REG 0x10 > @@ -87,6 +89,7 @@ > > struct sun6i_spi_cfg { > unsigned long fifo_depth; > + bool has_clk_ctl; > }; > > struct sun6i_spi { > @@ -260,7 +263,7 @@ static int sun6i_spi_transfer_one(struct spi_master *master, > struct spi_transfer *tfr) > { > struct sun6i_spi *sspi = spi_master_get_devdata(master); > - unsigned int mclk_rate, div, div_cdr1, div_cdr2, timeout; > + unsigned int div, div_cdr1, div_cdr2, timeout; > unsigned int start, end, tx_time; > unsigned int trig_level; > unsigned int tx_len = 0, rx_len = 0; > @@ -350,39 +353,66 @@ static int sun6i_spi_transfer_one(struct spi_master *master, > > sun6i_spi_write(sspi, SUN6I_TFR_CTL_REG, reg); > > - /* Ensure that we have a parent clock fast enough */ > - mclk_rate = clk_get_rate(sspi->mclk); > - if (mclk_rate < (2 * tfr->speed_hz)) { > - clk_set_rate(sspi->mclk, 2 * tfr->speed_hz); > - mclk_rate = clk_get_rate(sspi->mclk); > - } > + if (sspi->cfg->has_clk_ctl) { > + unsigned int mclk_rate = clk_get_rate(sspi->mclk); > > - /* > - * Setup clock divider. > - * > - * We have two choices there. Either we can use the clock > - * divide rate 1, which is calculated thanks to this formula: > - * SPI_CLK = MOD_CLK / (2 ^ cdr) > - * Or we can use CDR2, which is calculated with the formula: > - * SPI_CLK = MOD_CLK / (2 * (cdr + 1)) > - * Wether we use the former or the latter is set through the > - * DRS bit. > - * > - * First try CDR2, and if we can't reach the expected > - * frequency, fall back to CDR1. > - */ > - div_cdr1 = DIV_ROUND_UP(mclk_rate, tfr->speed_hz); > - div_cdr2 = DIV_ROUND_UP(div_cdr1, 2); > - if (div_cdr2 <= (SUN6I_CLK_CTL_CDR2_MASK + 1)) { > - reg = SUN6I_CLK_CTL_CDR2(div_cdr2 - 1) | SUN6I_CLK_CTL_DRS; > - tfr->effective_speed_hz = mclk_rate / (2 * div_cdr2); > + /* Ensure that we have a parent clock fast enough */ > + if (mclk_rate < (2 * tfr->speed_hz)) { > + clk_set_rate(sspi->mclk, 2 * tfr->speed_hz); > + mclk_rate = clk_get_rate(sspi->mclk); > + } > + > + /* > + * Setup clock divider. > + * > + * We have two choices there. Either we can use the clock > + * divide rate 1, which is calculated thanks to this formula: > + * SPI_CLK = MOD_CLK / (2 ^ cdr) > + * Or we can use CDR2, which is calculated with the formula: > + * SPI_CLK = MOD_CLK / (2 * (cdr + 1)) > + * Whether we use the former or the latter is set through the > + * DRS bit. > + * > + * First try CDR2, and if we can't reach the expected > + * frequency, fall back to CDR1. > + */ > + div_cdr1 = DIV_ROUND_UP(mclk_rate, tfr->speed_hz); > + div_cdr2 = DIV_ROUND_UP(div_cdr1, 2); > + if (div_cdr2 <= (SUN6I_CLK_CTL_CDR2_MASK + 1)) { > + reg = SUN6I_CLK_CTL_CDR2(div_cdr2 - 1) | SUN6I_CLK_CTL_DRS; > + tfr->effective_speed_hz = mclk_rate / (2 * div_cdr2); > + } else { > + div = min(SUN6I_CLK_CTL_CDR1_MASK, order_base_2(div_cdr1)); > + reg = SUN6I_CLK_CTL_CDR1(div); > + tfr->effective_speed_hz = mclk_rate / (1 << div); > + } > + > + sun6i_spi_write(sspi, SUN6I_CLK_CTL_REG, reg); > } else { > - div = min(SUN6I_CLK_CTL_CDR1_MASK, order_base_2(div_cdr1)); > - reg = SUN6I_CLK_CTL_CDR1(div); > - tfr->effective_speed_hz = mclk_rate / (1 << div); > + clk_set_rate(sspi->mclk, tfr->speed_hz); > + tfr->effective_speed_hz = clk_get_rate(sspi->mclk); > + > + /* > + * Configure work mode. > + * > + * There are three work modes depending on the controller clock > + * frequency: > + * - normal sample mode : CLK <= 24MHz SDM=1 SDC=0 > + * - delay half-cycle sample mode : CLK <= 40MHz SDM=0 SDC=0 > + * - delay one-cycle sample mode : CLK >= 80MHz SDM=0 SDC=1 > + */ > + reg = sun6i_spi_read(sspi, SUN6I_TFR_CTL_REG); > + > + if (tfr->effective_speed_hz <= 24000000) > + reg |= SUN6I_TFR_CTL_SDM; > + else if (tfr->effective_speed_hz >= 80000000) > + reg |= SUN6I_TFR_CTL_SDC; This case assumes that the SDM bit is zero already. I think just masking both bits off above, right after the read, is the easiest, then you can also lose the else branch below entirely. Cheers, Andre > + else > + reg &= ~(SUN6I_TFR_CTL_SDM | SUN6I_TFR_CTL_SDC); > + > + sun6i_spi_write(sspi, SUN6I_TFR_CTL_REG, reg); > } > > - sun6i_spi_write(sspi, SUN6I_CLK_CTL_REG, reg); > /* Finally enable the bus - doing so before might raise SCK to HIGH */ > reg = sun6i_spi_read(sspi, SUN6I_GBL_CTL_REG); > reg |= SUN6I_GBL_CTL_BUS_ENABLE; > @@ -701,10 +731,12 @@ static void sun6i_spi_remove(struct platform_device *pdev) > > static const struct sun6i_spi_cfg sun6i_a31_spi_cfg = { > .fifo_depth = SUN6I_FIFO_DEPTH, > + .has_clk_ctl = true, > }; > > static const struct sun6i_spi_cfg sun8i_h3_spi_cfg = { > .fifo_depth = SUN8I_FIFO_DEPTH, > + .has_clk_ctl = true, > }; > > static const struct of_device_id sun6i_spi_match[] = {