alpine_machdep_mp.c revision 289522
1/*-
2 * Copyright (c) 2013 Ruslan Bukin <br@bsdpad.com>
3 * Copyright (c) 2015 Semihalf
4 * All rights reserved.
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
8 * are met:
9 * 1. Redistributions of source code must retain the above copyright
10 *    notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 *    notice, this list of conditions and the following disclaimer in the
13 *    documentation and/or other materials provided with the distribution.
14 *
15 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
16 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25 * SUCH DAMAGE.
26 *
27 */
28
29#include <sys/cdefs.h>
30__FBSDID("$FreeBSD: head/sys/arm/annapurna/alpine/alpine_machdep_mp.c 289522 2015-10-18 16:54:34Z ian $");
31#include <sys/param.h>
32#include <sys/systm.h>
33#include <sys/bus.h>
34#include <sys/lock.h>
35#include <sys/mutex.h>
36#include <sys/smp.h>
37#include <sys/cpuset.h>
38
39#include <vm/vm.h>
40#include <vm/pmap.h>
41
42#include <machine/smp.h>
43#include <machine/fdt.h>
44#include <machine/intr.h>
45#include <machine/cpu-v6.h>
46
47#include <dev/fdt/fdt_common.h>
48#include <dev/ofw/ofw_cpu.h>
49#include <dev/ofw/ofw_bus_subr.h>
50
51#define AL_CPU_RESUME_WATERMARK_REG		0x00
52#define AL_CPU_RESUME_FLAGS_REG			0x04
53#define AL_CPU_RESUME_PCPU_RADDR_REG(cpu)	(0x08 + 0x04 + 8*(cpu))
54#define AL_CPU_RESUME_PCPU_FLAGS(cpu)		(0x08 + 8*(cpu))
55
56/* Per-CPU flags */
57#define AL_CPU_RESUME_FLG_PERCPU_DONT_RESUME	(1 << 2)
58
59/* The expected magic number for validating the resume addresses */
60#define AL_CPU_RESUME_MAGIC_NUM			0xf0e1d200
61#define AL_CPU_RESUME_MAGIC_NUM_MASK		0xffffff00
62
63/* The expected minimal version number for validating the capabilities */
64#define AL_CPU_RESUME_MIN_VER			0x000000c3
65#define AL_CPU_RESUME_MIN_VER_MASK		0x000000ff
66
67/* Field controlling the boot-up of companion cores */
68#define AL_NB_INIT_CONTROL		(0x8)
69#define AL_NB_CONFIG_STATUS_PWR_CTRL(cpu)	(0x2020 + (cpu)*0x100)
70
71#define SERDES_NUM_GROUPS	4
72#define SERDES_GROUP_SIZE	0x400
73
74extern bus_addr_t al_devmap_pa;
75extern bus_addr_t al_devmap_size;
76
77extern void mpentry(void);
78
79int alpine_serdes_resource_get(uint32_t group, bus_space_tag_t *tag,
80    bus_addr_t *baddr);
81static int platform_mp_get_core_cnt(void);
82static int alpine_get_cpu_resume_base(u_long *pbase, u_long *psize);
83static int alpine_get_nb_base(u_long *pbase, u_long *psize);
84static int alpine_get_serdes_base(u_long *pbase, u_long *psize);
85int alpine_serdes_resource_get(uint32_t group, bus_space_tag_t *tag,
86    bus_addr_t *baddr);
87static boolean_t alpine_validate_cpu(u_int, phandle_t, u_int, pcell_t *);
88
89static boolean_t
90alpine_validate_cpu(u_int id, phandle_t child, u_int addr_cell, pcell_t *reg)
91{
92	return fdt_is_compatible(child, "arm,cortex-a15");
93}
94
95static int
96platform_mp_get_core_cnt(void)
97{
98	static int ncores = 0;
99	int nchilds;
100	uint32_t reg;
101
102	/* Calculate ncores value only once */
103	if (ncores)
104		return (ncores);
105
106	reg = cp15_l2ctlr_get();
107	ncores = CPUV7_L2CTLR_NPROC(reg);
108
109	nchilds = ofw_cpu_early_foreach(alpine_validate_cpu, false);
110
111	/* Limit CPUs if DTS has configured less than available */
112	if ((nchilds > 0) && (nchilds < ncores)) {
113		printf("SMP: limiting number of active CPUs to %d out of %d\n",
114		    nchilds, ncores);
115		ncores = nchilds;
116	}
117
118	return (ncores);
119}
120
121void
122platform_mp_init_secondary(void)
123{
124
125	arm_pic_init_secondary();
126}
127
128void
129platform_mp_setmaxid(void)
130{
131	int core_cnt;
132
133	core_cnt = platform_mp_get_core_cnt();
134	mp_maxid = core_cnt - 1;
135}
136
137int
138platform_mp_probe(void)
139{
140	mp_ncpus = platform_mp_get_core_cnt();
141	return (1);
142}
143
144static int
145alpine_get_cpu_resume_base(u_long *pbase, u_long *psize)
146{
147	phandle_t node;
148	u_long base = 0;
149	u_long size = 0;
150
151	if (pbase == NULL || psize == NULL)
152		return (EINVAL);
153
154	if ((node = OF_finddevice("/")) == -1)
155		return (EFAULT);
156
157	if ((node =
158	    ofw_bus_find_compatible(node, "annapurna-labs,al-cpu-resume")) == 0)
159		return (EFAULT);
160
161	if (fdt_regsize(node, &base, &size))
162		return (EFAULT);
163
164	*pbase = base;
165	*psize = size;
166
167	return (0);
168}
169
170static int
171alpine_get_nb_base(u_long *pbase, u_long *psize)
172{
173	phandle_t node;
174	u_long base = 0;
175	u_long size = 0;
176
177	if (pbase == NULL || psize == NULL)
178		return (EINVAL);
179
180	if ((node = OF_finddevice("/")) == -1)
181		return (EFAULT);
182
183	if ((node =
184	    ofw_bus_find_compatible(node, "annapurna-labs,al-nb-service")) == 0)
185		return (EFAULT);
186
187	if (fdt_regsize(node, &base, &size))
188		return (EFAULT);
189
190	*pbase = base;
191	*psize = size;
192
193	return (0);
194}
195
196void
197platform_mp_start_ap(void)
198{
199	uint32_t physaddr;
200	vm_offset_t vaddr;
201	uint32_t val;
202	uint32_t start_mask;
203	u_long cpu_resume_base;
204	u_long nb_base;
205	u_long cpu_resume_size;
206	u_long nb_size;
207	bus_addr_t cpu_resume_baddr;
208	bus_addr_t nb_baddr;
209	int a;
210
211	if (alpine_get_cpu_resume_base(&cpu_resume_base, &cpu_resume_size))
212		panic("Couldn't resolve cpu_resume_base address\n");
213
214	if (alpine_get_nb_base(&nb_base, &nb_size))
215		panic("Couldn't resolve_nb_base address\n");
216
217	/* Proceed with start addresses for additional CPUs */
218	if (bus_space_map(fdtbus_bs_tag, al_devmap_pa + cpu_resume_base,
219	    cpu_resume_size, 0, &cpu_resume_baddr))
220		panic("Couldn't map CPU-resume area");
221	if (bus_space_map(fdtbus_bs_tag, al_devmap_pa + nb_base,
222	    nb_size, 0, &nb_baddr))
223		panic("Couldn't map NB-service area");
224
225	/* Proceed with start addresses for additional CPUs */
226	val = bus_space_read_4(fdtbus_bs_tag, cpu_resume_baddr,
227	    AL_CPU_RESUME_WATERMARK_REG);
228	if (((val & AL_CPU_RESUME_MAGIC_NUM_MASK) != AL_CPU_RESUME_MAGIC_NUM) ||
229	    ((val & AL_CPU_RESUME_MIN_VER_MASK) < AL_CPU_RESUME_MIN_VER)) {
230		panic("CPU-resume device is not compatible");
231	}
232
233	vaddr = (vm_offset_t)mpentry;
234	physaddr = pmap_kextract(vaddr);
235
236	for (a = 1; a < platform_mp_get_core_cnt(); a++) {
237		/* Power up the core */
238		bus_space_write_4(fdtbus_bs_tag, nb_baddr,
239		    AL_NB_CONFIG_STATUS_PWR_CTRL(a), 0);
240		mb();
241
242		/* Enable resume */
243		val = bus_space_read_4(fdtbus_bs_tag, cpu_resume_baddr,
244		    AL_CPU_RESUME_PCPU_FLAGS(a));
245		val &= ~AL_CPU_RESUME_FLG_PERCPU_DONT_RESUME;
246		bus_space_write_4(fdtbus_bs_tag, cpu_resume_baddr,
247		    AL_CPU_RESUME_PCPU_FLAGS(a), val);
248		mb();
249
250		/* Set resume physical address */
251		bus_space_write_4(fdtbus_bs_tag, cpu_resume_baddr,
252		    AL_CPU_RESUME_PCPU_RADDR_REG(a), physaddr);
253		mb();
254	}
255
256	/* Release cores from reset */
257	if (bus_space_map(fdtbus_bs_tag, al_devmap_pa + nb_base,
258	    nb_size, 0, &nb_baddr))
259		panic("Couldn't map NB-service area");
260
261	start_mask = (1 << platform_mp_get_core_cnt()) - 1;
262
263	/* Release cores from reset */
264	val = bus_space_read_4(fdtbus_bs_tag, nb_baddr, AL_NB_INIT_CONTROL);
265	val |= start_mask;
266	bus_space_write_4(fdtbus_bs_tag, nb_baddr, AL_NB_INIT_CONTROL, val);
267	dsb();
268
269	bus_space_unmap(fdtbus_bs_tag, nb_baddr, nb_size);
270	bus_space_unmap(fdtbus_bs_tag, cpu_resume_baddr, cpu_resume_size);
271}
272
273static int
274alpine_get_serdes_base(u_long *pbase, u_long *psize)
275{
276	phandle_t node;
277	u_long base = 0;
278	u_long size = 0;
279
280	if (pbase == NULL || psize == NULL)
281		return (EINVAL);
282
283	if ((node = OF_finddevice("/")) == -1)
284		return (EFAULT);
285
286	if ((node =
287	    ofw_bus_find_compatible(node, "annapurna-labs,al-serdes")) == 0)
288		return (EFAULT);
289
290	if (fdt_regsize(node, &base, &size))
291		return (EFAULT);
292
293	*pbase = base;
294	*psize = size;
295
296	return (0);
297}
298
299int
300alpine_serdes_resource_get(uint32_t group, bus_space_tag_t *tag, bus_addr_t *baddr)
301{
302	u_long serdes_base, serdes_size;
303	int ret;
304	static bus_addr_t baddr_mapped[SERDES_NUM_GROUPS];
305
306	if (group >= SERDES_NUM_GROUPS)
307		return (EINVAL);
308
309	if (baddr_mapped[group]) {
310		*tag = fdtbus_bs_tag;
311		*baddr = baddr_mapped[group];
312		return (0);
313	}
314
315	ret = alpine_get_serdes_base(&serdes_base, &serdes_size);
316	if (ret)
317		return (ret);
318
319	ret = bus_space_map(fdtbus_bs_tag,
320	    al_devmap_pa + serdes_base + group * SERDES_GROUP_SIZE,
321	    (SERDES_NUM_GROUPS - group) * SERDES_GROUP_SIZE, 0, baddr);
322	if (ret)
323		return (ret);
324
325	baddr_mapped[group] = *baddr;
326
327	return (0);
328}
329
330void
331platform_ipi_send(cpuset_t cpus, u_int ipi)
332{
333
334	pic_ipi_send(cpus, ipi);
335}
336