From: hermr2d2@gmail.com (Hermann Kraus)
To: linux-arm-kernel@lists.infradead.org
Subject: [PATCH v2] spi: spi-sun4i, spi-sun6i: Fix bit rate calculation
Date: Thu, 06 Aug 2015 22:51:10 +0200 [thread overview]
Message-ID: <op.x2yr7kbcltbbtz@localhost> (raw)
In-Reply-To: <20150804171931.GZ20873@sirena.org.uk>
Use requested bit rate instead of maximum possible and correctly calculate
the divider.
There are two different maximum bitrates. "max_speed_hz" which is the
maximum for a given SPI device and "speed_hz" in "struct spi_transfer"
which is the rate used for this transfer. "speed_hz" must always be
used as the actual transfer rate.
The divider must be calculated by calling ilog2 only once, because this
function truncates the fractional part. Calling it twice increases
the error so much that certain bit rates which are available from the
hardware can't be reached by the kernel.
The result of this calculation must be checked to fit into the register
size instead of being truncated silently.
Signed-off-by: Hermann Kraus <hermr2d2@gmail.com>
---
Changes to v1: Unconditionally use tfr->speed_hz as requested by Mark
Brown.
drivers/spi/spi-sun4i.c | 15 +++++++++++----
drivers/spi/spi-sun6i.c | 15 +++++++++++----
2 files changed, 22 insertions(+), 8 deletions(-)
diff --git a/drivers/spi/spi-sun4i.c b/drivers/spi/spi-sun4i.c
index fbb0a4d..729d70b 100644
--- a/drivers/spi/spi-sun4i.c
+++ b/drivers/spi/spi-sun4i.c
@@ -229,8 +229,8 @@ static int sun4i_spi_transfer_one(struct spi_master
*master,
/* Ensure that we have a parent clock fast enough */
mclk_rate = clk_get_rate(sspi->mclk);
- if (mclk_rate < (2 * spi->max_speed_hz)) {
- clk_set_rate(sspi->mclk, 2 * spi->max_speed_hz);
+ if (mclk_rate < (2 * tfr->speed_hz)) {
+ clk_set_rate(sspi->mclk, 2 * tfr->speed_hz);
mclk_rate = clk_get_rate(sspi->mclk);
}
@@ -248,14 +248,21 @@ static int sun4i_spi_transfer_one(struct spi_master
*master,
* First try CDR2, and if we can't reach the expected
* frequency, fall back to CDR1.
*/
- div = mclk_rate / (2 * spi->max_speed_hz);
+ div = mclk_rate / (2 * tfr->speed_hz);
if (div <= (SUN4I_CLK_CTL_CDR2_MASK + 1)) {
if (div > 0)
div--;
reg = SUN4I_CLK_CTL_CDR2(div) | SUN4I_CLK_CTL_DRS;
} else {
- div = ilog2(mclk_rate) - ilog2(spi->max_speed_hz);
+ /* ilog2 truncates the fractional part. Therefore we don't
+ * subtract 1 from the result. */
+ div = ilog2(mclk_rate / tfr->speed_hz + 1);
+ if (div & ~SUN4I_CLK_CTL_CDR1_MASK) {
+ /* Can't reach low enough bit rate. */
+ ret = -EINVAL;
+ goto out;
+ }
reg = SUN4I_CLK_CTL_CDR1(div);
}
diff --git a/drivers/spi/spi-sun6i.c b/drivers/spi/spi-sun6i.c
index ac48f59..2d3dcda 100644
--- a/drivers/spi/spi-sun6i.c
+++ b/drivers/spi/spi-sun6i.c
@@ -217,8 +217,8 @@ static int sun6i_spi_transfer_one(struct spi_master
*master,
/* Ensure that we have a parent clock fast enough */
mclk_rate = clk_get_rate(sspi->mclk);
- if (mclk_rate < (2 * spi->max_speed_hz)) {
- clk_set_rate(sspi->mclk, 2 * spi->max_speed_hz);
+ if (mclk_rate < (2 * tfr->speed_hz)) {
+ clk_set_rate(sspi->mclk, 2 * tfr->speed_hz);
mclk_rate = clk_get_rate(sspi->mclk);
}
@@ -236,14 +236,21 @@ static int sun6i_spi_transfer_one(struct spi_master
*master,
* First try CDR2, and if we can't reach the expected
* frequency, fall back to CDR1.
*/
- div = mclk_rate / (2 * spi->max_speed_hz);
+ div = mclk_rate / (2 * tfr->speed_hz);
if (div <= (SUN6I_CLK_CTL_CDR2_MASK + 1)) {
if (div > 0)
div--;
reg = SUN6I_CLK_CTL_CDR2(div) | SUN6I_CLK_CTL_DRS;
} else {
- div = ilog2(mclk_rate) - ilog2(spi->max_speed_hz);
+ /* ilog2 truncates the fractional part. Therefore we don't
+ * subtract 1 from the result. */
+ div = ilog2(mclk_rate / tfr->speed_hz + 1);
+ if (div & ~SUN6I_CLK_CTL_CDR1_MASK) {
+ /* Can't reach low enough bit rate. */
+ ret = -EINVAL;
+ goto out;
+ }
reg = SUN6I_CLK_CTL_CDR1(div);
}
--
2.1.4
next prev parent reply other threads:[~2015-08-06 20:51 UTC|newest]
Thread overview: 5+ messages / expand[flat|nested] mbox.gz Atom feed top
2015-08-01 22:06 [PATCH] spi: spi-sun4i, spi-sun6i: Fix bit rate calculation Hermann Kraus
2015-08-04 17:19 ` Mark Brown
2015-08-06 20:51 ` Hermann Kraus [this message]
2015-08-07 13:14 ` [PATCH v2] " Mark Brown
2015-08-07 13:16 ` Mark Brown
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=op.x2yr7kbcltbbtz@localhost \
--to=hermr2d2@gmail.com \
--cc=linux-arm-kernel@lists.infradead.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).