2019-05-27 08:55:21 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2017-07-10 18:00:26 -07:00
/*
* SMP initialisation and IPI support
* Based on arch / arm64 / kernel / smp . c
*
* Copyright ( C ) 2012 ARM Ltd .
* Copyright ( C ) 2015 Regents of the University of California
* Copyright ( C ) 2017 SiFive
*/
2019-06-27 12:53:00 -07:00
# include <linux/arch_topology.h>
2017-07-10 18:00:26 -07:00
# include <linux/module.h>
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/mm.h>
# include <linux/sched.h>
# include <linux/kernel_stat.h>
# include <linux/notifier.h>
# include <linux/cpu.h>
# include <linux/percpu.h>
# include <linux/delay.h>
# include <linux/err.h>
# include <linux/irq.h>
# include <linux/of.h>
# include <linux/sched/task_stack.h>
2018-10-02 12:15:02 -07:00
# include <linux/sched/mm.h>
2017-07-10 18:00:26 -07:00
# include <asm/irq.h>
# include <asm/mmu_context.h>
# include <asm/tlbflush.h>
# include <asm/sections.h>
# include <asm/sbi.h>
2019-10-17 15:00:17 -07:00
# include "head.h"
2017-07-10 18:00:26 -07:00
void * __cpu_up_stack_pointer [ NR_CPUS ] ;
void * __cpu_up_task_pointer [ NR_CPUS ] ;
2019-02-22 11:41:35 -08:00
static DECLARE_COMPLETION ( cpu_running ) ;
2017-07-10 18:00:26 -07:00
void __init smp_prepare_boot_cpu ( void )
{
2019-06-27 12:53:00 -07:00
init_cpu_topology ( ) ;
2017-07-10 18:00:26 -07:00
}
void __init smp_prepare_cpus ( unsigned int max_cpus )
{
2019-04-24 14:47:59 -07:00
int cpuid ;
/* This covers non-smp usecase mandated by "nosmp" option */
if ( max_cpus = = 0 )
return ;
for_each_possible_cpu ( cpuid ) {
if ( cpuid = = smp_processor_id ( ) )
continue ;
set_cpu_present ( cpuid , true ) ;
}
2017-07-10 18:00:26 -07:00
}
void __init setup_smp ( void )
{
2019-01-18 15:03:08 +01:00
struct device_node * dn ;
2018-10-02 12:15:01 -07:00
int hart ;
bool found_boot_cpu = false ;
2018-10-02 12:15:05 -07:00
int cpuid = 1 ;
2017-07-10 18:00:26 -07:00
2019-01-18 15:03:08 +01:00
for_each_of_cpu_node ( dn ) {
2018-10-02 12:15:00 -07:00
hart = riscv_of_processor_hartid ( dn ) ;
2019-01-07 15:16:35 +01:00
if ( hart < 0 )
2018-10-02 12:15:05 -07:00
continue ;
if ( hart = = cpuid_to_hartid_map ( 0 ) ) {
BUG_ON ( found_boot_cpu ) ;
found_boot_cpu = 1 ;
continue ;
2017-07-10 18:00:26 -07:00
}
2019-02-22 11:41:39 -08:00
if ( cpuid > = NR_CPUS ) {
pr_warn ( " Invalid cpuid [%d] for hartid [%d] \n " ,
cpuid , hart ) ;
break ;
}
2018-10-02 12:15:05 -07:00
cpuid_to_hartid_map ( cpuid ) = hart ;
cpuid + + ;
2017-07-10 18:00:26 -07:00
}
2018-10-02 12:15:01 -07:00
BUG_ON ( ! found_boot_cpu ) ;
2019-04-24 14:48:00 -07:00
if ( cpuid > nr_cpu_ids )
pr_warn ( " Total number of cpus [%d] is greater than nr_cpus option value [%d] \n " ,
cpuid , nr_cpu_ids ) ;
for ( cpuid = 1 ; cpuid < nr_cpu_ids ; cpuid + + ) {
if ( cpuid_to_hartid_map ( cpuid ) ! = INVALID_HARTID )
set_cpu_possible ( cpuid , true ) ;
}
2017-07-10 18:00:26 -07:00
}
int __cpu_up ( unsigned int cpu , struct task_struct * tidle )
{
2019-02-22 11:41:35 -08:00
int ret = 0 ;
2018-10-02 12:15:05 -07:00
int hartid = cpuid_to_hartid_map ( cpu ) ;
2017-07-10 18:00:26 -07:00
tidle - > thread_info . cpu = cpu ;
/*
* On RISC - V systems , all harts boot on their own accord . Our _start
* selects the first hart to boot the kernel and causes the remainder
* of the harts to spin in a loop waiting for their stack pointer to be
* setup by that main hart . Writing __cpu_up_stack_pointer signals to
* the spinning harts that they can continue the boot process .
*/
smp_mb ( ) ;
2018-10-02 12:15:05 -07:00
WRITE_ONCE ( __cpu_up_stack_pointer [ hartid ] ,
2018-10-02 12:15:03 -07:00
task_stack_page ( tidle ) + THREAD_SIZE ) ;
2018-10-02 12:15:05 -07:00
WRITE_ONCE ( __cpu_up_task_pointer [ hartid ] , tidle ) ;
2017-07-10 18:00:26 -07:00
2019-02-22 11:41:35 -08:00
lockdep_assert_held ( & cpu_running ) ;
wait_for_completion_timeout ( & cpu_running ,
msecs_to_jiffies ( 1000 ) ) ;
2017-07-10 18:00:26 -07:00
2019-02-22 11:41:35 -08:00
if ( ! cpu_online ( cpu ) ) {
pr_crit ( " CPU%u: failed to come online \n " , cpu ) ;
ret = - EIO ;
}
return ret ;
2017-07-10 18:00:26 -07:00
}
void __init smp_cpus_done ( unsigned int max_cpus )
{
}
/*
* C entry point for a secondary processor .
*/
asmlinkage void __init smp_callin ( void )
{
struct mm_struct * mm = & init_mm ;
/* All kernel threads share the same mm context. */
2018-10-02 12:15:02 -07:00
mmgrab ( mm ) ;
2017-07-10 18:00:26 -07:00
current - > active_mm = mm ;
trap_init ( ) ;
notify_cpu_starting ( smp_processor_id ( ) ) ;
2019-06-27 12:53:00 -07:00
update_siblings_masks ( smp_processor_id ( ) ) ;
2017-07-10 18:00:26 -07:00
set_cpu_online ( smp_processor_id ( ) , 1 ) ;
2018-10-02 12:14:57 -07:00
/*
* Remote TLB flushes are ignored while the CPU is offline , so emit
* a local TLB flush right now just in case .
*/
2017-07-10 18:00:26 -07:00
local_flush_tlb_all ( ) ;
2019-02-22 11:41:35 -08:00
complete ( & cpu_running ) ;
2018-10-02 12:14:58 -07:00
/*
* Disable preemption before enabling interrupts , so we don ' t try to
* schedule a CPU that hasn ' t actually started yet .
*/
2017-07-10 18:00:26 -07:00
preempt_disable ( ) ;
2018-10-02 12:14:58 -07:00
local_irq_enable ( ) ;
2017-07-10 18:00:26 -07:00
cpu_startup_entry ( CPUHP_AP_ONLINE_IDLE ) ;
}