2007-07-18 05:37:07 +04:00
/*
* Handle extern requests for shutdown , reboot and sysrq
*/
# include <linux/kernel.h>
# include <linux/err.h>
# include <linux/reboot.h>
# include <linux/sysrq.h>
2008-05-27 02:31:27 +04:00
# include <linux/stop_machine.h>
# include <linux/freezer.h>
2007-07-18 05:37:07 +04:00
# include <xen/xenbus.h>
2008-05-27 02:31:27 +04:00
# include <xen/grant_table.h>
# include <xen/events.h>
# include <xen/hvc-console.h>
# include <xen/xen-ops.h>
2007-07-18 05:37:07 +04:00
2008-05-27 02:31:27 +04:00
# include <asm/xen/hypercall.h>
# include <asm/xen/page.h>
enum shutdown_state {
SHUTDOWN_INVALID = - 1 ,
SHUTDOWN_POWEROFF = 0 ,
SHUTDOWN_SUSPEND = 2 ,
/* Code 3 is SHUTDOWN_CRASH, which we don't use because the domain can only
report a crash , not be instructed to crash !
HALT is the same as POWEROFF , as far as we ' re concerned . The tools use
the distinction when we return the reason code to them . */
SHUTDOWN_HALT = 4 ,
} ;
2007-07-18 05:37:07 +04:00
/* Ignore multiple shutdown requests. */
2008-05-27 02:31:27 +04:00
static enum shutdown_state shutting_down = SHUTDOWN_INVALID ;
2008-05-29 12:02:19 +04:00
# ifdef CONFIG_PM_SLEEP
2008-05-27 02:31:27 +04:00
static int xen_suspend ( void * data )
{
int * cancelled = data ;
2008-05-27 02:31:28 +04:00
int err ;
2008-05-27 02:31:27 +04:00
BUG_ON ( ! irqs_disabled ( ) ) ;
2009-02-22 20:38:50 +03:00
err = sysdev_suspend ( PMSG_SUSPEND ) ;
if ( err ) {
printk ( KERN_ERR " xen_suspend: sysdev_suspend failed: %d \n " ,
err ) ;
2009-05-25 00:05:42 +04:00
dpm_resume_noirq ( PMSG_RESUME ) ;
2009-02-22 20:38:50 +03:00
return err ;
}
2008-05-27 02:31:28 +04:00
2008-05-27 02:31:27 +04:00
xen_mm_pin_all ( ) ;
gnttab_suspend ( ) ;
xen_pre_suspend ( ) ;
/*
* This hypercall returns 1 if suspend was cancelled
* or the domain was merely checkpointed , and 0 if it
* is resuming in a new domain .
*/
* cancelled = HYPERVISOR_suspend ( virt_to_mfn ( xen_start_info ) ) ;
xen_post_suspend ( * cancelled ) ;
gnttab_resume ( ) ;
xen_mm_unpin_all ( ) ;
if ( ! * cancelled ) {
xen_irq_resume ( ) ;
xen_console_resume ( ) ;
2008-07-09 02:06:32 +04:00
xen_timer_resume ( ) ;
2008-05-27 02:31:27 +04:00
}
2009-03-25 20:46:42 +03:00
sysdev_resume ( ) ;
2009-05-25 00:05:42 +04:00
dpm_resume_noirq ( PMSG_RESUME ) ;
2009-03-25 20:46:42 +03:00
2008-05-27 02:31:27 +04:00
return 0 ;
}
static void do_suspend ( void )
{
int err ;
int cancelled = 1 ;
shutting_down = SHUTDOWN_SUSPEND ;
# ifdef CONFIG_PREEMPT
/* If the kernel is preemptible, we need to freeze all the processes
to prevent them from being in the middle of a pagetable update
during suspend . */
err = freeze_processes ( ) ;
if ( err ) {
printk ( KERN_ERR " xen suspend: freeze failed %d \n " , err ) ;
return ;
}
# endif
2009-05-25 00:05:42 +04:00
err = dpm_suspend_start ( PMSG_SUSPEND ) ;
2008-05-27 02:31:27 +04:00
if ( err ) {
2009-05-25 00:05:42 +04:00
printk ( KERN_ERR " xen suspend: dpm_suspend_start %d \n " , err ) ;
2008-05-27 02:31:27 +04:00
goto out ;
}
2009-02-09 23:05:50 +03:00
printk ( KERN_DEBUG " suspending xenstore... \n " ) ;
xs_suspend ( ) ;
2008-05-27 02:31:27 +04:00
2009-05-25 00:05:42 +04:00
err = dpm_suspend_noirq ( PMSG_SUSPEND ) ;
2009-03-17 00:34:06 +03:00
if ( err ) {
2009-05-25 00:05:42 +04:00
printk ( KERN_ERR " dpm_suspend_noirq failed: %d \n " , err ) ;
2009-03-17 00:34:06 +03:00
goto resume_devices ;
}
2009-01-11 08:58:09 +03:00
err = stop_machine ( xen_suspend , & cancelled , cpumask_of ( 0 ) ) ;
2008-05-27 02:31:27 +04:00
if ( err ) {
printk ( KERN_ERR " failed to start xen_suspend: %d \n " , err ) ;
goto out ;
}
2008-07-09 02:06:32 +04:00
if ( ! cancelled ) {
xen_arch_resume ( ) ;
2009-02-09 23:05:50 +03:00
xs_resume ( ) ;
2008-07-09 02:06:32 +04:00
} else
2009-02-09 23:05:50 +03:00
xs_suspend_cancel ( ) ;
2008-05-27 02:31:27 +04:00
2009-05-25 00:05:42 +04:00
dpm_resume_noirq ( PMSG_RESUME ) ;
2009-03-17 00:34:06 +03:00
resume_devices :
2009-05-25 00:05:42 +04:00
dpm_resume_end ( PMSG_RESUME ) ;
2008-05-27 02:31:27 +04:00
2008-05-27 02:31:28 +04:00
/* Make sure timer events get retriggered on all CPUs */
clock_was_set ( ) ;
2008-05-27 02:31:27 +04:00
out :
# ifdef CONFIG_PREEMPT
thaw_processes ( ) ;
# endif
shutting_down = SHUTDOWN_INVALID ;
}
2008-05-29 12:02:19 +04:00
# endif /* CONFIG_PM_SLEEP */
2007-07-18 05:37:07 +04:00
static void shutdown_handler ( struct xenbus_watch * watch ,
const char * * vec , unsigned int len )
{
char * str ;
struct xenbus_transaction xbt ;
int err ;
if ( shutting_down ! = SHUTDOWN_INVALID )
return ;
again :
err = xenbus_transaction_start ( & xbt ) ;
if ( err )
return ;
str = ( char * ) xenbus_read ( xbt , " control " , " shutdown " , NULL ) ;
/* Ignore read errors and empty reads. */
if ( XENBUS_IS_ERR_READ ( str ) ) {
xenbus_transaction_end ( xbt , 1 ) ;
return ;
}
xenbus_write ( xbt , " control " , " shutdown " , " " ) ;
err = xenbus_transaction_end ( xbt , 0 ) ;
if ( err = = - EAGAIN ) {
kfree ( str ) ;
goto again ;
}
if ( strcmp ( str , " poweroff " ) = = 0 | |
2008-05-27 02:31:27 +04:00
strcmp ( str , " halt " ) = = 0 ) {
shutting_down = SHUTDOWN_POWEROFF ;
2007-07-18 05:37:07 +04:00
orderly_poweroff ( false ) ;
2008-05-27 02:31:27 +04:00
} else if ( strcmp ( str , " reboot " ) = = 0 ) {
shutting_down = SHUTDOWN_POWEROFF ; /* ? */
2007-07-18 05:37:07 +04:00
ctrl_alt_del ( ) ;
2008-05-27 02:31:27 +04:00
# ifdef CONFIG_PM_SLEEP
} else if ( strcmp ( str , " suspend " ) = = 0 ) {
do_suspend ( ) ;
# endif
} else {
2007-07-18 05:37:07 +04:00
printk ( KERN_INFO " Ignoring shutdown request: %s \n " , str ) ;
shutting_down = SHUTDOWN_INVALID ;
}
kfree ( str ) ;
}
static void sysrq_handler ( struct xenbus_watch * watch , const char * * vec ,
unsigned int len )
{
char sysrq_key = ' \0 ' ;
struct xenbus_transaction xbt ;
int err ;
again :
err = xenbus_transaction_start ( & xbt ) ;
if ( err )
return ;
if ( ! xenbus_scanf ( xbt , " control " , " sysrq " , " %c " , & sysrq_key ) ) {
printk ( KERN_ERR " Unable to read sysrq code in "
" control/sysrq \n " ) ;
xenbus_transaction_end ( xbt , 1 ) ;
return ;
}
if ( sysrq_key ! = ' \0 ' )
xenbus_printf ( xbt , " control " , " sysrq " , " %c " , ' \0 ' ) ;
err = xenbus_transaction_end ( xbt , 0 ) ;
if ( err = = - EAGAIN )
goto again ;
if ( sysrq_key ! = ' \0 ' )
handle_sysrq ( sysrq_key , NULL ) ;
}
static struct xenbus_watch shutdown_watch = {
. node = " control/shutdown " ,
. callback = shutdown_handler
} ;
static struct xenbus_watch sysrq_watch = {
. node = " control/sysrq " ,
. callback = sysrq_handler
} ;
static int setup_shutdown_watcher ( void )
{
int err ;
err = register_xenbus_watch ( & shutdown_watch ) ;
if ( err ) {
printk ( KERN_ERR " Failed to set shutdown watcher \n " ) ;
return err ;
}
err = register_xenbus_watch ( & sysrq_watch ) ;
if ( err ) {
printk ( KERN_ERR " Failed to set sysrq watcher \n " ) ;
return err ;
}
return 0 ;
}
static int shutdown_event ( struct notifier_block * notifier ,
unsigned long event ,
void * data )
{
setup_shutdown_watcher ( ) ;
return NOTIFY_DONE ;
}
static int __init setup_shutdown_event ( void )
{
static struct notifier_block xenstore_notifier = {
. notifier_call = shutdown_event
} ;
register_xenstore_notifier ( & xenstore_notifier ) ;
return 0 ;
}
subsys_initcall ( setup_shutdown_event ) ;