mirror of
https://github.com/openwrt/openwrt.git
synced 2025-01-24 21:37:14 +00:00
b5f32064ed
Qualcomm Atheros IPQ807x is a modern WiSoC featuring: * Quad Core ARMv8 Cortex A-53 * @ 2.2 GHz (IPQ8072A/4A/6A/8A) Codename Hawkeye * @ 1.4 GHz (IPQ8070A/1A) Codename Acorn * Dual Band simultaneaous IEEE 802.11ax * 5G: 8x8/80 or 4x4/160MHz (IPQ8074A/8A) * 5G: 4x4/80 or 2x2/160MHz (IPQ8071A/2A/6A) * 5G: 2x2/80MHz (IPQ8070A) * 2G: 4x4/40MHz (IPQ8072A/4A/6A/8A) * 2G: 2x2/40MHz (IPQ8070A/1A) * 1x PSGMII via QCA8072/5 (Max 5x 1GbE ports) * 2x SGMII/USXGMII (1/2.5/5/10 GbE) on Hawkeye * 2x SGMII/USXGMII (1/2.5/5 GbE) on Acorn * DDR3L/4 32/16 bit up to 2400MT/s * SDIO 3.0/SD card 3.0/eMMC 5.1 * Dual USB 3.0 * One PCIe Gen2.1 and one PCIe Gen3.0 port (Single lane) * Parallel NAND (ONFI)/LCD * 6x QUP BLSP SPI/I2C/UART * I2S, PCM, and TDMA * HW PWM * 1.8V configurable GPIO * Companion PMP8074 PMIC via SPMI (GPIOS, RTC etc) Note that only v2 SOC models aka the ones ending with A suffix are supported, v1 models do not comply to the final 802.11ax and have lower clocks, lack the Gen3 PCIe etc. SoC itself has two UBI32 cores for the NSS offloading system, however currently no offloading is supported. Signed-off-by: Robert Marko <robimarko@gmail.com>
204 lines
5.4 KiB
Diff
204 lines
5.4 KiB
Diff
From 032be4f49dda786fea9e1501212f6cd09a7ded96 Mon Sep 17 00:00:00 2001
|
|
From: Christian Marangi <ansuelsmth@gmail.com>
|
|
Date: Thu, 3 Nov 2022 14:49:43 +0100
|
|
Subject: [PATCH] clk: qcom: clk-rcg2: introduce support for multiple conf for
|
|
same freq
|
|
|
|
Some RCG frequency can be reached by multiple configuration.
|
|
|
|
We currently declare multiple configuration for the same frequency but
|
|
that is not supported and always the first configuration will be taken.
|
|
|
|
These multiple configuration are needed as based on the current parent
|
|
configuration, it may be needed to use a different configuration to
|
|
reach the same frequency.
|
|
|
|
To handle this introduce 2 new macro, FM and C.
|
|
|
|
- FM is used to declare an empty freq_tbl with just the frequency and an
|
|
array of confs to insert all the config for the provided frequency.
|
|
|
|
- C is used to declare a fre_conf where src, pre_div, m and n are
|
|
provided.
|
|
|
|
The driver is changed to handle this special freq_tbl and select the
|
|
correct config by calculating the final rate and deciding based on the
|
|
one that is less different than the requested one.
|
|
|
|
Tested-by: Robert Marko <robimarko@gmail.com>
|
|
Signed-off-by: Christian Marangi <ansuelsmth@gmail.com>
|
|
---
|
|
drivers/clk/qcom/clk-rcg.h | 14 ++++++-
|
|
drivers/clk/qcom/clk-rcg2.c | 84 +++++++++++++++++++++++++++++++++----
|
|
2 files changed, 88 insertions(+), 10 deletions(-)
|
|
|
|
--- a/drivers/clk/qcom/clk-rcg.h
|
|
+++ b/drivers/clk/qcom/clk-rcg.h
|
|
@@ -7,7 +7,17 @@
|
|
#include <linux/clk-provider.h>
|
|
#include "clk-regmap.h"
|
|
|
|
-#define F(f, s, h, m, n) { (f), (s), (2 * (h) - 1), (m), (n) }
|
|
+#define F(f, s, h, m, n) { (f), (s), (2 * (h) - 1), (m), (n), 0, NULL }
|
|
+
|
|
+#define FM(_f, _confs) { .freq = (_f), .confs_num = ARRAY_SIZE(_confs), .confs = (_confs) }
|
|
+#define C(s, h, m, n) { (s), (2 * (h) - 1), (m), (n) }
|
|
+
|
|
+struct freq_conf {
|
|
+ u8 src;
|
|
+ u8 pre_div;
|
|
+ u16 m;
|
|
+ u16 n;
|
|
+};
|
|
|
|
struct freq_tbl {
|
|
unsigned long freq;
|
|
@@ -15,6 +25,8 @@ struct freq_tbl {
|
|
u8 pre_div;
|
|
u16 m;
|
|
u16 n;
|
|
+ int confs_num;
|
|
+ const struct freq_conf *confs;
|
|
};
|
|
|
|
/**
|
|
--- a/drivers/clk/qcom/clk-rcg2.c
|
|
+++ b/drivers/clk/qcom/clk-rcg2.c
|
|
@@ -209,11 +209,60 @@ clk_rcg2_recalc_rate(struct clk_hw *hw,
|
|
return __clk_rcg2_recalc_rate(hw, parent_rate, cfg);
|
|
}
|
|
|
|
+static void
|
|
+clk_rcg2_select_conf(struct clk_hw *hw, struct freq_tbl *f_tbl,
|
|
+ const struct freq_tbl *f, unsigned long req_rate)
|
|
+{
|
|
+ unsigned long best_rate = 0, parent_rate, rate;
|
|
+ const struct freq_conf *conf, *best_conf;
|
|
+ struct clk_rcg2 *rcg = to_clk_rcg2(hw);
|
|
+ struct clk_hw *p;
|
|
+ int index, i;
|
|
+
|
|
+ /* Search in each provided config the one that is near the wanted rate */
|
|
+ for (i = 0, conf = f->confs; i < f->confs_num; i++, conf++) {
|
|
+ index = qcom_find_src_index(hw, rcg->parent_map, conf->src);
|
|
+ if (index < 0)
|
|
+ continue;
|
|
+
|
|
+ p = clk_hw_get_parent_by_index(hw, index);
|
|
+ if (!p)
|
|
+ continue;
|
|
+
|
|
+ parent_rate = clk_hw_get_rate(p);
|
|
+ rate = calc_rate(parent_rate, conf->n, conf->m, conf->n, conf->pre_div);
|
|
+
|
|
+ if (rate == req_rate) {
|
|
+ best_conf = conf;
|
|
+ break;
|
|
+ }
|
|
+
|
|
+ if (abs(req_rate - rate) < abs(best_rate - rate)) {
|
|
+ best_rate = rate;
|
|
+ best_conf = conf;
|
|
+ }
|
|
+ }
|
|
+
|
|
+ /*
|
|
+ * Very unlikely.
|
|
+ * Force the first conf if we can't find a correct config.
|
|
+ */
|
|
+ if (unlikely(i == f->confs_num))
|
|
+ best_conf = f->confs;
|
|
+
|
|
+ /* Apply the config */
|
|
+ f_tbl->src = best_conf->src;
|
|
+ f_tbl->pre_div = best_conf->pre_div;
|
|
+ f_tbl->m = best_conf->m;
|
|
+ f_tbl->n = best_conf->n;
|
|
+}
|
|
+
|
|
static int _freq_tbl_determine_rate(struct clk_hw *hw, const struct freq_tbl *f,
|
|
struct clk_rate_request *req,
|
|
enum freq_policy policy)
|
|
{
|
|
unsigned long clk_flags, rate = req->rate;
|
|
+ struct freq_tbl f_tbl;
|
|
struct clk_hw *p;
|
|
struct clk_rcg2 *rcg = to_clk_rcg2(hw);
|
|
int index;
|
|
@@ -232,7 +281,15 @@ static int _freq_tbl_determine_rate(stru
|
|
if (!f)
|
|
return -EINVAL;
|
|
|
|
- index = qcom_find_src_index(hw, rcg->parent_map, f->src);
|
|
+ f_tbl = *f;
|
|
+ /*
|
|
+ * A single freq may be reached by multiple configuration.
|
|
+ * Try to find the bast one if we have this kind of freq_table.
|
|
+ */
|
|
+ if (f->confs)
|
|
+ clk_rcg2_select_conf(hw, &f_tbl, f, rate);
|
|
+
|
|
+ index = qcom_find_src_index(hw, rcg->parent_map, f_tbl.src);
|
|
if (index < 0)
|
|
return index;
|
|
|
|
@@ -242,18 +299,18 @@ static int _freq_tbl_determine_rate(stru
|
|
return -EINVAL;
|
|
|
|
if (clk_flags & CLK_SET_RATE_PARENT) {
|
|
- rate = f->freq;
|
|
- if (f->pre_div) {
|
|
+ rate = f_tbl.freq;
|
|
+ if (f_tbl.pre_div) {
|
|
if (!rate)
|
|
rate = req->rate;
|
|
rate /= 2;
|
|
- rate *= f->pre_div + 1;
|
|
+ rate *= f_tbl.pre_div + 1;
|
|
}
|
|
|
|
- if (f->n) {
|
|
+ if (f_tbl.n) {
|
|
u64 tmp = rate;
|
|
- tmp = tmp * f->n;
|
|
- do_div(tmp, f->m);
|
|
+ tmp = tmp * f_tbl.n;
|
|
+ do_div(tmp, f_tbl.m);
|
|
rate = tmp;
|
|
}
|
|
} else {
|
|
@@ -261,7 +318,7 @@ static int _freq_tbl_determine_rate(stru
|
|
}
|
|
req->best_parent_hw = p;
|
|
req->best_parent_rate = rate;
|
|
- req->rate = f->freq;
|
|
+ req->rate = f_tbl.freq;
|
|
|
|
return 0;
|
|
}
|
|
@@ -357,6 +414,7 @@ static int __clk_rcg2_set_rate(struct cl
|
|
{
|
|
struct clk_rcg2 *rcg = to_clk_rcg2(hw);
|
|
const struct freq_tbl *f;
|
|
+ struct freq_tbl f_tbl;
|
|
|
|
switch (policy) {
|
|
case FLOOR:
|
|
@@ -372,7 +430,15 @@ static int __clk_rcg2_set_rate(struct cl
|
|
if (!f)
|
|
return -EINVAL;
|
|
|
|
- return clk_rcg2_configure(rcg, f);
|
|
+ f_tbl = *f;
|
|
+ /*
|
|
+ * A single freq may be reached by multiple configuration.
|
|
+ * Try to find the best one if we have this kind of freq_table.
|
|
+ */
|
|
+ if (f->confs)
|
|
+ clk_rcg2_select_conf(hw, &f_tbl, f, rate);
|
|
+
|
|
+ return clk_rcg2_configure(rcg, &f_tbl);
|
|
}
|
|
|
|
static int clk_rcg2_set_rate(struct clk_hw *hw, unsigned long rate,
|