aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorRafael J. Wysocki <rafael.j.wysocki@intel.com>2014-02-03 16:30:06 -0500
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>2014-02-03 16:30:06 -0500
commit1b360f44d009059e446532f29c1a889951e72667 (patch)
treee30d762799671bda7d398fb8b6b0b0a31575d0cf
parentd42f5da2340083301dd2c48ff2d75f6ce4b30767 (diff)
ACPI / hotplug / PCI: Fix bridge removal race in handle_hotplug_event()
If a PCI bridge with an ACPIPHP context attached is removed via sysfs, the code path executed as a result is the following: pci_stop_and_remove_bus_device_locked pci_remove_bus pcibios_remove_bus acpi_pci_remove_bus acpiphp_remove_slots cleanup_bridge put_bridge free_bridge acpiphp_put_context (for each child, under context lock) kfree (child context) Now, if a hotplug notify is dispatched for one of the bridge's children and the timing is such that handle_hotplug_event() for that notify is executed while free_bridge() above is running, the get_bridge(context->func.parent) in handle_hotplug_event() will not really help, because it is too late to prevent the bridge from going away and the child's context may be freed before hotplug_event_work() scheduled from handle_hotplug_event() dereferences the pointer to it passed via the data argument. That will cause a kernel crash to happpen in hotplug_event_work(). To prevent that from happening, make handle_hotplug_event() check the is_going_away flag of the function's parent bridge (under acpiphp_context_lock) and bail out if it's set. Also, make cleanup_bridge() set the bridge's is_going_away flag under acpiphp_context_lock so that it cannot be changed between the check and the subsequent get_bridge(context->func.parent) in handle_hotplug_event(). Then, in the above scenario, handle_hotplug_event() will notice that context->func.parent->is_going_away is already set and it will exit immediately preventing the crash from happening. Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
-rw-r--r--drivers/pci/hotplug/acpiphp_glue.c18
1 files changed, 14 insertions, 4 deletions
diff --git a/drivers/pci/hotplug/acpiphp_glue.c b/drivers/pci/hotplug/acpiphp_glue.c
index 931d0b44eace..91eceaf3131b 100644
--- a/drivers/pci/hotplug/acpiphp_glue.c
+++ b/drivers/pci/hotplug/acpiphp_glue.c
@@ -441,7 +441,9 @@ static void cleanup_bridge(struct acpiphp_bridge *bridge)
441 list_del(&bridge->list); 441 list_del(&bridge->list);
442 mutex_unlock(&bridge_mutex); 442 mutex_unlock(&bridge_mutex);
443 443
444 mutex_lock(&acpiphp_context_lock);
444 bridge->is_going_away = true; 445 bridge->is_going_away = true;
446 mutex_unlock(&acpiphp_context_lock);
445} 447}
446 448
447/** 449/**
@@ -941,6 +943,7 @@ static void handle_hotplug_event(acpi_handle handle, u32 type, void *data)
941{ 943{
942 struct acpiphp_context *context; 944 struct acpiphp_context *context;
943 u32 ost_code = ACPI_OST_SC_SUCCESS; 945 u32 ost_code = ACPI_OST_SC_SUCCESS;
946 acpi_status status;
944 947
945 switch (type) { 948 switch (type) {
946 case ACPI_NOTIFY_BUS_CHECK: 949 case ACPI_NOTIFY_BUS_CHECK:
@@ -976,13 +979,20 @@ static void handle_hotplug_event(acpi_handle handle, u32 type, void *data)
976 979
977 mutex_lock(&acpiphp_context_lock); 980 mutex_lock(&acpiphp_context_lock);
978 context = acpiphp_get_context(handle); 981 context = acpiphp_get_context(handle);
979 if (context && !WARN_ON(context->handle != handle)) { 982 if (!context || WARN_ON(context->handle != handle)
980 get_bridge(context->func.parent); 983 || context->func.parent->is_going_away)
981 acpiphp_put_context(context); 984 goto err_out;
982 acpi_hotplug_execute(hotplug_event_work, context, type); 985
986 get_bridge(context->func.parent);
987 acpiphp_put_context(context);
988 status = acpi_hotplug_execute(hotplug_event_work, context, type);
989 if (ACPI_SUCCESS(status)) {
983 mutex_unlock(&acpiphp_context_lock); 990 mutex_unlock(&acpiphp_context_lock);
984 return; 991 return;
985 } 992 }
993 put_bridge(context->func.parent);
994
995 err_out:
986 mutex_unlock(&acpiphp_context_lock); 996 mutex_unlock(&acpiphp_context_lock);
987 ost_code = ACPI_OST_SC_NON_SPECIFIC_FAILURE; 997 ost_code = ACPI_OST_SC_NON_SPECIFIC_FAILURE;
988 998