2019-06-04 10:11:33 +02:00
/* SPDX-License-Identifier: GPL-2.0-only */
2013-01-18 15:12:16 +05:30
/*
* Copyright ( C ) 2004 , 2007 - 2010 , 2011 - 2012 Synopsys , Inc . ( www . synopsys . com )
*/
# ifndef __ASM_ARC_SMP_H
# define __ASM_ARC_SMP_H
2013-01-18 15:12:23 +05:30
# ifdef CONFIG_SMP
# include <linux/types.h>
# include <linux/init.h>
# include <linux/threads.h>
# define raw_smp_processor_id() (current_thread_info()->cpu)
/* including cpumask.h leads to cyclic deps hence this Forward declaration */
struct cpumask ;
/*
* APIs provided by arch SMP code to generic code
*/
extern void arch_send_call_function_single_ipi ( int cpu ) ;
extern void arch_send_call_function_ipi_mask ( const struct cpumask * mask ) ;
/*
* APIs provided by arch SMP code to rest of arch code
*/
extern void __init smp_init_cpus ( void ) ;
2013-10-23 10:16:38 +08:00
extern void first_lines_of_secondary ( void ) ;
2013-01-18 15:12:26 +05:30
extern const char * arc_platform_smp_cpuinfo ( void ) ;
2013-01-18 15:12:23 +05:30
/*
* API expected BY platform smp code ( FROM arch smp code )
*
* smp_ipi_irq_setup :
2016-11-08 10:08:31 +03:00
* Takes @ cpu and @ hwirq to which the arch - common ISR is hooked up
2013-01-18 15:12:23 +05:30
*/
2016-11-08 10:08:31 +03:00
extern int smp_ipi_irq_setup ( int cpu , irq_hw_number_t hwirq ) ;
2013-01-18 15:12:23 +05:30
/*
2013-01-18 15:12:26 +05:30
* struct plat_smp_ops - SMP callbacks provided by platform to ARC SMP
2013-01-18 15:12:23 +05:30
*
2013-01-18 15:12:26 +05:30
* @ info : SoC SMP specific info for / proc / cpuinfo etc
2015-10-12 16:28:55 +05:30
* @ init_early_smp : A SMP specific h / w block can init itself
* Could be common across platforms so not covered by
* mach_desc - > init_early ( )
2015-12-16 03:10:27 +02:00
* @ init_per_cpu : Called for each core so SMP h / w block driver can do
2015-10-14 14:38:02 +05:30
* any needed setup per cpu ( e . g . IPI request )
2013-01-18 15:12:26 +05:30
* @ cpu_kick : For Master to kickstart a cpu ( optionally at a PC )
2013-11-25 12:59:03 +05:30
* @ ipi_send : To send IPI to a @ cpu
2013-11-25 14:37:14 +05:30
* @ ips_clear : To clear IPI received at @ irq
2013-01-18 15:12:23 +05:30
*/
2013-01-18 15:12:26 +05:30
struct plat_smp_ops {
const char * info ;
2015-10-12 16:28:55 +05:30
void ( * init_early_smp ) ( void ) ;
2015-12-16 03:10:27 +02:00
void ( * init_per_cpu ) ( int cpu ) ;
2013-01-18 15:12:26 +05:30
void ( * cpu_kick ) ( int cpu , unsigned long pc ) ;
2013-11-25 12:59:03 +05:30
void ( * ipi_send ) ( int cpu ) ;
2013-11-25 14:37:14 +05:30
void ( * ipi_clear ) ( int irq ) ;
2013-01-18 15:12:26 +05:30
} ;
/* TBD: stop exporting it for direct population by platform */
extern struct plat_smp_ops plat_smp_ops ;
2013-01-18 15:12:23 +05:30
2014-09-04 10:57:33 +05:30
# else /* CONFIG_SMP */
static inline void smp_init_cpus ( void ) { }
static inline const char * arc_platform_smp_cpuinfo ( void )
{
return " " ;
}
# endif /* !CONFIG_SMP */
2013-01-18 15:12:23 +05:30
2013-01-18 15:12:16 +05:30
/*
* ARC700 doesn ' t support atomic Read - Modify - Write ops .
* Originally Interrupts had to be disabled around code to gaurantee atomicity .
* The LLOCK / SCOND insns allow writing interrupt - hassle - free based atomic ops
* based on retry - if - irq - in - atomic ( with hardware assist ) .
* However despite these , we provide the IRQ disabling variant
*
* ( 1 ) These insn were introduced only in 4.10 release . So for older released
* support needed .
2013-01-18 15:12:23 +05:30
*
2016-05-21 13:45:35 +02:00
* ( 2 ) In a SMP setup , the LLOCK / SCOND atomicity across CPUs needs to be
2013-01-18 15:12:23 +05:30
* gaurantted by the platform ( not something which core handles ) .
* Assuming a platform won ' t , SMP Linux needs to use spinlocks + local IRQ
* disabling for atomicity .
*
* However exported spinlock API is not usable due to cyclic hdr deps
* ( even after system . h disintegration upstream )
* asm / bitops . h - > linux / spinlock . h - > linux / preempt . h
* - > linux / thread_info . h - > linux / bitops . h - > asm / bitops . h
*
* So the workaround is to use the lowest level arch spinlock API .
* The exported spinlock API is smart enough to be NOP for ! CONFIG_SMP ,
* but same is not true for ARCH backend , hence the need for 2 variants
2013-01-18 15:12:16 +05:30
*/
# ifndef CONFIG_ARC_HAS_LLSC
# include <linux/irqflags.h>
2013-01-18 15:12:23 +05:30
# ifdef CONFIG_SMP
# include <asm/spinlock.h>
extern arch_spinlock_t smp_atomic_ops_lock ;
# define atomic_ops_lock(flags) do { \
local_irq_save ( flags ) ; \
arch_spin_lock ( & smp_atomic_ops_lock ) ; \
} while ( 0 )
# define atomic_ops_unlock(flags) do { \
arch_spin_unlock ( & smp_atomic_ops_lock ) ; \
local_irq_restore ( flags ) ; \
} while ( 0 )
# else /* !CONFIG_SMP */
2013-01-18 15:12:16 +05:30
# define atomic_ops_lock(flags) local_irq_save(flags)
# define atomic_ops_unlock(flags) local_irq_restore(flags)
2013-01-18 15:12:23 +05:30
# endif /* !CONFIG_SMP */
2013-01-18 15:12:16 +05:30
# endif /* !CONFIG_ARC_HAS_LLSC */
# endif