aboutsummaryrefslogtreecommitdiffstats
path: root/mm/memory_hotplug.c
diff options
context:
space:
mode:
authorToshi Kani <toshi.kani@hp.com>2013-09-11 17:21:49 -0400
committerLinus Torvalds <torvalds@linux-foundation.org>2013-09-11 18:57:39 -0400
commit27356f54c8c32609ff45b4ed333bb64fb2eef374 (patch)
treec17cbd22bc897bdf6644e242bbd086e4197ba3f5 /mm/memory_hotplug.c
parent15610c86fa83ff778eb80d3cfaa71d6acceb628a (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.c23
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
1073static 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 */
1074int __ref add_memory(int nid, u64 start, u64 size) 1091int __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 /*