root/arch/arm/mach-sti/platsmp.c
// SPDX-License-Identifier: GPL-2.0-only
/*
 *  arch/arm/mach-sti/platsmp.c
 *
 * Copyright (C) 2013 STMicroelectronics (R&D) Limited.
 *              http://www.st.com
 *
 * Cloned from linux/arch/arm/mach-vexpress/platsmp.c
 *
 *  Copyright (C) 2002 ARM Ltd.
 *  All Rights Reserved
 */
#include <linux/init.h>
#include <linux/errno.h>
#include <linux/delay.h>
#include <linux/smp.h>
#include <linux/io.h>
#include <linux/of.h>
#include <linux/of_address.h>
#include <linux/memblock.h>

#include <asm/cacheflush.h>
#include <asm/smp_plat.h>
#include <asm/smp_scu.h>

#include "smp.h"

static u32 __iomem *cpu_strt_ptr;

static int sti_boot_secondary(unsigned int cpu, struct task_struct *idle)
{
        unsigned long entry_pa = __pa_symbol(secondary_startup);

        /*
         * Secondary CPU is initialised and started by a U-BOOTROM firmware.
         * Secondary CPU is spinning and waiting for a write at cpu_strt_ptr.
         * Writing secondary_startup address at cpu_strt_ptr makes it to
         * jump directly to secondary_startup().
         */
        __raw_writel(entry_pa, cpu_strt_ptr);

        /* wmb so that data is actually written before cache flush is done */
        smp_wmb();
        sync_cache_w(cpu_strt_ptr);

        return 0;
}

static void __init sti_smp_prepare_cpus(unsigned int max_cpus)
{
        struct device_node *np;
        void __iomem *scu_base;
        u32 release_phys;
        int cpu;

        np = of_find_compatible_node(NULL, NULL, "arm,cortex-a9-scu");

        if (np) {
                scu_base = of_iomap(np, 0);
                scu_enable(scu_base);
                of_node_put(np);
        }

        if (max_cpus <= 1)
                return;

        for_each_possible_cpu(cpu) {

                np = of_get_cpu_node(cpu, NULL);

                if (!np)
                        continue;

                if (of_property_read_u32(np, "cpu-release-addr",
                                                &release_phys)) {
                        pr_err("CPU %d: missing or invalid cpu-release-addr "
                                "property\n", cpu);
                        continue;
                }

                /*
                 * cpu-release-addr is usually configured in SBC DMEM but can
                 * also be in RAM.
                 */

                if (!memblock_is_memory(release_phys))
                        cpu_strt_ptr =
                                ioremap(release_phys, sizeof(release_phys));
                else
                        cpu_strt_ptr =
                                (u32 __iomem *)phys_to_virt(release_phys);

                set_cpu_possible(cpu, true);
        }
}

const struct smp_operations sti_smp_ops __initconst = {
        .smp_prepare_cpus       = sti_smp_prepare_cpus,
        .smp_boot_secondary     = sti_boot_secondary,
};