All of lore.kernel.org
 help / color / mirror / Atom feed
From: Tero Kristo <kristo@kernel.org>
To: u-boot@lists.denx.de
Subject: [PATCHv4 01/26] lib: rational: copy the rational fraction lib routines from Linux
Date: Tue, 11 May 2021 11:30:39 +0300	[thread overview]
Message-ID: <20210511083104.10868-2-kristo@kernel.org> (raw)
In-Reply-To: <20210511083104.10868-1-kristo@kernel.org>

From: Tero Kristo <t-kristo@ti.com>

Copy the best rational approximation calculation routines from Linux.
Typical usecase for these routines is to calculate the M/N divider
values for PLLs to reach a specific clock rate.

This is based on linux kernel commit:
"lib/math/rational.c: fix possible incorrect result from rational
fractions helper"
(sha1: 323dd2c3ed0641f49e89b4e420f9eef5d3d5a881)

Signed-off-by: Tero Kristo <t-kristo@ti.com>
Reviewed-by: Tom Rini <trini@konsulko.com>
Signed-off-by: Tero Kristo <kristo@kernel.org>
---
 include/linux/rational.h | 20 ++++++++
 lib/Kconfig              |  7 +++
 lib/Makefile             |  2 +
 lib/rational.c           | 99 ++++++++++++++++++++++++++++++++++++++++
 4 files changed, 128 insertions(+)
 create mode 100644 include/linux/rational.h
 create mode 100644 lib/rational.c

diff --git a/include/linux/rational.h b/include/linux/rational.h
new file mode 100644
index 0000000000..33f5f5fc3e
--- /dev/null
+++ b/include/linux/rational.h
@@ -0,0 +1,20 @@
+/* SPDX-License-Identifier: GPL-2.0 */
+/*
+ * rational fractions
+ *
+ * Copyright (C) 2009 emlix GmbH, Oskar Schirmer <oskar@scara.com>
+ *
+ * helper functions when coping with rational numbers,
+ * e.g. when calculating optimum numerator/denominator pairs for
+ * pll configuration taking into account restricted register size
+ */
+
+#ifndef _LINUX_RATIONAL_H
+#define _LINUX_RATIONAL_H
+
+void rational_best_approximation(
+	unsigned long given_numerator, unsigned long given_denominator,
+	unsigned long max_numerator, unsigned long max_denominator,
+	unsigned long *best_numerator, unsigned long *best_denominator);
+
+#endif /* _LINUX_RATIONAL_H */
diff --git a/lib/Kconfig b/lib/Kconfig
index 6d2d41de30..68d58aa90d 100644
--- a/lib/Kconfig
+++ b/lib/Kconfig
@@ -663,6 +663,13 @@ config GENERATE_SMBIOS_TABLE
 	  See also SMBIOS_SYSINFO which allows SMBIOS values to be provided in
 	  the devicetree.
 
+config LIB_RATIONAL
+	bool "enable continued fraction calculation routines"
+
+config SPL_LIB_RATIONAL
+	bool "enable continued fraction calculation routines for SPL"
+	depends on SPL
+
 endmenu
 
 config ASN1_COMPILER
diff --git a/lib/Makefile b/lib/Makefile
index 6825671955..ea6bd0cb55 100644
--- a/lib/Makefile
+++ b/lib/Makefile
@@ -73,6 +73,8 @@ obj-$(CONFIG_$(SPL_)LZO) += lzo/
 obj-$(CONFIG_$(SPL_)LZMA) += lzma/
 obj-$(CONFIG_$(SPL_)LZ4) += lz4_wrapper.o
 
+obj-$(CONFIG_$(SPL_)LIB_RATIONAL) += rational.o
+
 obj-$(CONFIG_LIBAVB) += libavb/
 
 obj-$(CONFIG_$(SPL_TPL_)OF_LIBFDT) += libfdt/
diff --git a/lib/rational.c b/lib/rational.c
new file mode 100644
index 0000000000..316db3b590
--- /dev/null
+++ b/lib/rational.c
@@ -0,0 +1,99 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * rational fractions
+ *
+ * Copyright (C) 2009 emlix GmbH, Oskar Schirmer <oskar@scara.com>
+ * Copyright (C) 2019 Trent Piepho <tpiepho@gmail.com>
+ *
+ * helper functions when coping with rational numbers
+ */
+
+#include <linux/rational.h>
+#include <linux/compiler.h>
+#include <linux/kernel.h>
+
+/*
+ * calculate best rational approximation for a given fraction
+ * taking into account restricted register size, e.g. to find
+ * appropriate values for a pll with 5 bit denominator and
+ * 8 bit numerator register fields, trying to set up with a
+ * frequency ratio of 3.1415, one would say:
+ *
+ * rational_best_approximation(31415, 10000,
+ *		(1 << 8) - 1, (1 << 5) - 1, &n, &d);
+ *
+ * you may look@given_numerator as a fixed point number,
+ * with the fractional part size described in given_denominator.
+ *
+ * for theoretical background, see:
+ * http://en.wikipedia.org/wiki/Continued_fraction
+ */
+
+void rational_best_approximation(
+	unsigned long given_numerator, unsigned long given_denominator,
+	unsigned long max_numerator, unsigned long max_denominator,
+	unsigned long *best_numerator, unsigned long *best_denominator)
+{
+	/* n/d is the starting rational, which is continually
+	 * decreased each iteration using the Euclidean algorithm.
+	 *
+	 * dp is the value of d from the prior iteration.
+	 *
+	 * n2/d2, n1/d1, and n0/d0 are our successively more accurate
+	 * approximations of the rational.  They are, respectively,
+	 * the current, previous, and two prior iterations of it.
+	 *
+	 * a is current term of the continued fraction.
+	 */
+	unsigned long n, d, n0, d0, n1, d1, n2, d2;
+	n = given_numerator;
+	d = given_denominator;
+	n0 = d1 = 0;
+	n1 = d0 = 1;
+
+	for (;;) {
+		unsigned long dp, a;
+
+		if (d == 0)
+			break;
+		/* Find next term in continued fraction, 'a', via
+		 * Euclidean algorithm.
+		 */
+		dp = d;
+		a = n / d;
+		d = n % d;
+		n = dp;
+
+		/* Calculate the current rational approximation (aka
+		 * convergent), n2/d2, using the term just found and
+		 * the two prior approximations.
+		 */
+		n2 = n0 + a * n1;
+		d2 = d0 + a * d1;
+
+		/* If the current convergent exceeds the maxes, then
+		 * return either the previous convergent or the
+		 * largest semi-convergent, the final term of which is
+		 * found below as 't'.
+		 */
+		if ((n2 > max_numerator) || (d2 > max_denominator)) {
+			unsigned long t = min((max_numerator - n0) / n1,
+					      (max_denominator - d0) / d1);
+
+			/* This tests if the semi-convergent is closer
+			 * than the previous convergent.
+			 */
+			if (2u * t > a || (2u * t == a && d0 * dp > d1 * d)) {
+				n1 = n0 + t * n1;
+				d1 = d0 + t * d1;
+			}
+			break;
+		}
+		n0 = n1;
+		n1 = n2;
+		d0 = d1;
+		d1 = d2;
+	}
+	*best_numerator = n1;
+	*best_denominator = d1;
+}
-- 
2.17.1

  reply	other threads:[~2021-05-11  8:30 UTC|newest]

Thread overview: 30+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2021-05-11  8:30 [PATCHv4 00/26] J72xx: HSM rearch support series Tero Kristo
2021-05-11  8:30 ` Tero Kristo [this message]
2021-05-11  8:30 ` [PATCHv4 02/26] arm: mach-k3: introduce new config option for sysfw split Tero Kristo
2021-05-11  8:30 ` [PATCHv4 03/26] remoteproc: k3-r5: remove sysfw PM calls if not supported Tero Kristo
2021-05-11  8:30 ` [PATCHv4 04/26] common: fit: Update board_fit_image_post_process() to pass fit and node_offset Tero Kristo
2021-05-11  8:30 ` [PATCHv4 05/26] clk: fixed_rate: add API for directly registering fixed rate clocks Tero Kristo
2021-05-11  8:30 ` [PATCHv4 06/26] clk: fix clock tree dump to properly dump out every registered clock Tero Kristo
2021-05-11  8:30 ` [PATCHv4 07/26] clk: do not attempt to fetch clock pointer with null device Tero Kristo
2021-05-11  8:30 ` [PATCHv4 08/26] clk: add support for setting clk rate from cmdline Tero Kristo
2021-05-11  8:30 ` [PATCHv4 09/26] clk: sci-clk: fix return value of set_rate Tero Kristo
2021-05-11  8:30 ` [PATCHv4 10/26] clk: fix assigned-clocks to pass with deferring provider Tero Kristo
2021-05-11  8:30 ` [PATCHv4 11/26] clk: fix set_rate to clean up cached rates for the hierarchy Tero Kristo
2021-05-11  8:30 ` [PATCHv4 12/26] clk: add support for TI K3 SoC PLL Tero Kristo
2021-05-11  8:30 ` [PATCHv4 13/26] clk: add support for TI K3 SoC clocks Tero Kristo
2021-05-11  8:30 ` [PATCHv4 14/26] power: domain: Introduce driver for raw TI K3 PDs Tero Kristo
2021-05-11 22:44   ` Jaehoon Chung
2021-05-11  8:30 ` [PATCHv4 15/26] cmd: ti: pd: Add debug command for K3 power domains Tero Kristo
2021-05-11 22:44   ` Jaehoon Chung
2021-05-11  8:30 ` [PATCHv4 16/26] tools: k3_fit_atf: add DM binary to the FIT image Tero Kristo
2021-05-11  8:30 ` [PATCHv4 17/26] arm: mach-k3: Add platform data for j721e and j7200 Tero Kristo
2021-05-11  8:30 ` [PATCHv4 18/26] arm: mach-k3: add support for detecting firmware images from FIT Tero Kristo
2021-05-11  8:30 ` [PATCHv4 19/26] arm: mach-k3: do board config for PM only if supported Tero Kristo
2021-05-11  8:30 ` [PATCHv4 20/26] arm: mach-k3: common: Drop main r5 start Tero Kristo
2021-05-11  8:30 ` [PATCHv4 21/26] arm: mach-k3: sysfw-loader: pass boardcfg to sciserver Tero Kristo
2021-05-11  8:31 ` [PATCHv4 22/26] arm: mach-k3: j721e_init: Force early probe of clk-k3 driver Tero Kristo
2021-05-11  8:31 ` [PATCHv4 23/26] configs: j721e_evm_r5: Enable raw access power management features Tero Kristo
2021-05-11  8:31 ` [PATCHv4 24/26] configs: j7200_evm_r5: " Tero Kristo
2021-05-11  8:31 ` [PATCHv4 25/26] board: ti: j72xx: README: update build instructions and image formats Tero Kristo
2021-05-11  8:31 ` [PATCHv4 26/26] arm: dts: k3-j72xx: correct MCU timer1 frequency Tero Kristo
2021-06-02 16:26 ` [PATCHv4 00/26] J72xx: HSM rearch support series Lokesh Vutla

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=20210511083104.10868-2-kristo@kernel.org \
    --to=kristo@kernel.org \
    --cc=u-boot@lists.denx.de \
    /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 an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.