blob: f70f8e024239528eb627d9202f6a31516079ad91 [file] [log] [blame]
// SPDX-License-Identifier: GPL-2.0+
/*
* Copyright (c) 2023 SberDevices, Inc.
* Author: Alexey Romanov <avromanov@sberdevices.ru>
*/
#include <dm.h>
#include <asm/arch/sm.h>
#include <power-domain.h>
#include <power-domain-uclass.h>
#include <dt-bindings/power/meson-a1-power.h>
struct meson_secure_pwrc_domain_desc {
char *name;
size_t index;
};
struct meson_secure_pwrc_domain_data {
unsigned int count;
struct meson_secure_pwrc_domain_desc *domains;
};
struct meson_secure_pwrc_priv {
const struct meson_secure_pwrc_domain_data *data;
};
static int meson_secure_pwrc_on(struct power_domain *power_domain)
{
struct meson_secure_pwrc_priv *priv = dev_get_priv(power_domain->dev);
struct meson_secure_pwrc_domain_desc *pwrc_domain;
int err;
pwrc_domain = &priv->data->domains[power_domain->id];
err = meson_sm_pwrdm_on(pwrc_domain->index);
if (err) {
pr_err("meson_sm_pwrdm_on() failed (%d)\n", err);
return err;
}
pr_debug("enable %s power domain\n", pwrc_domain->name);
return 0;
}
static int meson_secure_pwrc_off(struct power_domain *power_domain)
{
struct meson_secure_pwrc_priv *priv = dev_get_priv(power_domain->dev);
struct meson_secure_pwrc_domain_desc *pwrc_domain;
int err;
pwrc_domain = &priv->data->domains[power_domain->id];
err = meson_sm_pwrdm_off(pwrc_domain->index);
if (err) {
pr_err("meson_sm_pwrdm_off() failed (%d)\n", err);
return err;
}
pr_debug("disable %s power domain\n", pwrc_domain->name);
return 0;
}
static int meson_secure_pwrc_of_xlate(struct power_domain *power_domain,
struct ofnode_phandle_args *args)
{
struct meson_secure_pwrc_priv *priv = dev_get_priv(power_domain->dev);
struct meson_secure_pwrc_domain_desc *pwrc_domain;
if (args->args_count < 1) {
pr_err("invalid args count: %d\n", args->args_count);
return -EINVAL;
}
power_domain->id = args->args[0];
if (power_domain->id >= priv->data->count) {
pr_err("domain with ID=%lu is invalid\n", power_domain->id);
return -EINVAL;
}
pwrc_domain = &priv->data->domains[power_domain->id];
if (!pwrc_domain->name) {
pr_err("domain with ID=%lu is invalid\n", power_domain->id);
return -EINVAL;
}
return 0;
}
#define SEC_PD(__name) \
[PWRC_##__name##_ID] = \
{ \
.name = #__name, \
.index = PWRC_##__name##_ID, \
}
static struct meson_secure_pwrc_domain_desc a1_pwrc_domains[] = {
SEC_PD(DSPA),
SEC_PD(DSPB),
SEC_PD(UART),
SEC_PD(DMC),
SEC_PD(I2C),
SEC_PD(PSRAM),
SEC_PD(ACODEC),
SEC_PD(AUDIO),
SEC_PD(OTP),
SEC_PD(DMA),
SEC_PD(SD_EMMC),
SEC_PD(RAMA),
SEC_PD(RAMB),
SEC_PD(IR),
SEC_PD(SPICC),
SEC_PD(SPIFC),
SEC_PD(USB),
SEC_PD(NIC),
SEC_PD(PDMIN),
SEC_PD(RSA),
};
struct power_domain_ops meson_secure_pwrc_ops = {
.on = meson_secure_pwrc_on,
.off = meson_secure_pwrc_off,
.of_xlate = meson_secure_pwrc_of_xlate,
};
static struct meson_secure_pwrc_domain_data meson_secure_a1_pwrc_data = {
.count = ARRAY_SIZE(a1_pwrc_domains),
.domains = a1_pwrc_domains,
};
static const struct udevice_id meson_secure_pwrc_ids[] = {
{
.compatible = "amlogic,meson-a1-pwrc",
.data = (unsigned long)&meson_secure_a1_pwrc_data,
},
{ }
};
static int meson_secure_pwrc_probe(struct udevice *dev)
{
struct meson_secure_pwrc_priv *priv = dev_get_priv(dev);
priv->data = (void *)dev_get_driver_data(dev);
if (!priv->data)
return -EINVAL;
return 0;
}
U_BOOT_DRIVER(meson_secure_pwrc) = {
.name = "meson_secure_pwrc",
.id = UCLASS_POWER_DOMAIN,
.of_match = meson_secure_pwrc_ids,
.probe = meson_secure_pwrc_probe,
.ops = &meson_secure_pwrc_ops,
.priv_auto = sizeof(struct meson_secure_pwrc_priv),
};