提交 4f256a79 编写于 作者: J Jia He 提交者: Xie XiuQi

mm: page_alloc: remain memblock_next_valid_pfn() on arm/arm64

hulk inclusion
category: performance
bugzilla: 11028
CVE: NA

-------------------------------------------------

Commit b92df1de ("mm: page_alloc: skip over regions of invalid pfns
where possible") optimized the loop in memmap_init_zone(). But it causes
possible panic bug. So Daniel Vacek reverted it later.

But as suggested by Daniel Vacek, it is fine to using memblock to skip
gaps and finding next valid frame with CONFIG_HAVE_ARCH_PFN_VALID.
Daniel said:
"On arm and arm64, memblock is used by default. But generic version of
pfn_valid() is based on mem sections and memblock_next_valid_pfn() does
not always return the next valid one but skips more resulting in some
valid frames to be skipped (as if they were invalid). And that's why
kernel was eventually crashing on some !arm machines."

About the performance consideration:
As said by James in b92df1de,
"I have tested this patch on a virtual model of a Samurai CPU
with a sparse memory map.  The kernel boot time drops from 109 to
62 seconds."

Thus it would be better if we remain memblock_next_valid_pfn on arm/arm64.
Suggested-by: NDaniel Vacek <neelx@redhat.com>
Signed-off-by: NJia He <jia.he@hxt-semitech.com>
Signed-off-by: NYang Yingliang <yangyingliang@huawei.com>
Reviewed-by: Nzhong jiang <zhongjiang@huawei.com>
Signed-off-by: NYang Yingliang <yangyingliang@huawei.com>
上级 1448c5ea
...@@ -1270,6 +1270,10 @@ static inline int pfn_present(unsigned long pfn) ...@@ -1270,6 +1270,10 @@ static inline int pfn_present(unsigned long pfn)
#endif #endif
#define early_pfn_valid(pfn) pfn_valid(pfn) #define early_pfn_valid(pfn) pfn_valid(pfn)
#ifdef CONFIG_HAVE_MEMBLOCK_PFN_VALID
extern unsigned long memblock_next_valid_pfn(unsigned long pfn);
#define next_valid_pfn(pfn) memblock_next_valid_pfn(pfn)
#endif
void sparse_init(void); void sparse_init(void);
#else #else
#define sparse_init() do {} while (0) #define sparse_init() do {} while (0)
...@@ -1291,6 +1295,11 @@ struct mminit_pfnnid_cache { ...@@ -1291,6 +1295,11 @@ struct mminit_pfnnid_cache {
#define early_pfn_valid(pfn) (1) #define early_pfn_valid(pfn) (1)
#endif #endif
/* fallback to default definitions*/
#ifndef next_valid_pfn
#define next_valid_pfn(pfn) (pfn + 1)
#endif
void memory_present(int nid, unsigned long start, unsigned long end); void memory_present(int nid, unsigned long start, unsigned long end);
/* /*
......
...@@ -1232,6 +1232,36 @@ int __init_memblock memblock_set_node(phys_addr_t base, phys_addr_t size, ...@@ -1232,6 +1232,36 @@ int __init_memblock memblock_set_node(phys_addr_t base, phys_addr_t size,
} }
#endif /* CONFIG_HAVE_MEMBLOCK_NODE_MAP */ #endif /* CONFIG_HAVE_MEMBLOCK_NODE_MAP */
#ifdef CONFIG_HAVE_MEMBLOCK_PFN_VALID
unsigned long __init_memblock memblock_next_valid_pfn(unsigned long pfn)
{
struct memblock_type *type = &memblock.memory;
unsigned int right = type->cnt;
unsigned int mid, left = 0;
phys_addr_t addr = PFN_PHYS(++pfn);
do {
mid = (right + left) / 2;
if (addr < type->regions[mid].base)
right = mid;
else if (addr >= (type->regions[mid].base +
type->regions[mid].size))
left = mid + 1;
else {
/* addr is within the region, so pfn is valid */
return pfn;
}
} while (left < right);
if (right == type->cnt)
return -1UL;
else
return PHYS_PFN(type->regions[right].base);
}
EXPORT_SYMBOL(memblock_next_valid_pfn);
#endif /*CONFIG_HAVE_MEMBLOCK_PFN_VALID*/
static phys_addr_t __init memblock_alloc_range_nid(phys_addr_t size, static phys_addr_t __init memblock_alloc_range_nid(phys_addr_t size,
phys_addr_t align, phys_addr_t start, phys_addr_t align, phys_addr_t start,
phys_addr_t end, int nid, phys_addr_t end, int nid,
......
...@@ -5549,8 +5549,11 @@ void __meminit memmap_init_zone(unsigned long size, int nid, unsigned long zone, ...@@ -5549,8 +5549,11 @@ void __meminit memmap_init_zone(unsigned long size, int nid, unsigned long zone,
if (context != MEMMAP_EARLY) if (context != MEMMAP_EARLY)
goto not_early; goto not_early;
if (!early_pfn_valid(pfn)) if (!early_pfn_valid(pfn)) {
pfn = next_valid_pfn(pfn) - 1;
continue; continue;
}
if (!early_pfn_in_nid(pfn, nid)) if (!early_pfn_in_nid(pfn, nid))
continue; continue;
if (!update_defer_init(pgdat, pfn, end_pfn, &nr_initialised)) if (!update_defer_init(pgdat, pfn, end_pfn, &nr_initialised))
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册