Release 4.11 arch/x86/mm/init_64.c
/*
* linux/arch/x86_64/mm/init.c
*
* Copyright (C) 1995 Linus Torvalds
* Copyright (C) 2000 Pavel Machek <pavel@ucw.cz>
* Copyright (C) 2002,2003 Andi Kleen <ak@suse.de>
*/
#include <linux/signal.h>
#include <linux/sched.h>
#include <linux/kernel.h>
#include <linux/errno.h>
#include <linux/string.h>
#include <linux/types.h>
#include <linux/ptrace.h>
#include <linux/mman.h>
#include <linux/mm.h>
#include <linux/swap.h>
#include <linux/smp.h>
#include <linux/init.h>
#include <linux/initrd.h>
#include <linux/pagemap.h>
#include <linux/bootmem.h>
#include <linux/memblock.h>
#include <linux/proc_fs.h>
#include <linux/pci.h>
#include <linux/pfn.h>
#include <linux/poison.h>
#include <linux/dma-mapping.h>
#include <linux/memory.h>
#include <linux/memory_hotplug.h>
#include <linux/memremap.h>
#include <linux/nmi.h>
#include <linux/gfp.h>
#include <linux/kcore.h>
#include <asm/processor.h>
#include <asm/bios_ebda.h>
#include <linux/uaccess.h>
#include <asm/pgtable.h>
#include <asm/pgalloc.h>
#include <asm/dma.h>
#include <asm/fixmap.h>
#include <asm/e820.h>
#include <asm/apic.h>
#include <asm/tlb.h>
#include <asm/mmu_context.h>
#include <asm/proto.h>
#include <asm/smp.h>
#include <asm/sections.h>
#include <asm/kdebug.h>
#include <asm/numa.h>
#include <asm/cacheflush.h>
#include <asm/init.h>
#include <asm/uv/uv.h>
#include <asm/setup.h>
#include "mm_internal.h"
#include "ident_map.c"
/*
* NOTE: pagetable_init alloc all the fixmap pagetables contiguous on the
* physical space so we can cache the place of the first one and move
* around without checking the pgd every time.
*/
pteval_t __supported_pte_mask __read_mostly = ~0;
EXPORT_SYMBOL_GPL(__supported_pte_mask);
int force_personality32;
/*
* noexec32=on|off
* Control non executable heap for 32bit processes.
* To control the stack too use noexec=off
*
* on PROT_READ does not imply PROT_EXEC for 32-bit processes (default)
* off PROT_READ implies PROT_EXEC
*/
static int __init nonx32_setup(char *str)
{
if (!strcmp(str, "on"))
force_personality32 &= ~READ_IMPLIES_EXEC;
else if (!strcmp(str, "off"))
force_personality32 |= READ_IMPLIES_EXEC;
return 1;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Yinghai Lu | 44 | 100.00% | 1 | 100.00% |
Total | 44 | 100.00% | 1 | 100.00% |
__setup("noexec32=", nonx32_setup);
/*
* When memory was added make sure all the processes MM have
* suitable PGD entries in the local PGD level page.
*/
void sync_global_pgds(unsigned long start, unsigned long end)
{
unsigned long address;
for (address = start; address <= end; address += PGDIR_SIZE) {
const pgd_t *pgd_ref = pgd_offset_k(address);
struct page *page;
if (pgd_none(*pgd_ref))
continue;
spin_lock(&pgd_lock);
list_for_each_entry(page, &pgd_list, lru) {
pgd_t *pgd;
spinlock_t *pgt_lock;
pgd = (pgd_t *)page_address(page) + pgd_index(address);
/* the pgt_lock only for Xen */
pgt_lock = &pgd_page_get_mm(page)->page_table_lock;
spin_lock(pgt_lock);
if (!pgd_none(*pgd_ref) && !pgd_none(*pgd))
BUG_ON(pgd_page_vaddr(*pgd)
!= pgd_page_vaddr(*pgd_ref));
if (pgd_none(*pgd))
set_pgd(pgd, *pgd_ref);
spin_unlock(pgt_lock);
}
spin_unlock(&pgd_lock);
}
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Haicheng Li | 117 | 68.42% | 1 | 20.00% |
Jeremy Fitzhardinge | 24 | 14.04% | 1 | 20.00% |
Yasuaki Ishimatsu | 23 | 13.45% | 1 | 20.00% |
Linus Torvalds | 4 | 2.34% | 1 | 20.00% |
Andrea Arcangeli | 3 | 1.75% | 1 | 20.00% |
Total | 171 | 100.00% | 5 | 100.00% |
/*
* NOTE: This function is marked __ref because it calls __init function
* (alloc_bootmem_pages). It's safe to do it ONLY when after_bootmem == 0.
*/
static __ref void *spp_getpage(void)
{
void *ptr;
if (after_bootmem)
ptr = (void *) get_zeroed_page(GFP_ATOMIC | __GFP_NOTRACK);
else
ptr = alloc_bootmem_pages(PAGE_SIZE);
if (!ptr || ((unsigned long)ptr & ~PAGE_MASK)) {
panic("set_pte_phys: cannot allocate page data %s\n",
after_bootmem ? "after bootmem" : "");
}
pr_debug("spp_getpage %p\n", ptr);
return ptr;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Andi Kleen | 71 | 91.03% | 2 | 28.57% |
Thomas Gleixner | 2 | 2.56% | 1 | 14.29% |
Vegard Nossum | 2 | 2.56% | 1 | 14.29% |
Andrew Morton | 1 | 1.28% | 1 | 14.29% |
Ingo Molnar | 1 | 1.28% | 1 | 14.29% |
Marcin Ślusarz | 1 | 1.28% | 1 | 14.29% |
Total | 78 | 100.00% | 7 | 100.00% |
static pud_t *fill_pud(pgd_t *pgd, unsigned long vaddr)
{
if (pgd_none(*pgd)) {
pud_t *pud = (pud_t *)spp_getpage();
pgd_populate(&init_mm, pgd, pud);
if (pud != pud_offset(pgd, 0))
printk(KERN_ERR "PAGETABLE BUG #00! %p <-> %p\n",
pud, pud_offset(pgd, 0));
}
return pud_offset(pgd, vaddr);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tejun Heo | 68 | 85.00% | 1 | 25.00% |
Andi Kleen | 9 | 11.25% | 2 | 50.00% |
Eduardo Pereira Habkost | 3 | 3.75% | 1 | 25.00% |
Total | 80 | 100.00% | 4 | 100.00% |
static pmd_t *fill_pmd(pud_t *pud, unsigned long vaddr)
{
if (pud_none(*pud)) {
pmd_t *pmd = (pmd_t *) spp_getpage();
pud_populate(&init_mm, pud, pmd);
if (pmd != pmd_offset(pud, 0))
printk(KERN_ERR "PAGETABLE BUG #01! %p <-> %p\n",
pmd, pmd_offset(pud, 0));
}
return pmd_offset(pud, vaddr);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Andi Kleen | 59 | 73.75% | 3 | 50.00% |
Tejun Heo | 16 | 20.00% | 1 | 16.67% |
Jeremy Fitzhardinge | 4 | 5.00% | 1 | 16.67% |
Ingo Molnar | 1 | 1.25% | 1 | 16.67% |
Total | 80 | 100.00% | 6 | 100.00% |
static pte_t *fill_pte(pmd_t *pmd, unsigned long vaddr)
{
if (pmd_none(*pmd)) {
pte_t *pte = (pte_t *) spp_getpage();
pmd_populate_kernel(&init_mm, pmd, pte);
if (pte != pte_offset_kernel(pmd, 0))
printk(KERN_ERR "PAGETABLE BUG #02!\n");
}
return pte_offset_kernel(pmd, vaddr);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Andi Kleen | 40 | 56.34% | 2 | 40.00% |
Tejun Heo | 26 | 36.62% | 1 | 20.00% |
Jeremy Fitzhardinge | 4 | 5.63% | 1 | 20.00% |
Ingo Molnar | 1 | 1.41% | 1 | 20.00% |
Total | 71 | 100.00% | 5 | 100.00% |
void set_pte_vaddr_pud(pud_t *pud_page, unsigned long vaddr, pte_t new_pte)
{
pud_t *pud;
pmd_t *pmd;
pte_t *pte;
pud = pud_page + pud_index(vaddr);
pmd = fill_pmd(pud, vaddr);
pte = fill_pte(pmd, vaddr);
set_pte(pte, new_pte);
/*
* It's enough to flush this one mapping.
* (PGE mappings get flushed as well)
*/
__flush_tlb_one(vaddr);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tejun Heo | 44 | 64.71% | 1 | 33.33% |
Andi Kleen | 24 | 35.29% | 2 | 66.67% |
Total | 68 | 100.00% | 3 | 100.00% |
void set_pte_vaddr(unsigned long vaddr, pte_t pteval)
{
pgd_t *pgd;
pud_t *pud_page;
pr_debug("set_pte_vaddr %lx to %lx\n", vaddr, native_pte_val(pteval));
pgd = pgd_offset_k(vaddr);
if (pgd_none(*pgd)) {
printk(KERN_ERR
"PGD FIXMAP MISSING, it should be setup in head.S!\n");
return;
}
pud_page = (pud_t*)pgd_page_vaddr(*pgd);
set_pte_vaddr_pud(pud_page, vaddr, pteval);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Eduardo Pereira Habkost | 76 | 98.70% | 1 | 50.00% |
Tejun Heo | 1 | 1.30% | 1 | 50.00% |
Total | 77 | 100.00% | 2 | 100.00% |
pmd_t * __init populate_extra_pmd(unsigned long vaddr)
{
pgd_t *pgd;
pud_t *pud;
pgd = pgd_offset_k(vaddr);
pud = fill_pud(pgd, vaddr);
return fill_pmd(pud, vaddr);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tejun Heo | 26 | 60.47% | 2 | 66.67% |
Jack Steiner | 17 | 39.53% | 1 | 33.33% |
Total | 43 | 100.00% | 3 | 100.00% |
pte_t * __init populate_extra_pte(unsigned long vaddr)
{
pmd_t *pmd;
pmd = populate_extra_pmd(vaddr);
return fill_pte(pmd, vaddr);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tejun Heo | 30 | 100.00% | 2 | 100.00% |
Total | 30 | 100.00% | 2 | 100.00% |
/*
* Create large page table mappings for a range of physical addresses.
*/
static void __init __init_extra_mapping(unsigned long phys, unsigned long size,
enum page_cache_mode cache)
{
pgd_t *pgd;
pud_t *pud;
pmd_t *pmd;
pgprot_t prot;
pgprot_val(prot) = pgprot_val(PAGE_KERNEL_LARGE) |
pgprot_val(pgprot_4k_2_large(cachemode2pgprot(cache)));
BUG_ON((phys & ~PMD_MASK) || (size & ~PMD_MASK));
for (; size; phys += PMD_SIZE, size -= PMD_SIZE) {
pgd = pgd_offset_k((unsigned long)__va(phys));
if (pgd_none(*pgd)) {
pud = (pud_t *) spp_getpage();
set_pgd(pgd, __pgd(__pa(pud) | _KERNPG_TABLE |
_PAGE_USER));
}
pud = pud_offset(pgd, (unsigned long)__va(phys));
if (pud_none(*pud)) {
pmd = (pmd_t *) spp_getpage();
set_pud(pud, __pud(__pa(pmd) | _KERNPG_TABLE |
_PAGE_USER));
}
pmd = pmd_offset(pud, phys);
BUG_ON(!pmd_none(*pmd));
set_pmd(pmd, __pmd(phys | pgprot_val(prot)));
}
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jack Steiner | 196 | 87.89% | 1 | 50.00% |
Juergen Gross | 27 | 12.11% | 1 | 50.00% |
Total | 223 | 100.00% | 2 | 100.00% |
void __init init_extra_mapping_wb(unsigned long phys, unsigned long size)
{
__init_extra_mapping(phys, size, _PAGE_CACHE_MODE_WB);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jack Steiner | 22 | 95.65% | 1 | 50.00% |
Juergen Gross | 1 | 4.35% | 1 | 50.00% |
Total | 23 | 100.00% | 2 | 100.00% |
void __init init_extra_mapping_uc(unsigned long phys, unsigned long size)
{
__init_extra_mapping(phys, size, _PAGE_CACHE_MODE_UC);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jack Steiner | 22 | 95.65% | 1 | 50.00% |
Juergen Gross | 1 | 4.35% | 1 | 50.00% |
Total | 23 | 100.00% | 2 | 100.00% |
/*
* The head.S code sets up the kernel high mapping:
*
* from __START_KERNEL_map to __START_KERNEL_map + size (== _end-_text)
*
* phys_base holds the negative offset to the kernel, which is added
* to the compile time generated pmds. This results in invalid pmds up
* to the point where we hit the physaddr 0 mapping.
*
* We limit the mappings to the region from _text to _brk_end. _brk_end
* is rounded up to the 2MB boundary. This catches the invalid pmds as
* well, as they are located before _text:
*/
void __init cleanup_highmap(void)
{
unsigned long vaddr = __START_KERNEL_map;
unsigned long vaddr_end = __START_KERNEL_map + KERNEL_IMAGE_SIZE;
unsigned long end = roundup((unsigned long)_brk_end, PMD_SIZE) - 1;
pmd_t *pmd = level2_kernel_pgt;
/*
* Native path, max_pfn_mapped is not set yet.
* Xen has valid max_pfn_mapped set in
* arch/x86/xen/mmu.c:xen_setup_kernel_pagetable().
*/
if (max_pfn_mapped)
vaddr_end = __START_KERNEL_map + (max_pfn_mapped << PAGE_SHIFT);
for (; vaddr + PMD_SIZE - 1 < vaddr_end; pmd++, vaddr += PMD_SIZE) {
if (pmd_none(*pmd))
continue;
if (vaddr < (unsigned long) _text || vaddr > end)
set_pmd(pmd, __pmd(0));
}
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Thomas Gleixner | 81 | 71.68% | 1 | 20.00% |
Yinghai Lu | 30 | 26.55% | 2 | 40.00% |
Joerg Roedel | 1 | 0.88% | 1 | 20.00% |
Hugh Dickins | 1 | 0.88% | 1 | 20.00% |
Total | 113 | 100.00% | 5 | 100.00% |
/*
* Create PTE level page table mapping for physical addresses.
* It returns the last physical address mapped.
*/
static unsigned long __meminit
phys_pte_init(pte_t *pte_page, unsigned long paddr, unsigned long paddr_end,
pgprot_t prot)
{
unsigned long pages = 0, paddr_next;
unsigned long paddr_last = paddr_end;
pte_t *pte;
int i;
pte = pte_page + pte_index(paddr);
i = pte_index(paddr);
for (; i < PTRS_PER_PTE; i++, paddr = paddr_next, pte++) {
paddr_next = (paddr & PAGE_MASK) + PAGE_SIZE;
if (paddr >= paddr_end) {
if (!after_bootmem &&
!e820_any_mapped(paddr & PAGE_MASK, paddr_next,
E820_RAM) &&
!e820_any_mapped(paddr & PAGE_MASK, paddr_next,
E820_RESERVED_KERN))
set_pte(pte, __pte(0));
continue;
}
/*
* We will re-use the existing mapping.
* Xen for example has some special requirements, like mapping
* pagetable pages as RO. So assume someone who pre-setup
* these mappings are more intelligent.
*/
if (!pte_none(*pte)) {
if (!after_bootmem)
pages++;
continue;
}
if (0)
pr_info(" pte=%p addr=%lx pte=%016lx\n", pte, paddr,
pfn_pte(paddr >> PAGE_SHIFT, PAGE_KERNEL).pte);
pages++;
set_pte(pte, pfn_pte(paddr >> PAGE_SHIFT, prot));
paddr_last = (paddr & PAGE_MASK) + PAGE_SIZE;
}
update_page_count(PG_LEVEL_4K, pages);
return paddr_last;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jeremy Fitzhardinge | 117 | 53.42% | 1 | 10.00% |
Yinghai Lu | 53 | 24.20% | 3 | 30.00% |
Thomas Garnier | 33 | 15.07% | 1 | 10.00% |
Suresh B. Siddha | 8 | 3.65% | 2 | 20.00% |
Jan Beulich | 5 | 2.28% | 1 | 10.00% |
Dave Hansen | 2 | 0.91% | 1 | 10.00% |
Andi Kleen | 1 | 0.46% | 1 | 10.00% |
Total | 219 | 100.00% | 10 | 100.00% |
/*
* Create PMD level page table mapping for physical addresses. The virtual
* and physical address have to be aligned at this level.
* It returns the last physical address mapped.
*/
static unsigned long __meminit
phys_pmd_init(pmd_t *pmd_page, unsigned long paddr, unsigned long paddr_end,
unsigned long page_size_mask, pgprot_t prot)
{
unsigned long pages = 0, paddr_next;
unsigned long paddr_last = paddr_end;
int i = pmd_index(paddr);
for (; i < PTRS_PER_PMD; i++, paddr = paddr_next) {
pmd_t *pmd = pmd_page + pmd_index(paddr);
pte_t *pte;
pgprot_t new_prot = prot;
paddr_next = (paddr & PMD_MASK) + PMD_SIZE;
if (paddr >= paddr_end) {
if (!after_bootmem &&
!e820_any_mapped(paddr & PMD_MASK, paddr_next,
E820_RAM) &&
!e820_any_mapped(paddr & PMD_MASK, paddr_next,
E820_RESERVED_KERN))
set_pmd(pmd, __pmd(0));
continue;
}
if (!pmd_none(*pmd)) {
if (!pmd_large(*pmd)) {
spin_lock(&init_mm.page_table_lock);
pte = (pte_t *)pmd_page_vaddr(*pmd);
paddr_last = phys_pte_init(pte, paddr,
paddr_end, prot);
spin_unlock(&init_mm.page_table_lock);
continue;
}
/*
* If we are ok with PG_LEVEL_2M mapping, then we will
* use the existing mapping,
*
* Otherwise, we will split the large page mapping but
* use the same existing protection bits except for
* large page, so that we don't violate Intel's TLB
* Application note (317080) which says, while changing
* the page sizes, new and old translations should
* not differ with respect to page frame and
* attributes.
*/
if (page_size_mask & (1 << PG_LEVEL_2M)) {
if (!after_bootmem)
pages++;
paddr_last = paddr_next;
continue;
}
new_prot = pte_pgprot(pte_clrhuge(*(pte_t *)pmd));
}
if (page_size_mask & (1<<PG_LEVEL_2M)) {
pages++;
spin_lock(&init_mm.page_table_lock);
set_pte((pte_t *)pmd,
pfn_pte((paddr & PMD_MASK) >> PAGE_SHIFT,
__pgprot(pgprot_val(prot) | _PAGE_PSE)));
spin_unlock(&init_mm.page_table_lock);
paddr_last = paddr_next;
continue;
}
pte = alloc_low_page();
paddr_last = phys_pte_init(pte, paddr, paddr_end, new_prot);
spin_lock(&init_mm.page_table_lock);
pmd_populate_kernel(&init_mm, pmd, pte);
spin_unlock(&init_mm.page_table_lock);
}
update_page_count(PG_LEVEL_2M, pages);
return paddr_last;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Yinghai Lu | 73 | 19.84% | 7 | 26.92% |
Andi Kleen | 72 | 19.57% | 6 | 23.08% |
Jan Beulich | 67 | 18.21% | 4 | 15.38% |
Jeremy Fitzhardinge | 48 | 13.04% | 2 | 7.69% |
Suresh B. Siddha | 44 | 11.96% | 1 | 3.85% |
Thomas Garnier | 29 | 7.88% | 1 | 3.85% |
Keith Mannthey | 24 | 6.52% | 1 | 3.85% |
Matt Tolentino | 5 | 1.36% | 1 | 3.85% |
Hugh Dickins | 3 | 0.82% | 1 | 3.85% |
Dave Hansen | 2 | 0.54% | 1 | 3.85% |
Thomas Gleixner | 1 | 0.27% | 1 | 3.85% |
Total | 368 | 100.00% | 26 | 100.00% |
/*
* Create PUD level page table mapping for physical addresses. The virtual
* and physical address do not have to be aligned at this level. KASLR can
* randomize virtual addresses up to this level.
* It returns the last physical address mapped.
*/
static unsigned long __meminit
phys_pud_init(pud_t *pud_page, unsigned long paddr, unsigned long paddr_end,
unsigned long page_size_mask)
{
unsigned long pages = 0, paddr_next;
unsigned long paddr_last = paddr_end;
unsigned long vaddr = (unsigned long)__va(paddr);
int i = pud_index(vaddr);
for (; i < PTRS_PER_PUD; i++, paddr = paddr_next) {
pud_t *pud;
pmd_t *pmd;
pgprot_t prot = PAGE_KERNEL;
vaddr = (unsigned long)__va(paddr);
pud = pud_page + pud_index(vaddr);
paddr_next = (paddr & PUD_MASK) + PUD_SIZE;
if (paddr >= paddr_end) {
if (!after_bootmem &&
!e820_any_mapped(paddr & PUD_MASK, paddr_next,
E820_RAM) &&
!e820_any_mapped(paddr & PUD_MASK, paddr_next,
E820_RESERVED_KERN))
set_pud(pud, __pud(0));
continue;
}
if (!pud_none(*pud)) {
if (!pud_large(*pud)) {
pmd = pmd_offset(pud, 0);
paddr_last = phys_pmd_init(pmd, paddr,
paddr_end,
page_size_mask,
prot);
__flush_tlb_all();
continue;
}
/*
* If we are ok with PG_LEVEL_1G mapping, then we will
* use the existing mapping.
*
* Otherwise, we will split the gbpage mapping but use
* the same existing protection bits except for large
* page, so that we don't violate Intel's TLB
* Application note (317080) which says, while changing
* the page sizes, new and old translations should
* not differ with respect to page frame and
* attributes.
*/
if (page_size_mask & (1 << PG_LEVEL_1G)) {
if (!after_bootmem)
pages++;
paddr_last = paddr_next;
continue;
}
prot = pte_pgprot(pte_clrhuge(*(pte_t *)pud));
}
if (page_size_mask & (1<<PG_LEVEL_1G)) {
pages++;
spin_lock(&init_mm.page_table_lock);
set_pte((pte_t *)pud,
pfn_pte((paddr & PUD_MASK) >> PAGE_SHIFT,
PAGE_KERNEL_LARGE));
spin_unlock(&init_mm.page_table_lock);
paddr_last = paddr_next;
continue;
}
pmd = alloc_low_page();
paddr_last = phys_pmd_init(pmd, paddr, paddr_end,
page_size_mask, prot);
spin_lock(&init_mm.page_table_lock);
pud_populate(&init_mm, pud, pmd);
spin_unlock(&init_mm.page_table_lock);
}
__flush_tlb_all();
update_page_count(PG_LEVEL_1G, pages);
return paddr_last;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Matt Tolentino | 78 | 20.86% | 1 | 4.17% |
Andi Kleen | 62 | 16.58% | 5 | 20.83% |
Thomas Garnier | 60 | 16.04% | 2 | 8.33% |
Yinghai Lu | 55 | 14.71% | 6 | 25.00% |
Jan Beulich | 42 | 11.23% | 3 | 12.50% |
Suresh B. Siddha | 38 | 10.16% | 2 | 8.33% |
Keith Mannthey | 24 | 6.42% | 1 | 4.17% |
Jeremy Fitzhardinge | 12 | 3.21% | 2 | 8.33% |
Dave Hansen | 2 | 0.53% | 1 | 4.17% |
Arjan van de Ven | 1 | 0.27% | 1 | 4.17% |
Total | 374 | 100.00% | 24 | 100.00% |
/*
* Create page table mapping for the physical memory for specific physical
* addresses. The virtual and physical addresses have to be aligned on PMD level
* down. It returns the last physical address mapped.
*/
unsigned long __meminit
kernel_physical_mapping_init(unsigned long paddr_start,
unsigned long paddr_end,
unsigned long page_size_mask)
{
bool pgd_changed = false;
unsigned long vaddr, vaddr_start, vaddr_end, vaddr_next, paddr_last;
paddr_last = paddr_end;
vaddr = (unsigned long)__va(paddr_start);
vaddr_end = (unsigned long)__va(paddr_end);
vaddr_start = vaddr;
for (; vaddr < vaddr_end; vaddr = vaddr_next) {
pgd_t *pgd = pgd_offset_k(vaddr);
pud_t *pud;
vaddr_next = (vaddr & PGDIR_MASK) + PGDIR_SIZE;
if (pgd_val(*pgd)) {
pud = (pud_t *)pgd_page_vaddr(*pgd);
paddr_last = phys_pud_init(pud, __pa(vaddr),
__pa(vaddr_end),
page_size_mask);
continue;
}
pud = alloc_low_page();
paddr_last = phys_pud_init(pud, __pa(vaddr), __pa(vaddr_end),
page_size_mask);
spin_lock(&init_mm.page_table_lock);
pgd_populate(&init_mm, pgd, pud);
spin_unlock(&init_mm.page_table_lock);
pgd_changed = true;
}
if (pgd_changed)
sync_global_pgds(vaddr_start, vaddr_end - 1);
__flush_tlb_all();
return paddr_last;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Pekka J Enberg | 39 | 17.97% | 1 | 3.85% |
Andi Kleen | 37 | 17.05% | 8 | 30.77% |
Yinghai Lu | 37 | 17.05% | 7 | 26.92% |
Thomas Garnier | 36 | 16.59% | 1 | 3.85% |
Haicheng Li | 19 | 8.76% | 1 | 3.85% |
Jeremy Fitzhardinge | 19 | 8.76% | 2 | 7.69% |
Matt Tolentino | 15 | 6.91% | 1 | 3.85% |
Jan Beulich | 8 | 3.69% | 1 | 3.85% |
Fengguang Wu | 2 | 0.92% | 1 | 3.85% |
Suresh B. Siddha | 2 | 0.92% | 1 | 3.85% |
Wen Congyang | 2 | 0.92% | 1 | 3.85% |
David Shaohua Li | 1 | 0.46% | 1 | 3.85% |
Total | 217 | 100.00% | 26 | 100.00% |
#ifndef CONFIG_NUMA
void __init initmem_init(void)
{
memblock_set_node(0, (phys_addr_t)ULLONG_MAX, &memblock.memory, 0);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Yinghai Lu | 13 | 52.00% | 1 | 25.00% |
Tejun Heo | 7 | 28.00% | 2 | 50.00% |
Tang Chen | 5 | 20.00% | 1 | 25.00% |
Total | 25 | 100.00% | 4 | 100.00% |
#endif
void __init paging_init(void)
{
sparse_memory_present_with_active_regions(MAX_NUMNODES);
sparse_init();
/*
* clear the default setting with node 0
* note: don't use nodes_clear here, that is really clearing when
* numa support is not compiled in, and later node_set_state
* will not set it back.
*/
node_clear_state(0, N_MEMORY);
if (N_MEMORY != N_NORMAL_MEMORY)
node_clear_state(0, N_NORMAL_MEMORY);
zone_sizes_init();
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Lai Jiangshan | 13 | 32.50% | 1 | 12.50% |
Andi Kleen | 9 | 22.50% | 2 | 25.00% |
Yinghai Lu | 8 | 20.00% | 2 | 25.00% |
Pekka J Enberg | 7 | 17.50% | 2 | 25.00% |
Matt Tolentino | 3 | 7.50% | 1 | 12.50% |
Total | 40 | 100.00% | 8 | 100.00% |
/*
* Memory hotplug specific functions
*/
#ifdef CONFIG_MEMORY_HOTPLUG
/*
* After memory hotplug the variables max_pfn, max_low_pfn and high_memory need
* updating.
*/
static void update_end_of_memory_vars(u64 start, u64 size)
{
unsigned long end_pfn = PFN_UP(start + size);
if (end_pfn > max_pfn) {
max_pfn = end_pfn;
max_low_pfn = end_pfn;
high_memory = (void *)__va(max_pfn * PAGE_SIZE - 1) + 1;
}
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Shaohui Zheng | 56 | 100.00% | 1 | 100.00% |
Total | 56 | 100.00% | 1 | 100.00% |
/*
* Memory is added always to NORMAL zone. This means you will never get
* additional DMA/DMA32 memory.
*/
int arch_add_memory(int nid, u64 start, u64 size, bool for_device)
{
struct pglist_data *pgdat = NODE_DATA(nid);
struct zone *zone = pgdat->node_zones +
zone_for_memory(nid, start, size, ZONE_NORMAL, for_device);
unsigned long start_pfn = start >> PAGE_SHIFT;
unsigned long nr_pages = size >> PAGE_SHIFT;
int ret;
init_memory_mapping(start, start + size);
ret = __add_pages(nid, zone, start_pfn, nr_pages);
WARN_ON_ONCE(ret);
/* update max_pfn, max_low_pfn and high_memory */
update_end_of_memory_vars(start, size);
return ret;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Matt Tolentino | 65 | 61.32% | 1 | 11.11% |
Wang Nan | 9 | 8.49% | 1 | 11.11% |
Keith Mannthey | 9 | 8.49% | 1 | 11.11% |
Shaohui Zheng | 8 | 7.55% | 1 | 11.11% |
Yasunori Goto | 5 | 4.72% | 1 | 11.11% |
Dan J Williams | 5 | 4.72% | 1 | 11.11% |
Gary Hade | 4 | 3.77% | 2 | 22.22% |
Christoph Lameter | 1 | 0.94% | 1 | 11.11% |
Total | 106 | 100.00% | 9 | 100.00% |
EXPORT_SYMBOL_GPL(arch_add_memory);
#define PAGE_INUSE 0xFD
static void __meminit free_pagetable(struct page *page, int order)
{
unsigned long magic;
unsigned int nr_pages = 1 << order;
struct vmem_altmap *altmap = to_vmem_altmap((unsigned long) page);
if (altmap) {
vmem_altmap_free(altmap, nr_pages);
return;
}
/* bootmem page has reserved flag */
if (PageReserved(page)) {
__ClearPageReserved(page);
magic = (unsigned long)page->freelist;
if (magic == SECTION_INFO || magic == MIX_SECTION_INFO) {
while (nr_pages--)
put_page_bootmem(page++);
} else
while (nr_pages--)
free_reserved_page(page++);
} else
free_pages((unsigned long)page_address(page), order);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Wen Congyang | 94 | 72.31% | 1 | 25.00% |
Dan J Williams | 28 | 21.54% | 1 | 25.00% |
Jiang Liu | 7 | 5.38% | 1 | 25.00% |
Yasuaki Ishimatsu | 1 | 0.77% | 1 | 25.00% |
Total | 130 | 100.00% | 4 | 100.00% |
static void __meminit free_pte_table(pte_t *pte_start, pmd_t *pmd)
{
pte_t *pte;
int i;
for (i = 0; i < PTRS_PER_PTE; i++) {
pte = pte_start + i;
if (!pte_none(*pte))
return;
}
/* free a pte talbe */
free_pagetable(pmd_page(*pmd), 0);
spin_lock(&init_mm.page_table_lock);
pmd_clear(pmd);
spin_unlock(&init_mm.page_table_lock);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Wen Congyang | 84 | 97.67% | 1 | 50.00% |
Dave Hansen | 2 | 2.33% | 1 | 50.00% |
Total | 86 | 100.00% | 2 | 100.00% |
static void __meminit free_pmd_table(pmd_t *pmd_start, pud_t *pud)
{
pmd_t *pmd;
int i;
for (i = 0; i < PTRS_PER_PMD; i++) {
pmd = pmd_start + i;
if (!pmd_none(*pmd))
return;
}
/* free a pmd talbe */
free_pagetable(pud_page(*pud), 0);
spin_lock(&init_mm.page_table_lock);
pud_clear(pud);
spin_unlock(&init_mm.page_table_lock);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Wen Congyang | 84 | 97.67% | 1 | 50.00% |
Dave Hansen | 2 | 2.33% | 1 | 50.00% |
Total | 86 | 100.00% | 2 | 100.00% |
static void __meminit
remove_pte_table(pte_t *pte_start, unsigned long addr, unsigned long end,
bool direct)
{
unsigned long next, pages = 0;
pte_t *pte;
void *page_addr;
phys_addr_t phys_addr;
pte = pte_start + pte_index(addr);
for (; addr < end; addr = next, pte++) {
next = (addr + PAGE_SIZE) & PAGE_MASK;
if (next > end)
next = end;
if (!pte_present(*pte))
continue;
/*
* We mapped [0,1G) memory as identity mapping when
* initializing, in arch/x86/kernel/head_64.S. These
* pagetables cannot be removed.
*/
phys_addr = pte_val(*pte) + (addr & PAGE_MASK);
if (phys_addr < (phys_addr_t)0x40000000)
return;
if (PAGE_ALIGNED(addr) && PAGE_ALIGNED(next)) {
/*
* Do not free direct mapping pages since they were
* freed when offlining, or simplely not in use.
*/
if (!direct)
free_pagetable(pte_page(*pte), 0);
spin_lock(&init_mm.page_table_lock);
pte_clear(&init_mm, addr, pte);
spin_unlock(&init_mm.page_table_lock);
/* For non-direct mapping, pages means nothing. */
pages++;
} else {
/*
* If we are here, we are freeing vmemmap pages since
* direct mapped memory ranges to be freed are aligned.
*
* If we are not removing the whole page, it means
* other page structs in this page are being used and
* we canot remove them. So fill the unused page_structs
* with 0xFD, and remove the page when it is wholly
* filled with 0xFD.
*/
memset((void *)addr, PAGE_INUSE, next - addr);
page_addr = page_address(pte_page(*pte));
if (!memchr_inv(page_addr, PAGE_INUSE, PAGE_SIZE)) {
free_pagetable(pte_page(*pte), 0);
spin_lock(&init_mm.page_table_lock);
pte_clear(&init_mm, addr, pte);
spin_unlock(&init_mm.page_table_lock);
}
}
}
/* Call free_pte_table() in remove_pmd_table(). */
flush_tlb_all();
if (direct)
update_page_count(PG_LEVEL_4K, -pages);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Wen Congyang | 277 | 99.28% | 1 | 50.00% |
Kefeng Wang | 2 | 0.72% | 1 | 50.00% |
Total | 279 | 100.00% | 2 | 100.00% |
static void __meminit
remove_pmd_table(pmd_t *pmd_start, unsigned long addr, unsigned long end,
bool direct)
{
unsigned long next, pages = 0;
pte_t *pte_base;
pmd_t *pmd;
void *page_addr;
pmd = pmd_start + pmd_index(addr);
for (; addr < end; addr = next, pmd++) {
next = pmd_addr_end(addr, end);
if (!pmd_present(*pmd))
continue;
if (pmd_large(*pmd)) {
if (IS_ALIGNED(addr, PMD_SIZE) &&
IS_ALIGNED(next, PMD_SIZE)) {
if (!direct)
free_pagetable(pmd_page(*pmd),
get_order(PMD_SIZE));
spin_lock(&init_mm.page_table_lock);
pmd_clear(pmd);
spin_unlock(&init_mm.page_table_lock);
pages++;
} else {
/* If here, we are freeing vmemmap pages. */
memset((void *)addr, PAGE_INUSE, next - addr);
page_addr = page_address(pmd_page(*pmd));
if (!memchr_inv(page_addr, PAGE_INUSE,
PMD_SIZE)) {
free_pagetable(pmd_page(*pmd),
get_order(PMD_SIZE));
spin_lock(&init_mm.page_table_lock);
pmd_clear(pmd);
spin_unlock(&init_mm.page_table_lock);
}
}
continue;
}
pte_base = (pte_t *)pmd_page_vaddr(*pmd);
remove_pte_table(pte_base, addr, next, direct);
free_pte_table(pte_base, pmd);
}
/* Call free_pmd_table() in remove_pud_table(). */
if (direct)
update_page_count(PG_LEVEL_2M, -pages);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Wen Congyang | 280 | 100.00% | 1 | 100.00% |
Total | 280 | 100.00% | 1 | 100.00% |
static void __meminit
remove_pud_table(pud_t *pud_start, unsigned long addr, unsigned long end,
bool direct)
{
unsigned long next, pages = 0;
pmd_t *pmd_base;
pud_t *pud;
void *page_addr;
pud = pud_start + pud_index(addr);
for (; addr < end; addr = next, pud++) {
next = pud_addr_end(addr, end);
if (!pud_present(*pud))
continue;
if (pud_large(*pud)) {
if (IS_ALIGNED(addr, PUD_SIZE) &&
IS_ALIGNED(next, PUD_SIZE)) {
if (!direct)
free_pagetable(pud_page(*pud),
get_order(PUD_SIZE));
spin_lock(&init_mm.page_table_lock);
pud_clear(pud);
spin_unlock(&init_mm.page_table_lock);
pages++;
} else {
/* If here, we are freeing vmemmap pages. */
memset((void *)addr, PAGE_INUSE, next - addr);
page_addr = page_address(pud_page(*pud));
if (!memchr_inv(page_addr, PAGE_INUSE,
PUD_SIZE)) {
free_pagetable(pud_page(*pud),
get_order(PUD_SIZE));
spin_lock(&init_mm.page_table_lock);
pud_clear(pud);
spin_unlock(&init_mm.page_table_lock);
}
}
continue;
}
pmd_base = (pmd_t *)pud_page_vaddr(*pud);
remove_pmd_table(pmd_base, addr, next, direct);
free_pmd_table(pmd_base, pud);
}
if (direct)
update_page_count(PG_LEVEL_1G, -pages);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Wen Congyang | 279 | 100.00% | 1 | 100.00% |
Total | 279 | 100.00% | 1 | 100.00% |
/* start and end are both virtual address. */
static void __meminit
remove_pagetable(unsigned long start, unsigned long end, bool direct)
{
unsigned long next;
unsigned long addr;
pgd_t *pgd;
pud_t *pud;
for (addr = start; addr < end; addr = next) {
next = pgd_addr_end(addr, end);
pgd = pgd_offset_k(addr);
if (!pgd_present(*pgd))
continue;
pud = (pud_t *)pgd_page_vaddr(*pgd);
remove_pud_table(pud, addr, next, direct);
}
flush_tlb_all();
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Wen Congyang | 90 | 88.24% | 1 | 50.00% |
Yasuaki Ishimatsu | 12 | 11.76% | 1 | 50.00% |
Total | 102 | 100.00% | 2 | 100.00% |
void __ref vmemmap_free(unsigned long start, unsigned long end)
{
remove_pagetable(start, end, false);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tang Chen | 20 | 86.96% | 1 | 50.00% |
Johannes Weiner | 3 | 13.04% | 1 | 50.00% |
Total | 23 | 100.00% | 2 | 100.00% |
#ifdef CONFIG_MEMORY_HOTREMOVE
static void __meminit
kernel_physical_mapping_remove(unsigned long start, unsigned long end)
{
start = (unsigned long)__va(start);
end = (unsigned long)__va(end);
remove_pagetable(start, end, true);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Tang Chen | 46 | 100.00% | 1 | 100.00% |
Total | 46 | 100.00% | 1 | 100.00% |
int __ref arch_remove_memory(u64 start, u64 size)
{
unsigned long start_pfn = start >> PAGE_SHIFT;
unsigned long nr_pages = size >> PAGE_SHIFT;
struct page *page = pfn_to_page(start_pfn);
struct vmem_altmap *altmap;
struct zone *zone;
int ret;
/* With altmap the first mapped page is offset from @start */
altmap = to_vmem_altmap((unsigned long) page);
if (altmap)
page += vmem_altmap_offset(altmap);
zone = page_zone(page);
ret = __remove_pages(zone, start_pfn, nr_pages);
WARN_ON_ONCE(ret);
kernel_physical_mapping_remove(start, start + size);
return ret;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Wen Congyang | 60 | 55.05% | 1 | 33.33% |
Dan J Williams | 46 | 42.20% | 1 | 33.33% |
Tang Chen | 3 | 2.75% | 1 | 33.33% |
Total | 109 | 100.00% | 3 | 100.00% |
#endif
#endif /* CONFIG_MEMORY_HOTPLUG */
static struct kcore_list kcore_vsyscall;
static void __init register_page_bootmem_info(void)
{
#ifdef CONFIG_NUMA
int i;
for_each_online_node(i)
register_page_bootmem_info_node(NODE_DATA(i));
#endif
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Yinghai Lu | 29 | 100.00% | 1 | 100.00% |
Total | 29 | 100.00% | 1 | 100.00% |
void __init mem_init(void)
{
pci_iommu_alloc();
/* clear_bss() already clear the empty_zero_page */
register_page_bootmem_info();
/* this will put all memory onto the freelists */
free_all_bootmem();
after_bootmem = 1;
/* Register memory areas for /proc/kcore */
kclist_add(&kcore_vsyscall, (void *)VSYSCALL_ADDR,
PAGE_SIZE, KCORE_OTHER);
mem_init_print_info(NULL);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Andi Kleen | 33 | 73.33% | 3 | 27.27% |
Jiang Liu | 3 | 6.67% | 2 | 18.18% |
Kamezawa Hiroyuki | 2 | 4.44% | 1 | 9.09% |
Muli Ben-Yehuda | 2 | 4.44% | 1 | 9.09% |
Yinghai Lu | 2 | 4.44% | 2 | 18.18% |
Andrew Lutomirski | 2 | 4.44% | 1 | 9.09% |
Jon Mason | 1 | 2.22% | 1 | 9.09% |
Total | 45 | 100.00% | 11 | 100.00% |
int kernel_set_to_readonly;
void set_kernel_text_rw(void)
{
unsigned long start = PFN_ALIGN(_text);
unsigned long end = PFN_ALIGN(__stop___ex_table);
if (!kernel_set_to_readonly)
return;
pr_debug("Set kernel text: %lx - %lx for read write\n",
start, end);
/*
* Make the kernel identity mapping for text RW. Kernel text
* mapping will always be RO. Refer to the comment in
* static_protections() in pageattr.c
*/
set_memory_rw(start, (end - start) >> PAGE_SHIFT);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Steven Rostedt | 51 | 94.44% | 1 | 33.33% |
Suresh B. Siddha | 3 | 5.56% | 2 | 66.67% |
Total | 54 | 100.00% | 3 | 100.00% |
void set_kernel_text_ro(void)
{
unsigned long start = PFN_ALIGN(_text);
unsigned long end = PFN_ALIGN(__stop___ex_table);
if (!kernel_set_to_readonly)
return;
pr_debug("Set kernel text: %lx - %lx for read only\n",
start, end);
/*
* Set the kernel identity mapping for text RO.
*/
set_memory_ro(start, (end - start) >> PAGE_SHIFT);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Steven Rostedt | 51 | 94.44% | 1 | 33.33% |
Suresh B. Siddha | 3 | 5.56% | 2 | 66.67% |
Total | 54 | 100.00% | 3 | 100.00% |
void mark_rodata_ro(void)
{
unsigned long start = PFN_ALIGN(_text);
unsigned long rodata_start = PFN_ALIGN(__start_rodata);
unsigned long end = (unsigned long) &__end_rodata_hpage_align;
unsigned long text_end = PFN_ALIGN(&__stop___ex_table);
unsigned long rodata_end = PFN_ALIGN(&__end_rodata);
unsigned long all_end;
printk(KERN_INFO "Write protecting the kernel read-only data: %luk\n",
(end - start) >> 10);
set_memory_ro(start, (end - start) >> PAGE_SHIFT);
kernel_set_to_readonly = 1;
/*
* The rodata/data/bss/brk section (but not the kernel text!)
* should also be not-executable.
*
* We align all_end to PMD_SIZE because the existing mapping
* is a full PMD. If we would align _brk_end to PAGE_SIZE we
* split the PMD and the reminder between _brk_end and the end
* of the PMD will remain mapped executable.
*
* Any PMD which was setup after the one which covers _brk_end
* has been zapped already via cleanup_highmem().
*/
all_end = roundup((unsigned long)_brk_end, PMD_SIZE);
set_memory_nx(text_end, (all_end - text_end) >> PAGE_SHIFT);
#ifdef CONFIG_CPA_DEBUG
printk(KERN_INFO "Testing CPA: undo %lx-%lx\n", start, end);
set_memory_rw(start, (end-start) >> PAGE_SHIFT);
printk(KERN_INFO "Testing CPA: again\n");
set_memory_ro(start, (end-start) >> PAGE_SHIFT);
#endif
free_init_pages("unused kernel",
(unsigned long) __va(__pa_symbol(text_end)),
(unsigned long) __va(__pa_symbol(rodata_start)));
free_init_pages("unused kernel",
(unsigned long) __va(__pa_symbol(rodata_end)),
(unsigned long) __va(__pa_symbol(_sdata)));
debug_checkwx();
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Suresh B. Siddha | 80 | 35.40% | 2 | 11.11% |
Andi Kleen | 43 | 19.03% | 1 | 5.56% |
Arjan van de Ven | 38 | 16.81% | 3 | 16.67% |
Alexander Duyck | 14 | 6.19% | 1 | 5.56% |
Kees Cook | 14 | 6.19% | 1 | 5.56% |
Steven Rostedt | 10 | 4.42% | 2 | 11.11% |
Linus Torvalds | 8 | 3.54% | 1 | 5.56% |
Stephen D. Smalley | 5 | 2.21% | 2 | 11.11% |
Mathieu Desnoyers | 4 | 1.77% | 1 | 5.56% |
Jan Beulich | 4 | 1.77% | 1 | 5.56% |
Jiang Liu | 2 | 0.88% | 1 | 5.56% |
Yinghai Lu | 2 | 0.88% | 1 | 5.56% |
Ingo Molnar | 2 | 0.88% | 1 | 5.56% |
Total | 226 | 100.00% | 18 | 100.00% |
int kern_addr_valid(unsigned long addr)
{
unsigned long above = ((long)addr) >> __VIRTUAL_MASK_SHIFT;
pgd_t *pgd;
pud_t *pud;
pmd_t *pmd;
pte_t *pte;
if (above != 0 && above != -1UL)
return 0;
pgd = pgd_offset_k(addr);
if (pgd_none(*pgd))
return 0;
pud = pud_offset(pgd, addr);
if (pud_none(*pud))
return 0;
if (pud_large(*pud))
return pfn_valid(pud_pfn(*pud));
pmd = pmd_offset(pud, addr);
if (pmd_none(*pmd))
return 0;
if (pmd_large(*pmd))
return pfn_valid(pmd_pfn(*pmd));
pte = pte_offset_kernel(pmd, addr);
if (pte_none(*pte))
return 0;
return pfn_valid(pte_pfn(*pte));
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Andi Kleen | 158 | 89.77% | 3 | 75.00% |
Mel Gorman | 18 | 10.23% | 1 | 25.00% |
Total | 176 | 100.00% | 4 | 100.00% |
static unsigned long probe_memory_block_size(void)
{
unsigned long bz = MIN_MEMORY_BLOCK_SIZE;
/* if system is UV or has 64GB of RAM or more, use large blocks */
if (is_uv_system() || ((max_pfn << PAGE_SHIFT) >= (64UL << 30)))
bz = 2UL << 30; /* 2GB */
pr_info("x86/mm: Memory block size: %ldMB\n", bz >> 20);
return bz;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Yinghai Lu | 30 | 55.56% | 1 | 25.00% |
Seth Jennings | 15 | 27.78% | 1 | 25.00% |
Nathan Fontenot | 8 | 14.81% | 1 | 25.00% |
Daniel J Blueman | 1 | 1.85% | 1 | 25.00% |
Total | 54 | 100.00% | 4 | 100.00% |
static unsigned long memory_block_size_probed;
unsigned long memory_block_size_bytes(void)
{
if (!memory_block_size_probed)
memory_block_size_probed = probe_memory_block_size();
return memory_block_size_probed;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Yinghai Lu | 20 | 95.24% | 1 | 50.00% |
Nathan Fontenot | 1 | 4.76% | 1 | 50.00% |
Total | 21 | 100.00% | 2 | 100.00% |
#ifdef CONFIG_SPARSEMEM_VMEMMAP
/*
* Initialise the sparsemem vmemmap using huge-pages at the PMD level.
*/
static long __meminitdata addr_start, addr_end;
static void __meminitdata *p_start, *p_end;
static int __meminitdata node_start;
static int __meminit vmemmap_populate_hugepages(unsigned long start,
unsigned long end, int node, struct vmem_altmap *altmap)
{
unsigned long addr;
unsigned long next;
pgd_t *pgd;
pud_t *pud;
pmd_t *pmd;
for (addr = start; addr < end; addr = next) {
next = pmd_addr_end(addr, end);
pgd = vmemmap_pgd_populate(addr, node);
if (!pgd)
return -ENOMEM;
pud = vmemmap_pud_populate(pgd, addr, node);
if (!pud)
return -ENOMEM;
pmd = pmd_offset(pud, addr);
if (pmd_none(*pmd)) {
void *p;
p = __vmemmap_alloc_block_buf(PMD_SIZE, node, altmap);
if (p) {
pte_t entry;
entry = pfn_pte(__pa(p) >> PAGE_SHIFT,
PAGE_KERNEL_LARGE);
set_pmd(pmd, __pmd(pte_val(entry)));
/* check to see if we have contiguous blocks */
if (p_end != p || node_start != node) {
if (p_start)
pr_debug(" [%lx-%lx] PMD -> [%p-%p] on node %d\n",
addr_start, addr_end-1, p_start, p_end-1, node_start);
addr_start = addr;
node_start = node;
p_start = p;
}
addr_end = addr + PMD_SIZE;
p_end = p + PMD_SIZE;
continue;
} else if (altmap)
return -ENOMEM; /* no fallback */
} else if (pmd_large(*pmd)) {
vmemmap_verify((pte_t *)pmd, node, addr, next);
continue;
}
pr_warn_once("vmemmap: falling back to regular page backing\n");
if (vmemmap_populate_basepages(addr, next, node))
return -ENOMEM;
}
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Christoph Lameter | 161 | 54.21% | 1 | 10.00% |
Johannes Weiner | 62 | 20.88% | 3 | 30.00% |
Yinghai Lu | 50 | 16.84% | 2 | 20.00% |
Dan J Williams | 19 | 6.40% | 2 | 20.00% |
Thomas Gleixner | 4 | 1.35% | 1 | 10.00% |
Jeremy Fitzhardinge | 1 | 0.34% | 1 | 10.00% |
Total | 297 | 100.00% | 10 | 100.00% |
int __meminit vmemmap_populate(unsigned long start, unsigned long end, int node)
{
struct vmem_altmap *altmap = to_vmem_altmap(start);
int err;
if (boot_cpu_has(X86_FEATURE_PSE))
err = vmemmap_populate_hugepages(start, end, node, altmap);
else if (altmap) {
pr_err_once("%s: no cpu support for altmap allocations\n",
__func__);
err = -ENOMEM;
} else
err = vmemmap_populate_basepages(start, end, node);
if (!err)
sync_global_pgds(start, end - 1);
return err;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Johannes Weiner | 52 | 53.06% | 2 | 28.57% |
Dan J Williams | 31 | 31.63% | 1 | 14.29% |
Haicheng Li | 6 | 6.12% | 1 | 14.29% |
Borislav Petkov | 4 | 4.08% | 1 | 14.29% |
Christoph Lameter | 3 | 3.06% | 1 | 14.29% |
Wen Congyang | 2 | 2.04% | 1 | 14.29% |
Total | 98 | 100.00% | 7 | 100.00% |
#if defined(CONFIG_MEMORY_HOTPLUG_SPARSE) && defined(CONFIG_HAVE_BOOTMEM_INFO_NODE)
void register_page_bootmem_memmap(unsigned long section_nr,
struct page *start_page, unsigned long size)
{
unsigned long addr = (unsigned long)start_page;
unsigned long end = (unsigned long)(start_page + size);
unsigned long next;
pgd_t *pgd;
pud_t *pud;
pmd_t *pmd;
unsigned int nr_pages;
struct page *page;
for (; addr < end; addr = next) {
pte_t *pte = NULL;
pgd = pgd_offset_k(addr);
if (pgd_none(*pgd)) {
next = (addr + PAGE_SIZE) & PAGE_MASK;
continue;
}
get_page_bootmem(section_nr, pgd_page(*pgd), MIX_SECTION_INFO);
pud = pud_offset(pgd, addr);
if (pud_none(*pud)) {
next = (addr + PAGE_SIZE) & PAGE_MASK;
continue;
}
get_page_bootmem(section_nr, pud_page(*pud), MIX_SECTION_INFO);
if (!boot_cpu_has(X86_FEATURE_PSE)) {
next = (addr + PAGE_SIZE) & PAGE_MASK;
pmd = pmd_offset(pud, addr);
if (pmd_none(*pmd))
continue;
get_page_bootmem(section_nr, pmd_page(*pmd),
MIX_SECTION_INFO);
pte = pte_offset_kernel(pmd, addr);
if (pte_none(*pte))
continue;
get_page_bootmem(section_nr, pte_page(*pte),
SECTION_INFO);
} else {
next = pmd_addr_end(addr, end);
pmd = pmd_offset(pud, addr);
if (pmd_none(*pmd))
continue;
nr_pages = 1 << (get_order(PMD_SIZE));
page = pmd_page(*pmd);
while (nr_pages--)
get_page_bootmem(section_nr, page++,
SECTION_INFO);
}
}
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Yasuaki Ishimatsu | 312 | 98.73% | 1 | 50.00% |
Borislav Petkov | 4 | 1.27% | 1 | 50.00% |
Total | 316 | 100.00% | 2 | 100.00% |
#endif
void __meminit vmemmap_populate_print_last(void)
{
if (p_start) {
pr_debug(" [%lx-%lx] PMD -> [%p-%p] on node %d\n",
addr_start, addr_end-1, p_start, p_end-1, node_start);
p_start = NULL;
p_end = NULL;
node_start = 0;
}
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Yinghai Lu | 44 | 97.78% | 1 | 50.00% |
Dan J Williams | 1 | 2.22% | 1 | 50.00% |
Total | 45 | 100.00% | 2 | 100.00% |
#endif
Overall Contributors
Person | Tokens | Prop | Commits | CommitProp |
Wen Congyang | 1259 | 22.38% | 3 | 1.85% |
Andi Kleen | 716 | 12.73% | 21 | 12.96% |
Yinghai Lu | 546 | 9.70% | 25 | 15.43% |
Yasuaki Ishimatsu | 361 | 6.42% | 4 | 2.47% |
Jack Steiner | 258 | 4.59% | 1 | 0.62% |
Jeremy Fitzhardinge | 230 | 4.09% | 6 | 3.70% |
Tejun Heo | 221 | 3.93% | 5 | 3.09% |
Suresh B. Siddha | 178 | 3.16% | 6 | 3.70% |
Matt Tolentino | 173 | 3.08% | 1 | 0.62% |
Christoph Lameter | 169 | 3.00% | 2 | 1.23% |
Thomas Garnier | 162 | 2.88% | 2 | 1.23% |
Haicheng Li | 142 | 2.52% | 2 | 1.23% |
Dan J Williams | 133 | 2.36% | 3 | 1.85% |
Jan Beulich | 129 | 2.29% | 5 | 3.09% |
Johannes Weiner | 117 | 2.08% | 3 | 1.85% |
Steven Rostedt | 115 | 2.04% | 2 | 1.23% |
Thomas Gleixner | 97 | 1.72% | 5 | 3.09% |
Eduardo Pereira Habkost | 79 | 1.40% | 1 | 0.62% |
Tang Chen | 77 | 1.37% | 4 | 2.47% |
Shaohui Zheng | 65 | 1.16% | 1 | 0.62% |
Keith Mannthey | 59 | 1.05% | 2 | 1.23% |
Pekka J Enberg | 51 | 0.91% | 4 | 2.47% |
Arjan van de Ven | 39 | 0.69% | 4 | 2.47% |
Juergen Gross | 29 | 0.52% | 1 | 0.62% |
Seth Jennings | 18 | 0.32% | 1 | 0.62% |
Mel Gorman | 18 | 0.32% | 1 | 0.62% |
Kees Cook | 14 | 0.25% | 1 | 0.62% |
Alexander Duyck | 14 | 0.25% | 1 | 0.62% |
Lai Jiangshan | 13 | 0.23% | 1 | 0.62% |
Nathan Fontenot | 13 | 0.23% | 1 | 0.62% |
Linus Torvalds | 13 | 0.23% | 3 | 1.85% |
Jiang Liu | 12 | 0.21% | 4 | 2.47% |
Dave Hansen | 10 | 0.18% | 1 | 0.62% |
Ingo Molnar | 9 | 0.16% | 3 | 1.85% |
Yasunori Goto | 9 | 0.16% | 1 | 0.62% |
Wang Nan | 9 | 0.16% | 1 | 0.62% |
Borislav Petkov | 8 | 0.14% | 1 | 0.62% |
Stephen D. Smalley | 5 | 0.09% | 2 | 1.23% |
Muli Ben-Yehuda | 5 | 0.09% | 1 | 0.62% |
Mathieu Desnoyers | 4 | 0.07% | 1 | 0.62% |
Gary Hade | 4 | 0.07% | 2 | 1.23% |
Hugh Dickins | 4 | 0.07% | 2 | 1.23% |
Konrad Rzeszutek Wilk | 3 | 0.05% | 1 | 0.62% |
David Howells | 3 | 0.05% | 1 | 0.62% |
Randy Dunlap | 3 | 0.05% | 1 | 0.62% |
Andrea Arcangeli | 3 | 0.05% | 1 | 0.62% |
Harvey Harrison | 3 | 0.05% | 1 | 0.62% |
Benjamin Herrenschmidt | 3 | 0.05% | 1 | 0.62% |
Vegard Nossum | 2 | 0.04% | 1 | 0.62% |
Kamezawa Hiroyuki | 2 | 0.04% | 1 | 0.62% |
Kefeng Wang | 2 | 0.04% | 1 | 0.62% |
Marcin Ślusarz | 2 | 0.04% | 1 | 0.62% |
Fengguang Wu | 2 | 0.04% | 1 | 0.62% |
Andrew Lutomirski | 2 | 0.04% | 1 | 0.62% |
Kirill A. Shutemov | 1 | 0.02% | 1 | 0.62% |
Zhang Yanfei | 1 | 0.02% | 1 | 0.62% |
David Vrabel | 1 | 0.02% | 1 | 0.62% |
Daniel J Blueman | 1 | 0.02% | 1 | 0.62% |
Pavel Machek | 1 | 0.02% | 1 | 0.62% |
David Shaohua Li | 1 | 0.02% | 1 | 0.62% |
Joerg Roedel | 1 | 0.02% | 1 | 0.62% |
Andrew Morton | 1 | 0.02% | 1 | 0.62% |
Jon Mason | 1 | 0.02% | 1 | 0.62% |
Total | 5626 | 100.00% | 162 | 100.00% |
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.