
On Wed, 6 Mar 2024 at 06:23, Volodymyr Babchuk Volodymyr_Babchuk@epam.com wrote:
Now sub-drivers for particular SoCs can register them as power domain drivers. This is needed for upcoming SM8150 support, because it needs to power up the Ethernet module.
Signed-off-by: Volodymyr Babchuk volodymyr_babchuk@epam.com
Changes in v2:
- Reworked qcom_cc_bind() function
- Added timeout to qcom_power_set()
- Minor fixes in register names and formatting
drivers/clk/qcom/clock-qcom.c | 128 ++++++++++++++++++++++++++++++---- drivers/clk/qcom/clock-qcom.h | 6 ++ 2 files changed, 121 insertions(+), 13 deletions(-)
diff --git a/drivers/clk/qcom/clock-qcom.c b/drivers/clk/qcom/clock-qcom.c index 729d190c54..c3f8d96183 100644 --- a/drivers/clk/qcom/clock-qcom.c +++ b/drivers/clk/qcom/clock-qcom.c @@ -23,6 +23,7 @@ #include <linux/delay.h> #include <linux/bitops.h> #include <reset-uclass.h> +#include <power-domain-uclass.h>
#include "clock-qcom.h"
@@ -30,6 +31,13 @@ #define CBCR_BRANCH_ENABLE_BIT BIT(0) #define CBCR_BRANCH_OFF_BIT BIT(31)
+#define GDSC_SW_COLLAPSE_MASK BIT(0) +#define GDSC_POWER_DOWN_COMPLETE BIT(15) +#define GDSC_POWER_UP_COMPLETE BIT(16) +#define GDSC_PWR_ON_MASK BIT(31) +#define CFG_GDSCR_OFFSET 0x4 +#define GDSC_STATUS_POLL_TIMEOUT_US 1500
/* Enable clock controlled by CBC soft macro */ void clk_enable_cbc(phys_addr_t cbcr) { @@ -223,7 +231,7 @@ U_BOOT_DRIVER(qcom_clk) = { int qcom_cc_bind(struct udevice *parent) { struct msm_clk_data *data = (struct msm_clk_data *)dev_get_driver_data(parent);
struct udevice *clkdev, *rstdev;
struct udevice *clkdev = NULL, *rstdev = NULL, *pwrdev; struct driver *drv; int ret;
@@ -238,20 +246,41 @@ int qcom_cc_bind(struct udevice *parent) if (ret) return ret;
/* Bail out early if resets are not specified for this platform */
if (!data->resets)
return ret;
if (data->resets) {
/* Get a handle to the common reset handler */
drv = lists_driver_lookup_name("qcom_reset");
if (!drv) {
ret = -ENOENT;
goto unbind_clkdev;
}
/* Register the reset controller */
ret = device_bind_with_driver_data(parent, drv, "qcom_reset", (ulong)data,
dev_ofnode(parent), &rstdev);
if (ret)
goto unbind_clkdev;
}
/* Get a handle to the common reset handler */
drv = lists_driver_lookup_name("qcom_reset");
if (!drv)
return -ENOENT;
if (data->power_domains) {
/* Get a handle to the common power domain handler */
drv = lists_driver_lookup_name("qcom_power");
if (!drv) {
ret = -ENOENT;
goto unbind_rstdev;
}
/* Register the power domain controller */
ret = device_bind_with_driver_data(parent, drv, "qcom_power", (ulong)data,
dev_ofnode(parent), &pwrdev);
if (ret)
goto unbind_rstdev;
}
/* Register the reset controller */
ret = device_bind_with_driver_data(parent, drv, "qcom_reset", (ulong)data,
dev_ofnode(parent), &rstdev);
if (ret)
device_unbind(clkdev);
return 0;
+unbind_rstdev:
device_unbind(rstdev);
+unbind_clkdev:
device_unbind(clkdev); return ret;
} @@ -306,3 +335,76 @@ U_BOOT_DRIVER(qcom_reset) = { .ops = &qcom_reset_ops, .probe = qcom_reset_probe, };
+static int qcom_power_set(struct power_domain *pwr, bool on) +{
struct msm_clk_data *data = (struct msm_clk_data *)dev_get_driver_data(pwr->dev);
void __iomem *base = dev_get_priv(pwr->dev);
unsigned long timeout;
const struct qcom_power_map *map;
u32 value;
if (pwr->id >= data->num_power_domains)
return -ENODEV;
map = &data->power_domains[pwr->id];
if (!map->reg)
return -ENODEV;
value = readl(base + map->reg);
if (on)
value &= ~GDSC_SW_COLLAPSE_MASK;
else
value |= GDSC_SW_COLLAPSE_MASK;
writel(value, base + map->reg);
timeout = timer_get_us() + GDSC_STATUS_POLL_TIMEOUT_US;
/* Wait for power on */
while (timeout > timer_get_us()) {
value = readl(base + map->reg + CFG_GDSCR_OFFSET);
You should be able to reuse readl_poll_timeout() here instead.
-Sumit
if (on) {
if ((value & GDSC_POWER_UP_COMPLETE) ||
(value & GDSC_PWR_ON_MASK))
return 0;
} else {
if (value & GDSC_POWER_DOWN_COMPLETE ||
!(value & GDSC_PWR_ON_MASK))
return 0;
}
}
return -ETIMEDOUT;
+}
+static int qcom_power_on(struct power_domain *pwr) +{
return qcom_power_set(pwr, true);
+}
+static int qcom_power_off(struct power_domain *pwr) +{
return qcom_power_set(pwr, false);
+}
+static const struct power_domain_ops qcom_power_ops = {
.on = qcom_power_on,
.off = qcom_power_off,
+};
+static int qcom_power_probe(struct udevice *dev) +{
/* Set our priv pointer to the base address */
dev_set_priv(dev, (void *)dev_read_addr(dev));
return 0;
+}
+U_BOOT_DRIVER(qcom_power) = {
.name = "qcom_power",
.id = UCLASS_POWER_DOMAIN,
.ops = &qcom_power_ops,
.probe = qcom_power_probe,
+}; diff --git a/drivers/clk/qcom/clock-qcom.h b/drivers/clk/qcom/clock-qcom.h index 01088c1901..12a1eaec2b 100644 --- a/drivers/clk/qcom/clock-qcom.h +++ b/drivers/clk/qcom/clock-qcom.h @@ -59,9 +59,15 @@ struct qcom_reset_map { u8 bit; };
+struct qcom_power_map {
unsigned int reg;
+};
struct clk;
struct msm_clk_data {
const struct qcom_power_map *power_domains;
unsigned long num_power_domains; const struct qcom_reset_map *resets; unsigned long num_resets; const struct gate_clk *clks;
-- 2.43.0