Commit 303bfdb1 authored by Rafael J. Wysocki's avatar Rafael J. Wysocki

ACPI / scan: Add second pass of companion offlining to hot-remove code

As indicated by comments in mm/memory_hotplug.c:remove_memory(),
if CONFIG_MEMCG is set, it may not be possible to offline all of the
memory blocks held by one module (FRU) in one pass (because one of
them may be used by the others to store page cgroup in that case
and that block has to be offlined before the other ones).

To handle that arguably corner case, add a second pass of companion
device offlining to acpi_scan_hot_remove() and make it ignore errors
returned in the first pass (and make it skip the second pass if the
first one is successful).
Signed-off-by: default avatarRafael J. Wysocki <rafael.j.wysocki@intel.com>
Acked-by: default avatarToshi Kani <toshi.kani@hp.com>
parent ea50be59
...@@ -131,6 +131,7 @@ static acpi_status acpi_bus_offline_companions(acpi_handle handle, u32 lvl, ...@@ -131,6 +131,7 @@ static acpi_status acpi_bus_offline_companions(acpi_handle handle, u32 lvl,
{ {
struct acpi_device *device = NULL; struct acpi_device *device = NULL;
struct acpi_device_physical_node *pn; struct acpi_device_physical_node *pn;
bool second_pass = (bool)data;
acpi_status status = AE_OK; acpi_status status = AE_OK;
if (acpi_bus_get_device(handle, &device)) if (acpi_bus_get_device(handle, &device))
...@@ -141,15 +142,26 @@ static acpi_status acpi_bus_offline_companions(acpi_handle handle, u32 lvl, ...@@ -141,15 +142,26 @@ static acpi_status acpi_bus_offline_companions(acpi_handle handle, u32 lvl,
list_for_each_entry(pn, &device->physical_node_list, node) { list_for_each_entry(pn, &device->physical_node_list, node) {
int ret; int ret;
if (second_pass) {
/* Skip devices offlined by the first pass. */
if (pn->put_online)
continue;
} else {
pn->put_online = false;
}
ret = device_offline(pn->dev); ret = device_offline(pn->dev);
if (acpi_force_hot_remove) if (acpi_force_hot_remove)
continue; continue;
if (ret < 0) { if (ret >= 0) {
status = AE_ERROR; pn->put_online = !ret;
break; } else {
*ret_p = pn->dev;
if (second_pass) {
status = AE_ERROR;
break;
}
} }
pn->put_online = !ret;
} }
mutex_unlock(&device->physical_node_lock); mutex_unlock(&device->physical_node_lock);
...@@ -185,6 +197,7 @@ static int acpi_scan_hot_remove(struct acpi_device *device) ...@@ -185,6 +197,7 @@ static int acpi_scan_hot_remove(struct acpi_device *device)
acpi_handle not_used; acpi_handle not_used;
struct acpi_object_list arg_list; struct acpi_object_list arg_list;
union acpi_object arg; union acpi_object arg;
struct device *errdev;
acpi_status status; acpi_status status;
unsigned long long sta; unsigned long long sta;
...@@ -197,22 +210,42 @@ static int acpi_scan_hot_remove(struct acpi_device *device) ...@@ -197,22 +210,42 @@ static int acpi_scan_hot_remove(struct acpi_device *device)
lock_device_hotplug(); lock_device_hotplug();
status = acpi_walk_namespace(ACPI_TYPE_ANY, handle, ACPI_UINT32_MAX, /*
NULL, acpi_bus_offline_companions, NULL, * Carry out two passes here and ignore errors in the first pass,
NULL); * because if the devices in question are memory blocks and
if (ACPI_SUCCESS(status) || acpi_force_hot_remove) * CONFIG_MEMCG is set, one of the blocks may hold data structures
status = acpi_bus_offline_companions(handle, 0, NULL, NULL); * that the other blocks depend on, but it is not known in advance which
* block holds them.
if (ACPI_FAILURE(status) && !acpi_force_hot_remove) { *
acpi_bus_online_companions(handle, 0, NULL, NULL); * If the first pass is successful, the second one isn't needed, though.
*/
errdev = NULL;
acpi_walk_namespace(ACPI_TYPE_ANY, handle, ACPI_UINT32_MAX,
NULL, acpi_bus_offline_companions,
(void *)false, (void **)&errdev);
acpi_bus_offline_companions(handle, 0, (void *)false, (void **)&errdev);
if (errdev) {
errdev = NULL;
acpi_walk_namespace(ACPI_TYPE_ANY, handle, ACPI_UINT32_MAX, acpi_walk_namespace(ACPI_TYPE_ANY, handle, ACPI_UINT32_MAX,
acpi_bus_online_companions, NULL, NULL, NULL, acpi_bus_offline_companions,
NULL); (void *)true , (void **)&errdev);
if (!errdev || acpi_force_hot_remove)
unlock_device_hotplug(); acpi_bus_offline_companions(handle, 0, (void *)true,
(void **)&errdev);
put_device(&device->dev);
return -EBUSY; if (errdev && !acpi_force_hot_remove) {
dev_warn(errdev, "Offline failed.\n");
acpi_bus_online_companions(handle, 0, NULL, NULL);
acpi_walk_namespace(ACPI_TYPE_ANY, handle,
ACPI_UINT32_MAX,
acpi_bus_online_companions, NULL,
NULL, NULL);
unlock_device_hotplug();
put_device(&device->dev);
return -EBUSY;
}
} }
ACPI_DEBUG_PRINT((ACPI_DB_INFO, ACPI_DEBUG_PRINT((ACPI_DB_INFO,
......
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