diff options
Diffstat (limited to 'drivers/xen/manage.c')
| -rw-r--r-- | drivers/xen/manage.c | 153 | 
1 files changed, 86 insertions, 67 deletions
| diff --git a/drivers/xen/manage.c b/drivers/xen/manage.c index 24177272bcb..ebb292859b5 100644 --- a/drivers/xen/manage.c +++ b/drivers/xen/manage.c | |||
| @@ -34,42 +34,38 @@ enum shutdown_state { | |||
| 34 | /* Ignore multiple shutdown requests. */ | 34 | /* Ignore multiple shutdown requests. */ | 
| 35 | static enum shutdown_state shutting_down = SHUTDOWN_INVALID; | 35 | static enum shutdown_state shutting_down = SHUTDOWN_INVALID; | 
| 36 | 36 | ||
| 37 | #ifdef CONFIG_PM_SLEEP | 37 | struct suspend_info { | 
| 38 | static int xen_hvm_suspend(void *data) | 38 | int cancelled; | 
| 39 | { | 39 | unsigned long arg; /* extra hypercall argument */ | 
| 40 | int err; | 40 | void (*pre)(void); | 
| 41 | struct sched_shutdown r = { .reason = SHUTDOWN_suspend }; | 41 | void (*post)(int cancelled); | 
| 42 | int *cancelled = data; | 42 | }; | 
| 43 | |||
| 44 | BUG_ON(!irqs_disabled()); | ||
| 45 | |||
| 46 | err = sysdev_suspend(PMSG_SUSPEND); | ||
| 47 | if (err) { | ||
| 48 | printk(KERN_ERR "xen_hvm_suspend: sysdev_suspend failed: %d\n", | ||
| 49 | err); | ||
| 50 | return err; | ||
| 51 | } | ||
| 52 | |||
| 53 | *cancelled = HYPERVISOR_sched_op(SCHEDOP_shutdown, &r); | ||
| 54 | 43 | ||
| 55 | xen_hvm_post_suspend(*cancelled); | 44 | static void xen_hvm_post_suspend(int cancelled) | 
| 45 | { | ||
| 46 | xen_arch_hvm_post_suspend(cancelled); | ||
| 56 | gnttab_resume(); | 47 | gnttab_resume(); | 
| 48 | } | ||
| 57 | 49 | ||
| 58 | if (!*cancelled) { | 50 | static void xen_pre_suspend(void) | 
| 59 | xen_irq_resume(); | 51 | { | 
| 60 | xen_console_resume(); | 52 | xen_mm_pin_all(); | 
| 61 | xen_timer_resume(); | 53 | gnttab_suspend(); | 
| 62 | } | 54 | xen_arch_pre_suspend(); | 
| 63 | 55 | } | |
| 64 | sysdev_resume(); | ||
| 65 | 56 | ||
| 66 | return 0; | 57 | static void xen_post_suspend(int cancelled) | 
| 58 | { | ||
| 59 | xen_arch_post_suspend(cancelled); | ||
| 60 | gnttab_resume(); | ||
| 61 | xen_mm_unpin_all(); | ||
| 67 | } | 62 | } | 
| 68 | 63 | ||
| 64 | #ifdef CONFIG_PM_SLEEP | ||
| 69 | static int xen_suspend(void *data) | 65 | static int xen_suspend(void *data) | 
| 70 | { | 66 | { | 
| 67 | struct suspend_info *si = data; | ||
| 71 | int err; | 68 | int err; | 
| 72 | int *cancelled = data; | ||
| 73 | 69 | ||
| 74 | BUG_ON(!irqs_disabled()); | 70 | BUG_ON(!irqs_disabled()); | 
| 75 | 71 | ||
| @@ -80,22 +76,20 @@ static int xen_suspend(void *data) | |||
| 80 | return err; | 76 | return err; | 
| 81 | } | 77 | } | 
| 82 | 78 | ||
| 83 | xen_mm_pin_all(); | 79 | if (si->pre) | 
| 84 | gnttab_suspend(); | 80 | si->pre(); | 
| 85 | xen_pre_suspend(); | ||
| 86 | 81 | ||
| 87 | /* | 82 | /* | 
| 88 | * This hypercall returns 1 if suspend was cancelled | 83 | * This hypercall returns 1 if suspend was cancelled | 
| 89 | * or the domain was merely checkpointed, and 0 if it | 84 | * or the domain was merely checkpointed, and 0 if it | 
| 90 | * is resuming in a new domain. | 85 | * is resuming in a new domain. | 
| 91 | */ | 86 | */ | 
| 92 | *cancelled = HYPERVISOR_suspend(virt_to_mfn(xen_start_info)); | 87 | si->cancelled = HYPERVISOR_suspend(si->arg); | 
| 93 | 88 | ||
| 94 | xen_post_suspend(*cancelled); | 89 | if (si->post) | 
| 95 | gnttab_resume(); | 90 | si->post(si->cancelled); | 
| 96 | xen_mm_unpin_all(); | ||
| 97 | 91 | ||
| 98 | if (!*cancelled) { | 92 | if (!si->cancelled) { | 
| 99 | xen_irq_resume(); | 93 | xen_irq_resume(); | 
| 100 | xen_console_resume(); | 94 | xen_console_resume(); | 
| 101 | xen_timer_resume(); | 95 | xen_timer_resume(); | 
| @@ -109,7 +103,7 @@ static int xen_suspend(void *data) | |||
| 109 | static void do_suspend(void) | 103 | static void do_suspend(void) | 
| 110 | { | 104 | { | 
| 111 | int err; | 105 | int err; | 
| 112 | int cancelled = 1; | 106 | struct suspend_info si; | 
| 113 | 107 | ||
| 114 | shutting_down = SHUTDOWN_SUSPEND; | 108 | shutting_down = SHUTDOWN_SUSPEND; | 
| 115 | 109 | ||
| @@ -139,20 +133,29 @@ static void do_suspend(void) | |||
| 139 | goto out_resume; | 133 | goto out_resume; | 
| 140 | } | 134 | } | 
| 141 | 135 | ||
| 142 | if (xen_hvm_domain()) | 136 | si.cancelled = 1; | 
| 143 | err = stop_machine(xen_hvm_suspend, &cancelled, cpumask_of(0)); | 137 | |
| 144 | else | 138 | if (xen_hvm_domain()) { | 
| 145 | err = stop_machine(xen_suspend, &cancelled, cpumask_of(0)); | 139 | si.arg = 0UL; | 
| 140 | si.pre = NULL; | ||
| 141 | si.post = &xen_hvm_post_suspend; | ||
| 142 | } else { | ||
| 143 | si.arg = virt_to_mfn(xen_start_info); | ||
| 144 | si.pre = &xen_pre_suspend; | ||
| 145 | si.post = &xen_post_suspend; | ||
| 146 | } | ||
| 147 | |||
| 148 | err = stop_machine(xen_suspend, &si, cpumask_of(0)); | ||
| 146 | 149 | ||
| 147 | dpm_resume_noirq(PMSG_RESUME); | 150 | dpm_resume_noirq(PMSG_RESUME); | 
| 148 | 151 | ||
| 149 | if (err) { | 152 | if (err) { | 
| 150 | printk(KERN_ERR "failed to start xen_suspend: %d\n", err); | 153 | printk(KERN_ERR "failed to start xen_suspend: %d\n", err); | 
| 151 | cancelled = 1; | 154 | si.cancelled = 1; | 
| 152 | } | 155 | } | 
| 153 | 156 | ||
| 154 | out_resume: | 157 | out_resume: | 
| 155 | if (!cancelled) { | 158 | if (!si.cancelled) { | 
| 156 | xen_arch_resume(); | 159 | xen_arch_resume(); | 
| 157 | xs_resume(); | 160 | xs_resume(); | 
| 158 | } else | 161 | } else | 
| @@ -172,12 +175,39 @@ out: | |||
| 172 | } | 175 | } | 
| 173 | #endif /* CONFIG_PM_SLEEP */ | 176 | #endif /* CONFIG_PM_SLEEP */ | 
| 174 | 177 | ||
| 178 | struct shutdown_handler { | ||
| 179 | const char *command; | ||
| 180 | void (*cb)(void); | ||
| 181 | }; | ||
| 182 | |||
| 183 | static void do_poweroff(void) | ||
| 184 | { | ||
| 185 | shutting_down = SHUTDOWN_POWEROFF; | ||
| 186 | orderly_poweroff(false); | ||
| 187 | } | ||
| 188 | |||
| 189 | static void do_reboot(void) | ||
| 190 | { | ||
| 191 | shutting_down = SHUTDOWN_POWEROFF; /* ? */ | ||
| 192 | ctrl_alt_del(); | ||
| 193 | } | ||
| 194 | |||
| 175 | static void shutdown_handler(struct xenbus_watch *watch, | 195 | static void shutdown_handler(struct xenbus_watch *watch, | 
| 176 | const char **vec, unsigned int len) | 196 | const char **vec, unsigned int len) | 
| 177 | { | 197 | { | 
| 178 | char *str; | 198 | char *str; | 
| 179 | struct xenbus_transaction xbt; | 199 | struct xenbus_transaction xbt; | 
| 180 | int err; | 200 | int err; | 
| 201 | static struct shutdown_handler handlers[] = { | ||
| 202 | { "poweroff", do_poweroff }, | ||
| 203 | { "halt", do_poweroff }, | ||
| 204 | { "reboot", do_reboot }, | ||
| 205 | #ifdef CONFIG_PM_SLEEP | ||
| 206 | { "suspend", do_suspend }, | ||
| 207 | #endif | ||
| 208 | {NULL, NULL}, | ||
| 209 | }; | ||
| 210 | static struct shutdown_handler *handler; | ||
| 181 | 211 | ||
| 182 | if (shutting_down != SHUTDOWN_INVALID) | 212 | if (shutting_down != SHUTDOWN_INVALID) | 
| 183 | return; | 213 | return; | 
| @@ -194,7 +224,14 @@ static void shutdown_handler(struct xenbus_watch *watch, | |||
| 194 | return; | 224 | return; | 
| 195 | } | 225 | } | 
| 196 | 226 | ||
| 197 | xenbus_write(xbt, "control", "shutdown", ""); | 227 | for (handler = &handlers[0]; handler->command; handler++) { | 
| 228 | if (strcmp(str, handler->command) == 0) | ||
| 229 | break; | ||
| 230 | } | ||
| 231 | |||
| 232 | /* Only acknowledge commands which we are prepared to handle. */ | ||
| 233 | if (handler->cb) | ||
| 234 | xenbus_write(xbt, "control", "shutdown", ""); | ||
| 198 | 235 | ||
| 199 | err = xenbus_transaction_end(xbt, 0); | 236 | err = xenbus_transaction_end(xbt, 0); | 
| 200 | if (err == -EAGAIN) { | 237 | if (err == -EAGAIN) { | 
| @@ -202,17 +239,8 @@ static void shutdown_handler(struct xenbus_watch *watch, | |||
| 202 | goto again; | 239 | goto again; | 
| 203 | } | 240 | } | 
| 204 | 241 | ||
| 205 | if (strcmp(str, "poweroff") == 0 || | 242 | if (handler->cb) { | 
| 206 | strcmp(str, "halt") == 0) { | 243 | handler->cb(); | 
| 207 | shutting_down = SHUTDOWN_POWEROFF; | ||
| 208 | orderly_poweroff(false); | ||
| 209 | } else if (strcmp(str, "reboot") == 0) { | ||
| 210 | shutting_down = SHUTDOWN_POWEROFF; /* ? */ | ||
| 211 | ctrl_alt_del(); | ||
| 212 | #ifdef CONFIG_PM_SLEEP | ||
| 213 | } else if (strcmp(str, "suspend") == 0) { | ||
| 214 | do_suspend(); | ||
| 215 | #endif | ||
| 216 | } else { | 244 | } else { | 
| 217 | printk(KERN_INFO "Ignoring shutdown request: %s\n", str); | 245 | printk(KERN_INFO "Ignoring shutdown request: %s\n", str); | 
| 218 | shutting_down = SHUTDOWN_INVALID; | 246 | shutting_down = SHUTDOWN_INVALID; | 
| @@ -291,27 +319,18 @@ static int shutdown_event(struct notifier_block *notifier, | |||
| 291 | return NOTIFY_DONE; | 319 | return NOTIFY_DONE; | 
| 292 | } | 320 | } | 
| 293 | 321 | ||
| 294 | static int __init __setup_shutdown_event(void) | ||
| 295 | { | ||
| 296 | /* Delay initialization in the PV on HVM case */ | ||
| 297 | if (xen_hvm_domain()) | ||
| 298 | return 0; | ||
| 299 | |||
| 300 | if (!xen_pv_domain()) | ||
| 301 | return -ENODEV; | ||
| 302 | |||
| 303 | return xen_setup_shutdown_event(); | ||
| 304 | } | ||
| 305 | |||
| 306 | int xen_setup_shutdown_event(void) | 322 | int xen_setup_shutdown_event(void) | 
| 307 | { | 323 | { | 
| 308 | static struct notifier_block xenstore_notifier = { | 324 | static struct notifier_block xenstore_notifier = { | 
| 309 | .notifier_call = shutdown_event | 325 | .notifier_call = shutdown_event | 
| 310 | }; | 326 | }; | 
| 327 | |||
| 328 | if (!xen_domain()) | ||
| 329 | return -ENODEV; | ||
| 311 | register_xenstore_notifier(&xenstore_notifier); | 330 | register_xenstore_notifier(&xenstore_notifier); | 
| 312 | 331 | ||
| 313 | return 0; | 332 | return 0; | 
| 314 | } | 333 | } | 
| 315 | EXPORT_SYMBOL_GPL(xen_setup_shutdown_event); | 334 | EXPORT_SYMBOL_GPL(xen_setup_shutdown_event); | 
| 316 | 335 | ||
| 317 | subsys_initcall(__setup_shutdown_event); | 336 | subsys_initcall(xen_setup_shutdown_event); | 
