diff options
Diffstat (limited to 'arch/s390/kernel/smp.c')
-rw-r--r-- | arch/s390/kernel/smp.c | 23 |
1 files changed, 23 insertions, 0 deletions
diff --git a/arch/s390/kernel/smp.c b/arch/s390/kernel/smp.c index 6ab16ac64d29..e4572601e91e 100644 --- a/arch/s390/kernel/smp.c +++ b/arch/s390/kernel/smp.c | |||
@@ -97,6 +97,29 @@ static inline int cpu_stopped(int cpu) | |||
97 | return raw_cpu_stopped(cpu_logical_map(cpu)); | 97 | return raw_cpu_stopped(cpu_logical_map(cpu)); |
98 | } | 98 | } |
99 | 99 | ||
100 | /* | ||
101 | * Ensure that PSW restart is done on an online CPU | ||
102 | */ | ||
103 | void smp_restart_with_online_cpu(void) | ||
104 | { | ||
105 | int cpu; | ||
106 | |||
107 | for_each_online_cpu(cpu) { | ||
108 | if (stap() == __cpu_logical_map[cpu]) { | ||
109 | /* We are online: Enable DAT again and return */ | ||
110 | __load_psw_mask(psw_kernel_bits & ~PSW_MASK_MCHECK); | ||
111 | return; | ||
112 | } | ||
113 | } | ||
114 | /* We are not online: Do PSW restart on an online CPU */ | ||
115 | while (sigp(cpu, sigp_restart) == sigp_busy) | ||
116 | cpu_relax(); | ||
117 | /* And stop ourself */ | ||
118 | while (raw_sigp(stap(), sigp_stop) == sigp_busy) | ||
119 | cpu_relax(); | ||
120 | for (;;); | ||
121 | } | ||
122 | |||
100 | void smp_switch_to_ipl_cpu(void (*func)(void *), void *data) | 123 | void smp_switch_to_ipl_cpu(void (*func)(void *), void *data) |
101 | { | 124 | { |
102 | struct _lowcore *lc, *current_lc; | 125 | struct _lowcore *lc, *current_lc; |