blob: 2126b9d27c3401e5ee518b490f8cb045fc0a29cd [file] [log] [blame]
Greg Kroah-Hartmanb2441312017-11-01 15:07:57 +01001// SPDX-License-Identifier: GPL-2.0
Joe Perchesc767a542012-05-21 19:50:07 -07002#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
3
Paul Gortmaker186f4362016-07-13 20:18:56 -04004#include <linux/export.h>
Adrian Bunkcd6ed522006-12-06 20:40:06 -08005#include <linux/reboot.h>
Miguel Boton4d022e32008-01-30 13:32:51 +01006#include <linux/init.h>
7#include <linux/pm.h>
8#include <linux/efi.h>
Paul Mackerras6c6c51e2009-08-03 22:47:32 +10009#include <linux/dmi.h>
Alexey Dobriyand43c36d2009-10-07 17:09:06 +040010#include <linux/sched.h>
Shane Wang69575d32009-09-01 18:25:07 -070011#include <linux/tboot.h>
Jean Delvareca4445642011-03-25 15:20:14 +010012#include <linux/delay.h>
Josh Poimboeufc207aee2017-06-28 10:11:06 -050013#include <linux/frame.h>
Miguel Boton4d022e32008-01-30 13:32:51 +010014#include <acpi/reboot.h>
15#include <asm/io.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070016#include <asm/apic.h>
Jiang Liu8643e282014-10-27 16:12:04 +080017#include <asm/io_apic.h>
Zachary Amsden4d37e7e2005-09-03 15:56:38 -070018#include <asm/desc.h>
Miguel Boton4d022e32008-01-30 13:32:51 +010019#include <asm/hpet.h>
Jeremy Fitzhardinge68db0652008-03-17 16:37:13 -070020#include <asm/pgtable.h>
Dmitri Vorobiev44126202008-04-28 03:15:59 +040021#include <asm/proto.h>
Jeremy Fitzhardinge973efae2007-05-02 19:27:06 +020022#include <asm/reboot_fixups.h>
Jeremy Fitzhardinge07f33312007-05-02 19:27:11 +020023#include <asm/reboot.h>
Jaswinder Singh Rajput82487712008-12-27 18:32:28 +053024#include <asm/pci_x86.h>
Eduardo Habkostd1767202008-11-17 19:03:24 -020025#include <asm/virtext.h>
Jaswinder Singh Rajput96b89dc62009-01-07 21:35:48 +053026#include <asm/cpu.h>
Don Zickusc410b832011-01-06 16:18:50 -050027#include <asm/nmi.h>
H. Peter Anvin65051392012-06-16 21:47:37 -070028#include <asm/smp.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070029
H. Peter Anvin65051392012-06-16 21:47:37 -070030#include <linux/ctype.h>
31#include <linux/mc146818rtc.h>
32#include <asm/realmode.h>
33#include <asm/x86_init.h>
Matt Fleming44be28e2014-06-13 12:39:55 +010034#include <asm/efi.h>
Miguel Boton4d022e32008-01-30 13:32:51 +010035
Linus Torvalds1da177e2005-04-16 15:20:36 -070036/*
37 * Power off function, if any
38 */
39void (*pm_power_off)(void);
Alexey Dobriyan129f6942005-06-23 00:08:33 -070040EXPORT_SYMBOL(pm_power_off);
Linus Torvalds1da177e2005-04-16 15:20:36 -070041
Michael D Labriola144d1022012-02-01 10:06:34 -050042/*
43 * This is set if we need to go through the 'emergency' path.
Eduardo Habkostd1767202008-11-17 19:03:24 -020044 * When machine_emergency_restart() is called, we may be on
45 * an inconsistent state and won't be able to do a clean cleanup
46 */
47static int reboot_emergency;
48
H. Peter Anvin14d7ca52008-11-11 16:19:48 -080049/* This is set by the PCI code if either type 1 or type 2 PCI is detected */
50bool port_cf9_safe = false;
51
Michael D Labriola144d1022012-02-01 10:06:34 -050052/*
Linus Torvalds1da177e2005-04-16 15:20:36 -070053 * Reboot options and system auto-detection code provided by
54 * Dell Inc. so their systems "just work". :-)
55 */
56
57/*
Alex Hung4d581252016-07-14 18:05:56 +080058 * Some machines require the "reboot=a" commandline options
59 */
60static int __init set_acpi_reboot(const struct dmi_system_id *d)
61{
62 if (reboot_type != BOOT_ACPI) {
63 reboot_type = BOOT_ACPI;
64 pr_info("%s series board detected. Selecting %s-method for reboots.\n",
65 d->ident, "ACPI");
66 }
67 return 0;
68}
69
70/*
Peter Chubb1ef03892011-12-05 16:53:53 +030071 * Some machines require the "reboot=b" or "reboot=k" commandline options,
Miguel Boton4d022e32008-01-30 13:32:51 +010072 * this quirk makes that automatic.
Linus Torvalds1da177e2005-04-16 15:20:36 -070073 */
Jeff Garzik18552562007-10-03 15:15:40 -040074static int __init set_bios_reboot(const struct dmi_system_id *d)
Linus Torvalds1da177e2005-04-16 15:20:36 -070075{
Miguel Boton4d022e32008-01-30 13:32:51 +010076 if (reboot_type != BOOT_BIOS) {
77 reboot_type = BOOT_BIOS;
Joe Perchesc767a542012-05-21 19:50:07 -070078 pr_info("%s series board detected. Selecting %s-method for reboots.\n",
Lan Tianyu6d9153b2013-10-24 15:11:33 +080079 d->ident, "BIOS");
Linus Torvalds1da177e2005-04-16 15:20:36 -070080 }
81 return 0;
82}
83
H. Peter Anvin65051392012-06-16 21:47:37 -070084void __noreturn machine_real_restart(unsigned int type)
Michael D Labriola57b16592012-02-01 10:05:00 -050085{
Michael D Labriola57b16592012-02-01 10:05:00 -050086 local_irq_disable();
87
Michael D Labriola144d1022012-02-01 10:06:34 -050088 /*
89 * Write zero to CMOS register number 0x0f, which the BIOS POST
90 * routine will recognize as telling it to do a proper reboot. (Well
91 * that's what this book in front of me says -- it may only apply to
92 * the Phoenix BIOS though, it's not clear). At the same time,
93 * disable NMIs by setting the top bit in the CMOS address register,
94 * as we're about to do peculiar things to the CPU. I'm not sure if
95 * `outb_p' is needed instead of just `outb'. Use it to be on the
96 * safe side. (Yes, CMOS_WRITE does outb_p's. - Paul G.)
Michael D Labriola57b16592012-02-01 10:05:00 -050097 */
98 spin_lock(&rtc_lock);
99 CMOS_WRITE(0x00, 0x8f);
100 spin_unlock(&rtc_lock);
101
102 /*
103 * Switch back to the initial page table.
104 */
H. Peter Anvin65051392012-06-16 21:47:37 -0700105#ifdef CONFIG_X86_32
Michael D Labriola57b16592012-02-01 10:05:00 -0500106 load_cr3(initial_page_table);
H. Peter Anvin65051392012-06-16 21:47:37 -0700107#else
108 write_cr3(real_mode_header->trampoline_pgd);
Andy Lutomirski924c6b92017-10-08 21:53:05 -0700109
110 /* Exiting long mode will fail if CR4.PCIDE is set. */
111 if (static_cpu_has(X86_FEATURE_PCID))
112 cr4_clear_bits(X86_CR4_PCIDE);
H. Peter Anvin65051392012-06-16 21:47:37 -0700113#endif
Michael D Labriola57b16592012-02-01 10:05:00 -0500114
Michael D Labriola57b16592012-02-01 10:05:00 -0500115 /* Jump to the identity-mapped low memory code */
H. Peter Anvin65051392012-06-16 21:47:37 -0700116#ifdef CONFIG_X86_32
117 asm volatile("jmpl *%0" : :
118 "rm" (real_mode_header->machine_real_restart_asm),
119 "a" (type));
120#else
121 asm volatile("ljmpl *%0" : :
122 "m" (real_mode_header->machine_real_restart_asm),
123 "D" (type));
124#endif
125 unreachable();
Michael D Labriola57b16592012-02-01 10:05:00 -0500126}
127#ifdef CONFIG_APM_MODULE
128EXPORT_SYMBOL(machine_real_restart);
129#endif
Josh Poimboeufc207aee2017-06-28 10:11:06 -0500130STACK_FRAME_NON_STANDARD(machine_real_restart);
Michael D Labriola57b16592012-02-01 10:05:00 -0500131
Michael D Labriola57b16592012-02-01 10:05:00 -0500132/*
133 * Some Apple MacBook and MacBookPro's needs reboot=p to be able to reboot
134 */
135static int __init set_pci_reboot(const struct dmi_system_id *d)
136{
Ingo Molnar5be44a62014-04-04 08:41:26 +0200137 if (reboot_type != BOOT_CF9_FORCE) {
138 reboot_type = BOOT_CF9_FORCE;
Joe Perchesc767a542012-05-21 19:50:07 -0700139 pr_info("%s series board detected. Selecting %s-method for reboots.\n",
Lan Tianyu6d9153b2013-10-24 15:11:33 +0800140 d->ident, "PCI");
Michael D Labriola57b16592012-02-01 10:05:00 -0500141 }
142 return 0;
143}
144
Peter Chubb1ef03892011-12-05 16:53:53 +0300145static int __init set_kbd_reboot(const struct dmi_system_id *d)
146{
147 if (reboot_type != BOOT_KBD) {
148 reboot_type = BOOT_KBD;
Joe Perchesc767a542012-05-21 19:50:07 -0700149 pr_info("%s series board detected. Selecting %s-method for reboot.\n",
Lan Tianyu6d9153b2013-10-24 15:11:33 +0800150 d->ident, "KBD");
Peter Chubb1ef03892011-12-05 16:53:53 +0300151 }
152 return 0;
153}
154
Michael D Labriola144d1022012-02-01 10:06:34 -0500155/*
H. Peter Anvin65051392012-06-16 21:47:37 -0700156 * This is a single dmi_table handling all reboot quirks.
Michael D Labriola57b16592012-02-01 10:05:00 -0500157 */
Christoph Hellwig6faadbb2017-09-14 11:59:30 +0200158static const struct dmi_system_id reboot_dmi_table[] __initconst = {
Michael D Labriola57b16592012-02-01 10:05:00 -0500159
Dave Jonese56e57f2013-10-01 16:36:55 -0400160 /* Acer */
Michael D Labriola144d1022012-02-01 10:06:34 -0500161 { /* Handle reboot issue on Acer Aspire one */
Peter Chubb1ef03892011-12-05 16:53:53 +0300162 .callback = set_kbd_reboot,
Peter Chubbb49c78d2011-07-06 10:56:30 +1000163 .ident = "Acer Aspire One A110",
164 .matches = {
165 DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
166 DMI_MATCH(DMI_PRODUCT_NAME, "AOA110"),
167 },
168 },
Dave Jonese56e57f2013-10-01 16:36:55 -0400169
170 /* Apple */
Shunichi Fuji3e03bbe2009-08-11 03:34:40 +0900171 { /* Handle problems with rebooting on Apple MacBook5 */
Paul Mackerras6c6c51e2009-08-03 22:47:32 +1000172 .callback = set_pci_reboot,
Shunichi Fuji3e03bbe2009-08-11 03:34:40 +0900173 .ident = "Apple MacBook5",
Paul Mackerras6c6c51e2009-08-03 22:47:32 +1000174 .matches = {
175 DMI_MATCH(DMI_SYS_VENDOR, "Apple Inc."),
Shunichi Fuji3e03bbe2009-08-11 03:34:40 +0900176 DMI_MATCH(DMI_PRODUCT_NAME, "MacBook5"),
Paul Mackerras6c6c51e2009-08-03 22:47:32 +1000177 },
178 },
Shunichi Fuji3e03bbe2009-08-11 03:34:40 +0900179 { /* Handle problems with rebooting on Apple MacBookPro5 */
Ozan Çağlayan498cdbf2009-08-04 19:39:31 +0300180 .callback = set_pci_reboot,
Shunichi Fuji3e03bbe2009-08-11 03:34:40 +0900181 .ident = "Apple MacBookPro5",
Ozan Çağlayan498cdbf2009-08-04 19:39:31 +0300182 .matches = {
183 DMI_MATCH(DMI_SYS_VENDOR, "Apple Inc."),
Shunichi Fuji3e03bbe2009-08-11 03:34:40 +0900184 DMI_MATCH(DMI_PRODUCT_NAME, "MacBookPro5"),
Ozan Çağlayan498cdbf2009-08-04 19:39:31 +0300185 },
186 },
Gottfried Haider05154752009-11-02 11:51:11 +0100187 { /* Handle problems with rebooting on Apple Macmini3,1 */
188 .callback = set_pci_reboot,
189 .ident = "Apple Macmini3,1",
190 .matches = {
191 DMI_MATCH(DMI_SYS_VENDOR, "Apple Inc."),
192 DMI_MATCH(DMI_PRODUCT_NAME, "Macmini3,1"),
193 },
194 },
Justin P. Mattock0a832322010-02-16 15:17:29 -0800195 { /* Handle problems with rebooting on the iMac9,1. */
196 .callback = set_pci_reboot,
197 .ident = "Apple iMac9,1",
198 .matches = {
199 DMI_MATCH(DMI_SYS_VENDOR, "Apple Inc."),
200 DMI_MATCH(DMI_PRODUCT_NAME, "iMac9,1"),
201 },
202 },
Mario Kleiner2f0c0b22015-12-18 20:24:06 +0100203 { /* Handle problems with rebooting on the iMac10,1. */
204 .callback = set_pci_reboot,
205 .ident = "Apple iMac10,1",
206 .matches = {
207 DMI_MATCH(DMI_SYS_VENDOR, "Apple Inc."),
208 DMI_MATCH(DMI_PRODUCT_NAME, "iMac10,1"),
209 },
210 },
Dave Jonese56e57f2013-10-01 16:36:55 -0400211
Stefan Lippers-Hollmann80313b32015-03-30 22:44:27 +0200212 /* ASRock */
213 { /* Handle problems with rebooting on ASRock Q1900DC-ITX */
214 .callback = set_pci_reboot,
215 .ident = "ASRock Q1900DC-ITX",
216 .matches = {
217 DMI_MATCH(DMI_BOARD_VENDOR, "ASRock"),
218 DMI_MATCH(DMI_BOARD_NAME, "Q1900DC-ITX"),
219 },
220 },
221
Dave Jonese56e57f2013-10-01 16:36:55 -0400222 /* ASUS */
223 { /* Handle problems with rebooting on ASUS P4S800 */
224 .callback = set_bios_reboot,
225 .ident = "ASUS P4S800",
226 .matches = {
227 DMI_MATCH(DMI_BOARD_VENDOR, "ASUSTeK Computer INC."),
228 DMI_MATCH(DMI_BOARD_NAME, "P4S800"),
229 },
230 },
Matjaz Hegedic90b28de2017-02-19 19:32:48 +0100231 { /* Handle problems with rebooting on ASUS EeeBook X205TA */
232 .callback = set_acpi_reboot,
233 .ident = "ASUS EeeBook X205TA",
234 .matches = {
235 DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
Matjaz Hegedicbba8376a2017-03-09 14:00:17 +0100236 DMI_MATCH(DMI_PRODUCT_NAME, "X205TA"),
Matjaz Hegedic90b28de2017-02-19 19:32:48 +0100237 },
238 },
Matjaz Hegedic3b3e7852017-03-05 19:16:44 +0100239 { /* Handle problems with rebooting on ASUS EeeBook X205TAW */
240 .callback = set_acpi_reboot,
241 .ident = "ASUS EeeBook X205TAW",
242 .matches = {
243 DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK COMPUTER INC."),
244 DMI_MATCH(DMI_PRODUCT_NAME, "X205TAW"),
245 },
246 },
Dave Jonese56e57f2013-10-01 16:36:55 -0400247
Christian Gmeineraadca6f2014-05-07 09:01:54 +0200248 /* Certec */
249 { /* Handle problems with rebooting on Certec BPC600 */
250 .callback = set_pci_reboot,
251 .ident = "Certec BPC600",
252 .matches = {
253 DMI_MATCH(DMI_SYS_VENDOR, "Certec"),
254 DMI_MATCH(DMI_PRODUCT_NAME, "BPC600"),
255 },
256 },
257
Dave Jonese56e57f2013-10-01 16:36:55 -0400258 /* Dell */
259 { /* Handle problems with rebooting on Dell DXP061 */
260 .callback = set_bios_reboot,
261 .ident = "Dell DXP061",
Maxime Ripard3628c3f2011-06-28 15:57:31 +0200262 .matches = {
263 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
Dave Jonese56e57f2013-10-01 16:36:55 -0400264 DMI_MATCH(DMI_PRODUCT_NAME, "Dell DXP061"),
265 },
266 },
267 { /* Handle problems with rebooting on Dell E520's */
268 .callback = set_bios_reboot,
269 .ident = "Dell E520",
270 .matches = {
271 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
272 DMI_MATCH(DMI_PRODUCT_NAME, "Dell DM061"),
Maxime Ripard3628c3f2011-06-28 15:57:31 +0200273 },
274 },
Ville Syrjälä8412da752013-10-04 15:16:04 +0300275 { /* Handle problems with rebooting on the Latitude E5410. */
276 .callback = set_pci_reboot,
277 .ident = "Dell Latitude E5410",
278 .matches = {
279 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
280 DMI_MATCH(DMI_PRODUCT_NAME, "Latitude E5410"),
281 },
282 },
Daniel J Bluemanb7798d22011-05-13 09:04:59 +0800283 { /* Handle problems with rebooting on the Latitude E5420. */
284 .callback = set_pci_reboot,
285 .ident = "Dell Latitude E5420",
286 .matches = {
287 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
288 DMI_MATCH(DMI_PRODUCT_NAME, "Latitude E5420"),
289 },
290 },
Dave Jonese56e57f2013-10-01 16:36:55 -0400291 { /* Handle problems with rebooting on the Latitude E6320. */
292 .callback = set_pci_reboot,
293 .ident = "Dell Latitude E6320",
294 .matches = {
295 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
296 DMI_MATCH(DMI_PRODUCT_NAME, "Latitude E6320"),
297 },
298 },
H. Peter Anvina5368772011-07-21 11:22:21 -0700299 { /* Handle problems with rebooting on the Latitude E6420. */
300 .callback = set_pci_reboot,
301 .ident = "Dell Latitude E6420",
302 .matches = {
303 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
304 DMI_MATCH(DMI_PRODUCT_NAME, "Latitude E6420"),
305 },
306 },
Dave Jonese56e57f2013-10-01 16:36:55 -0400307 { /* Handle problems with rebooting on Dell Optiplex 330 with 0KP561 */
308 .callback = set_bios_reboot,
309 .ident = "Dell OptiPlex 330",
310 .matches = {
311 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
312 DMI_MATCH(DMI_PRODUCT_NAME, "OptiPlex 330"),
313 DMI_MATCH(DMI_BOARD_NAME, "0KP561"),
314 },
315 },
316 { /* Handle problems with rebooting on Dell Optiplex 360 with 0T656F */
317 .callback = set_bios_reboot,
318 .ident = "Dell OptiPlex 360",
319 .matches = {
320 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
321 DMI_MATCH(DMI_PRODUCT_NAME, "OptiPlex 360"),
322 DMI_MATCH(DMI_BOARD_NAME, "0T656F"),
323 },
324 },
325 { /* Handle problems with rebooting on Dell Optiplex 745's SFF */
326 .callback = set_bios_reboot,
327 .ident = "Dell OptiPlex 745",
328 .matches = {
329 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
330 DMI_MATCH(DMI_PRODUCT_NAME, "OptiPlex 745"),
331 },
332 },
333 { /* Handle problems with rebooting on Dell Optiplex 745's DFF */
334 .callback = set_bios_reboot,
335 .ident = "Dell OptiPlex 745",
336 .matches = {
337 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
338 DMI_MATCH(DMI_PRODUCT_NAME, "OptiPlex 745"),
339 DMI_MATCH(DMI_BOARD_NAME, "0MM599"),
340 },
341 },
342 { /* Handle problems with rebooting on Dell Optiplex 745 with 0KW626 */
343 .callback = set_bios_reboot,
344 .ident = "Dell OptiPlex 745",
345 .matches = {
346 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
347 DMI_MATCH(DMI_PRODUCT_NAME, "OptiPlex 745"),
348 DMI_MATCH(DMI_BOARD_NAME, "0KW626"),
349 },
350 },
351 { /* Handle problems with rebooting on Dell OptiPlex 760 with 0G919G */
352 .callback = set_bios_reboot,
353 .ident = "Dell OptiPlex 760",
354 .matches = {
355 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
356 DMI_MATCH(DMI_PRODUCT_NAME, "OptiPlex 760"),
357 DMI_MATCH(DMI_BOARD_NAME, "0G919G"),
358 },
359 },
Rafael J. Wysocki6be30bb2011-11-16 00:19:51 +0100360 { /* Handle problems with rebooting on the OptiPlex 990. */
361 .callback = set_pci_reboot,
362 .ident = "Dell OptiPlex 990",
363 .matches = {
364 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
365 DMI_MATCH(DMI_PRODUCT_NAME, "OptiPlex 990"),
366 },
367 },
Dave Jonese56e57f2013-10-01 16:36:55 -0400368 { /* Handle problems with rebooting on Dell 300's */
369 .callback = set_bios_reboot,
370 .ident = "Dell PowerEdge 300",
Zhang Rui76eb9a32012-02-20 14:20:06 +0800371 .matches = {
Dave Jonese56e57f2013-10-01 16:36:55 -0400372 DMI_MATCH(DMI_SYS_VENDOR, "Dell Computer Corporation"),
373 DMI_MATCH(DMI_PRODUCT_NAME, "PowerEdge 300/"),
374 },
375 },
376 { /* Handle problems with rebooting on Dell 1300's */
377 .callback = set_bios_reboot,
378 .ident = "Dell PowerEdge 1300",
379 .matches = {
380 DMI_MATCH(DMI_SYS_VENDOR, "Dell Computer Corporation"),
381 DMI_MATCH(DMI_PRODUCT_NAME, "PowerEdge 1300/"),
382 },
383 },
384 { /* Handle problems with rebooting on Dell 2400's */
385 .callback = set_bios_reboot,
386 .ident = "Dell PowerEdge 2400",
387 .matches = {
388 DMI_MATCH(DMI_SYS_VENDOR, "Dell Computer Corporation"),
389 DMI_MATCH(DMI_PRODUCT_NAME, "PowerEdge 2400"),
Zhang Rui76eb9a32012-02-20 14:20:06 +0800390 },
391 },
Masoud Sharbiani4f0acd32013-09-20 15:59:07 -0700392 { /* Handle problems with rebooting on the Dell PowerEdge C6100. */
393 .callback = set_pci_reboot,
394 .ident = "Dell PowerEdge C6100",
395 .matches = {
Masoud Sharbiani4f0acd32013-09-20 15:59:07 -0700396 DMI_MATCH(DMI_SYS_VENDOR, "Dell"),
397 DMI_MATCH(DMI_PRODUCT_NAME, "C6100"),
398 },
399 },
Paul Mackerras6c6c51e2009-08-03 22:47:32 +1000400 { /* Handle problems with rebooting on the Precision M6600. */
401 .callback = set_pci_reboot,
402 .ident = "Dell Precision M6600",
403 .matches = {
404 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
405 DMI_MATCH(DMI_PRODUCT_NAME, "Precision M6600"),
406 },
407 },
Dave Jonese56e57f2013-10-01 16:36:55 -0400408 { /* Handle problems with rebooting on Dell T5400's */
409 .callback = set_bios_reboot,
410 .ident = "Dell Precision T5400",
Michael D Labriola57b16592012-02-01 10:05:00 -0500411 .matches = {
Paul Mackerras6c6c51e2009-08-03 22:47:32 +1000412 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
Dave Jonese56e57f2013-10-01 16:36:55 -0400413 DMI_MATCH(DMI_PRODUCT_NAME, "Precision WorkStation T5400"),
Michael D Labriola144d1022012-02-01 10:06:34 -0500414 },
Michael D Labriola59556332012-01-29 14:17:22 -0500415 },
Dave Jonese56e57f2013-10-01 16:36:55 -0400416 { /* Handle problems with rebooting on Dell T7400's */
417 .callback = set_bios_reboot,
418 .ident = "Dell Precision T7400",
Paul Mackerras6c6c51e2009-08-03 22:47:32 +1000419 .matches = {
Dave Jonese56e57f2013-10-01 16:36:55 -0400420 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
421 DMI_MATCH(DMI_PRODUCT_NAME, "Precision WorkStation T7400"),
Paul Mackerras6c6c51e2009-08-03 22:47:32 +1000422 },
Miguel Boton4d022e32008-01-30 13:32:51 +0100423 },
Dave Jonese56e57f2013-10-01 16:36:55 -0400424 { /* Handle problems with rebooting on Dell XPS710 */
425 .callback = set_bios_reboot,
426 .ident = "Dell XPS710",
427 .matches = {
428 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
429 DMI_MATCH(DMI_PRODUCT_NAME, "Dell XPS710"),
430 },
431 },
Alex Hung4d581252016-07-14 18:05:56 +0800432 { /* Handle problems with rebooting on Dell Optiplex 7450 AIO */
433 .callback = set_acpi_reboot,
434 .ident = "Dell OptiPlex 7450 AIO",
435 .matches = {
436 DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
437 DMI_MATCH(DMI_PRODUCT_NAME, "OptiPlex 7450 AIO"),
438 },
439 },
Dave Jonese56e57f2013-10-01 16:36:55 -0400440
441 /* Hewlett-Packard */
442 { /* Handle problems with rebooting on HP laptops */
443 .callback = set_bios_reboot,
444 .ident = "HP Compaq Laptop",
445 .matches = {
446 DMI_MATCH(DMI_SYS_VENDOR, "Hewlett-Packard"),
447 DMI_MATCH(DMI_PRODUCT_NAME, "HP Compaq"),
448 },
449 },
450
451 /* Sony */
452 { /* Handle problems with rebooting on Sony VGN-Z540N */
453 .callback = set_bios_reboot,
454 .ident = "Sony VGN-Z540N",
455 .matches = {
456 DMI_MATCH(DMI_SYS_VENDOR, "Sony Corporation"),
457 DMI_MATCH(DMI_PRODUCT_NAME, "VGN-Z540N"),
458 },
459 },
460
Miguel Boton4d022e32008-01-30 13:32:51 +0100461 { }
462};
Don Zickus9c48f1c2011-09-30 15:06:21 -0400463
Eduardo Habkostd1767202008-11-17 19:03:24 -0200464static int __init reboot_init(void)
465{
Matt Fleming44be28e2014-06-13 12:39:55 +0100466 int rv;
467
Eduardo Habkostd1767202008-11-17 19:03:24 -0200468 /*
469 * Only do the DMI check if reboot_type hasn't been overridden
470 * on the command line
471 */
Matt Fleming44be28e2014-06-13 12:39:55 +0100472 if (!reboot_default)
473 return 0;
474
475 /*
476 * The DMI quirks table takes precedence. If no quirks entry
Stefan Assmann4ecf7192017-07-24 14:22:48 +0200477 * matches and the ACPI Hardware Reduced bit is set and EFI
478 * runtime services are enabled, force EFI reboot.
Matt Fleming44be28e2014-06-13 12:39:55 +0100479 */
480 rv = dmi_check_system(reboot_dmi_table);
481
Stefan Assmann4ecf7192017-07-24 14:22:48 +0200482 if (!rv && efi_reboot_required() && !efi_runtime_disabled())
Matt Fleming44be28e2014-06-13 12:39:55 +0100483 reboot_type = BOOT_EFI;
484
Eduardo Habkostd1767202008-11-17 19:03:24 -0200485 return 0;
486}
487core_initcall(reboot_init);
488
489static inline void kb_wait(void)
490{
491 int i;
492
493 for (i = 0; i < 0x10000; i++) {
Ingo Molnar7432d142008-03-06 18:29:43 +0100494 if ((inb(0x64) & 0x02) == 0)
495 break;
496 udelay(2);
497 }
Jody Belka416e2d62008-02-12 23:37:48 +0000498}
Miguel Boton4d022e32008-01-30 13:32:51 +0100499
500static void vmxoff_nmi(int cpu, struct pt_regs *regs)
501{
Eduardo Habkostd1767202008-11-17 19:03:24 -0200502 cpu_emergency_vmxoff();
503}
504
Michael D Labriola144d1022012-02-01 10:06:34 -0500505/* Use NMIs as IPIs to tell all CPUs to disable virtualization */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700506static void emergency_vmx_disable_all(void)
507{
508 /* Just make sure we won't change CPUs while doing this */
509 local_irq_disable();
Zachary Amsden4d37e7e2005-09-03 15:56:38 -0700510
Michael D Labriola144d1022012-02-01 10:06:34 -0500511 /*
512 * We need to disable VMX on all CPUs before rebooting, otherwise
Linus Torvalds1da177e2005-04-16 15:20:36 -0700513 * we risk hanging up the machine, because the CPU ignore INIT
514 * signals when VMX is enabled.
515 *
516 * We can't take any locks and we may be on an inconsistent
517 * state, so we use NMIs as IPIs to tell the other CPUs to disable
518 * VMX and halt.
519 *
520 * For safety, we will avoid running the nmi_shootdown_cpus()
521 * stuff unnecessarily, but we don't have a way to check
522 * if other CPUs have VMX enabled. So we will call it only if the
523 * CPU we are running on has VMX enabled.
524 *
525 * We will miss cases where VMX is not enabled on all CPUs. This
526 * shouldn't do much harm because KVM always enable VMX on all
527 * CPUs anyway. But we can miss it on the small window where KVM
528 * is still enabling VMX.
529 */
530 if (cpu_has_vmx() && cpu_vmx_enabled()) {
Michael D Labriola144d1022012-02-01 10:06:34 -0500531 /* Disable VMX on this CPU. */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700532 cpu_vmxoff();
Alexey Dobriyan129f6942005-06-23 00:08:33 -0700533
534 /* Halt and disable VMX on the other CPUs */
535 nmi_shootdown_cpus(vmxoff_nmi);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700536
Jeremy Fitzhardinge07f33312007-05-02 19:27:11 +0200537 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700538}
Miguel Boton4d022e32008-01-30 13:32:51 +0100539
540
541void __attribute__((weak)) mach_reboot_fixups(void)
542{
543}
544
Matthew Garrett660e34c2011-04-04 13:55:05 -0400545/*
Ingo Molnar5be44a62014-04-04 08:41:26 +0200546 * To the best of our knowledge Windows compatible x86 hardware expects
547 * the following on reboot:
Matthew Garrett660e34c2011-04-04 13:55:05 -0400548 *
549 * 1) If the FADT has the ACPI reboot register flag set, try it
550 * 2) If still alive, write to the keyboard controller
551 * 3) If still alive, write to the ACPI reboot register again
552 * 4) If still alive, write to the keyboard controller again
Li, Aubreya4f19872014-03-02 18:39:02 +0800553 * 5) If still alive, call the EFI runtime service to reboot
Ingo Molnar5be44a62014-04-04 08:41:26 +0200554 * 6) If no EFI runtime service, call the BIOS to do a reboot
Matthew Garrett660e34c2011-04-04 13:55:05 -0400555 *
Ingo Molnar5be44a62014-04-04 08:41:26 +0200556 * We default to following the same pattern. We also have
557 * two other reboot methods: 'triple fault' and 'PCI', which
558 * can be triggered via the reboot= kernel boot option or
559 * via quirks.
560 *
561 * This means that this function can never return, it can misbehave
562 * by not rebooting properly and hanging.
Matthew Garrett660e34c2011-04-04 13:55:05 -0400563 */
Miguel Boton4d022e32008-01-30 13:32:51 +0100564static void native_machine_emergency_restart(void)
565{
566 int i;
Matthew Garrett660e34c2011-04-04 13:55:05 -0400567 int attempt = 0;
568 int orig_reboot_type = reboot_type;
Robin Holtedf2b132013-07-08 16:01:35 -0700569 unsigned short mode;
Miguel Boton4d022e32008-01-30 13:32:51 +0100570
571 if (reboot_emergency)
572 emergency_vmx_disable_all();
573
Joseph Cihula840c2ba2009-06-30 19:31:02 -0700574 tboot_shutdown(TB_SHUTDOWN_REBOOT);
575
Miguel Boton4d022e32008-01-30 13:32:51 +0100576 /* Tell the BIOS if we want cold or warm reboot */
Robin Holtedf2b132013-07-08 16:01:35 -0700577 mode = reboot_mode == REBOOT_WARM ? 0x1234 : 0;
578 *((unsigned short *)__va(0x472)) = mode;
Miguel Boton4d022e32008-01-30 13:32:51 +0100579
Matt Fleming87615a32016-04-25 21:07:00 +0100580 /*
581 * If an EFI capsule has been registered with the firmware then
582 * override the reboot= parameter.
583 */
584 if (efi_capsule_pending(NULL)) {
585 pr_info("EFI capsule is pending, forcing EFI reboot.\n");
586 reboot_type = BOOT_EFI;
587 }
588
Miguel Boton4d022e32008-01-30 13:32:51 +0100589 for (;;) {
590 /* Could also try the reset bit in the Hammer NB */
591 switch (reboot_type) {
Ingo Molnar5be44a62014-04-04 08:41:26 +0200592 case BOOT_ACPI:
593 acpi_reboot();
594 reboot_type = BOOT_KBD;
595 break;
596
Miguel Boton4d022e32008-01-30 13:32:51 +0100597 case BOOT_KBD:
Michael D Labriola144d1022012-02-01 10:06:34 -0500598 mach_reboot_fixups(); /* For board specific fixups */
Ingo Molnar7432d142008-03-06 18:29:43 +0100599
Miguel Boton4d022e32008-01-30 13:32:51 +0100600 for (i = 0; i < 10; i++) {
601 kb_wait();
602 udelay(50);
Michael D Labriola144d1022012-02-01 10:06:34 -0500603 outb(0xfe, 0x64); /* Pulse reset low */
Miguel Boton4d022e32008-01-30 13:32:51 +0100604 udelay(50);
605 }
Matthew Garrett660e34c2011-04-04 13:55:05 -0400606 if (attempt == 0 && orig_reboot_type == BOOT_ACPI) {
607 attempt = 1;
608 reboot_type = BOOT_ACPI;
609 } else {
Li, Aubreya4f19872014-03-02 18:39:02 +0800610 reboot_type = BOOT_EFI;
Matthew Garrett660e34c2011-04-04 13:55:05 -0400611 }
612 break;
Miguel Boton4d022e32008-01-30 13:32:51 +0100613
Miguel Boton4d022e32008-01-30 13:32:51 +0100614 case BOOT_EFI:
Matt Fleming8562c992014-06-13 12:22:22 +0100615 efi_reboot(reboot_mode, NULL);
Ingo Molnar5be44a62014-04-04 08:41:26 +0200616 reboot_type = BOOT_BIOS;
H. Peter Anvin14d7ca52008-11-11 16:19:48 -0800617 break;
Miguel Boton4d022e32008-01-30 13:32:51 +0100618
Ingo Molnar5be44a62014-04-04 08:41:26 +0200619 case BOOT_BIOS:
620 machine_real_restart(MRR_BIOS);
621
622 /* We're probably dead after this, but... */
623 reboot_type = BOOT_CF9_SAFE;
624 break;
625
626 case BOOT_CF9_FORCE:
H. Peter Anvin14d7ca52008-11-11 16:19:48 -0800627 port_cf9_safe = true;
Michael D Labriola144d1022012-02-01 10:06:34 -0500628 /* Fall through */
H. Peter Anvin14d7ca52008-11-11 16:19:48 -0800629
Ingo Molnar5be44a62014-04-04 08:41:26 +0200630 case BOOT_CF9_SAFE:
H. Peter Anvin14d7ca52008-11-11 16:19:48 -0800631 if (port_cf9_safe) {
Ingo Molnar5be44a62014-04-04 08:41:26 +0200632 u8 reboot_code = reboot_mode == REBOOT_WARM ? 0x06 : 0x0E;
Li Fei16c21ae2013-08-21 16:13:57 +0800633 u8 cf9 = inb(0xcf9) & ~reboot_code;
H. Peter Anvin14d7ca52008-11-11 16:19:48 -0800634 outb(cf9|2, 0xcf9); /* Request hard reset */
635 udelay(50);
Li Fei16c21ae2013-08-21 16:13:57 +0800636 /* Actually do the reset */
637 outb(cf9|reboot_code, 0xcf9);
H. Peter Anvin14d7ca52008-11-11 16:19:48 -0800638 udelay(50);
639 }
Ingo Molnar5be44a62014-04-04 08:41:26 +0200640 reboot_type = BOOT_TRIPLE;
641 break;
642
643 case BOOT_TRIPLE:
Thomas Gleixnere802a512017-08-28 08:47:46 +0200644 idt_invalidate(NULL);
Ingo Molnar5be44a62014-04-04 08:41:26 +0200645 __asm__ __volatile__("int3");
646
647 /* We're probably dead after this, but... */
648 reboot_type = BOOT_KBD;
Miguel Boton4d022e32008-01-30 13:32:51 +0100649 break;
650 }
651 }
652}
653
Glauber Costa3c62c622008-03-17 16:08:39 -0300654void native_machine_shutdown(void)
Miguel Boton4d022e32008-01-30 13:32:51 +0100655{
656 /* Stop the cpus and apics */
Fenghua Yu522e6642013-10-23 18:30:12 -0700657#ifdef CONFIG_X86_IO_APIC
Fenghua Yu28854322013-12-04 16:07:49 -0800658 /*
659 * Disabling IO APIC before local APIC is a workaround for
660 * erratum AVR31 in "Intel Atom Processor C2000 Product Family
661 * Specification Update". In this situation, interrupts that target
662 * a Logical Processor whose Local APIC is either in the process of
663 * being hardware disabled or software disabled are neither delivered
664 * nor discarded. When this erratum occurs, the processor may hang.
665 *
666 * Even without the erratum, it still makes sense to quiet IO APIC
667 * before disabling Local APIC.
668 */
Fenghua Yu522e6642013-10-23 18:30:12 -0700669 disable_IO_APIC();
670#endif
671
Linus Torvalds1da177e2005-04-16 15:20:36 -0700672#ifdef CONFIG_SMP
Michael D Labriola144d1022012-02-01 10:06:34 -0500673 /*
Robin Holt1b3a5d02013-07-08 16:01:42 -0700674 * Stop all of the others. Also disable the local irq to
675 * not receive the per-cpu timer interrupt which may trigger
676 * scheduler's load balance.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700677 */
Feng Tang55c844a2012-05-30 23:15:41 +0800678 local_irq_disable();
Alok Kataria76fac072010-10-11 14:37:08 -0700679 stop_other_cpus();
Miguel Boton4d022e32008-01-30 13:32:51 +0100680#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -0700681
682 lapic_shutdown();
683
OGAWA Hirofumic86c7fb2007-12-03 17:17:10 +0100684#ifdef CONFIG_HPET_TIMER
685 hpet_disable();
686#endif
Miguel Boton4d022e32008-01-30 13:32:51 +0100687
688#ifdef CONFIG_X86_64
FUJITA Tomonori338bac52009-10-27 16:34:44 +0900689 x86_platform.iommu_shutdown();
Miguel Boton4d022e32008-01-30 13:32:51 +0100690#endif
Eric W. Biedermandd2a1302005-06-25 14:57:55 -0700691}
692
Eduardo Habkostd1767202008-11-17 19:03:24 -0200693static void __machine_emergency_restart(int emergency)
694{
695 reboot_emergency = emergency;
696 machine_ops.emergency_restart();
697}
698
Jody Belka416e2d62008-02-12 23:37:48 +0000699static void native_machine_restart(char *__unused)
Jeremy Fitzhardinge973efae2007-05-02 19:27:06 +0200700{
Joe Perchesc767a542012-05-21 19:50:07 -0700701 pr_notice("machine restart\n");
Jeremy Fitzhardinge973efae2007-05-02 19:27:06 +0200702
Miguel Boton4d022e32008-01-30 13:32:51 +0100703 if (!reboot_force)
Eric W. Biederman6e3fbee2006-01-11 22:43:12 +0100704 machine_shutdown();
Eduardo Habkostd1767202008-11-17 19:03:24 -0200705 __machine_emergency_restart(0);
Jeremy Fitzhardinge07f33312007-05-02 19:27:11 +0200706}
707
Jody Belka416e2d62008-02-12 23:37:48 +0000708static void native_machine_halt(void)
Jeremy Fitzhardinge07f33312007-05-02 19:27:11 +0200709{
Michael D Labriola144d1022012-02-01 10:06:34 -0500710 /* Stop other cpus and apics */
Ivan Vecerad3ec5ca2008-11-11 14:33:44 +0100711 machine_shutdown();
712
Joseph Cihula840c2ba2009-06-30 19:31:02 -0700713 tboot_shutdown(TB_SHUTDOWN_HALT);
714
Ivan Vecerad3ec5ca2008-11-11 14:33:44 +0100715 stop_this_cpu(NULL);
Jeremy Fitzhardinge07f33312007-05-02 19:27:11 +0200716}
Miguel Boton4d022e32008-01-30 13:32:51 +0100717
Jody Belka416e2d62008-02-12 23:37:48 +0000718static void native_machine_power_off(void)
Miguel Boton4d022e32008-01-30 13:32:51 +0100719{
720 if (pm_power_off) {
721 if (!reboot_force)
722 machine_shutdown();
723 pm_power_off();
724 }
Michael D Labriola144d1022012-02-01 10:06:34 -0500725 /* A fallback in case there is no PM info available */
Joseph Cihula840c2ba2009-06-30 19:31:02 -0700726 tboot_shutdown(TB_SHUTDOWN_HALT);
Miguel Boton4d022e32008-01-30 13:32:51 +0100727}
728
Kees Cook404f6aa2016-08-08 16:29:06 -0700729struct machine_ops machine_ops __ro_after_init = {
Jody Belka416e2d62008-02-12 23:37:48 +0000730 .power_off = native_machine_power_off,
731 .shutdown = native_machine_shutdown,
732 .emergency_restart = native_machine_emergency_restart,
733 .restart = native_machine_restart,
Glauber Costaed23dc62008-03-17 16:08:38 -0300734 .halt = native_machine_halt,
Dave Young2965faa2015-09-09 15:38:55 -0700735#ifdef CONFIG_KEXEC_CORE
Glauber Costaed23dc62008-03-17 16:08:38 -0300736 .crash_shutdown = native_machine_crash_shutdown,
737#endif
Miguel Boton4d022e32008-01-30 13:32:51 +0100738};
Jody Belka416e2d62008-02-12 23:37:48 +0000739
740void machine_power_off(void)
741{
742 machine_ops.power_off();
743}
744
745void machine_shutdown(void)
746{
747 machine_ops.shutdown();
748}
749
750void machine_emergency_restart(void)
751{
Eduardo Habkostd1767202008-11-17 19:03:24 -0200752 __machine_emergency_restart(1);
Jody Belka416e2d62008-02-12 23:37:48 +0000753}
754
755void machine_restart(char *cmd)
756{
757 machine_ops.restart(cmd);
758}
759
760void machine_halt(void)
761{
762 machine_ops.halt();
763}
764
Dave Young2965faa2015-09-09 15:38:55 -0700765#ifdef CONFIG_KEXEC_CORE
Glauber Costaed23dc62008-03-17 16:08:38 -0300766void machine_crash_shutdown(struct pt_regs *regs)
767{
768 machine_ops.crash_shutdown(regs);
769}
770#endif
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200771
772
Xunlei Pang5bc32952017-03-13 10:50:19 +0100773/* This is the CPU performing the emergency shutdown work. */
774int crashing_cpu = -1;
775
Eduardo Habkostbb8dd272008-11-12 11:34:43 -0200776#if defined(CONFIG_SMP)
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200777
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200778static nmi_shootdown_cb shootdown_callback;
779
780static atomic_t waiting_for_crash_ipi;
Hidehiro Kawai58c56612015-12-14 11:19:10 +0100781static int crash_ipi_issued;
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200782
Don Zickus9c48f1c2011-09-30 15:06:21 -0400783static int crash_nmi_callback(unsigned int val, struct pt_regs *regs)
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200784{
785 int cpu;
786
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200787 cpu = raw_smp_processor_id();
788
Michael D Labriola144d1022012-02-01 10:06:34 -0500789 /*
790 * Don't do anything if this handler is invoked on crashing cpu.
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200791 * Otherwise, system will completely hang. Crashing cpu can get
792 * an NMI if system was initially booted with nmi_watchdog parameter.
793 */
794 if (cpu == crashing_cpu)
Don Zickus9c48f1c2011-09-30 15:06:21 -0400795 return NMI_HANDLED;
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200796 local_irq_disable();
797
Don Zickus9c48f1c2011-09-30 15:06:21 -0400798 shootdown_callback(cpu, regs);
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200799
800 atomic_dec(&waiting_for_crash_ipi);
801 /* Assume hlt works */
802 halt();
803 for (;;)
804 cpu_relax();
805
Don Zickus9c48f1c2011-09-30 15:06:21 -0400806 return NMI_HANDLED;
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200807}
808
809static void smp_send_nmi_allbutself(void)
810{
Ingo Molnardac5f412009-01-28 15:42:24 +0100811 apic->send_IPI_allbutself(NMI_VECTOR);
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200812}
813
Michael D Labriola144d1022012-02-01 10:06:34 -0500814/*
815 * Halt all other CPUs, calling the specified function on each of them
Eduardo Habkostbb8dd272008-11-12 11:34:43 -0200816 *
817 * This function can be used to halt all other CPUs on crash
818 * or emergency reboot time. The function passed as parameter
819 * will be called inside a NMI handler on all CPUs.
820 */
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200821void nmi_shootdown_cpus(nmi_shootdown_cb callback)
822{
823 unsigned long msecs;
Eduardo Habkostc415b3d2008-11-12 11:34:44 -0200824 local_irq_disable();
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200825
Michael D Labriola144d1022012-02-01 10:06:34 -0500826 /* Make a note of crashing cpu. Will be used in NMI callback. */
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200827 crashing_cpu = safe_smp_processor_id();
828
829 shootdown_callback = callback;
830
831 atomic_set(&waiting_for_crash_ipi, num_online_cpus() - 1);
832 /* Would it be better to replace the trap vector here? */
Don Zickus9c48f1c2011-09-30 15:06:21 -0400833 if (register_nmi_handler(NMI_LOCAL, crash_nmi_callback,
834 NMI_FLAG_FIRST, "crash"))
Michael D Labriola144d1022012-02-01 10:06:34 -0500835 return; /* Return what? */
836 /*
837 * Ensure the new callback function is set before sending
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200838 * out the NMI
839 */
840 wmb();
841
842 smp_send_nmi_allbutself();
843
Hidehiro Kawai58c56612015-12-14 11:19:10 +0100844 /* Kick CPUs looping in NMI context. */
845 WRITE_ONCE(crash_ipi_issued, 1);
846
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200847 msecs = 1000; /* Wait at most a second for the other cpus to stop */
848 while ((atomic_read(&waiting_for_crash_ipi) > 0) && msecs) {
849 mdelay(1);
850 msecs--;
851 }
852
853 /* Leave the nmi callback set */
854}
Hidehiro Kawai58c56612015-12-14 11:19:10 +0100855
Hidehiro Kawaib279d672015-12-14 11:19:13 +0100856/*
857 * Check if the crash dumping IPI got issued and if so, call its callback
858 * directly. This function is used when we have already been in NMI handler.
859 * It doesn't return.
860 */
861void run_crash_ipi_callback(struct pt_regs *regs)
862{
863 if (crash_ipi_issued)
864 crash_nmi_callback(0, regs);
865}
866
Hidehiro Kawai58c56612015-12-14 11:19:10 +0100867/* Override the weak function in kernel/panic.c */
868void nmi_panic_self_stop(struct pt_regs *regs)
869{
870 while (1) {
Hidehiro Kawaib279d672015-12-14 11:19:13 +0100871 /* If no CPU is preparing crash dump, we simply loop here. */
872 run_crash_ipi_callback(regs);
Hidehiro Kawai58c56612015-12-14 11:19:10 +0100873 cpu_relax();
874 }
875}
876
Eduardo Habkostbb8dd272008-11-12 11:34:43 -0200877#else /* !CONFIG_SMP */
878void nmi_shootdown_cpus(nmi_shootdown_cb callback)
879{
880 /* No other CPUs to shoot down */
881}
Hidehiro Kawaib279d672015-12-14 11:19:13 +0100882
883void run_crash_ipi_callback(struct pt_regs *regs)
884{
885}
Eduardo Habkost2ddded22008-11-12 11:34:42 -0200886#endif