Commit 59401ccc authored by Rafael J. Wysocki's avatar Rafael J. Wysocki

ACPI / dock: Rework the handling of notifications

The ACPI dock driver uses register_acpi_bus_notifier() which
installs a notifier triggered globally for all system notifications.
That first of all is inefficient, because the dock driver is only
interested in notifications associated with the devices it handles,
but it has to handle all system notifies for all devices.  Moreover,
it does that even if no docking stations are present in the system
(CONFIG_ACPI_DOCK set is sufficient for that to happen).  Besides,
that is inconvenient, because it requires the driver to do extra work
for each notification to find the target dock station object.

For these reasons, rework the dock driver to install a notify
handler individually for each dock station in the system using
acpi_install_notify_handler().  This allows the dock station
object to be passed directly to the notify handler and makes it
possible to simplify the dock driver quite a bit.  It also
reduces the overhead related to the handling of all system
notifies when CONFIG_ACPI_DOCK is set.
Signed-off-by: default avatarRafael J. Wysocki <rafael.j.wysocki@intel.com>
Acked-by: default avatarYinghai Lu <yinghai@kernel.org>
parent 4ec24065
...@@ -607,18 +607,17 @@ static int handle_eject_request(struct dock_station *ds, u32 event) ...@@ -607,18 +607,17 @@ static int handle_eject_request(struct dock_station *ds, u32 event)
/** /**
* dock_notify - act upon an acpi dock notification * dock_notify - act upon an acpi dock notification
* @handle: the dock station handle * @ds: dock station
* @event: the acpi event * @event: the acpi event
* @data: our driver data struct
* *
* If we are notified to dock, then check to see if the dock is * If we are notified to dock, then check to see if the dock is
* present and then dock. Notify all drivers of the dock event, * present and then dock. Notify all drivers of the dock event,
* and then hotplug and devices that may need hotplugging. * and then hotplug and devices that may need hotplugging.
*/ */
static void dock_notify(acpi_handle handle, u32 event, void *data) static void dock_notify(struct dock_station *ds, u32 event)
{ {
struct dock_station *ds = data; acpi_handle handle = ds->handle;
struct acpi_device *tmp; struct acpi_device *ad;
int surprise_removal = 0; int surprise_removal = 0;
/* /*
...@@ -641,8 +640,7 @@ static void dock_notify(acpi_handle handle, u32 event, void *data) ...@@ -641,8 +640,7 @@ static void dock_notify(acpi_handle handle, u32 event, void *data)
switch (event) { switch (event) {
case ACPI_NOTIFY_BUS_CHECK: case ACPI_NOTIFY_BUS_CHECK:
case ACPI_NOTIFY_DEVICE_CHECK: case ACPI_NOTIFY_DEVICE_CHECK:
if (!dock_in_progress(ds) && acpi_bus_get_device(ds->handle, if (!dock_in_progress(ds) && acpi_bus_get_device(handle, &ad)) {
&tmp)) {
begin_dock(ds); begin_dock(ds);
dock(ds); dock(ds);
if (!dock_present(ds)) { if (!dock_present(ds)) {
...@@ -679,9 +677,8 @@ static void dock_notify(acpi_handle handle, u32 event, void *data) ...@@ -679,9 +677,8 @@ static void dock_notify(acpi_handle handle, u32 event, void *data)
} }
struct dock_data { struct dock_data {
acpi_handle handle;
unsigned long event;
struct dock_station *ds; struct dock_station *ds;
u32 event;
}; };
static void acpi_dock_deferred_cb(void *context) static void acpi_dock_deferred_cb(void *context)
...@@ -689,52 +686,31 @@ static void acpi_dock_deferred_cb(void *context) ...@@ -689,52 +686,31 @@ static void acpi_dock_deferred_cb(void *context)
struct dock_data *data = context; struct dock_data *data = context;
acpi_scan_lock_acquire(); acpi_scan_lock_acquire();
dock_notify(data->handle, data->event, data->ds); dock_notify(data->ds, data->event);
acpi_scan_lock_release(); acpi_scan_lock_release();
kfree(data); kfree(data);
} }
static int acpi_dock_notifier_call(struct notifier_block *this, static void dock_notify_handler(acpi_handle handle, u32 event, void *data)
unsigned long event, void *data)
{ {
struct dock_station *dock_station; struct dock_data *dd;
acpi_handle handle = data;
if (event != ACPI_NOTIFY_BUS_CHECK && event != ACPI_NOTIFY_DEVICE_CHECK if (event != ACPI_NOTIFY_BUS_CHECK && event != ACPI_NOTIFY_DEVICE_CHECK
&& event != ACPI_NOTIFY_EJECT_REQUEST) && event != ACPI_NOTIFY_EJECT_REQUEST)
return 0; return;
acpi_scan_lock_acquire();
list_for_each_entry(dock_station, &dock_stations, sibling) {
if (dock_station->handle == handle) {
struct dock_data *dd;
acpi_status status;
dd = kmalloc(sizeof(*dd), GFP_KERNEL); dd = kmalloc(sizeof(*dd), GFP_KERNEL);
if (!dd) if (dd) {
break; acpi_status status;
dd->handle = handle; dd->ds = data;
dd->event = event; dd->event = event;
dd->ds = dock_station; status = acpi_os_hotplug_execute(acpi_dock_deferred_cb, dd);
status = acpi_os_hotplug_execute(acpi_dock_deferred_cb,
dd);
if (ACPI_FAILURE(status)) if (ACPI_FAILURE(status))
kfree(dd); kfree(dd);
break;
} }
}
acpi_scan_lock_release();
return 0;
} }
static struct notifier_block dock_acpi_notifier = {
.notifier_call = acpi_dock_notifier_call,
};
/** /**
* find_dock_devices - find devices on the dock station * find_dock_devices - find devices on the dock station
* @handle: the handle of the device we are examining * @handle: the handle of the device we are examining
...@@ -868,6 +844,7 @@ static int __init dock_add(acpi_handle handle) ...@@ -868,6 +844,7 @@ static int __init dock_add(acpi_handle handle)
int ret, id; int ret, id;
struct dock_station ds, *dock_station; struct dock_station ds, *dock_station;
struct platform_device *dd; struct platform_device *dd;
acpi_status status;
id = dock_station_count; id = dock_station_count;
memset(&ds, 0, sizeof(ds)); memset(&ds, 0, sizeof(ds));
...@@ -908,6 +885,11 @@ static int __init dock_add(acpi_handle handle) ...@@ -908,6 +885,11 @@ static int __init dock_add(acpi_handle handle)
if (ret) if (ret)
goto err_rmgroup; goto err_rmgroup;
status = acpi_install_notify_handler(handle, ACPI_SYSTEM_NOTIFY,
dock_notify_handler, dock_station);
if (ACPI_FAILURE(status))
goto err_rmgroup;
dock_station_count++; dock_station_count++;
list_add(&dock_station->sibling, &dock_stations); list_add(&dock_station->sibling, &dock_stations);
return 0; return 0;
...@@ -953,7 +935,6 @@ void __init acpi_dock_init(void) ...@@ -953,7 +935,6 @@ void __init acpi_dock_init(void)
} }
ATOMIC_INIT_NOTIFIER_HEAD(&dock_notifier_list); ATOMIC_INIT_NOTIFIER_HEAD(&dock_notifier_list);
register_acpi_bus_notifier(&dock_acpi_notifier);
pr_info(PREFIX "%s: %d docks/bays found\n", pr_info(PREFIX "%s: %d docks/bays found\n",
ACPI_DOCK_DRIVER_DESCRIPTION, dock_station_count); ACPI_DOCK_DRIVER_DESCRIPTION, dock_station_count);
} }
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