mirror of
				git://git.openwrt.org/openwrt/openwrt.git
				synced 2025-11-03 22:44:27 -05:00 
			
		
		
		
	ipq806x: introduce dedicated krait cpufreq
- Drop cpufreq patchs that tweak the cpufreq-dt driver - Add dedicated krait cpufreq driver Signed-off-by: Ansuel Smith <ansuelsmth@gmail.com>
This commit is contained in:
		
							parent
							
								
									e17fb62293
								
							
						
					
					
						commit
						5dbbefcbcc
					
				@ -1,153 +0,0 @@
 | 
				
			|||||||
From: Sylwester Nawrocki <s.nawrocki@samsung.com>
 | 
					 | 
				
			||||||
To: krzk@kernel.org, vireshk@kernel.org, robh+dt@kernel.org
 | 
					 | 
				
			||||||
Cc: sboyd@kernel.org, roger.lu@mediatek.com,
 | 
					 | 
				
			||||||
	linux-pm@vger.kernel.org, linux-arm-kernel@lists.infradead.org,
 | 
					 | 
				
			||||||
	linux-samsung-soc@vger.kernel.org, devicetree@vger.kernel.org,
 | 
					 | 
				
			||||||
	b.zolnierkie@samsung.com, m.szyprowski@samsung.com,
 | 
					 | 
				
			||||||
	Stephen Boyd <sboyd@codeaurora.org>,
 | 
					 | 
				
			||||||
	Sylwester Nawrocki <s.nawrocki@samsung.com>
 | 
					 | 
				
			||||||
Subject: [PATCH v5 1/4] PM / OPP: Support adjusting OPP voltages at runtime
 | 
					 | 
				
			||||||
Date: Wed, 16 Oct 2019 16:57:53 +0200
 | 
					 | 
				
			||||||
Message-ID: <20191016145756.16004-2-s.nawrocki@samsung.com> (raw)
 | 
					 | 
				
			||||||
In-Reply-To: <20191016145756.16004-1-s.nawrocki@samsung.com>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
From: Stephen Boyd <sboyd@codeaurora.org>
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
On some SoCs the Adaptive Voltage Scaling (AVS) technique is
 | 
					 | 
				
			||||||
employed to optimize the operating voltage of a device. At a
 | 
					 | 
				
			||||||
given frequency, the hardware monitors dynamic factors and either
 | 
					 | 
				
			||||||
makes a suggestion for how much to adjust a voltage for the
 | 
					 | 
				
			||||||
current frequency, or it automatically adjusts the voltage
 | 
					 | 
				
			||||||
without software intervention. Add an API to the OPP library for
 | 
					 | 
				
			||||||
the former case, so that AVS type devices can update the voltages
 | 
					 | 
				
			||||||
for an OPP when the hardware determines the voltage should
 | 
					 | 
				
			||||||
change. The assumption is that drivers like CPUfreq or devfreq
 | 
					 | 
				
			||||||
will register for the OPP notifiers and adjust the voltage
 | 
					 | 
				
			||||||
according to suggestions that AVS makes.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
This patch is derived from [1] submitted by Stephen.
 | 
					 | 
				
			||||||
[1] https://lore.kernel.org/patchwork/patch/599279/
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
Signed-off-by: Stephen Boyd <sboyd@codeaurora.org>
 | 
					 | 
				
			||||||
Signed-off-by: Roger Lu <roger.lu@mediatek.com>
 | 
					 | 
				
			||||||
[s.nawrocki@samsung.com: added handling of OPP min/max voltage]
 | 
					 | 
				
			||||||
Signed-off-by: Sylwester Nawrocki <s.nawrocki@samsung.com>
 | 
					 | 
				
			||||||
---
 | 
					 | 
				
			||||||
 drivers/opp/core.c     | 69 ++++++++++++++++++++++++++++++++++++++++++
 | 
					 | 
				
			||||||
 include/linux/pm_opp.h | 13 ++++++++
 | 
					 | 
				
			||||||
 2 files changed, 82 insertions(+)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
--- a/drivers/opp/core.c
 | 
					 | 
				
			||||||
+++ b/drivers/opp/core.c
 | 
					 | 
				
			||||||
@@ -2102,6 +2102,75 @@ put_table:
 | 
					 | 
				
			||||||
 }
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 /**
 | 
					 | 
				
			||||||
+ * dev_pm_opp_adjust_voltage() - helper to change the voltage of an OPP
 | 
					 | 
				
			||||||
+ * @dev:		device for which we do this operation
 | 
					 | 
				
			||||||
+ * @freq:		OPP frequency to adjust voltage of
 | 
					 | 
				
			||||||
+ * @u_volt:		new OPP target voltage
 | 
					 | 
				
			||||||
+ * @u_volt_min:		new OPP min voltage
 | 
					 | 
				
			||||||
+ * @u_volt_max:		new OPP max voltage
 | 
					 | 
				
			||||||
+ *
 | 
					 | 
				
			||||||
+ * Return: -EINVAL for bad pointers, -ENOMEM if no memory available for the
 | 
					 | 
				
			||||||
+ * copy operation, returns 0 if no modifcation was done OR modification was
 | 
					 | 
				
			||||||
+ * successful.
 | 
					 | 
				
			||||||
+ */
 | 
					 | 
				
			||||||
+int dev_pm_opp_adjust_voltage(struct device *dev, unsigned long freq,
 | 
					 | 
				
			||||||
+			      unsigned long u_volt, unsigned long u_volt_min,
 | 
					 | 
				
			||||||
+			      unsigned long u_volt_max)
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+{
 | 
					 | 
				
			||||||
+	struct opp_table *opp_table;
 | 
					 | 
				
			||||||
+	struct dev_pm_opp *tmp_opp, *opp = ERR_PTR(-ENODEV);
 | 
					 | 
				
			||||||
+	int r = 0;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	/* Find the opp_table */
 | 
					 | 
				
			||||||
+	opp_table = _find_opp_table(dev);
 | 
					 | 
				
			||||||
+	if (IS_ERR(opp_table)) {
 | 
					 | 
				
			||||||
+		r = PTR_ERR(opp_table);
 | 
					 | 
				
			||||||
+		dev_warn(dev, "%s: Device OPP not found (%d)\n", __func__, r);
 | 
					 | 
				
			||||||
+		return r;
 | 
					 | 
				
			||||||
+	}
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	mutex_lock(&opp_table->lock);
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	/* Do we have the frequency? */
 | 
					 | 
				
			||||||
+	list_for_each_entry(tmp_opp, &opp_table->opp_list, node) {
 | 
					 | 
				
			||||||
+		if (tmp_opp->rate == freq) {
 | 
					 | 
				
			||||||
+			opp = tmp_opp;
 | 
					 | 
				
			||||||
+			break;
 | 
					 | 
				
			||||||
+		}
 | 
					 | 
				
			||||||
+	}
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	if (IS_ERR(opp)) {
 | 
					 | 
				
			||||||
+		r = PTR_ERR(opp);
 | 
					 | 
				
			||||||
+		goto adjust_unlock;
 | 
					 | 
				
			||||||
+	}
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	/* Is update really needed? */
 | 
					 | 
				
			||||||
+	if (opp->supplies->u_volt == u_volt)
 | 
					 | 
				
			||||||
+		goto adjust_unlock;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	opp->supplies->u_volt = u_volt;
 | 
					 | 
				
			||||||
+	opp->supplies->u_volt_min = u_volt_min;
 | 
					 | 
				
			||||||
+	opp->supplies->u_volt_max = u_volt_max;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	dev_pm_opp_get(opp);
 | 
					 | 
				
			||||||
+	mutex_unlock(&opp_table->lock);
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	/* Notify the voltage change of the OPP */
 | 
					 | 
				
			||||||
+	blocking_notifier_call_chain(&opp_table->head, OPP_EVENT_ADJUST_VOLTAGE,
 | 
					 | 
				
			||||||
+				     opp);
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	dev_pm_opp_put(opp);
 | 
					 | 
				
			||||||
+	goto adjust_put_table;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+adjust_unlock:
 | 
					 | 
				
			||||||
+	mutex_unlock(&opp_table->lock);
 | 
					 | 
				
			||||||
+adjust_put_table:
 | 
					 | 
				
			||||||
+	dev_pm_opp_put_opp_table(opp_table);
 | 
					 | 
				
			||||||
+	return r;
 | 
					 | 
				
			||||||
+}
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+/**
 | 
					 | 
				
			||||||
  * dev_pm_opp_enable() - Enable a specific OPP
 | 
					 | 
				
			||||||
  * @dev:	device for which we do this operation
 | 
					 | 
				
			||||||
  * @freq:	OPP frequency to enable
 | 
					 | 
				
			||||||
--- a/include/linux/pm_opp.h
 | 
					 | 
				
			||||||
+++ b/include/linux/pm_opp.h
 | 
					 | 
				
			||||||
@@ -22,6 +22,7 @@ struct opp_table;
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 enum dev_pm_opp_event {
 | 
					 | 
				
			||||||
 	OPP_EVENT_ADD, OPP_EVENT_REMOVE, OPP_EVENT_ENABLE, OPP_EVENT_DISABLE,
 | 
					 | 
				
			||||||
+	OPP_EVENT_ADJUST_VOLTAGE,
 | 
					 | 
				
			||||||
 };
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 /**
 | 
					 | 
				
			||||||
@@ -113,6 +114,10 @@ int dev_pm_opp_add(struct device *dev, u
 | 
					 | 
				
			||||||
 void dev_pm_opp_remove(struct device *dev, unsigned long freq);
 | 
					 | 
				
			||||||
 void dev_pm_opp_remove_all_dynamic(struct device *dev);
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
+int dev_pm_opp_adjust_voltage(struct device *dev, unsigned long freq,
 | 
					 | 
				
			||||||
+			      unsigned long u_volt, unsigned long u_volt_min,
 | 
					 | 
				
			||||||
+			      unsigned long u_volt_max);
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
 int dev_pm_opp_enable(struct device *dev, unsigned long freq);
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 int dev_pm_opp_disable(struct device *dev, unsigned long freq);
 | 
					 | 
				
			||||||
@@ -242,6 +247,14 @@ static inline void dev_pm_opp_remove_all
 | 
					 | 
				
			||||||
 {
 | 
					 | 
				
			||||||
 }
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
+static inline int
 | 
					 | 
				
			||||||
+dev_pm_opp_adjust_voltage(struct device *dev, unsigned long freq,
 | 
					 | 
				
			||||||
+			  unsigned long u_volt, unsigned long u_volt_min,
 | 
					 | 
				
			||||||
+			  unsigned long u_volt_max)
 | 
					 | 
				
			||||||
+{
 | 
					 | 
				
			||||||
+	return 0;
 | 
					 | 
				
			||||||
+}
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
 static inline int dev_pm_opp_enable(struct device *dev, unsigned long freq)
 | 
					 | 
				
			||||||
 {
 | 
					 | 
				
			||||||
 	return 0;
 | 
					 | 
				
			||||||
@ -1,52 +0,0 @@
 | 
				
			|||||||
From d06ca5e7a3cf726f5be5ffd96e93ccd798b8c09a Mon Sep 17 00:00:00 2001
 | 
					 | 
				
			||||||
From: Georgi Djakov <georgi.djakov@linaro.org>
 | 
					 | 
				
			||||||
Date: Thu, 12 May 2016 14:41:33 +0300
 | 
					 | 
				
			||||||
Subject: [PATCH 51/69] PM / OPP: Add a helper to get an opp regulator for
 | 
					 | 
				
			||||||
 device
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
Signed-off-by: Georgi Djakov <georgi.djakov@linaro.org>
 | 
					 | 
				
			||||||
---
 | 
					 | 
				
			||||||
 drivers/opp/core.c            | 21 +++++++++++++++++++++
 | 
					 | 
				
			||||||
 include/linux/pm_opp.h        |  1 +
 | 
					 | 
				
			||||||
 2 files changed, 22 insertions(+)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
--- a/drivers/opp/core.c
 | 
					 | 
				
			||||||
+++ b/drivers/opp/core.c
 | 
					 | 
				
			||||||
@@ -127,6 +127,27 @@ unsigned long dev_pm_opp_get_freq(struct
 | 
					 | 
				
			||||||
 }
 | 
					 | 
				
			||||||
 EXPORT_SYMBOL_GPL(dev_pm_opp_get_freq);
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
+struct regulator *dev_pm_opp_get_regulator(struct device *dev)
 | 
					 | 
				
			||||||
+{
 | 
					 | 
				
			||||||
+	struct opp_table *opp_table;
 | 
					 | 
				
			||||||
+	struct regulator *reg;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	rcu_read_lock();
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	opp_table = _find_opp_table(dev);
 | 
					 | 
				
			||||||
+	if (IS_ERR(opp_table)) {
 | 
					 | 
				
			||||||
+		rcu_read_unlock();
 | 
					 | 
				
			||||||
+		return ERR_CAST(opp_table);
 | 
					 | 
				
			||||||
+	}
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	reg = opp_table->regulators[0];
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	rcu_read_unlock();
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	return reg;
 | 
					 | 
				
			||||||
+}
 | 
					 | 
				
			||||||
+EXPORT_SYMBOL_GPL(dev_pm_opp_get_regulator);
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
 /**
 | 
					 | 
				
			||||||
  * dev_pm_opp_get_level() - Gets the level corresponding to an available opp
 | 
					 | 
				
			||||||
  * @opp:	opp for which level value has to be returned for
 | 
					 | 
				
			||||||
--- a/include/linux/pm_opp.h
 | 
					 | 
				
			||||||
+++ b/include/linux/pm_opp.h
 | 
					 | 
				
			||||||
@@ -83,6 +83,7 @@ void dev_pm_opp_put_opp_table(struct opp
 | 
					 | 
				
			||||||
 unsigned long dev_pm_opp_get_voltage(struct dev_pm_opp *opp);
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 unsigned long dev_pm_opp_get_freq(struct dev_pm_opp *opp);
 | 
					 | 
				
			||||||
+struct regulator *dev_pm_opp_get_regulator(struct device *dev);
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 unsigned int dev_pm_opp_get_level(struct dev_pm_opp *opp);
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
@ -1,47 +0,0 @@
 | 
				
			|||||||
From 4533c285c2aedce6d4434d7b877066de3b1ecb33 Mon Sep 17 00:00:00 2001
 | 
					 | 
				
			||||||
From: Georgi Djakov <georgi.djakov@linaro.org>
 | 
					 | 
				
			||||||
Date: Thu, 25 Aug 2016 18:43:35 +0300
 | 
					 | 
				
			||||||
Subject: [PATCH 52/69] PM / OPP: Update the voltage tolerance when adjusting
 | 
					 | 
				
			||||||
 the OPP
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
When the voltage is adjusted, the voltage tolerance is not updated.
 | 
					 | 
				
			||||||
This can lead to situations where the voltage min value is greater
 | 
					 | 
				
			||||||
than the voltage max value. The final result is triggering a BUG()
 | 
					 | 
				
			||||||
in the regulator core.
 | 
					 | 
				
			||||||
Fix this by updating the voltage tolerance values too.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
Signed-off-by: Georgi Djakov <georgi.djakov@linaro.org>
 | 
					 | 
				
			||||||
---
 | 
					 | 
				
			||||||
 drivers/opp/core.c | 5 +++++
 | 
					 | 
				
			||||||
 1 file changed, 5 insertions(+)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
--- a/drivers/opp/core.c
 | 
					 | 
				
			||||||
+++ b/drivers/opp/core.c
 | 
					 | 
				
			||||||
@@ -2142,6 +2142,7 @@ int dev_pm_opp_adjust_voltage(struct dev
 | 
					 | 
				
			||||||
 	struct opp_table *opp_table;
 | 
					 | 
				
			||||||
 	struct dev_pm_opp *tmp_opp, *opp = ERR_PTR(-ENODEV);
 | 
					 | 
				
			||||||
 	int r = 0;
 | 
					 | 
				
			||||||
+	unsigned long tol;
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 	/* Find the opp_table */
 | 
					 | 
				
			||||||
 	opp_table = _find_opp_table(dev);
 | 
					 | 
				
			||||||
@@ -2171,8 +2172,17 @@ int dev_pm_opp_adjust_voltage(struct dev
 | 
					 | 
				
			||||||
 		goto adjust_unlock;
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 	opp->supplies->u_volt = u_volt;
 | 
					 | 
				
			||||||
-	opp->supplies->u_volt_min = u_volt_min;
 | 
					 | 
				
			||||||
-	opp->supplies->u_volt_max = u_volt_max;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	tol = u_volt * opp_table->voltage_tolerance_v1 / 100;
 | 
					 | 
				
			||||||
+	if ( u_volt_min == u_volt )
 | 
					 | 
				
			||||||
+		opp->supplies->u_volt_min = u_volt - tol;
 | 
					 | 
				
			||||||
+	else
 | 
					 | 
				
			||||||
+		opp->supplies->u_volt_min = u_volt_min;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	if ( u_volt_max == u_volt )
 | 
					 | 
				
			||||||
+		opp->supplies->u_volt_max = u_volt + tol;
 | 
					 | 
				
			||||||
+	else
 | 
					 | 
				
			||||||
+		opp->supplies->u_volt_max = u_volt_max;
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 	dev_pm_opp_get(opp);
 | 
					 | 
				
			||||||
 	mutex_unlock(&opp_table->lock);
 | 
					 | 
				
			||||||
@ -1,118 +0,0 @@
 | 
				
			|||||||
From 10577f74c35bd395951d1b2382c8d821089b5745 Mon Sep 17 00:00:00 2001
 | 
					 | 
				
			||||||
From: Stephen Boyd <sboyd@codeaurora.org>
 | 
					 | 
				
			||||||
Date: Fri, 18 Sep 2015 17:52:08 -0700
 | 
					 | 
				
			||||||
Subject: [PATCH 54/69] cpufreq-dt: Handle OPP voltage adjust events
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
On some SoCs the Adaptive Voltage Scaling (AVS) technique is
 | 
					 | 
				
			||||||
employed to optimize the operating voltage of a device. At a
 | 
					 | 
				
			||||||
given frequency, the hardware monitors dynamic factors and either
 | 
					 | 
				
			||||||
makes a suggestion for how much to adjust a voltage for the
 | 
					 | 
				
			||||||
current frequency, or it automatically adjusts the voltage
 | 
					 | 
				
			||||||
without software intervention.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
In the former case, an AVS driver will call
 | 
					 | 
				
			||||||
dev_pm_opp_modify_voltage() and update the voltage for the
 | 
					 | 
				
			||||||
particular OPP the CPUs are using. Add an OPP notifier to
 | 
					 | 
				
			||||||
cpufreq-dt so that we can adjust the voltage of the CPU when AVS
 | 
					 | 
				
			||||||
updates the OPP.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
Signed-off-by: Stephen Boyd <sboyd@codeaurora.org>
 | 
					 | 
				
			||||||
Acked-by: Viresh Kumar <viresh.kumar@linaro.org>
 | 
					 | 
				
			||||||
Signed-off-by: Georgi Djakov <georgi.djakov@linaro.org>
 | 
					 | 
				
			||||||
---
 | 
					 | 
				
			||||||
 drivers/cpufreq/cpufreq-dt.c | 68 ++++++++++++++++++++++++++++++++++++++++++--
 | 
					 | 
				
			||||||
 1 file changed, 65 insertions(+), 3 deletions(-)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
--- a/drivers/cpufreq/cpufreq-dt.c
 | 
					 | 
				
			||||||
+++ b/drivers/cpufreq/cpufreq-dt.c
 | 
					 | 
				
			||||||
@@ -27,6 +27,9 @@ struct private_data {
 | 
					 | 
				
			||||||
 	struct opp_table *opp_table;
 | 
					 | 
				
			||||||
 	struct device *cpu_dev;
 | 
					 | 
				
			||||||
 	const char *reg_name;
 | 
					 | 
				
			||||||
+	struct notifier_block opp_nb;
 | 
					 | 
				
			||||||
+	struct mutex lock;
 | 
					 | 
				
			||||||
+	unsigned long opp_freq;
 | 
					 | 
				
			||||||
 	bool have_static_opps;
 | 
					 | 
				
			||||||
 };
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
@@ -42,12 +45,15 @@ static int set_target(struct cpufreq_pol
 | 
					 | 
				
			||||||
 	unsigned long freq = policy->freq_table[index].frequency;
 | 
					 | 
				
			||||||
 	int ret;
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
+	mutex_lock(&priv->lock);
 | 
					 | 
				
			||||||
 	ret = dev_pm_opp_set_rate(priv->cpu_dev, freq * 1000);
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 	if (!ret) {
 | 
					 | 
				
			||||||
+		priv->opp_freq = freq * 1000;
 | 
					 | 
				
			||||||
 		arch_set_freq_scale(policy->related_cpus, freq,
 | 
					 | 
				
			||||||
 				    policy->cpuinfo.max_freq);
 | 
					 | 
				
			||||||
 	}
 | 
					 | 
				
			||||||
+	mutex_unlock(&priv->lock);
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 	return ret;
 | 
					 | 
				
			||||||
 }
 | 
					 | 
				
			||||||
@@ -90,6 +96,39 @@ node_put:
 | 
					 | 
				
			||||||
 	return name;
 | 
					 | 
				
			||||||
 }
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
+static int opp_notifier(struct notifier_block *nb, unsigned long event,
 | 
					 | 
				
			||||||
+			void *data)
 | 
					 | 
				
			||||||
+{
 | 
					 | 
				
			||||||
+	struct dev_pm_opp *opp = data;
 | 
					 | 
				
			||||||
+	struct private_data *priv = container_of(nb, struct private_data,
 | 
					 | 
				
			||||||
+						 opp_nb);
 | 
					 | 
				
			||||||
+	struct device *cpu_dev = priv->cpu_dev;
 | 
					 | 
				
			||||||
+	struct regulator *cpu_reg;
 | 
					 | 
				
			||||||
+	unsigned long volt, freq;
 | 
					 | 
				
			||||||
+	int ret = 0;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	if (event == OPP_EVENT_ADJUST_VOLTAGE) {
 | 
					 | 
				
			||||||
+		cpu_reg = dev_pm_opp_get_regulator(cpu_dev);
 | 
					 | 
				
			||||||
+		if (IS_ERR(cpu_reg)) {
 | 
					 | 
				
			||||||
+			ret = PTR_ERR(cpu_reg);
 | 
					 | 
				
			||||||
+			goto out;
 | 
					 | 
				
			||||||
+		}
 | 
					 | 
				
			||||||
+		volt = dev_pm_opp_get_voltage(opp);
 | 
					 | 
				
			||||||
+		freq = dev_pm_opp_get_freq(opp);
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+		mutex_lock(&priv->lock);
 | 
					 | 
				
			||||||
+		if (freq == priv->opp_freq) {
 | 
					 | 
				
			||||||
+			ret = regulator_set_voltage_triplet(cpu_reg, volt, volt, volt);
 | 
					 | 
				
			||||||
+		}
 | 
					 | 
				
			||||||
+		mutex_unlock(&priv->lock);
 | 
					 | 
				
			||||||
+		if (ret)
 | 
					 | 
				
			||||||
+			dev_err(cpu_dev, "failed to scale voltage: %d\n", ret);
 | 
					 | 
				
			||||||
+	}
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+out:
 | 
					 | 
				
			||||||
+	return notifier_from_errno(ret);
 | 
					 | 
				
			||||||
+}
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
 static int resources_available(void)
 | 
					 | 
				
			||||||
 {
 | 
					 | 
				
			||||||
 	struct device *cpu_dev;
 | 
					 | 
				
			||||||
@@ -246,10 +285,14 @@ static int cpufreq_init(struct cpufreq_p
 | 
					 | 
				
			||||||
 				__func__, ret);
 | 
					 | 
				
			||||||
 	}
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
+	mutex_init(&priv->lock);
 | 
					 | 
				
			||||||
+	priv->opp_nb.notifier_call = opp_notifier;
 | 
					 | 
				
			||||||
+	dev_pm_opp_register_notifier(cpu_dev, &priv->opp_nb);
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
 	ret = dev_pm_opp_init_cpufreq_table(cpu_dev, &freq_table);
 | 
					 | 
				
			||||||
 	if (ret) {
 | 
					 | 
				
			||||||
 		dev_err(cpu_dev, "failed to init cpufreq table: %d\n", ret);
 | 
					 | 
				
			||||||
-		goto out_free_opp;
 | 
					 | 
				
			||||||
+		goto out_unregister_nb;
 | 
					 | 
				
			||||||
 	}
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 	priv->cpu_dev = cpu_dev;
 | 
					 | 
				
			||||||
@@ -281,6 +324,8 @@ static int cpufreq_init(struct cpufreq_p
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 out_free_cpufreq_table:
 | 
					 | 
				
			||||||
 	dev_pm_opp_free_cpufreq_table(cpu_dev, &freq_table);
 | 
					 | 
				
			||||||
+out_unregister_nb:
 | 
					 | 
				
			||||||
+	dev_pm_opp_unregister_notifier(cpu_dev, &priv->opp_nb);
 | 
					 | 
				
			||||||
 out_free_opp:
 | 
					 | 
				
			||||||
 	if (priv->have_static_opps)
 | 
					 | 
				
			||||||
 		dev_pm_opp_of_cpumask_remove_table(policy->cpus);
 | 
					 | 
				
			||||||
@ -1,199 +0,0 @@
 | 
				
			|||||||
--- a/drivers/cpufreq/cpufreq-dt.c
 | 
					 | 
				
			||||||
+++ b/drivers/cpufreq/cpufreq-dt.c
 | 
					 | 
				
			||||||
@@ -39,20 +39,85 @@ static struct freq_attr *cpufreq_dt_attr
 | 
					 | 
				
			||||||
 	NULL,
 | 
					 | 
				
			||||||
 };
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
+struct shared_data {
 | 
					 | 
				
			||||||
+	unsigned long *curr_freq_table;
 | 
					 | 
				
			||||||
+	unsigned long curr_l2_freq;
 | 
					 | 
				
			||||||
+	unsigned long curr_l2_volt;
 | 
					 | 
				
			||||||
+};
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+static struct shared_data *cpus_shared_data;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
 static int set_target(struct cpufreq_policy *policy, unsigned int index)
 | 
					 | 
				
			||||||
 {
 | 
					 | 
				
			||||||
 	struct private_data *priv = policy->driver_data;
 | 
					 | 
				
			||||||
 	unsigned long freq = policy->freq_table[index].frequency;
 | 
					 | 
				
			||||||
+	struct clk *l2_clk = policy->l2_clk;
 | 
					 | 
				
			||||||
+	struct regulator *l2_regulator = policy->l2_regulator;
 | 
					 | 
				
			||||||
+	unsigned long l2_freq, target_l2_freq;
 | 
					 | 
				
			||||||
+	unsigned long l2_vol, target_l2_volt;
 | 
					 | 
				
			||||||
+	unsigned long target_freq;
 | 
					 | 
				
			||||||
 	int ret;
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 	mutex_lock(&priv->lock);
 | 
					 | 
				
			||||||
 	ret = dev_pm_opp_set_rate(priv->cpu_dev, freq * 1000);
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 	if (!ret) {
 | 
					 | 
				
			||||||
+		if (policy->l2_rate_set) {
 | 
					 | 
				
			||||||
+			int cpu, l2_index, tol = 0;
 | 
					 | 
				
			||||||
+			target_freq = freq * 1000;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+			cpus_shared_data->curr_freq_table[policy->cpu] = target_freq;
 | 
					 | 
				
			||||||
+			for_each_present_cpu(cpu)
 | 
					 | 
				
			||||||
+				if (cpu != policy->cpu)
 | 
					 | 
				
			||||||
+					target_freq = max(target_freq, cpus_shared_data->curr_freq_table[cpu]);
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+			for (l2_index = 2; l2_index > 0; l2_index--)
 | 
					 | 
				
			||||||
+				if (target_freq >= policy->l2_cpufreq[l2_index])
 | 
					 | 
				
			||||||
+					break;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+			l2_freq = cpus_shared_data->curr_l2_freq;
 | 
					 | 
				
			||||||
+			target_l2_freq = policy->l2_rate[l2_index];
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+			if (l2_freq != target_l2_freq) {
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+				/*
 | 
					 | 
				
			||||||
+				 * Set to idle bin if switching from normal to high bin 
 | 
					 | 
				
			||||||
+				 * or vice versa
 | 
					 | 
				
			||||||
+				 */
 | 
					 | 
				
			||||||
+				if ( (l2_index == 2 && l2_freq == policy->l2_rate[1]) ||
 | 
					 | 
				
			||||||
+					 (l2_index == 1 && l2_freq == policy->l2_rate[2]) ) {
 | 
					 | 
				
			||||||
+					ret = clk_set_rate(l2_clk, policy->l2_rate[0]);
 | 
					 | 
				
			||||||
+					if (ret)
 | 
					 | 
				
			||||||
+						goto exit;
 | 
					 | 
				
			||||||
+				}
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+				/* scale l2 with the core */
 | 
					 | 
				
			||||||
+				ret = clk_set_rate(l2_clk, target_l2_freq);
 | 
					 | 
				
			||||||
+				if (ret)
 | 
					 | 
				
			||||||
+					goto exit;
 | 
					 | 
				
			||||||
+				cpus_shared_data->curr_l2_freq = target_l2_freq;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+				if (policy->l2_volt_set) {
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+					l2_vol = cpus_shared_data->curr_l2_volt;
 | 
					 | 
				
			||||||
+					target_l2_volt = policy->l2_volt[l2_index];
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+					if (l2_vol != target_l2_volt) {
 | 
					 | 
				
			||||||
+						tol = target_l2_volt * policy->l2_volt_tol / 100;
 | 
					 | 
				
			||||||
+						ret = regulator_set_voltage_tol(l2_regulator,target_l2_volt,tol);
 | 
					 | 
				
			||||||
+						if (ret)
 | 
					 | 
				
			||||||
+							goto exit;
 | 
					 | 
				
			||||||
+						cpus_shared_data->curr_l2_volt = target_l2_volt;
 | 
					 | 
				
			||||||
+					}
 | 
					 | 
				
			||||||
+				}
 | 
					 | 
				
			||||||
+			}
 | 
					 | 
				
			||||||
+		}
 | 
					 | 
				
			||||||
 		priv->opp_freq = freq * 1000;
 | 
					 | 
				
			||||||
 		arch_set_freq_scale(policy->related_cpus, freq,
 | 
					 | 
				
			||||||
 				    policy->cpuinfo.max_freq);
 | 
					 | 
				
			||||||
 	}
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+exit:
 | 
					 | 
				
			||||||
 	mutex_unlock(&priv->lock);
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 	return ret;
 | 
					 | 
				
			||||||
@@ -195,6 +260,9 @@ static int cpufreq_init(struct cpufreq_p
 | 
					 | 
				
			||||||
 	bool fallback = false;
 | 
					 | 
				
			||||||
 	const char *name;
 | 
					 | 
				
			||||||
 	int ret;
 | 
					 | 
				
			||||||
+	struct device_node *np, *l2_np;
 | 
					 | 
				
			||||||
+	struct clk *l2_clk = NULL;
 | 
					 | 
				
			||||||
+	struct regulator *l2_regulator = NULL;
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 	cpu_dev = get_cpu_device(policy->cpu);
 | 
					 | 
				
			||||||
 	if (!cpu_dev) {
 | 
					 | 
				
			||||||
@@ -302,6 +370,57 @@ static int cpufreq_init(struct cpufreq_p
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 	policy->suspend_freq = dev_pm_opp_get_suspend_opp_freq(cpu_dev) / 1000;
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
+	policy->l2_rate_set = false;
 | 
					 | 
				
			||||||
+	policy->l2_volt_set = false;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	l2_clk = clk_get(cpu_dev, "l2");
 | 
					 | 
				
			||||||
+	if (!IS_ERR(l2_clk))
 | 
					 | 
				
			||||||
+		policy->l2_clk = l2_clk;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	l2_np = of_find_node_by_name(NULL, "qcom,l2");
 | 
					 | 
				
			||||||
+	if (l2_np) {
 | 
					 | 
				
			||||||
+		struct device_node *vdd;
 | 
					 | 
				
			||||||
+		np = of_node_get(priv->cpu_dev->of_node);
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+		if (np)
 | 
					 | 
				
			||||||
+			of_property_read_u32(np, "voltage-tolerance", &policy->l2_volt_tol);
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+		of_property_read_u32_array(l2_np, "qcom,l2-rates", policy->l2_rate, 3);
 | 
					 | 
				
			||||||
+		if (policy->l2_rate[0] && policy->l2_rate[1] && policy->l2_rate[2]) {
 | 
					 | 
				
			||||||
+			policy->l2_rate_set = true;
 | 
					 | 
				
			||||||
+			of_property_read_u32_array(l2_np, "qcom,l2-cpufreq", policy->l2_cpufreq, 3);
 | 
					 | 
				
			||||||
+			of_property_read_u32_array(l2_np, "qcom,l2-volt", policy->l2_volt, 3);
 | 
					 | 
				
			||||||
+		} else
 | 
					 | 
				
			||||||
+			pr_warn("L2: failed to parse L2 rates\n");
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+		if (!policy->l2_cpufreq[0] && !policy->l2_cpufreq[1] && 
 | 
					 | 
				
			||||||
+			!policy->l2_cpufreq[2] && policy->l2_rate_set) {
 | 
					 | 
				
			||||||
+			int i;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+			pr_warn("L2: failed to parse target cpu freq, using defaults\n");
 | 
					 | 
				
			||||||
+			for (i = 0; i < 3; i++)
 | 
					 | 
				
			||||||
+				policy->l2_cpufreq[i] = policy->l2_rate[i];
 | 
					 | 
				
			||||||
+		}
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+		if (policy->l2_volt[0] && policy->l2_volt[1] && policy->l2_volt[2] &&
 | 
					 | 
				
			||||||
+			policy->l2_volt_tol && policy->l2_rate_set) {
 | 
					 | 
				
			||||||
+			vdd = of_parse_phandle(l2_np, "qcom,l2-supply", 0);
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+			if (vdd) {
 | 
					 | 
				
			||||||
+				l2_regulator = devm_regulator_get(cpu_dev, vdd->name);
 | 
					 | 
				
			||||||
+				if (!IS_ERR(l2_regulator)) {
 | 
					 | 
				
			||||||
+					policy->l2_regulator = l2_regulator;
 | 
					 | 
				
			||||||
+					policy->l2_volt_set = true;
 | 
					 | 
				
			||||||
+				} else {
 | 
					 | 
				
			||||||
+					pr_warn("failed to get l2 supply\n");
 | 
					 | 
				
			||||||
+					l2_regulator = NULL;
 | 
					 | 
				
			||||||
+				}
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+				of_node_put(vdd);
 | 
					 | 
				
			||||||
+			}
 | 
					 | 
				
			||||||
+		}
 | 
					 | 
				
			||||||
+	}
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
 	/* Support turbo/boost mode */
 | 
					 | 
				
			||||||
 	if (policy_has_boost_freq(policy)) {
 | 
					 | 
				
			||||||
 		/* This gets disabled by core on driver unregister */
 | 
					 | 
				
			||||||
@@ -401,6 +520,14 @@ static int dt_cpufreq_probe(struct platf
 | 
					 | 
				
			||||||
 	if (ret)
 | 
					 | 
				
			||||||
 		return ret;
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
+	cpus_shared_data = kzalloc(sizeof(struct shared_data), GFP_KERNEL);
 | 
					 | 
				
			||||||
+	if (!cpus_shared_data)
 | 
					 | 
				
			||||||
+		return -ENOMEM;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+	cpus_shared_data->curr_freq_table = kzalloc(sizeof(int) * CONFIG_NR_CPUS, GFP_KERNEL);
 | 
					 | 
				
			||||||
+	if (!cpus_shared_data->curr_freq_table)
 | 
					 | 
				
			||||||
+		return -ENOMEM;
 | 
					 | 
				
			||||||
+
 | 
					 | 
				
			||||||
 	if (data) {
 | 
					 | 
				
			||||||
 		if (data->have_governor_per_policy)
 | 
					 | 
				
			||||||
 			dt_cpufreq_driver.flags |= CPUFREQ_HAVE_GOVERNOR_PER_POLICY;
 | 
					 | 
				
			||||||
@@ -419,6 +546,8 @@ static int dt_cpufreq_probe(struct platf
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 static int dt_cpufreq_remove(struct platform_device *pdev)
 | 
					 | 
				
			||||||
 {
 | 
					 | 
				
			||||||
+	kfree(cpus_shared_data->curr_freq_table);
 | 
					 | 
				
			||||||
+	kfree(cpus_shared_data);
 | 
					 | 
				
			||||||
 	cpufreq_unregister_driver(&dt_cpufreq_driver);
 | 
					 | 
				
			||||||
 	return 0;
 | 
					 | 
				
			||||||
 }
 | 
					 | 
				
			||||||
--- a/include/linux/cpufreq.h
 | 
					 | 
				
			||||||
+++ b/include/linux/cpufreq.h
 | 
					 | 
				
			||||||
@@ -58,7 +58,15 @@ struct cpufreq_policy {
 | 
					 | 
				
			||||||
 						should set cpufreq */
 | 
					 | 
				
			||||||
 	unsigned int		cpu;    /* cpu managing this policy, must be online */
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
-	struct clk		*clk;
 | 
					 | 
				
			||||||
+	struct clk			*clk;
 | 
					 | 
				
			||||||
+	struct clk			*l2_clk; /* L2 clock */
 | 
					 | 
				
			||||||
+	struct regulator	*l2_regulator; /* L2 supply */
 | 
					 | 
				
			||||||
+	unsigned int		l2_rate[3]; /* L2 bus clock rate */
 | 
					 | 
				
			||||||
+	bool				l2_rate_set;
 | 
					 | 
				
			||||||
+	unsigned int		l2_cpufreq[3]; /* L2 target CPU frequency */
 | 
					 | 
				
			||||||
+	unsigned int		l2_volt[3]; /* L2 voltage array */
 | 
					 | 
				
			||||||
+	bool				l2_volt_set;
 | 
					 | 
				
			||||||
+	unsigned int		l2_volt_tol; /* L2 voltage tolerance */
 | 
					 | 
				
			||||||
 	struct cpufreq_cpuinfo	cpuinfo;/* see above */
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 	unsigned int		min;    /* in kHz */
 | 
					 | 
				
			||||||
@ -1,23 +0,0 @@
 | 
				
			|||||||
From 001a8dcb56ced58c518aaa10a4f0ba5e878705b6 Mon Sep 17 00:00:00 2001
 | 
					 | 
				
			||||||
From: Georgi Djakov <georgi.djakov@linaro.org>
 | 
					 | 
				
			||||||
Date: Tue, 17 May 2016 16:15:43 +0300
 | 
					 | 
				
			||||||
Subject: [PATCH 56/69] cpufreq-dt: Add missing rcu locks
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
Signed-off-by: Georgi Djakov <georgi.djakov@linaro.org>
 | 
					 | 
				
			||||||
---
 | 
					 | 
				
			||||||
 drivers/cpufreq/cpufreq-dt.c | 2 ++
 | 
					 | 
				
			||||||
 1 file changed, 2 insertions(+)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
--- a/drivers/cpufreq/cpufreq-dt.c
 | 
					 | 
				
			||||||
+++ b/drivers/cpufreq/cpufreq-dt.c
 | 
					 | 
				
			||||||
@@ -178,8 +178,10 @@ static int opp_notifier(struct notifier_
 | 
					 | 
				
			||||||
 			ret = PTR_ERR(cpu_reg);
 | 
					 | 
				
			||||||
 			goto out;
 | 
					 | 
				
			||||||
 		}
 | 
					 | 
				
			||||||
+		rcu_read_lock();
 | 
					 | 
				
			||||||
 		volt = dev_pm_opp_get_voltage(opp);
 | 
					 | 
				
			||||||
 		freq = dev_pm_opp_get_freq(opp);
 | 
					 | 
				
			||||||
+		rcu_read_unlock();
 | 
					 | 
				
			||||||
 
 | 
					 | 
				
			||||||
 		mutex_lock(&priv->lock);
 | 
					 | 
				
			||||||
 		if (freq == priv->opp_freq) {
 | 
					 | 
				
			||||||
@ -0,0 +1,56 @@
 | 
				
			|||||||
 | 
					From a206d4061f1cc2c5cd17ee45c53a0ba711e48e6d Mon Sep 17 00:00:00 2001
 | 
				
			||||||
 | 
					From: Ansuel Smith <ansuelsmth@gmail.com>
 | 
				
			||||||
 | 
					Date: Sun, 7 Feb 2021 16:42:52 +0100
 | 
				
			||||||
 | 
					Subject: [PATCH 3/3] drivers: cpufreq: qcom-cpufreq-nvmem: support specific
 | 
				
			||||||
 | 
					 cpufreq driver
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Add support for specific cpufreq driver for qcom-cpufreq-nvmem driver.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Signed-off-by: Ansuel Smith <ansuelsmth@gmail.com>
 | 
				
			||||||
 | 
					---
 | 
				
			||||||
 | 
					 drivers/cpufreq/qcom-cpufreq-nvmem.c | 15 +++++++++++++++
 | 
				
			||||||
 | 
					 1 file changed, 15 insertions(+)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					diff --git a/drivers/cpufreq/qcom-cpufreq-nvmem.c b/drivers/cpufreq/qcom-cpufreq-nvmem.c
 | 
				
			||||||
 | 
					index d1744b5d9619..4866c74ead0f 100644
 | 
				
			||||||
 | 
					--- a/drivers/cpufreq/qcom-cpufreq-nvmem.c
 | 
				
			||||||
 | 
					+++ b/drivers/cpufreq/qcom-cpufreq-nvmem.c
 | 
				
			||||||
 | 
					@@ -52,6 +52,7 @@ struct qcom_cpufreq_match_data {
 | 
				
			||||||
 | 
					 			   char **pvs_name,
 | 
				
			||||||
 | 
					 			   struct qcom_cpufreq_drv *drv);
 | 
				
			||||||
 | 
					 	const char **genpd_names;
 | 
				
			||||||
 | 
					+	const char *cpufreq_driver;
 | 
				
			||||||
 | 
					 };
 | 
				
			||||||
 | 
					 
 | 
				
			||||||
 | 
					 struct qcom_cpufreq_drv {
 | 
				
			||||||
 | 
					@@ -250,6 +251,7 @@ static const struct qcom_cpufreq_match_data match_data_kryo = {
 | 
				
			||||||
 | 
					 
 | 
				
			||||||
 | 
					 static const struct qcom_cpufreq_match_data match_data_krait = {
 | 
				
			||||||
 | 
					 	.get_version = qcom_cpufreq_krait_name_version,
 | 
				
			||||||
 | 
					+	.cpufreq_driver = "krait-cpufreq",
 | 
				
			||||||
 | 
					 };
 | 
				
			||||||
 | 
					 
 | 
				
			||||||
 | 
					 static const char *qcs404_genpd_names[] = { "cpr", NULL };
 | 
				
			||||||
 | 
					@@ -385,6 +387,19 @@ static int qcom_cpufreq_probe(struct platform_device *pdev)
 | 
				
			||||||
 | 
					 		}
 | 
				
			||||||
 | 
					 	}
 | 
				
			||||||
 | 
					 
 | 
				
			||||||
 | 
					+	if (drv->data->cpufreq_driver) {
 | 
				
			||||||
 | 
					+		cpufreq_dt_pdev = platform_device_register_simple(
 | 
				
			||||||
 | 
					+			drv->data->cpufreq_driver, -1, NULL, 0);
 | 
				
			||||||
 | 
					+		if (!IS_ERR(cpufreq_dt_pdev)) {
 | 
				
			||||||
 | 
					+			platform_set_drvdata(pdev, drv);
 | 
				
			||||||
 | 
					+			return 0;
 | 
				
			||||||
 | 
					+		} else {
 | 
				
			||||||
 | 
					+			dev_err(cpu_dev,
 | 
				
			||||||
 | 
					+				"Failed to register dedicated %s cpufreq\n",
 | 
				
			||||||
 | 
					+				drv->data->cpufreq_driver);
 | 
				
			||||||
 | 
					+		}
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					 	cpufreq_dt_pdev = platform_device_register_simple("cpufreq-dt", -1,
 | 
				
			||||||
 | 
					 							  NULL, 0);
 | 
				
			||||||
 | 
					 	if (!IS_ERR(cpufreq_dt_pdev)) {
 | 
				
			||||||
 | 
					-- 
 | 
				
			||||||
 | 
					2.29.2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -0,0 +1,679 @@
 | 
				
			|||||||
 | 
					From cc41a266280cad0b55319e614167c88dff344248 Mon Sep 17 00:00:00 2001
 | 
				
			||||||
 | 
					From: Ansuel Smith <ansuelsmth@gmail.com>
 | 
				
			||||||
 | 
					Date: Sat, 22 Feb 2020 16:33:10 +0100
 | 
				
			||||||
 | 
					Subject: [PATCH 1/8] cpufreq: add Krait dedicated scaling driver
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					This new driver is based on generic cpufreq-dt driver.
 | 
				
			||||||
 | 
					Krait SoCs have 2-4 cpu and one shared L2 cache that can
 | 
				
			||||||
 | 
					operate at different frequency based on the maximum cpu clk
 | 
				
			||||||
 | 
					across all core.
 | 
				
			||||||
 | 
					L2 frequency and voltage are scaled on every frequency change
 | 
				
			||||||
 | 
					if needed. On Krait SoCs is present a bug that can cause
 | 
				
			||||||
 | 
					transition problem between frequency bin, to workaround this
 | 
				
			||||||
 | 
					on more than one transition, the L2 frequency is first set to the
 | 
				
			||||||
 | 
					base rate and then to the target rate.
 | 
				
			||||||
 | 
					The L2 frequency use the OPP framework and use the opp-level
 | 
				
			||||||
 | 
					bindings to link the l2 freq to different cpu freq. This is needed
 | 
				
			||||||
 | 
					as the Krait l2 clk are note mapped 1:1 to the core clks and some
 | 
				
			||||||
 | 
					of the l2 clk is set based on a range of the cpu clks. If the driver
 | 
				
			||||||
 | 
					find a broken config (for example no opp-level set) the l2 scaling is
 | 
				
			||||||
 | 
					skipped.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Signed-off-by: Ansuel Smith <ansuelsmth@gmail.com>
 | 
				
			||||||
 | 
					---
 | 
				
			||||||
 | 
					 drivers/cpufreq/Kconfig.arm          |  14 +-
 | 
				
			||||||
 | 
					 drivers/cpufreq/Makefile             |   2 +
 | 
				
			||||||
 | 
					 drivers/cpufreq/qcom-cpufreq-krait.c | 589 +++++++++++++++++++++++++++
 | 
				
			||||||
 | 
					 3 files changed, 604 insertions(+), 1 deletion(-)
 | 
				
			||||||
 | 
					 create mode 100644 drivers/cpufreq/qcom-cpufreq-krait.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--- a/drivers/cpufreq/Kconfig.arm
 | 
				
			||||||
 | 
					+++ b/drivers/cpufreq/Kconfig.arm
 | 
				
			||||||
 | 
					@@ -150,6 +150,18 @@ config ARM_QCOM_CPUFREQ_HW
 | 
				
			||||||
 | 
					 	  The driver implements the cpufreq interface for this HW engine.
 | 
				
			||||||
 | 
					 	  Say Y if you want to support CPUFreq HW.
 | 
				
			||||||
 | 
					 
 | 
				
			||||||
 | 
					+config ARM_QCOM_CPUFREQ_KRAIT
 | 
				
			||||||
 | 
					+	tristate "CPU Frequency scaling support for Krait SoCs"
 | 
				
			||||||
 | 
					+	depends on ARCH_QCOM || COMPILE_TEST
 | 
				
			||||||
 | 
					+	select PM_OPP
 | 
				
			||||||
 | 
					+	select ARM_QCOM_CPUFREQ_NVMEM
 | 
				
			||||||
 | 
					+	help
 | 
				
			||||||
 | 
					+	  This adds the CPUFreq driver for Qualcomm Krait SoC based boards.
 | 
				
			||||||
 | 
					+	  This scale the cache clk and regulator based on the different cpu
 | 
				
			||||||
 | 
					+	  clks when scaling the different cores clk.
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	  If in doubt, say N.
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					 config ARM_RASPBERRYPI_CPUFREQ
 | 
				
			||||||
 | 
					 	tristate "Raspberry Pi cpufreq support"
 | 
				
			||||||
 | 
					 	depends on CLK_RASPBERRYPI || COMPILE_TEST
 | 
				
			||||||
 | 
					@@ -339,4 +351,4 @@ config ARM_PXA2xx_CPUFREQ
 | 
				
			||||||
 | 
					 	help
 | 
				
			||||||
 | 
					 	  This add the CPUFreq driver support for Intel PXA2xx SOCs.
 | 
				
			||||||
 | 
					 
 | 
				
			||||||
 | 
					-	  If in doubt, say N.
 | 
				
			||||||
 | 
					+	  If in doubt, say N.
 | 
				
			||||||
 | 
					\ No newline at end of file
 | 
				
			||||||
 | 
					--- a/drivers/cpufreq/Makefile
 | 
				
			||||||
 | 
					+++ b/drivers/cpufreq/Makefile
 | 
				
			||||||
 | 
					@@ -63,6 +63,7 @@ obj-$(CONFIG_ARM_PXA2xx_CPUFREQ)	+= pxa2
 | 
				
			||||||
 | 
					 obj-$(CONFIG_PXA3xx)			+= pxa3xx-cpufreq.o
 | 
				
			||||||
 | 
					 obj-$(CONFIG_ARM_QCOM_CPUFREQ_HW)	+= qcom-cpufreq-hw.o
 | 
				
			||||||
 | 
					 obj-$(CONFIG_ARM_QCOM_CPUFREQ_NVMEM)	+= qcom-cpufreq-nvmem.o
 | 
				
			||||||
 | 
					+obj-$(CONFIG_ARM_QCOM_CPUFREQ_KRAIT)	+= qcom-cpufreq-krait.o
 | 
				
			||||||
 | 
					 obj-$(CONFIG_ARM_RASPBERRYPI_CPUFREQ) 	+= raspberrypi-cpufreq.o
 | 
				
			||||||
 | 
					 obj-$(CONFIG_ARM_S3C2410_CPUFREQ)	+= s3c2410-cpufreq.o
 | 
				
			||||||
 | 
					 obj-$(CONFIG_ARM_S3C2412_CPUFREQ)	+= s3c2412-cpufreq.o
 | 
				
			||||||
 | 
					@@ -86,6 +87,7 @@ obj-$(CONFIG_ARM_TEGRA186_CPUFREQ)	+= te
 | 
				
			||||||
 | 
					 obj-$(CONFIG_ARM_TEGRA194_CPUFREQ)	+= tegra194-cpufreq.o
 | 
				
			||||||
 | 
					 obj-$(CONFIG_ARM_TI_CPUFREQ)		+= ti-cpufreq.o
 | 
				
			||||||
 | 
					 obj-$(CONFIG_ARM_VEXPRESS_SPC_CPUFREQ)	+= vexpress-spc-cpufreq.o
 | 
				
			||||||
 | 
					+obj-$(CONFIG_ARM_KRAIT_CPUFREQ)		+= krait-cpufreq.o
 | 
				
			||||||
 | 
					 
 | 
				
			||||||
 | 
					 
 | 
				
			||||||
 | 
					 ##################################################################################
 | 
				
			||||||
 | 
					--- /dev/null
 | 
				
			||||||
 | 
					+++ b/drivers/cpufreq/qcom-cpufreq-krait.c
 | 
				
			||||||
 | 
					@@ -0,0 +1,601 @@
 | 
				
			||||||
 | 
					+// SPDX-License-Identifier: GPL-2.0
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+#include <linux/clk.h>
 | 
				
			||||||
 | 
					+#include <linux/cpu.h>
 | 
				
			||||||
 | 
					+#include <linux/cpufreq.h>
 | 
				
			||||||
 | 
					+#include <linux/cpumask.h>
 | 
				
			||||||
 | 
					+#include <linux/err.h>
 | 
				
			||||||
 | 
					+#include <linux/module.h>
 | 
				
			||||||
 | 
					+#include <linux/of.h>
 | 
				
			||||||
 | 
					+#include <linux/of_device.h>
 | 
				
			||||||
 | 
					+#include <linux/pm_opp.h>
 | 
				
			||||||
 | 
					+#include <linux/platform_device.h>
 | 
				
			||||||
 | 
					+#include <linux/regulator/consumer.h>
 | 
				
			||||||
 | 
					+#include <linux/slab.h>
 | 
				
			||||||
 | 
					+#include <linux/thermal.h>
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+#include "cpufreq-dt.h"
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static struct platform_device *l2_pdev;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+struct private_data {
 | 
				
			||||||
 | 
					+	struct opp_table *opp_table;
 | 
				
			||||||
 | 
					+	struct device *cpu_dev;
 | 
				
			||||||
 | 
					+	const char *reg_name;
 | 
				
			||||||
 | 
					+	bool have_static_opps;
 | 
				
			||||||
 | 
					+};
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static int set_target(struct cpufreq_policy *policy, unsigned int index)
 | 
				
			||||||
 | 
					+{
 | 
				
			||||||
 | 
					+	struct private_data *priv = policy->driver_data;
 | 
				
			||||||
 | 
					+	unsigned long freq = policy->freq_table[index].frequency;
 | 
				
			||||||
 | 
					+	unsigned long target_freq = freq * 1000;
 | 
				
			||||||
 | 
					+	struct dev_pm_opp *opp;
 | 
				
			||||||
 | 
					+	unsigned int level;
 | 
				
			||||||
 | 
					+	int cpu, ret;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	if (l2_pdev) {
 | 
				
			||||||
 | 
					+		/* find the max freq across all core */
 | 
				
			||||||
 | 
					+		for_each_present_cpu(cpu)
 | 
				
			||||||
 | 
					+			if (cpu != index)
 | 
				
			||||||
 | 
					+				target_freq = max(
 | 
				
			||||||
 | 
					+					target_freq,
 | 
				
			||||||
 | 
					+					(unsigned long)cpufreq_quick_get(cpu));
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+		opp = dev_pm_opp_find_freq_exact(priv->cpu_dev, target_freq,
 | 
				
			||||||
 | 
					+						 true);
 | 
				
			||||||
 | 
					+		if (IS_ERR(opp)) {
 | 
				
			||||||
 | 
					+			dev_err(&l2_pdev->dev, "failed to find OPP for %ld\n",
 | 
				
			||||||
 | 
					+				target_freq);
 | 
				
			||||||
 | 
					+			return PTR_ERR(opp);
 | 
				
			||||||
 | 
					+		}
 | 
				
			||||||
 | 
					+		level = dev_pm_opp_get_level(opp);
 | 
				
			||||||
 | 
					+		dev_pm_opp_put(opp);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+		opp = dev_pm_opp_find_level_exact(&l2_pdev->dev, level);
 | 
				
			||||||
 | 
					+		if (IS_ERR(opp)) {
 | 
				
			||||||
 | 
					+			dev_err(&l2_pdev->dev,
 | 
				
			||||||
 | 
					+				"failed to find level OPP for %d\n", level);
 | 
				
			||||||
 | 
					+			return PTR_ERR(opp);
 | 
				
			||||||
 | 
					+		}
 | 
				
			||||||
 | 
					+		target_freq = dev_pm_opp_get_freq(opp);
 | 
				
			||||||
 | 
					+		dev_pm_opp_put(opp);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+		ret = dev_pm_opp_set_rate(&l2_pdev->dev, target_freq);
 | 
				
			||||||
 | 
					+		if (ret)
 | 
				
			||||||
 | 
					+			return ret;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+		/*
 | 
				
			||||||
 | 
					+		 * Hardware constraint:
 | 
				
			||||||
 | 
					+		 * Krait CPU cannot operate at 384MHz with L2 at 1Ghz.
 | 
				
			||||||
 | 
					+		 * Assume index 0 with the idle freq and level > 0 as 
 | 
				
			||||||
 | 
					+		 * any L2 freq > 384MHz.
 | 
				
			||||||
 | 
					+		 * Skip CPU freq change in this corner case.
 | 
				
			||||||
 | 
					+		 */
 | 
				
			||||||
 | 
					+		if (unlikely(index == 0 && level != 0)) {
 | 
				
			||||||
 | 
					+			dev_err(priv->cpu_dev, "Krait CPU can't operate at idle freq with L2 at 1GHz");
 | 
				
			||||||
 | 
					+			return -EINVAL;
 | 
				
			||||||
 | 
					+		}
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	ret = dev_pm_opp_set_rate(priv->cpu_dev, freq * 1000);
 | 
				
			||||||
 | 
					+	if (ret)
 | 
				
			||||||
 | 
					+		return ret;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	arch_set_freq_scale(policy->related_cpus, freq,
 | 
				
			||||||
 | 
					+			    policy->cpuinfo.max_freq);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	return 0;
 | 
				
			||||||
 | 
					+}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+/*
 | 
				
			||||||
 | 
					+ * An earlier version of opp-v1 bindings used to name the regulator
 | 
				
			||||||
 | 
					+ * "cpu0-supply", we still need to handle that for backwards compatibility.
 | 
				
			||||||
 | 
					+ */
 | 
				
			||||||
 | 
					+static const char *find_supply_name(struct device *dev)
 | 
				
			||||||
 | 
					+{
 | 
				
			||||||
 | 
					+	struct device_node *np;
 | 
				
			||||||
 | 
					+	struct property *pp;
 | 
				
			||||||
 | 
					+	int cpu = dev->id;
 | 
				
			||||||
 | 
					+	const char *name = NULL;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	np = of_node_get(dev->of_node);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/* This must be valid for sure */
 | 
				
			||||||
 | 
					+	if (WARN_ON(!np))
 | 
				
			||||||
 | 
					+		return NULL;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/* Try "cpu0" for older DTs */
 | 
				
			||||||
 | 
					+	if (!cpu) {
 | 
				
			||||||
 | 
					+		pp = of_find_property(np, "cpu0-supply", NULL);
 | 
				
			||||||
 | 
					+		if (pp) {
 | 
				
			||||||
 | 
					+			name = "cpu0";
 | 
				
			||||||
 | 
					+			goto node_put;
 | 
				
			||||||
 | 
					+		}
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	pp = of_find_property(np, "cpu-supply", NULL);
 | 
				
			||||||
 | 
					+	if (pp) {
 | 
				
			||||||
 | 
					+		name = "cpu";
 | 
				
			||||||
 | 
					+		goto node_put;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	dev_dbg(dev, "no regulator for cpu%d\n", cpu);
 | 
				
			||||||
 | 
					+node_put:
 | 
				
			||||||
 | 
					+	of_node_put(np);
 | 
				
			||||||
 | 
					+	return name;
 | 
				
			||||||
 | 
					+}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static int resources_available(void)
 | 
				
			||||||
 | 
					+{
 | 
				
			||||||
 | 
					+	struct device *cpu_dev;
 | 
				
			||||||
 | 
					+	struct regulator *cpu_reg;
 | 
				
			||||||
 | 
					+	struct clk *cpu_clk;
 | 
				
			||||||
 | 
					+	int ret = 0;
 | 
				
			||||||
 | 
					+	const char *name;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	cpu_dev = get_cpu_device(0);
 | 
				
			||||||
 | 
					+	if (!cpu_dev) {
 | 
				
			||||||
 | 
					+		pr_err("failed to get cpu0 device\n");
 | 
				
			||||||
 | 
					+		return -ENODEV;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	cpu_clk = clk_get(cpu_dev, NULL);
 | 
				
			||||||
 | 
					+	ret = PTR_ERR_OR_ZERO(cpu_clk);
 | 
				
			||||||
 | 
					+	if (ret) {
 | 
				
			||||||
 | 
					+		/*
 | 
				
			||||||
 | 
					+		 * If cpu's clk node is present, but clock is not yet
 | 
				
			||||||
 | 
					+		 * registered, we should try defering probe.
 | 
				
			||||||
 | 
					+		 */
 | 
				
			||||||
 | 
					+		if (ret == -EPROBE_DEFER)
 | 
				
			||||||
 | 
					+			dev_dbg(cpu_dev, "clock not ready, retry\n");
 | 
				
			||||||
 | 
					+		else
 | 
				
			||||||
 | 
					+			dev_err(cpu_dev, "failed to get clock: %d\n", ret);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+		return ret;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	clk_put(cpu_clk);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	name = find_supply_name(cpu_dev);
 | 
				
			||||||
 | 
					+	/* Platform doesn't require regulator */
 | 
				
			||||||
 | 
					+	if (!name)
 | 
				
			||||||
 | 
					+		return 0;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	cpu_reg = regulator_get_optional(cpu_dev, name);
 | 
				
			||||||
 | 
					+	ret = PTR_ERR_OR_ZERO(cpu_reg);
 | 
				
			||||||
 | 
					+	if (ret) {
 | 
				
			||||||
 | 
					+		/*
 | 
				
			||||||
 | 
					+		 * If cpu's regulator supply node is present, but regulator is
 | 
				
			||||||
 | 
					+		 * not yet registered, we should try defering probe.
 | 
				
			||||||
 | 
					+		 */
 | 
				
			||||||
 | 
					+		if (ret == -EPROBE_DEFER)
 | 
				
			||||||
 | 
					+			dev_dbg(cpu_dev, "cpu0 regulator not ready, retry\n");
 | 
				
			||||||
 | 
					+		else
 | 
				
			||||||
 | 
					+			dev_dbg(cpu_dev, "no regulator for cpu0: %d\n", ret);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+		return ret;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	regulator_put(cpu_reg);
 | 
				
			||||||
 | 
					+	return 0;
 | 
				
			||||||
 | 
					+}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static int cpufreq_init(struct cpufreq_policy *policy)
 | 
				
			||||||
 | 
					+{
 | 
				
			||||||
 | 
					+	struct cpufreq_frequency_table *freq_table;
 | 
				
			||||||
 | 
					+	struct opp_table *opp_table = NULL;
 | 
				
			||||||
 | 
					+	unsigned int transition_latency;
 | 
				
			||||||
 | 
					+	struct private_data *priv;
 | 
				
			||||||
 | 
					+	struct device *cpu_dev;
 | 
				
			||||||
 | 
					+	bool fallback = false;
 | 
				
			||||||
 | 
					+	struct clk *cpu_clk;
 | 
				
			||||||
 | 
					+	const char *name;
 | 
				
			||||||
 | 
					+	int ret;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	cpu_dev = get_cpu_device(policy->cpu);
 | 
				
			||||||
 | 
					+	if (!cpu_dev) {
 | 
				
			||||||
 | 
					+		pr_err("failed to get cpu%d device\n", policy->cpu);
 | 
				
			||||||
 | 
					+		return -ENODEV;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	cpu_clk = clk_get(cpu_dev, NULL);
 | 
				
			||||||
 | 
					+	if (IS_ERR(cpu_clk)) {
 | 
				
			||||||
 | 
					+		ret = PTR_ERR(cpu_clk);
 | 
				
			||||||
 | 
					+		dev_err(cpu_dev, "%s: failed to get clk: %d\n", __func__, ret);
 | 
				
			||||||
 | 
					+		return ret;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/* Get OPP-sharing information from "operating-points-v2" bindings */
 | 
				
			||||||
 | 
					+	ret = dev_pm_opp_of_get_sharing_cpus(cpu_dev, policy->cpus);
 | 
				
			||||||
 | 
					+	if (ret) {
 | 
				
			||||||
 | 
					+		if (ret != -ENOENT)
 | 
				
			||||||
 | 
					+			goto out_put_clk;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+		/*
 | 
				
			||||||
 | 
					+		 * operating-points-v2 not supported, fallback to old method of
 | 
				
			||||||
 | 
					+		 * finding shared-OPPs for backward compatibility if the
 | 
				
			||||||
 | 
					+		 * platform hasn't set sharing CPUs.
 | 
				
			||||||
 | 
					+		 */
 | 
				
			||||||
 | 
					+		if (dev_pm_opp_get_sharing_cpus(cpu_dev, policy->cpus))
 | 
				
			||||||
 | 
					+			fallback = true;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/*
 | 
				
			||||||
 | 
					+	 * OPP layer will be taking care of regulators now, but it needs to know
 | 
				
			||||||
 | 
					+	 * the name of the regulator first.
 | 
				
			||||||
 | 
					+	 */
 | 
				
			||||||
 | 
					+	name = find_supply_name(cpu_dev);
 | 
				
			||||||
 | 
					+	if (name) {
 | 
				
			||||||
 | 
					+		opp_table = dev_pm_opp_set_regulators(cpu_dev, &name, 1);
 | 
				
			||||||
 | 
					+		if (IS_ERR(opp_table)) {
 | 
				
			||||||
 | 
					+			ret = PTR_ERR(opp_table);
 | 
				
			||||||
 | 
					+			dev_err(cpu_dev,
 | 
				
			||||||
 | 
					+				"Failed to set regulator for cpu%d: %d\n",
 | 
				
			||||||
 | 
					+				policy->cpu, ret);
 | 
				
			||||||
 | 
					+			goto out_put_clk;
 | 
				
			||||||
 | 
					+		}
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	priv = kzalloc(sizeof(*priv), GFP_KERNEL);
 | 
				
			||||||
 | 
					+	if (!priv) {
 | 
				
			||||||
 | 
					+		ret = -ENOMEM;
 | 
				
			||||||
 | 
					+		goto out_put_regulator;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	priv->reg_name = name;
 | 
				
			||||||
 | 
					+	priv->opp_table = opp_table;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/*
 | 
				
			||||||
 | 
					+	 * Initialize OPP tables for all policy->cpus. They will be shared by
 | 
				
			||||||
 | 
					+	 * all CPUs which have marked their CPUs shared with OPP bindings.
 | 
				
			||||||
 | 
					+	 *
 | 
				
			||||||
 | 
					+	 * For platforms not using operating-points-v2 bindings, we do this
 | 
				
			||||||
 | 
					+	 * before updating policy->cpus. Otherwise, we will end up creating
 | 
				
			||||||
 | 
					+	 * duplicate OPPs for policy->cpus.
 | 
				
			||||||
 | 
					+	 *
 | 
				
			||||||
 | 
					+	 * OPPs might be populated at runtime, don't check for error here
 | 
				
			||||||
 | 
					+	 */
 | 
				
			||||||
 | 
					+	if (!dev_pm_opp_of_cpumask_add_table(policy->cpus))
 | 
				
			||||||
 | 
					+		priv->have_static_opps = true;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/*
 | 
				
			||||||
 | 
					+	 * But we need OPP table to function so if it is not there let's
 | 
				
			||||||
 | 
					+	 * give platform code chance to provide it for us.
 | 
				
			||||||
 | 
					+	 */
 | 
				
			||||||
 | 
					+	ret = dev_pm_opp_get_opp_count(cpu_dev);
 | 
				
			||||||
 | 
					+	if (ret <= 0) {
 | 
				
			||||||
 | 
					+		dev_dbg(cpu_dev, "OPP table is not ready, deferring probe\n");
 | 
				
			||||||
 | 
					+		ret = -EPROBE_DEFER;
 | 
				
			||||||
 | 
					+		goto out_free_opp;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	if (fallback) {
 | 
				
			||||||
 | 
					+		cpumask_setall(policy->cpus);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+		/*
 | 
				
			||||||
 | 
					+		 * OPP tables are initialized only for policy->cpu, do it for
 | 
				
			||||||
 | 
					+		 * others as well.
 | 
				
			||||||
 | 
					+		 */
 | 
				
			||||||
 | 
					+		ret = dev_pm_opp_set_sharing_cpus(cpu_dev, policy->cpus);
 | 
				
			||||||
 | 
					+		if (ret)
 | 
				
			||||||
 | 
					+			dev_err(cpu_dev,
 | 
				
			||||||
 | 
					+				"%s: failed to mark OPPs as shared: %d\n",
 | 
				
			||||||
 | 
					+				__func__, ret);
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	ret = dev_pm_opp_init_cpufreq_table(cpu_dev, &freq_table);
 | 
				
			||||||
 | 
					+	if (ret) {
 | 
				
			||||||
 | 
					+		dev_err(cpu_dev, "failed to init cpufreq table: %d\n", ret);
 | 
				
			||||||
 | 
					+		goto out_free_opp;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	priv->cpu_dev = cpu_dev;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	policy->driver_data = priv;
 | 
				
			||||||
 | 
					+	policy->clk = cpu_clk;
 | 
				
			||||||
 | 
					+	policy->freq_table = freq_table;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	policy->suspend_freq = dev_pm_opp_get_suspend_opp_freq(cpu_dev) / 1000;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	transition_latency = dev_pm_opp_get_max_transition_latency(cpu_dev);
 | 
				
			||||||
 | 
					+	if (!transition_latency)
 | 
				
			||||||
 | 
					+		transition_latency = CPUFREQ_ETERNAL;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	policy->cpuinfo.transition_latency = transition_latency;
 | 
				
			||||||
 | 
					+	policy->dvfs_possible_from_any_cpu = true;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	dev_pm_opp_of_register_em(cpu_dev, policy->cpus);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	return 0;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+out_free_opp:
 | 
				
			||||||
 | 
					+	if (priv->have_static_opps)
 | 
				
			||||||
 | 
					+		dev_pm_opp_of_cpumask_remove_table(policy->cpus);
 | 
				
			||||||
 | 
					+	kfree(priv);
 | 
				
			||||||
 | 
					+out_put_regulator:
 | 
				
			||||||
 | 
					+	if (name)
 | 
				
			||||||
 | 
					+		dev_pm_opp_put_regulators(opp_table);
 | 
				
			||||||
 | 
					+out_put_clk:
 | 
				
			||||||
 | 
					+	clk_put(cpu_clk);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	return ret;
 | 
				
			||||||
 | 
					+}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static int cpufreq_online(struct cpufreq_policy *policy)
 | 
				
			||||||
 | 
					+{
 | 
				
			||||||
 | 
					+	/* We did light-weight tear down earlier, nothing to do here */
 | 
				
			||||||
 | 
					+	return 0;
 | 
				
			||||||
 | 
					+}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static int cpufreq_offline(struct cpufreq_policy *policy)
 | 
				
			||||||
 | 
					+{
 | 
				
			||||||
 | 
					+	/*
 | 
				
			||||||
 | 
					+	 * Preserve policy->driver_data and don't free resources on light-weight
 | 
				
			||||||
 | 
					+	 * tear down.
 | 
				
			||||||
 | 
					+	 */
 | 
				
			||||||
 | 
					+	return 0;
 | 
				
			||||||
 | 
					+}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static int cpufreq_exit(struct cpufreq_policy *policy)
 | 
				
			||||||
 | 
					+{
 | 
				
			||||||
 | 
					+	struct private_data *priv = policy->driver_data;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	dev_pm_opp_free_cpufreq_table(priv->cpu_dev, &policy->freq_table);
 | 
				
			||||||
 | 
					+	if (priv->have_static_opps)
 | 
				
			||||||
 | 
					+		dev_pm_opp_of_cpumask_remove_table(policy->related_cpus);
 | 
				
			||||||
 | 
					+	if (priv->reg_name)
 | 
				
			||||||
 | 
					+		dev_pm_opp_put_regulators(priv->opp_table);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	clk_put(policy->clk);
 | 
				
			||||||
 | 
					+	kfree(priv);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	return 0;
 | 
				
			||||||
 | 
					+}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static struct cpufreq_driver krait_cpufreq_driver = {
 | 
				
			||||||
 | 
					+	.flags = CPUFREQ_STICKY | CPUFREQ_NEED_INITIAL_FREQ_CHECK |
 | 
				
			||||||
 | 
					+		 CPUFREQ_IS_COOLING_DEV,
 | 
				
			||||||
 | 
					+	.verify = cpufreq_generic_frequency_table_verify,
 | 
				
			||||||
 | 
					+	.target_index = set_target,
 | 
				
			||||||
 | 
					+	.get = cpufreq_generic_get,
 | 
				
			||||||
 | 
					+	.init = cpufreq_init,
 | 
				
			||||||
 | 
					+	.exit = cpufreq_exit,
 | 
				
			||||||
 | 
					+	.online = cpufreq_online,
 | 
				
			||||||
 | 
					+	.offline = cpufreq_offline,
 | 
				
			||||||
 | 
					+	.name = "krait-cpufreq",
 | 
				
			||||||
 | 
					+	.suspend = cpufreq_generic_suspend,
 | 
				
			||||||
 | 
					+};
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+struct krait_data {
 | 
				
			||||||
 | 
					+	unsigned long idle_freq;
 | 
				
			||||||
 | 
					+	bool regulator_enabled;
 | 
				
			||||||
 | 
					+};
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static int krait_cache_set_opp(struct dev_pm_set_opp_data *data)
 | 
				
			||||||
 | 
					+{
 | 
				
			||||||
 | 
					+	unsigned long old_freq = data->old_opp.rate, freq = data->new_opp.rate;
 | 
				
			||||||
 | 
					+	struct dev_pm_opp_supply *supply = &data->new_opp.supplies[0];
 | 
				
			||||||
 | 
					+	struct regulator *reg = data->regulators[0];
 | 
				
			||||||
 | 
					+	struct clk *clk = data->clk;
 | 
				
			||||||
 | 
					+	struct krait_data *kdata;
 | 
				
			||||||
 | 
					+	unsigned long idle_freq;
 | 
				
			||||||
 | 
					+	int ret;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	kdata = (struct krait_data *)dev_get_drvdata(data->dev);
 | 
				
			||||||
 | 
					+	idle_freq = kdata->idle_freq;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/* Scaling up? Scale voltage before frequency */
 | 
				
			||||||
 | 
					+	if (freq >= old_freq) {
 | 
				
			||||||
 | 
					+		ret = regulator_set_voltage_triplet(reg, supply->u_volt_min,
 | 
				
			||||||
 | 
					+						    supply->u_volt,
 | 
				
			||||||
 | 
					+						    supply->u_volt_max);
 | 
				
			||||||
 | 
					+		if (ret)
 | 
				
			||||||
 | 
					+			goto exit;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/*
 | 
				
			||||||
 | 
					+	 * Set to idle bin if switching from normal to high bin
 | 
				
			||||||
 | 
					+	 * or vice versa. It has been notice that a bug is triggered
 | 
				
			||||||
 | 
					+	 * in cache scaling when more than one bin is scaled, to fix
 | 
				
			||||||
 | 
					+	 * this we first need to transition to the base rate and then
 | 
				
			||||||
 | 
					+	 * to target rate
 | 
				
			||||||
 | 
					+	 */
 | 
				
			||||||
 | 
					+	if (likely(freq != idle_freq && old_freq != idle_freq)) {
 | 
				
			||||||
 | 
					+		ret = clk_set_rate(clk, idle_freq);
 | 
				
			||||||
 | 
					+		if (ret)
 | 
				
			||||||
 | 
					+			goto exit;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	ret = clk_set_rate(clk, freq);
 | 
				
			||||||
 | 
					+	if (ret)
 | 
				
			||||||
 | 
					+		goto exit;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/* Scaling down? Scale voltage after frequency */
 | 
				
			||||||
 | 
					+	if (freq < old_freq) {
 | 
				
			||||||
 | 
					+		ret = regulator_set_voltage_triplet(reg, supply->u_volt_min,
 | 
				
			||||||
 | 
					+						    supply->u_volt,
 | 
				
			||||||
 | 
					+						    supply->u_volt_max);
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	if (unlikely(!kdata->regulator_enabled)) {
 | 
				
			||||||
 | 
					+		ret = regulator_enable(reg);
 | 
				
			||||||
 | 
					+		if (ret < 0)
 | 
				
			||||||
 | 
					+			dev_warn(data->dev, "Failed to enable regulator: %d", ret);
 | 
				
			||||||
 | 
					+		else
 | 
				
			||||||
 | 
					+			kdata->regulator_enabled = true;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+exit:
 | 
				
			||||||
 | 
					+	return ret;
 | 
				
			||||||
 | 
					+};
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static int krait_cache_probe(struct platform_device *pdev)
 | 
				
			||||||
 | 
					+{
 | 
				
			||||||
 | 
					+	struct device *dev = &pdev->dev;
 | 
				
			||||||
 | 
					+	struct krait_data *data;
 | 
				
			||||||
 | 
					+	struct opp_table *table;
 | 
				
			||||||
 | 
					+	struct dev_pm_opp *opp;
 | 
				
			||||||
 | 
					+	struct device *cpu_dev;
 | 
				
			||||||
 | 
					+	int ret;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	data = devm_kzalloc(dev, sizeof(*data), GFP_KERNEL);
 | 
				
			||||||
 | 
					+	if (!data)
 | 
				
			||||||
 | 
					+		return -ENOMEM;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	table = dev_pm_opp_set_regulators(dev, (const char *[]){ "l2" }, 1);
 | 
				
			||||||
 | 
					+	if (IS_ERR(table)) {
 | 
				
			||||||
 | 
					+		ret = PTR_ERR(table);
 | 
				
			||||||
 | 
					+		if (ret != -EPROBE_DEFER)
 | 
				
			||||||
 | 
					+			dev_err(dev, "failed to set regulators %d\n", ret);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+		return ret;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	ret = PTR_ERR_OR_ZERO(
 | 
				
			||||||
 | 
					+		dev_pm_opp_register_set_opp_helper(dev, krait_cache_set_opp));
 | 
				
			||||||
 | 
					+	if (ret)
 | 
				
			||||||
 | 
					+		return ret;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	ret = dev_pm_opp_of_add_table(dev);
 | 
				
			||||||
 | 
					+	if (ret) {
 | 
				
			||||||
 | 
					+		dev_err(dev, "failed to parse L2 freq thresholds\n");
 | 
				
			||||||
 | 
					+		return ret;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	opp = dev_pm_opp_find_freq_ceil(dev, &data->idle_freq);
 | 
				
			||||||
 | 
					+	dev_pm_opp_put(opp);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/*
 | 
				
			||||||
 | 
					+	 * Check opp-level configuration
 | 
				
			||||||
 | 
					+	 * At least 2 level must be set or the cache will always be scaled
 | 
				
			||||||
 | 
					+	 * the idle freq causing some performance problem
 | 
				
			||||||
 | 
					+	 *
 | 
				
			||||||
 | 
					+	 * In case of invalid configuration, the l2 scaling is skipped
 | 
				
			||||||
 | 
					+	 */
 | 
				
			||||||
 | 
					+	cpu_dev = get_cpu_device(0);
 | 
				
			||||||
 | 
					+	if (!cpu_dev) {
 | 
				
			||||||
 | 
					+		pr_err("failed to get cpu0 device\n");
 | 
				
			||||||
 | 
					+		return -ENODEV;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/*
 | 
				
			||||||
 | 
					+	 * Check if we have at least opp-level 1, 0 should always be set to
 | 
				
			||||||
 | 
					+	 * the idle freq
 | 
				
			||||||
 | 
					+	 */
 | 
				
			||||||
 | 
					+	opp = dev_pm_opp_find_level_exact(dev, 1);
 | 
				
			||||||
 | 
					+	if (IS_ERR(opp)) {
 | 
				
			||||||
 | 
					+		dev_err(dev,
 | 
				
			||||||
 | 
					+			"Invalid configuration found of l2 opp. Can't find opp-level 1");
 | 
				
			||||||
 | 
					+		goto invalid_conf;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+	dev_pm_opp_put(opp);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/*
 | 
				
			||||||
 | 
					+	 * Check if we have at least opp-level 1 in the cpu opp, 0 should always
 | 
				
			||||||
 | 
					+	 * be set to the idle freq
 | 
				
			||||||
 | 
					+	 */
 | 
				
			||||||
 | 
					+	opp = dev_pm_opp_find_level_exact(cpu_dev, 1);
 | 
				
			||||||
 | 
					+	if (IS_ERR(opp)) {
 | 
				
			||||||
 | 
					+		dev_err(dev,
 | 
				
			||||||
 | 
					+			"Invalid configuration found of cpu opp. Can't find opp-level 1");
 | 
				
			||||||
 | 
					+		goto invalid_conf;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+	dev_pm_opp_put(opp);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	platform_set_drvdata(pdev, data);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/* The l2 scaling is enabled by linking the cpufreq driver */
 | 
				
			||||||
 | 
					+	l2_pdev = pdev;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	return 0;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+invalid_conf:
 | 
				
			||||||
 | 
					+	dev_pm_opp_remove_table(dev);
 | 
				
			||||||
 | 
					+	dev_pm_opp_put_regulators(table);
 | 
				
			||||||
 | 
					+	dev_pm_opp_unregister_set_opp_helper(table);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	return -EINVAL;
 | 
				
			||||||
 | 
					+};
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static int krait_cache_remove(struct platform_device *pdev)
 | 
				
			||||||
 | 
					+{
 | 
				
			||||||
 | 
					+	struct device *dev = &pdev->dev;
 | 
				
			||||||
 | 
					+	struct opp_table *table = dev_pm_opp_get_opp_table(dev);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	dev_pm_opp_remove_table(dev);
 | 
				
			||||||
 | 
					+	dev_pm_opp_put_regulators(table);
 | 
				
			||||||
 | 
					+	dev_pm_opp_unregister_set_opp_helper(table);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	return 0;
 | 
				
			||||||
 | 
					+};
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static const struct of_device_id krait_cache_match_table[] = {
 | 
				
			||||||
 | 
					+	{ .compatible = "qcom,krait-cache" },
 | 
				
			||||||
 | 
					+	{}
 | 
				
			||||||
 | 
					+};
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static struct platform_driver krait_cache_driver = {
 | 
				
			||||||
 | 
					+	.driver = {
 | 
				
			||||||
 | 
					+		.name	= "krait-cache",
 | 
				
			||||||
 | 
					+		.of_match_table = krait_cache_match_table,
 | 
				
			||||||
 | 
					+	},
 | 
				
			||||||
 | 
					+	.probe		= krait_cache_probe,
 | 
				
			||||||
 | 
					+	.remove		= krait_cache_remove,
 | 
				
			||||||
 | 
					+};
 | 
				
			||||||
 | 
					+module_platform_driver(krait_cache_driver);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static int krait_cpufreq_probe(struct platform_device *pdev)
 | 
				
			||||||
 | 
					+{
 | 
				
			||||||
 | 
					+	struct cpufreq_dt_platform_data *data = dev_get_platdata(&pdev->dev);
 | 
				
			||||||
 | 
					+	int ret;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	/*
 | 
				
			||||||
 | 
					+	 * All per-cluster (CPUs sharing clock/voltages) initialization is done
 | 
				
			||||||
 | 
					+	 * from ->init(). In probe(), we just need to make sure that clk and
 | 
				
			||||||
 | 
					+	 * regulators are available. Else defer probe and retry.
 | 
				
			||||||
 | 
					+	 *
 | 
				
			||||||
 | 
					+	 * FIXME: Is checking this only for CPU0 sufficient ?
 | 
				
			||||||
 | 
					+	 */
 | 
				
			||||||
 | 
					+	ret = resources_available();
 | 
				
			||||||
 | 
					+	if (ret)
 | 
				
			||||||
 | 
					+		return ret;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	if (data) {
 | 
				
			||||||
 | 
					+		if (data->have_governor_per_policy)
 | 
				
			||||||
 | 
					+			krait_cpufreq_driver.flags |=
 | 
				
			||||||
 | 
					+				CPUFREQ_HAVE_GOVERNOR_PER_POLICY;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+		krait_cpufreq_driver.resume = data->resume;
 | 
				
			||||||
 | 
					+		if (data->suspend)
 | 
				
			||||||
 | 
					+			krait_cpufreq_driver.suspend = data->suspend;
 | 
				
			||||||
 | 
					+	}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	ret = cpufreq_register_driver(&krait_cpufreq_driver);
 | 
				
			||||||
 | 
					+	if (ret)
 | 
				
			||||||
 | 
					+		dev_err(&pdev->dev, "failed register driver: %d\n", ret);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+	return ret;
 | 
				
			||||||
 | 
					+}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static int krait_cpufreq_remove(struct platform_device *pdev)
 | 
				
			||||||
 | 
					+{
 | 
				
			||||||
 | 
					+	cpufreq_unregister_driver(&krait_cpufreq_driver);
 | 
				
			||||||
 | 
					+	return 0;
 | 
				
			||||||
 | 
					+}
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+static struct platform_driver krait_cpufreq_platdrv = {
 | 
				
			||||||
 | 
					+	.driver = {
 | 
				
			||||||
 | 
					+		.name	= "krait-cpufreq",
 | 
				
			||||||
 | 
					+	},
 | 
				
			||||||
 | 
					+	.probe		= krait_cpufreq_probe,
 | 
				
			||||||
 | 
					+	.remove		= krait_cpufreq_remove,
 | 
				
			||||||
 | 
					+};
 | 
				
			||||||
 | 
					+module_platform_driver(krait_cpufreq_platdrv);
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+MODULE_ALIAS("platform:krait-cpufreq");
 | 
				
			||||||
 | 
					+MODULE_AUTHOR("Ansuel Smith <ansuelsmth@gmail.com>");
 | 
				
			||||||
 | 
					+MODULE_DESCRIPTION("Dedicated Krait SoC cpufreq driver");
 | 
				
			||||||
 | 
					+MODULE_LICENSE("GPL");
 | 
				
			||||||
@ -0,0 +1,243 @@
 | 
				
			|||||||
 | 
					From c9ecd920324a647bf1f2b47f771c8f599cc7b551 Mon Sep 17 00:00:00 2001
 | 
				
			||||||
 | 
					From: Ansuel Smith <ansuelsmth@gmail.com>
 | 
				
			||||||
 | 
					Date: Sat, 22 Feb 2020 18:02:17 +0100
 | 
				
			||||||
 | 
					Subject: [PATCH 2/8] Documentation: cpufreq: add qcom,krait-cache bindings
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Document dedicated cpufreq for Krait CPUs.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Signed-off-by: Ansuel Smith <ansuelsmth@gmail.com>
 | 
				
			||||||
 | 
					---
 | 
				
			||||||
 | 
					 .../bindings/cpufreq/qcom-cpufreq-krait.yaml  | 221 ++++++++++++++++++
 | 
				
			||||||
 | 
					 1 file changed, 221 insertions(+)
 | 
				
			||||||
 | 
					 create mode 100644 Documentation/devicetree/bindings/cpufreq/qcom-cpufreq-krait.yaml
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					diff --git a/Documentation/devicetree/bindings/cpufreq/qcom-cpufreq-krait.yaml b/Documentation/devicetree/bindings/cpufreq/qcom-cpufreq-krait.yaml
 | 
				
			||||||
 | 
					new file mode 100644
 | 
				
			||||||
 | 
					index 000000000000..f6bcca863d9a
 | 
				
			||||||
 | 
					--- /dev/null
 | 
				
			||||||
 | 
					+++ b/Documentation/devicetree/bindings/cpufreq/qcom-cpufreq-krait.yaml
 | 
				
			||||||
 | 
					@@ -0,0 +1,221 @@
 | 
				
			||||||
 | 
					+# SPDX-License-Identifier: GPL-2.0
 | 
				
			||||||
 | 
					+%YAML 1.2
 | 
				
			||||||
 | 
					+---
 | 
				
			||||||
 | 
					+$id: http://devicetree.org/schemas/cpufreq/qcom-cpufreq-krait.yaml#
 | 
				
			||||||
 | 
					+$schema: http://devicetree.org/meta-schemas/core.yaml#
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+title: CPU Frequency scaling driver for Krait SoCs
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+maintainers:
 | 
				
			||||||
 | 
					+  - Ansuel Smith <ansuelsmth@gmail.com>
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+description: |
 | 
				
			||||||
 | 
					+  The krait cpufreq driver is a dedicated frequency scaling driver
 | 
				
			||||||
 | 
					+  based on cpufreq-dt generic driver that scale L2 cache and the
 | 
				
			||||||
 | 
					+  cores. TEST
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+  The L2 cache is scaled based on the max clk across all cores and
 | 
				
			||||||
 | 
					+  the clock is decided based on the opp-level set in the device tree.
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+  Different core freq can be linked to a specific l2 freq and the driver
 | 
				
			||||||
 | 
					+  on frequency change will scale the core and the l2 clk based of the 
 | 
				
			||||||
 | 
					+  linked freq.
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+  On Krait SoC is present a bug and on every L2 clk change the driver
 | 
				
			||||||
 | 
					+  needs to set the clk to the idle freq before changing it to the new value.
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+  This requires the qcom cpufreq nvmem driver to parse the different opp
 | 
				
			||||||
 | 
					+  core clk and an additional opp table for the l2 scaling.
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+  If the driver detect broken config (for example missing opp-level) the
 | 
				
			||||||
 | 
					+  cpufreq driver skips the l2 scaling
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+  Referring to this example opp-level can be used to link a range of cpu freq
 | 
				
			||||||
 | 
					+  to a specific l2 freq:
 | 
				
			||||||
 | 
					+    cpu opp freq 384000000 has opp-level 0
 | 
				
			||||||
 | 
					+    l2 opp freq 384000000 has opp-level 0
 | 
				
			||||||
 | 
					+    The driver will scale l2 to 384000000
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+    cpu opp freq 600000000-1000000000 has opp-level 1
 | 
				
			||||||
 | 
					+    l2 opp freq 1000000000 has opp-level 1
 | 
				
			||||||
 | 
					+    The driver will scale l2 to 1000000000
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+allOf:
 | 
				
			||||||
 | 
					+  - $ref: /schemas/cache-controller.yaml#
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+select:
 | 
				
			||||||
 | 
					+  properties:
 | 
				
			||||||
 | 
					+    compatible:
 | 
				
			||||||
 | 
					+      items:
 | 
				
			||||||
 | 
					+        - enum:
 | 
				
			||||||
 | 
					+            - qcom,krait-cache
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+  required:
 | 
				
			||||||
 | 
					+    - compatible
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+properties:
 | 
				
			||||||
 | 
					+  compatible:
 | 
				
			||||||
 | 
					+    items:
 | 
				
			||||||
 | 
					+      - const: qcom,krait-cache
 | 
				
			||||||
 | 
					+      - const: cache
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+  cache-level:
 | 
				
			||||||
 | 
					+    const: 2
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+  clocks:
 | 
				
			||||||
 | 
					+    maxItems: 1
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+  clock-names:
 | 
				
			||||||
 | 
					+    const: l2
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+  l2-supply: true
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+  operating-points-v2: true
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+required:
 | 
				
			||||||
 | 
					+  - compatible
 | 
				
			||||||
 | 
					+  - cache-level
 | 
				
			||||||
 | 
					+  - clocks
 | 
				
			||||||
 | 
					+  - clock-names
 | 
				
			||||||
 | 
					+  - l2-supply
 | 
				
			||||||
 | 
					+  - operating-points-v2
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+additionalProperties: false
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+examples:
 | 
				
			||||||
 | 
					+  - |
 | 
				
			||||||
 | 
					+    cpus {
 | 
				
			||||||
 | 
					+      #address-cells = <1>;
 | 
				
			||||||
 | 
					+      #size-cells = <0>;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+      cpu0: cpu@0 {
 | 
				
			||||||
 | 
					+        compatible = "qcom,krait";
 | 
				
			||||||
 | 
					+        enable-method = "qcom,kpss-acc-v1";
 | 
				
			||||||
 | 
					+        device_type = "cpu";
 | 
				
			||||||
 | 
					+        reg = <0>;
 | 
				
			||||||
 | 
					+        next-level-cache = <&L2>;
 | 
				
			||||||
 | 
					+        qcom,acc = <&acc0>;
 | 
				
			||||||
 | 
					+        qcom,saw = <&saw0>;
 | 
				
			||||||
 | 
					+        clocks = <&kraitcc 0>, <&kraitcc 4>;
 | 
				
			||||||
 | 
					+        clock-names = "cpu", "l2";
 | 
				
			||||||
 | 
					+        clock-latency = <100000>;
 | 
				
			||||||
 | 
					+        cpu-supply = <&smb208_s2a>;
 | 
				
			||||||
 | 
					+        operating-points-v2 = <&opp_table0>;
 | 
				
			||||||
 | 
					+        voltage-tolerance = <5>;
 | 
				
			||||||
 | 
					+        cooling-min-state = <0>;
 | 
				
			||||||
 | 
					+        cooling-max-state = <10>;
 | 
				
			||||||
 | 
					+        #cooling-cells = <2>;
 | 
				
			||||||
 | 
					+        cpu-idle-states = <&CPU_SPC>;
 | 
				
			||||||
 | 
					+      };
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+      /* ... */
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+    };
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+    opp_table0: opp_table0 {
 | 
				
			||||||
 | 
					+      compatible = "operating-points-v2-kryo-cpu";
 | 
				
			||||||
 | 
					+      nvmem-cells = <&speedbin_efuse>;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+      opp-384000000 {
 | 
				
			||||||
 | 
					+        opp-hz = /bits/ 64 <384000000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs0-v0 = <1000000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs1-v0 = <925000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs2-v0 = <875000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs3-v0 = <800000>;
 | 
				
			||||||
 | 
					+        opp-supported-hw = <0x1>;
 | 
				
			||||||
 | 
					+        clock-latency-ns = <100000>;
 | 
				
			||||||
 | 
					+        opp-level = <0>;
 | 
				
			||||||
 | 
					+      };
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+      opp-600000000 {
 | 
				
			||||||
 | 
					+        opp-hz = /bits/ 64 <600000000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs0-v0 = <1050000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs1-v0 = <975000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs2-v0 = <925000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs3-v0 = <850000>;
 | 
				
			||||||
 | 
					+        opp-supported-hw = <0x1>;
 | 
				
			||||||
 | 
					+        clock-latency-ns = <100000>;
 | 
				
			||||||
 | 
					+        opp-level = <1>;
 | 
				
			||||||
 | 
					+      };
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+      opp-800000000 {
 | 
				
			||||||
 | 
					+        opp-hz = /bits/ 64 <800000000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs0-v0 = <1100000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs1-v0 = <1025000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs2-v0 = <995000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs3-v0 = <900000>;
 | 
				
			||||||
 | 
					+        opp-supported-hw = <0x1>;
 | 
				
			||||||
 | 
					+        clock-latency-ns = <100000>;
 | 
				
			||||||
 | 
					+        opp-level = <1>;
 | 
				
			||||||
 | 
					+      };
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+      opp-1000000000 {
 | 
				
			||||||
 | 
					+        opp-hz = /bits/ 64 <1000000000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs0-v0 = <1150000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs1-v0 = <1075000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs2-v0 = <1025000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs3-v0 = <950000>;
 | 
				
			||||||
 | 
					+        opp-supported-hw = <0x1>;
 | 
				
			||||||
 | 
					+        clock-latency-ns = <100000>;
 | 
				
			||||||
 | 
					+        opp-level = <1>;
 | 
				
			||||||
 | 
					+      };
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+      opp-1200000000 {
 | 
				
			||||||
 | 
					+        opp-hz = /bits/ 64 <1200000000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs0-v0 = <1200000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs1-v0 = <1125000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs2-v0 = <1075000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs3-v0 = <1000000>;
 | 
				
			||||||
 | 
					+        opp-supported-hw = <0x1>;
 | 
				
			||||||
 | 
					+        clock-latency-ns = <100000>;
 | 
				
			||||||
 | 
					+        opp-level = <2>;
 | 
				
			||||||
 | 
					+      };
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+      opp-1400000000 {
 | 
				
			||||||
 | 
					+        opp-hz = /bits/ 64 <1400000000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs0-v0 = <1250000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs1-v0 = <1175000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs2-v0 = <1125000>;
 | 
				
			||||||
 | 
					+        opp-microvolt-speed0-pvs3-v0 = <1050000>;
 | 
				
			||||||
 | 
					+        opp-supported-hw = <0x1>;
 | 
				
			||||||
 | 
					+        clock-latency-ns = <100000>;
 | 
				
			||||||
 | 
					+        opp-level = <2>;
 | 
				
			||||||
 | 
					+      };
 | 
				
			||||||
 | 
					+    };
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+    opp_table_l2: opp_table_l2 {
 | 
				
			||||||
 | 
					+      compatible = "operating-points-v2";
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+      opp-384000000 {
 | 
				
			||||||
 | 
					+        opp-hz = /bits/ 64 <384000000>;
 | 
				
			||||||
 | 
					+        opp-microvolt = <1100000>;
 | 
				
			||||||
 | 
					+        clock-latency-ns = <100000>;
 | 
				
			||||||
 | 
					+        opp-level = <0>;
 | 
				
			||||||
 | 
					+      };
 | 
				
			||||||
 | 
					+      opp-1000000000 {
 | 
				
			||||||
 | 
					+        opp-hz = /bits/ 64 <1000000000>;
 | 
				
			||||||
 | 
					+        opp-microvolt = <1100000>;
 | 
				
			||||||
 | 
					+        clock-latency-ns = <100000>;
 | 
				
			||||||
 | 
					+        opp-level = <1>;
 | 
				
			||||||
 | 
					+      };
 | 
				
			||||||
 | 
					+      opp-1200000000 {
 | 
				
			||||||
 | 
					+        opp-hz = /bits/ 64 <1200000000>;
 | 
				
			||||||
 | 
					+        opp-microvolt = <1150000>;
 | 
				
			||||||
 | 
					+        clock-latency-ns = <100000>;
 | 
				
			||||||
 | 
					+        opp-level = <2>;
 | 
				
			||||||
 | 
					+      };
 | 
				
			||||||
 | 
					+    };
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+    soc {
 | 
				
			||||||
 | 
					+      L2: l2-cache {
 | 
				
			||||||
 | 
					+        compatible = "qcom,krait-cache", "cache";
 | 
				
			||||||
 | 
					+        cache-level = <2>;
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+        clocks = <&kraitcc 4>;
 | 
				
			||||||
 | 
					+        clock-names = "l2";
 | 
				
			||||||
 | 
					+        l2-supply = <&smb208_s1a>;
 | 
				
			||||||
 | 
					+        operating-points-v2 = <&opp_table_l2>;
 | 
				
			||||||
 | 
					+      };
 | 
				
			||||||
 | 
					+    };
 | 
				
			||||||
 | 
					+
 | 
				
			||||||
 | 
					+...
 | 
				
			||||||
 | 
					-- 
 | 
				
			||||||
 | 
					2.29.2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -217,9 +217,9 @@
 | 
				
			|||||||
+int scale_fabrics(unsigned long max_cpu_freq);
 | 
					+int scale_fabrics(unsigned long max_cpu_freq);
 | 
				
			||||||
+
 | 
					+
 | 
				
			||||||
+#endif
 | 
					+#endif
 | 
				
			||||||
--- a/drivers/cpufreq/cpufreq-dt.c
 | 
					--- a/drivers/cpufreq/qcom-cpufreq-krait.c
 | 
				
			||||||
+++ b/drivers/cpufreq/cpufreq-dt.c
 | 
					+++ b/drivers/cpufreq/qcom-cpufreq-krait.c
 | 
				
			||||||
@@ -20,6 +20,7 @@
 | 
					@@ -15,6 +15,7 @@
 | 
				
			||||||
 #include <linux/regulator/consumer.h>
 | 
					 #include <linux/regulator/consumer.h>
 | 
				
			||||||
 #include <linux/slab.h>
 | 
					 #include <linux/slab.h>
 | 
				
			||||||
 #include <linux/thermal.h>
 | 
					 #include <linux/thermal.h>
 | 
				
			||||||
@ -227,17 +227,17 @@
 | 
				
			|||||||
 
 | 
					 
 | 
				
			||||||
 #include "cpufreq-dt.h"
 | 
					 #include "cpufreq-dt.h"
 | 
				
			||||||
 
 | 
					 
 | 
				
			||||||
@@ -111,6 +112,13 @@ static int set_target(struct cpufreq_pol
 | 
					@@ -58,6 +59,13 @@ static int set_target(struct cpufreq_policy *policy, unsigned int index)
 | 
				
			||||||
 					}
 | 
					 		level = dev_pm_opp_get_level(opp);
 | 
				
			||||||
 				}
 | 
					 		dev_pm_opp_put(opp);
 | 
				
			||||||
 			}
 | 
					 
 | 
				
			||||||
+
 | 
					 | 
				
			||||||
+		/*
 | 
					+		/*
 | 
				
			||||||
+		 * Scale fabrics with max freq across all cores
 | 
					+		 * Scale fabrics with max freq across all cores
 | 
				
			||||||
+		 */
 | 
					+		 */
 | 
				
			||||||
+		ret = scale_fabrics(target_freq);
 | 
					+		ret = scale_fabrics(target_freq);
 | 
				
			||||||
+		if (ret)
 | 
					+		if (ret)
 | 
				
			||||||
+				goto exit;
 | 
					+			return ret;
 | 
				
			||||||
 		}
 | 
					+
 | 
				
			||||||
 		priv->opp_freq = freq * 1000;
 | 
					 		opp = dev_pm_opp_find_level_exact(&l2_pdev->dev, level);
 | 
				
			||||||
 		arch_set_freq_scale(policy->related_cpus, freq,
 | 
					 		if (IS_ERR(opp)) {
 | 
				
			||||||
 | 
					 			dev_err(&l2_pdev->dev,
 | 
				
			||||||
		Loading…
	
		Reference in New Issue
	
	Block a user