1b40ce02aSNathan Whitehorn /*- 2b40ce02aSNathan Whitehorn * Copyright (c) 2008-2009 Semihalf, Rafal Jaworowski 3b40ce02aSNathan Whitehorn * All rights reserved. 4b40ce02aSNathan Whitehorn * 5b40ce02aSNathan Whitehorn * Redistribution and use in source and binary forms, with or without 6b40ce02aSNathan Whitehorn * modification, are permitted provided that the following conditions 7b40ce02aSNathan Whitehorn * are met: 8b40ce02aSNathan Whitehorn * 9b40ce02aSNathan Whitehorn * 1. Redistributions of source code must retain the above copyright 10b40ce02aSNathan Whitehorn * notice, this list of conditions and the following disclaimer. 11b40ce02aSNathan Whitehorn * 2. Redistributions in binary form must reproduce the above copyright 12b40ce02aSNathan Whitehorn * notice, this list of conditions and the following disclaimer in the 13b40ce02aSNathan Whitehorn * documentation and/or other materials provided with the distribution. 14b40ce02aSNathan Whitehorn * 15b40ce02aSNathan Whitehorn * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 16b40ce02aSNathan Whitehorn * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 17b40ce02aSNathan Whitehorn * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 18b40ce02aSNathan Whitehorn * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 19b40ce02aSNathan Whitehorn * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 20b40ce02aSNathan Whitehorn * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 21b40ce02aSNathan Whitehorn * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 22b40ce02aSNathan Whitehorn * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 23b40ce02aSNathan Whitehorn * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 24b40ce02aSNathan Whitehorn * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 25b40ce02aSNathan Whitehorn */ 26b40ce02aSNathan Whitehorn 27b40ce02aSNathan Whitehorn #include <sys/cdefs.h> 28b40ce02aSNathan Whitehorn __FBSDID("$FreeBSD$"); 29b40ce02aSNathan Whitehorn 30b40ce02aSNathan Whitehorn #include <sys/param.h> 31b40ce02aSNathan Whitehorn #include <sys/systm.h> 32b40ce02aSNathan Whitehorn #include <sys/kernel.h> 33b40ce02aSNathan Whitehorn #include <sys/bus.h> 34b40ce02aSNathan Whitehorn #include <sys/pcpu.h> 35b40ce02aSNathan Whitehorn #include <sys/proc.h> 36b40ce02aSNathan Whitehorn #include <sys/smp.h> 37b40ce02aSNathan Whitehorn 38b40ce02aSNathan Whitehorn #include <machine/bus.h> 39b40ce02aSNathan Whitehorn #include <machine/cpu.h> 40b40ce02aSNathan Whitehorn #include <machine/hid.h> 41b40ce02aSNathan Whitehorn #include <machine/platform.h> 42b40ce02aSNathan Whitehorn #include <machine/platformvar.h> 43b40ce02aSNathan Whitehorn #include <machine/smp.h> 44b40ce02aSNathan Whitehorn #include <machine/spr.h> 45b40ce02aSNathan Whitehorn #include <machine/vmparam.h> 46b40ce02aSNathan Whitehorn 47*d1d3233eSRafal Jaworowski #include <dev/fdt/fdt_common.h> 48*d1d3233eSRafal Jaworowski #include <dev/ofw/ofw_bus.h> 49*d1d3233eSRafal Jaworowski #include <dev/ofw/ofw_bus_subr.h> 50*d1d3233eSRafal Jaworowski #include <dev/ofw/openfirm.h> 51*d1d3233eSRafal Jaworowski 52b40ce02aSNathan Whitehorn #include <powerpc/mpc85xx/mpc85xx.h> 53b40ce02aSNathan Whitehorn 54b40ce02aSNathan Whitehorn #include "platform_if.h" 55b40ce02aSNathan Whitehorn 5628bb01e5SRafal Jaworowski #ifdef SMP 5728bb01e5SRafal Jaworowski extern void *ap_pcpu; 5828bb01e5SRafal Jaworowski extern uint8_t __boot_page[]; /* Boot page body */ 5928bb01e5SRafal Jaworowski extern uint32_t kernload; /* Kernel physical load address */ 6028bb01e5SRafal Jaworowski #endif 6128bb01e5SRafal Jaworowski 622b7b2d79SRafal Jaworowski static int cpu, maxcpu; 63b40ce02aSNathan Whitehorn 64b40ce02aSNathan Whitehorn static int bare_probe(platform_t); 65b40ce02aSNathan Whitehorn static void bare_mem_regions(platform_t, struct mem_region **phys, int *physsz, 66b40ce02aSNathan Whitehorn struct mem_region **avail, int *availsz); 67b40ce02aSNathan Whitehorn static u_long bare_timebase_freq(platform_t, struct cpuref *cpuref); 68b40ce02aSNathan Whitehorn static int bare_smp_first_cpu(platform_t, struct cpuref *cpuref); 69b40ce02aSNathan Whitehorn static int bare_smp_next_cpu(platform_t, struct cpuref *cpuref); 70b40ce02aSNathan Whitehorn static int bare_smp_get_bsp(platform_t, struct cpuref *cpuref); 71b40ce02aSNathan Whitehorn static int bare_smp_start_cpu(platform_t, struct pcpu *cpu); 72b40ce02aSNathan Whitehorn 73b40ce02aSNathan Whitehorn static platform_method_t bare_methods[] = { 74b40ce02aSNathan Whitehorn PLATFORMMETHOD(platform_probe, bare_probe), 75b40ce02aSNathan Whitehorn PLATFORMMETHOD(platform_mem_regions, bare_mem_regions), 76b40ce02aSNathan Whitehorn PLATFORMMETHOD(platform_timebase_freq, bare_timebase_freq), 77b40ce02aSNathan Whitehorn 78b40ce02aSNathan Whitehorn PLATFORMMETHOD(platform_smp_first_cpu, bare_smp_first_cpu), 79b40ce02aSNathan Whitehorn PLATFORMMETHOD(platform_smp_next_cpu, bare_smp_next_cpu), 80b40ce02aSNathan Whitehorn PLATFORMMETHOD(platform_smp_get_bsp, bare_smp_get_bsp), 81b40ce02aSNathan Whitehorn PLATFORMMETHOD(platform_smp_start_cpu, bare_smp_start_cpu), 82b40ce02aSNathan Whitehorn 83b40ce02aSNathan Whitehorn { 0, 0 } 84b40ce02aSNathan Whitehorn }; 85b40ce02aSNathan Whitehorn 86b40ce02aSNathan Whitehorn static platform_def_t bare_platform = { 87b40ce02aSNathan Whitehorn "bare metal", 88b40ce02aSNathan Whitehorn bare_methods, 89b40ce02aSNathan Whitehorn 0 90b40ce02aSNathan Whitehorn }; 91b40ce02aSNathan Whitehorn 92b40ce02aSNathan Whitehorn PLATFORM_DEF(bare_platform); 93b40ce02aSNathan Whitehorn 94b40ce02aSNathan Whitehorn static int 95b40ce02aSNathan Whitehorn bare_probe(platform_t plat) 96b40ce02aSNathan Whitehorn { 97*d1d3233eSRafal Jaworowski uint32_t ver, sr; 98*d1d3233eSRafal Jaworowski int i, law_max, tgt; 992b7b2d79SRafal Jaworowski 1002b7b2d79SRafal Jaworowski ver = SVR_VER(mfspr(SPR_SVR)); 1012b7b2d79SRafal Jaworowski if (ver == SVR_MPC8572E || ver == SVR_MPC8572) 1022b7b2d79SRafal Jaworowski maxcpu = 2; 1032b7b2d79SRafal Jaworowski else 1042b7b2d79SRafal Jaworowski maxcpu = 1; 105b40ce02aSNathan Whitehorn 106*d1d3233eSRafal Jaworowski /* 107*d1d3233eSRafal Jaworowski * Clear local access windows. Skip DRAM entries, so we don't shoot 108*d1d3233eSRafal Jaworowski * ourselves in the foot. 109*d1d3233eSRafal Jaworowski */ 110*d1d3233eSRafal Jaworowski law_max = law_getmax(); 111*d1d3233eSRafal Jaworowski for (i = 0; i < law_max; i++) { 112*d1d3233eSRafal Jaworowski sr = ccsr_read4(OCP85XX_LAWSR(i)); 113*d1d3233eSRafal Jaworowski if ((sr & 0x80000000) == 0) 114*d1d3233eSRafal Jaworowski continue; 115*d1d3233eSRafal Jaworowski tgt = (sr & 0x01f00000) >> 20; 116*d1d3233eSRafal Jaworowski if (tgt == OCP85XX_TGTIF_RAM1 || tgt == OCP85XX_TGTIF_RAM2 || 117*d1d3233eSRafal Jaworowski tgt == OCP85XX_TGTIF_RAM_INTL) 118*d1d3233eSRafal Jaworowski continue; 119*d1d3233eSRafal Jaworowski 120*d1d3233eSRafal Jaworowski ccsr_write4(OCP85XX_LAWSR(i), sr & 0x7fffffff); 121*d1d3233eSRafal Jaworowski } 122*d1d3233eSRafal Jaworowski 123b40ce02aSNathan Whitehorn return (BUS_PROBE_GENERIC); 124b40ce02aSNathan Whitehorn } 125b40ce02aSNathan Whitehorn 126b40ce02aSNathan Whitehorn #define MEM_REGIONS 8 127b40ce02aSNathan Whitehorn static struct mem_region avail_regions[MEM_REGIONS]; 128b40ce02aSNathan Whitehorn 129b40ce02aSNathan Whitehorn void 130b40ce02aSNathan Whitehorn bare_mem_regions(platform_t plat, struct mem_region **phys, int *physsz, 131b40ce02aSNathan Whitehorn struct mem_region **avail, int *availsz) 132b40ce02aSNathan Whitehorn { 133*d1d3233eSRafal Jaworowski uint32_t memsize; 134*d1d3233eSRafal Jaworowski int i, rv; 135b40ce02aSNathan Whitehorn 136*d1d3233eSRafal Jaworowski rv = fdt_get_mem_regions(avail_regions, availsz, &memsize); 137*d1d3233eSRafal Jaworowski 138*d1d3233eSRafal Jaworowski if (rv != 0) 139*d1d3233eSRafal Jaworowski return; 140*d1d3233eSRafal Jaworowski 141*d1d3233eSRafal Jaworowski for (i = 0; i < *availsz; i++) { 142*d1d3233eSRafal Jaworowski if (avail_regions[i].mr_start < 1048576) { 143*d1d3233eSRafal Jaworowski avail_regions[i].mr_size = 144*d1d3233eSRafal Jaworowski avail_regions[i].mr_size - 145*d1d3233eSRafal Jaworowski (1048576 - avail_regions[i].mr_start); 146b40ce02aSNathan Whitehorn avail_regions[i].mr_start = 1048576; 147b40ce02aSNathan Whitehorn } 148b40ce02aSNathan Whitehorn } 149b40ce02aSNathan Whitehorn *avail = avail_regions; 150b40ce02aSNathan Whitehorn 151b40ce02aSNathan Whitehorn /* On the bare metal platform phys == avail memory */ 152b40ce02aSNathan Whitehorn *physsz = *availsz; 153b40ce02aSNathan Whitehorn *phys = *avail; 154b40ce02aSNathan Whitehorn } 155b40ce02aSNathan Whitehorn 156b40ce02aSNathan Whitehorn static u_long 157b40ce02aSNathan Whitehorn bare_timebase_freq(platform_t plat, struct cpuref *cpuref) 158b40ce02aSNathan Whitehorn { 159b40ce02aSNathan Whitehorn u_long ticks = -1; 160*d1d3233eSRafal Jaworowski phandle_t cpus, child; 161*d1d3233eSRafal Jaworowski pcell_t freq; 162b40ce02aSNathan Whitehorn 163*d1d3233eSRafal Jaworowski if ((cpus = OF_finddevice("/cpus")) == 0) 164*d1d3233eSRafal Jaworowski goto out; 165*d1d3233eSRafal Jaworowski 166*d1d3233eSRafal Jaworowski if ((child = OF_child(cpus)) == 0) 167*d1d3233eSRafal Jaworowski goto out; 168*d1d3233eSRafal Jaworowski 169*d1d3233eSRafal Jaworowski if (OF_getprop(child, "bus-frequency", (void *)&freq, 170*d1d3233eSRafal Jaworowski sizeof(freq)) <= 0) 171*d1d3233eSRafal Jaworowski goto out; 172b40ce02aSNathan Whitehorn /* 173b40ce02aSNathan Whitehorn * Time Base and Decrementer are updated every 8 CCB bus clocks. 174b40ce02aSNathan Whitehorn * HID0[SEL_TBCLK] = 0 175b40ce02aSNathan Whitehorn */ 176*d1d3233eSRafal Jaworowski ticks = freq / 8; 177*d1d3233eSRafal Jaworowski out: 178b40ce02aSNathan Whitehorn if (ticks <= 0) 179b40ce02aSNathan Whitehorn panic("Unable to determine timebase frequency!"); 180b40ce02aSNathan Whitehorn 181b40ce02aSNathan Whitehorn return (ticks); 182b40ce02aSNathan Whitehorn } 183b40ce02aSNathan Whitehorn 184b40ce02aSNathan Whitehorn static int 185b40ce02aSNathan Whitehorn bare_smp_first_cpu(platform_t plat, struct cpuref *cpuref) 186b40ce02aSNathan Whitehorn { 187b40ce02aSNathan Whitehorn 188b40ce02aSNathan Whitehorn cpu = 0; 189b40ce02aSNathan Whitehorn cpuref->cr_cpuid = cpu; 190b40ce02aSNathan Whitehorn cpuref->cr_hwref = cpuref->cr_cpuid; 191b40ce02aSNathan Whitehorn if (bootverbose) 192b40ce02aSNathan Whitehorn printf("powerpc_smp_first_cpu: cpuid %d\n", cpuref->cr_cpuid); 193b40ce02aSNathan Whitehorn cpu++; 194b40ce02aSNathan Whitehorn 195b40ce02aSNathan Whitehorn return (0); 196b40ce02aSNathan Whitehorn } 197b40ce02aSNathan Whitehorn 198b40ce02aSNathan Whitehorn static int 199b40ce02aSNathan Whitehorn bare_smp_next_cpu(platform_t plat, struct cpuref *cpuref) 200b40ce02aSNathan Whitehorn { 201b40ce02aSNathan Whitehorn 2022b7b2d79SRafal Jaworowski if (cpu >= maxcpu) 203b40ce02aSNathan Whitehorn return (ENOENT); 204b40ce02aSNathan Whitehorn 205b40ce02aSNathan Whitehorn cpuref->cr_cpuid = cpu++; 206b40ce02aSNathan Whitehorn cpuref->cr_hwref = cpuref->cr_cpuid; 207b40ce02aSNathan Whitehorn if (bootverbose) 208b40ce02aSNathan Whitehorn printf("powerpc_smp_next_cpu: cpuid %d\n", cpuref->cr_cpuid); 209b40ce02aSNathan Whitehorn 210b40ce02aSNathan Whitehorn return (0); 211b40ce02aSNathan Whitehorn } 212b40ce02aSNathan Whitehorn 213b40ce02aSNathan Whitehorn static int 214b40ce02aSNathan Whitehorn bare_smp_get_bsp(platform_t plat, struct cpuref *cpuref) 215b40ce02aSNathan Whitehorn { 216b40ce02aSNathan Whitehorn 217b40ce02aSNathan Whitehorn cpuref->cr_cpuid = mfspr(SPR_PIR); 218b40ce02aSNathan Whitehorn cpuref->cr_hwref = cpuref->cr_cpuid; 219b40ce02aSNathan Whitehorn 220b40ce02aSNathan Whitehorn return (0); 221b40ce02aSNathan Whitehorn } 222b40ce02aSNathan Whitehorn 223b40ce02aSNathan Whitehorn static int 224b40ce02aSNathan Whitehorn bare_smp_start_cpu(platform_t plat, struct pcpu *pc) 225b40ce02aSNathan Whitehorn { 22628bb01e5SRafal Jaworowski #ifdef SMP 22728bb01e5SRafal Jaworowski uint32_t bptr, eebpcr; 22828bb01e5SRafal Jaworowski int timeout; 229b40ce02aSNathan Whitehorn 23028bb01e5SRafal Jaworowski eebpcr = ccsr_read4(OCP85XX_EEBPCR); 23128bb01e5SRafal Jaworowski if ((eebpcr & (pc->pc_cpumask << 24)) != 0) { 23228bb01e5SRafal Jaworowski printf("%s: CPU=%d already out of hold-off state!\n", 23328bb01e5SRafal Jaworowski __func__, pc->pc_cpuid); 23428bb01e5SRafal Jaworowski return (ENXIO); 23528bb01e5SRafal Jaworowski } 23628bb01e5SRafal Jaworowski 23728bb01e5SRafal Jaworowski ap_pcpu = pc; 23828bb01e5SRafal Jaworowski __asm __volatile("msync; isync"); 23928bb01e5SRafal Jaworowski 24028bb01e5SRafal Jaworowski /* 24128bb01e5SRafal Jaworowski * Set BPTR to the physical address of the boot page 24228bb01e5SRafal Jaworowski */ 24328bb01e5SRafal Jaworowski bptr = ((uint32_t)__boot_page - KERNBASE) + kernload; 24428bb01e5SRafal Jaworowski ccsr_write4(OCP85XX_BPTR, (bptr >> 12) | 0x80000000); 24528bb01e5SRafal Jaworowski 24628bb01e5SRafal Jaworowski /* 24728bb01e5SRafal Jaworowski * Release AP from hold-off state 24828bb01e5SRafal Jaworowski */ 24928bb01e5SRafal Jaworowski eebpcr |= (pc->pc_cpumask << 24); 25028bb01e5SRafal Jaworowski ccsr_write4(OCP85XX_EEBPCR, eebpcr); 25128bb01e5SRafal Jaworowski __asm __volatile("isync; msync"); 25228bb01e5SRafal Jaworowski 25328bb01e5SRafal Jaworowski timeout = 500; 25428bb01e5SRafal Jaworowski while (!pc->pc_awake && timeout--) 25528bb01e5SRafal Jaworowski DELAY(1000); /* wait 1ms */ 25628bb01e5SRafal Jaworowski 25728bb01e5SRafal Jaworowski return ((pc->pc_awake) ? 0 : EBUSY); 25828bb01e5SRafal Jaworowski #else 259b40ce02aSNathan Whitehorn /* No SMP support */ 260b40ce02aSNathan Whitehorn return (ENXIO); 26128bb01e5SRafal Jaworowski #endif 262b40ce02aSNathan Whitehorn } 263