mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2024-11-01 17:08:10 +00:00
4b09700244
Patch series "mm/memory_hotplug: "auto-movable" online policy and memory groups", v3. I. Goal The goal of this series is improving in-kernel auto-online support. It tackles the fundamental problems that: 1) We can create zone imbalances when onlining all memory blindly to ZONE_MOVABLE, in the worst case crashing the system. We have to know upfront how much memory we are going to hotplug such that we can safely enable auto-onlining of all hotplugged memory to ZONE_MOVABLE via "online_movable". This is far from practical and only applicable in limited setups -- like inside VMs under the RHV/oVirt hypervisor which will never hotplug more than 3 times the boot memory (and the limitation is only in place due to the Linux limitation). 2) We see more setups that implement dynamic VM resizing, hot(un)plugging memory to resize VM memory. In these setups, we might hotplug a lot of memory, but it might happen in various small steps in both directions (e.g., 2 GiB -> 8 GiB -> 4 GiB -> 16 GiB ...). virtio-mem is the primary driver of this upstream right now, performing such dynamic resizing NUMA-aware via multiple virtio-mem devices. Onlining all hotplugged memory to ZONE_NORMAL means we basically have no hotunplug guarantees. Onlining all to ZONE_MOVABLE means we can easily run into zone imbalances when growing a VM. We want a mixture, and we want as much memory as reasonable/configured in ZONE_MOVABLE. Details regarding zone imbalances can be found at [1]. 3) Memory devices consist of 1..X memory block devices, however, the kernel doesn't really track the relationship. Consequently, also user space has no idea. We want to make per-device decisions. As one example, for memory hotunplug it doesn't make sense to use a mixture of zones within a single DIMM: we want all MOVABLE if possible, otherwise all !MOVABLE, because any !MOVABLE part will easily block the whole DIMM from getting hotunplugged. As another example, virtio-mem operates on individual units that span 1..X memory blocks. Similar to a DIMM, we want a unit to either be all MOVABLE or !MOVABLE. A "unit" can be thought of like a DIMM, however, all units of a virtio-mem device logically belong together and are managed (added/removed) by a single driver. We want as much memory of a virtio-mem device to be MOVABLE as possible. 4) We want memory onlining to be done right from the kernel while adding memory, not triggered by user space via udev rules; for example, this is reqired for fast memory hotplug for drivers that add individual memory blocks, like virito-mem. We want a way to configure a policy in the kernel and avoid implementing advanced policies in user space. The auto-onlining support we have in the kernel is not sufficient. All we have is a) online everything MOVABLE (online_movable) b) online everything !MOVABLE (online_kernel) c) keep zones contiguous (online). This series allows configuring c) to mean instead "online movable if possible according to the coniguration, driven by a maximum MOVABLE:KERNEL ratio" -- a new onlining policy. II. Approach This series does 3 things: 1) Introduces the "auto-movable" online policy that initially operates on individual memory blocks only. It uses a maximum MOVABLE:KERNEL ratio to make a decision whether a memory block will be onlined to ZONE_MOVABLE or not. However, in the basic form, hotplugged KERNEL memory does not allow for more MOVABLE memory (details in the patches). CMA memory is treated like MOVABLE memory. 2) Introduces static (e.g., DIMM) and dynamic (e.g., virtio-mem) memory groups and uses group information to make decisions in the "auto-movable" online policy across memory blocks of a single memory device (modeled as memory group). More details can be found in patch #3 or in the DIMM example below. 3) Maximizes ZONE_MOVABLE memory within dynamic memory groups, by allowing ZONE_NORMAL memory within a dynamic memory group to allow for more ZONE_MOVABLE memory within the same memory group. The target use case is dynamic VM resizing using virtio-mem. See the virtio-mem example below. I remember that the basic idea of using a ratio to implement a policy in the kernel was once mentioned by Vitaly Kuznetsov, but I might be wrong (I lost the pointer to that discussion). For me, the main use case is using it along with virtio-mem (and DIMMs / ppc64 dlpar where necessary) for dynamic resizing of VMs, increasing the amount of memory we can hotunplug reliably again if we might eventually hotplug a lot of memory to a VM. III. Target Usage The target usage will be: 1) Linux boots with "mhp_default_online_type=offline" 2) User space (e.g., systemd unit) configures memory onlining (according to a config file and system properties), for example: * Setting memory_hotplug.online_policy=auto-movable * Setting memory_hotplug.auto_movable_ratio=301 * Setting memory_hotplug.auto_movable_numa_aware=true 3) User space enabled auto onlining via "echo online > /sys/devices/system/memory/auto_online_blocks" 4) User space triggers manual onlining of all already-offline memory blocks (go over offline memory blocks and set them to "online") IV. Example For DIMMs, hotplugging 4 GiB DIMMs to a 4 GiB VM with a configured ratio of 301% results in the following layout: Memory block 0-15: DMA32 (early) Memory block 32-47: Normal (early) Memory block 48-79: Movable (DIMM 0) Memory block 80-111: Movable (DIMM 1) Memory block 112-143: Movable (DIMM 2) Memory block 144-275: Normal (DIMM 3) Memory block 176-207: Normal (DIMM 4) ... all Normal (-> hotplugged Normal memory does not allow for more Movable memory) For virtio-mem, using a simple, single virtio-mem device with a 4 GiB VM will result in the following layout: Memory block 0-15: DMA32 (early) Memory block 32-47: Normal (early) Memory block 48-143: Movable (virtio-mem, first 12 GiB) Memory block 144: Normal (virtio-mem, next 128 MiB) Memory block 145-147: Movable (virtio-mem, next 384 MiB) Memory block 148: Normal (virtio-mem, next 128 MiB) Memory block 149-151: Movable (virtio-mem, next 384 MiB) ... Normal/Movable mixture as above (-> hotplugged Normal memory allows for more Movable memory within the same device) Which gives us maximum flexibility when dynamically growing/shrinking a VM in smaller steps. V. Doc Update I'll update the memory-hotplug.rst documentation, once the overhaul [1] is usptream. Until then, details can be found in patch #2. VI. Future Work 1) Use memory groups for ppc64 dlpar 2) Being able to specify a portion of (early) kernel memory that will be excluded from the ratio. Like "128 MiB globally/per node" are excluded. This might be helpful when starting VMs with extremely small memory footprint (e.g., 128 MiB) and hotplugging memory later -- not wanting the first hotplugged units getting onlined to ZONE_MOVABLE. One alternative would be a trigger to not consider ZONE_DMA memory in the ratio. We'll have to see if this is really rrequired. 3) Indicate to user space that MOVABLE might be a bad idea -- especially relevant when memory ballooning without support for balloon compaction is active. This patch (of 9): For implementing a new memory onlining policy, which determines when to online memory blocks to ZONE_MOVABLE semi-automatically, we need the number of present early (boot) pages -- present pages excluding hotplugged pages. Let's track these pages per zone. Pass a page instead of the zone to adjust_present_page_count(), similar as adjust_managed_page_count() and derive the zone from the page. It's worth noting that a memory block to be offlined/onlined is either completely "early" or "not early". add_memory() and friends can only add complete memory blocks and we only online/offline complete (individual) memory blocks. Link: https://lkml.kernel.org/r/20210806124715.17090-1-david@redhat.com Link: https://lkml.kernel.org/r/20210806124715.17090-2-david@redhat.com Signed-off-by: David Hildenbrand <david@redhat.com> Cc: Vitaly Kuznetsov <vkuznets@redhat.com> Cc: "Michael S. Tsirkin" <mst@redhat.com> Cc: Jason Wang <jasowang@redhat.com> Cc: Marek Kedzierski <mkedzier@redhat.com> Cc: Hui Zhu <teawater@gmail.com> Cc: Pankaj Gupta <pankaj.gupta.linux@gmail.com> Cc: Wei Yang <richard.weiyang@linux.alibaba.com> Cc: Oscar Salvador <osalvador@suse.de> Cc: Michal Hocko <mhocko@kernel.org> Cc: Dan Williams <dan.j.williams@intel.com> Cc: Anshuman Khandual <anshuman.khandual@arm.com> Cc: Dave Hansen <dave.hansen@linux.intel.com> Cc: Vlastimil Babka <vbabka@suse.cz> Cc: Mike Rapoport <rppt@kernel.org> Cc: "Rafael J. Wysocki" <rjw@rjwysocki.net> Cc: Len Brown <lenb@kernel.org> Cc: Pavel Tatashin <pasha.tatashin@soleen.com> Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org> Cc: Rafael J. Wysocki <rafael.j.wysocki@intel.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
893 lines
22 KiB
C
893 lines
22 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Memory subsystem support
|
|
*
|
|
* Written by Matt Tolentino <matthew.e.tolentino@intel.com>
|
|
* Dave Hansen <haveblue@us.ibm.com>
|
|
*
|
|
* This file provides the necessary infrastructure to represent
|
|
* a SPARSEMEM-memory-model system's physical memory in /sysfs.
|
|
* All arch-independent code that assumes MEMORY_HOTPLUG requires
|
|
* SPARSEMEM should be contained here, or in mm/memory_hotplug.c.
|
|
*/
|
|
|
|
#include <linux/module.h>
|
|
#include <linux/init.h>
|
|
#include <linux/topology.h>
|
|
#include <linux/capability.h>
|
|
#include <linux/device.h>
|
|
#include <linux/memory.h>
|
|
#include <linux/memory_hotplug.h>
|
|
#include <linux/mm.h>
|
|
#include <linux/stat.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/xarray.h>
|
|
|
|
#include <linux/atomic.h>
|
|
#include <linux/uaccess.h>
|
|
|
|
#define MEMORY_CLASS_NAME "memory"
|
|
|
|
static const char *const online_type_to_str[] = {
|
|
[MMOP_OFFLINE] = "offline",
|
|
[MMOP_ONLINE] = "online",
|
|
[MMOP_ONLINE_KERNEL] = "online_kernel",
|
|
[MMOP_ONLINE_MOVABLE] = "online_movable",
|
|
};
|
|
|
|
int mhp_online_type_from_str(const char *str)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(online_type_to_str); i++) {
|
|
if (sysfs_streq(str, online_type_to_str[i]))
|
|
return i;
|
|
}
|
|
return -EINVAL;
|
|
}
|
|
|
|
#define to_memory_block(dev) container_of(dev, struct memory_block, dev)
|
|
|
|
static int sections_per_block;
|
|
|
|
static inline unsigned long memory_block_id(unsigned long section_nr)
|
|
{
|
|
return section_nr / sections_per_block;
|
|
}
|
|
|
|
static inline unsigned long pfn_to_block_id(unsigned long pfn)
|
|
{
|
|
return memory_block_id(pfn_to_section_nr(pfn));
|
|
}
|
|
|
|
static inline unsigned long phys_to_block_id(unsigned long phys)
|
|
{
|
|
return pfn_to_block_id(PFN_DOWN(phys));
|
|
}
|
|
|
|
static int memory_subsys_online(struct device *dev);
|
|
static int memory_subsys_offline(struct device *dev);
|
|
|
|
static struct bus_type memory_subsys = {
|
|
.name = MEMORY_CLASS_NAME,
|
|
.dev_name = MEMORY_CLASS_NAME,
|
|
.online = memory_subsys_online,
|
|
.offline = memory_subsys_offline,
|
|
};
|
|
|
|
/*
|
|
* Memory blocks are cached in a local radix tree to avoid
|
|
* a costly linear search for the corresponding device on
|
|
* the subsystem bus.
|
|
*/
|
|
static DEFINE_XARRAY(memory_blocks);
|
|
|
|
static BLOCKING_NOTIFIER_HEAD(memory_chain);
|
|
|
|
int register_memory_notifier(struct notifier_block *nb)
|
|
{
|
|
return blocking_notifier_chain_register(&memory_chain, nb);
|
|
}
|
|
EXPORT_SYMBOL(register_memory_notifier);
|
|
|
|
void unregister_memory_notifier(struct notifier_block *nb)
|
|
{
|
|
blocking_notifier_chain_unregister(&memory_chain, nb);
|
|
}
|
|
EXPORT_SYMBOL(unregister_memory_notifier);
|
|
|
|
static void memory_block_release(struct device *dev)
|
|
{
|
|
struct memory_block *mem = to_memory_block(dev);
|
|
|
|
kfree(mem);
|
|
}
|
|
|
|
unsigned long __weak memory_block_size_bytes(void)
|
|
{
|
|
return MIN_MEMORY_BLOCK_SIZE;
|
|
}
|
|
EXPORT_SYMBOL_GPL(memory_block_size_bytes);
|
|
|
|
/*
|
|
* Show the first physical section index (number) of this memory block.
|
|
*/
|
|
static ssize_t phys_index_show(struct device *dev,
|
|
struct device_attribute *attr, char *buf)
|
|
{
|
|
struct memory_block *mem = to_memory_block(dev);
|
|
unsigned long phys_index;
|
|
|
|
phys_index = mem->start_section_nr / sections_per_block;
|
|
|
|
return sysfs_emit(buf, "%08lx\n", phys_index);
|
|
}
|
|
|
|
/*
|
|
* Legacy interface that we cannot remove. Always indicate "removable"
|
|
* with CONFIG_MEMORY_HOTREMOVE - bad heuristic.
|
|
*/
|
|
static ssize_t removable_show(struct device *dev, struct device_attribute *attr,
|
|
char *buf)
|
|
{
|
|
return sysfs_emit(buf, "%d\n", (int)IS_ENABLED(CONFIG_MEMORY_HOTREMOVE));
|
|
}
|
|
|
|
/*
|
|
* online, offline, going offline, etc.
|
|
*/
|
|
static ssize_t state_show(struct device *dev, struct device_attribute *attr,
|
|
char *buf)
|
|
{
|
|
struct memory_block *mem = to_memory_block(dev);
|
|
const char *output;
|
|
|
|
/*
|
|
* We can probably put these states in a nice little array
|
|
* so that they're not open-coded
|
|
*/
|
|
switch (mem->state) {
|
|
case MEM_ONLINE:
|
|
output = "online";
|
|
break;
|
|
case MEM_OFFLINE:
|
|
output = "offline";
|
|
break;
|
|
case MEM_GOING_OFFLINE:
|
|
output = "going-offline";
|
|
break;
|
|
default:
|
|
WARN_ON(1);
|
|
return sysfs_emit(buf, "ERROR-UNKNOWN-%ld\n", mem->state);
|
|
}
|
|
|
|
return sysfs_emit(buf, "%s\n", output);
|
|
}
|
|
|
|
int memory_notify(unsigned long val, void *v)
|
|
{
|
|
return blocking_notifier_call_chain(&memory_chain, val, v);
|
|
}
|
|
|
|
static int memory_block_online(struct memory_block *mem)
|
|
{
|
|
unsigned long start_pfn = section_nr_to_pfn(mem->start_section_nr);
|
|
unsigned long nr_pages = PAGES_PER_SECTION * sections_per_block;
|
|
unsigned long nr_vmemmap_pages = mem->nr_vmemmap_pages;
|
|
struct zone *zone;
|
|
int ret;
|
|
|
|
zone = zone_for_pfn_range(mem->online_type, mem->nid, start_pfn, nr_pages);
|
|
|
|
/*
|
|
* Although vmemmap pages have a different lifecycle than the pages
|
|
* they describe (they remain until the memory is unplugged), doing
|
|
* their initialization and accounting at memory onlining/offlining
|
|
* stage helps to keep accounting easier to follow - e.g vmemmaps
|
|
* belong to the same zone as the memory they backed.
|
|
*/
|
|
if (nr_vmemmap_pages) {
|
|
ret = mhp_init_memmap_on_memory(start_pfn, nr_vmemmap_pages, zone);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
|
|
ret = online_pages(start_pfn + nr_vmemmap_pages,
|
|
nr_pages - nr_vmemmap_pages, zone);
|
|
if (ret) {
|
|
if (nr_vmemmap_pages)
|
|
mhp_deinit_memmap_on_memory(start_pfn, nr_vmemmap_pages);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Account once onlining succeeded. If the zone was unpopulated, it is
|
|
* now already properly populated.
|
|
*/
|
|
if (nr_vmemmap_pages)
|
|
adjust_present_page_count(pfn_to_page(start_pfn),
|
|
nr_vmemmap_pages);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int memory_block_offline(struct memory_block *mem)
|
|
{
|
|
unsigned long start_pfn = section_nr_to_pfn(mem->start_section_nr);
|
|
unsigned long nr_pages = PAGES_PER_SECTION * sections_per_block;
|
|
unsigned long nr_vmemmap_pages = mem->nr_vmemmap_pages;
|
|
int ret;
|
|
|
|
/*
|
|
* Unaccount before offlining, such that unpopulated zone and kthreads
|
|
* can properly be torn down in offline_pages().
|
|
*/
|
|
if (nr_vmemmap_pages)
|
|
adjust_present_page_count(pfn_to_page(start_pfn),
|
|
-nr_vmemmap_pages);
|
|
|
|
ret = offline_pages(start_pfn + nr_vmemmap_pages,
|
|
nr_pages - nr_vmemmap_pages);
|
|
if (ret) {
|
|
/* offline_pages() failed. Account back. */
|
|
if (nr_vmemmap_pages)
|
|
adjust_present_page_count(pfn_to_page(start_pfn),
|
|
nr_vmemmap_pages);
|
|
return ret;
|
|
}
|
|
|
|
if (nr_vmemmap_pages)
|
|
mhp_deinit_memmap_on_memory(start_pfn, nr_vmemmap_pages);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* MEMORY_HOTPLUG depends on SPARSEMEM in mm/Kconfig, so it is
|
|
* OK to have direct references to sparsemem variables in here.
|
|
*/
|
|
static int
|
|
memory_block_action(struct memory_block *mem, unsigned long action)
|
|
{
|
|
int ret;
|
|
|
|
switch (action) {
|
|
case MEM_ONLINE:
|
|
ret = memory_block_online(mem);
|
|
break;
|
|
case MEM_OFFLINE:
|
|
ret = memory_block_offline(mem);
|
|
break;
|
|
default:
|
|
WARN(1, KERN_WARNING "%s(%ld, %ld) unknown action: "
|
|
"%ld\n", __func__, mem->start_section_nr, action, action);
|
|
ret = -EINVAL;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int memory_block_change_state(struct memory_block *mem,
|
|
unsigned long to_state, unsigned long from_state_req)
|
|
{
|
|
int ret = 0;
|
|
|
|
if (mem->state != from_state_req)
|
|
return -EINVAL;
|
|
|
|
if (to_state == MEM_OFFLINE)
|
|
mem->state = MEM_GOING_OFFLINE;
|
|
|
|
ret = memory_block_action(mem, to_state);
|
|
mem->state = ret ? from_state_req : to_state;
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* The device lock serializes operations on memory_subsys_[online|offline] */
|
|
static int memory_subsys_online(struct device *dev)
|
|
{
|
|
struct memory_block *mem = to_memory_block(dev);
|
|
int ret;
|
|
|
|
if (mem->state == MEM_ONLINE)
|
|
return 0;
|
|
|
|
/*
|
|
* When called via device_online() without configuring the online_type,
|
|
* we want to default to MMOP_ONLINE.
|
|
*/
|
|
if (mem->online_type == MMOP_OFFLINE)
|
|
mem->online_type = MMOP_ONLINE;
|
|
|
|
ret = memory_block_change_state(mem, MEM_ONLINE, MEM_OFFLINE);
|
|
mem->online_type = MMOP_OFFLINE;
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int memory_subsys_offline(struct device *dev)
|
|
{
|
|
struct memory_block *mem = to_memory_block(dev);
|
|
|
|
if (mem->state == MEM_OFFLINE)
|
|
return 0;
|
|
|
|
return memory_block_change_state(mem, MEM_OFFLINE, MEM_ONLINE);
|
|
}
|
|
|
|
static ssize_t state_store(struct device *dev, struct device_attribute *attr,
|
|
const char *buf, size_t count)
|
|
{
|
|
const int online_type = mhp_online_type_from_str(buf);
|
|
struct memory_block *mem = to_memory_block(dev);
|
|
int ret;
|
|
|
|
if (online_type < 0)
|
|
return -EINVAL;
|
|
|
|
ret = lock_device_hotplug_sysfs();
|
|
if (ret)
|
|
return ret;
|
|
|
|
switch (online_type) {
|
|
case MMOP_ONLINE_KERNEL:
|
|
case MMOP_ONLINE_MOVABLE:
|
|
case MMOP_ONLINE:
|
|
/* mem->online_type is protected by device_hotplug_lock */
|
|
mem->online_type = online_type;
|
|
ret = device_online(&mem->dev);
|
|
break;
|
|
case MMOP_OFFLINE:
|
|
ret = device_offline(&mem->dev);
|
|
break;
|
|
default:
|
|
ret = -EINVAL; /* should never happen */
|
|
}
|
|
|
|
unlock_device_hotplug();
|
|
|
|
if (ret < 0)
|
|
return ret;
|
|
if (ret)
|
|
return -EINVAL;
|
|
|
|
return count;
|
|
}
|
|
|
|
/*
|
|
* Legacy interface that we cannot remove: s390x exposes the storage increment
|
|
* covered by a memory block, allowing for identifying which memory blocks
|
|
* comprise a storage increment. Since a memory block spans complete
|
|
* storage increments nowadays, this interface is basically unused. Other
|
|
* archs never exposed != 0.
|
|
*/
|
|
static ssize_t phys_device_show(struct device *dev,
|
|
struct device_attribute *attr, char *buf)
|
|
{
|
|
struct memory_block *mem = to_memory_block(dev);
|
|
unsigned long start_pfn = section_nr_to_pfn(mem->start_section_nr);
|
|
|
|
return sysfs_emit(buf, "%d\n",
|
|
arch_get_memory_phys_device(start_pfn));
|
|
}
|
|
|
|
#ifdef CONFIG_MEMORY_HOTREMOVE
|
|
static int print_allowed_zone(char *buf, int len, int nid,
|
|
unsigned long start_pfn, unsigned long nr_pages,
|
|
int online_type, struct zone *default_zone)
|
|
{
|
|
struct zone *zone;
|
|
|
|
zone = zone_for_pfn_range(online_type, nid, start_pfn, nr_pages);
|
|
if (zone == default_zone)
|
|
return 0;
|
|
|
|
return sysfs_emit_at(buf, len, " %s", zone->name);
|
|
}
|
|
|
|
static ssize_t valid_zones_show(struct device *dev,
|
|
struct device_attribute *attr, char *buf)
|
|
{
|
|
struct memory_block *mem = to_memory_block(dev);
|
|
unsigned long start_pfn = section_nr_to_pfn(mem->start_section_nr);
|
|
unsigned long nr_pages = PAGES_PER_SECTION * sections_per_block;
|
|
struct zone *default_zone;
|
|
int len = 0;
|
|
int nid;
|
|
|
|
/*
|
|
* Check the existing zone. Make sure that we do that only on the
|
|
* online nodes otherwise the page_zone is not reliable
|
|
*/
|
|
if (mem->state == MEM_ONLINE) {
|
|
/*
|
|
* The block contains more than one zone can not be offlined.
|
|
* This can happen e.g. for ZONE_DMA and ZONE_DMA32
|
|
*/
|
|
default_zone = test_pages_in_a_zone(start_pfn,
|
|
start_pfn + nr_pages);
|
|
if (!default_zone)
|
|
return sysfs_emit(buf, "%s\n", "none");
|
|
len += sysfs_emit_at(buf, len, "%s", default_zone->name);
|
|
goto out;
|
|
}
|
|
|
|
nid = mem->nid;
|
|
default_zone = zone_for_pfn_range(MMOP_ONLINE, nid, start_pfn,
|
|
nr_pages);
|
|
|
|
len += sysfs_emit_at(buf, len, "%s", default_zone->name);
|
|
len += print_allowed_zone(buf, len, nid, start_pfn, nr_pages,
|
|
MMOP_ONLINE_KERNEL, default_zone);
|
|
len += print_allowed_zone(buf, len, nid, start_pfn, nr_pages,
|
|
MMOP_ONLINE_MOVABLE, default_zone);
|
|
out:
|
|
len += sysfs_emit_at(buf, len, "\n");
|
|
return len;
|
|
}
|
|
static DEVICE_ATTR_RO(valid_zones);
|
|
#endif
|
|
|
|
static DEVICE_ATTR_RO(phys_index);
|
|
static DEVICE_ATTR_RW(state);
|
|
static DEVICE_ATTR_RO(phys_device);
|
|
static DEVICE_ATTR_RO(removable);
|
|
|
|
/*
|
|
* Show the memory block size (shared by all memory blocks).
|
|
*/
|
|
static ssize_t block_size_bytes_show(struct device *dev,
|
|
struct device_attribute *attr, char *buf)
|
|
{
|
|
return sysfs_emit(buf, "%lx\n", memory_block_size_bytes());
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(block_size_bytes);
|
|
|
|
/*
|
|
* Memory auto online policy.
|
|
*/
|
|
|
|
static ssize_t auto_online_blocks_show(struct device *dev,
|
|
struct device_attribute *attr, char *buf)
|
|
{
|
|
return sysfs_emit(buf, "%s\n",
|
|
online_type_to_str[mhp_default_online_type]);
|
|
}
|
|
|
|
static ssize_t auto_online_blocks_store(struct device *dev,
|
|
struct device_attribute *attr,
|
|
const char *buf, size_t count)
|
|
{
|
|
const int online_type = mhp_online_type_from_str(buf);
|
|
|
|
if (online_type < 0)
|
|
return -EINVAL;
|
|
|
|
mhp_default_online_type = online_type;
|
|
return count;
|
|
}
|
|
|
|
static DEVICE_ATTR_RW(auto_online_blocks);
|
|
|
|
/*
|
|
* Some architectures will have custom drivers to do this, and
|
|
* will not need to do it from userspace. The fake hot-add code
|
|
* as well as ppc64 will do all of their discovery in userspace
|
|
* and will require this interface.
|
|
*/
|
|
#ifdef CONFIG_ARCH_MEMORY_PROBE
|
|
static ssize_t probe_store(struct device *dev, struct device_attribute *attr,
|
|
const char *buf, size_t count)
|
|
{
|
|
u64 phys_addr;
|
|
int nid, ret;
|
|
unsigned long pages_per_block = PAGES_PER_SECTION * sections_per_block;
|
|
|
|
ret = kstrtoull(buf, 0, &phys_addr);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (phys_addr & ((pages_per_block << PAGE_SHIFT) - 1))
|
|
return -EINVAL;
|
|
|
|
ret = lock_device_hotplug_sysfs();
|
|
if (ret)
|
|
return ret;
|
|
|
|
nid = memory_add_physaddr_to_nid(phys_addr);
|
|
ret = __add_memory(nid, phys_addr,
|
|
MIN_MEMORY_BLOCK_SIZE * sections_per_block,
|
|
MHP_NONE);
|
|
|
|
if (ret)
|
|
goto out;
|
|
|
|
ret = count;
|
|
out:
|
|
unlock_device_hotplug();
|
|
return ret;
|
|
}
|
|
|
|
static DEVICE_ATTR_WO(probe);
|
|
#endif
|
|
|
|
#ifdef CONFIG_MEMORY_FAILURE
|
|
/*
|
|
* Support for offlining pages of memory
|
|
*/
|
|
|
|
/* Soft offline a page */
|
|
static ssize_t soft_offline_page_store(struct device *dev,
|
|
struct device_attribute *attr,
|
|
const char *buf, size_t count)
|
|
{
|
|
int ret;
|
|
u64 pfn;
|
|
if (!capable(CAP_SYS_ADMIN))
|
|
return -EPERM;
|
|
if (kstrtoull(buf, 0, &pfn) < 0)
|
|
return -EINVAL;
|
|
pfn >>= PAGE_SHIFT;
|
|
ret = soft_offline_page(pfn, 0);
|
|
return ret == 0 ? count : ret;
|
|
}
|
|
|
|
/* Forcibly offline a page, including killing processes. */
|
|
static ssize_t hard_offline_page_store(struct device *dev,
|
|
struct device_attribute *attr,
|
|
const char *buf, size_t count)
|
|
{
|
|
int ret;
|
|
u64 pfn;
|
|
if (!capable(CAP_SYS_ADMIN))
|
|
return -EPERM;
|
|
if (kstrtoull(buf, 0, &pfn) < 0)
|
|
return -EINVAL;
|
|
pfn >>= PAGE_SHIFT;
|
|
ret = memory_failure(pfn, 0);
|
|
return ret ? ret : count;
|
|
}
|
|
|
|
static DEVICE_ATTR_WO(soft_offline_page);
|
|
static DEVICE_ATTR_WO(hard_offline_page);
|
|
#endif
|
|
|
|
/* See phys_device_show(). */
|
|
int __weak arch_get_memory_phys_device(unsigned long start_pfn)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* A reference for the returned memory block device is acquired.
|
|
*
|
|
* Called under device_hotplug_lock.
|
|
*/
|
|
static struct memory_block *find_memory_block_by_id(unsigned long block_id)
|
|
{
|
|
struct memory_block *mem;
|
|
|
|
mem = xa_load(&memory_blocks, block_id);
|
|
if (mem)
|
|
get_device(&mem->dev);
|
|
return mem;
|
|
}
|
|
|
|
/*
|
|
* Called under device_hotplug_lock.
|
|
*/
|
|
struct memory_block *find_memory_block(struct mem_section *section)
|
|
{
|
|
unsigned long block_id = memory_block_id(__section_nr(section));
|
|
|
|
return find_memory_block_by_id(block_id);
|
|
}
|
|
|
|
static struct attribute *memory_memblk_attrs[] = {
|
|
&dev_attr_phys_index.attr,
|
|
&dev_attr_state.attr,
|
|
&dev_attr_phys_device.attr,
|
|
&dev_attr_removable.attr,
|
|
#ifdef CONFIG_MEMORY_HOTREMOVE
|
|
&dev_attr_valid_zones.attr,
|
|
#endif
|
|
NULL
|
|
};
|
|
|
|
static const struct attribute_group memory_memblk_attr_group = {
|
|
.attrs = memory_memblk_attrs,
|
|
};
|
|
|
|
static const struct attribute_group *memory_memblk_attr_groups[] = {
|
|
&memory_memblk_attr_group,
|
|
NULL,
|
|
};
|
|
|
|
/*
|
|
* register_memory - Setup a sysfs device for a memory block
|
|
*/
|
|
static
|
|
int register_memory(struct memory_block *memory)
|
|
{
|
|
int ret;
|
|
|
|
memory->dev.bus = &memory_subsys;
|
|
memory->dev.id = memory->start_section_nr / sections_per_block;
|
|
memory->dev.release = memory_block_release;
|
|
memory->dev.groups = memory_memblk_attr_groups;
|
|
memory->dev.offline = memory->state == MEM_OFFLINE;
|
|
|
|
ret = device_register(&memory->dev);
|
|
if (ret) {
|
|
put_device(&memory->dev);
|
|
return ret;
|
|
}
|
|
ret = xa_err(xa_store(&memory_blocks, memory->dev.id, memory,
|
|
GFP_KERNEL));
|
|
if (ret) {
|
|
put_device(&memory->dev);
|
|
device_unregister(&memory->dev);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static int init_memory_block(unsigned long block_id, unsigned long state,
|
|
unsigned long nr_vmemmap_pages)
|
|
{
|
|
struct memory_block *mem;
|
|
int ret = 0;
|
|
|
|
mem = find_memory_block_by_id(block_id);
|
|
if (mem) {
|
|
put_device(&mem->dev);
|
|
return -EEXIST;
|
|
}
|
|
mem = kzalloc(sizeof(*mem), GFP_KERNEL);
|
|
if (!mem)
|
|
return -ENOMEM;
|
|
|
|
mem->start_section_nr = block_id * sections_per_block;
|
|
mem->state = state;
|
|
mem->nid = NUMA_NO_NODE;
|
|
mem->nr_vmemmap_pages = nr_vmemmap_pages;
|
|
|
|
ret = register_memory(mem);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int add_memory_block(unsigned long base_section_nr)
|
|
{
|
|
int section_count = 0;
|
|
unsigned long nr;
|
|
|
|
for (nr = base_section_nr; nr < base_section_nr + sections_per_block;
|
|
nr++)
|
|
if (present_section_nr(nr))
|
|
section_count++;
|
|
|
|
if (section_count == 0)
|
|
return 0;
|
|
return init_memory_block(memory_block_id(base_section_nr),
|
|
MEM_ONLINE, 0);
|
|
}
|
|
|
|
static void unregister_memory(struct memory_block *memory)
|
|
{
|
|
if (WARN_ON_ONCE(memory->dev.bus != &memory_subsys))
|
|
return;
|
|
|
|
WARN_ON(xa_erase(&memory_blocks, memory->dev.id) == NULL);
|
|
|
|
/* drop the ref. we got via find_memory_block() */
|
|
put_device(&memory->dev);
|
|
device_unregister(&memory->dev);
|
|
}
|
|
|
|
/*
|
|
* Create memory block devices for the given memory area. Start and size
|
|
* have to be aligned to memory block granularity. Memory block devices
|
|
* will be initialized as offline.
|
|
*
|
|
* Called under device_hotplug_lock.
|
|
*/
|
|
int create_memory_block_devices(unsigned long start, unsigned long size,
|
|
unsigned long vmemmap_pages)
|
|
{
|
|
const unsigned long start_block_id = pfn_to_block_id(PFN_DOWN(start));
|
|
unsigned long end_block_id = pfn_to_block_id(PFN_DOWN(start + size));
|
|
struct memory_block *mem;
|
|
unsigned long block_id;
|
|
int ret = 0;
|
|
|
|
if (WARN_ON_ONCE(!IS_ALIGNED(start, memory_block_size_bytes()) ||
|
|
!IS_ALIGNED(size, memory_block_size_bytes())))
|
|
return -EINVAL;
|
|
|
|
for (block_id = start_block_id; block_id != end_block_id; block_id++) {
|
|
ret = init_memory_block(block_id, MEM_OFFLINE, vmemmap_pages);
|
|
if (ret)
|
|
break;
|
|
}
|
|
if (ret) {
|
|
end_block_id = block_id;
|
|
for (block_id = start_block_id; block_id != end_block_id;
|
|
block_id++) {
|
|
mem = find_memory_block_by_id(block_id);
|
|
if (WARN_ON_ONCE(!mem))
|
|
continue;
|
|
unregister_memory(mem);
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Remove memory block devices for the given memory area. Start and size
|
|
* have to be aligned to memory block granularity. Memory block devices
|
|
* have to be offline.
|
|
*
|
|
* Called under device_hotplug_lock.
|
|
*/
|
|
void remove_memory_block_devices(unsigned long start, unsigned long size)
|
|
{
|
|
const unsigned long start_block_id = pfn_to_block_id(PFN_DOWN(start));
|
|
const unsigned long end_block_id = pfn_to_block_id(PFN_DOWN(start + size));
|
|
struct memory_block *mem;
|
|
unsigned long block_id;
|
|
|
|
if (WARN_ON_ONCE(!IS_ALIGNED(start, memory_block_size_bytes()) ||
|
|
!IS_ALIGNED(size, memory_block_size_bytes())))
|
|
return;
|
|
|
|
for (block_id = start_block_id; block_id != end_block_id; block_id++) {
|
|
mem = find_memory_block_by_id(block_id);
|
|
if (WARN_ON_ONCE(!mem))
|
|
continue;
|
|
unregister_memory_block_under_nodes(mem);
|
|
unregister_memory(mem);
|
|
}
|
|
}
|
|
|
|
/* return true if the memory block is offlined, otherwise, return false */
|
|
bool is_memblock_offlined(struct memory_block *mem)
|
|
{
|
|
return mem->state == MEM_OFFLINE;
|
|
}
|
|
|
|
static struct attribute *memory_root_attrs[] = {
|
|
#ifdef CONFIG_ARCH_MEMORY_PROBE
|
|
&dev_attr_probe.attr,
|
|
#endif
|
|
|
|
#ifdef CONFIG_MEMORY_FAILURE
|
|
&dev_attr_soft_offline_page.attr,
|
|
&dev_attr_hard_offline_page.attr,
|
|
#endif
|
|
|
|
&dev_attr_block_size_bytes.attr,
|
|
&dev_attr_auto_online_blocks.attr,
|
|
NULL
|
|
};
|
|
|
|
static const struct attribute_group memory_root_attr_group = {
|
|
.attrs = memory_root_attrs,
|
|
};
|
|
|
|
static const struct attribute_group *memory_root_attr_groups[] = {
|
|
&memory_root_attr_group,
|
|
NULL,
|
|
};
|
|
|
|
/*
|
|
* Initialize the sysfs support for memory devices. At the time this function
|
|
* is called, we cannot have concurrent creation/deletion of memory block
|
|
* devices, the device_hotplug_lock is not needed.
|
|
*/
|
|
void __init memory_dev_init(void)
|
|
{
|
|
int ret;
|
|
unsigned long block_sz, nr;
|
|
|
|
/* Validate the configured memory block size */
|
|
block_sz = memory_block_size_bytes();
|
|
if (!is_power_of_2(block_sz) || block_sz < MIN_MEMORY_BLOCK_SIZE)
|
|
panic("Memory block size not suitable: 0x%lx\n", block_sz);
|
|
sections_per_block = block_sz / MIN_MEMORY_BLOCK_SIZE;
|
|
|
|
ret = subsys_system_register(&memory_subsys, memory_root_attr_groups);
|
|
if (ret)
|
|
panic("%s() failed to register subsystem: %d\n", __func__, ret);
|
|
|
|
/*
|
|
* Create entries for memory sections that were found
|
|
* during boot and have been initialized
|
|
*/
|
|
for (nr = 0; nr <= __highest_present_section_nr;
|
|
nr += sections_per_block) {
|
|
ret = add_memory_block(nr);
|
|
if (ret)
|
|
panic("%s() failed to add memory block: %d\n", __func__,
|
|
ret);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* walk_memory_blocks - walk through all present memory blocks overlapped
|
|
* by the range [start, start + size)
|
|
*
|
|
* @start: start address of the memory range
|
|
* @size: size of the memory range
|
|
* @arg: argument passed to func
|
|
* @func: callback for each memory section walked
|
|
*
|
|
* This function walks through all present memory blocks overlapped by the
|
|
* range [start, start + size), calling func on each memory block.
|
|
*
|
|
* In case func() returns an error, walking is aborted and the error is
|
|
* returned.
|
|
*
|
|
* Called under device_hotplug_lock.
|
|
*/
|
|
int walk_memory_blocks(unsigned long start, unsigned long size,
|
|
void *arg, walk_memory_blocks_func_t func)
|
|
{
|
|
const unsigned long start_block_id = phys_to_block_id(start);
|
|
const unsigned long end_block_id = phys_to_block_id(start + size - 1);
|
|
struct memory_block *mem;
|
|
unsigned long block_id;
|
|
int ret = 0;
|
|
|
|
if (!size)
|
|
return 0;
|
|
|
|
for (block_id = start_block_id; block_id <= end_block_id; block_id++) {
|
|
mem = find_memory_block_by_id(block_id);
|
|
if (!mem)
|
|
continue;
|
|
|
|
ret = func(mem, arg);
|
|
put_device(&mem->dev);
|
|
if (ret)
|
|
break;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
struct for_each_memory_block_cb_data {
|
|
walk_memory_blocks_func_t func;
|
|
void *arg;
|
|
};
|
|
|
|
static int for_each_memory_block_cb(struct device *dev, void *data)
|
|
{
|
|
struct memory_block *mem = to_memory_block(dev);
|
|
struct for_each_memory_block_cb_data *cb_data = data;
|
|
|
|
return cb_data->func(mem, cb_data->arg);
|
|
}
|
|
|
|
/**
|
|
* for_each_memory_block - walk through all present memory blocks
|
|
*
|
|
* @arg: argument passed to func
|
|
* @func: callback for each memory block walked
|
|
*
|
|
* This function walks through all present memory blocks, calling func on
|
|
* each memory block.
|
|
*
|
|
* In case func() returns an error, walking is aborted and the error is
|
|
* returned.
|
|
*/
|
|
int for_each_memory_block(void *arg, walk_memory_blocks_func_t func)
|
|
{
|
|
struct for_each_memory_block_cb_data cb_data = {
|
|
.func = func,
|
|
.arg = arg,
|
|
};
|
|
|
|
return bus_for_each_dev(&memory_subsys, NULL, &cb_data,
|
|
for_each_memory_block_cb);
|
|
}
|