2005-04-16 15:20:36 -07:00
/*
* include / asm - s390 / smp . h
*
* S390 version
* Copyright ( C ) 1999 IBM Deutschland Entwicklung GmbH , IBM Corporation
* Author ( s ) : Denis Joseph Barrow ( djbarrow @ de . ibm . com , barrow_dj @ yahoo . com ) ,
* Martin Schwidefsky ( schwidefsky @ de . ibm . com )
* Heiko Carstens ( heiko . carstens @ de . ibm . com )
*/
# ifndef __ASM_SMP_H
# define __ASM_SMP_H
# include <linux/threads.h>
# include <linux/cpumask.h>
# include <linux/bitops.h>
# if defined(__KERNEL__) && defined(CONFIG_SMP) && !defined(__ASSEMBLY__)
# include <asm/lowcore.h>
# include <asm/sigp.h>
2006-12-04 15:40:33 +01:00
# include <asm/ptrace.h>
2005-04-16 15:20:36 -07:00
/*
s390 specific smp . c headers
*/
typedef struct
{
int intresting ;
sigp_ccode ccode ;
__u32 status ;
__u16 cpu ;
} sigp_info ;
2007-02-05 21:16:47 +01:00
extern void machine_restart_smp ( char * ) ;
extern void machine_halt_smp ( void ) ;
extern void machine_power_off_smp ( void ) ;
2005-04-16 15:20:36 -07:00
# define NO_PROC_ID 0xFF /* No processor magic marker */
/*
* This magic constant controls our willingness to transfer
* a process across CPUs . Such a transfer incurs misses on the L1
* cache , and on a P6 or P5 with multiple L2 caches L2 hits . My
* gut feeling is this will vary by board in value . For a board
* with separate L2 cache it probably depends also on the RSS , and
* for a board with shared L2 cache it ought to decay fast as other
* processes are run .
*/
# define PROC_CHANGE_PENALTY 20 /* Schedule penalty */
2005-06-21 17:14:34 -07:00
# define raw_smp_processor_id() (S390_lowcore.cpu_data.cpu_nr)
2005-04-16 15:20:36 -07:00
2005-11-08 21:34:42 -08:00
static inline __u16 hard_smp_processor_id ( void )
2005-04-16 15:20:36 -07:00
{
__u16 cpu_address ;
2006-09-28 16:56:43 +02:00
asm volatile ( " stap %0 " : " =m " ( cpu_address ) ) ;
2005-04-16 15:20:36 -07:00
return cpu_address ;
}
/*
* returns 1 if cpu is in stopped / check stopped state or not operational
* returns 0 otherwise
*/
static inline int
smp_cpu_not_running ( int cpu )
{
__u32 status ;
switch ( signal_processor_ps ( & status , 0 , cpu , sigp_sense ) ) {
case sigp_order_code_accepted :
case sigp_status_stored :
/* Check for stopped and check stop state */
if ( status & 0x50 )
return 1 ;
break ;
case sigp_not_operational :
return 1 ;
default :
break ;
}
return 0 ;
}
# define cpu_logical_map(cpu) (cpu)
extern int __cpu_disable ( void ) ;
extern void __cpu_die ( unsigned int cpu ) ;
extern void cpu_die ( void ) __attribute__ ( ( noreturn ) ) ;
extern int __cpu_up ( unsigned int cpu ) ;
2008-01-26 14:11:27 +01:00
extern int smp_call_function_mask ( cpumask_t mask , void ( * func ) ( void * ) ,
void * info , int wait ) ;
2005-04-16 15:20:36 -07:00
# endif
# ifndef CONFIG_SMP
2006-12-04 15:40:33 +01:00
static inline void smp_send_stop ( void )
{
/* Disable all interrupts/machine checks */
2007-02-05 21:18:17 +01:00
__load_psw_mask ( psw_kernel_bits & ~ PSW_MASK_MCHECK ) ;
2006-12-04 15:40:33 +01:00
}
2007-05-09 02:33:25 -07:00
# define hard_smp_processor_id() 0
2006-02-11 17:56:01 -08:00
# define smp_cpu_not_running(cpu) 1
2005-04-16 15:20:36 -07:00
# endif
2007-04-27 16:01:49 +02:00
extern union save_area * zfcpdump_save_areas [ NR_CPUS + 1 ] ;
2005-04-16 15:20:36 -07:00
# endif