diff options
author | Toshi Kani <toshi.kani@hp.com> | 2013-09-11 17:21:49 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2013-09-11 18:57:39 -0400 |
commit | 27356f54c8c32609ff45b4ed333bb64fb2eef374 (patch) | |
tree | c17cbd22bc897bdf6644e242bbd086e4197ba3f5 /mm/memory_hotplug.c | |
parent | 15610c86fa83ff778eb80d3cfaa71d6acceb628a (diff) |
mm/hotplug: verify hotplug memory range
add_memory() and remove_memory() can only handle a memory range aligned
with section. There are problems when an unaligned range is added and
then deleted as follows:
- add_memory() with an unaligned range succeeds, but __add_pages()
called from add_memory() adds a whole section of pages even though
a given memory range is less than the section size.
- remove_memory() to the added unaligned range hits BUG_ON() in
__remove_pages().
This patch changes add_memory() and remove_memory() to check if a given
memory range is aligned with section at the beginning. As the result,
add_memory() fails with -EINVAL when a given range is unaligned, and does
not add such memory range. This prevents remove_memory() to be called
with an unaligned range as well. Note that remove_memory() has to use
BUG_ON() since this function cannot fail.
[akpm@linux-foundation.org: avoid printk warnings]
Signed-off-by: Toshi Kani <toshi.kani@hp.com>
Acked-by: KOSAKI Motohiro <kosaki.motohiro@jp.fujitsu.com>
Reviewed-by: Tang Chen <tangchen@cn.fujitsu.com>
Reviewed-by: Wanpeng Li <liwanp@linux.vnet.ibm.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'mm/memory_hotplug.c')
-rw-r--r-- | mm/memory_hotplug.c | 23 |
1 files changed, 23 insertions, 0 deletions
diff --git a/mm/memory_hotplug.c b/mm/memory_hotplug.c index 46b489cacdd8..247d66675a91 100644 --- a/mm/memory_hotplug.c +++ b/mm/memory_hotplug.c | |||
@@ -1070,6 +1070,23 @@ out: | |||
1070 | return ret; | 1070 | return ret; |
1071 | } | 1071 | } |
1072 | 1072 | ||
1073 | static int check_hotplug_memory_range(u64 start, u64 size) | ||
1074 | { | ||
1075 | u64 start_pfn = start >> PAGE_SHIFT; | ||
1076 | u64 nr_pages = size >> PAGE_SHIFT; | ||
1077 | |||
1078 | /* Memory range must be aligned with section */ | ||
1079 | if ((start_pfn & ~PAGE_SECTION_MASK) || | ||
1080 | (nr_pages % PAGES_PER_SECTION) || (!nr_pages)) { | ||
1081 | pr_err("Section-unaligned hotplug range: start 0x%llx, size 0x%llx\n", | ||
1082 | (unsigned long long)start, | ||
1083 | (unsigned long long)size); | ||
1084 | return -EINVAL; | ||
1085 | } | ||
1086 | |||
1087 | return 0; | ||
1088 | } | ||
1089 | |||
1073 | /* we are OK calling __meminit stuff here - we have CONFIG_MEMORY_HOTPLUG */ | 1090 | /* we are OK calling __meminit stuff here - we have CONFIG_MEMORY_HOTPLUG */ |
1074 | int __ref add_memory(int nid, u64 start, u64 size) | 1091 | int __ref add_memory(int nid, u64 start, u64 size) |
1075 | { | 1092 | { |
@@ -1079,6 +1096,10 @@ int __ref add_memory(int nid, u64 start, u64 size) | |||
1079 | struct resource *res; | 1096 | struct resource *res; |
1080 | int ret; | 1097 | int ret; |
1081 | 1098 | ||
1099 | ret = check_hotplug_memory_range(start, size); | ||
1100 | if (ret) | ||
1101 | return ret; | ||
1102 | |||
1082 | lock_memory_hotplug(); | 1103 | lock_memory_hotplug(); |
1083 | 1104 | ||
1084 | res = register_memory_resource(start, size); | 1105 | res = register_memory_resource(start, size); |
@@ -1786,6 +1807,8 @@ void __ref remove_memory(int nid, u64 start, u64 size) | |||
1786 | { | 1807 | { |
1787 | int ret; | 1808 | int ret; |
1788 | 1809 | ||
1810 | BUG_ON(check_hotplug_memory_range(start, size)); | ||
1811 | |||
1789 | lock_memory_hotplug(); | 1812 | lock_memory_hotplug(); |
1790 | 1813 | ||
1791 | /* | 1814 | /* |