162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * SMP support for R-Mobile / SH-Mobile - r8a7779 portion 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (C) 2011 Renesas Solutions Corp. 662306a36Sopenharmony_ci * Copyright (C) 2011 Magnus Damm 762306a36Sopenharmony_ci */ 862306a36Sopenharmony_ci#include <linux/kernel.h> 962306a36Sopenharmony_ci#include <linux/init.h> 1062306a36Sopenharmony_ci#include <linux/smp.h> 1162306a36Sopenharmony_ci#include <linux/spinlock.h> 1262306a36Sopenharmony_ci#include <linux/io.h> 1362306a36Sopenharmony_ci#include <linux/delay.h> 1462306a36Sopenharmony_ci#include <linux/soc/renesas/rcar-sysc.h> 1562306a36Sopenharmony_ci 1662306a36Sopenharmony_ci#include <asm/cacheflush.h> 1762306a36Sopenharmony_ci#include <asm/smp_plat.h> 1862306a36Sopenharmony_ci#include <asm/smp_scu.h> 1962306a36Sopenharmony_ci 2062306a36Sopenharmony_ci#include "common.h" 2162306a36Sopenharmony_ci#include "r8a7779.h" 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci#define HPBREG_BASE 0xfe700000 2462306a36Sopenharmony_ci#define AVECR 0x0040 /* ARM Reset Vector Address Register */ 2562306a36Sopenharmony_ci 2662306a36Sopenharmony_ci#define R8A7779_SCU_BASE 0xf0000000 2762306a36Sopenharmony_ci 2862306a36Sopenharmony_cistatic int r8a7779_boot_secondary(unsigned int cpu, struct task_struct *idle) 2962306a36Sopenharmony_ci{ 3062306a36Sopenharmony_ci int ret = -EIO; 3162306a36Sopenharmony_ci 3262306a36Sopenharmony_ci cpu = cpu_logical_map(cpu); 3362306a36Sopenharmony_ci if (cpu) 3462306a36Sopenharmony_ci ret = rcar_sysc_power_up_cpu(cpu); 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_ci return ret; 3762306a36Sopenharmony_ci} 3862306a36Sopenharmony_ci 3962306a36Sopenharmony_cistatic void __init r8a7779_smp_prepare_cpus(unsigned int max_cpus) 4062306a36Sopenharmony_ci{ 4162306a36Sopenharmony_ci void __iomem *base = ioremap(HPBREG_BASE, 0x1000); 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_ci /* Map the reset vector (in headsmp-scu.S, headsmp.S) */ 4462306a36Sopenharmony_ci writel(__pa(shmobile_boot_vector), base + AVECR); 4562306a36Sopenharmony_ci 4662306a36Sopenharmony_ci /* setup r8a7779 specific SCU bits */ 4762306a36Sopenharmony_ci shmobile_smp_scu_prepare_cpus(R8A7779_SCU_BASE, max_cpus); 4862306a36Sopenharmony_ci 4962306a36Sopenharmony_ci iounmap(base); 5062306a36Sopenharmony_ci} 5162306a36Sopenharmony_ci 5262306a36Sopenharmony_ci#ifdef CONFIG_HOTPLUG_CPU 5362306a36Sopenharmony_cistatic int r8a7779_platform_cpu_kill(unsigned int cpu) 5462306a36Sopenharmony_ci{ 5562306a36Sopenharmony_ci int ret = -EIO; 5662306a36Sopenharmony_ci 5762306a36Sopenharmony_ci cpu = cpu_logical_map(cpu); 5862306a36Sopenharmony_ci if (cpu) 5962306a36Sopenharmony_ci ret = rcar_sysc_power_down_cpu(cpu); 6062306a36Sopenharmony_ci 6162306a36Sopenharmony_ci return ret ? ret : 1; 6262306a36Sopenharmony_ci} 6362306a36Sopenharmony_ci 6462306a36Sopenharmony_cistatic int r8a7779_cpu_kill(unsigned int cpu) 6562306a36Sopenharmony_ci{ 6662306a36Sopenharmony_ci if (shmobile_smp_scu_cpu_kill(cpu)) 6762306a36Sopenharmony_ci return r8a7779_platform_cpu_kill(cpu); 6862306a36Sopenharmony_ci 6962306a36Sopenharmony_ci return 0; 7062306a36Sopenharmony_ci} 7162306a36Sopenharmony_ci#endif /* CONFIG_HOTPLUG_CPU */ 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ciconst struct smp_operations r8a7779_smp_ops __initconst = { 7462306a36Sopenharmony_ci .smp_prepare_cpus = r8a7779_smp_prepare_cpus, 7562306a36Sopenharmony_ci .smp_boot_secondary = r8a7779_boot_secondary, 7662306a36Sopenharmony_ci#ifdef CONFIG_HOTPLUG_CPU 7762306a36Sopenharmony_ci .cpu_die = shmobile_smp_scu_cpu_die, 7862306a36Sopenharmony_ci .cpu_kill = r8a7779_cpu_kill, 7962306a36Sopenharmony_ci#endif 8062306a36Sopenharmony_ci}; 81