1
2
3
4
5
6
7
8
9
10
11
12#include <linux/kernel_stat.h>
13#include <linux/init.h>
14#include <linux/errno.h>
15#include <linux/hardirq.h>
16#include <linux/log2.h>
17#include <linux/kprobes.h>
18#include <linux/kmemleak.h>
19#include <linux/time.h>
20#include <linux/module.h>
21#include <linux/sched/signal.h>
22
23#include <linux/export.h>
24#include <asm/lowcore.h>
25#include <asm/smp.h>
26#include <asm/stp.h>
27#include <asm/cputime.h>
28#include <asm/nmi.h>
29#include <asm/crw.h>
30#include <asm/switch_to.h>
31#include <asm/ctl_reg.h>
32#include <asm/asm-offsets.h>
33#include <linux/kvm_host.h>
34
35struct mcck_struct {
36 unsigned int kill_task : 1;
37 unsigned int channel_report : 1;
38 unsigned int warning : 1;
39 unsigned int stp_queue : 1;
40 unsigned long mcck_code;
41};
42
43static DEFINE_PER_CPU(struct mcck_struct, cpu_mcck);
44static struct kmem_cache *mcesa_cache;
45static unsigned long mcesa_origin_lc;
46
47static inline int nmi_needs_mcesa(void)
48{
49 return MACHINE_HAS_VX || MACHINE_HAS_GS;
50}
51
52static inline unsigned long nmi_get_mcesa_size(void)
53{
54 if (MACHINE_HAS_GS)
55 return MCESA_MAX_SIZE;
56 return MCESA_MIN_SIZE;
57}
58
59
60
61
62
63
64
65static struct mcesa boot_mcesa __initdata __aligned(MCESA_MAX_SIZE);
66
67void __init nmi_alloc_boot_cpu(struct lowcore *lc)
68{
69 if (!nmi_needs_mcesa())
70 return;
71 lc->mcesad = (unsigned long) &boot_mcesa;
72 if (MACHINE_HAS_GS)
73 lc->mcesad |= ilog2(MCESA_MAX_SIZE);
74}
75
76static int __init nmi_init(void)
77{
78 unsigned long origin, cr0, size;
79
80 if (!nmi_needs_mcesa())
81 return 0;
82 size = nmi_get_mcesa_size();
83 if (size > MCESA_MIN_SIZE)
84 mcesa_origin_lc = ilog2(size);
85
86 mcesa_cache = kmem_cache_create("nmi_save_areas", size, size, 0, NULL);
87 if (!mcesa_cache)
88 panic("Couldn't create nmi save area cache");
89 origin = (unsigned long) kmem_cache_alloc(mcesa_cache, GFP_KERNEL);
90 if (!origin)
91 panic("Couldn't allocate nmi save area");
92
93 kmemleak_not_leak((void *) origin);
94 __ctl_store(cr0, 0, 0);
95 __ctl_clear_bit(0, 28);
96
97 S390_lowcore.mcesad = origin | mcesa_origin_lc;
98 __ctl_load(cr0, 0, 0);
99 return 0;
100}
101early_initcall(nmi_init);
102
103int nmi_alloc_per_cpu(struct lowcore *lc)
104{
105 unsigned long origin;
106
107 if (!nmi_needs_mcesa())
108 return 0;
109 origin = (unsigned long) kmem_cache_alloc(mcesa_cache, GFP_KERNEL);
110 if (!origin)
111 return -ENOMEM;
112
113 kmemleak_not_leak((void *) origin);
114 lc->mcesad = origin | mcesa_origin_lc;
115 return 0;
116}
117
118void nmi_free_per_cpu(struct lowcore *lc)
119{
120 if (!nmi_needs_mcesa())
121 return;
122 kmem_cache_free(mcesa_cache, (void *)(lc->mcesad & MCESA_ORIGIN_MASK));
123}
124
125static notrace void s390_handle_damage(void)
126{
127 smp_emergency_stop();
128 disabled_wait();
129 while (1);
130}
131NOKPROBE_SYMBOL(s390_handle_damage);
132
133
134
135
136
137void __s390_handle_mcck(void)
138{
139 struct mcck_struct mcck;
140
141
142
143
144
145
146 local_mcck_disable();
147 mcck = *this_cpu_ptr(&cpu_mcck);
148 memset(this_cpu_ptr(&cpu_mcck), 0, sizeof(mcck));
149 local_mcck_enable();
150
151 if (mcck.channel_report)
152 crw_handle_channel_report();
153
154
155
156
157
158
159
160
161
162 if (mcck.warning) {
163 static int mchchk_wng_posted = 0;
164
165
166 __ctl_clear_bit(14, 24);
167 if (xchg(&mchchk_wng_posted, 1) == 0)
168 kill_cad_pid(SIGPWR, 1);
169 }
170 if (mcck.stp_queue)
171 stp_queue_work();
172 if (mcck.kill_task) {
173 local_irq_enable();
174 printk(KERN_EMERG "mcck: Terminating task because of machine "
175 "malfunction (code 0x%016lx).\n", mcck.mcck_code);
176 printk(KERN_EMERG "mcck: task: %s, pid: %d.\n",
177 current->comm, current->pid);
178 do_exit(SIGSEGV);
179 }
180}
181
182void noinstr s390_handle_mcck(void)
183{
184 trace_hardirqs_off();
185 __s390_handle_mcck();
186 trace_hardirqs_on();
187}
188
189
190
191
192static int notrace s390_check_registers(union mci mci, int umode)
193{
194 union ctlreg2 cr2;
195 int kill_task;
196
197 kill_task = 0;
198
199 if (!mci.gr) {
200
201
202
203
204 if (!umode)
205 s390_handle_damage();
206 kill_task = 1;
207 }
208
209 if (!mci.cr) {
210
211
212
213
214 s390_handle_damage();
215 }
216 if (!mci.fp) {
217
218
219
220
221
222
223 if (S390_lowcore.fpu_flags & KERNEL_VXR_V0V7)
224 s390_handle_damage();
225 if (!test_cpu_flag(CIF_FPU))
226 kill_task = 1;
227 }
228 if (!mci.fc) {
229
230
231
232
233
234
235
236 if (S390_lowcore.fpu_flags & KERNEL_FPC)
237 s390_handle_damage();
238 if (!test_cpu_flag(CIF_FPU))
239 kill_task = 1;
240 }
241
242 if (MACHINE_HAS_VX) {
243 if (!mci.vr) {
244
245
246
247
248
249
250 if (S390_lowcore.fpu_flags & KERNEL_VXR)
251 s390_handle_damage();
252 if (!test_cpu_flag(CIF_FPU))
253 kill_task = 1;
254 }
255 }
256
257 if (!mci.ar) {
258
259
260
261
262 kill_task = 1;
263 }
264
265 cr2.val = S390_lowcore.cregs_save_area[2];
266 if (cr2.gse) {
267 if (!mci.gs) {
268
269
270
271
272
273 kill_task = 1;
274 }
275 }
276
277 if (!mci.wp) {
278
279
280
281
282 s390_handle_damage();
283 }
284
285 if (!mci.ia && !umode) {
286
287
288
289
290 s390_handle_damage();
291 }
292
293 if (!mci.ms || !mci.pm || !mci.ia)
294 kill_task = 1;
295
296 return kill_task;
297}
298NOKPROBE_SYMBOL(s390_check_registers);
299
300
301
302
303static void notrace s390_backup_mcck_info(struct pt_regs *regs)
304{
305 struct mcck_volatile_info *mcck_backup;
306 struct sie_page *sie_page;
307
308
309 struct kvm_s390_sie_block *sie_block =
310 (struct kvm_s390_sie_block *) regs->gprs[14];
311
312 if (sie_block == NULL)
313
314 s390_handle_damage();
315
316 sie_page = container_of(sie_block, struct sie_page, sie_block);
317 mcck_backup = &sie_page->mcck_info;
318 mcck_backup->mcic = S390_lowcore.mcck_interruption_code &
319 ~(MCCK_CODE_CP | MCCK_CODE_EXT_DAMAGE);
320 mcck_backup->ext_damage_code = S390_lowcore.external_damage_code;
321 mcck_backup->failing_storage_address
322 = S390_lowcore.failing_storage_address;
323}
324NOKPROBE_SYMBOL(s390_backup_mcck_info);
325
326#define MAX_IPD_COUNT 29
327#define MAX_IPD_TIME (5 * 60 * USEC_PER_SEC)
328
329#define ED_STP_ISLAND 6
330#define ED_STP_SYNC 7
331
332#define MCCK_CODE_NO_GUEST (MCCK_CODE_CP | MCCK_CODE_EXT_DAMAGE)
333
334
335
336
337int notrace s390_do_machine_check(struct pt_regs *regs)
338{
339 static int ipd_count;
340 static DEFINE_SPINLOCK(ipd_lock);
341 static unsigned long long last_ipd;
342 struct mcck_struct *mcck;
343 unsigned long long tmp;
344 union mci mci;
345 unsigned long mcck_dam_code;
346 int mcck_pending = 0;
347
348 nmi_enter();
349
350 if (user_mode(regs))
351 update_timer_mcck();
352 inc_irq_stat(NMI_NMI);
353 mci.val = S390_lowcore.mcck_interruption_code;
354 mcck = this_cpu_ptr(&cpu_mcck);
355
356 if (mci.sd) {
357
358 s390_handle_damage();
359 }
360
361
362
363
364
365
366 if (mci.pd && !test_cpu_flag(CIF_MCCK_GUEST)) {
367 if (mci.b) {
368
369 u64 z_mcic, o_mcic, t_mcic;
370 z_mcic = (1ULL<<63 | 1ULL<<59 | 1ULL<<29);
371 o_mcic = (1ULL<<43 | 1ULL<<42 | 1ULL<<41 | 1ULL<<40 |
372 1ULL<<36 | 1ULL<<35 | 1ULL<<34 | 1ULL<<32 |
373 1ULL<<30 | 1ULL<<21 | 1ULL<<20 | 1ULL<<17 |
374 1ULL<<16);
375 t_mcic = mci.val;
376
377 if (((t_mcic & z_mcic) != 0) ||
378 ((t_mcic & o_mcic) != o_mcic)) {
379 s390_handle_damage();
380 }
381
382
383
384
385
386 spin_lock(&ipd_lock);
387 tmp = get_tod_clock();
388 if (((tmp - last_ipd) >> 12) < MAX_IPD_TIME)
389 ipd_count++;
390 else
391 ipd_count = 1;
392 last_ipd = tmp;
393 if (ipd_count == MAX_IPD_COUNT)
394 s390_handle_damage();
395 spin_unlock(&ipd_lock);
396 } else {
397
398 s390_handle_damage();
399 }
400 }
401 if (s390_check_registers(mci, user_mode(regs))) {
402
403
404
405
406 mcck->kill_task = 1;
407 mcck->mcck_code = mci.val;
408 mcck_pending = 1;
409 }
410
411
412
413
414
415 if (test_cpu_flag(CIF_MCCK_GUEST))
416 s390_backup_mcck_info(regs);
417
418 if (mci.cd) {
419
420 s390_handle_damage();
421 }
422 if (mci.ed && mci.ec) {
423
424 if (S390_lowcore.external_damage_code & (1U << ED_STP_SYNC))
425 mcck->stp_queue |= stp_sync_check();
426 if (S390_lowcore.external_damage_code & (1U << ED_STP_ISLAND))
427 mcck->stp_queue |= stp_island_check();
428 mcck_pending = 1;
429 }
430
431
432
433
434
435 if (!test_cpu_flag(CIF_MCCK_GUEST)) {
436 if (mci.se)
437
438 s390_handle_damage();
439 if (mci.ke)
440
441 s390_handle_damage();
442 if (mci.ds && mci.fa)
443
444 s390_handle_damage();
445 }
446 if (mci.cp) {
447
448 mcck->channel_report = 1;
449 mcck_pending = 1;
450 }
451 if (mci.w) {
452
453 mcck->warning = 1;
454 mcck_pending = 1;
455 }
456
457
458
459
460
461
462 mcck_dam_code = (mci.val & MCIC_SUBCLASS_MASK);
463 if (test_cpu_flag(CIF_MCCK_GUEST) &&
464 (mcck_dam_code & MCCK_CODE_NO_GUEST) != mcck_dam_code) {
465
466 *((long *)(regs->gprs[15] + __SF_SIE_REASON)) = -EINTR;
467 }
468 clear_cpu_flag(CIF_MCCK_GUEST);
469
470 if (user_mode(regs) && mcck_pending) {
471 nmi_exit();
472 return 1;
473 }
474
475 if (mcck_pending)
476 schedule_mcck_handler();
477
478 nmi_exit();
479 return 0;
480}
481NOKPROBE_SYMBOL(s390_do_machine_check);
482
483static int __init machine_check_init(void)
484{
485 ctl_set_bit(14, 25);
486 ctl_set_bit(14, 27);
487 ctl_set_bit(14, 24);
488 return 0;
489}
490early_initcall(machine_check_init);
491