Commit 2f2df895 authored by Nicolas Pitre's avatar Nicolas Pitre

ARM: vexpress/dcscb: do not hardcode number of CPUs per cluster

If 4 CPUs are assumed, the A15x1-A7x1 model configuration would never
shut down the initial cluster as the 0xf reset bit mask will never be
observed.  Let's construct this mask based on the provided information
in the DCSCB config register for the number of CPUs per cluster.
Signed-off-by: default avatarNicolas Pitre <nico@linaro.org>
Reviewed-by: default avatarSantosh Shilimkar <santosh.shilimkar@ti.com>
Acked-by: default avatarPawel Moll <pawel.moll@arm.com>
parent 13eae144
...@@ -45,10 +45,12 @@ static arch_spinlock_t dcscb_lock = __ARCH_SPIN_LOCK_UNLOCKED; ...@@ -45,10 +45,12 @@ static arch_spinlock_t dcscb_lock = __ARCH_SPIN_LOCK_UNLOCKED;
static void __iomem *dcscb_base; static void __iomem *dcscb_base;
static int dcscb_use_count[4][2]; static int dcscb_use_count[4][2];
static int dcscb_allcpus_mask[2];
static int dcscb_power_up(unsigned int cpu, unsigned int cluster) static int dcscb_power_up(unsigned int cpu, unsigned int cluster)
{ {
unsigned int rst_hold, cpumask = (1 << cpu); unsigned int rst_hold, cpumask = (1 << cpu);
unsigned int all_mask = dcscb_allcpus_mask[cluster];
pr_debug("%s: cpu %u cluster %u\n", __func__, cpu, cluster); pr_debug("%s: cpu %u cluster %u\n", __func__, cpu, cluster);
if (cpu >= 4 || cluster >= 2) if (cpu >= 4 || cluster >= 2)
...@@ -67,7 +69,7 @@ static int dcscb_power_up(unsigned int cpu, unsigned int cluster) ...@@ -67,7 +69,7 @@ static int dcscb_power_up(unsigned int cpu, unsigned int cluster)
if (rst_hold & (1 << 8)) { if (rst_hold & (1 << 8)) {
/* remove cluster reset and add individual CPU's reset */ /* remove cluster reset and add individual CPU's reset */
rst_hold &= ~(1 << 8); rst_hold &= ~(1 << 8);
rst_hold |= 0xf; rst_hold |= all_mask;
} }
rst_hold &= ~(cpumask | (cpumask << 4)); rst_hold &= ~(cpumask | (cpumask << 4));
writel_relaxed(rst_hold, dcscb_base + RST_HOLD0 + cluster * 4); writel_relaxed(rst_hold, dcscb_base + RST_HOLD0 + cluster * 4);
...@@ -91,13 +93,14 @@ static int dcscb_power_up(unsigned int cpu, unsigned int cluster) ...@@ -91,13 +93,14 @@ static int dcscb_power_up(unsigned int cpu, unsigned int cluster)
static void dcscb_power_down(void) static void dcscb_power_down(void)
{ {
unsigned int mpidr, cpu, cluster, rst_hold, cpumask; unsigned int mpidr, cpu, cluster, rst_hold, cpumask, all_mask;
bool last_man = false, skip_wfi = false; bool last_man = false, skip_wfi = false;
mpidr = read_cpuid_mpidr(); mpidr = read_cpuid_mpidr();
cpu = MPIDR_AFFINITY_LEVEL(mpidr, 0); cpu = MPIDR_AFFINITY_LEVEL(mpidr, 0);
cluster = MPIDR_AFFINITY_LEVEL(mpidr, 1); cluster = MPIDR_AFFINITY_LEVEL(mpidr, 1);
cpumask = (1 << cpu); cpumask = (1 << cpu);
all_mask = dcscb_allcpus_mask[cluster];
pr_debug("%s: cpu %u cluster %u\n", __func__, cpu, cluster); pr_debug("%s: cpu %u cluster %u\n", __func__, cpu, cluster);
BUG_ON(cpu >= 4 || cluster >= 2); BUG_ON(cpu >= 4 || cluster >= 2);
...@@ -107,7 +110,7 @@ static void dcscb_power_down(void) ...@@ -107,7 +110,7 @@ static void dcscb_power_down(void)
if (dcscb_use_count[cpu][cluster] == 0) { if (dcscb_use_count[cpu][cluster] == 0) {
rst_hold = readl_relaxed(dcscb_base + RST_HOLD0 + cluster * 4); rst_hold = readl_relaxed(dcscb_base + RST_HOLD0 + cluster * 4);
rst_hold |= cpumask; rst_hold |= cpumask;
if (((rst_hold | (rst_hold >> 4)) & 0xf) == 0xf) { if (((rst_hold | (rst_hold >> 4)) & all_mask) == all_mask) {
rst_hold |= (1 << 8); rst_hold |= (1 << 8);
last_man = true; last_man = true;
} }
...@@ -177,6 +180,7 @@ static void __init dcscb_usage_count_init(void) ...@@ -177,6 +180,7 @@ static void __init dcscb_usage_count_init(void)
static int __init dcscb_init(void) static int __init dcscb_init(void)
{ {
struct device_node *node; struct device_node *node;
unsigned int cfg;
int ret; int ret;
node = of_find_compatible_node(NULL, NULL, "arm,rtsm,dcscb"); node = of_find_compatible_node(NULL, NULL, "arm,rtsm,dcscb");
...@@ -185,7 +189,9 @@ static int __init dcscb_init(void) ...@@ -185,7 +189,9 @@ static int __init dcscb_init(void)
dcscb_base = of_iomap(node, 0); dcscb_base = of_iomap(node, 0);
if (!dcscb_base) if (!dcscb_base)
return -EADDRNOTAVAIL; return -EADDRNOTAVAIL;
cfg = readl_relaxed(dcscb_base + DCS_CFG_R);
dcscb_allcpus_mask[0] = (1 << (((cfg >> 16) >> (0 << 2)) & 0xf)) - 1;
dcscb_allcpus_mask[1] = (1 << (((cfg >> 16) >> (1 << 2)) & 0xf)) - 1;
dcscb_usage_count_init(); dcscb_usage_count_init();
ret = mcpm_platform_register(&dcscb_power_ops); ret = mcpm_platform_register(&dcscb_power_ops);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment