162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Marvell Dove PMU support
462306a36Sopenharmony_ci */
562306a36Sopenharmony_ci#include <linux/io.h>
662306a36Sopenharmony_ci#include <linux/irq.h>
762306a36Sopenharmony_ci#include <linux/irqdomain.h>
862306a36Sopenharmony_ci#include <linux/of.h>
962306a36Sopenharmony_ci#include <linux/of_irq.h>
1062306a36Sopenharmony_ci#include <linux/of_address.h>
1162306a36Sopenharmony_ci#include <linux/platform_device.h>
1262306a36Sopenharmony_ci#include <linux/pm_domain.h>
1362306a36Sopenharmony_ci#include <linux/reset.h>
1462306a36Sopenharmony_ci#include <linux/reset-controller.h>
1562306a36Sopenharmony_ci#include <linux/sched.h>
1662306a36Sopenharmony_ci#include <linux/slab.h>
1762306a36Sopenharmony_ci#include <linux/soc/dove/pmu.h>
1862306a36Sopenharmony_ci#include <linux/spinlock.h>
1962306a36Sopenharmony_ci
2062306a36Sopenharmony_ci#define NR_PMU_IRQS		7
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_ci#define PMC_SW_RST		0x30
2362306a36Sopenharmony_ci#define PMC_IRQ_CAUSE		0x50
2462306a36Sopenharmony_ci#define PMC_IRQ_MASK		0x54
2562306a36Sopenharmony_ci
2662306a36Sopenharmony_ci#define PMU_PWR			0x10
2762306a36Sopenharmony_ci#define PMU_ISO			0x58
2862306a36Sopenharmony_ci
2962306a36Sopenharmony_cistruct pmu_data {
3062306a36Sopenharmony_ci	spinlock_t lock;
3162306a36Sopenharmony_ci	struct device_node *of_node;
3262306a36Sopenharmony_ci	void __iomem *pmc_base;
3362306a36Sopenharmony_ci	void __iomem *pmu_base;
3462306a36Sopenharmony_ci	struct irq_chip_generic *irq_gc;
3562306a36Sopenharmony_ci	struct irq_domain *irq_domain;
3662306a36Sopenharmony_ci#ifdef CONFIG_RESET_CONTROLLER
3762306a36Sopenharmony_ci	struct reset_controller_dev reset;
3862306a36Sopenharmony_ci#endif
3962306a36Sopenharmony_ci};
4062306a36Sopenharmony_ci
4162306a36Sopenharmony_ci/*
4262306a36Sopenharmony_ci * The PMU contains a register to reset various subsystems within the
4362306a36Sopenharmony_ci * SoC.  Export this as a reset controller.
4462306a36Sopenharmony_ci */
4562306a36Sopenharmony_ci#ifdef CONFIG_RESET_CONTROLLER
4662306a36Sopenharmony_ci#define rcdev_to_pmu(rcdev) container_of(rcdev, struct pmu_data, reset)
4762306a36Sopenharmony_ci
4862306a36Sopenharmony_cistatic int pmu_reset_reset(struct reset_controller_dev *rc, unsigned long id)
4962306a36Sopenharmony_ci{
5062306a36Sopenharmony_ci	struct pmu_data *pmu = rcdev_to_pmu(rc);
5162306a36Sopenharmony_ci	unsigned long flags;
5262306a36Sopenharmony_ci	u32 val;
5362306a36Sopenharmony_ci
5462306a36Sopenharmony_ci	spin_lock_irqsave(&pmu->lock, flags);
5562306a36Sopenharmony_ci	val = readl_relaxed(pmu->pmc_base + PMC_SW_RST);
5662306a36Sopenharmony_ci	writel_relaxed(val & ~BIT(id), pmu->pmc_base + PMC_SW_RST);
5762306a36Sopenharmony_ci	writel_relaxed(val | BIT(id), pmu->pmc_base + PMC_SW_RST);
5862306a36Sopenharmony_ci	spin_unlock_irqrestore(&pmu->lock, flags);
5962306a36Sopenharmony_ci
6062306a36Sopenharmony_ci	return 0;
6162306a36Sopenharmony_ci}
6262306a36Sopenharmony_ci
6362306a36Sopenharmony_cistatic int pmu_reset_assert(struct reset_controller_dev *rc, unsigned long id)
6462306a36Sopenharmony_ci{
6562306a36Sopenharmony_ci	struct pmu_data *pmu = rcdev_to_pmu(rc);
6662306a36Sopenharmony_ci	unsigned long flags;
6762306a36Sopenharmony_ci	u32 val = ~BIT(id);
6862306a36Sopenharmony_ci
6962306a36Sopenharmony_ci	spin_lock_irqsave(&pmu->lock, flags);
7062306a36Sopenharmony_ci	val &= readl_relaxed(pmu->pmc_base + PMC_SW_RST);
7162306a36Sopenharmony_ci	writel_relaxed(val, pmu->pmc_base + PMC_SW_RST);
7262306a36Sopenharmony_ci	spin_unlock_irqrestore(&pmu->lock, flags);
7362306a36Sopenharmony_ci
7462306a36Sopenharmony_ci	return 0;
7562306a36Sopenharmony_ci}
7662306a36Sopenharmony_ci
7762306a36Sopenharmony_cistatic int pmu_reset_deassert(struct reset_controller_dev *rc, unsigned long id)
7862306a36Sopenharmony_ci{
7962306a36Sopenharmony_ci	struct pmu_data *pmu = rcdev_to_pmu(rc);
8062306a36Sopenharmony_ci	unsigned long flags;
8162306a36Sopenharmony_ci	u32 val = BIT(id);
8262306a36Sopenharmony_ci
8362306a36Sopenharmony_ci	spin_lock_irqsave(&pmu->lock, flags);
8462306a36Sopenharmony_ci	val |= readl_relaxed(pmu->pmc_base + PMC_SW_RST);
8562306a36Sopenharmony_ci	writel_relaxed(val, pmu->pmc_base + PMC_SW_RST);
8662306a36Sopenharmony_ci	spin_unlock_irqrestore(&pmu->lock, flags);
8762306a36Sopenharmony_ci
8862306a36Sopenharmony_ci	return 0;
8962306a36Sopenharmony_ci}
9062306a36Sopenharmony_ci
9162306a36Sopenharmony_cistatic const struct reset_control_ops pmu_reset_ops = {
9262306a36Sopenharmony_ci	.reset = pmu_reset_reset,
9362306a36Sopenharmony_ci	.assert = pmu_reset_assert,
9462306a36Sopenharmony_ci	.deassert = pmu_reset_deassert,
9562306a36Sopenharmony_ci};
9662306a36Sopenharmony_ci
9762306a36Sopenharmony_cistatic struct reset_controller_dev pmu_reset __initdata = {
9862306a36Sopenharmony_ci	.ops = &pmu_reset_ops,
9962306a36Sopenharmony_ci	.owner = THIS_MODULE,
10062306a36Sopenharmony_ci	.nr_resets = 32,
10162306a36Sopenharmony_ci};
10262306a36Sopenharmony_ci
10362306a36Sopenharmony_cistatic void __init pmu_reset_init(struct pmu_data *pmu)
10462306a36Sopenharmony_ci{
10562306a36Sopenharmony_ci	int ret;
10662306a36Sopenharmony_ci
10762306a36Sopenharmony_ci	pmu->reset = pmu_reset;
10862306a36Sopenharmony_ci	pmu->reset.of_node = pmu->of_node;
10962306a36Sopenharmony_ci
11062306a36Sopenharmony_ci	ret = reset_controller_register(&pmu->reset);
11162306a36Sopenharmony_ci	if (ret)
11262306a36Sopenharmony_ci		pr_err("pmu: %s failed: %d\n", "reset_controller_register", ret);
11362306a36Sopenharmony_ci}
11462306a36Sopenharmony_ci#else
11562306a36Sopenharmony_cistatic void __init pmu_reset_init(struct pmu_data *pmu)
11662306a36Sopenharmony_ci{
11762306a36Sopenharmony_ci}
11862306a36Sopenharmony_ci#endif
11962306a36Sopenharmony_ci
12062306a36Sopenharmony_cistruct pmu_domain {
12162306a36Sopenharmony_ci	struct pmu_data *pmu;
12262306a36Sopenharmony_ci	u32 pwr_mask;
12362306a36Sopenharmony_ci	u32 rst_mask;
12462306a36Sopenharmony_ci	u32 iso_mask;
12562306a36Sopenharmony_ci	struct generic_pm_domain base;
12662306a36Sopenharmony_ci};
12762306a36Sopenharmony_ci
12862306a36Sopenharmony_ci#define to_pmu_domain(dom) container_of(dom, struct pmu_domain, base)
12962306a36Sopenharmony_ci
13062306a36Sopenharmony_ci/*
13162306a36Sopenharmony_ci * This deals with the "old" Marvell sequence of bringing a power domain
13262306a36Sopenharmony_ci * down/up, which is: apply power, release reset, disable isolators.
13362306a36Sopenharmony_ci *
13462306a36Sopenharmony_ci * Later devices apparantly use a different sequence: power up, disable
13562306a36Sopenharmony_ci * isolators, assert repair signal, enable SRMA clock, enable AXI clock,
13662306a36Sopenharmony_ci * enable module clock, deassert reset.
13762306a36Sopenharmony_ci *
13862306a36Sopenharmony_ci * Note: reading the assembly, it seems that the IO accessors have an
13962306a36Sopenharmony_ci * unfortunate side-effect - they cause memory already read into registers
14062306a36Sopenharmony_ci * for the if () to be re-read for the bit-set or bit-clear operation.
14162306a36Sopenharmony_ci * The code is written to avoid this.
14262306a36Sopenharmony_ci */
14362306a36Sopenharmony_cistatic int pmu_domain_power_off(struct generic_pm_domain *domain)
14462306a36Sopenharmony_ci{
14562306a36Sopenharmony_ci	struct pmu_domain *pmu_dom = to_pmu_domain(domain);
14662306a36Sopenharmony_ci	struct pmu_data *pmu = pmu_dom->pmu;
14762306a36Sopenharmony_ci	unsigned long flags;
14862306a36Sopenharmony_ci	unsigned int val;
14962306a36Sopenharmony_ci	void __iomem *pmu_base = pmu->pmu_base;
15062306a36Sopenharmony_ci	void __iomem *pmc_base = pmu->pmc_base;
15162306a36Sopenharmony_ci
15262306a36Sopenharmony_ci	spin_lock_irqsave(&pmu->lock, flags);
15362306a36Sopenharmony_ci
15462306a36Sopenharmony_ci	/* Enable isolators */
15562306a36Sopenharmony_ci	if (pmu_dom->iso_mask) {
15662306a36Sopenharmony_ci		val = ~pmu_dom->iso_mask;
15762306a36Sopenharmony_ci		val &= readl_relaxed(pmu_base + PMU_ISO);
15862306a36Sopenharmony_ci		writel_relaxed(val, pmu_base + PMU_ISO);
15962306a36Sopenharmony_ci	}
16062306a36Sopenharmony_ci
16162306a36Sopenharmony_ci	/* Reset unit */
16262306a36Sopenharmony_ci	if (pmu_dom->rst_mask) {
16362306a36Sopenharmony_ci		val = ~pmu_dom->rst_mask;
16462306a36Sopenharmony_ci		val &= readl_relaxed(pmc_base + PMC_SW_RST);
16562306a36Sopenharmony_ci		writel_relaxed(val, pmc_base + PMC_SW_RST);
16662306a36Sopenharmony_ci	}
16762306a36Sopenharmony_ci
16862306a36Sopenharmony_ci	/* Power down */
16962306a36Sopenharmony_ci	val = readl_relaxed(pmu_base + PMU_PWR) | pmu_dom->pwr_mask;
17062306a36Sopenharmony_ci	writel_relaxed(val, pmu_base + PMU_PWR);
17162306a36Sopenharmony_ci
17262306a36Sopenharmony_ci	spin_unlock_irqrestore(&pmu->lock, flags);
17362306a36Sopenharmony_ci
17462306a36Sopenharmony_ci	return 0;
17562306a36Sopenharmony_ci}
17662306a36Sopenharmony_ci
17762306a36Sopenharmony_cistatic int pmu_domain_power_on(struct generic_pm_domain *domain)
17862306a36Sopenharmony_ci{
17962306a36Sopenharmony_ci	struct pmu_domain *pmu_dom = to_pmu_domain(domain);
18062306a36Sopenharmony_ci	struct pmu_data *pmu = pmu_dom->pmu;
18162306a36Sopenharmony_ci	unsigned long flags;
18262306a36Sopenharmony_ci	unsigned int val;
18362306a36Sopenharmony_ci	void __iomem *pmu_base = pmu->pmu_base;
18462306a36Sopenharmony_ci	void __iomem *pmc_base = pmu->pmc_base;
18562306a36Sopenharmony_ci
18662306a36Sopenharmony_ci	spin_lock_irqsave(&pmu->lock, flags);
18762306a36Sopenharmony_ci
18862306a36Sopenharmony_ci	/* Power on */
18962306a36Sopenharmony_ci	val = ~pmu_dom->pwr_mask & readl_relaxed(pmu_base + PMU_PWR);
19062306a36Sopenharmony_ci	writel_relaxed(val, pmu_base + PMU_PWR);
19162306a36Sopenharmony_ci
19262306a36Sopenharmony_ci	/* Release reset */
19362306a36Sopenharmony_ci	if (pmu_dom->rst_mask) {
19462306a36Sopenharmony_ci		val = pmu_dom->rst_mask;
19562306a36Sopenharmony_ci		val |= readl_relaxed(pmc_base + PMC_SW_RST);
19662306a36Sopenharmony_ci		writel_relaxed(val, pmc_base + PMC_SW_RST);
19762306a36Sopenharmony_ci	}
19862306a36Sopenharmony_ci
19962306a36Sopenharmony_ci	/* Disable isolators */
20062306a36Sopenharmony_ci	if (pmu_dom->iso_mask) {
20162306a36Sopenharmony_ci		val = pmu_dom->iso_mask;
20262306a36Sopenharmony_ci		val |= readl_relaxed(pmu_base + PMU_ISO);
20362306a36Sopenharmony_ci		writel_relaxed(val, pmu_base + PMU_ISO);
20462306a36Sopenharmony_ci	}
20562306a36Sopenharmony_ci
20662306a36Sopenharmony_ci	spin_unlock_irqrestore(&pmu->lock, flags);
20762306a36Sopenharmony_ci
20862306a36Sopenharmony_ci	return 0;
20962306a36Sopenharmony_ci}
21062306a36Sopenharmony_ci
21162306a36Sopenharmony_cistatic void __pmu_domain_register(struct pmu_domain *domain,
21262306a36Sopenharmony_ci	struct device_node *np)
21362306a36Sopenharmony_ci{
21462306a36Sopenharmony_ci	unsigned int val = readl_relaxed(domain->pmu->pmu_base + PMU_PWR);
21562306a36Sopenharmony_ci
21662306a36Sopenharmony_ci	domain->base.power_off = pmu_domain_power_off;
21762306a36Sopenharmony_ci	domain->base.power_on = pmu_domain_power_on;
21862306a36Sopenharmony_ci
21962306a36Sopenharmony_ci	pm_genpd_init(&domain->base, NULL, !(val & domain->pwr_mask));
22062306a36Sopenharmony_ci
22162306a36Sopenharmony_ci	if (np)
22262306a36Sopenharmony_ci		of_genpd_add_provider_simple(np, &domain->base);
22362306a36Sopenharmony_ci}
22462306a36Sopenharmony_ci
22562306a36Sopenharmony_ci/* PMU IRQ controller */
22662306a36Sopenharmony_cistatic void pmu_irq_handler(struct irq_desc *desc)
22762306a36Sopenharmony_ci{
22862306a36Sopenharmony_ci	struct pmu_data *pmu = irq_desc_get_handler_data(desc);
22962306a36Sopenharmony_ci	struct irq_chip_generic *gc = pmu->irq_gc;
23062306a36Sopenharmony_ci	struct irq_domain *domain = pmu->irq_domain;
23162306a36Sopenharmony_ci	void __iomem *base = gc->reg_base;
23262306a36Sopenharmony_ci	u32 stat = readl_relaxed(base + PMC_IRQ_CAUSE) & gc->mask_cache;
23362306a36Sopenharmony_ci	u32 done = ~0;
23462306a36Sopenharmony_ci
23562306a36Sopenharmony_ci	if (stat == 0) {
23662306a36Sopenharmony_ci		handle_bad_irq(desc);
23762306a36Sopenharmony_ci		return;
23862306a36Sopenharmony_ci	}
23962306a36Sopenharmony_ci
24062306a36Sopenharmony_ci	while (stat) {
24162306a36Sopenharmony_ci		u32 hwirq = fls(stat) - 1;
24262306a36Sopenharmony_ci
24362306a36Sopenharmony_ci		stat &= ~(1 << hwirq);
24462306a36Sopenharmony_ci		done &= ~(1 << hwirq);
24562306a36Sopenharmony_ci
24662306a36Sopenharmony_ci		generic_handle_irq(irq_find_mapping(domain, hwirq));
24762306a36Sopenharmony_ci	}
24862306a36Sopenharmony_ci
24962306a36Sopenharmony_ci	/*
25062306a36Sopenharmony_ci	 * The PMU mask register is not RW0C: it is RW.  This means that
25162306a36Sopenharmony_ci	 * the bits take whatever value is written to them; if you write
25262306a36Sopenharmony_ci	 * a '1', you will set the interrupt.
25362306a36Sopenharmony_ci	 *
25462306a36Sopenharmony_ci	 * Unfortunately this means there is NO race free way to clear
25562306a36Sopenharmony_ci	 * these interrupts.
25662306a36Sopenharmony_ci	 *
25762306a36Sopenharmony_ci	 * So, let's structure the code so that the window is as small as
25862306a36Sopenharmony_ci	 * possible.
25962306a36Sopenharmony_ci	 */
26062306a36Sopenharmony_ci	irq_gc_lock(gc);
26162306a36Sopenharmony_ci	done &= readl_relaxed(base + PMC_IRQ_CAUSE);
26262306a36Sopenharmony_ci	writel_relaxed(done, base + PMC_IRQ_CAUSE);
26362306a36Sopenharmony_ci	irq_gc_unlock(gc);
26462306a36Sopenharmony_ci}
26562306a36Sopenharmony_ci
26662306a36Sopenharmony_cistatic int __init dove_init_pmu_irq(struct pmu_data *pmu, int irq)
26762306a36Sopenharmony_ci{
26862306a36Sopenharmony_ci	const char *name = "pmu_irq";
26962306a36Sopenharmony_ci	struct irq_chip_generic *gc;
27062306a36Sopenharmony_ci	struct irq_domain *domain;
27162306a36Sopenharmony_ci	int ret;
27262306a36Sopenharmony_ci
27362306a36Sopenharmony_ci	/* mask and clear all interrupts */
27462306a36Sopenharmony_ci	writel(0, pmu->pmc_base + PMC_IRQ_MASK);
27562306a36Sopenharmony_ci	writel(0, pmu->pmc_base + PMC_IRQ_CAUSE);
27662306a36Sopenharmony_ci
27762306a36Sopenharmony_ci	domain = irq_domain_add_linear(pmu->of_node, NR_PMU_IRQS,
27862306a36Sopenharmony_ci				       &irq_generic_chip_ops, NULL);
27962306a36Sopenharmony_ci	if (!domain) {
28062306a36Sopenharmony_ci		pr_err("%s: unable to add irq domain\n", name);
28162306a36Sopenharmony_ci		return -ENOMEM;
28262306a36Sopenharmony_ci	}
28362306a36Sopenharmony_ci
28462306a36Sopenharmony_ci	ret = irq_alloc_domain_generic_chips(domain, NR_PMU_IRQS, 1, name,
28562306a36Sopenharmony_ci					     handle_level_irq,
28662306a36Sopenharmony_ci					     IRQ_NOREQUEST | IRQ_NOPROBE, 0,
28762306a36Sopenharmony_ci					     IRQ_GC_INIT_MASK_CACHE);
28862306a36Sopenharmony_ci	if (ret) {
28962306a36Sopenharmony_ci		pr_err("%s: unable to alloc irq domain gc: %d\n", name, ret);
29062306a36Sopenharmony_ci		irq_domain_remove(domain);
29162306a36Sopenharmony_ci		return ret;
29262306a36Sopenharmony_ci	}
29362306a36Sopenharmony_ci
29462306a36Sopenharmony_ci	gc = irq_get_domain_generic_chip(domain, 0);
29562306a36Sopenharmony_ci	gc->reg_base = pmu->pmc_base;
29662306a36Sopenharmony_ci	gc->chip_types[0].regs.mask = PMC_IRQ_MASK;
29762306a36Sopenharmony_ci	gc->chip_types[0].chip.irq_mask = irq_gc_mask_clr_bit;
29862306a36Sopenharmony_ci	gc->chip_types[0].chip.irq_unmask = irq_gc_mask_set_bit;
29962306a36Sopenharmony_ci
30062306a36Sopenharmony_ci	pmu->irq_domain = domain;
30162306a36Sopenharmony_ci	pmu->irq_gc = gc;
30262306a36Sopenharmony_ci
30362306a36Sopenharmony_ci	irq_set_handler_data(irq, pmu);
30462306a36Sopenharmony_ci	irq_set_chained_handler(irq, pmu_irq_handler);
30562306a36Sopenharmony_ci
30662306a36Sopenharmony_ci	return 0;
30762306a36Sopenharmony_ci}
30862306a36Sopenharmony_ci
30962306a36Sopenharmony_ciint __init dove_init_pmu_legacy(const struct dove_pmu_initdata *initdata)
31062306a36Sopenharmony_ci{
31162306a36Sopenharmony_ci	const struct dove_pmu_domain_initdata *domain_initdata;
31262306a36Sopenharmony_ci	struct pmu_data *pmu;
31362306a36Sopenharmony_ci	int ret;
31462306a36Sopenharmony_ci
31562306a36Sopenharmony_ci	pmu = kzalloc(sizeof(*pmu), GFP_KERNEL);
31662306a36Sopenharmony_ci	if (!pmu)
31762306a36Sopenharmony_ci		return -ENOMEM;
31862306a36Sopenharmony_ci
31962306a36Sopenharmony_ci	spin_lock_init(&pmu->lock);
32062306a36Sopenharmony_ci	pmu->pmc_base = initdata->pmc_base;
32162306a36Sopenharmony_ci	pmu->pmu_base = initdata->pmu_base;
32262306a36Sopenharmony_ci
32362306a36Sopenharmony_ci	pmu_reset_init(pmu);
32462306a36Sopenharmony_ci	for (domain_initdata = initdata->domains; domain_initdata->name;
32562306a36Sopenharmony_ci	     domain_initdata++) {
32662306a36Sopenharmony_ci		struct pmu_domain *domain;
32762306a36Sopenharmony_ci
32862306a36Sopenharmony_ci		domain = kzalloc(sizeof(*domain), GFP_KERNEL);
32962306a36Sopenharmony_ci		if (domain) {
33062306a36Sopenharmony_ci			domain->pmu = pmu;
33162306a36Sopenharmony_ci			domain->pwr_mask = domain_initdata->pwr_mask;
33262306a36Sopenharmony_ci			domain->rst_mask = domain_initdata->rst_mask;
33362306a36Sopenharmony_ci			domain->iso_mask = domain_initdata->iso_mask;
33462306a36Sopenharmony_ci			domain->base.name = domain_initdata->name;
33562306a36Sopenharmony_ci
33662306a36Sopenharmony_ci			__pmu_domain_register(domain, NULL);
33762306a36Sopenharmony_ci		}
33862306a36Sopenharmony_ci	}
33962306a36Sopenharmony_ci
34062306a36Sopenharmony_ci	ret = dove_init_pmu_irq(pmu, initdata->irq);
34162306a36Sopenharmony_ci	if (ret)
34262306a36Sopenharmony_ci		pr_err("dove_init_pmu_irq() failed: %d\n", ret);
34362306a36Sopenharmony_ci
34462306a36Sopenharmony_ci	if (pmu->irq_domain)
34562306a36Sopenharmony_ci		irq_domain_associate_many(pmu->irq_domain,
34662306a36Sopenharmony_ci					  initdata->irq_domain_start,
34762306a36Sopenharmony_ci					  0, NR_PMU_IRQS);
34862306a36Sopenharmony_ci
34962306a36Sopenharmony_ci	return 0;
35062306a36Sopenharmony_ci}
35162306a36Sopenharmony_ci
35262306a36Sopenharmony_ci/*
35362306a36Sopenharmony_ci * pmu: power-manager@d0000 {
35462306a36Sopenharmony_ci *	compatible = "marvell,dove-pmu";
35562306a36Sopenharmony_ci *	reg = <0xd0000 0x8000> <0xd8000 0x8000>;
35662306a36Sopenharmony_ci *	interrupts = <33>;
35762306a36Sopenharmony_ci *	interrupt-controller;
35862306a36Sopenharmony_ci *	#reset-cells = 1;
35962306a36Sopenharmony_ci *	vpu_domain: vpu-domain {
36062306a36Sopenharmony_ci *		#power-domain-cells = <0>;
36162306a36Sopenharmony_ci *		marvell,pmu_pwr_mask = <0x00000008>;
36262306a36Sopenharmony_ci *		marvell,pmu_iso_mask = <0x00000001>;
36362306a36Sopenharmony_ci *		resets = <&pmu 16>;
36462306a36Sopenharmony_ci *	};
36562306a36Sopenharmony_ci *	gpu_domain: gpu-domain {
36662306a36Sopenharmony_ci *		#power-domain-cells = <0>;
36762306a36Sopenharmony_ci *		marvell,pmu_pwr_mask = <0x00000004>;
36862306a36Sopenharmony_ci *		marvell,pmu_iso_mask = <0x00000002>;
36962306a36Sopenharmony_ci *		resets = <&pmu 18>;
37062306a36Sopenharmony_ci *	};
37162306a36Sopenharmony_ci * };
37262306a36Sopenharmony_ci */
37362306a36Sopenharmony_ciint __init dove_init_pmu(void)
37462306a36Sopenharmony_ci{
37562306a36Sopenharmony_ci	struct device_node *np_pmu, *domains_node, *np;
37662306a36Sopenharmony_ci	struct pmu_data *pmu;
37762306a36Sopenharmony_ci	int ret, parent_irq;
37862306a36Sopenharmony_ci
37962306a36Sopenharmony_ci	/* Lookup the PMU node */
38062306a36Sopenharmony_ci	np_pmu = of_find_compatible_node(NULL, NULL, "marvell,dove-pmu");
38162306a36Sopenharmony_ci	if (!np_pmu)
38262306a36Sopenharmony_ci		return 0;
38362306a36Sopenharmony_ci
38462306a36Sopenharmony_ci	domains_node = of_get_child_by_name(np_pmu, "domains");
38562306a36Sopenharmony_ci	if (!domains_node) {
38662306a36Sopenharmony_ci		pr_err("%pOFn: failed to find domains sub-node\n", np_pmu);
38762306a36Sopenharmony_ci		return 0;
38862306a36Sopenharmony_ci	}
38962306a36Sopenharmony_ci
39062306a36Sopenharmony_ci	pmu = kzalloc(sizeof(*pmu), GFP_KERNEL);
39162306a36Sopenharmony_ci	if (!pmu)
39262306a36Sopenharmony_ci		return -ENOMEM;
39362306a36Sopenharmony_ci
39462306a36Sopenharmony_ci	spin_lock_init(&pmu->lock);
39562306a36Sopenharmony_ci	pmu->of_node = np_pmu;
39662306a36Sopenharmony_ci	pmu->pmc_base = of_iomap(pmu->of_node, 0);
39762306a36Sopenharmony_ci	pmu->pmu_base = of_iomap(pmu->of_node, 1);
39862306a36Sopenharmony_ci	if (!pmu->pmc_base || !pmu->pmu_base) {
39962306a36Sopenharmony_ci		pr_err("%pOFn: failed to map PMU\n", np_pmu);
40062306a36Sopenharmony_ci		iounmap(pmu->pmu_base);
40162306a36Sopenharmony_ci		iounmap(pmu->pmc_base);
40262306a36Sopenharmony_ci		kfree(pmu);
40362306a36Sopenharmony_ci		return -ENOMEM;
40462306a36Sopenharmony_ci	}
40562306a36Sopenharmony_ci
40662306a36Sopenharmony_ci	pmu_reset_init(pmu);
40762306a36Sopenharmony_ci
40862306a36Sopenharmony_ci	for_each_available_child_of_node(domains_node, np) {
40962306a36Sopenharmony_ci		struct of_phandle_args args;
41062306a36Sopenharmony_ci		struct pmu_domain *domain;
41162306a36Sopenharmony_ci
41262306a36Sopenharmony_ci		domain = kzalloc(sizeof(*domain), GFP_KERNEL);
41362306a36Sopenharmony_ci		if (!domain)
41462306a36Sopenharmony_ci			break;
41562306a36Sopenharmony_ci
41662306a36Sopenharmony_ci		domain->pmu = pmu;
41762306a36Sopenharmony_ci		domain->base.name = kasprintf(GFP_KERNEL, "%pOFn", np);
41862306a36Sopenharmony_ci		if (!domain->base.name) {
41962306a36Sopenharmony_ci			kfree(domain);
42062306a36Sopenharmony_ci			break;
42162306a36Sopenharmony_ci		}
42262306a36Sopenharmony_ci
42362306a36Sopenharmony_ci		of_property_read_u32(np, "marvell,pmu_pwr_mask",
42462306a36Sopenharmony_ci				     &domain->pwr_mask);
42562306a36Sopenharmony_ci		of_property_read_u32(np, "marvell,pmu_iso_mask",
42662306a36Sopenharmony_ci				     &domain->iso_mask);
42762306a36Sopenharmony_ci
42862306a36Sopenharmony_ci		/*
42962306a36Sopenharmony_ci		 * We parse the reset controller property directly here
43062306a36Sopenharmony_ci		 * to ensure that we can operate when the reset controller
43162306a36Sopenharmony_ci		 * support is not configured into the kernel.
43262306a36Sopenharmony_ci		 */
43362306a36Sopenharmony_ci		ret = of_parse_phandle_with_args(np, "resets", "#reset-cells",
43462306a36Sopenharmony_ci						 0, &args);
43562306a36Sopenharmony_ci		if (ret == 0) {
43662306a36Sopenharmony_ci			if (args.np == pmu->of_node)
43762306a36Sopenharmony_ci				domain->rst_mask = BIT(args.args[0]);
43862306a36Sopenharmony_ci			of_node_put(args.np);
43962306a36Sopenharmony_ci		}
44062306a36Sopenharmony_ci
44162306a36Sopenharmony_ci		__pmu_domain_register(domain, np);
44262306a36Sopenharmony_ci	}
44362306a36Sopenharmony_ci
44462306a36Sopenharmony_ci	/* Loss of the interrupt controller is not a fatal error. */
44562306a36Sopenharmony_ci	parent_irq = irq_of_parse_and_map(pmu->of_node, 0);
44662306a36Sopenharmony_ci	if (!parent_irq) {
44762306a36Sopenharmony_ci		pr_err("%pOFn: no interrupt specified\n", np_pmu);
44862306a36Sopenharmony_ci	} else {
44962306a36Sopenharmony_ci		ret = dove_init_pmu_irq(pmu, parent_irq);
45062306a36Sopenharmony_ci		if (ret)
45162306a36Sopenharmony_ci			pr_err("dove_init_pmu_irq() failed: %d\n", ret);
45262306a36Sopenharmony_ci	}
45362306a36Sopenharmony_ci
45462306a36Sopenharmony_ci	return 0;
45562306a36Sopenharmony_ci}
456