xref: /OK3568_Linux_fs/kernel/arch/arm/mach-zx/platsmp.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  * Copyright 2014 Linaro Ltd.
4*4882a593Smuzhiyun  * Copyright (C) 2014 ZTE Corporation.
5*4882a593Smuzhiyun  */
6*4882a593Smuzhiyun 
7*4882a593Smuzhiyun #include <linux/delay.h>
8*4882a593Smuzhiyun #include <linux/errno.h>
9*4882a593Smuzhiyun #include <linux/init.h>
10*4882a593Smuzhiyun #include <linux/io.h>
11*4882a593Smuzhiyun #include <linux/jiffies.h>
12*4882a593Smuzhiyun #include <linux/of.h>
13*4882a593Smuzhiyun #include <linux/of_address.h>
14*4882a593Smuzhiyun #include <linux/smp.h>
15*4882a593Smuzhiyun 
16*4882a593Smuzhiyun #include <asm/cacheflush.h>
17*4882a593Smuzhiyun #include <asm/cp15.h>
18*4882a593Smuzhiyun #include <asm/fncpy.h>
19*4882a593Smuzhiyun #include <asm/proc-fns.h>
20*4882a593Smuzhiyun #include <asm/smp_scu.h>
21*4882a593Smuzhiyun #include <asm/smp_plat.h>
22*4882a593Smuzhiyun 
23*4882a593Smuzhiyun #include "core.h"
24*4882a593Smuzhiyun 
25*4882a593Smuzhiyun #define AON_SYS_CTRL_RESERVED1		0xa8
26*4882a593Smuzhiyun 
27*4882a593Smuzhiyun #define BUS_MATRIX_REMAP_CONFIG		0x00
28*4882a593Smuzhiyun 
29*4882a593Smuzhiyun #define PCU_CPU0_CTRL			0x00
30*4882a593Smuzhiyun #define PCU_CPU1_CTRL			0x04
31*4882a593Smuzhiyun #define PCU_CPU1_ST			0x0c
32*4882a593Smuzhiyun #define PCU_GLOBAL_CTRL			0x14
33*4882a593Smuzhiyun #define PCU_EXPEND_CONTROL		0x34
34*4882a593Smuzhiyun 
35*4882a593Smuzhiyun #define ZX_IRAM_BASE			0x00200000
36*4882a593Smuzhiyun 
37*4882a593Smuzhiyun static void __iomem *pcu_base;
38*4882a593Smuzhiyun static void __iomem *matrix_base;
39*4882a593Smuzhiyun static void __iomem *scu_base;
40*4882a593Smuzhiyun 
zx_smp_prepare_cpus(unsigned int max_cpus)41*4882a593Smuzhiyun void __init zx_smp_prepare_cpus(unsigned int max_cpus)
42*4882a593Smuzhiyun {
43*4882a593Smuzhiyun 	struct device_node *np;
44*4882a593Smuzhiyun 	unsigned long base = 0;
45*4882a593Smuzhiyun 	void __iomem *aonsysctrl_base;
46*4882a593Smuzhiyun 	void __iomem *sys_iram;
47*4882a593Smuzhiyun 
48*4882a593Smuzhiyun 	base = scu_a9_get_base();
49*4882a593Smuzhiyun 	scu_base = ioremap(base, SZ_256);
50*4882a593Smuzhiyun 	if (!scu_base) {
51*4882a593Smuzhiyun 		pr_err("%s: failed to map scu\n", __func__);
52*4882a593Smuzhiyun 		return;
53*4882a593Smuzhiyun 	}
54*4882a593Smuzhiyun 
55*4882a593Smuzhiyun 	scu_enable(scu_base);
56*4882a593Smuzhiyun 
57*4882a593Smuzhiyun 	np = of_find_compatible_node(NULL, NULL, "zte,sysctrl");
58*4882a593Smuzhiyun 	if (!np) {
59*4882a593Smuzhiyun 		pr_err("%s: failed to find sysctrl node\n", __func__);
60*4882a593Smuzhiyun 		return;
61*4882a593Smuzhiyun 	}
62*4882a593Smuzhiyun 
63*4882a593Smuzhiyun 	aonsysctrl_base = of_iomap(np, 0);
64*4882a593Smuzhiyun 	if (!aonsysctrl_base) {
65*4882a593Smuzhiyun 		pr_err("%s: failed to map aonsysctrl\n", __func__);
66*4882a593Smuzhiyun 		of_node_put(np);
67*4882a593Smuzhiyun 		return;
68*4882a593Smuzhiyun 	}
69*4882a593Smuzhiyun 
70*4882a593Smuzhiyun 	/*
71*4882a593Smuzhiyun 	 * Write the address of secondary startup into the
72*4882a593Smuzhiyun 	 * system-wide flags register. The BootMonitor waits
73*4882a593Smuzhiyun 	 * until it receives a soft interrupt, and then the
74*4882a593Smuzhiyun 	 * secondary CPU branches to this address.
75*4882a593Smuzhiyun 	 */
76*4882a593Smuzhiyun 	__raw_writel(__pa_symbol(zx_secondary_startup),
77*4882a593Smuzhiyun 		     aonsysctrl_base + AON_SYS_CTRL_RESERVED1);
78*4882a593Smuzhiyun 
79*4882a593Smuzhiyun 	iounmap(aonsysctrl_base);
80*4882a593Smuzhiyun 	of_node_put(np);
81*4882a593Smuzhiyun 
82*4882a593Smuzhiyun 	np = of_find_compatible_node(NULL, NULL, "zte,zx296702-pcu");
83*4882a593Smuzhiyun 	pcu_base = of_iomap(np, 0);
84*4882a593Smuzhiyun 	of_node_put(np);
85*4882a593Smuzhiyun 	WARN_ON(!pcu_base);
86*4882a593Smuzhiyun 
87*4882a593Smuzhiyun 	np = of_find_compatible_node(NULL, NULL, "zte,zx-bus-matrix");
88*4882a593Smuzhiyun 	matrix_base = of_iomap(np, 0);
89*4882a593Smuzhiyun 	of_node_put(np);
90*4882a593Smuzhiyun 	WARN_ON(!matrix_base);
91*4882a593Smuzhiyun 
92*4882a593Smuzhiyun 	/* Map the first 4 KB IRAM for suspend usage */
93*4882a593Smuzhiyun 	sys_iram = __arm_ioremap_exec(ZX_IRAM_BASE, PAGE_SIZE, false);
94*4882a593Smuzhiyun 	zx_secondary_startup_pa = __pa_symbol(zx_secondary_startup);
95*4882a593Smuzhiyun 	fncpy(sys_iram, &zx_resume_jump, zx_suspend_iram_sz);
96*4882a593Smuzhiyun }
97*4882a593Smuzhiyun 
zx_boot_secondary(unsigned int cpu,struct task_struct * idle)98*4882a593Smuzhiyun static int zx_boot_secondary(unsigned int cpu, struct task_struct *idle)
99*4882a593Smuzhiyun {
100*4882a593Smuzhiyun 	static bool first_boot = true;
101*4882a593Smuzhiyun 
102*4882a593Smuzhiyun 	if (first_boot) {
103*4882a593Smuzhiyun 		arch_send_wakeup_ipi_mask(cpumask_of(cpu));
104*4882a593Smuzhiyun 		first_boot = false;
105*4882a593Smuzhiyun 		return 0;
106*4882a593Smuzhiyun 	}
107*4882a593Smuzhiyun 
108*4882a593Smuzhiyun 	/* Swap the base address mapping between IRAM and IROM */
109*4882a593Smuzhiyun 	writel_relaxed(0x1, matrix_base + BUS_MATRIX_REMAP_CONFIG);
110*4882a593Smuzhiyun 
111*4882a593Smuzhiyun 	/* Power on CPU1 */
112*4882a593Smuzhiyun 	writel_relaxed(0x0, pcu_base + PCU_CPU1_CTRL);
113*4882a593Smuzhiyun 
114*4882a593Smuzhiyun 	/* Wait for power on ack */
115*4882a593Smuzhiyun 	while (readl_relaxed(pcu_base + PCU_CPU1_ST) & 0x4)
116*4882a593Smuzhiyun 		cpu_relax();
117*4882a593Smuzhiyun 
118*4882a593Smuzhiyun 	/* Swap back the mapping of IRAM and IROM */
119*4882a593Smuzhiyun 	writel_relaxed(0x0, matrix_base + BUS_MATRIX_REMAP_CONFIG);
120*4882a593Smuzhiyun 
121*4882a593Smuzhiyun 	return 0;
122*4882a593Smuzhiyun }
123*4882a593Smuzhiyun 
124*4882a593Smuzhiyun #ifdef CONFIG_HOTPLUG_CPU
cpu_enter_lowpower(void)125*4882a593Smuzhiyun static inline void cpu_enter_lowpower(void)
126*4882a593Smuzhiyun {
127*4882a593Smuzhiyun 	unsigned int v;
128*4882a593Smuzhiyun 
129*4882a593Smuzhiyun 	asm volatile(
130*4882a593Smuzhiyun 		"mcr	p15, 0, %1, c7, c5, 0\n"
131*4882a593Smuzhiyun 	"	mcr	p15, 0, %1, c7, c10, 4\n"
132*4882a593Smuzhiyun 	/*
133*4882a593Smuzhiyun 	 * Turn off coherency
134*4882a593Smuzhiyun 	 */
135*4882a593Smuzhiyun 	"	mrc	p15, 0, %0, c1, c0, 1\n"
136*4882a593Smuzhiyun 	"	bic	%0, %0, %3\n"
137*4882a593Smuzhiyun 	"	mcr	p15, 0, %0, c1, c0, 1\n"
138*4882a593Smuzhiyun 	"	mrc	p15, 0, %0, c1, c0, 0\n"
139*4882a593Smuzhiyun 	"	bic	%0, %0, %2\n"
140*4882a593Smuzhiyun 	"	mcr	p15, 0, %0, c1, c0, 0\n"
141*4882a593Smuzhiyun 	  : "=&r" (v)
142*4882a593Smuzhiyun 	  : "r" (0), "Ir" (CR_C), "Ir" (0x40)
143*4882a593Smuzhiyun 	  : "cc");
144*4882a593Smuzhiyun }
145*4882a593Smuzhiyun 
zx_cpu_kill(unsigned int cpu)146*4882a593Smuzhiyun static int zx_cpu_kill(unsigned int cpu)
147*4882a593Smuzhiyun {
148*4882a593Smuzhiyun 	unsigned long timeout = jiffies + msecs_to_jiffies(2000);
149*4882a593Smuzhiyun 
150*4882a593Smuzhiyun 	writel_relaxed(0x2, pcu_base + PCU_CPU1_CTRL);
151*4882a593Smuzhiyun 
152*4882a593Smuzhiyun 	while ((readl_relaxed(pcu_base + PCU_CPU1_ST) & 0x3) != 0x0) {
153*4882a593Smuzhiyun 		if (time_after(jiffies, timeout)) {
154*4882a593Smuzhiyun 			pr_err("*** cpu1 poweroff timeout\n");
155*4882a593Smuzhiyun 			break;
156*4882a593Smuzhiyun 		}
157*4882a593Smuzhiyun 	}
158*4882a593Smuzhiyun 	return 1;
159*4882a593Smuzhiyun }
160*4882a593Smuzhiyun 
zx_cpu_die(unsigned int cpu)161*4882a593Smuzhiyun static void zx_cpu_die(unsigned int cpu)
162*4882a593Smuzhiyun {
163*4882a593Smuzhiyun 	scu_power_mode(scu_base, SCU_PM_POWEROFF);
164*4882a593Smuzhiyun 	cpu_enter_lowpower();
165*4882a593Smuzhiyun 
166*4882a593Smuzhiyun 	while (1)
167*4882a593Smuzhiyun 		cpu_do_idle();
168*4882a593Smuzhiyun }
169*4882a593Smuzhiyun #endif
170*4882a593Smuzhiyun 
zx_secondary_init(unsigned int cpu)171*4882a593Smuzhiyun static void zx_secondary_init(unsigned int cpu)
172*4882a593Smuzhiyun {
173*4882a593Smuzhiyun 	scu_power_mode(scu_base, SCU_PM_NORMAL);
174*4882a593Smuzhiyun }
175*4882a593Smuzhiyun 
176*4882a593Smuzhiyun static const struct smp_operations zx_smp_ops __initconst = {
177*4882a593Smuzhiyun 	.smp_prepare_cpus	= zx_smp_prepare_cpus,
178*4882a593Smuzhiyun 	.smp_secondary_init	= zx_secondary_init,
179*4882a593Smuzhiyun 	.smp_boot_secondary	= zx_boot_secondary,
180*4882a593Smuzhiyun #ifdef CONFIG_HOTPLUG_CPU
181*4882a593Smuzhiyun 	.cpu_kill		= zx_cpu_kill,
182*4882a593Smuzhiyun 	.cpu_die		= zx_cpu_die,
183*4882a593Smuzhiyun #endif
184*4882a593Smuzhiyun };
185*4882a593Smuzhiyun 
186*4882a593Smuzhiyun CPU_METHOD_OF_DECLARE(zx_smp, "zte,zx296702-smp", &zx_smp_ops);
187