| // SPDX-License-Identifier: GPL-2.0 |
| /* |
| * ip30-smp.c: SMP on IP30 architecture. |
| * Based off of the original IP30 SMP code, with inspiration from ip27-smp.c |
| * and smp-bmips.c. |
| * |
| * Copyright (C) 2005-2007 Stanislaw Skowronek <skylark@unaligned.org> |
| * 2006-2007, 2014-2015 Joshua Kinard <kumba@gentoo.org> |
| * 2009 Johannes Dickgreber <tanzy@gmx.de> |
| */ |
| |
| #include <linux/init.h> |
| #include <linux/sched.h> |
| #include <linux/sched/task_stack.h> |
| |
| #include <asm/time.h> |
| #include <asm/sgi/heart.h> |
| |
| #include "ip30-common.h" |
| |
| #define MPCONF_MAGIC 0xbaddeed2 |
| #define MPCONF_ADDR 0xa800000000000600L |
| #define MPCONF_SIZE 0x80 |
| #define MPCONF(x) (MPCONF_ADDR + (x) * MPCONF_SIZE) |
| |
| /* HEART can theoretically do 4 CPUs, but only 2 are physically possible */ |
| #define MP_NCPU 2 |
| |
| struct mpconf { |
| u32 magic; |
| u32 prid; |
| u32 physid; |
| u32 virtid; |
| u32 scachesz; |
| u16 fanloads; |
| u16 res; |
| void *launch; |
| void *rendezvous; |
| u64 res2[3]; |
| void *stackaddr; |
| void *lnch_parm; |
| void *rndv_parm; |
| u32 idleflag; |
| }; |
| |
| static void ip30_smp_send_ipi_single(int cpu, u32 action) |
| { |
| int irq; |
| |
| switch (action) { |
| case SMP_RESCHEDULE_YOURSELF: |
| irq = HEART_L2_INT_RESCHED_CPU_0; |
| break; |
| case SMP_CALL_FUNCTION: |
| irq = HEART_L2_INT_CALL_CPU_0; |
| break; |
| default: |
| panic("IP30: Unknown action value in %s!\n", __func__); |
| } |
| |
| irq += cpu; |
| |
| /* Poke the other CPU -- it's got mail! */ |
| heart_write(BIT_ULL(irq), &heart_regs->set_isr); |
| } |
| |
| static void ip30_smp_send_ipi_mask(const struct cpumask *mask, u32 action) |
| { |
| u32 i; |
| |
| for_each_cpu(i, mask) |
| ip30_smp_send_ipi_single(i, action); |
| } |
| |
| static void __init ip30_smp_setup(void) |
| { |
| int i; |
| int ncpu = 0; |
| struct mpconf *mpc; |
| |
| init_cpu_possible(cpumask_of(0)); |
| |
| /* Scan the MPCONF structure and enumerate available CPUs. */ |
| for (i = 0; i < MP_NCPU; i++) { |
| mpc = (struct mpconf *)MPCONF(i); |
| if (mpc->magic == MPCONF_MAGIC) { |
| set_cpu_possible(i, true); |
| __cpu_number_map[i] = ++ncpu; |
| __cpu_logical_map[ncpu] = i; |
| pr_info("IP30: Slot: %d, PrID: %.8x, PhyID: %d, VirtID: %d\n", |
| i, mpc->prid, mpc->physid, mpc->virtid); |
| } |
| } |
| pr_info("IP30: Detected %d CPU(s) present.\n", ncpu); |
| |
| /* |
| * Set the coherency algorithm to '5' (cacheable coherent |
| * exclusive on write). This is needed on IP30 SMP, especially |
| * for R14000 CPUs, otherwise, instruction bus errors will |
| * occur upon reaching userland. |
| */ |
| change_c0_config(CONF_CM_CMASK, CONF_CM_CACHABLE_COW); |
| } |
| |
| static void __init ip30_smp_prepare_cpus(unsigned int max_cpus) |
| { |
| /* nothing to do here */ |
| } |
| |
| static int __init ip30_smp_boot_secondary(int cpu, struct task_struct *idle) |
| { |
| struct mpconf *mpc = (struct mpconf *)MPCONF(cpu); |
| |
| /* Stack pointer (sp). */ |
| mpc->stackaddr = (void *)__KSTK_TOS(idle); |
| |
| /* Global pointer (gp). */ |
| mpc->lnch_parm = task_thread_info(idle); |
| |
| mb(); /* make sure stack and lparm are written */ |
| |
| /* Boot CPUx. */ |
| mpc->launch = smp_bootstrap; |
| |
| /* CPUx now executes smp_bootstrap, then ip30_smp_finish */ |
| return 0; |
| } |
| |
| static void __init ip30_smp_init_cpu(void) |
| { |
| ip30_per_cpu_init(); |
| } |
| |
| static void __init ip30_smp_finish(void) |
| { |
| enable_percpu_irq(get_c0_compare_int(), IRQ_TYPE_NONE); |
| local_irq_enable(); |
| } |
| |
| struct plat_smp_ops __read_mostly ip30_smp_ops = { |
| .send_ipi_single = ip30_smp_send_ipi_single, |
| .send_ipi_mask = ip30_smp_send_ipi_mask, |
| .smp_setup = ip30_smp_setup, |
| .prepare_cpus = ip30_smp_prepare_cpus, |
| .boot_secondary = ip30_smp_boot_secondary, |
| .init_secondary = ip30_smp_init_cpu, |
| .smp_finish = ip30_smp_finish, |
| .prepare_boot_cpu = ip30_smp_init_cpu, |
| }; |