aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/acpi
diff options
context:
space:
mode:
authorRafael J. Wysocki <rafael.j.wysocki@intel.com>2013-11-06 19:45:40 -0500
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>2013-11-07 13:28:48 -0500
commit7b98118aaa5d75644c48f41fc5d0cc181e478383 (patch)
tree9ab1e04ce0ee5ce816d744e50db1a3e2ba607575 /drivers/acpi
parent176a88d79d6b5aebabaff16734e8b3107efcaaad (diff)
ACPI / hotplug: Consolidate deferred execution of ACPI hotplug routines
There are two different interfaces for queuing up work items on the ACPI hotplug workqueue, alloc_acpi_hp_work() used by PCI and PCI host bridge hotplug code and acpi_os_hotplug_execute() used by the common ACPI hotplug code and docking stations. They both are somewhat cumbersome to use and work slightly differently. The users of alloc_acpi_hp_work() have to submit a work function that will extract the necessary data items from a struct acpi_hp_work object allocated by alloc_acpi_hp_work() and then will free that object, while it would be more straightforward to simply use a work function with one more argument and let the interface take care of the execution details. The users of acpi_os_hotplug_execute() also have to deal with the fact that it takes only one argument in addition to the work function pointer, although acpi_os_execute_deferred() actually takes care of the allocation and freeing of memory, so it would have been able to pass more arguments to the work function if it hadn't been constrained by the connection with acpi_os_execute(). Moreover, while alloc_acpi_hp_work() makes GFP_KERNEL memory allocations, which is correct, because hotplug work items are always queued up from process context, acpi_os_hotplug_execute() uses GFP_ATOMIC, as that is needed by acpi_os_execute(). Also, acpi_os_execute_deferred() queued up by it waits for the ACPI event workqueues to flush before executing the work function, whereas alloc_acpi_hp_work() can't do anything similar. That leads to somewhat arbitrary differences in behavior between various ACPI hotplug code paths and has to be straightened up. For this reason, replace both alloc_acpi_hp_work() and acpi_os_hotplug_execute() with a single interface, acpi_hotplug_execute(), combining their behavior and being more friendly to its users than any of the two. Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com> Tested-by: Mika Westerberg <mika.westerberg@linux.intel.com>
Diffstat (limited to 'drivers/acpi')
-rw-r--r--drivers/acpi/dock.c25
-rw-r--r--drivers/acpi/internal.h2
-rw-r--r--drivers/acpi/osl.c96
-rw-r--r--drivers/acpi/pci_root.c14
-rw-r--r--drivers/acpi/scan.c43
5 files changed, 61 insertions, 119 deletions
diff --git a/drivers/acpi/dock.c b/drivers/acpi/dock.c
index 05ea4be01a83..eab7d1145bfa 100644
--- a/drivers/acpi/dock.c
+++ b/drivers/acpi/dock.c
@@ -671,39 +671,20 @@ static void dock_notify(struct dock_station *ds, u32 event)
671 } 671 }
672} 672}
673 673
674struct dock_data { 674static void acpi_dock_deferred_cb(void *data, u32 event)
675 struct dock_station *ds;
676 u32 event;
677};
678
679static void acpi_dock_deferred_cb(void *context)
680{ 675{
681 struct dock_data *data = context;
682
683 acpi_scan_lock_acquire(); 676 acpi_scan_lock_acquire();
684 dock_notify(data->ds, data->event); 677 dock_notify(data, event);
685 acpi_scan_lock_release(); 678 acpi_scan_lock_release();
686 kfree(data);
687} 679}
688 680
689static void dock_notify_handler(acpi_handle handle, u32 event, void *data) 681static void dock_notify_handler(acpi_handle handle, u32 event, void *data)
690{ 682{
691 struct dock_data *dd;
692
693 if (event != ACPI_NOTIFY_BUS_CHECK && event != ACPI_NOTIFY_DEVICE_CHECK 683 if (event != ACPI_NOTIFY_BUS_CHECK && event != ACPI_NOTIFY_DEVICE_CHECK
694 && event != ACPI_NOTIFY_EJECT_REQUEST) 684 && event != ACPI_NOTIFY_EJECT_REQUEST)
695 return; 685 return;
696 686
697 dd = kmalloc(sizeof(*dd), GFP_KERNEL); 687 acpi_hotplug_execute(acpi_dock_deferred_cb, data, event);
698 if (dd) {
699 acpi_status status;
700
701 dd->ds = data;
702 dd->event = event;
703 status = acpi_os_hotplug_execute(acpi_dock_deferred_cb, dd);
704 if (ACPI_FAILURE(status))
705 kfree(dd);
706 }
707} 688}
708 689
709/** 690/**
diff --git a/drivers/acpi/internal.h b/drivers/acpi/internal.h
index 80cd1a10c4c3..c6db680c826c 100644
--- a/drivers/acpi/internal.h
+++ b/drivers/acpi/internal.h
@@ -87,7 +87,7 @@ void acpi_device_add_finalize(struct acpi_device *device);
87void acpi_free_pnp_ids(struct acpi_device_pnp *pnp); 87void acpi_free_pnp_ids(struct acpi_device_pnp *pnp);
88int acpi_bind_one(struct device *dev, acpi_handle handle); 88int acpi_bind_one(struct device *dev, acpi_handle handle);
89int acpi_unbind_one(struct device *dev); 89int acpi_unbind_one(struct device *dev);
90void acpi_bus_device_eject(struct acpi_device *device, u32 ost_src); 90void acpi_bus_device_eject(void *data, u32 ost_src);
91 91
92/* -------------------------------------------------------------------------- 92/* --------------------------------------------------------------------------
93 Power Resource 93 Power Resource
diff --git a/drivers/acpi/osl.c b/drivers/acpi/osl.c
index e5f416c7f66e..cfc3e260a688 100644
--- a/drivers/acpi/osl.c
+++ b/drivers/acpi/osl.c
@@ -61,7 +61,6 @@ struct acpi_os_dpc {
61 acpi_osd_exec_callback function; 61 acpi_osd_exec_callback function;
62 void *context; 62 void *context;
63 struct work_struct work; 63 struct work_struct work;
64 int wait;
65}; 64};
66 65
67#ifdef CONFIG_ACPI_CUSTOM_DSDT 66#ifdef CONFIG_ACPI_CUSTOM_DSDT
@@ -1067,9 +1066,6 @@ static void acpi_os_execute_deferred(struct work_struct *work)
1067{ 1066{
1068 struct acpi_os_dpc *dpc = container_of(work, struct acpi_os_dpc, work); 1067 struct acpi_os_dpc *dpc = container_of(work, struct acpi_os_dpc, work);
1069 1068
1070 if (dpc->wait)
1071 acpi_os_wait_events_complete();
1072
1073 dpc->function(dpc->context); 1069 dpc->function(dpc->context);
1074 kfree(dpc); 1070 kfree(dpc);
1075} 1071}
@@ -1089,8 +1085,8 @@ static void acpi_os_execute_deferred(struct work_struct *work)
1089 * 1085 *
1090 ******************************************************************************/ 1086 ******************************************************************************/
1091 1087
1092static acpi_status __acpi_os_execute(acpi_execute_type type, 1088acpi_status acpi_os_execute(acpi_execute_type type,
1093 acpi_osd_exec_callback function, void *context, int hp) 1089 acpi_osd_exec_callback function, void *context)
1094{ 1090{
1095 acpi_status status = AE_OK; 1091 acpi_status status = AE_OK;
1096 struct acpi_os_dpc *dpc; 1092 struct acpi_os_dpc *dpc;
@@ -1117,20 +1113,11 @@ static acpi_status __acpi_os_execute(acpi_execute_type type,
1117 dpc->context = context; 1113 dpc->context = context;
1118 1114
1119 /* 1115 /*
1120 * We can't run hotplug code in keventd_wq/kacpid_wq/kacpid_notify_wq
1121 * because the hotplug code may call driver .remove() functions,
1122 * which invoke flush_scheduled_work/acpi_os_wait_events_complete
1123 * to flush these workqueues.
1124 *
1125 * To prevent lockdep from complaining unnecessarily, make sure that 1116 * To prevent lockdep from complaining unnecessarily, make sure that
1126 * there is a different static lockdep key for each workqueue by using 1117 * there is a different static lockdep key for each workqueue by using
1127 * INIT_WORK() for each of them separately. 1118 * INIT_WORK() for each of them separately.
1128 */ 1119 */
1129 if (hp) { 1120 if (type == OSL_NOTIFY_HANDLER) {
1130 queue = kacpi_hotplug_wq;
1131 dpc->wait = 1;
1132 INIT_WORK(&dpc->work, acpi_os_execute_deferred);
1133 } else if (type == OSL_NOTIFY_HANDLER) {
1134 queue = kacpi_notify_wq; 1121 queue = kacpi_notify_wq;
1135 INIT_WORK(&dpc->work, acpi_os_execute_deferred); 1122 INIT_WORK(&dpc->work, acpi_os_execute_deferred);
1136 } else { 1123 } else {
@@ -1155,28 +1142,59 @@ static acpi_status __acpi_os_execute(acpi_execute_type type,
1155 } 1142 }
1156 return status; 1143 return status;
1157} 1144}
1145EXPORT_SYMBOL(acpi_os_execute);
1158 1146
1159acpi_status acpi_os_execute(acpi_execute_type type, 1147void acpi_os_wait_events_complete(void)
1160 acpi_osd_exec_callback function, void *context)
1161{ 1148{
1162 return __acpi_os_execute(type, function, context, 0); 1149 flush_workqueue(kacpid_wq);
1150 flush_workqueue(kacpi_notify_wq);
1163} 1151}
1164EXPORT_SYMBOL(acpi_os_execute);
1165 1152
1166acpi_status acpi_os_hotplug_execute(acpi_osd_exec_callback function, 1153struct acpi_hp_work {
1167 void *context) 1154 struct work_struct work;
1155 acpi_hp_callback func;
1156 void *data;
1157 u32 src;
1158};
1159
1160static void acpi_hotplug_work_fn(struct work_struct *work)
1168{ 1161{
1169 return __acpi_os_execute(0, function, context, 1); 1162 struct acpi_hp_work *hpw = container_of(work, struct acpi_hp_work, work);
1163
1164 acpi_os_wait_events_complete();
1165 hpw->func(hpw->data, hpw->src);
1166 kfree(hpw);
1170} 1167}
1171EXPORT_SYMBOL(acpi_os_hotplug_execute);
1172 1168
1173void acpi_os_wait_events_complete(void) 1169acpi_status acpi_hotplug_execute(acpi_hp_callback func, void *data, u32 src)
1174{ 1170{
1175 flush_workqueue(kacpid_wq); 1171 struct acpi_hp_work *hpw;
1176 flush_workqueue(kacpi_notify_wq); 1172
1173 ACPI_DEBUG_PRINT((ACPI_DB_EXEC,
1174 "Scheduling function [%p(%p, %u)] for deferred execution.\n",
1175 func, data, src));
1176
1177 hpw = kmalloc(sizeof(*hpw), GFP_KERNEL);
1178 if (!hpw)
1179 return AE_NO_MEMORY;
1180
1181 INIT_WORK(&hpw->work, acpi_hotplug_work_fn);
1182 hpw->func = func;
1183 hpw->data = data;
1184 hpw->src = src;
1185 /*
1186 * We can't run hotplug code in kacpid_wq/kacpid_notify_wq etc., because
1187 * the hotplug code may call driver .remove() functions, which may
1188 * invoke flush_scheduled_work()/acpi_os_wait_events_complete() to flush
1189 * these workqueues.
1190 */
1191 if (!queue_work(kacpi_hotplug_wq, &hpw->work)) {
1192 kfree(hpw);
1193 return AE_ERROR;
1194 }
1195 return AE_OK;
1177} 1196}
1178 1197
1179EXPORT_SYMBOL(acpi_os_wait_events_complete);
1180 1198
1181acpi_status 1199acpi_status
1182acpi_os_create_semaphore(u32 max_units, u32 initial_units, acpi_handle * handle) 1200acpi_os_create_semaphore(u32 max_units, u32 initial_units, acpi_handle * handle)
@@ -1825,25 +1843,3 @@ void acpi_os_set_prepare_extended_sleep(int (*func)(u8 sleep_state,
1825{ 1843{
1826 __acpi_os_prepare_extended_sleep = func; 1844 __acpi_os_prepare_extended_sleep = func;
1827} 1845}
1828
1829
1830void alloc_acpi_hp_work(acpi_handle handle, u32 type, void *context,
1831 void (*func)(struct work_struct *work))
1832{
1833 struct acpi_hp_work *hp_work;
1834 int ret;
1835
1836 hp_work = kmalloc(sizeof(*hp_work), GFP_KERNEL);
1837 if (!hp_work)
1838 return;
1839
1840 hp_work->handle = handle;
1841 hp_work->type = type;
1842 hp_work->context = context;
1843
1844 INIT_WORK(&hp_work->work, func);
1845 ret = queue_work(kacpi_hotplug_wq, &hp_work->work);
1846 if (!ret)
1847 kfree(hp_work);
1848}
1849EXPORT_SYMBOL_GPL(alloc_acpi_hp_work);
diff --git a/drivers/acpi/pci_root.c b/drivers/acpi/pci_root.c
index d77f0bf7eda0..417876bce854 100644
--- a/drivers/acpi/pci_root.c
+++ b/drivers/acpi/pci_root.c
@@ -592,17 +592,10 @@ static void handle_root_bridge_insertion(acpi_handle handle)
592 acpi_handle_err(handle, "cannot add bridge to acpi list\n"); 592 acpi_handle_err(handle, "cannot add bridge to acpi list\n");
593} 593}
594 594
595static void _handle_hotplug_event_root(struct work_struct *work) 595static void hotplug_event_root(void *data, u32 type)
596{ 596{
597 acpi_handle handle = data;
597 struct acpi_pci_root *root; 598 struct acpi_pci_root *root;
598 struct acpi_hp_work *hp_work;
599 acpi_handle handle;
600 u32 type;
601
602 hp_work = container_of(work, struct acpi_hp_work, work);
603 handle = hp_work->handle;
604 type = hp_work->type;
605 kfree(hp_work); /* allocated in handle_hotplug_event_bridge */
606 599
607 acpi_scan_lock_acquire(); 600 acpi_scan_lock_acquire();
608 601
@@ -654,8 +647,7 @@ static void _handle_hotplug_event_root(struct work_struct *work)
654static void handle_hotplug_event_root(acpi_handle handle, u32 type, 647static void handle_hotplug_event_root(acpi_handle handle, u32 type,
655 void *context) 648 void *context)
656{ 649{
657 alloc_acpi_hp_work(handle, type, context, 650 acpi_hotplug_execute(hotplug_event_root, handle, type);
658 _handle_hotplug_event_root);
659} 651}
660 652
661static acpi_status __init 653static acpi_status __init
diff --git a/drivers/acpi/scan.c b/drivers/acpi/scan.c
index 5cd7f8c5666a..276cde70a514 100644
--- a/drivers/acpi/scan.c
+++ b/drivers/acpi/scan.c
@@ -285,8 +285,9 @@ static int acpi_scan_hot_remove(struct acpi_device *device)
285 return 0; 285 return 0;
286} 286}
287 287
288void acpi_bus_device_eject(struct acpi_device *device, u32 ost_src) 288void acpi_bus_device_eject(void *data, u32 ost_src)
289{ 289{
290 struct acpi_device *device = data;
290 acpi_handle handle = device->handle; 291 acpi_handle handle = device->handle;
291 struct acpi_scan_handler *handler; 292 struct acpi_scan_handler *handler;
292 u32 ost_code = ACPI_OST_SC_NON_SPECIFIC_FAILURE; 293 u32 ost_code = ACPI_OST_SC_NON_SPECIFIC_FAILURE;
@@ -327,8 +328,9 @@ void acpi_bus_device_eject(struct acpi_device *device, u32 ost_src)
327 goto out; 328 goto out;
328} 329}
329 330
330static void acpi_scan_bus_device_check(acpi_handle handle, u32 ost_source) 331static void acpi_scan_bus_device_check(void *data, u32 ost_source)
331{ 332{
333 acpi_handle handle = data;
332 struct acpi_device *device = NULL; 334 struct acpi_device *device = NULL;
333 u32 ost_code = ACPI_OST_SC_NON_SPECIFIC_FAILURE; 335 u32 ost_code = ACPI_OST_SC_NON_SPECIFIC_FAILURE;
334 int error; 336 int error;
@@ -363,18 +365,6 @@ static void acpi_scan_bus_device_check(acpi_handle handle, u32 ost_source)
363 unlock_device_hotplug(); 365 unlock_device_hotplug();
364} 366}
365 367
366static void acpi_scan_bus_check(void *context)
367{
368 acpi_scan_bus_device_check((acpi_handle)context,
369 ACPI_NOTIFY_BUS_CHECK);
370}
371
372static void acpi_scan_device_check(void *context)
373{
374 acpi_scan_bus_device_check((acpi_handle)context,
375 ACPI_NOTIFY_DEVICE_CHECK);
376}
377
378static void acpi_hotplug_unsupported(acpi_handle handle, u32 type) 368static void acpi_hotplug_unsupported(acpi_handle handle, u32 type)
379{ 369{
380 u32 ost_status; 370 u32 ost_status;
@@ -403,18 +393,8 @@ static void acpi_hotplug_unsupported(acpi_handle handle, u32 type)
403 acpi_evaluate_hotplug_ost(handle, type, ost_status, NULL); 393 acpi_evaluate_hotplug_ost(handle, type, ost_status, NULL);
404} 394}
405 395
406/**
407 * acpi_bus_hot_remove_device: Hot-remove a device and its children.
408 * @context: Address of the ACPI device object to hot-remove.
409 */
410static void acpi_bus_hot_remove_device(void *context)
411{
412 acpi_bus_device_eject(context, ACPI_NOTIFY_EJECT_REQUEST);
413}
414
415static void acpi_hotplug_notify_cb(acpi_handle handle, u32 type, void *data) 396static void acpi_hotplug_notify_cb(acpi_handle handle, u32 type, void *data)
416{ 397{
417 acpi_osd_exec_callback callback;
418 struct acpi_scan_handler *handler = data; 398 struct acpi_scan_handler *handler = data;
419 struct acpi_device *adev; 399 struct acpi_device *adev;
420 acpi_status status; 400 acpi_status status;
@@ -425,11 +405,9 @@ static void acpi_hotplug_notify_cb(acpi_handle handle, u32 type, void *data)
425 switch (type) { 405 switch (type) {
426 case ACPI_NOTIFY_BUS_CHECK: 406 case ACPI_NOTIFY_BUS_CHECK:
427 acpi_handle_debug(handle, "ACPI_NOTIFY_BUS_CHECK event\n"); 407 acpi_handle_debug(handle, "ACPI_NOTIFY_BUS_CHECK event\n");
428 callback = acpi_scan_bus_check;
429 break; 408 break;
430 case ACPI_NOTIFY_DEVICE_CHECK: 409 case ACPI_NOTIFY_DEVICE_CHECK:
431 acpi_handle_debug(handle, "ACPI_NOTIFY_DEVICE_CHECK event\n"); 410 acpi_handle_debug(handle, "ACPI_NOTIFY_DEVICE_CHECK event\n");
432 callback = acpi_scan_device_check;
433 break; 411 break;
434 case ACPI_NOTIFY_EJECT_REQUEST: 412 case ACPI_NOTIFY_EJECT_REQUEST:
435 acpi_handle_debug(handle, "ACPI_NOTIFY_EJECT_REQUEST event\n"); 413 acpi_handle_debug(handle, "ACPI_NOTIFY_EJECT_REQUEST event\n");
@@ -438,8 +416,7 @@ static void acpi_hotplug_notify_cb(acpi_handle handle, u32 type, void *data)
438 goto err_out; 416 goto err_out;
439 417
440 get_device(&adev->dev); 418 get_device(&adev->dev);
441 callback = acpi_bus_hot_remove_device; 419 status = acpi_hotplug_execute(acpi_bus_device_eject, adev, type);
442 status = acpi_os_hotplug_execute(callback, adev);
443 if (ACPI_SUCCESS(status)) 420 if (ACPI_SUCCESS(status))
444 return; 421 return;
445 422
@@ -449,7 +426,7 @@ static void acpi_hotplug_notify_cb(acpi_handle handle, u32 type, void *data)
449 /* non-hotplug event; possibly handled by other handler */ 426 /* non-hotplug event; possibly handled by other handler */
450 return; 427 return;
451 } 428 }
452 status = acpi_os_hotplug_execute(callback, handle); 429 status = acpi_hotplug_execute(acpi_scan_bus_device_check, handle, type);
453 if (ACPI_SUCCESS(status)) 430 if (ACPI_SUCCESS(status))
454 return; 431 return;
455 432
@@ -484,11 +461,6 @@ static ssize_t power_state_show(struct device *dev,
484 461
485static DEVICE_ATTR(power_state, 0444, power_state_show, NULL); 462static DEVICE_ATTR(power_state, 0444, power_state_show, NULL);
486 463
487static void acpi_eject_store_work(void *context)
488{
489 acpi_bus_device_eject(context, ACPI_OST_EC_OSPM_EJECT);
490}
491
492static ssize_t 464static ssize_t
493acpi_eject_store(struct device *d, struct device_attribute *attr, 465acpi_eject_store(struct device *d, struct device_attribute *attr,
494 const char *buf, size_t count) 466 const char *buf, size_t count)
@@ -511,7 +483,8 @@ acpi_eject_store(struct device *d, struct device_attribute *attr,
511 acpi_evaluate_hotplug_ost(acpi_device->handle, ACPI_OST_EC_OSPM_EJECT, 483 acpi_evaluate_hotplug_ost(acpi_device->handle, ACPI_OST_EC_OSPM_EJECT,
512 ACPI_OST_SC_EJECT_IN_PROGRESS, NULL); 484 ACPI_OST_SC_EJECT_IN_PROGRESS, NULL);
513 get_device(&acpi_device->dev); 485 get_device(&acpi_device->dev);
514 status = acpi_os_hotplug_execute(acpi_eject_store_work, acpi_device); 486 status = acpi_hotplug_execute(acpi_bus_device_eject, acpi_device,
487 ACPI_OST_EC_OSPM_EJECT);
515 if (ACPI_SUCCESS(status)) 488 if (ACPI_SUCCESS(status))
516 return count; 489 return count;
517 490