| /* | 
 |  * Copyright (C) 2016 Neil Armstrong <narmstrong@baylibre.com> | 
 |  * Copyright (C) 2013 Ma Haijun <mahaijuns@gmail.com> | 
 |  * Copyright (C) 2002 ARM Ltd. | 
 |  * All Rights Reserved | 
 |  * | 
 |  * This program is free software; you can redistribute it and/or modify | 
 |  * it under the terms of the GNU General Public License version 2 as | 
 |  * published by the Free Software Foundation. | 
 |  */ | 
 | #include <linux/io.h> | 
 | #include <linux/delay.h> | 
 | #include <linux/of.h> | 
 | #include <linux/of_address.h> | 
 |  | 
 | #include <asm/cacheflush.h> | 
 | #include <asm/cp15.h> | 
 | #include <asm/smp_plat.h> | 
 | #include <asm/smp_scu.h> | 
 |  | 
 | extern void ox820_secondary_startup(void); | 
 | extern void ox820_cpu_die(unsigned int cpu); | 
 |  | 
 | static void __iomem *cpu_ctrl; | 
 | static void __iomem *gic_cpu_ctrl; | 
 |  | 
 | #define HOLDINGPEN_CPU_OFFSET		0xc8 | 
 | #define HOLDINGPEN_LOCATION_OFFSET	0xc4 | 
 |  | 
 | #define GIC_NCPU_OFFSET(cpu)		(0x100 + (cpu)*0x100) | 
 | #define GIC_CPU_CTRL			0x00 | 
 | #define GIC_CPU_CTRL_ENABLE		1 | 
 |  | 
 | int __init ox820_boot_secondary(unsigned int cpu, struct task_struct *idle) | 
 | { | 
 | 	/* | 
 | 	 * Write the address of secondary startup into the | 
 | 	 * system-wide flags register. The BootMonitor waits | 
 | 	 * until it receives a soft interrupt, and then the | 
 | 	 * secondary CPU branches to this address. | 
 | 	 */ | 
 | 	writel(virt_to_phys(ox820_secondary_startup), | 
 | 			cpu_ctrl + HOLDINGPEN_LOCATION_OFFSET); | 
 |  | 
 | 	writel(cpu, cpu_ctrl + HOLDINGPEN_CPU_OFFSET); | 
 |  | 
 | 	/* | 
 | 	 * Enable GIC cpu interface in CPU Interface Control Register | 
 | 	 */ | 
 | 	writel(GIC_CPU_CTRL_ENABLE, | 
 | 		gic_cpu_ctrl + GIC_NCPU_OFFSET(cpu) + GIC_CPU_CTRL); | 
 |  | 
 | 	/* | 
 | 	 * Send the secondary CPU a soft interrupt, thereby causing | 
 | 	 * the boot monitor to read the system wide flags register, | 
 | 	 * and branch to the address found there. | 
 | 	 */ | 
 | 	arch_send_wakeup_ipi_mask(cpumask_of(cpu)); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void __init ox820_smp_prepare_cpus(unsigned int max_cpus) | 
 | { | 
 | 	struct device_node *np; | 
 | 	void __iomem *scu_base; | 
 |  | 
 | 	np = of_find_compatible_node(NULL, NULL, "arm,arm11mp-scu"); | 
 | 	scu_base = of_iomap(np, 0); | 
 | 	of_node_put(np); | 
 | 	if (!scu_base) | 
 | 		return; | 
 |  | 
 | 	/* Remap CPU Interrupt Interface Registers */ | 
 | 	np = of_find_compatible_node(NULL, NULL, "arm,arm11mp-gic"); | 
 | 	gic_cpu_ctrl = of_iomap(np, 1); | 
 | 	of_node_put(np); | 
 | 	if (!gic_cpu_ctrl) | 
 | 		goto unmap_scu; | 
 |  | 
 | 	np = of_find_compatible_node(NULL, NULL, "oxsemi,ox820-sys-ctrl"); | 
 | 	cpu_ctrl = of_iomap(np, 0); | 
 | 	of_node_put(np); | 
 | 	if (!cpu_ctrl) | 
 | 		goto unmap_scu; | 
 |  | 
 | 	scu_enable(scu_base); | 
 | 	flush_cache_all(); | 
 |  | 
 | unmap_scu: | 
 | 	iounmap(scu_base); | 
 | } | 
 |  | 
 | static const struct smp_operations ox820_smp_ops __initconst = { | 
 | 	.smp_prepare_cpus	= ox820_smp_prepare_cpus, | 
 | 	.smp_boot_secondary	= ox820_boot_secondary, | 
 | #ifdef CONFIG_HOTPLUG_CPU | 
 | 	.cpu_die		= ox820_cpu_die, | 
 | #endif | 
 | }; | 
 |  | 
 | CPU_METHOD_OF_DECLARE(ox820_smp, "oxsemi,ox820-smp", &ox820_smp_ops); |