Commit 3a4c63f5 authored by Georgi Djakov's avatar Georgi Djakov

Merge branch 'icc-rpm' into icc-next

This patch set is to address two clock rate setting issues.

The first patch is to fix a potential cached clock rate mismatching
issue, the issue can lead to the clock rate is missed to be set. Note,
since this potential issue requires specific time window and certain
condition (consumers need to request the same bandwidth) to produce,
the patch is based on analysis but not a real trace log.

The second patch is an extension to cache clock rates for active and
sleep clocks separately, with this change it gives us possibility to set
active and sleep clock with different clock rates.

* icc-rpm
  interconnect: qcom: icc-rpm: Fix for cached clock rate
  interconnect: qcom: icc-rpm: Cache every clock rate

Link: https://lore.kernel.org/r/20220416031029.693211-1-leo.yan@linaro.orgSigned-off-by: default avatarGeorgi Djakov <djakov@kernel.org>
parents 314cf651 97f7d384
...@@ -274,20 +274,19 @@ static int qcom_icc_set(struct icc_node *src, struct icc_node *dst) ...@@ -274,20 +274,19 @@ static int qcom_icc_set(struct icc_node *src, struct icc_node *dst)
do_div(rate, qn->buswidth); do_div(rate, qn->buswidth);
rate = min_t(u64, rate, LONG_MAX); rate = min_t(u64, rate, LONG_MAX);
if (qn->rate == rate)
return 0;
for (i = 0; i < qp->num_clks; i++) { for (i = 0; i < qp->num_clks; i++) {
if (qp->bus_clk_rate[i] == rate)
continue;
ret = clk_set_rate(qp->bus_clks[i].clk, rate); ret = clk_set_rate(qp->bus_clks[i].clk, rate);
if (ret) { if (ret) {
pr_err("%s clk_set_rate error: %d\n", pr_err("%s clk_set_rate error: %d\n",
qp->bus_clks[i].id, ret); qp->bus_clks[i].id, ret);
return ret; return ret;
} }
qp->bus_clk_rate[i] = rate;
} }
qn->rate = rate;
return 0; return 0;
} }
...@@ -332,6 +331,11 @@ int qnoc_probe(struct platform_device *pdev) ...@@ -332,6 +331,11 @@ int qnoc_probe(struct platform_device *pdev)
if (!qp) if (!qp)
return -ENOMEM; return -ENOMEM;
qp->bus_clk_rate = devm_kcalloc(dev, cd_num, sizeof(*qp->bus_clk_rate),
GFP_KERNEL);
if (!qp->bus_clk_rate)
return -ENOMEM;
data = devm_kzalloc(dev, struct_size(data, nodes, num_nodes), data = devm_kzalloc(dev, struct_size(data, nodes, num_nodes),
GFP_KERNEL); GFP_KERNEL);
if (!data) if (!data)
......
...@@ -26,6 +26,7 @@ enum qcom_icc_type { ...@@ -26,6 +26,7 @@ enum qcom_icc_type {
* @type: the ICC provider type * @type: the ICC provider type
* @qos_offset: offset to QoS registers * @qos_offset: offset to QoS registers
* @regmap: regmap for QoS registers read/write access * @regmap: regmap for QoS registers read/write access
* @bus_clk_rate: bus clock rate in Hz
*/ */
struct qcom_icc_provider { struct qcom_icc_provider {
struct icc_provider provider; struct icc_provider provider;
...@@ -33,6 +34,7 @@ struct qcom_icc_provider { ...@@ -33,6 +34,7 @@ struct qcom_icc_provider {
enum qcom_icc_type type; enum qcom_icc_type type;
struct regmap *regmap; struct regmap *regmap;
unsigned int qos_offset; unsigned int qos_offset;
u64 *bus_clk_rate;
struct clk_bulk_data bus_clks[]; struct clk_bulk_data bus_clks[];
}; };
...@@ -66,7 +68,6 @@ struct qcom_icc_qos { ...@@ -66,7 +68,6 @@ struct qcom_icc_qos {
* @mas_rpm_id: RPM id for devices that are bus masters * @mas_rpm_id: RPM id for devices that are bus masters
* @slv_rpm_id: RPM id for devices that are bus slaves * @slv_rpm_id: RPM id for devices that are bus slaves
* @qos: NoC QoS setting parameters * @qos: NoC QoS setting parameters
* @rate: current bus clock rate in Hz
*/ */
struct qcom_icc_node { struct qcom_icc_node {
unsigned char *name; unsigned char *name;
...@@ -77,7 +78,6 @@ struct qcom_icc_node { ...@@ -77,7 +78,6 @@ struct qcom_icc_node {
int mas_rpm_id; int mas_rpm_id;
int slv_rpm_id; int slv_rpm_id;
struct qcom_icc_qos qos; struct qcom_icc_qos qos;
u64 rate;
}; };
struct qcom_icc_desc { struct qcom_icc_desc {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment