2005-04-17 02:20:36 +04:00
# include <linux/module.h>
2006-12-07 07:40:06 +03:00
# include <linux/reboot.h>
2008-01-30 15:32:51 +03:00
# include <linux/init.h>
# include <linux/pm.h>
# include <linux/efi.h>
2009-08-03 16:47:32 +04:00
# include <linux/dmi.h>
2009-10-07 17:09:06 +04:00
# include <linux/sched.h>
2009-09-02 05:25:07 +04:00
# include <linux/tboot.h>
2008-01-30 15:32:51 +03:00
# include <acpi/reboot.h>
# include <asm/io.h>
2005-04-17 02:20:36 +04:00
# include <asm/apic.h>
2005-09-04 02:56:38 +04:00
# include <asm/desc.h>
2008-01-30 15:32:51 +03:00
# include <asm/hpet.h>
2008-03-18 02:37:13 +03:00
# include <asm/pgtable.h>
2008-04-28 03:15:59 +04:00
# include <asm/proto.h>
2007-05-02 21:27:06 +04:00
# include <asm/reboot_fixups.h>
2007-05-02 21:27:11 +04:00
# include <asm/reboot.h>
2008-12-27 16:02:28 +03:00
# include <asm/pci_x86.h>
2008-11-18 00:03:24 +03:00
# include <asm/virtext.h>
2009-01-07 19:05:48 +03:00
# include <asm/cpu.h>
2005-04-17 02:20:36 +04:00
2008-01-30 15:32:51 +03:00
# ifdef CONFIG_X86_32
# include <linux / ctype.h>
# include <linux / mc146818rtc.h>
# else
2009-10-27 10:34:44 +03:00
# include <asm / x86_init.h>
2008-01-30 15:32:51 +03:00
# endif
2005-04-17 02:20:36 +04:00
/*
* Power off function , if any
*/
void ( * pm_power_off ) ( void ) ;
2005-06-23 11:08:33 +04:00
EXPORT_SYMBOL ( pm_power_off ) ;
2005-04-17 02:20:36 +04:00
2008-05-12 17:43:38 +04:00
static const struct desc_ptr no_idt = { } ;
2005-04-17 02:20:36 +04:00
static int reboot_mode ;
2008-11-04 17:52:44 +03:00
enum reboot_type reboot_type = BOOT_KBD ;
2008-01-30 15:32:51 +03:00
int reboot_force ;
2005-04-17 02:20:36 +04:00
2008-01-30 15:32:51 +03:00
# if defined(CONFIG_X86_32) && defined(CONFIG_SMP)
2005-04-17 02:20:36 +04:00
static int reboot_cpu = - 1 ;
# endif
2008-01-30 15:32:51 +03:00
2008-11-18 00:03:24 +03:00
/* This is set if we need to go through the 'emergency' path.
* When machine_emergency_restart ( ) is called , we may be on
* an inconsistent state and won ' t be able to do a clean cleanup
*/
static int reboot_emergency ;
2008-11-12 03:19:48 +03:00
/* This is set by the PCI code if either type 1 or type 2 PCI is detected */
bool port_cf9_safe = false ;
/* reboot=b[ios] | s[mp] | t[riple] | k[bd] | e[fi] [, [w]arm | [c]old] | p[ci]
2008-01-30 15:32:51 +03:00
warm Don ' t set the cold reboot flag
cold Set the cold reboot flag
bios Reboot by jumping through the BIOS ( only for X86_32 )
smp Reboot by executing reset on BSP or other CPU ( only for X86_32 )
triple Force a triple fault ( init )
kbd Use the keyboard controller . cold reset ( default )
acpi Use the RESET_REG in the FADT
efi Use efi reset_system runtime service
2008-11-12 03:19:48 +03:00
pci Use the so - called " PCI reset register " , CF9
2008-01-30 15:32:51 +03:00
force Avoid anything that could hang .
*/
2005-04-17 02:20:36 +04:00
static int __init reboot_setup ( char * str )
{
2008-01-30 15:32:51 +03:00
for ( ; ; ) {
2005-04-17 02:20:36 +04:00
switch ( * str ) {
2008-01-30 15:32:51 +03:00
case ' w ' :
2005-04-17 02:20:36 +04:00
reboot_mode = 0x1234 ;
break ;
2008-01-30 15:32:51 +03:00
case ' c ' :
reboot_mode = 0 ;
2005-04-17 02:20:36 +04:00
break ;
2008-01-30 15:32:51 +03:00
# ifdef CONFIG_X86_32
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_SMP
2008-01-30 15:32:51 +03:00
case ' s ' :
2005-09-13 12:25:48 +04:00
if ( isdigit ( * ( str + 1 ) ) ) {
2005-04-17 02:20:36 +04:00
reboot_cpu = ( int ) ( * ( str + 1 ) - ' 0 ' ) ;
2005-09-13 12:25:48 +04:00
if ( isdigit ( * ( str + 2 ) ) )
2005-04-17 02:20:36 +04:00
reboot_cpu = reboot_cpu * 10 + ( int ) ( * ( str + 2 ) - ' 0 ' ) ;
}
2008-01-30 15:32:51 +03:00
/* we will leave sorting out the final value
when we are ready to reboot , since we might not
have set up boot_cpu_id or smp_num_cpu */
2005-04-17 02:20:36 +04:00
break ;
2008-01-30 15:32:51 +03:00
# endif /* CONFIG_SMP */
case ' b ' :
2005-04-17 02:20:36 +04:00
# endif
2008-01-30 15:32:51 +03:00
case ' a ' :
case ' k ' :
case ' t ' :
case ' e ' :
2008-11-12 03:19:48 +03:00
case ' p ' :
2008-01-30 15:32:51 +03:00
reboot_type = * str ;
break ;
case ' f ' :
reboot_force = 1 ;
break ;
2005-04-17 02:20:36 +04:00
}
2008-01-30 15:32:51 +03:00
str = strchr ( str , ' , ' ) ;
if ( str )
2005-04-17 02:20:36 +04:00
str + + ;
else
break ;
}
return 1 ;
}
__setup ( " reboot= " , reboot_setup ) ;
2008-01-30 15:32:51 +03:00
# ifdef CONFIG_X86_32
2005-04-17 02:20:36 +04:00
/*
* Reboot options and system auto - detection code provided by
* Dell Inc . so their systems " just work " . : - )
*/
/*
2008-01-30 15:32:51 +03:00
* Some machines require the " reboot=b " commandline option ,
* this quirk makes that automatic .
2005-04-17 02:20:36 +04:00
*/
2007-10-03 23:15:40 +04:00
static int __init set_bios_reboot ( const struct dmi_system_id * d )
2005-04-17 02:20:36 +04:00
{
2008-01-30 15:32:51 +03:00
if ( reboot_type ! = BOOT_BIOS ) {
reboot_type = BOOT_BIOS ;
2005-04-17 02:20:36 +04:00
printk ( KERN_INFO " %s series board detected. Selecting BIOS-method for reboots. \n " , d - > ident ) ;
}
return 0 ;
}
static struct dmi_system_id __initdata reboot_dmi_table [ ] = {
2007-06-01 11:46:40 +04:00
{ /* Handle problems with rebooting on Dell E520's */
. callback = set_bios_reboot ,
. ident = " Dell E520 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Dell DM061 " ) ,
} ,
} ,
2005-04-17 02:20:36 +04:00
{ /* Handle problems with rebooting on Dell 1300's */
2005-06-26 01:57:55 +04:00
. callback = set_bios_reboot ,
2005-04-17 02:20:36 +04:00
. ident = " Dell PowerEdge 1300 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Computer Corporation " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " PowerEdge 1300/ " ) ,
} ,
} ,
{ /* Handle problems with rebooting on Dell 300's */
. callback = set_bios_reboot ,
. ident = " Dell PowerEdge 300 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Computer Corporation " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " PowerEdge 300/ " ) ,
} ,
} ,
2007-07-21 19:11:11 +04:00
{ /* Handle problems with rebooting on Dell Optiplex 745's SFF*/
. callback = set_bios_reboot ,
. ident = " Dell OptiPlex 745 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " OptiPlex 745 " ) ,
} ,
} ,
2008-03-05 02:05:41 +03:00
{ /* Handle problems with rebooting on Dell Optiplex 745's DFF*/
. callback = set_bios_reboot ,
. ident = " Dell OptiPlex 745 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " OptiPlex 745 " ) ,
DMI_MATCH ( DMI_BOARD_NAME , " 0MM599 " ) ,
} ,
} ,
2008-03-12 18:27:56 +03:00
{ /* Handle problems with rebooting on Dell Optiplex 745 with 0KW626 */
. callback = set_bios_reboot ,
. ident = " Dell OptiPlex 745 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " OptiPlex 745 " ) ,
DMI_MATCH ( DMI_BOARD_NAME , " 0KW626 " ) ,
} ,
} ,
2008-11-14 09:55:51 +03:00
{ /* Handle problems with rebooting on Dell Optiplex 330 with 0KP561 */
. callback = set_bios_reboot ,
. ident = " Dell OptiPlex 330 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " OptiPlex 330 " ) ,
DMI_MATCH ( DMI_BOARD_NAME , " 0KP561 " ) ,
} ,
} ,
2009-06-05 14:02:38 +04:00
{ /* Handle problems with rebooting on Dell Optiplex 360 with 0T656F */
. callback = set_bios_reboot ,
. ident = " Dell OptiPlex 360 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " OptiPlex 360 " ) ,
DMI_MATCH ( DMI_BOARD_NAME , " 0T656F " ) ,
} ,
} ,
2010-01-28 02:29:18 +03:00
{ /* Handle problems with rebooting on Dell OptiPlex 760 with 0G919G*/
. callback = set_bios_reboot ,
. ident = " Dell OptiPlex 760 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " OptiPlex 760 " ) ,
DMI_MATCH ( DMI_BOARD_NAME , " 0G919G " ) ,
} ,
} ,
2005-04-17 02:20:36 +04:00
{ /* Handle problems with rebooting on Dell 2400's */
. callback = set_bios_reboot ,
. ident = " Dell PowerEdge 2400 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Computer Corporation " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " PowerEdge 2400 " ) ,
} ,
} ,
2008-07-17 15:50:15 +04:00
{ /* Handle problems with rebooting on Dell T5400's */
. callback = set_bios_reboot ,
. ident = " Dell Precision T5400 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Precision WorkStation T5400 " ) ,
} ,
} ,
2010-06-19 17:32:25 +04:00
{ /* Handle problems with rebooting on Dell T7400's */
. callback = set_bios_reboot ,
. ident = " Dell Precision T7400 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Precision WorkStation T7400 " ) ,
} ,
} ,
2006-01-06 11:12:20 +03:00
{ /* Handle problems with rebooting on HP laptops */
2005-11-30 06:34:35 +03:00
. callback = set_bios_reboot ,
2006-01-06 11:12:20 +03:00
. ident = " HP Compaq Laptop " ,
2005-11-30 06:34:35 +03:00
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Hewlett-Packard " ) ,
2006-01-06 11:12:20 +03:00
DMI_MATCH ( DMI_PRODUCT_NAME , " HP Compaq " ) ,
2005-11-30 06:34:35 +03:00
} ,
} ,
2009-03-04 22:53:00 +03:00
{ /* Handle problems with rebooting on Dell XPS710 */
. callback = set_bios_reboot ,
. ident = " Dell XPS710 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Dell XPS710 " ) ,
} ,
} ,
2009-03-26 23:45:28 +03:00
{ /* Handle problems with rebooting on Dell DXP061 */
. callback = set_bios_reboot ,
. ident = " Dell DXP061 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Dell Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Dell DXP061 " ) ,
} ,
} ,
2009-05-22 07:35:50 +04:00
{ /* Handle problems with rebooting on Sony VGN-Z540N */
. callback = set_bios_reboot ,
. ident = " Sony VGN-Z540N " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Sony Corporation " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " VGN-Z540N " ) ,
} ,
} ,
2009-07-20 19:48:17 +04:00
{ /* Handle problems with rebooting on CompuLab SBC-FITPC2 */
. callback = set_bios_reboot ,
. ident = " CompuLab SBC-FITPC2 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " CompuLab " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " SBC-FITPC2 " ) ,
} ,
} ,
2009-12-05 02:42:22 +03:00
{ /* Handle problems with rebooting on ASUS P4S800 */
. callback = set_bios_reboot ,
. ident = " ASUS P4S800 " ,
. matches = {
DMI_MATCH ( DMI_BOARD_VENDOR , " ASUSTeK Computer INC. " ) ,
DMI_MATCH ( DMI_BOARD_NAME , " P4S800 " ) ,
} ,
} ,
2005-04-17 02:20:36 +04:00
{ }
} ;
static int __init reboot_init ( void )
{
dmi_check_system ( reboot_dmi_table ) ;
return 0 ;
}
core_initcall ( reboot_init ) ;
/* The following code and data reboots the machine by switching to real
mode and jumping to the BIOS reset entry point , as if the CPU has
really been reset . The previous version asked the keyboard
controller to pulse the CPU reset line , which is more thorough , but
doesn ' t work with at least one type of 486 motherboard . It is easy
to stop this code working ; hence the copious comments . */
2008-05-12 17:43:38 +04:00
static const unsigned long long
2005-04-17 02:20:36 +04:00
real_mode_gdt_entries [ 3 ] =
{
0x0000000000000000ULL , /* Null descriptor */
2008-05-12 17:43:38 +04:00
0x00009b000000ffffULL , /* 16-bit real-mode 64k code at 0x00000000 */
0x000093000100ffffULL /* 16-bit real-mode 64k data at 0x00000100 */
2005-04-17 02:20:36 +04:00
} ;
2008-05-12 17:43:38 +04:00
static const struct desc_ptr
2006-09-26 10:32:22 +04:00
real_mode_gdt = { sizeof ( real_mode_gdt_entries ) - 1 , ( long ) real_mode_gdt_entries } ,
2008-01-30 15:32:51 +03:00
real_mode_idt = { 0x3ff , 0 } ;
2005-04-17 02:20:36 +04:00
/* This is 16-bit protected mode code to disable paging and the cache,
switch to real mode and jump to the BIOS reset code .
The instruction that switches to real mode by writing to CR0 must be
followed immediately by a far jump instruction , which set CS to a
valid value for real mode , and flushes the prefetch queue to avoid
running instructions that have already been decoded in protected
mode .
Clears all the flags except ET , especially PG ( paging ) , PE
( protected - mode enable ) and TS ( task switch for coprocessor state
save ) . Flushes the TLB after paging has been disabled . Sets CD and
NW , to disable the cache on a 486 , and invalidates the cache . This
is more like the state of a 486 after reset . I don ' t know if
something else should be done for other chips .
More could be done here to set up the registers as if a CPU reset had
occurred ; hopefully real BIOSs don ' t assume much . */
2008-05-12 17:43:38 +04:00
static const unsigned char real_mode_switch [ ] =
2005-04-17 02:20:36 +04:00
{
0x66 , 0x0f , 0x20 , 0xc0 , /* movl %cr0,%eax */
0x66 , 0x83 , 0xe0 , 0x11 , /* andl $0x00000011,%eax */
0x66 , 0x0d , 0x00 , 0x00 , 0x00 , 0x60 , /* orl $0x60000000,%eax */
0x66 , 0x0f , 0x22 , 0xc0 , /* movl %eax,%cr0 */
0x66 , 0x0f , 0x22 , 0xd8 , /* movl %eax,%cr3 */
0x66 , 0x0f , 0x20 , 0xc3 , /* movl %cr0,%ebx */
0x66 , 0x81 , 0xe3 , 0x00 , 0x00 , 0x00 , 0x60 , /* andl $0x60000000,%ebx */
0x74 , 0x02 , /* jz f */
0x0f , 0x09 , /* wbinvd */
0x24 , 0x10 , /* f: andb $0x10,al */
0x66 , 0x0f , 0x22 , 0xc0 /* movl %eax,%cr0 */
} ;
2008-05-12 17:43:38 +04:00
static const unsigned char jump_to_bios [ ] =
2005-04-17 02:20:36 +04:00
{
0xea , 0x00 , 0x00 , 0xff , 0xff /* ljmp $0xffff,$0x0000 */
} ;
/*
* Switch to real mode and then execute the code
* specified by the code and length parameters .
* We assume that length will aways be less that 100 !
*/
2008-05-12 17:43:38 +04:00
void machine_real_restart ( const unsigned char * code , int length )
2005-04-17 02:20:36 +04:00
{
local_irq_disable ( ) ;
/* Write zero to CMOS register number 0x0f, which the BIOS POST
routine will recognize as telling it to do a proper reboot . ( Well
that ' s what this book in front of me says - - it may only apply to
the Phoenix BIOS though , it ' s not clear ) . At the same time ,
disable NMIs by setting the top bit in the CMOS address register ,
as we ' re about to do peculiar things to the CPU . I ' m not sure if
` outb_p ' is needed instead of just ` outb ' . Use it to be on the
safe side . ( Yes , CMOS_WRITE does outb_p ' s . - Paul G . )
*/
2007-05-02 21:27:18 +04:00
spin_lock ( & rtc_lock ) ;
2005-04-17 02:20:36 +04:00
CMOS_WRITE ( 0x00 , 0x8f ) ;
2007-05-02 21:27:18 +04:00
spin_unlock ( & rtc_lock ) ;
2005-04-17 02:20:36 +04:00
/* Remap the kernel at virtual address zero, as well as offset zero
from the kernel segment . This assumes the kernel segment starts at
virtual address PAGE_OFFSET . */
2008-03-18 02:37:13 +03:00
memcpy ( swapper_pg_dir , swapper_pg_dir + KERNEL_PGD_BOUNDARY ,
2008-01-30 15:32:51 +03:00
sizeof ( swapper_pg_dir [ 0 ] ) * KERNEL_PGD_PTRS ) ;
2005-04-17 02:20:36 +04:00
/*
* Use ` swapper_pg_dir ' as our page directory .
*/
load_cr3 ( swapper_pg_dir ) ;
/* Write 0x1234 to absolute memory location 0x472. The BIOS reads
this on booting to tell it to " Bypass memory test (also warm
boot ) " . This seems like a fairly standard thing that gets set by
REBOOT . COM programs , and the previous reset routine did this
too . */
* ( ( unsigned short * ) 0x472 ) = reboot_mode ;
/* For the switch to real mode, copy some code to low memory. It has
to be in the first 64 k because it is running in 16 - bit mode , and it
has to have the same physical and virtual address , because it turns
off paging . Copy it near the end of the first page , out of the way
of BIOS variables . */
2008-01-30 15:32:51 +03:00
memcpy ( ( void * ) ( 0x1000 - sizeof ( real_mode_switch ) - 100 ) ,
2005-04-17 02:20:36 +04:00
real_mode_switch , sizeof ( real_mode_switch ) ) ;
2008-01-30 15:32:51 +03:00
memcpy ( ( void * ) ( 0x1000 - 100 ) , code , length ) ;
2005-04-17 02:20:36 +04:00
/* Set up the IDT for real mode. */
2005-09-04 02:56:38 +04:00
load_idt ( & real_mode_idt ) ;
2005-04-17 02:20:36 +04:00
/* Set up a GDT from which we can load segment descriptors for real
mode . The GDT is not used in real mode ; it is just needed here to
prepare the descriptors . */
2005-09-04 02:56:38 +04:00
load_gdt ( & real_mode_gdt ) ;
2005-04-17 02:20:36 +04:00
/* Load the data segment registers, and thus the descriptors ready for
real mode . The base address of each segment is 0x100 , 16 times the
selector value being loaded here . This is so that the segment
registers don ' t have to be reloaded after switching to real mode :
the values are consistent for real mode operation already . */
__asm__ __volatile__ ( " movl $0x0010,%%eax \n "
" \t movl %%eax,%%ds \n "
" \t movl %%eax,%%es \n "
" \t movl %%eax,%%fs \n "
" \t movl %%eax,%%gs \n "
" \t movl %%eax,%%ss " : : : " eax " ) ;
/* Jump to the 16-bit code that we copied earlier. It disables paging
and the cache , switches to real mode , and jumps to the BIOS reset
entry point . */
__asm__ __volatile__ ( " ljmp $0x0008,%0 "
:
2008-01-30 15:32:51 +03:00
: " i " ( ( void * ) ( 0x1000 - sizeof ( real_mode_switch ) - 100 ) ) ) ;
2005-04-17 02:20:36 +04:00
}
2005-06-23 11:08:33 +04:00
# ifdef CONFIG_APM_MODULE
EXPORT_SYMBOL ( machine_real_restart ) ;
# endif
2005-04-17 02:20:36 +04:00
2008-01-30 15:32:51 +03:00
# endif /* CONFIG_X86_32 */
2009-08-03 16:47:32 +04:00
/*
2009-08-04 20:39:31 +04:00
* Some Apple MacBook and MacBookPro ' s needs reboot = p to be able to reboot
2009-08-03 16:47:32 +04:00
*/
static int __init set_pci_reboot ( const struct dmi_system_id * d )
{
if ( reboot_type ! = BOOT_CF9 ) {
reboot_type = BOOT_CF9 ;
printk ( KERN_INFO " %s series board detected. "
" Selecting PCI-method for reboots. \n " , d - > ident ) ;
}
return 0 ;
}
static struct dmi_system_id __initdata pci_reboot_dmi_table [ ] = {
2009-08-10 22:34:40 +04:00
{ /* Handle problems with rebooting on Apple MacBook5 */
2009-08-03 16:47:32 +04:00
. callback = set_pci_reboot ,
2009-08-10 22:34:40 +04:00
. ident = " Apple MacBook5 " ,
2009-08-03 16:47:32 +04:00
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Apple Inc. " ) ,
2009-08-10 22:34:40 +04:00
DMI_MATCH ( DMI_PRODUCT_NAME , " MacBook5 " ) ,
2009-08-03 16:47:32 +04:00
} ,
} ,
2009-08-10 22:34:40 +04:00
{ /* Handle problems with rebooting on Apple MacBookPro5 */
2009-08-04 20:39:31 +04:00
. callback = set_pci_reboot ,
2009-08-10 22:34:40 +04:00
. ident = " Apple MacBookPro5 " ,
2009-08-04 20:39:31 +04:00
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Apple Inc. " ) ,
2009-08-10 22:34:40 +04:00
DMI_MATCH ( DMI_PRODUCT_NAME , " MacBookPro5 " ) ,
2009-08-04 20:39:31 +04:00
} ,
} ,
2009-11-02 13:51:11 +03:00
{ /* Handle problems with rebooting on Apple Macmini3,1 */
. callback = set_pci_reboot ,
. ident = " Apple Macmini3,1 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Apple Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " Macmini3,1 " ) ,
} ,
} ,
2010-02-17 02:17:29 +03:00
{ /* Handle problems with rebooting on the iMac9,1. */
. callback = set_pci_reboot ,
. ident = " Apple iMac9,1 " ,
. matches = {
DMI_MATCH ( DMI_SYS_VENDOR , " Apple Inc. " ) ,
DMI_MATCH ( DMI_PRODUCT_NAME , " iMac9,1 " ) ,
} ,
} ,
2009-08-03 16:47:32 +04:00
{ }
} ;
static int __init pci_reboot_init ( void )
{
dmi_check_system ( pci_reboot_dmi_table ) ;
return 0 ;
}
core_initcall ( pci_reboot_init ) ;
2008-01-30 15:32:51 +03:00
static inline void kb_wait ( void )
{
int i ;
2008-01-30 15:33:25 +03:00
for ( i = 0 ; i < 0x10000 ; i + + ) {
if ( ( inb ( 0x64 ) & 0x02 ) = = 0 )
2008-01-30 15:32:51 +03:00
break ;
2008-01-30 15:33:25 +03:00
udelay ( 2 ) ;
}
2008-01-30 15:32:51 +03:00
}
2008-11-18 00:03:24 +03:00
static void vmxoff_nmi ( int cpu , struct die_args * args )
{
cpu_emergency_vmxoff ( ) ;
}
/* Use NMIs as IPIs to tell all CPUs to disable virtualization
*/
static void emergency_vmx_disable_all ( void )
{
/* Just make sure we won't change CPUs while doing this */
local_irq_disable ( ) ;
/* We need to disable VMX on all CPUs before rebooting, otherwise
* we risk hanging up the machine , because the CPU ignore INIT
* signals when VMX is enabled .
*
* We can ' t take any locks and we may be on an inconsistent
* state , so we use NMIs as IPIs to tell the other CPUs to disable
* VMX and halt .
*
* For safety , we will avoid running the nmi_shootdown_cpus ( )
* stuff unnecessarily , but we don ' t have a way to check
* if other CPUs have VMX enabled . So we will call it only if the
* CPU we are running on has VMX enabled .
*
* We will miss cases where VMX is not enabled on all CPUs . This
* shouldn ' t do much harm because KVM always enable VMX on all
* CPUs anyway . But we can miss it on the small window where KVM
* is still enabling VMX .
*/
if ( cpu_has_vmx ( ) & & cpu_vmx_enabled ( ) ) {
/* Disable VMX on this CPU.
*/
cpu_vmxoff ( ) ;
/* Halt and disable VMX on the other CPUs */
nmi_shootdown_cpus ( vmxoff_nmi ) ;
}
}
2008-03-06 20:29:43 +03:00
void __attribute__ ( ( weak ) ) mach_reboot_fixups ( void )
{
}
2008-02-13 02:37:48 +03:00
static void native_machine_emergency_restart ( void )
2005-04-17 02:20:36 +04:00
{
2008-01-30 15:32:51 +03:00
int i ;
2008-11-18 00:03:24 +03:00
if ( reboot_emergency )
emergency_vmx_disable_all ( ) ;
2009-07-01 06:31:02 +04:00
tboot_shutdown ( TB_SHUTDOWN_REBOOT ) ;
2008-01-30 15:32:51 +03:00
/* Tell the BIOS if we want cold or warm reboot */
* ( ( unsigned short * ) __va ( 0x472 ) ) = reboot_mode ;
for ( ; ; ) {
/* Could also try the reset bit in the Hammer NB */
switch ( reboot_type ) {
case BOOT_KBD :
2008-03-06 20:29:43 +03:00
mach_reboot_fixups ( ) ; /* for board specific fixups */
2008-01-30 15:32:51 +03:00
for ( i = 0 ; i < 10 ; i + + ) {
kb_wait ( ) ;
udelay ( 50 ) ;
outb ( 0xfe , 0x64 ) ; /* pulse reset low */
udelay ( 50 ) ;
}
case BOOT_TRIPLE :
2008-05-12 17:43:38 +04:00
load_idt ( & no_idt ) ;
2008-01-30 15:32:51 +03:00
__asm__ __volatile__ ( " int3 " ) ;
reboot_type = BOOT_KBD ;
break ;
# ifdef CONFIG_X86_32
case BOOT_BIOS :
machine_real_restart ( jump_to_bios , sizeof ( jump_to_bios ) ) ;
reboot_type = BOOT_KBD ;
break ;
# endif
case BOOT_ACPI :
acpi_reboot ( ) ;
reboot_type = BOOT_KBD ;
break ;
case BOOT_EFI :
if ( efi_enabled )
2008-11-12 03:19:48 +03:00
efi . reset_system ( reboot_mode ?
EFI_RESET_WARM :
EFI_RESET_COLD ,
2008-01-30 15:32:51 +03:00
EFI_SUCCESS , 0 , NULL ) ;
2008-11-24 11:50:09 +03:00
reboot_type = BOOT_KBD ;
2008-11-12 03:19:48 +03:00
break ;
2008-01-30 15:32:51 +03:00
2008-11-12 03:19:48 +03:00
case BOOT_CF9 :
port_cf9_safe = true ;
/* fall through */
2008-01-30 15:32:51 +03:00
2008-11-12 03:19:48 +03:00
case BOOT_CF9_COND :
if ( port_cf9_safe ) {
u8 cf9 = inb ( 0xcf9 ) & ~ 6 ;
outb ( cf9 | 2 , 0xcf9 ) ; /* Request hard reset */
udelay ( 50 ) ;
outb ( cf9 | 6 , 0xcf9 ) ; /* Actually do the reset */
udelay ( 50 ) ;
}
2008-01-30 15:32:51 +03:00
reboot_type = BOOT_KBD ;
break ;
}
}
}
2008-03-17 22:08:39 +03:00
void native_machine_shutdown ( void )
2008-01-30 15:32:51 +03:00
{
/* Stop the cpus and apics */
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_SMP
2005-06-26 01:57:55 +04:00
/* The boot cpu is always logical cpu 0 */
2008-07-16 01:14:30 +04:00
int reboot_cpu_id = 0 ;
2005-06-26 01:57:55 +04:00
2008-01-30 15:32:51 +03:00
# ifdef CONFIG_X86_32
2005-06-26 01:57:55 +04:00
/* See if there has been given a command line override */
2009-01-01 05:08:46 +03:00
if ( ( reboot_cpu ! = - 1 ) & & ( reboot_cpu < nr_cpu_ids ) & &
2008-07-25 05:21:31 +04:00
cpu_online ( reboot_cpu ) )
2005-06-26 01:57:55 +04:00
reboot_cpu_id = reboot_cpu ;
2008-01-30 15:32:51 +03:00
# endif
2005-04-17 02:20:36 +04:00
2008-01-30 15:32:51 +03:00
/* Make certain the cpu I'm about to reboot on is online */
2008-07-25 05:21:31 +04:00
if ( ! cpu_online ( reboot_cpu_id ) )
2005-06-26 01:57:55 +04:00
reboot_cpu_id = smp_processor_id ( ) ;
/* Make certain I only run on the appropriate processor */
2009-01-01 05:08:46 +03:00
set_cpus_allowed_ptr ( current , cpumask_of ( reboot_cpu_id ) ) ;
2005-06-26 01:57:55 +04:00
2008-01-30 15:32:51 +03:00
/* O.K Now that I'm on the appropriate processor,
* stop all of the others .
2005-04-17 02:20:36 +04:00
*/
smp_send_stop ( ) ;
2008-01-30 15:32:51 +03:00
# endif
2005-04-17 02:20:36 +04:00
lapic_shutdown ( ) ;
# ifdef CONFIG_X86_IO_APIC
disable_IO_APIC ( ) ;
# endif
2008-01-30 15:32:51 +03:00
2007-12-03 19:17:10 +03:00
# ifdef CONFIG_HPET_TIMER
hpet_disable ( ) ;
# endif
2005-06-26 01:57:55 +04:00
2008-01-30 15:32:51 +03:00
# ifdef CONFIG_X86_64
2009-10-27 10:34:44 +03:00
x86_platform . iommu_shutdown ( ) ;
2008-01-30 15:32:51 +03:00
# endif
2007-05-02 21:27:06 +04:00
}
2008-11-18 00:03:24 +03:00
static void __machine_emergency_restart ( int emergency )
{
reboot_emergency = emergency ;
machine_ops . emergency_restart ( ) ;
}
2008-02-13 02:37:48 +03:00
static void native_machine_restart ( char * __unused )
2005-06-26 01:57:55 +04:00
{
2008-01-30 15:32:51 +03:00
printk ( " machine restart \n " ) ;
2005-04-17 02:20:36 +04:00
2008-01-30 15:32:51 +03:00
if ( ! reboot_force )
machine_shutdown ( ) ;
2008-11-18 00:03:24 +03:00
__machine_emergency_restart ( 0 ) ;
2005-07-26 21:41:26 +04:00
}
2008-02-13 02:37:48 +03:00
static void native_machine_halt ( void )
2005-04-17 02:20:36 +04:00
{
2008-11-11 16:33:44 +03:00
/* stop other cpus and apics */
machine_shutdown ( ) ;
2009-07-01 06:31:02 +04:00
tboot_shutdown ( TB_SHUTDOWN_HALT ) ;
2008-11-11 16:33:44 +03:00
/* stop this cpu */
stop_this_cpu ( NULL ) ;
2005-04-17 02:20:36 +04:00
}
2008-02-13 02:37:48 +03:00
static void native_machine_power_off ( void )
2005-04-17 02:20:36 +04:00
{
2006-01-12 00:43:12 +03:00
if ( pm_power_off ) {
2008-01-30 15:32:51 +03:00
if ( ! reboot_force )
machine_shutdown ( ) ;
2005-04-17 02:20:36 +04:00
pm_power_off ( ) ;
2006-01-12 00:43:12 +03:00
}
2009-07-01 06:31:02 +04:00
/* a fallback in case there is no PM info available */
tboot_shutdown ( TB_SHUTDOWN_HALT ) ;
2005-04-17 02:20:36 +04:00
}
2007-05-02 21:27:11 +04:00
struct machine_ops machine_ops = {
2008-02-13 02:37:48 +03:00
. power_off = native_machine_power_off ,
. shutdown = native_machine_shutdown ,
. emergency_restart = native_machine_emergency_restart ,
. restart = native_machine_restart ,
2008-03-17 22:08:38 +03:00
. halt = native_machine_halt ,
# ifdef CONFIG_KEXEC
. crash_shutdown = native_machine_crash_shutdown ,
# endif
2007-05-02 21:27:11 +04:00
} ;
2008-02-13 02:37:48 +03:00
void machine_power_off ( void )
{
machine_ops . power_off ( ) ;
}
void machine_shutdown ( void )
{
machine_ops . shutdown ( ) ;
}
void machine_emergency_restart ( void )
{
2008-11-18 00:03:24 +03:00
__machine_emergency_restart ( 1 ) ;
2008-02-13 02:37:48 +03:00
}
void machine_restart ( char * cmd )
{
machine_ops . restart ( cmd ) ;
}
void machine_halt ( void )
{
machine_ops . halt ( ) ;
}
2008-03-17 22:08:38 +03:00
# ifdef CONFIG_KEXEC
void machine_crash_shutdown ( struct pt_regs * regs )
{
machine_ops . crash_shutdown ( regs ) ;
}
# endif
2008-11-12 16:34:42 +03:00
2008-11-12 16:34:43 +03:00
# if defined(CONFIG_SMP)
2008-11-12 16:34:42 +03:00
/* This keeps a track of which one is crashing cpu. */
static int crashing_cpu ;
static nmi_shootdown_cb shootdown_callback ;
static atomic_t waiting_for_crash_ipi ;
static int crash_nmi_callback ( struct notifier_block * self ,
unsigned long val , void * data )
{
int cpu ;
if ( val ! = DIE_NMI_IPI )
return NOTIFY_OK ;
cpu = raw_smp_processor_id ( ) ;
/* Don't do anything if this handler is invoked on crashing cpu.
* Otherwise , system will completely hang . Crashing cpu can get
* an NMI if system was initially booted with nmi_watchdog parameter .
*/
if ( cpu = = crashing_cpu )
return NOTIFY_STOP ;
local_irq_disable ( ) ;
shootdown_callback ( cpu , ( struct die_args * ) data ) ;
atomic_dec ( & waiting_for_crash_ipi ) ;
/* Assume hlt works */
halt ( ) ;
for ( ; ; )
cpu_relax ( ) ;
return 1 ;
}
static void smp_send_nmi_allbutself ( void )
{
2009-01-28 17:42:24 +03:00
apic - > send_IPI_allbutself ( NMI_VECTOR ) ;
2008-11-12 16:34:42 +03:00
}
static struct notifier_block crash_nmi_nb = {
. notifier_call = crash_nmi_callback ,
} ;
2008-11-12 16:34:43 +03:00
/* Halt all other CPUs, calling the specified function on each of them
*
* This function can be used to halt all other CPUs on crash
* or emergency reboot time . The function passed as parameter
* will be called inside a NMI handler on all CPUs .
*/
2008-11-12 16:34:42 +03:00
void nmi_shootdown_cpus ( nmi_shootdown_cb callback )
{
unsigned long msecs ;
2008-11-12 16:34:44 +03:00
local_irq_disable ( ) ;
2008-11-12 16:34:42 +03:00
/* Make a note of crashing cpu. Will be used in NMI callback.*/
crashing_cpu = safe_smp_processor_id ( ) ;
shootdown_callback = callback ;
atomic_set ( & waiting_for_crash_ipi , num_online_cpus ( ) - 1 ) ;
/* Would it be better to replace the trap vector here? */
if ( register_die_notifier ( & crash_nmi_nb ) )
return ; /* return what? */
/* Ensure the new callback function is set before sending
* out the NMI
*/
wmb ( ) ;
smp_send_nmi_allbutself ( ) ;
msecs = 1000 ; /* Wait at most a second for the other cpus to stop */
while ( ( atomic_read ( & waiting_for_crash_ipi ) > 0 ) & & msecs ) {
mdelay ( 1 ) ;
msecs - - ;
}
/* Leave the nmi callback set */
}
2008-11-12 16:34:43 +03:00
# else /* !CONFIG_SMP */
void nmi_shootdown_cpus ( nmi_shootdown_cb callback )
{
/* No other CPUs to shoot down */
}
2008-11-12 16:34:42 +03:00
# endif