2007-10-16 16:24:13 +08:00
|
|
|
/*
|
|
|
|
* Virtual Memory Map support
|
|
|
|
*
|
2008-07-05 00:59:22 +08:00
|
|
|
* (C) 2007 sgi. Christoph Lameter.
|
2007-10-16 16:24:13 +08:00
|
|
|
*
|
|
|
|
* Virtual memory maps allow VM primitives pfn_to_page, page_to_pfn,
|
|
|
|
* virt_to_page, page_address() to be implemented as a base offset
|
|
|
|
* calculation without memory access.
|
|
|
|
*
|
|
|
|
* However, virtual mappings need a page table and TLBs. Many Linux
|
|
|
|
* architectures already map their physical space using 1-1 mappings
|
tree-wide: fix comment/printk typos
"gadget", "through", "command", "maintain", "maintain", "controller", "address",
"between", "initiali[zs]e", "instead", "function", "select", "already",
"equal", "access", "management", "hierarchy", "registration", "interest",
"relative", "memory", "offset", "already",
Signed-off-by: Uwe Kleine-König <u.kleine-koenig@pengutronix.de>
Signed-off-by: Jiri Kosina <jkosina@suse.cz>
2010-11-02 03:38:34 +08:00
|
|
|
* via TLBs. For those arches the virtual memory map is essentially
|
2007-10-16 16:24:13 +08:00
|
|
|
* for free if we use the same page size as the 1-1 mappings. In that
|
|
|
|
* case the overhead consists of a few additional pages that are
|
|
|
|
* allocated to create a view of memory for vmemmap.
|
|
|
|
*
|
2007-10-16 16:24:14 +08:00
|
|
|
* The architecture is expected to provide a vmemmap_populate() function
|
|
|
|
* to instantiate the mapping.
|
2007-10-16 16:24:13 +08:00
|
|
|
*/
|
|
|
|
#include <linux/mm.h>
|
|
|
|
#include <linux/mmzone.h>
|
|
|
|
#include <linux/bootmem.h>
|
2016-01-16 08:56:22 +08:00
|
|
|
#include <linux/memremap.h>
|
2007-10-16 16:24:13 +08:00
|
|
|
#include <linux/highmem.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 16:04:11 +08:00
|
|
|
#include <linux/slab.h>
|
2007-10-16 16:24:13 +08:00
|
|
|
#include <linux/spinlock.h>
|
|
|
|
#include <linux/vmalloc.h>
|
2007-10-30 05:37:19 +08:00
|
|
|
#include <linux/sched.h>
|
2007-10-16 16:24:13 +08:00
|
|
|
#include <asm/dma.h>
|
|
|
|
#include <asm/pgalloc.h>
|
|
|
|
#include <asm/pgtable.h>
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate a block of memory to be used to back the virtual memory map
|
|
|
|
* or to back the page tables that are used to create the mapping.
|
|
|
|
* Uses the main allocators if they are available, else bootmem.
|
|
|
|
*/
|
2007-11-29 08:21:57 +08:00
|
|
|
|
|
|
|
static void * __init_refok __earlyonly_bootmem_alloc(int node,
|
|
|
|
unsigned long size,
|
|
|
|
unsigned long align,
|
|
|
|
unsigned long goal)
|
|
|
|
{
|
2014-01-22 07:50:34 +08:00
|
|
|
return memblock_virt_alloc_try_nid(size, align, goal,
|
|
|
|
BOOTMEM_ALLOC_ACCESSIBLE, node);
|
2007-11-29 08:21:57 +08:00
|
|
|
}
|
|
|
|
|
2010-02-10 17:20:22 +08:00
|
|
|
static void *vmemmap_buf;
|
|
|
|
static void *vmemmap_buf_end;
|
2007-11-29 08:21:57 +08:00
|
|
|
|
2007-10-16 16:24:13 +08:00
|
|
|
void * __meminit vmemmap_alloc_block(unsigned long size, int node)
|
|
|
|
{
|
|
|
|
/* If the main allocator is up use that, fallback to bootmem. */
|
|
|
|
if (slab_is_available()) {
|
2009-09-22 08:01:19 +08:00
|
|
|
struct page *page;
|
|
|
|
|
|
|
|
if (node_state(node, N_HIGH_MEMORY))
|
2013-04-30 06:07:49 +08:00
|
|
|
page = alloc_pages_node(
|
|
|
|
node, GFP_KERNEL | __GFP_ZERO | __GFP_REPEAT,
|
|
|
|
get_order(size));
|
2009-09-22 08:01:19 +08:00
|
|
|
else
|
2013-04-30 06:07:49 +08:00
|
|
|
page = alloc_pages(
|
|
|
|
GFP_KERNEL | __GFP_ZERO | __GFP_REPEAT,
|
2009-09-22 08:01:19 +08:00
|
|
|
get_order(size));
|
2007-10-16 16:24:13 +08:00
|
|
|
if (page)
|
|
|
|
return page_address(page);
|
|
|
|
return NULL;
|
|
|
|
} else
|
2007-11-29 08:21:57 +08:00
|
|
|
return __earlyonly_bootmem_alloc(node, size, size,
|
2007-10-16 16:24:13 +08:00
|
|
|
__pa(MAX_DMA_ADDRESS));
|
|
|
|
}
|
|
|
|
|
2010-02-10 17:20:22 +08:00
|
|
|
/* need to make sure size is all the same during early stage */
|
2016-01-16 08:56:22 +08:00
|
|
|
static void * __meminit alloc_block_buf(unsigned long size, int node)
|
2010-02-10 17:20:22 +08:00
|
|
|
{
|
|
|
|
void *ptr;
|
|
|
|
|
|
|
|
if (!vmemmap_buf)
|
|
|
|
return vmemmap_alloc_block(size, node);
|
|
|
|
|
|
|
|
/* take the from buf */
|
|
|
|
ptr = (void *)ALIGN((unsigned long)vmemmap_buf, size);
|
|
|
|
if (ptr + size > vmemmap_buf_end)
|
|
|
|
return vmemmap_alloc_block(size, node);
|
|
|
|
|
|
|
|
vmemmap_buf = ptr + size;
|
|
|
|
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
2016-01-16 08:56:22 +08:00
|
|
|
static unsigned long __meminit vmem_altmap_next_pfn(struct vmem_altmap *altmap)
|
|
|
|
{
|
|
|
|
return altmap->base_pfn + altmap->reserve + altmap->alloc
|
|
|
|
+ altmap->align;
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned long __meminit vmem_altmap_nr_free(struct vmem_altmap *altmap)
|
|
|
|
{
|
|
|
|
unsigned long allocated = altmap->alloc + altmap->align;
|
|
|
|
|
|
|
|
if (altmap->free > allocated)
|
|
|
|
return altmap->free - allocated;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* vmem_altmap_alloc - allocate pages from the vmem_altmap reservation
|
|
|
|
* @altmap - reserved page pool for the allocation
|
|
|
|
* @nr_pfns - size (in pages) of the allocation
|
|
|
|
*
|
|
|
|
* Allocations are aligned to the size of the request
|
|
|
|
*/
|
|
|
|
static unsigned long __meminit vmem_altmap_alloc(struct vmem_altmap *altmap,
|
|
|
|
unsigned long nr_pfns)
|
|
|
|
{
|
|
|
|
unsigned long pfn = vmem_altmap_next_pfn(altmap);
|
|
|
|
unsigned long nr_align;
|
|
|
|
|
|
|
|
nr_align = 1UL << find_first_bit(&nr_pfns, BITS_PER_LONG);
|
|
|
|
nr_align = ALIGN(pfn, nr_align) - pfn;
|
|
|
|
|
|
|
|
if (nr_pfns + nr_align > vmem_altmap_nr_free(altmap))
|
|
|
|
return ULONG_MAX;
|
|
|
|
altmap->alloc += nr_pfns;
|
|
|
|
altmap->align += nr_align;
|
|
|
|
return pfn + nr_align;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void * __meminit altmap_alloc_block_buf(unsigned long size,
|
|
|
|
struct vmem_altmap *altmap)
|
|
|
|
{
|
|
|
|
unsigned long pfn, nr_pfns;
|
|
|
|
void *ptr;
|
|
|
|
|
|
|
|
if (size & ~PAGE_MASK) {
|
|
|
|
pr_warn_once("%s: allocations must be multiple of PAGE_SIZE (%ld)\n",
|
|
|
|
__func__, size);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
nr_pfns = size >> PAGE_SHIFT;
|
|
|
|
pfn = vmem_altmap_alloc(altmap, nr_pfns);
|
|
|
|
if (pfn < ULONG_MAX)
|
|
|
|
ptr = __va(__pfn_to_phys(pfn));
|
|
|
|
else
|
|
|
|
ptr = NULL;
|
|
|
|
pr_debug("%s: pfn: %#lx alloc: %ld align: %ld nr: %#lx\n",
|
|
|
|
__func__, pfn, altmap->alloc, altmap->align, nr_pfns);
|
|
|
|
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* need to make sure size is all the same during early stage */
|
|
|
|
void * __meminit __vmemmap_alloc_block_buf(unsigned long size, int node,
|
|
|
|
struct vmem_altmap *altmap)
|
|
|
|
{
|
|
|
|
if (altmap)
|
|
|
|
return altmap_alloc_block_buf(size, altmap);
|
|
|
|
return alloc_block_buf(size, node);
|
|
|
|
}
|
|
|
|
|
2007-10-16 16:24:13 +08:00
|
|
|
void __meminit vmemmap_verify(pte_t *pte, int node,
|
|
|
|
unsigned long start, unsigned long end)
|
|
|
|
{
|
|
|
|
unsigned long pfn = pte_pfn(*pte);
|
|
|
|
int actual_node = early_pfn_to_nid(pfn);
|
|
|
|
|
2008-11-07 04:53:31 +08:00
|
|
|
if (node_distance(actual_node, node) > LOCAL_DISTANCE)
|
2007-10-16 16:24:13 +08:00
|
|
|
printk(KERN_WARNING "[%lx-%lx] potential offnode "
|
|
|
|
"page_structs\n", start, end - 1);
|
|
|
|
}
|
|
|
|
|
2007-10-16 16:24:14 +08:00
|
|
|
pte_t * __meminit vmemmap_pte_populate(pmd_t *pmd, unsigned long addr, int node)
|
2007-10-16 16:24:13 +08:00
|
|
|
{
|
2007-10-16 16:24:14 +08:00
|
|
|
pte_t *pte = pte_offset_kernel(pmd, addr);
|
|
|
|
if (pte_none(*pte)) {
|
|
|
|
pte_t entry;
|
2016-01-16 08:56:22 +08:00
|
|
|
void *p = alloc_block_buf(PAGE_SIZE, node);
|
2007-10-16 16:24:14 +08:00
|
|
|
if (!p)
|
2008-03-29 11:07:28 +08:00
|
|
|
return NULL;
|
2007-10-16 16:24:14 +08:00
|
|
|
entry = pfn_pte(__pa(p) >> PAGE_SHIFT, PAGE_KERNEL);
|
|
|
|
set_pte_at(&init_mm, addr, pte, entry);
|
|
|
|
}
|
|
|
|
return pte;
|
2007-10-16 16:24:13 +08:00
|
|
|
}
|
|
|
|
|
2007-10-16 16:24:14 +08:00
|
|
|
pmd_t * __meminit vmemmap_pmd_populate(pud_t *pud, unsigned long addr, int node)
|
2007-10-16 16:24:13 +08:00
|
|
|
{
|
2007-10-16 16:24:14 +08:00
|
|
|
pmd_t *pmd = pmd_offset(pud, addr);
|
|
|
|
if (pmd_none(*pmd)) {
|
|
|
|
void *p = vmemmap_alloc_block(PAGE_SIZE, node);
|
|
|
|
if (!p)
|
2008-03-29 11:07:28 +08:00
|
|
|
return NULL;
|
2007-10-16 16:24:14 +08:00
|
|
|
pmd_populate_kernel(&init_mm, pmd, p);
|
2007-10-16 16:24:13 +08:00
|
|
|
}
|
2007-10-16 16:24:14 +08:00
|
|
|
return pmd;
|
2007-10-16 16:24:13 +08:00
|
|
|
}
|
|
|
|
|
2007-10-16 16:24:14 +08:00
|
|
|
pud_t * __meminit vmemmap_pud_populate(pgd_t *pgd, unsigned long addr, int node)
|
2007-10-16 16:24:13 +08:00
|
|
|
{
|
2007-10-16 16:24:14 +08:00
|
|
|
pud_t *pud = pud_offset(pgd, addr);
|
|
|
|
if (pud_none(*pud)) {
|
|
|
|
void *p = vmemmap_alloc_block(PAGE_SIZE, node);
|
|
|
|
if (!p)
|
2008-03-29 11:07:28 +08:00
|
|
|
return NULL;
|
2007-10-16 16:24:14 +08:00
|
|
|
pud_populate(&init_mm, pud, p);
|
|
|
|
}
|
|
|
|
return pud;
|
|
|
|
}
|
2007-10-16 16:24:13 +08:00
|
|
|
|
2007-10-16 16:24:14 +08:00
|
|
|
pgd_t * __meminit vmemmap_pgd_populate(unsigned long addr, int node)
|
|
|
|
{
|
|
|
|
pgd_t *pgd = pgd_offset_k(addr);
|
|
|
|
if (pgd_none(*pgd)) {
|
|
|
|
void *p = vmemmap_alloc_block(PAGE_SIZE, node);
|
|
|
|
if (!p)
|
2008-03-29 11:07:28 +08:00
|
|
|
return NULL;
|
2007-10-16 16:24:14 +08:00
|
|
|
pgd_populate(&init_mm, pgd, p);
|
2007-10-16 16:24:13 +08:00
|
|
|
}
|
2007-10-16 16:24:14 +08:00
|
|
|
return pgd;
|
2007-10-16 16:24:13 +08:00
|
|
|
}
|
|
|
|
|
2013-04-30 06:07:50 +08:00
|
|
|
int __meminit vmemmap_populate_basepages(unsigned long start,
|
|
|
|
unsigned long end, int node)
|
2007-10-16 16:24:13 +08:00
|
|
|
{
|
2013-04-30 06:07:50 +08:00
|
|
|
unsigned long addr = start;
|
2007-10-16 16:24:14 +08:00
|
|
|
pgd_t *pgd;
|
|
|
|
pud_t *pud;
|
|
|
|
pmd_t *pmd;
|
|
|
|
pte_t *pte;
|
2007-10-16 16:24:13 +08:00
|
|
|
|
2007-10-16 16:24:14 +08:00
|
|
|
for (; addr < end; addr += PAGE_SIZE) {
|
|
|
|
pgd = vmemmap_pgd_populate(addr, node);
|
|
|
|
if (!pgd)
|
|
|
|
return -ENOMEM;
|
|
|
|
pud = vmemmap_pud_populate(pgd, addr, node);
|
|
|
|
if (!pud)
|
|
|
|
return -ENOMEM;
|
|
|
|
pmd = vmemmap_pmd_populate(pud, addr, node);
|
|
|
|
if (!pmd)
|
|
|
|
return -ENOMEM;
|
|
|
|
pte = vmemmap_pte_populate(pmd, addr, node);
|
|
|
|
if (!pte)
|
|
|
|
return -ENOMEM;
|
|
|
|
vmemmap_verify(pte, node, addr, addr + PAGE_SIZE);
|
2007-10-16 16:24:13 +08:00
|
|
|
}
|
2007-10-16 16:24:14 +08:00
|
|
|
|
|
|
|
return 0;
|
2007-10-16 16:24:13 +08:00
|
|
|
}
|
|
|
|
|
2007-10-16 16:26:14 +08:00
|
|
|
struct page * __meminit sparse_mem_map_populate(unsigned long pnum, int nid)
|
2007-10-16 16:24:13 +08:00
|
|
|
{
|
2013-04-30 06:07:50 +08:00
|
|
|
unsigned long start;
|
|
|
|
unsigned long end;
|
|
|
|
struct page *map;
|
|
|
|
|
|
|
|
map = pfn_to_page(pnum * PAGES_PER_SECTION);
|
|
|
|
start = (unsigned long)map;
|
|
|
|
end = (unsigned long)(map + PAGES_PER_SECTION);
|
|
|
|
|
|
|
|
if (vmemmap_populate(start, end, nid))
|
2007-10-16 16:24:13 +08:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return map;
|
|
|
|
}
|
2010-02-10 17:20:22 +08:00
|
|
|
|
|
|
|
void __init sparse_mem_maps_populate_node(struct page **map_map,
|
|
|
|
unsigned long pnum_begin,
|
|
|
|
unsigned long pnum_end,
|
|
|
|
unsigned long map_count, int nodeid)
|
|
|
|
{
|
|
|
|
unsigned long pnum;
|
|
|
|
unsigned long size = sizeof(struct page) * PAGES_PER_SECTION;
|
|
|
|
void *vmemmap_buf_start;
|
|
|
|
|
|
|
|
size = ALIGN(size, PMD_SIZE);
|
|
|
|
vmemmap_buf_start = __earlyonly_bootmem_alloc(nodeid, size * map_count,
|
|
|
|
PMD_SIZE, __pa(MAX_DMA_ADDRESS));
|
|
|
|
|
|
|
|
if (vmemmap_buf_start) {
|
|
|
|
vmemmap_buf = vmemmap_buf_start;
|
|
|
|
vmemmap_buf_end = vmemmap_buf_start + size * map_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (pnum = pnum_begin; pnum < pnum_end; pnum++) {
|
|
|
|
struct mem_section *ms;
|
|
|
|
|
|
|
|
if (!present_section_nr(pnum))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
map_map[pnum] = sparse_mem_map_populate(pnum, nodeid);
|
|
|
|
if (map_map[pnum])
|
|
|
|
continue;
|
|
|
|
ms = __nr_to_section(pnum);
|
|
|
|
printk(KERN_ERR "%s: sparsemem memory map backing failed "
|
|
|
|
"some memory will not be available.\n", __func__);
|
|
|
|
ms->section_mem_map = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vmemmap_buf_start) {
|
|
|
|
/* need to free left buf */
|
2014-01-22 07:50:34 +08:00
|
|
|
memblock_free_early(__pa(vmemmap_buf),
|
|
|
|
vmemmap_buf_end - vmemmap_buf);
|
2010-02-10 17:20:22 +08:00
|
|
|
vmemmap_buf = NULL;
|
|
|
|
vmemmap_buf_end = NULL;
|
|
|
|
}
|
|
|
|
}
|