acpi sleep: Must acquire hypercall_deadlock_mutex when a domain
authorKeir Fraser <keir.fraser@citrix.com>
Thu, 15 Apr 2010 12:06:48 +0000 (13:06 +0100)
committerKeir Fraser <keir.fraser@citrix.com>
Thu, 15 Apr 2010 12:06:48 +0000 (13:06 +0100)
freezes its own vcpus.

Signed-off-by: Keir Fraser <keir.fraser@citrix.com>
xen/arch/x86/acpi/power.c

index 7978e01de2c52a9fb328da07110c1149b2021c74..4fbfd5da9acb432d809097319b98027aef04f51f 100644 (file)
@@ -74,11 +74,14 @@ static void device_power_up(void)
     console_resume();
 }
 
-static void freeze_domains(void)
+static int freeze_domains(void)
 {
     struct domain *d;
     struct vcpu *v;
 
+    if ( !spin_trylock(&current->domain->hypercall_deadlock_mutex) )
+        return -EBUSY;
+
     rcu_read_lock(&domlist_read_lock);
     for_each_domain ( d )
     {
@@ -91,6 +94,10 @@ static void freeze_domains(void)
         }
     }
     rcu_read_unlock(&domlist_read_lock);
+
+    spin_unlock(&current->domain->hypercall_deadlock_mutex);
+
+    return 0;
 }
 
 static void thaw_domains(void)
@@ -254,16 +261,22 @@ int acpi_enter_sleep(struct xenpf_enter_acpi_sleep *sleep)
 
     printk(XENLOG_INFO "Preparing system for ACPI S%d state.", state);
 
-    freeze_domains();
+    rc = freeze_domains();
+    if ( rc )
+        goto unlock_and_fail;
 
     rc = continue_hypercall_on_cpu(0, enter_state_helper, &acpi_sinfo);
     if ( rc )
     {
         /* Continuation will not execute: undo our own work so far. */
         thaw_domains();
-        spin_unlock(&pm_lock);
+        goto unlock_and_fail;
     }
 
+    return 0;
+
+ unlock_and_fail:
+    spin_unlock(&pm_lock);
     return rc;
 }