Commit 37f90877 authored by Rafael J. Wysocki's avatar Rafael J. Wysocki

ACPI / dock: Walk list in reverse order during removal of devices

If there are indirect dependencies between devices in a dock
station's dependent devices list, they may be broken if the devices
are removed in the same order in which they have been added.

For this reason, make the code in handle_eject_request() walk the
list of dependent devices in reverse order.
Signed-off-by: default avatarRafael J. Wysocki <rafael.j.wysocki@intel.com>
Acked-by: default avatarYinghai Lu <yinghai@kernel.org>
parent 96c0a4d4
...@@ -337,9 +337,29 @@ static void dock_remove_acpi_device(acpi_handle handle) ...@@ -337,9 +337,29 @@ static void dock_remove_acpi_device(acpi_handle handle)
} }
/** /**
* hotplug_dock_devices - insert or remove devices on the dock station * hot_remove_dock_devices - Remove dock station devices.
* @ds: Dock station.
*/
static void hot_remove_dock_devices(struct dock_station *ds)
{
struct dock_dependent_device *dd;
/*
* Walk the list in reverse order so that devices that have been added
* last are removed first (in case there are some indirect dependencies
* between them).
*/
list_for_each_entry_reverse(dd, &ds->dependent_devices, list)
dock_hotplug_event(dd, ACPI_NOTIFY_EJECT_REQUEST, false);
list_for_each_entry_reverse(dd, &ds->dependent_devices, list)
dock_remove_acpi_device(dd->handle);
}
/**
* hotplug_dock_devices - Insert devices on a dock station.
* @ds: the dock station * @ds: the dock station
* @event: either bus check or eject request * @event: either bus check or device check request
* *
* Some devices on the dock station need to have drivers called * Some devices on the dock station need to have drivers called
* to perform hotplug operations after a dock event has occurred. * to perform hotplug operations after a dock event has occurred.
...@@ -350,24 +370,17 @@ static void hotplug_dock_devices(struct dock_station *ds, u32 event) ...@@ -350,24 +370,17 @@ static void hotplug_dock_devices(struct dock_station *ds, u32 event)
{ {
struct dock_dependent_device *dd; struct dock_dependent_device *dd;
/* /* Call driver specific hotplug functions. */
* First call driver specific hotplug functions
*/
list_for_each_entry(dd, &ds->dependent_devices, list) list_for_each_entry(dd, &ds->dependent_devices, list)
dock_hotplug_event(dd, event, false); dock_hotplug_event(dd, event, false);
/* /*
* Now make sure that an acpi_device is created for each * Now make sure that an acpi_device is created for each dependent
* dependent device, or removed if this is an eject request. * device. That will cause scan handlers to be attached to device
* This will cause acpi_drivers to be stopped/started if they * objects or acpi_drivers to be stopped/started if they are present.
* exist
*/ */
list_for_each_entry(dd, &ds->dependent_devices, list) { list_for_each_entry(dd, &ds->dependent_devices, list)
if (event == ACPI_NOTIFY_EJECT_REQUEST) dock_create_acpi_device(dd->handle);
dock_remove_acpi_device(dd->handle);
else
dock_create_acpi_device(dd->handle);
}
} }
static void dock_event(struct dock_station *ds, u32 event, int num) static void dock_event(struct dock_station *ds, u32 event, int num)
...@@ -588,7 +601,7 @@ static int handle_eject_request(struct dock_station *ds, u32 event) ...@@ -588,7 +601,7 @@ static int handle_eject_request(struct dock_station *ds, u32 event)
*/ */
dock_event(ds, event, UNDOCK_EVENT); dock_event(ds, event, UNDOCK_EVENT);
hotplug_dock_devices(ds, ACPI_NOTIFY_EJECT_REQUEST); hot_remove_dock_devices(ds);
undock(ds); undock(ds);
acpi_evaluate_lck(ds->handle, 0); acpi_evaluate_lck(ds->handle, 0);
acpi_evaluate_ej0(ds->handle); acpi_evaluate_ej0(ds->handle);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment