2008-04-19 03:33:50 +07:00
|
|
|
/*
|
|
|
|
* pseries Memory Hotplug infrastructure.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2008 Badari Pulavarty, IBM Corporation
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version
|
|
|
|
* 2 of the License, or (at your option) any later version.
|
|
|
|
*/
|
|
|
|
|
powerpc/pseries: Create new device hotplug entry point
The current hotplug (or dlpar) of devices (the process is generally the
same for memory, cpu, and pci) on PowerVM systems is initiated
from the HMC, which communicates the request to the partitions through
the RSCT framework. The RSCT framework then invokes the drmgr command.
The drmgr command performs the hotplug operation by doing some pieces,
such as most of the rtas calls and device tree parsing, in userspace
and make requests to the kernel to online/offline the device, update the
device tree and add/remove the device.
For PowerKVM the approach for device hotplug is to follow what is currently
being done for pci hotplug. A hotplug request is initiated from the host.
QEMU then generates an EPOW interrupt to the guest which causes the guest
to make the rtas,check-exception call. In QEMU, the rtas,check-exception call
returns a rtas hotplug event to the guest.
Please note that the current pci hotplug path for PowerKVM involves the
kernel receiving the rtas hotplug event, passing it to rtas_errd in
userspace, and having rtas_errd invoke drmgr. The drmgr command then
handles the request as described above for PowerVM systems.
There is no need for this circuitous route, we should just handle the entire
hotplug of devices in the kernel. What I am planning is to enable this
by moving the code to handle hotplug from drmgr into the kernel to
provide a single path for handling device hotplug for both PowerVM and
PowerKVM systems. This patch provides the common iframework and entry point.
For PowerKVM a future update to the kernel rtas code will recognize rtas
hotplug events returned from rtas,check-exception calls and use the common
entry point to handle hotplug of the device.
For PowerVM systems, This patch creates /sys/kernel/dlpar that can be
used by the drmgr command to initiate hotplug requests. In order to do
this a string of the format "<resource> <action> <id_type> <id>" is
written to this file. The string consists of a resource (cpu, memory, pci,
phb), an action (add or remove), an id_type (count, drc index, drc name),
and the corresponding id. The kernel will parse the string and create a
rtas hotplug section that can be passed to the common entry point for
handling hotplug requests.
It should be noted that there is no chance of updating how we receive
hotplug (dlpar) requests from the HMC on PowerVM systems.
Signed-off-by: Nathan Fontenot <nfont@linux.vnet.ibm.com>
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
2015-02-11 02:47:02 +07:00
|
|
|
#define pr_fmt(fmt) "pseries-hotplug-mem: " fmt
|
|
|
|
|
2008-04-19 03:33:50 +07:00
|
|
|
#include <linux/of.h>
|
2013-09-26 19:40:04 +07:00
|
|
|
#include <linux/of_address.h>
|
2010-07-12 11:36:09 +07:00
|
|
|
#include <linux/memblock.h>
|
2011-06-14 07:57:51 +07:00
|
|
|
#include <linux/memory.h>
|
2014-01-27 23:54:06 +07:00
|
|
|
#include <linux/memory_hotplug.h>
|
2015-02-11 02:48:25 +07:00
|
|
|
#include <linux/slab.h>
|
2011-06-14 07:57:51 +07:00
|
|
|
|
2008-04-19 03:33:50 +07:00
|
|
|
#include <asm/firmware.h>
|
|
|
|
#include <asm/machdep.h>
|
2013-09-26 19:40:04 +07:00
|
|
|
#include <asm/prom.h>
|
2009-02-08 21:49:39 +07:00
|
|
|
#include <asm/sparsemem.h>
|
2017-06-02 00:21:26 +07:00
|
|
|
#include <asm/fadump.h>
|
2017-12-01 23:47:31 +07:00
|
|
|
#include <asm/drmem.h>
|
2014-08-20 05:55:19 +07:00
|
|
|
#include "pseries.h"
|
2008-04-19 03:33:50 +07:00
|
|
|
|
2015-02-11 02:48:25 +07:00
|
|
|
static bool rtas_hp_event;
|
|
|
|
|
2014-06-04 14:50:47 +07:00
|
|
|
unsigned long pseries_memory_block_size(void)
|
2011-01-20 23:45:20 +07:00
|
|
|
{
|
|
|
|
struct device_node *np;
|
2011-06-14 07:57:51 +07:00
|
|
|
unsigned int memblock_size = MIN_MEMORY_BLOCK_SIZE;
|
|
|
|
struct resource r;
|
2011-01-20 23:45:20 +07:00
|
|
|
|
|
|
|
np = of_find_node_by_path("/ibm,dynamic-reconfiguration-memory");
|
|
|
|
if (np) {
|
2011-06-14 07:57:51 +07:00
|
|
|
const __be64 *size;
|
2011-01-20 23:45:20 +07:00
|
|
|
|
|
|
|
size = of_get_property(np, "ibm,lmb-size", NULL);
|
2011-06-14 07:57:51 +07:00
|
|
|
if (size)
|
|
|
|
memblock_size = be64_to_cpup(size);
|
2011-01-20 23:45:20 +07:00
|
|
|
of_node_put(np);
|
2011-06-14 07:57:51 +07:00
|
|
|
} else if (machine_is(pseries)) {
|
|
|
|
/* This fallback really only applies to pseries */
|
2011-01-20 23:45:20 +07:00
|
|
|
unsigned int memzero_size = 0;
|
|
|
|
|
|
|
|
np = of_find_node_by_path("/memory@0");
|
|
|
|
if (np) {
|
2011-06-14 07:57:51 +07:00
|
|
|
if (!of_address_to_resource(np, 0, &r))
|
|
|
|
memzero_size = resource_size(&r);
|
2011-01-20 23:45:20 +07:00
|
|
|
of_node_put(np);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memzero_size) {
|
|
|
|
/* We now know the size of memory@0, use this to find
|
|
|
|
* the first memoryblock and get its size.
|
|
|
|
*/
|
|
|
|
char buf[64];
|
|
|
|
|
|
|
|
sprintf(buf, "/memory@%x", memzero_size);
|
|
|
|
np = of_find_node_by_path(buf);
|
|
|
|
if (np) {
|
2011-06-14 07:57:51 +07:00
|
|
|
if (!of_address_to_resource(np, 0, &r))
|
|
|
|
memblock_size = resource_size(&r);
|
2011-01-20 23:45:20 +07:00
|
|
|
of_node_put(np);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return memblock_size;
|
|
|
|
}
|
|
|
|
|
2016-06-20 21:00:39 +07:00
|
|
|
static void dlpar_free_property(struct property *prop)
|
2015-02-11 02:48:25 +07:00
|
|
|
{
|
|
|
|
kfree(prop->name);
|
|
|
|
kfree(prop->value);
|
|
|
|
kfree(prop);
|
|
|
|
}
|
|
|
|
|
2016-06-20 21:00:39 +07:00
|
|
|
static struct property *dlpar_clone_property(struct property *prop,
|
|
|
|
u32 prop_size)
|
|
|
|
{
|
|
|
|
struct property *new_prop;
|
|
|
|
|
|
|
|
new_prop = kzalloc(sizeof(*new_prop), GFP_KERNEL);
|
|
|
|
if (!new_prop)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
new_prop->name = kstrdup(prop->name, GFP_KERNEL);
|
|
|
|
new_prop->value = kzalloc(prop_size, GFP_KERNEL);
|
|
|
|
if (!new_prop->name || !new_prop->value) {
|
|
|
|
dlpar_free_property(new_prop);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(new_prop->value, prop->value, prop->length);
|
|
|
|
new_prop->length = prop_size;
|
|
|
|
|
|
|
|
of_property_set_flag(new_prop, OF_DYNAMIC);
|
|
|
|
return new_prop;
|
|
|
|
}
|
|
|
|
|
2018-10-09 20:59:13 +07:00
|
|
|
static bool find_aa_index(struct device_node *dr_node,
|
|
|
|
struct property *ala_prop,
|
|
|
|
const u32 *lmb_assoc, u32 *aa_index)
|
2016-06-20 21:01:38 +07:00
|
|
|
{
|
2018-10-09 20:59:13 +07:00
|
|
|
u32 *assoc_arrays, new_prop_size;
|
|
|
|
struct property *new_prop;
|
2016-06-20 21:01:38 +07:00
|
|
|
int aa_arrays, aa_array_entries, aa_array_sz;
|
|
|
|
int i, index;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The ibm,associativity-lookup-arrays property is defined to be
|
|
|
|
* a 32-bit value specifying the number of associativity arrays
|
|
|
|
* followed by a 32-bitvalue specifying the number of entries per
|
|
|
|
* array, followed by the associativity arrays.
|
|
|
|
*/
|
|
|
|
assoc_arrays = ala_prop->value;
|
|
|
|
|
|
|
|
aa_arrays = be32_to_cpu(assoc_arrays[0]);
|
|
|
|
aa_array_entries = be32_to_cpu(assoc_arrays[1]);
|
|
|
|
aa_array_sz = aa_array_entries * sizeof(u32);
|
|
|
|
|
|
|
|
for (i = 0; i < aa_arrays; i++) {
|
|
|
|
index = (i * aa_array_entries) + 2;
|
|
|
|
|
|
|
|
if (memcmp(&assoc_arrays[index], &lmb_assoc[1], aa_array_sz))
|
|
|
|
continue;
|
|
|
|
|
2018-10-09 20:59:13 +07:00
|
|
|
*aa_index = i;
|
|
|
|
return true;
|
2016-06-20 21:01:38 +07:00
|
|
|
}
|
|
|
|
|
2018-10-09 20:59:13 +07:00
|
|
|
new_prop_size = ala_prop->length + aa_array_sz;
|
|
|
|
new_prop = dlpar_clone_property(ala_prop, new_prop_size);
|
|
|
|
if (!new_prop)
|
|
|
|
return false;
|
2016-06-20 21:01:38 +07:00
|
|
|
|
2018-10-09 20:59:13 +07:00
|
|
|
assoc_arrays = new_prop->value;
|
2016-06-20 21:01:38 +07:00
|
|
|
|
2018-10-09 20:59:13 +07:00
|
|
|
/* increment the number of entries in the lookup array */
|
|
|
|
assoc_arrays[0] = cpu_to_be32(aa_arrays + 1);
|
2016-06-20 21:01:38 +07:00
|
|
|
|
2018-10-09 20:59:13 +07:00
|
|
|
/* copy the new associativity into the lookup array */
|
|
|
|
index = aa_arrays * aa_array_entries + 2;
|
|
|
|
memcpy(&assoc_arrays[index], &lmb_assoc[1], aa_array_sz);
|
2016-06-20 21:01:38 +07:00
|
|
|
|
2018-10-09 20:59:13 +07:00
|
|
|
of_update_property(dr_node, new_prop);
|
2016-06-20 21:01:38 +07:00
|
|
|
|
2018-10-09 20:59:13 +07:00
|
|
|
/*
|
|
|
|
* The associativity lookup array index for this lmb is
|
|
|
|
* number of entries - 1 since we added its associativity
|
|
|
|
* to the end of the lookup array.
|
|
|
|
*/
|
|
|
|
*aa_index = be32_to_cpu(assoc_arrays[0]) - 1;
|
|
|
|
return true;
|
2016-06-20 21:01:38 +07:00
|
|
|
}
|
|
|
|
|
2018-04-21 03:29:48 +07:00
|
|
|
static int update_lmb_associativity_index(struct drmem_lmb *lmb)
|
2016-02-11 00:12:13 +07:00
|
|
|
{
|
|
|
|
struct device_node *parent, *lmb_node, *dr_node;
|
2016-06-20 21:01:38 +07:00
|
|
|
struct property *ala_prop;
|
2016-02-11 00:12:13 +07:00
|
|
|
const u32 *lmb_assoc;
|
|
|
|
u32 aa_index;
|
2018-10-09 20:59:13 +07:00
|
|
|
bool found;
|
2016-02-11 00:12:13 +07:00
|
|
|
|
|
|
|
parent = of_find_node_by_path("/");
|
|
|
|
if (!parent)
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
lmb_node = dlpar_configure_connector(cpu_to_be32(lmb->drc_index),
|
|
|
|
parent);
|
|
|
|
of_node_put(parent);
|
|
|
|
if (!lmb_node)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
lmb_assoc = of_get_property(lmb_node, "ibm,associativity", NULL);
|
|
|
|
if (!lmb_assoc) {
|
|
|
|
dlpar_free_cc_nodes(lmb_node);
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
dr_node = of_find_node_by_path("/ibm,dynamic-reconfiguration-memory");
|
|
|
|
if (!dr_node) {
|
|
|
|
dlpar_free_cc_nodes(lmb_node);
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
2016-06-20 21:01:38 +07:00
|
|
|
ala_prop = of_find_property(dr_node, "ibm,associativity-lookup-arrays",
|
|
|
|
NULL);
|
|
|
|
if (!ala_prop) {
|
|
|
|
of_node_put(dr_node);
|
2016-02-11 00:12:13 +07:00
|
|
|
dlpar_free_cc_nodes(lmb_node);
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
2018-10-09 20:59:13 +07:00
|
|
|
found = find_aa_index(dr_node, ala_prop, lmb_assoc, &aa_index);
|
2016-02-11 00:12:13 +07:00
|
|
|
|
2018-11-27 15:16:44 +07:00
|
|
|
of_node_put(dr_node);
|
2016-02-11 00:12:13 +07:00
|
|
|
dlpar_free_cc_nodes(lmb_node);
|
|
|
|
|
2018-10-09 20:59:13 +07:00
|
|
|
if (!found) {
|
2018-04-21 03:29:48 +07:00
|
|
|
pr_err("Could not find LMB associativity\n");
|
|
|
|
return -1;
|
2016-02-11 00:12:13 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
lmb->aa_index = aa_index;
|
2018-04-21 03:29:48 +07:00
|
|
|
return 0;
|
2016-02-11 00:12:13 +07:00
|
|
|
}
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static struct memory_block *lmb_to_memblock(struct drmem_lmb *lmb)
|
2017-02-16 01:45:30 +07:00
|
|
|
{
|
|
|
|
unsigned long section_nr;
|
|
|
|
struct mem_section *mem_sect;
|
|
|
|
struct memory_block *mem_block;
|
|
|
|
|
|
|
|
section_nr = pfn_to_section_nr(PFN_DOWN(lmb->base_addr));
|
|
|
|
mem_sect = __nr_to_section(section_nr);
|
|
|
|
|
|
|
|
mem_block = find_memory_block(mem_sect);
|
|
|
|
return mem_block;
|
|
|
|
}
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static int get_lmb_range(u32 drc_index, int n_lmbs,
|
|
|
|
struct drmem_lmb **start_lmb,
|
|
|
|
struct drmem_lmb **end_lmb)
|
|
|
|
{
|
|
|
|
struct drmem_lmb *lmb, *start, *end;
|
|
|
|
struct drmem_lmb *last_lmb;
|
|
|
|
|
|
|
|
start = NULL;
|
|
|
|
for_each_drmem_lmb(lmb) {
|
|
|
|
if (lmb->drc_index == drc_index) {
|
|
|
|
start = lmb;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!start)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
end = &start[n_lmbs - 1];
|
|
|
|
|
|
|
|
last_lmb = &drmem_info->lmbs[drmem_info->n_lmbs - 1];
|
|
|
|
if (end > last_lmb)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
*start_lmb = start;
|
|
|
|
*end_lmb = end;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int dlpar_change_lmb_state(struct drmem_lmb *lmb, bool online)
|
2017-08-03 01:03:22 +07:00
|
|
|
{
|
|
|
|
struct memory_block *mem_block;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
mem_block = lmb_to_memblock(lmb);
|
|
|
|
if (!mem_block)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (online && mem_block->dev.offline)
|
|
|
|
rc = device_online(&mem_block->dev);
|
|
|
|
else if (!online && !mem_block->dev.offline)
|
|
|
|
rc = device_offline(&mem_block->dev);
|
|
|
|
else
|
|
|
|
rc = 0;
|
|
|
|
|
|
|
|
put_device(&mem_block->dev);
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_online_lmb(struct drmem_lmb *lmb)
|
2017-08-03 01:03:22 +07:00
|
|
|
{
|
|
|
|
return dlpar_change_lmb_state(lmb, true);
|
|
|
|
}
|
|
|
|
|
2013-04-30 05:08:22 +07:00
|
|
|
#ifdef CONFIG_MEMORY_HOTREMOVE
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_offline_lmb(struct drmem_lmb *lmb)
|
2017-08-03 01:03:22 +07:00
|
|
|
{
|
|
|
|
return dlpar_change_lmb_state(lmb, false);
|
|
|
|
}
|
|
|
|
|
2014-01-27 23:54:06 +07:00
|
|
|
static int pseries_remove_memblock(unsigned long base, unsigned int memblock_size)
|
|
|
|
{
|
|
|
|
unsigned long block_sz, start_pfn;
|
|
|
|
int sections_per_block;
|
|
|
|
int i, nid;
|
2008-07-03 10:20:58 +07:00
|
|
|
|
2008-10-01 16:44:02 +07:00
|
|
|
start_pfn = base >> PAGE_SHIFT;
|
2008-10-13 15:42:00 +07:00
|
|
|
|
powerpc/pseries: Protect remove_memory() with device hotplug lock
While testing memory hot-remove, I found following dead lock:
Process #1141 is drmgr, trying to remove some memory, i.e. memory499.
It holds the memory_hotplug_mutex, and blocks when trying to remove file
"online" under dir memory499, in kernfs_drain(), at
wait_event(root->deactivate_waitq,
atomic_read(&kn->active) == KN_DEACTIVATED_BIAS);
Process #1120 is trying to online memory499 by
echo 1 > memory499/online
In .kernfs_fop_write, it uses kernfs_get_active() to increase
&kn->active, thus blocking process #1141. While itself is blocked later
when trying to acquire memory_hotplug_mutex, which is held by process
The backtrace of both processes are shown below:
[<c000000001b18600>] 0xc000000001b18600
[<c000000000015044>] .__switch_to+0x144/0x200
[<c000000000263ca4>] .online_pages+0x74/0x7b0
[<c00000000055b40c>] .memory_subsys_online+0x9c/0x150
[<c00000000053cbe8>] .device_online+0xb8/0x120
[<c00000000053cd04>] .online_store+0xb4/0xc0
[<c000000000538ce4>] .dev_attr_store+0x64/0xa0
[<c00000000030f4ec>] .sysfs_kf_write+0x7c/0xb0
[<c00000000030e574>] .kernfs_fop_write+0x154/0x1e0
[<c000000000268450>] .vfs_write+0xe0/0x260
[<c000000000269144>] .SyS_write+0x64/0x110
[<c000000000009ffc>] syscall_exit+0x0/0x7c
[<c000000001b18600>] 0xc000000001b18600
[<c000000000015044>] .__switch_to+0x144/0x200
[<c00000000030be14>] .__kernfs_remove+0x204/0x300
[<c00000000030d428>] .kernfs_remove_by_name_ns+0x68/0xf0
[<c00000000030fb38>] .sysfs_remove_file_ns+0x38/0x60
[<c000000000539354>] .device_remove_attrs+0x54/0xc0
[<c000000000539fd8>] .device_del+0x158/0x250
[<c00000000053a104>] .device_unregister+0x34/0xa0
[<c00000000055bc14>] .unregister_memory_section+0x164/0x170
[<c00000000024ee18>] .__remove_pages+0x108/0x4c0
[<c00000000004b590>] .arch_remove_memory+0x60/0xc0
[<c00000000026446c>] .remove_memory+0x8c/0xe0
[<c00000000007f9f4>] .pseries_remove_memblock+0xd4/0x160
[<c00000000007fcfc>] .pseries_memory_notifier+0x27c/0x290
[<c0000000008ae6cc>] .notifier_call_chain+0x8c/0x100
[<c0000000000d858c>] .__blocking_notifier_call_chain+0x6c/0xe0
[<c00000000071ddec>] .of_property_notify+0x7c/0xc0
[<c00000000071ed3c>] .of_update_property+0x3c/0x1b0
[<c0000000000756cc>] .ofdt_write+0x3dc/0x740
[<c0000000002f60fc>] .proc_reg_write+0xac/0x110
[<c000000000268450>] .vfs_write+0xe0/0x260
[<c000000000269144>] .SyS_write+0x64/0x110
[<c000000000009ffc>] syscall_exit+0x0/0x7c
This patch uses lock_device_hotplug() to protect remove_memory() called
in pseries_remove_memblock(), which is also stated before function
remove_memory():
* NOTE: The caller must call lock_device_hotplug() to serialize hotplug
* and online/offline operations before this call, as required by
* try_offline_node().
*/
void __ref remove_memory(int nid, u64 start, u64 size)
With this lock held, the other process(#1120 above) trying to online the
memory block will retry the system call when calling
lock_device_hotplug_sysfs(), and finally find No such device error.
Signed-off-by: Li Zhong <zhong@linux.vnet.ibm.com>
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
2014-04-10 15:25:31 +07:00
|
|
|
lock_device_hotplug();
|
|
|
|
|
|
|
|
if (!pfn_valid(start_pfn))
|
|
|
|
goto out;
|
2008-10-13 15:42:00 +07:00
|
|
|
|
2014-06-04 14:50:47 +07:00
|
|
|
block_sz = pseries_memory_block_size();
|
2014-01-27 23:54:06 +07:00
|
|
|
sections_per_block = block_sz / MIN_MEMORY_BLOCK_SIZE;
|
|
|
|
nid = memory_add_physaddr_to_nid(base);
|
2008-04-19 03:33:50 +07:00
|
|
|
|
2014-01-27 23:54:06 +07:00
|
|
|
for (i = 0; i < sections_per_block; i++) {
|
mm/memory_hotplug: make remove_memory() take the device_hotplug_lock
Patch series "mm: online/offline_pages called w.o. mem_hotplug_lock", v3.
Reading through the code and studying how mem_hotplug_lock is to be used,
I noticed that there are two places where we can end up calling
device_online()/device_offline() - online_pages()/offline_pages() without
the mem_hotplug_lock. And there are other places where we call
device_online()/device_offline() without the device_hotplug_lock.
While e.g.
echo "online" > /sys/devices/system/memory/memory9/state
is fine, e.g.
echo 1 > /sys/devices/system/memory/memory9/online
Will not take the mem_hotplug_lock. However the device_lock() and
device_hotplug_lock.
E.g. via memory_probe_store(), we can end up calling
add_memory()->online_pages() without the device_hotplug_lock. So we can
have concurrent callers in online_pages(). We e.g. touch in
online_pages() basically unprotected zone->present_pages then.
Looks like there is a longer history to that (see Patch #2 for details),
and fixing it to work the way it was intended is not really possible. We
would e.g. have to take the mem_hotplug_lock in device/base/core.c, which
sounds wrong.
Summary: We had a lock inversion on mem_hotplug_lock and device_lock().
More details can be found in patch 3 and patch 6.
I propose the general rules (documentation added in patch 6):
1. add_memory/add_memory_resource() must only be called with
device_hotplug_lock.
2. remove_memory() must only be called with device_hotplug_lock. This is
already documented and holds for all callers.
3. device_online()/device_offline() must only be called with
device_hotplug_lock. This is already documented and true for now in core
code. Other callers (related to memory hotplug) have to be fixed up.
4. mem_hotplug_lock is taken inside of add_memory/remove_memory/
online_pages/offline_pages.
To me, this looks way cleaner than what we have right now (and easier to
verify). And looking at the documentation of remove_memory, using
lock_device_hotplug also for add_memory() feels natural.
This patch (of 6):
remove_memory() is exported right now but requires the
device_hotplug_lock, which is not exported. So let's provide a variant
that takes the lock and only export that one.
The lock is already held in
arch/powerpc/platforms/pseries/hotplug-memory.c
drivers/acpi/acpi_memhotplug.c
arch/powerpc/platforms/powernv/memtrace.c
Apart from that, there are not other users in the tree.
Link: http://lkml.kernel.org/r/20180925091457.28651-2-david@redhat.com
Signed-off-by: David Hildenbrand <david@redhat.com>
Reviewed-by: Pavel Tatashin <pavel.tatashin@microsoft.com>
Reviewed-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Reviewed-by: Rashmica Gupta <rashmica.g@gmail.com>
Reviewed-by: Oscar Salvador <osalvador@suse.de>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Michael Ellerman <mpe@ellerman.id.au>
Cc: "Rafael J. Wysocki" <rjw@rjwysocki.net>
Cc: Len Brown <lenb@kernel.org>
Cc: Rashmica Gupta <rashmica.g@gmail.com>
Cc: Michael Neuling <mikey@neuling.org>
Cc: Balbir Singh <bsingharora@gmail.com>
Cc: Nathan Fontenot <nfont@linux.vnet.ibm.com>
Cc: John Allen <jallen@linux.vnet.ibm.com>
Cc: Michal Hocko <mhocko@suse.com>
Cc: Dan Williams <dan.j.williams@intel.com>
Cc: Joonsoo Kim <iamjoonsoo.kim@lge.com>
Cc: Vlastimil Babka <vbabka@suse.cz>
Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Cc: YASUAKI ISHIMATSU <yasu.isimatu@gmail.com>
Cc: Mathieu Malaterre <malat@debian.org>
Cc: Boris Ostrovsky <boris.ostrovsky@oracle.com>
Cc: Haiyang Zhang <haiyangz@microsoft.com>
Cc: Heiko Carstens <heiko.carstens@de.ibm.com>
Cc: Jonathan Corbet <corbet@lwn.net>
Cc: Juergen Gross <jgross@suse.com>
Cc: Kate Stewart <kstewart@linuxfoundation.org>
Cc: "K. Y. Srinivasan" <kys@microsoft.com>
Cc: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: Philippe Ombredanne <pombredanne@nexb.com>
Cc: Stephen Hemminger <sthemmin@microsoft.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2018-10-31 05:10:18 +07:00
|
|
|
__remove_memory(nid, base, MIN_MEMORY_BLOCK_SIZE);
|
2014-01-27 23:54:06 +07:00
|
|
|
base += MIN_MEMORY_BLOCK_SIZE;
|
2012-10-09 06:34:14 +07:00
|
|
|
}
|
2008-04-19 03:33:50 +07:00
|
|
|
|
powerpc/pseries: Protect remove_memory() with device hotplug lock
While testing memory hot-remove, I found following dead lock:
Process #1141 is drmgr, trying to remove some memory, i.e. memory499.
It holds the memory_hotplug_mutex, and blocks when trying to remove file
"online" under dir memory499, in kernfs_drain(), at
wait_event(root->deactivate_waitq,
atomic_read(&kn->active) == KN_DEACTIVATED_BIAS);
Process #1120 is trying to online memory499 by
echo 1 > memory499/online
In .kernfs_fop_write, it uses kernfs_get_active() to increase
&kn->active, thus blocking process #1141. While itself is blocked later
when trying to acquire memory_hotplug_mutex, which is held by process
The backtrace of both processes are shown below:
[<c000000001b18600>] 0xc000000001b18600
[<c000000000015044>] .__switch_to+0x144/0x200
[<c000000000263ca4>] .online_pages+0x74/0x7b0
[<c00000000055b40c>] .memory_subsys_online+0x9c/0x150
[<c00000000053cbe8>] .device_online+0xb8/0x120
[<c00000000053cd04>] .online_store+0xb4/0xc0
[<c000000000538ce4>] .dev_attr_store+0x64/0xa0
[<c00000000030f4ec>] .sysfs_kf_write+0x7c/0xb0
[<c00000000030e574>] .kernfs_fop_write+0x154/0x1e0
[<c000000000268450>] .vfs_write+0xe0/0x260
[<c000000000269144>] .SyS_write+0x64/0x110
[<c000000000009ffc>] syscall_exit+0x0/0x7c
[<c000000001b18600>] 0xc000000001b18600
[<c000000000015044>] .__switch_to+0x144/0x200
[<c00000000030be14>] .__kernfs_remove+0x204/0x300
[<c00000000030d428>] .kernfs_remove_by_name_ns+0x68/0xf0
[<c00000000030fb38>] .sysfs_remove_file_ns+0x38/0x60
[<c000000000539354>] .device_remove_attrs+0x54/0xc0
[<c000000000539fd8>] .device_del+0x158/0x250
[<c00000000053a104>] .device_unregister+0x34/0xa0
[<c00000000055bc14>] .unregister_memory_section+0x164/0x170
[<c00000000024ee18>] .__remove_pages+0x108/0x4c0
[<c00000000004b590>] .arch_remove_memory+0x60/0xc0
[<c00000000026446c>] .remove_memory+0x8c/0xe0
[<c00000000007f9f4>] .pseries_remove_memblock+0xd4/0x160
[<c00000000007fcfc>] .pseries_memory_notifier+0x27c/0x290
[<c0000000008ae6cc>] .notifier_call_chain+0x8c/0x100
[<c0000000000d858c>] .__blocking_notifier_call_chain+0x6c/0xe0
[<c00000000071ddec>] .of_property_notify+0x7c/0xc0
[<c00000000071ed3c>] .of_update_property+0x3c/0x1b0
[<c0000000000756cc>] .ofdt_write+0x3dc/0x740
[<c0000000002f60fc>] .proc_reg_write+0xac/0x110
[<c000000000268450>] .vfs_write+0xe0/0x260
[<c000000000269144>] .SyS_write+0x64/0x110
[<c000000000009ffc>] syscall_exit+0x0/0x7c
This patch uses lock_device_hotplug() to protect remove_memory() called
in pseries_remove_memblock(), which is also stated before function
remove_memory():
* NOTE: The caller must call lock_device_hotplug() to serialize hotplug
* and online/offline operations before this call, as required by
* try_offline_node().
*/
void __ref remove_memory(int nid, u64 start, u64 size)
With this lock held, the other process(#1120 above) trying to online the
memory block will retry the system call when calling
lock_device_hotplug_sysfs(), and finally find No such device error.
Signed-off-by: Li Zhong <zhong@linux.vnet.ibm.com>
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
2014-04-10 15:25:31 +07:00
|
|
|
out:
|
2014-01-27 23:54:06 +07:00
|
|
|
/* Update memory regions for memory remove */
|
2010-07-12 11:36:09 +07:00
|
|
|
memblock_remove(base, memblock_size);
|
powerpc/pseries: Protect remove_memory() with device hotplug lock
While testing memory hot-remove, I found following dead lock:
Process #1141 is drmgr, trying to remove some memory, i.e. memory499.
It holds the memory_hotplug_mutex, and blocks when trying to remove file
"online" under dir memory499, in kernfs_drain(), at
wait_event(root->deactivate_waitq,
atomic_read(&kn->active) == KN_DEACTIVATED_BIAS);
Process #1120 is trying to online memory499 by
echo 1 > memory499/online
In .kernfs_fop_write, it uses kernfs_get_active() to increase
&kn->active, thus blocking process #1141. While itself is blocked later
when trying to acquire memory_hotplug_mutex, which is held by process
The backtrace of both processes are shown below:
[<c000000001b18600>] 0xc000000001b18600
[<c000000000015044>] .__switch_to+0x144/0x200
[<c000000000263ca4>] .online_pages+0x74/0x7b0
[<c00000000055b40c>] .memory_subsys_online+0x9c/0x150
[<c00000000053cbe8>] .device_online+0xb8/0x120
[<c00000000053cd04>] .online_store+0xb4/0xc0
[<c000000000538ce4>] .dev_attr_store+0x64/0xa0
[<c00000000030f4ec>] .sysfs_kf_write+0x7c/0xb0
[<c00000000030e574>] .kernfs_fop_write+0x154/0x1e0
[<c000000000268450>] .vfs_write+0xe0/0x260
[<c000000000269144>] .SyS_write+0x64/0x110
[<c000000000009ffc>] syscall_exit+0x0/0x7c
[<c000000001b18600>] 0xc000000001b18600
[<c000000000015044>] .__switch_to+0x144/0x200
[<c00000000030be14>] .__kernfs_remove+0x204/0x300
[<c00000000030d428>] .kernfs_remove_by_name_ns+0x68/0xf0
[<c00000000030fb38>] .sysfs_remove_file_ns+0x38/0x60
[<c000000000539354>] .device_remove_attrs+0x54/0xc0
[<c000000000539fd8>] .device_del+0x158/0x250
[<c00000000053a104>] .device_unregister+0x34/0xa0
[<c00000000055bc14>] .unregister_memory_section+0x164/0x170
[<c00000000024ee18>] .__remove_pages+0x108/0x4c0
[<c00000000004b590>] .arch_remove_memory+0x60/0xc0
[<c00000000026446c>] .remove_memory+0x8c/0xe0
[<c00000000007f9f4>] .pseries_remove_memblock+0xd4/0x160
[<c00000000007fcfc>] .pseries_memory_notifier+0x27c/0x290
[<c0000000008ae6cc>] .notifier_call_chain+0x8c/0x100
[<c0000000000d858c>] .__blocking_notifier_call_chain+0x6c/0xe0
[<c00000000071ddec>] .of_property_notify+0x7c/0xc0
[<c00000000071ed3c>] .of_update_property+0x3c/0x1b0
[<c0000000000756cc>] .ofdt_write+0x3dc/0x740
[<c0000000002f60fc>] .proc_reg_write+0xac/0x110
[<c000000000268450>] .vfs_write+0xe0/0x260
[<c000000000269144>] .SyS_write+0x64/0x110
[<c000000000009ffc>] syscall_exit+0x0/0x7c
This patch uses lock_device_hotplug() to protect remove_memory() called
in pseries_remove_memblock(), which is also stated before function
remove_memory():
* NOTE: The caller must call lock_device_hotplug() to serialize hotplug
* and online/offline operations before this call, as required by
* try_offline_node().
*/
void __ref remove_memory(int nid, u64 start, u64 size)
With this lock held, the other process(#1120 above) trying to online the
memory block will retry the system call when calling
lock_device_hotplug_sysfs(), and finally find No such device error.
Signed-off-by: Li Zhong <zhong@linux.vnet.ibm.com>
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
2014-04-10 15:25:31 +07:00
|
|
|
unlock_device_hotplug();
|
2014-01-27 23:54:06 +07:00
|
|
|
return 0;
|
2008-04-19 03:33:50 +07:00
|
|
|
}
|
|
|
|
|
2014-01-27 23:54:06 +07:00
|
|
|
static int pseries_remove_mem_node(struct device_node *np)
|
2008-07-03 10:22:39 +07:00
|
|
|
{
|
2014-08-20 03:44:57 +07:00
|
|
|
const __be32 *regs;
|
2008-07-03 10:22:39 +07:00
|
|
|
unsigned long base;
|
2010-07-23 07:35:52 +07:00
|
|
|
unsigned int lmb_size;
|
2008-07-03 10:22:39 +07:00
|
|
|
int ret = -EINVAL;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check to see if we are actually removing memory
|
|
|
|
*/
|
2018-11-17 05:11:00 +07:00
|
|
|
if (!of_node_is_type(np, "memory"))
|
2008-07-03 10:22:39 +07:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
/*
|
2014-08-07 12:11:58 +07:00
|
|
|
* Find the base address and size of the memblock
|
2008-07-03 10:22:39 +07:00
|
|
|
*/
|
|
|
|
regs = of_get_property(np, "reg", NULL);
|
|
|
|
if (!regs)
|
|
|
|
return ret;
|
|
|
|
|
2014-08-20 03:44:57 +07:00
|
|
|
base = be64_to_cpu(*(unsigned long *)regs);
|
|
|
|
lmb_size = be32_to_cpu(regs[3]);
|
2008-07-03 10:22:39 +07:00
|
|
|
|
2014-01-27 23:54:06 +07:00
|
|
|
pseries_remove_memblock(base, lmb_size);
|
|
|
|
return 0;
|
2008-07-03 10:22:39 +07:00
|
|
|
}
|
2015-02-11 02:49:22 +07:00
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static bool lmb_is_removable(struct drmem_lmb *lmb)
|
2015-02-11 02:49:22 +07:00
|
|
|
{
|
|
|
|
int i, scns_per_block;
|
|
|
|
int rc = 1;
|
|
|
|
unsigned long pfn, block_sz;
|
|
|
|
u64 phys_addr;
|
|
|
|
|
|
|
|
if (!(lmb->flags & DRCONF_MEM_ASSIGNED))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
block_sz = memory_block_size_bytes();
|
|
|
|
scns_per_block = block_sz / MIN_MEMORY_BLOCK_SIZE;
|
|
|
|
phys_addr = lmb->base_addr;
|
|
|
|
|
2017-06-02 00:21:26 +07:00
|
|
|
#ifdef CONFIG_FA_DUMP
|
2018-08-20 15:17:32 +07:00
|
|
|
/*
|
|
|
|
* Don't hot-remove memory that falls in fadump boot memory area
|
|
|
|
* and memory that is reserved for capturing old kernel memory.
|
|
|
|
*/
|
|
|
|
if (is_fadump_memory_area(phys_addr, block_sz))
|
2017-06-02 00:21:26 +07:00
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
|
2015-02-11 02:49:22 +07:00
|
|
|
for (i = 0; i < scns_per_block; i++) {
|
|
|
|
pfn = PFN_DOWN(phys_addr);
|
|
|
|
if (!pfn_present(pfn))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
rc &= is_mem_section_removable(pfn, PAGES_PER_SECTION);
|
|
|
|
phys_addr += MIN_MEMORY_BLOCK_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc ? true : false;
|
|
|
|
}
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_add_lmb(struct drmem_lmb *);
|
2015-02-11 02:49:22 +07:00
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_remove_lmb(struct drmem_lmb *lmb)
|
2015-02-11 02:49:22 +07:00
|
|
|
{
|
|
|
|
unsigned long block_sz;
|
|
|
|
int nid, rc;
|
|
|
|
|
|
|
|
if (!lmb_is_removable(lmb))
|
|
|
|
return -EINVAL;
|
|
|
|
|
2017-08-03 01:03:22 +07:00
|
|
|
rc = dlpar_offline_lmb(lmb);
|
2015-02-11 02:49:22 +07:00
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
|
|
|
|
block_sz = pseries_memory_block_size();
|
|
|
|
nid = memory_add_physaddr_to_nid(lmb->base_addr);
|
|
|
|
|
mm/memory_hotplug: make remove_memory() take the device_hotplug_lock
Patch series "mm: online/offline_pages called w.o. mem_hotplug_lock", v3.
Reading through the code and studying how mem_hotplug_lock is to be used,
I noticed that there are two places where we can end up calling
device_online()/device_offline() - online_pages()/offline_pages() without
the mem_hotplug_lock. And there are other places where we call
device_online()/device_offline() without the device_hotplug_lock.
While e.g.
echo "online" > /sys/devices/system/memory/memory9/state
is fine, e.g.
echo 1 > /sys/devices/system/memory/memory9/online
Will not take the mem_hotplug_lock. However the device_lock() and
device_hotplug_lock.
E.g. via memory_probe_store(), we can end up calling
add_memory()->online_pages() without the device_hotplug_lock. So we can
have concurrent callers in online_pages(). We e.g. touch in
online_pages() basically unprotected zone->present_pages then.
Looks like there is a longer history to that (see Patch #2 for details),
and fixing it to work the way it was intended is not really possible. We
would e.g. have to take the mem_hotplug_lock in device/base/core.c, which
sounds wrong.
Summary: We had a lock inversion on mem_hotplug_lock and device_lock().
More details can be found in patch 3 and patch 6.
I propose the general rules (documentation added in patch 6):
1. add_memory/add_memory_resource() must only be called with
device_hotplug_lock.
2. remove_memory() must only be called with device_hotplug_lock. This is
already documented and holds for all callers.
3. device_online()/device_offline() must only be called with
device_hotplug_lock. This is already documented and true for now in core
code. Other callers (related to memory hotplug) have to be fixed up.
4. mem_hotplug_lock is taken inside of add_memory/remove_memory/
online_pages/offline_pages.
To me, this looks way cleaner than what we have right now (and easier to
verify). And looking at the documentation of remove_memory, using
lock_device_hotplug also for add_memory() feels natural.
This patch (of 6):
remove_memory() is exported right now but requires the
device_hotplug_lock, which is not exported. So let's provide a variant
that takes the lock and only export that one.
The lock is already held in
arch/powerpc/platforms/pseries/hotplug-memory.c
drivers/acpi/acpi_memhotplug.c
arch/powerpc/platforms/powernv/memtrace.c
Apart from that, there are not other users in the tree.
Link: http://lkml.kernel.org/r/20180925091457.28651-2-david@redhat.com
Signed-off-by: David Hildenbrand <david@redhat.com>
Reviewed-by: Pavel Tatashin <pavel.tatashin@microsoft.com>
Reviewed-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Reviewed-by: Rashmica Gupta <rashmica.g@gmail.com>
Reviewed-by: Oscar Salvador <osalvador@suse.de>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Michael Ellerman <mpe@ellerman.id.au>
Cc: "Rafael J. Wysocki" <rjw@rjwysocki.net>
Cc: Len Brown <lenb@kernel.org>
Cc: Rashmica Gupta <rashmica.g@gmail.com>
Cc: Michael Neuling <mikey@neuling.org>
Cc: Balbir Singh <bsingharora@gmail.com>
Cc: Nathan Fontenot <nfont@linux.vnet.ibm.com>
Cc: John Allen <jallen@linux.vnet.ibm.com>
Cc: Michal Hocko <mhocko@suse.com>
Cc: Dan Williams <dan.j.williams@intel.com>
Cc: Joonsoo Kim <iamjoonsoo.kim@lge.com>
Cc: Vlastimil Babka <vbabka@suse.cz>
Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Cc: YASUAKI ISHIMATSU <yasu.isimatu@gmail.com>
Cc: Mathieu Malaterre <malat@debian.org>
Cc: Boris Ostrovsky <boris.ostrovsky@oracle.com>
Cc: Haiyang Zhang <haiyangz@microsoft.com>
Cc: Heiko Carstens <heiko.carstens@de.ibm.com>
Cc: Jonathan Corbet <corbet@lwn.net>
Cc: Juergen Gross <jgross@suse.com>
Cc: Kate Stewart <kstewart@linuxfoundation.org>
Cc: "K. Y. Srinivasan" <kys@microsoft.com>
Cc: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: Philippe Ombredanne <pombredanne@nexb.com>
Cc: Stephen Hemminger <sthemmin@microsoft.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2018-10-31 05:10:18 +07:00
|
|
|
__remove_memory(nid, lmb->base_addr, block_sz);
|
2015-02-11 02:49:22 +07:00
|
|
|
|
|
|
|
/* Update memory regions for memory remove */
|
|
|
|
memblock_remove(lmb->base_addr, block_sz);
|
|
|
|
|
2018-04-21 03:29:48 +07:00
|
|
|
invalidate_lmb_associativity_index(lmb);
|
|
|
|
lmb->flags &= ~DRCONF_MEM_ASSIGNED;
|
|
|
|
|
2015-02-11 02:49:22 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_memory_remove_by_count(u32 lmbs_to_remove)
|
2015-02-11 02:49:22 +07:00
|
|
|
{
|
2017-12-01 23:47:31 +07:00
|
|
|
struct drmem_lmb *lmb;
|
2015-02-11 02:49:22 +07:00
|
|
|
int lmbs_removed = 0;
|
|
|
|
int lmbs_available = 0;
|
2017-12-01 23:47:31 +07:00
|
|
|
int rc;
|
2015-02-11 02:49:22 +07:00
|
|
|
|
|
|
|
pr_info("Attempting to hot-remove %d LMB(s)\n", lmbs_to_remove);
|
|
|
|
|
|
|
|
if (lmbs_to_remove == 0)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* Validate that there are enough LMBs to satisfy the request */
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb(lmb) {
|
|
|
|
if (lmb_is_removable(lmb))
|
2015-02-11 02:49:22 +07:00
|
|
|
lmbs_available++;
|
2017-12-01 23:47:31 +07:00
|
|
|
|
|
|
|
if (lmbs_available == lmbs_to_remove)
|
|
|
|
break;
|
2015-02-11 02:49:22 +07:00
|
|
|
}
|
|
|
|
|
2016-11-28 23:50:45 +07:00
|
|
|
if (lmbs_available < lmbs_to_remove) {
|
|
|
|
pr_info("Not enough LMBs available (%d of %d) to satisfy request\n",
|
|
|
|
lmbs_available, lmbs_to_remove);
|
2015-02-11 02:49:22 +07:00
|
|
|
return -EINVAL;
|
2016-11-28 23:50:45 +07:00
|
|
|
}
|
2015-02-11 02:49:22 +07:00
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb(lmb) {
|
|
|
|
rc = dlpar_remove_lmb(lmb);
|
2015-02-11 02:49:22 +07:00
|
|
|
if (rc)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Mark this lmb so we can add it later if all of the
|
|
|
|
* requested LMBs cannot be removed.
|
|
|
|
*/
|
2017-12-01 23:47:31 +07:00
|
|
|
drmem_mark_lmb_reserved(lmb);
|
|
|
|
|
|
|
|
lmbs_removed++;
|
|
|
|
if (lmbs_removed == lmbs_to_remove)
|
|
|
|
break;
|
2015-02-11 02:49:22 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (lmbs_removed != lmbs_to_remove) {
|
|
|
|
pr_err("Memory hot-remove failed, adding LMB's back\n");
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb(lmb) {
|
|
|
|
if (!drmem_lmb_reserved(lmb))
|
2015-02-11 02:49:22 +07:00
|
|
|
continue;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_add_lmb(lmb);
|
2015-02-11 02:49:22 +07:00
|
|
|
if (rc)
|
|
|
|
pr_err("Failed to add LMB back, drc index %x\n",
|
2017-12-01 23:47:31 +07:00
|
|
|
lmb->drc_index);
|
2015-02-11 02:49:22 +07:00
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
drmem_remove_lmb_reservation(lmb);
|
2015-02-11 02:49:22 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
rc = -EINVAL;
|
|
|
|
} else {
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb(lmb) {
|
|
|
|
if (!drmem_lmb_reserved(lmb))
|
2015-02-11 02:49:22 +07:00
|
|
|
continue;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
dlpar_release_drc(lmb->drc_index);
|
2015-02-11 02:49:22 +07:00
|
|
|
pr_info("Memory at %llx was hot-removed\n",
|
2017-12-01 23:47:31 +07:00
|
|
|
lmb->base_addr);
|
2015-02-11 02:49:22 +07:00
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
drmem_remove_lmb_reservation(lmb);
|
2015-02-11 02:49:22 +07:00
|
|
|
}
|
|
|
|
rc = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_memory_remove_by_index(u32 drc_index)
|
2015-02-11 02:49:22 +07:00
|
|
|
{
|
2017-12-01 23:47:31 +07:00
|
|
|
struct drmem_lmb *lmb;
|
2015-02-11 02:49:22 +07:00
|
|
|
int lmb_found;
|
2017-12-01 23:47:31 +07:00
|
|
|
int rc;
|
2015-02-11 02:49:22 +07:00
|
|
|
|
|
|
|
pr_info("Attempting to hot-remove LMB, drc index %x\n", drc_index);
|
|
|
|
|
|
|
|
lmb_found = 0;
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb(lmb) {
|
|
|
|
if (lmb->drc_index == drc_index) {
|
2015-02-11 02:49:22 +07:00
|
|
|
lmb_found = 1;
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_remove_lmb(lmb);
|
2017-01-07 02:25:53 +07:00
|
|
|
if (!rc)
|
2017-12-01 23:47:31 +07:00
|
|
|
dlpar_release_drc(lmb->drc_index);
|
2017-01-07 02:25:53 +07:00
|
|
|
|
2015-02-11 02:49:22 +07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!lmb_found)
|
|
|
|
rc = -EINVAL;
|
|
|
|
|
|
|
|
if (rc)
|
|
|
|
pr_info("Failed to hot-remove memory at %llx\n",
|
2017-12-01 23:47:31 +07:00
|
|
|
lmb->base_addr);
|
2015-02-11 02:49:22 +07:00
|
|
|
else
|
2017-12-01 23:47:31 +07:00
|
|
|
pr_info("Memory at %llx was hot-removed\n", lmb->base_addr);
|
2015-02-11 02:49:22 +07:00
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_memory_readd_by_index(u32 drc_index)
|
2017-01-07 02:27:26 +07:00
|
|
|
{
|
2017-12-01 23:47:31 +07:00
|
|
|
struct drmem_lmb *lmb;
|
2017-01-07 02:27:26 +07:00
|
|
|
int lmb_found;
|
2017-12-01 23:47:31 +07:00
|
|
|
int rc;
|
2017-01-07 02:27:26 +07:00
|
|
|
|
|
|
|
pr_info("Attempting to update LMB, drc index %x\n", drc_index);
|
|
|
|
|
|
|
|
lmb_found = 0;
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb(lmb) {
|
|
|
|
if (lmb->drc_index == drc_index) {
|
2017-01-07 02:27:26 +07:00
|
|
|
lmb_found = 1;
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_remove_lmb(lmb);
|
2017-01-07 02:27:26 +07:00
|
|
|
if (!rc) {
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_add_lmb(lmb);
|
2017-01-07 02:27:26 +07:00
|
|
|
if (rc)
|
2017-12-01 23:47:31 +07:00
|
|
|
dlpar_release_drc(lmb->drc_index);
|
2017-01-07 02:27:26 +07:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!lmb_found)
|
|
|
|
rc = -EINVAL;
|
|
|
|
|
|
|
|
if (rc)
|
|
|
|
pr_info("Failed to update memory at %llx\n",
|
2017-12-01 23:47:31 +07:00
|
|
|
lmb->base_addr);
|
2017-01-07 02:27:26 +07:00
|
|
|
else
|
2017-12-01 23:47:31 +07:00
|
|
|
pr_info("Memory at %llx was updated\n", lmb->base_addr);
|
2017-01-07 02:27:26 +07:00
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
2017-02-16 01:46:18 +07:00
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_memory_remove_by_ic(u32 lmbs_to_remove, u32 drc_index)
|
2017-02-16 01:46:18 +07:00
|
|
|
{
|
2017-12-01 23:47:31 +07:00
|
|
|
struct drmem_lmb *lmb, *start_lmb, *end_lmb;
|
|
|
|
int lmbs_available = 0;
|
|
|
|
int rc;
|
2017-02-16 01:46:18 +07:00
|
|
|
|
|
|
|
pr_info("Attempting to hot-remove %u LMB(s) at %x\n",
|
|
|
|
lmbs_to_remove, drc_index);
|
|
|
|
|
|
|
|
if (lmbs_to_remove == 0)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = get_lmb_range(drc_index, lmbs_to_remove, &start_lmb, &end_lmb);
|
|
|
|
if (rc)
|
2017-02-16 01:46:18 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* Validate that there are enough LMBs to satisfy the request */
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb_in_range(lmb, start_lmb, end_lmb) {
|
|
|
|
if (lmb->flags & DRCONF_MEM_RESERVED)
|
2017-02-16 01:46:18 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
lmbs_available++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lmbs_available < lmbs_to_remove)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb_in_range(lmb, start_lmb, end_lmb) {
|
|
|
|
if (!(lmb->flags & DRCONF_MEM_ASSIGNED))
|
2017-02-16 01:46:18 +07:00
|
|
|
continue;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_remove_lmb(lmb);
|
2017-02-16 01:46:18 +07:00
|
|
|
if (rc)
|
|
|
|
break;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
drmem_mark_lmb_reserved(lmb);
|
2017-02-16 01:46:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (rc) {
|
|
|
|
pr_err("Memory indexed-count-remove failed, adding any removed LMBs\n");
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
|
|
|
|
for_each_drmem_lmb_in_range(lmb, start_lmb, end_lmb) {
|
|
|
|
if (!drmem_lmb_reserved(lmb))
|
2017-02-16 01:46:18 +07:00
|
|
|
continue;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_add_lmb(lmb);
|
2017-02-16 01:46:18 +07:00
|
|
|
if (rc)
|
|
|
|
pr_err("Failed to add LMB, drc index %x\n",
|
2017-12-01 23:47:31 +07:00
|
|
|
lmb->drc_index);
|
2017-02-16 01:46:18 +07:00
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
drmem_remove_lmb_reservation(lmb);
|
2017-02-16 01:46:18 +07:00
|
|
|
}
|
|
|
|
rc = -EINVAL;
|
|
|
|
} else {
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb_in_range(lmb, start_lmb, end_lmb) {
|
|
|
|
if (!drmem_lmb_reserved(lmb))
|
2017-02-16 01:46:18 +07:00
|
|
|
continue;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
dlpar_release_drc(lmb->drc_index);
|
2017-02-16 01:46:18 +07:00
|
|
|
pr_info("Memory at %llx (drc index %x) was hot-removed\n",
|
2017-12-01 23:47:31 +07:00
|
|
|
lmb->base_addr, lmb->drc_index);
|
2017-02-16 01:46:18 +07:00
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
drmem_remove_lmb_reservation(lmb);
|
2017-02-16 01:46:18 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2013-04-30 05:08:22 +07:00
|
|
|
#else
|
|
|
|
static inline int pseries_remove_memblock(unsigned long base,
|
|
|
|
unsigned int memblock_size)
|
|
|
|
{
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
2014-01-27 23:54:06 +07:00
|
|
|
static inline int pseries_remove_mem_node(struct device_node *np)
|
2013-04-30 05:08:22 +07:00
|
|
|
{
|
2014-08-11 16:16:19 +07:00
|
|
|
return 0;
|
2013-04-30 05:08:22 +07:00
|
|
|
}
|
2015-02-11 02:49:22 +07:00
|
|
|
static inline int dlpar_memory_remove(struct pseries_hp_errorlog *hp_elog)
|
|
|
|
{
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_remove_lmb(struct drmem_lmb *lmb)
|
2015-04-14 14:01:56 +07:00
|
|
|
{
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_memory_remove_by_count(u32 lmbs_to_remove)
|
2015-04-14 14:01:56 +07:00
|
|
|
{
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_memory_remove_by_index(u32 drc_index)
|
2015-04-14 14:01:56 +07:00
|
|
|
{
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_memory_readd_by_index(u32 drc_index)
|
2017-02-16 07:27:00 +07:00
|
|
|
{
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
2017-02-16 01:46:18 +07:00
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_memory_remove_by_ic(u32 lmbs_to_remove, u32 drc_index)
|
2017-02-16 01:46:18 +07:00
|
|
|
{
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
}
|
2013-04-30 05:08:22 +07:00
|
|
|
#endif /* CONFIG_MEMORY_HOTREMOVE */
|
2008-07-03 10:22:39 +07:00
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_add_lmb(struct drmem_lmb *lmb)
|
2015-02-11 02:48:25 +07:00
|
|
|
{
|
|
|
|
unsigned long block_sz;
|
|
|
|
int nid, rc;
|
|
|
|
|
2016-02-11 00:10:44 +07:00
|
|
|
if (lmb->flags & DRCONF_MEM_ASSIGNED)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2018-04-21 03:29:48 +07:00
|
|
|
rc = update_lmb_associativity_index(lmb);
|
2016-02-11 00:12:13 +07:00
|
|
|
if (rc) {
|
|
|
|
dlpar_release_drc(lmb->drc_index);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2016-06-30 00:20:30 +07:00
|
|
|
block_sz = memory_block_size_bytes();
|
|
|
|
|
|
|
|
/* Find the node id for this address */
|
|
|
|
nid = memory_add_physaddr_to_nid(lmb->base_addr);
|
|
|
|
|
|
|
|
/* Add the memory */
|
2018-10-31 05:10:24 +07:00
|
|
|
rc = __add_memory(nid, lmb->base_addr, block_sz);
|
2017-02-16 01:45:30 +07:00
|
|
|
if (rc) {
|
2018-04-21 03:29:48 +07:00
|
|
|
invalidate_lmb_associativity_index(lmb);
|
2017-02-16 01:45:30 +07:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = dlpar_online_lmb(lmb);
|
|
|
|
if (rc) {
|
mm/memory_hotplug: make remove_memory() take the device_hotplug_lock
Patch series "mm: online/offline_pages called w.o. mem_hotplug_lock", v3.
Reading through the code and studying how mem_hotplug_lock is to be used,
I noticed that there are two places where we can end up calling
device_online()/device_offline() - online_pages()/offline_pages() without
the mem_hotplug_lock. And there are other places where we call
device_online()/device_offline() without the device_hotplug_lock.
While e.g.
echo "online" > /sys/devices/system/memory/memory9/state
is fine, e.g.
echo 1 > /sys/devices/system/memory/memory9/online
Will not take the mem_hotplug_lock. However the device_lock() and
device_hotplug_lock.
E.g. via memory_probe_store(), we can end up calling
add_memory()->online_pages() without the device_hotplug_lock. So we can
have concurrent callers in online_pages(). We e.g. touch in
online_pages() basically unprotected zone->present_pages then.
Looks like there is a longer history to that (see Patch #2 for details),
and fixing it to work the way it was intended is not really possible. We
would e.g. have to take the mem_hotplug_lock in device/base/core.c, which
sounds wrong.
Summary: We had a lock inversion on mem_hotplug_lock and device_lock().
More details can be found in patch 3 and patch 6.
I propose the general rules (documentation added in patch 6):
1. add_memory/add_memory_resource() must only be called with
device_hotplug_lock.
2. remove_memory() must only be called with device_hotplug_lock. This is
already documented and holds for all callers.
3. device_online()/device_offline() must only be called with
device_hotplug_lock. This is already documented and true for now in core
code. Other callers (related to memory hotplug) have to be fixed up.
4. mem_hotplug_lock is taken inside of add_memory/remove_memory/
online_pages/offline_pages.
To me, this looks way cleaner than what we have right now (and easier to
verify). And looking at the documentation of remove_memory, using
lock_device_hotplug also for add_memory() feels natural.
This patch (of 6):
remove_memory() is exported right now but requires the
device_hotplug_lock, which is not exported. So let's provide a variant
that takes the lock and only export that one.
The lock is already held in
arch/powerpc/platforms/pseries/hotplug-memory.c
drivers/acpi/acpi_memhotplug.c
arch/powerpc/platforms/powernv/memtrace.c
Apart from that, there are not other users in the tree.
Link: http://lkml.kernel.org/r/20180925091457.28651-2-david@redhat.com
Signed-off-by: David Hildenbrand <david@redhat.com>
Reviewed-by: Pavel Tatashin <pavel.tatashin@microsoft.com>
Reviewed-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Reviewed-by: Rashmica Gupta <rashmica.g@gmail.com>
Reviewed-by: Oscar Salvador <osalvador@suse.de>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Michael Ellerman <mpe@ellerman.id.au>
Cc: "Rafael J. Wysocki" <rjw@rjwysocki.net>
Cc: Len Brown <lenb@kernel.org>
Cc: Rashmica Gupta <rashmica.g@gmail.com>
Cc: Michael Neuling <mikey@neuling.org>
Cc: Balbir Singh <bsingharora@gmail.com>
Cc: Nathan Fontenot <nfont@linux.vnet.ibm.com>
Cc: John Allen <jallen@linux.vnet.ibm.com>
Cc: Michal Hocko <mhocko@suse.com>
Cc: Dan Williams <dan.j.williams@intel.com>
Cc: Joonsoo Kim <iamjoonsoo.kim@lge.com>
Cc: Vlastimil Babka <vbabka@suse.cz>
Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Cc: YASUAKI ISHIMATSU <yasu.isimatu@gmail.com>
Cc: Mathieu Malaterre <malat@debian.org>
Cc: Boris Ostrovsky <boris.ostrovsky@oracle.com>
Cc: Haiyang Zhang <haiyangz@microsoft.com>
Cc: Heiko Carstens <heiko.carstens@de.ibm.com>
Cc: Jonathan Corbet <corbet@lwn.net>
Cc: Juergen Gross <jgross@suse.com>
Cc: Kate Stewart <kstewart@linuxfoundation.org>
Cc: "K. Y. Srinivasan" <kys@microsoft.com>
Cc: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: Philippe Ombredanne <pombredanne@nexb.com>
Cc: Stephen Hemminger <sthemmin@microsoft.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2018-10-31 05:10:18 +07:00
|
|
|
__remove_memory(nid, lmb->base_addr, block_sz);
|
2018-04-21 03:29:48 +07:00
|
|
|
invalidate_lmb_associativity_index(lmb);
|
2017-02-16 01:45:30 +07:00
|
|
|
} else {
|
2016-06-30 00:20:30 +07:00
|
|
|
lmb->flags |= DRCONF_MEM_ASSIGNED;
|
2017-02-16 01:45:30 +07:00
|
|
|
}
|
2016-02-11 00:10:44 +07:00
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_memory_add_by_count(u32 lmbs_to_add)
|
2015-02-11 02:48:25 +07:00
|
|
|
{
|
2017-12-01 23:47:31 +07:00
|
|
|
struct drmem_lmb *lmb;
|
2015-02-11 02:48:25 +07:00
|
|
|
int lmbs_available = 0;
|
|
|
|
int lmbs_added = 0;
|
2017-12-01 23:47:31 +07:00
|
|
|
int rc;
|
2015-02-11 02:48:25 +07:00
|
|
|
|
|
|
|
pr_info("Attempting to hot-add %d LMB(s)\n", lmbs_to_add);
|
|
|
|
|
|
|
|
if (lmbs_to_add == 0)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* Validate that there are enough LMBs to satisfy the request */
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb(lmb) {
|
|
|
|
if (!(lmb->flags & DRCONF_MEM_ASSIGNED))
|
2015-02-11 02:48:25 +07:00
|
|
|
lmbs_available++;
|
2017-12-01 23:47:31 +07:00
|
|
|
|
|
|
|
if (lmbs_available == lmbs_to_add)
|
|
|
|
break;
|
2015-02-11 02:48:25 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (lmbs_available < lmbs_to_add)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb(lmb) {
|
|
|
|
if (lmb->flags & DRCONF_MEM_ASSIGNED)
|
2017-08-24 00:18:43 +07:00
|
|
|
continue;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_acquire_drc(lmb->drc_index);
|
2015-02-11 02:48:25 +07:00
|
|
|
if (rc)
|
|
|
|
continue;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_add_lmb(lmb);
|
2017-01-07 02:25:53 +07:00
|
|
|
if (rc) {
|
2017-12-01 23:47:31 +07:00
|
|
|
dlpar_release_drc(lmb->drc_index);
|
2017-01-07 02:25:53 +07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-02-11 02:48:25 +07:00
|
|
|
/* Mark this lmb so we can remove it later if all of the
|
|
|
|
* requested LMBs cannot be added.
|
|
|
|
*/
|
2017-12-01 23:47:31 +07:00
|
|
|
drmem_mark_lmb_reserved(lmb);
|
|
|
|
|
|
|
|
lmbs_added++;
|
|
|
|
if (lmbs_added == lmbs_to_add)
|
|
|
|
break;
|
2015-02-11 02:48:25 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (lmbs_added != lmbs_to_add) {
|
2015-02-11 02:49:22 +07:00
|
|
|
pr_err("Memory hot-add failed, removing any added LMBs\n");
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb(lmb) {
|
|
|
|
if (!drmem_lmb_reserved(lmb))
|
2015-02-11 02:49:22 +07:00
|
|
|
continue;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_remove_lmb(lmb);
|
2015-02-11 02:49:22 +07:00
|
|
|
if (rc)
|
|
|
|
pr_err("Failed to remove LMB, drc index %x\n",
|
2017-12-01 23:47:31 +07:00
|
|
|
lmb->drc_index);
|
2017-01-07 02:25:53 +07:00
|
|
|
else
|
2017-12-01 23:47:31 +07:00
|
|
|
dlpar_release_drc(lmb->drc_index);
|
|
|
|
|
|
|
|
drmem_remove_lmb_reservation(lmb);
|
2015-02-11 02:49:22 +07:00
|
|
|
}
|
2015-02-11 02:48:25 +07:00
|
|
|
rc = -EINVAL;
|
|
|
|
} else {
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb(lmb) {
|
|
|
|
if (!drmem_lmb_reserved(lmb))
|
2015-02-11 02:48:25 +07:00
|
|
|
continue;
|
|
|
|
|
|
|
|
pr_info("Memory at %llx (drc index %x) was hot-added\n",
|
2017-12-01 23:47:31 +07:00
|
|
|
lmb->base_addr, lmb->drc_index);
|
|
|
|
drmem_remove_lmb_reservation(lmb);
|
2015-02-11 02:48:25 +07:00
|
|
|
}
|
2017-08-24 00:18:43 +07:00
|
|
|
rc = 0;
|
2015-02-11 02:48:25 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_memory_add_by_index(u32 drc_index)
|
2015-02-11 02:48:25 +07:00
|
|
|
{
|
2017-12-01 23:47:31 +07:00
|
|
|
struct drmem_lmb *lmb;
|
|
|
|
int rc, lmb_found;
|
2015-02-11 02:48:25 +07:00
|
|
|
|
|
|
|
pr_info("Attempting to hot-add LMB, drc index %x\n", drc_index);
|
|
|
|
|
|
|
|
lmb_found = 0;
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb(lmb) {
|
|
|
|
if (lmb->drc_index == drc_index) {
|
2015-02-11 02:48:25 +07:00
|
|
|
lmb_found = 1;
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_acquire_drc(lmb->drc_index);
|
2017-01-07 02:25:53 +07:00
|
|
|
if (!rc) {
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_add_lmb(lmb);
|
2017-01-07 02:25:53 +07:00
|
|
|
if (rc)
|
2017-12-01 23:47:31 +07:00
|
|
|
dlpar_release_drc(lmb->drc_index);
|
2017-01-07 02:25:53 +07:00
|
|
|
}
|
|
|
|
|
2015-02-11 02:48:25 +07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!lmb_found)
|
|
|
|
rc = -EINVAL;
|
|
|
|
|
|
|
|
if (rc)
|
|
|
|
pr_info("Failed to hot-add memory, drc index %x\n", drc_index);
|
|
|
|
else
|
|
|
|
pr_info("Memory at %llx (drc index %x) was hot-added\n",
|
2017-12-01 23:47:31 +07:00
|
|
|
lmb->base_addr, drc_index);
|
2015-02-11 02:48:25 +07:00
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
static int dlpar_memory_add_by_ic(u32 lmbs_to_add, u32 drc_index)
|
2017-02-16 01:45:56 +07:00
|
|
|
{
|
2017-12-01 23:47:31 +07:00
|
|
|
struct drmem_lmb *lmb, *start_lmb, *end_lmb;
|
|
|
|
int lmbs_available = 0;
|
|
|
|
int rc;
|
2017-02-16 01:45:56 +07:00
|
|
|
|
|
|
|
pr_info("Attempting to hot-add %u LMB(s) at index %x\n",
|
|
|
|
lmbs_to_add, drc_index);
|
|
|
|
|
|
|
|
if (lmbs_to_add == 0)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = get_lmb_range(drc_index, lmbs_to_add, &start_lmb, &end_lmb);
|
|
|
|
if (rc)
|
2017-02-16 01:45:56 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* Validate that the LMBs in this range are not reserved */
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb_in_range(lmb, start_lmb, end_lmb) {
|
|
|
|
if (lmb->flags & DRCONF_MEM_RESERVED)
|
2017-02-16 01:45:56 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
lmbs_available++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lmbs_available < lmbs_to_add)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb_in_range(lmb, start_lmb, end_lmb) {
|
|
|
|
if (lmb->flags & DRCONF_MEM_ASSIGNED)
|
2017-02-16 01:45:56 +07:00
|
|
|
continue;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_acquire_drc(lmb->drc_index);
|
2017-02-16 01:45:56 +07:00
|
|
|
if (rc)
|
|
|
|
break;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_add_lmb(lmb);
|
2017-02-16 01:45:56 +07:00
|
|
|
if (rc) {
|
2017-12-01 23:47:31 +07:00
|
|
|
dlpar_release_drc(lmb->drc_index);
|
2017-02-16 01:45:56 +07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
drmem_mark_lmb_reserved(lmb);
|
2017-02-16 01:45:56 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (rc) {
|
|
|
|
pr_err("Memory indexed-count-add failed, removing any added LMBs\n");
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb_in_range(lmb, start_lmb, end_lmb) {
|
|
|
|
if (!drmem_lmb_reserved(lmb))
|
2017-02-16 01:45:56 +07:00
|
|
|
continue;
|
|
|
|
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_remove_lmb(lmb);
|
2017-02-16 01:45:56 +07:00
|
|
|
if (rc)
|
|
|
|
pr_err("Failed to remove LMB, drc index %x\n",
|
2017-12-01 23:47:31 +07:00
|
|
|
lmb->drc_index);
|
2017-02-16 01:45:56 +07:00
|
|
|
else
|
2017-12-01 23:47:31 +07:00
|
|
|
dlpar_release_drc(lmb->drc_index);
|
|
|
|
|
|
|
|
drmem_remove_lmb_reservation(lmb);
|
2017-02-16 01:45:56 +07:00
|
|
|
}
|
|
|
|
rc = -EINVAL;
|
|
|
|
} else {
|
2017-12-01 23:47:31 +07:00
|
|
|
for_each_drmem_lmb_in_range(lmb, start_lmb, end_lmb) {
|
|
|
|
if (!drmem_lmb_reserved(lmb))
|
2017-02-16 01:45:56 +07:00
|
|
|
continue;
|
|
|
|
|
|
|
|
pr_info("Memory at %llx (drc index %x) was hot-added\n",
|
2017-12-01 23:47:31 +07:00
|
|
|
lmb->base_addr, lmb->drc_index);
|
|
|
|
drmem_remove_lmb_reservation(lmb);
|
2017-02-16 01:45:56 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
powerpc/pseries: Create new device hotplug entry point
The current hotplug (or dlpar) of devices (the process is generally the
same for memory, cpu, and pci) on PowerVM systems is initiated
from the HMC, which communicates the request to the partitions through
the RSCT framework. The RSCT framework then invokes the drmgr command.
The drmgr command performs the hotplug operation by doing some pieces,
such as most of the rtas calls and device tree parsing, in userspace
and make requests to the kernel to online/offline the device, update the
device tree and add/remove the device.
For PowerKVM the approach for device hotplug is to follow what is currently
being done for pci hotplug. A hotplug request is initiated from the host.
QEMU then generates an EPOW interrupt to the guest which causes the guest
to make the rtas,check-exception call. In QEMU, the rtas,check-exception call
returns a rtas hotplug event to the guest.
Please note that the current pci hotplug path for PowerKVM involves the
kernel receiving the rtas hotplug event, passing it to rtas_errd in
userspace, and having rtas_errd invoke drmgr. The drmgr command then
handles the request as described above for PowerVM systems.
There is no need for this circuitous route, we should just handle the entire
hotplug of devices in the kernel. What I am planning is to enable this
by moving the code to handle hotplug from drmgr into the kernel to
provide a single path for handling device hotplug for both PowerVM and
PowerKVM systems. This patch provides the common iframework and entry point.
For PowerKVM a future update to the kernel rtas code will recognize rtas
hotplug events returned from rtas,check-exception calls and use the common
entry point to handle hotplug of the device.
For PowerVM systems, This patch creates /sys/kernel/dlpar that can be
used by the drmgr command to initiate hotplug requests. In order to do
this a string of the format "<resource> <action> <id_type> <id>" is
written to this file. The string consists of a resource (cpu, memory, pci,
phb), an action (add or remove), an id_type (count, drc index, drc name),
and the corresponding id. The kernel will parse the string and create a
rtas hotplug section that can be passed to the common entry point for
handling hotplug requests.
It should be noted that there is no chance of updating how we receive
hotplug (dlpar) requests from the HMC on PowerVM systems.
Signed-off-by: Nathan Fontenot <nfont@linux.vnet.ibm.com>
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
2015-02-11 02:47:02 +07:00
|
|
|
int dlpar_memory(struct pseries_hp_errorlog *hp_elog)
|
|
|
|
{
|
2015-02-11 02:48:25 +07:00
|
|
|
u32 count, drc_index;
|
|
|
|
int rc;
|
|
|
|
|
powerpc/pseries: Create new device hotplug entry point
The current hotplug (or dlpar) of devices (the process is generally the
same for memory, cpu, and pci) on PowerVM systems is initiated
from the HMC, which communicates the request to the partitions through
the RSCT framework. The RSCT framework then invokes the drmgr command.
The drmgr command performs the hotplug operation by doing some pieces,
such as most of the rtas calls and device tree parsing, in userspace
and make requests to the kernel to online/offline the device, update the
device tree and add/remove the device.
For PowerKVM the approach for device hotplug is to follow what is currently
being done for pci hotplug. A hotplug request is initiated from the host.
QEMU then generates an EPOW interrupt to the guest which causes the guest
to make the rtas,check-exception call. In QEMU, the rtas,check-exception call
returns a rtas hotplug event to the guest.
Please note that the current pci hotplug path for PowerKVM involves the
kernel receiving the rtas hotplug event, passing it to rtas_errd in
userspace, and having rtas_errd invoke drmgr. The drmgr command then
handles the request as described above for PowerVM systems.
There is no need for this circuitous route, we should just handle the entire
hotplug of devices in the kernel. What I am planning is to enable this
by moving the code to handle hotplug from drmgr into the kernel to
provide a single path for handling device hotplug for both PowerVM and
PowerKVM systems. This patch provides the common iframework and entry point.
For PowerKVM a future update to the kernel rtas code will recognize rtas
hotplug events returned from rtas,check-exception calls and use the common
entry point to handle hotplug of the device.
For PowerVM systems, This patch creates /sys/kernel/dlpar that can be
used by the drmgr command to initiate hotplug requests. In order to do
this a string of the format "<resource> <action> <id_type> <id>" is
written to this file. The string consists of a resource (cpu, memory, pci,
phb), an action (add or remove), an id_type (count, drc index, drc name),
and the corresponding id. The kernel will parse the string and create a
rtas hotplug section that can be passed to the common entry point for
handling hotplug requests.
It should be noted that there is no chance of updating how we receive
hotplug (dlpar) requests from the HMC on PowerVM systems.
Signed-off-by: Nathan Fontenot <nfont@linux.vnet.ibm.com>
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
2015-02-11 02:47:02 +07:00
|
|
|
lock_device_hotplug();
|
|
|
|
|
|
|
|
switch (hp_elog->action) {
|
2015-02-11 02:48:25 +07:00
|
|
|
case PSERIES_HP_ELOG_ACTION_ADD:
|
2017-02-16 01:45:56 +07:00
|
|
|
if (hp_elog->id_type == PSERIES_HP_ELOG_ID_DRC_COUNT) {
|
|
|
|
count = hp_elog->_drc_u.drc_count;
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_memory_add_by_count(count);
|
2017-02-16 01:45:56 +07:00
|
|
|
} else if (hp_elog->id_type == PSERIES_HP_ELOG_ID_DRC_INDEX) {
|
|
|
|
drc_index = hp_elog->_drc_u.drc_index;
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_memory_add_by_index(drc_index);
|
2017-02-16 01:45:56 +07:00
|
|
|
} else if (hp_elog->id_type == PSERIES_HP_ELOG_ID_DRC_IC) {
|
|
|
|
count = hp_elog->_drc_u.ic.count;
|
|
|
|
drc_index = hp_elog->_drc_u.ic.index;
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_memory_add_by_ic(count, drc_index);
|
2017-02-16 01:45:56 +07:00
|
|
|
} else {
|
2015-02-11 02:48:25 +07:00
|
|
|
rc = -EINVAL;
|
2017-02-16 01:45:56 +07:00
|
|
|
}
|
|
|
|
|
2015-02-11 02:48:25 +07:00
|
|
|
break;
|
2015-02-11 02:49:22 +07:00
|
|
|
case PSERIES_HP_ELOG_ACTION_REMOVE:
|
2017-02-16 01:45:56 +07:00
|
|
|
if (hp_elog->id_type == PSERIES_HP_ELOG_ID_DRC_COUNT) {
|
|
|
|
count = hp_elog->_drc_u.drc_count;
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_memory_remove_by_count(count);
|
2017-02-16 01:45:56 +07:00
|
|
|
} else if (hp_elog->id_type == PSERIES_HP_ELOG_ID_DRC_INDEX) {
|
|
|
|
drc_index = hp_elog->_drc_u.drc_index;
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_memory_remove_by_index(drc_index);
|
2017-02-16 01:46:18 +07:00
|
|
|
} else if (hp_elog->id_type == PSERIES_HP_ELOG_ID_DRC_IC) {
|
|
|
|
count = hp_elog->_drc_u.ic.count;
|
|
|
|
drc_index = hp_elog->_drc_u.ic.index;
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_memory_remove_by_ic(count, drc_index);
|
2017-02-16 01:45:56 +07:00
|
|
|
} else {
|
2015-02-11 02:49:22 +07:00
|
|
|
rc = -EINVAL;
|
2017-02-16 01:45:56 +07:00
|
|
|
}
|
|
|
|
|
2015-02-11 02:49:22 +07:00
|
|
|
break;
|
2017-01-07 02:27:26 +07:00
|
|
|
case PSERIES_HP_ELOG_ACTION_READD:
|
2017-02-16 01:45:56 +07:00
|
|
|
drc_index = hp_elog->_drc_u.drc_index;
|
2017-12-01 23:47:31 +07:00
|
|
|
rc = dlpar_memory_readd_by_index(drc_index);
|
2017-01-07 02:27:26 +07:00
|
|
|
break;
|
powerpc/pseries: Create new device hotplug entry point
The current hotplug (or dlpar) of devices (the process is generally the
same for memory, cpu, and pci) on PowerVM systems is initiated
from the HMC, which communicates the request to the partitions through
the RSCT framework. The RSCT framework then invokes the drmgr command.
The drmgr command performs the hotplug operation by doing some pieces,
such as most of the rtas calls and device tree parsing, in userspace
and make requests to the kernel to online/offline the device, update the
device tree and add/remove the device.
For PowerKVM the approach for device hotplug is to follow what is currently
being done for pci hotplug. A hotplug request is initiated from the host.
QEMU then generates an EPOW interrupt to the guest which causes the guest
to make the rtas,check-exception call. In QEMU, the rtas,check-exception call
returns a rtas hotplug event to the guest.
Please note that the current pci hotplug path for PowerKVM involves the
kernel receiving the rtas hotplug event, passing it to rtas_errd in
userspace, and having rtas_errd invoke drmgr. The drmgr command then
handles the request as described above for PowerVM systems.
There is no need for this circuitous route, we should just handle the entire
hotplug of devices in the kernel. What I am planning is to enable this
by moving the code to handle hotplug from drmgr into the kernel to
provide a single path for handling device hotplug for both PowerVM and
PowerKVM systems. This patch provides the common iframework and entry point.
For PowerKVM a future update to the kernel rtas code will recognize rtas
hotplug events returned from rtas,check-exception calls and use the common
entry point to handle hotplug of the device.
For PowerVM systems, This patch creates /sys/kernel/dlpar that can be
used by the drmgr command to initiate hotplug requests. In order to do
this a string of the format "<resource> <action> <id_type> <id>" is
written to this file. The string consists of a resource (cpu, memory, pci,
phb), an action (add or remove), an id_type (count, drc index, drc name),
and the corresponding id. The kernel will parse the string and create a
rtas hotplug section that can be passed to the common entry point for
handling hotplug requests.
It should be noted that there is no chance of updating how we receive
hotplug (dlpar) requests from the HMC on PowerVM systems.
Signed-off-by: Nathan Fontenot <nfont@linux.vnet.ibm.com>
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
2015-02-11 02:47:02 +07:00
|
|
|
default:
|
|
|
|
pr_err("Invalid action (%d) specified\n", hp_elog->action);
|
|
|
|
rc = -EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-04-21 03:29:48 +07:00
|
|
|
if (!rc) {
|
|
|
|
rtas_hp_event = true;
|
|
|
|
rc = drmem_update_dt();
|
|
|
|
rtas_hp_event = false;
|
|
|
|
}
|
|
|
|
|
powerpc/pseries: Create new device hotplug entry point
The current hotplug (or dlpar) of devices (the process is generally the
same for memory, cpu, and pci) on PowerVM systems is initiated
from the HMC, which communicates the request to the partitions through
the RSCT framework. The RSCT framework then invokes the drmgr command.
The drmgr command performs the hotplug operation by doing some pieces,
such as most of the rtas calls and device tree parsing, in userspace
and make requests to the kernel to online/offline the device, update the
device tree and add/remove the device.
For PowerKVM the approach for device hotplug is to follow what is currently
being done for pci hotplug. A hotplug request is initiated from the host.
QEMU then generates an EPOW interrupt to the guest which causes the guest
to make the rtas,check-exception call. In QEMU, the rtas,check-exception call
returns a rtas hotplug event to the guest.
Please note that the current pci hotplug path for PowerKVM involves the
kernel receiving the rtas hotplug event, passing it to rtas_errd in
userspace, and having rtas_errd invoke drmgr. The drmgr command then
handles the request as described above for PowerVM systems.
There is no need for this circuitous route, we should just handle the entire
hotplug of devices in the kernel. What I am planning is to enable this
by moving the code to handle hotplug from drmgr into the kernel to
provide a single path for handling device hotplug for both PowerVM and
PowerKVM systems. This patch provides the common iframework and entry point.
For PowerKVM a future update to the kernel rtas code will recognize rtas
hotplug events returned from rtas,check-exception calls and use the common
entry point to handle hotplug of the device.
For PowerVM systems, This patch creates /sys/kernel/dlpar that can be
used by the drmgr command to initiate hotplug requests. In order to do
this a string of the format "<resource> <action> <id_type> <id>" is
written to this file. The string consists of a resource (cpu, memory, pci,
phb), an action (add or remove), an id_type (count, drc index, drc name),
and the corresponding id. The kernel will parse the string and create a
rtas hotplug section that can be passed to the common entry point for
handling hotplug requests.
It should be noted that there is no chance of updating how we receive
hotplug (dlpar) requests from the HMC on PowerVM systems.
Signed-off-by: Nathan Fontenot <nfont@linux.vnet.ibm.com>
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
2015-02-11 02:47:02 +07:00
|
|
|
unlock_device_hotplug();
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2014-01-27 23:54:06 +07:00
|
|
|
static int pseries_add_mem_node(struct device_node *np)
|
2008-04-19 03:33:52 +07:00
|
|
|
{
|
2014-08-20 03:44:57 +07:00
|
|
|
const __be32 *regs;
|
2008-07-03 10:20:58 +07:00
|
|
|
unsigned long base;
|
2010-07-23 07:35:52 +07:00
|
|
|
unsigned int lmb_size;
|
2008-04-19 03:33:52 +07:00
|
|
|
int ret = -EINVAL;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check to see if we are actually adding memory
|
|
|
|
*/
|
2018-11-17 05:11:00 +07:00
|
|
|
if (!of_node_is_type(np, "memory"))
|
2008-04-19 03:33:52 +07:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
/*
|
2010-07-12 11:36:09 +07:00
|
|
|
* Find the base and size of the memblock
|
2008-04-19 03:33:52 +07:00
|
|
|
*/
|
|
|
|
regs = of_get_property(np, "reg", NULL);
|
|
|
|
if (!regs)
|
|
|
|
return ret;
|
|
|
|
|
2014-08-20 03:44:57 +07:00
|
|
|
base = be64_to_cpu(*(unsigned long *)regs);
|
|
|
|
lmb_size = be32_to_cpu(regs[3]);
|
2008-04-19 03:33:52 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Update memory region to represent the memory add
|
|
|
|
*/
|
2010-07-23 07:35:52 +07:00
|
|
|
ret = memblock_add(base, lmb_size);
|
2008-07-03 10:22:39 +07:00
|
|
|
return (ret < 0) ? -EINVAL : 0;
|
|
|
|
}
|
|
|
|
|
2014-11-25 00:58:01 +07:00
|
|
|
static int pseries_update_drconf_memory(struct of_reconfig_data *pr)
|
2008-07-03 10:22:39 +07:00
|
|
|
{
|
2017-12-01 23:47:42 +07:00
|
|
|
struct of_drconf_cell_v1 *new_drmem, *old_drmem;
|
2011-01-20 23:45:20 +07:00
|
|
|
unsigned long memblock_size;
|
2012-10-02 23:57:57 +07:00
|
|
|
u32 entries;
|
2014-08-20 03:44:57 +07:00
|
|
|
__be32 *p;
|
2012-10-02 23:57:57 +07:00
|
|
|
int i, rc = -EINVAL;
|
2008-07-03 10:22:39 +07:00
|
|
|
|
2015-02-11 02:48:25 +07:00
|
|
|
if (rtas_hp_event)
|
|
|
|
return 0;
|
|
|
|
|
2014-06-04 14:50:47 +07:00
|
|
|
memblock_size = pseries_memory_block_size();
|
2011-01-20 23:45:20 +07:00
|
|
|
if (!memblock_size)
|
2008-07-03 10:22:39 +07:00
|
|
|
return -EINVAL;
|
|
|
|
|
2014-08-20 03:44:57 +07:00
|
|
|
p = (__be32 *) pr->old_prop->value;
|
2012-10-02 23:57:57 +07:00
|
|
|
if (!p)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* The first int of the property is the number of lmb's described
|
|
|
|
* by the property. This is followed by an array of of_drconf_cell
|
2014-08-07 12:11:58 +07:00
|
|
|
* entries. Get the number of entries and skip to the array of
|
2012-10-02 23:57:57 +07:00
|
|
|
* of_drconf_cell's.
|
|
|
|
*/
|
2014-08-20 03:44:57 +07:00
|
|
|
entries = be32_to_cpu(*p++);
|
2017-12-01 23:47:42 +07:00
|
|
|
old_drmem = (struct of_drconf_cell_v1 *)p;
|
2012-10-02 23:57:57 +07:00
|
|
|
|
2014-08-20 03:44:57 +07:00
|
|
|
p = (__be32 *)pr->prop->value;
|
2012-10-02 23:57:57 +07:00
|
|
|
p++;
|
2017-12-01 23:47:42 +07:00
|
|
|
new_drmem = (struct of_drconf_cell_v1 *)p;
|
2012-10-02 23:57:57 +07:00
|
|
|
|
|
|
|
for (i = 0; i < entries; i++) {
|
2014-08-20 03:44:57 +07:00
|
|
|
if ((be32_to_cpu(old_drmem[i].flags) & DRCONF_MEM_ASSIGNED) &&
|
|
|
|
(!(be32_to_cpu(new_drmem[i].flags) & DRCONF_MEM_ASSIGNED))) {
|
|
|
|
rc = pseries_remove_memblock(
|
|
|
|
be64_to_cpu(old_drmem[i].base_addr),
|
2012-10-02 23:57:57 +07:00
|
|
|
memblock_size);
|
|
|
|
break;
|
2014-08-20 03:44:57 +07:00
|
|
|
} else if ((!(be32_to_cpu(old_drmem[i].flags) &
|
|
|
|
DRCONF_MEM_ASSIGNED)) &&
|
|
|
|
(be32_to_cpu(new_drmem[i].flags) &
|
|
|
|
DRCONF_MEM_ASSIGNED)) {
|
|
|
|
rc = memblock_add(be64_to_cpu(old_drmem[i].base_addr),
|
2012-10-02 23:57:57 +07:00
|
|
|
memblock_size);
|
|
|
|
rc = (rc < 0) ? -EINVAL : 0;
|
|
|
|
break;
|
|
|
|
}
|
2008-07-03 10:22:39 +07:00
|
|
|
}
|
|
|
|
return rc;
|
2008-04-19 03:33:52 +07:00
|
|
|
}
|
|
|
|
|
2008-04-19 03:33:50 +07:00
|
|
|
static int pseries_memory_notifier(struct notifier_block *nb,
|
2014-11-25 00:58:01 +07:00
|
|
|
unsigned long action, void *data)
|
2008-04-19 03:33:50 +07:00
|
|
|
{
|
2014-11-25 00:58:01 +07:00
|
|
|
struct of_reconfig_data *rd = data;
|
2011-06-21 10:35:56 +07:00
|
|
|
int err = 0;
|
2008-04-19 03:33:50 +07:00
|
|
|
|
|
|
|
switch (action) {
|
2012-10-02 23:57:57 +07:00
|
|
|
case OF_RECONFIG_ATTACH_NODE:
|
2014-11-25 00:58:01 +07:00
|
|
|
err = pseries_add_mem_node(rd->dn);
|
2008-04-19 03:33:50 +07:00
|
|
|
break;
|
2012-10-02 23:57:57 +07:00
|
|
|
case OF_RECONFIG_DETACH_NODE:
|
2014-11-25 00:58:01 +07:00
|
|
|
err = pseries_remove_mem_node(rd->dn);
|
2008-04-19 03:33:50 +07:00
|
|
|
break;
|
2012-10-02 23:57:57 +07:00
|
|
|
case OF_RECONFIG_UPDATE_PROPERTY:
|
2014-11-25 00:58:01 +07:00
|
|
|
if (!strcmp(rd->prop->name, "ibm,dynamic-memory"))
|
|
|
|
err = pseries_update_drconf_memory(rd);
|
2008-04-19 03:33:50 +07:00
|
|
|
break;
|
|
|
|
}
|
2011-06-21 10:35:56 +07:00
|
|
|
return notifier_from_errno(err);
|
2008-04-19 03:33:50 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static struct notifier_block pseries_mem_nb = {
|
|
|
|
.notifier_call = pseries_memory_notifier,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int __init pseries_memory_hotplug_init(void)
|
|
|
|
{
|
|
|
|
if (firmware_has_feature(FW_FEATURE_LPAR))
|
2012-10-02 23:57:57 +07:00
|
|
|
of_reconfig_notifier_register(&pseries_mem_nb);
|
2008-04-19 03:33:50 +07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
machine_device_initcall(pseries, pseries_memory_hotplug_init);
|