提交 540557b9 编写于 作者: A Andy Whitcroft 提交者: Linus Torvalds

sparsemem: record when a section has a valid mem_map

We have flags to indicate whether a section actually has a valid mem_map
associated with it.  This is never set and we rely solely on the present bit
to indicate a section is valid.  By definition a section is not valid if it
has no mem_map and there is a window during init where the present bit is set
but there is no mem_map, during which pfn_valid() will return true
incorrectly.

Use the existing SECTION_HAS_MEM_MAP flag to indicate the presence of a valid
mem_map.  Switch valid_section{,_nr} and pfn_valid() to this bit.  Add a new
present_section{,_nr} and pfn_present() interfaces for those users who care to
know that a section is going to be valid.

[akpm@linux-foundation.org: coding-syle fixes]
Signed-off-by: NAndy Whitcroft <apw@shadowen.org>
Acked-by: NMel Gorman <mel@csn.ul.ie>
Cc: Christoph Lameter <clameter@sgi.com>
Cc: "Luck, Tony" <tony.luck@intel.com>
Cc: Andi Kleen <ak@suse.de>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Signed-off-by: NAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: NLinus Torvalds <torvalds@linux-foundation.org>
上级 cd881a6b
...@@ -238,7 +238,7 @@ store_mem_state(struct sys_device *dev, const char *buf, size_t count) ...@@ -238,7 +238,7 @@ store_mem_state(struct sys_device *dev, const char *buf, size_t count)
mem = container_of(dev, struct memory_block, sysdev); mem = container_of(dev, struct memory_block, sysdev);
phys_section_nr = mem->phys_index; phys_section_nr = mem->phys_index;
if (!valid_section_nr(phys_section_nr)) if (!present_section_nr(phys_section_nr))
goto out; goto out;
if (!strncmp(buf, "online", min((int)count, 6))) if (!strncmp(buf, "online", min((int)count, 6)))
...@@ -418,7 +418,7 @@ int register_new_memory(struct mem_section *section) ...@@ -418,7 +418,7 @@ int register_new_memory(struct mem_section *section)
int unregister_memory_section(struct mem_section *section) int unregister_memory_section(struct mem_section *section)
{ {
if (!valid_section(section)) if (!present_section(section))
return -EINVAL; return -EINVAL;
return remove_memory_block(0, section, 0); return remove_memory_block(0, section, 0);
...@@ -443,7 +443,7 @@ int __init memory_dev_init(void) ...@@ -443,7 +443,7 @@ int __init memory_dev_init(void)
* during boot and have been initialized * during boot and have been initialized
*/ */
for (i = 0; i < NR_MEM_SECTIONS; i++) { for (i = 0; i < NR_MEM_SECTIONS; i++) {
if (!valid_section_nr(i)) if (!present_section_nr(i))
continue; continue;
err = add_memory_block(0, __nr_to_section(i), MEM_ONLINE, 0); err = add_memory_block(0, __nr_to_section(i), MEM_ONLINE, 0);
if (!ret) if (!ret)
......
...@@ -771,12 +771,17 @@ static inline struct page *__section_mem_map_addr(struct mem_section *section) ...@@ -771,12 +771,17 @@ static inline struct page *__section_mem_map_addr(struct mem_section *section)
return (struct page *)map; return (struct page *)map;
} }
static inline int valid_section(struct mem_section *section) static inline int present_section(struct mem_section *section)
{ {
return (section && (section->section_mem_map & SECTION_MARKED_PRESENT)); return (section && (section->section_mem_map & SECTION_MARKED_PRESENT));
} }
static inline int section_has_mem_map(struct mem_section *section) static inline int present_section_nr(unsigned long nr)
{
return present_section(__nr_to_section(nr));
}
static inline int valid_section(struct mem_section *section)
{ {
return (section && (section->section_mem_map & SECTION_HAS_MEM_MAP)); return (section && (section->section_mem_map & SECTION_HAS_MEM_MAP));
} }
...@@ -798,6 +803,13 @@ static inline int pfn_valid(unsigned long pfn) ...@@ -798,6 +803,13 @@ static inline int pfn_valid(unsigned long pfn)
return valid_section(__nr_to_section(pfn_to_section_nr(pfn))); return valid_section(__nr_to_section(pfn_to_section_nr(pfn)));
} }
static inline int pfn_present(unsigned long pfn)
{
if (pfn_to_section_nr(pfn) >= NR_MEM_SECTIONS)
return 0;
return present_section(__nr_to_section(pfn_to_section_nr(pfn)));
}
/* /*
* These are _only_ used during initialisation, therefore they * These are _only_ used during initialisation, therefore they
* can use __initdata ... They could have names to indicate * can use __initdata ... They could have names to indicate
......
...@@ -176,7 +176,7 @@ unsigned long __init node_memmap_size_bytes(int nid, unsigned long start_pfn, ...@@ -176,7 +176,7 @@ unsigned long __init node_memmap_size_bytes(int nid, unsigned long start_pfn,
if (nid != early_pfn_to_nid(pfn)) if (nid != early_pfn_to_nid(pfn))
continue; continue;
if (pfn_valid(pfn)) if (pfn_present(pfn))
nr_pages += PAGES_PER_SECTION; nr_pages += PAGES_PER_SECTION;
} }
...@@ -206,11 +206,12 @@ struct page *sparse_decode_mem_map(unsigned long coded_mem_map, unsigned long pn ...@@ -206,11 +206,12 @@ struct page *sparse_decode_mem_map(unsigned long coded_mem_map, unsigned long pn
static int __meminit sparse_init_one_section(struct mem_section *ms, static int __meminit sparse_init_one_section(struct mem_section *ms,
unsigned long pnum, struct page *mem_map) unsigned long pnum, struct page *mem_map)
{ {
if (!valid_section(ms)) if (!present_section(ms))
return -EINVAL; return -EINVAL;
ms->section_mem_map &= ~SECTION_MAP_MASK; ms->section_mem_map &= ~SECTION_MAP_MASK;
ms->section_mem_map |= sparse_encode_mem_map(mem_map, pnum); ms->section_mem_map |= sparse_encode_mem_map(mem_map, pnum) |
SECTION_HAS_MEM_MAP;
return 1; return 1;
} }
...@@ -256,7 +257,7 @@ void __init sparse_init(void) ...@@ -256,7 +257,7 @@ void __init sparse_init(void)
struct page *map; struct page *map;
for (pnum = 0; pnum < NR_MEM_SECTIONS; pnum++) { for (pnum = 0; pnum < NR_MEM_SECTIONS; pnum++) {
if (!valid_section_nr(pnum)) if (!present_section_nr(pnum))
continue; continue;
map = sparse_early_mem_map_alloc(pnum); map = sparse_early_mem_map_alloc(pnum);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册