cregit-Linux how code gets into the kernel

Release 4.10 arch/x86/mm/init_32.c

Directory: arch/x86/mm
/*
 *
 *  Copyright (C) 1995  Linus Torvalds
 *
 *  Support of BIGMEM added by Gerhard Wichert, Siemens AG, July 1999
 */

#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/hugetlb.h>
#include <linux/swap.h>
#include <linux/smp.h>
#include <linux/init.h>
#include <linux/highmem.h>
#include <linux/pagemap.h>
#include <linux/pci.h>
#include <linux/pfn.h>
#include <linux/poison.h>
#include <linux/bootmem.h>
#include <linux/memblock.h>
#include <linux/proc_fs.h>
#include <linux/memory_hotplug.h>
#include <linux/initrd.h>
#include <linux/cpumask.h>
#include <linux/gfp.h>

#include <asm/asm.h>
#include <asm/bios_ebda.h>
#include <asm/processor.h>
#include <linux/uaccess.h>
#include <asm/pgtable.h>
#include <asm/dma.h>
#include <asm/fixmap.h>
#include <asm/e820.h>
#include <asm/apic.h>
#include <asm/bugs.h>
#include <asm/tlb.h>
#include <asm/tlbflush.h>
#include <asm/olpc_ofw.h>
#include <asm/pgalloc.h>
#include <asm/sections.h>
#include <asm/paravirt.h>
#include <asm/setup.h>
#include <asm/cacheflush.h>
#include <asm/page_types.h>
#include <asm/init.h>

#include "mm_internal.h"



unsigned long highstart_pfn, highend_pfn;

static noinline int do_test_wp_bit(void);


bool __read_mostly __vmalloc_start_set = false;

/*
 * Creates a middle page table and puts a pointer to it in the
 * given global directory entry. This only returns the gd entry
 * in non-PAE compilation mode, since the middle layer is folded.
 */

static pmd_t * __init one_md_table_init(pgd_t *pgd) { pud_t *pud; pmd_t *pmd_table; #ifdef CONFIG_X86_PAE if (!(pgd_val(*pgd) & _PAGE_PRESENT)) { pmd_table = (pmd_t *)alloc_low_page(); paravirt_alloc_pmd(&init_mm, __pa(pmd_table) >> PAGE_SHIFT); set_pgd(pgd, __pgd(__pa(pmd_table) | _PAGE_PRESENT)); pud = pud_offset(pgd, 0); BUG_ON(pmd_table != pmd_offset(pud, 0)); return pmd_table; } #endif pud = pud_offset(pgd, 0); pmd_table = pmd_offset(pud, 0); return pmd_table; }

Contributors

PersonTokensPropCommitsCommitProp
dan alonidan aloni3125.41%15.56%
andi kleenandi kleen2419.67%15.56%
jeremy fitzhardingejeremy fitzhardinge2117.21%316.67%
pre-gitpre-git1915.57%633.33%
zachary amsdenzachary amsden97.38%15.56%
yinghai luyinghai lu86.56%15.56%
ingo molnaringo molnar32.46%15.56%
zhao leizhao lei32.46%15.56%
linus torvaldslinus torvalds21.64%15.56%
william lee irwin iiiwilliam lee irwin iii10.82%15.56%
jan beulichjan beulich10.82%15.56%
Total122100.00%18100.00%

/* * Create a page table and place a pointer to it in a middle page * directory entry: */
static pte_t * __init one_page_table_init(pmd_t *pmd) { if (!(pmd_val(*pmd) & _PAGE_PRESENT)) { pte_t *page_table = (pte_t *)alloc_low_page(); paravirt_alloc_pte(&init_mm, __pa(page_table) >> PAGE_SHIFT); set_pmd(pmd, __pmd(__pa(page_table) | _PAGE_TABLE)); BUG_ON(page_table != pte_offset_kernel(pmd, 0)); } return pte_offset_kernel(pmd, 0); }

Contributors

PersonTokensPropCommitsCommitProp
dan alonidan aloni2832.56%17.69%
brian gerstbrian gerst1719.77%17.69%
jeremy fitzhardingejeremy fitzhardinge1315.12%323.08%
pre-gitpre-git1112.79%323.08%
zachary amsdenzachary amsden910.47%17.69%
yinghai luyinghai lu66.98%215.38%
linus torvaldslinus torvalds11.16%17.69%
jan beulichjan beulich11.16%17.69%
Total86100.00%13100.00%


pmd_t * __init populate_extra_pmd(unsigned long vaddr) { int pgd_idx = pgd_index(vaddr); int pmd_idx = pmd_index(vaddr); return one_md_table_init(swapper_pg_dir + pgd_idx) + pmd_idx; }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo37100.00%2100.00%
Total37100.00%2100.00%


pte_t * __init populate_extra_pte(unsigned long vaddr) { int pte_idx = pte_index(vaddr); pmd_t *pmd; pmd = populate_extra_pmd(vaddr); return one_page_table_init(pmd) + pte_idx; }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo38100.00%2100.00%
Total38100.00%2100.00%


static unsigned long __init page_table_range_init_count(unsigned long start, unsigned long end) { unsigned long count = 0; #ifdef CONFIG_HIGHMEM int pmd_idx_kmap_begin = fix_to_virt(FIX_KMAP_END) >> PMD_SHIFT; int pmd_idx_kmap_end = fix_to_virt(FIX_KMAP_BEGIN) >> PMD_SHIFT; int pgd_idx, pmd_idx; unsigned long vaddr; if (pmd_idx_kmap_begin == pmd_idx_kmap_end) return 0; vaddr = start; pgd_idx = pgd_index(vaddr); pmd_idx = pmd_index(vaddr); for ( ; (pgd_idx < PTRS_PER_PGD) && (vaddr != end); pgd_idx++) { for (; (pmd_idx < PTRS_PER_PMD) && (vaddr != end); pmd_idx++) { if ((vaddr >> PMD_SHIFT) >= pmd_idx_kmap_begin && (vaddr >> PMD_SHIFT) <= pmd_idx_kmap_end) count++; vaddr += PMD_SIZE; } pmd_idx = 0; } #endif return count; }

Contributors

PersonTokensPropCommitsCommitProp
yinghai luyinghai lu14895.48%150.00%
minfei huangminfei huang74.52%150.00%
Total155100.00%2100.00%


static pte_t *__init page_table_kmap_check(pte_t *pte, pmd_t *pmd, unsigned long vaddr, pte_t *lastpte, void **adr) { #ifdef CONFIG_HIGHMEM /* * Something (early fixmap) may already have put a pte * page here, which causes the page table allocation * to become nonlinear. Attempt to fix it, and if it * is still nonlinear then we have to bug. */ int pmd_idx_kmap_begin = fix_to_virt(FIX_KMAP_END) >> PMD_SHIFT; int pmd_idx_kmap_end = fix_to_virt(FIX_KMAP_BEGIN) >> PMD_SHIFT; if (pmd_idx_kmap_begin != pmd_idx_kmap_end && (vaddr >> PMD_SHIFT) >= pmd_idx_kmap_begin && (vaddr >> PMD_SHIFT) <= pmd_idx_kmap_end) { pte_t *newpte; int i; BUG_ON(after_bootmem); newpte = *adr; for (i = 0; i < PTRS_PER_PTE; i++) set_pte(newpte + i, pte[i]); *adr = (void *)(((unsigned long)(*adr)) + PAGE_SIZE); paravirt_alloc_pte(&init_mm, __pa(newpte) >> PAGE_SHIFT); set_pmd(pmd, __pmd(__pa(newpte)|_PAGE_TABLE)); BUG_ON(newpte != pte_offset_kernel(pmd, 0)); __flush_tlb_all(); paravirt_release_pte(__pa(pte) >> PAGE_SHIFT); pte = newpte; } BUG_ON(vaddr < fix_to_virt(FIX_KMAP_BEGIN - 1) && vaddr > fix_to_virt(FIX_KMAP_END) && lastpte && lastpte + PTRS_PER_PTE != pte); #endif return pte; }

Contributors

PersonTokensPropCommitsCommitProp
jan beulichjan beulich20086.96%133.33%
yinghai luyinghai lu2912.61%133.33%
pekka j enbergpekka j enberg10.43%133.33%
Total230100.00%3100.00%

/* * This function initializes a certain range of kernel virtual memory * with new bootmem page tables, everywhere page tables are missing in * the given range. * * NOTE: The pagetables are allocated contiguous on the physical space * so we can cache the place of the first one and move around without * checking the pgd every time. */
static void __init page_table_range_init(unsigned long start, unsigned long end, pgd_t *pgd_base) { int pgd_idx, pmd_idx; unsigned long vaddr; pgd_t *pgd; pmd_t *pmd; pte_t *pte = NULL; unsigned long count = page_table_range_init_count(start, end); void *adr = NULL; if (count) adr = alloc_low_pages(count); vaddr = start; pgd_idx = pgd_index(vaddr); pmd_idx = pmd_index(vaddr); pgd = pgd_base + pgd_idx; for ( ; (pgd_idx < PTRS_PER_PGD) && (vaddr != end); pgd++, pgd_idx++) { pmd = one_md_table_init(pgd); pmd = pmd + pmd_index(vaddr); for (; (pmd_idx < PTRS_PER_PMD) && (vaddr != end); pmd++, pmd_idx++) { pte = page_table_kmap_check(one_page_table_init(pmd), pmd, vaddr, pte, &adr); vaddr += PMD_SIZE; } pmd_idx = 0; } }

Contributors

PersonTokensPropCommitsCommitProp
pre-gitpre-git10356.28%220.00%
yinghai luyinghai lu3116.94%110.00%
jan beulichjan beulich179.29%110.00%
andrew mortonandrew morton126.56%220.00%
ingo molnaringo molnar84.37%110.00%
andi kleenandi kleen52.73%110.00%
jeremy fitzhardingejeremy fitzhardinge52.73%110.00%
dan alonidan aloni21.09%110.00%
Total183100.00%10100.00%


static inline int is_kernel_text(unsigned long addr) { if (addr >= (unsigned long)_text && addr <= (unsigned long)__init_end) return 1; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
ingo molnaringo molnar3085.71%150.00%
matthieu castetmatthieu castet514.29%150.00%
Total35100.00%2100.00%

/* * This maps the physical memory to kernel virtual address space, a total * of max_low_pfn pages, by creating page tables starting from address * PAGE_OFFSET: */
unsigned long __init kernel_physical_mapping_init(unsigned long start, unsigned long end, unsigned long page_size_mask) { int use_pse = page_size_mask == (1<<PG_LEVEL_2M); unsigned long last_map_addr = end; unsigned long start_pfn, end_pfn; pgd_t *pgd_base = swapper_pg_dir; int pgd_idx, pmd_idx, pte_ofs; unsigned long pfn; pgd_t *pgd; pmd_t *pmd; pte_t *pte; unsigned pages_2m, pages_4k; int mapping_iter; start_pfn = start >> PAGE_SHIFT; end_pfn = end >> PAGE_SHIFT; /* * First iteration will setup identity mapping using large/small pages * based on use_pse, with other attributes same as set by * the early code in head_32.S * * Second iteration will setup the appropriate attributes (NX, GLOBAL..) * as desired for the kernel identity mapping. * * This two pass mechanism conforms to the TLB app note which says: * * "Software should not write to a paging-structure entry in a way * that would change, for any linear address, both the page size * and either the page frame or attributes." */ mapping_iter = 1; if (!boot_cpu_has(X86_FEATURE_PSE)) use_pse = 0; repeat: pages_2m = pages_4k = 0; pfn = start_pfn; pgd_idx = pgd_index((pfn<<PAGE_SHIFT) + PAGE_OFFSET); pgd = pgd_base + pgd_idx; for (; pgd_idx < PTRS_PER_PGD; pgd++, pgd_idx++) { pmd = one_md_table_init(pgd); if (pfn >= end_pfn) continue; #ifdef CONFIG_X86_PAE pmd_idx = pmd_index((pfn<<PAGE_SHIFT) + PAGE_OFFSET); pmd += pmd_idx; #else pmd_idx = 0; #endif for (; pmd_idx < PTRS_PER_PMD && pfn < end_pfn; pmd++, pmd_idx++) { unsigned int addr = pfn * PAGE_SIZE + PAGE_OFFSET; /* * Map with big pages if possible, otherwise * create normal page tables: */ if (use_pse) { unsigned int addr2; pgprot_t prot = PAGE_KERNEL_LARGE; /* * first pass will use the same initial * identity mapping attribute + _PAGE_PSE. */ pgprot_t init_prot = __pgprot(PTE_IDENT_ATTR | _PAGE_PSE); pfn &= PMD_MASK >> PAGE_SHIFT; addr2 = (pfn + PTRS_PER_PTE-1) * PAGE_SIZE + PAGE_OFFSET + PAGE_SIZE-1; if (is_kernel_text(addr) || is_kernel_text(addr2)) prot = PAGE_KERNEL_LARGE_EXEC; pages_2m++; if (mapping_iter == 1) set_pmd(pmd, pfn_pmd(pfn, init_prot)); else set_pmd(pmd, pfn_pmd(pfn, prot)); pfn += PTRS_PER_PTE; continue; } pte = one_page_table_init(pmd); pte_ofs = pte_index((pfn<<PAGE_SHIFT) + PAGE_OFFSET); pte += pte_ofs; for (; pte_ofs < PTRS_PER_PTE && pfn < end_pfn; pte++, pfn++, pte_ofs++, addr += PAGE_SIZE) { pgprot_t prot = PAGE_KERNEL; /* * first pass will use the same initial * identity mapping attribute. */ pgprot_t init_prot = __pgprot(PTE_IDENT_ATTR); if (is_kernel_text(addr)) prot = PAGE_KERNEL_EXEC; pages_4k++; if (mapping_iter == 1) { set_pte(pte, pfn_pte(pfn, init_prot)); last_map_addr = (pfn << PAGE_SHIFT) + PAGE_SIZE; } else set_pte(pte, pfn_pte(pfn, prot)); } } } if (mapping_iter == 1) { /* * update direct mapping page count only in the first * iteration. */ update_page_count(PG_LEVEL_2M, pages_2m); update_page_count(PG_LEVEL_4K, pages_4k); /* * local global flush tlb, which will flush the previous * mappings present in both small and large page TLB's. */ __flush_tlb_all(); /* * Second iteration will set the actual desired PTE attributes. */ mapping_iter = 2; goto repeat; } return last_map_addr; }

Contributors

PersonTokensPropCommitsCommitProp
suresh siddhasuresh siddha9519.39%12.94%
pre-gitpre-git8918.16%1235.29%
yinghai luyinghai lu7916.12%411.76%
ingo molnaringo molnar6413.06%25.88%
pekka j enbergpekka j enberg6012.24%38.82%
brian gerstbrian gerst265.31%25.88%
andi kleenandi kleen255.10%12.94%
jeremy fitzhardingejeremy fitzhardinge214.29%25.88%
andrew mortonandrew morton142.86%38.82%
dan alonidan aloni91.84%12.94%
borislav petkovborislav petkov40.82%12.94%
arjan van de venarjan van de ven30.61%12.94%
roman zippelroman zippel10.20%12.94%
Total490100.00%34100.00%

pte_t *kmap_pte;
static inline pte_t *kmap_get_fixmap_pte(unsigned long vaddr) { return pte_offset_kernel(pmd_offset(pud_offset(pgd_offset_k(vaddr), vaddr), vaddr), vaddr); }

Contributors

PersonTokensPropCommitsCommitProp
ingo molnaringo molnar2987.88%150.00%
dan alonidan aloni412.12%150.00%
Total33100.00%2100.00%


static void __init kmap_init(void) { unsigned long kmap_vstart; /* * Cache the first kmap pte: */ kmap_vstart = __fix_to_virt(FIX_KMAP_BEGIN); kmap_pte = kmap_get_fixmap_pte(kmap_vstart); }

Contributors

PersonTokensPropCommitsCommitProp
dan alonidan aloni1346.43%116.67%
pre-gitpre-git1035.71%233.33%
linus torvaldslinus torvalds310.71%116.67%
adrian bunkadrian bunk13.57%116.67%
ingo molnaringo molnar13.57%116.67%
Total28100.00%6100.00%

#ifdef CONFIG_HIGHMEM
static void __init permanent_kmaps_init(pgd_t *pgd_base) { unsigned long vaddr; pgd_t *pgd; pud_t *pud; pmd_t *pmd; pte_t *pte; vaddr = PKMAP_BASE; page_table_range_init(vaddr, vaddr + PAGE_SIZE*LAST_PKMAP, pgd_base); pgd = swapper_pg_dir + pgd_index(vaddr); pud = pud_offset(pgd, vaddr); pmd = pmd_offset(pud, vaddr); pte = pte_offset_kernel(pmd, vaddr); pkmap_page_table = pte; }

Contributors

PersonTokensPropCommitsCommitProp
pre-gitpre-git4854.55%853.33%
dan alonidan aloni1719.32%16.67%
andi kleenandi kleen1415.91%16.67%
ingo molnaringo molnar55.68%213.33%
brian gerstbrian gerst22.27%16.67%
adrian bunkadrian bunk11.14%16.67%
andrew mortonandrew morton11.14%16.67%
Total88100.00%15100.00%


void __init add_highpages_with_active_regions(int nid, unsigned long start_pfn, unsigned long end_pfn) { phys_addr_t start, end; u64 i; for_each_free_mem_range(i, nid, MEMBLOCK_NONE, &start, &end, NULL) { unsigned long pfn = clamp_t(unsigned long, PFN_UP(start), start_pfn, end_pfn); unsigned long e_pfn = clamp_t(unsigned long, PFN_DOWN(end), start_pfn, end_pfn); for ( ; pfn < e_pfn; pfn++) if (pfn_valid(pfn)) free_highmem_page(pfn_to_page(pfn)); } }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo4647.92%112.50%
yinghai luyinghai lu2323.96%225.00%
andrew mortonandrew morton1717.71%225.00%
ingo molnaringo molnar77.29%112.50%
tony lucktony luck22.08%112.50%
jiang liujiang liu11.04%112.50%
Total96100.00%8100.00%

#else
static inline void permanent_kmaps_init(pgd_t *pgd_base) { }

Contributors

PersonTokensPropCommitsCommitProp
ingo bruecklingo brueckl660.00%150.00%
dan alonidan aloni440.00%150.00%
Total10100.00%2100.00%

#endif /* CONFIG_HIGHMEM */
void __init native_pagetable_init(void) { unsigned long pfn, va; pgd_t *pgd, *base = swapper_pg_dir; pud_t *pud; pmd_t *pmd; pte_t *pte; /* * Remove any mappings which extend past the end of physical * memory from the boot time page table. * In virtual address space, we should have at least two pages * from VMALLOC_END to pkmap or fixmap according to VMALLOC_END * definition. And max_low_pfn is set to VMALLOC_END physical * address. If initial memory mapping is doing right job, we * should have pte used near max_low_pfn or one pmd is not present. */ for (pfn = max_low_pfn; pfn < 1<<(32-PAGE_SHIFT); pfn++) { va = PAGE_OFFSET + (pfn<<PAGE_SHIFT); pgd = base + pgd_index(va); if (!pgd_present(*pgd)) break; pud = pud_offset(pgd, va); pmd = pmd_offset(pud, va); if (!pmd_present(*pmd)) break; /* should not be large page here */ if (pmd_large(*pmd)) { pr_warn("try to clear pte for ram above max_low_pfn: pfn: %lx pmd: %p pmd phys: %lx, but pmd is big page and is not using pte !\n", pfn, pmd, __pa(pmd)); BUG_ON(1); } pte = pte_offset_kernel(pmd, va); if (!pte_present(*pte)) break; printk(KERN_DEBUG "clearing pte for ram above max_low_pfn: pfn: %lx pmd: %p pmd phys: %lx pte: %p pte phys: %lx\n", pfn, pmd, __pa(pmd), pte, __pa(pte)); pte_clear(NULL, va, pte); } paravirt_alloc_pmd(&init_mm, __pa(base) >> PAGE_SHIFT); paging_init(); }

Contributors

PersonTokensPropCommitsCommitProp
ian campbellian campbell10650.48%17.69%
yinghai luyinghai lu5325.24%17.69%
jeremy fitzhardingejeremy fitzhardinge167.62%323.08%
pre-gitpre-git136.19%215.38%
attilio raoattilio rao104.76%323.08%
zachary amsdenzachary amsden83.81%17.69%
dan alonidan aloni31.43%17.69%
linus torvaldslinus torvalds10.48%17.69%
Total210100.00%13100.00%

/* * Build a proper pagetable for the kernel mappings. Up until this * point, we've been running on some set of pagetables constructed by * the boot process. * * If we're booting on native hardware, this will be a pagetable * constructed in arch/x86/kernel/head_32.S. The root of the * pagetable will be swapper_pg_dir. * * If we're booting paravirtualized under a hypervisor, then there are * more options: we may already be running PAE, and the pagetable may * or may not be based in swapper_pg_dir. In any case, * paravirt_pagetable_init() will set up swapper_pg_dir * appropriately for the rest of the initialization to work. * * In general, pagetable_init() assumes that the pagetable may already * be partially populated, and so it avoids stomping on any existing * mappings. */
void __init early_ioremap_page_table_range_init(void) { pgd_t *pgd_base = swapper_pg_dir; unsigned long vaddr, end; /* * Fixed mappings, only the page table structure has to be * created - mappings will be set by set_fixmap(): */ vaddr = __fix_to_virt(__end_of_fixed_addresses - 1) & PMD_MASK; end = (FIXADDR_TOP + PMD_SIZE - 1) & PMD_MASK; page_table_range_init(vaddr, end, pgd_base); early_ioremap_reset(); }

Contributors

PersonTokensPropCommitsCommitProp
jeremy fitzhardingejeremy fitzhardinge1933.93%112.50%
dan alonidan aloni1628.57%112.50%
pekka j enbergpekka j enberg610.71%112.50%
ingo molnaringo molnar610.71%112.50%
pre-gitpre-git47.14%112.50%
huang yinghuang ying35.36%225.00%
yinghai luyinghai lu23.57%112.50%
Total56100.00%8100.00%


static void __init pagetable_init(void) { pgd_t *pgd_base = swapper_pg_dir; permanent_kmaps_init(pgd_base); }

Contributors

PersonTokensPropCommitsCommitProp
yinghai luyinghai lu1470.00%133.33%
dan alonidan aloni525.00%133.33%
pre-gitpre-git15.00%133.33%
Total20100.00%3100.00%

pteval_t __supported_pte_mask __read_mostly = ~(_PAGE_NX | _PAGE_GLOBAL); EXPORT_SYMBOL_GPL(__supported_pte_mask); /* user-defined highmem size */ static unsigned int highmem_pages = -1; /* * highmem=size forces highmem to be exactly 'size' bytes. * This works even on boxes that have no highmem otherwise. * This also works to reduce highmem size on bigger boxes. */
static int __init parse_highmem(char *arg) { if (!arg) return -EINVAL; highmem_pages = memparse(arg, &arg) >> PAGE_SHIFT; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
yinghai luyinghai lu35100.00%1100.00%
Total35100.00%1100.00%

early_param("highmem", parse_highmem); #define MSG_HIGHMEM_TOO_BIG \ "highmem size (%luMB) is bigger than pages available (%luMB)!\n" #define MSG_LOWMEM_TOO_SMALL \ "highmem size (%luMB) results in <64MB lowmem, ignoring it!\n" /* * All of RAM fits into lowmem - but if user wants highmem * artificially via the highmem=x boot parameter then create * it: */
static void __init lowmem_pfn_init(void) { /* max_low_pfn is 0, we already have early_res support */ max_low_pfn = max_pfn; if (highmem_pages == -1) highmem_pages = 0; #ifdef CONFIG_HIGHMEM if (highmem_pages >= max_pfn) { printk(KERN_ERR MSG_HIGHMEM_TOO_BIG, pages_to_mb(highmem_pages), pages_to_mb(max_pfn)); highmem_pages = 0; } if (highmem_pages) { if (max_low_pfn - highmem_pages < 64*1024*1024/PAGE_SIZE) { printk(KERN_ERR MSG_LOWMEM_TOO_SMALL, pages_to_mb(highmem_pages)); highmem_pages = 0; } max_low_pfn -= highmem_pages; } #else if (highmem_pages) printk(KERN_ERR "ignoring highmem size on non-highmem kernel!\n"); #endif }

Contributors

PersonTokensPropCommitsCommitProp
ingo molnaringo molnar6666.00%116.67%
yinghai luyinghai lu3434.00%583.33%
Total100100.00%6100.00%

#define MSG_HIGHMEM_TOO_SMALL \ "only %luMB highmem pages available, ignoring highmem size of %luMB!\n" #define MSG_HIGHMEM_TRIMMED \ "Warning: only 4GB will be used. Use a HIGHMEM64G enabled kernel!\n" /* * We have more RAM than fits into lowmem - we try to put it into * highmem, also taking the highmem=x boot parameter into account: */
static void __init highmem_pfn_init(void) { max_low_pfn = MAXMEM_PFN; if (highmem_pages == -1) highmem_pages = max_pfn - MAXMEM_PFN; if (highmem_pages + MAXMEM_PFN < max_pfn) max_pfn = MAXMEM_PFN + highmem_pages; if (highmem_pages + MAXMEM_PFN > max_pfn) { printk(KERN_WARNING MSG_HIGHMEM_TOO_SMALL, pages_to_mb(max_pfn - MAXMEM_PFN), pages_to_mb(highmem_pages)); highmem_pages = 0; } #ifndef CONFIG_HIGHMEM /* Maximum memory usable is what is directly addressable */ printk(KERN_WARNING "Warning only %ldMB will be used.\n", MAXMEM>>20); if (max_pfn > MAX_NONPAE_PFN) printk(KERN_WARNING "Use a HIGHMEM64G enabled kernel.\n"); else printk(KERN_WARNING "Use a HIGHMEM enabled kernel.\n"); max_pfn = MAXMEM_PFN; #else /* !CONFIG_HIGHMEM */ #ifndef CONFIG_HIGHMEM64G if (max_pfn > MAX_NONPAE_PFN) { max_pfn = MAX_NONPAE_PFN; printk(KERN_WARNING MSG_HIGHMEM_TRIMMED); } #endif /* !CONFIG_HIGHMEM64G */ #endif /* !CONFIG_HIGHMEM */ }

Contributors

PersonTokensPropCommitsCommitProp
yinghai luyinghai lu12193.80%466.67%
ingo molnaringo molnar86.20%233.33%
Total129100.00%6100.00%

/* * Determine low and high memory ranges: */
void __init find_low_pfn_range(void) { /* it could update max_pfn */ if (max_pfn <= MAXMEM_PFN) lowmem_pfn_init(); else highmem_pfn_init(); }

Contributors

PersonTokensPropCommitsCommitProp
ingo molnaringo molnar1463.64%266.67%
yinghai luyinghai lu836.36%133.33%
Total22100.00%3100.00%

#ifndef CONFIG_NEED_MULTIPLE_NODES
void __init initmem_init(void) { #ifdef CONFIG_HIGHMEM highstart_pfn = highend_pfn = max_pfn; if (max_pfn > max_low_pfn) highstart_pfn = max_low_pfn; printk(KERN_NOTICE "%ldMB HIGHMEM available.\n", pages_to_mb(highend_pfn - highstart_pfn)); high_memory = (void *) __va(highstart_pfn * PAGE_SIZE - 1) + 1; #else high_memory = (void *) __va(max_low_pfn * PAGE_SIZE - 1) + 1; #endif memblock_set_node(0, (phys_addr_t)ULLONG_MAX, &memblock.memory, 0); sparse_memory_present_with_active_regions(0); #ifdef CONFIG_FLATMEM max_mapnr = IS_ENABLED(CONFIG_HIGHMEM) ? highend_pfn : max_low_pfn; #endif __vmalloc_start_set = true; printk(KERN_NOTICE "%ldMB LOWMEM available.\n", pages_to_mb(max_low_pfn)); setup_bootmem_allocator(); }

Contributors

PersonTokensPropCommitsCommitProp
yinghai luyinghai lu9973.88%228.57%
tejun heotejun heo1813.43%228.57%
jiang liujiang liu85.97%114.29%
tang chentang chen53.73%114.29%
jeremy fitzhardingejeremy fitzhardinge42.99%114.29%
Total134100.00%7100.00%

#endif /* !CONFIG_NEED_MULTIPLE_NODES */
void __init setup_bootmem_allocator(void) { printk(KERN_INFO " mapped low ram: 0 - %08lx\n", max_pfn_mapped<<PAGE_SHIFT); printk(KERN_INFO " low ram: 0 - %08lx\n", max_low_pfn<<PAGE_SHIFT); }

Contributors

PersonTokensPropCommitsCommitProp
yinghai luyinghai lu1864.29%444.44%
dan alonidan aloni517.86%111.11%
pre-gitpre-git310.71%222.22%
ingo molnaringo molnar27.14%222.22%
Total28100.00%9100.00%

/* * paging_init() sets up the page tables - note that the first 8MB are * already mapped by head.S. * * This routines also unmaps the page at virtual kernel address 0, so * that we can trap those pesky NULL-reference errors in the kernel. */
void __init paging_init(void) { pagetable_init(); __flush_tlb_all(); kmap_init(); /* * NOTE: at this point the bootmem allocator is fully available. */ olpc_dt_build_devicetree(); sparse_memory_present_with_active_regions(MAX_NUMNODES); sparse_init(); zone_sizes_init(); }

Contributors

PersonTokensPropCommitsCommitProp
yinghai luyinghai lu1959.38%233.33%
tejun heotejun heo515.62%116.67%
pre-gitpre-git39.38%116.67%
andres salomonandres salomon39.38%116.67%
dan alonidan aloni26.25%116.67%
Total32100.00%6100.00%

/* * Test if the WP bit works in supervisor mode. It isn't supported on 386's * and also on some strange 486's. All 586+'s are OK. This used to involve * black magic jumps to work around some nasty CPU bugs, but fortunately the * switch to using exceptions got rid of all that. */
static void __init test_wp_bit(void) { printk(KERN_INFO "Checking if this processor honours the WP bit even in supervisor mode..."); /* Any page-aligned address will do, the test is non-destructive */ __set_fixmap(FIX_WP_TEST, __pa(&swapper_pg_dir), PAGE_KERNEL_RO); boot_cpu_data.wp_works_ok = do_test_wp_bit(); clear_fixmap(FIX_WP_TEST); if (!boot_cpu_data.wp_works_ok) { printk(KERN_CONT "No.\n"); panic("Linux doesn't support CPUs with broken WP."); } else { printk(KERN_CONT "Ok.\n"); } }

Contributors

PersonTokensPropCommitsCommitProp
dan alonidan aloni2840.00%110.00%
pre-gitpre-git1318.57%330.00%
linus torvaldslinus torvalds1217.14%110.00%
brian gerstbrian gerst1014.29%110.00%
ingo molnaringo molnar45.71%110.00%
h. peter anvinh. peter anvin22.86%220.00%
adrian bunkadrian bunk11.43%110.00%
Total70100.00%10100.00%


void __init mem_init(void) { pci_iommu_alloc(); #ifdef CONFIG_FLATMEM BUG_ON(!mem_map); #endif /* * With CONFIG_DEBUG_PAGEALLOC initialization of highmem pages has to * be done before free_all_bootmem(). Memblock use free low memory for * temporary data (see find_range_array()) and for this purpose can use * pages that was already passed to the buddy allocator, hence marked as * not accessible in the page tables when compiled with * CONFIG_DEBUG_PAGEALLOC. Otherwise order of initialization is not * important here. */ set_highmem_pages_init(); /* this will put all low memory onto the freelists */ free_all_bootmem(); after_bootmem = 1; mem_init_print_info(NULL); printk(KERN_INFO "virtual kernel memory layout:\n" " fixmap : 0x%08lx - 0x%08lx (%4ld kB)\n" #ifdef CONFIG_HIGHMEM " pkmap : 0x%08lx - 0x%08lx (%4ld kB)\n" #endif " vmalloc : 0x%08lx - 0x%08lx (%4ld MB)\n" " lowmem : 0x%08lx - 0x%08lx (%4ld MB)\n" " .init : 0x%08lx - 0x%08lx (%4ld kB)\n" " .data : 0x%08lx - 0x%08lx (%4ld kB)\n" " .text : 0x%08lx - 0x%08lx (%4ld kB)\n", FIXADDR_START, FIXADDR_TOP, (FIXADDR_TOP - FIXADDR_START) >> 10, #ifdef CONFIG_HIGHMEM PKMAP_BASE, PKMAP_BASE+LAST_PKMAP*PAGE_SIZE, (LAST_PKMAP*PAGE_SIZE) >> 10, #endif VMALLOC_START, VMALLOC_END, (VMALLOC_END - VMALLOC_START) >> 20, (unsigned long)__va(0), (unsigned long)high_memory, ((unsigned long)high_memory - (unsigned long)__va(0)) >> 20, (unsigned long)&__init_begin, (unsigned long)&__init_end, ((unsigned long)&__init_end - (unsigned long)&__init_begin) >> 10, (unsigned long)&_etext, (unsigned long)&_edata, ((unsigned long)&_edata - (unsigned long)&_etext) >> 10, (unsigned long)&_text, (unsigned long)&_etext, ((unsigned long)&_etext - (unsigned long)&_text) >> 10); /* * Check boundaries twice: Some fundamental inconsistencies can * be detected at build time already. */ #define __FIXADDR_TOP (-PAGE_SIZE) #ifdef CONFIG_HIGHMEM BUILD_BUG_ON(PKMAP_BASE + LAST_PKMAP*PAGE_SIZE > FIXADDR_START); BUILD_BUG_ON(VMALLOC_END > PKMAP_BASE); #endif #define high_memory (-128UL << 20) BUILD_BUG_ON(VMALLOC_START >= VMALLOC_END); #undef high_memory #undef __FIXADDR_TOP #ifdef CONFIG_HIGHMEM BUG_ON(PKMAP_BASE + LAST_PKMAP*PAGE_SIZE > FIXADDR_START); BUG_ON(VMALLOC_END > PKMAP_BASE); #endif BUG_ON(VMALLOC_START >= VMALLOC_END); BUG_ON((unsigned long)high_memory > VMALLOC_START); if (boot_cpu_data.wp_works_ok < 0) test_wp_bit(); }

Contributors

PersonTokensPropCommitsCommitProp
jeremy fitzhardingejeremy fitzhardinge23571.43%215.38%
jan beulichjan beulich4613.98%17.69%
dan alonidan aloni195.78%17.69%
richard hendersonrichard henderson92.74%17.69%
stanislaw gruszkastanislaw gruszka41.22%17.69%
yinghai luyinghai lu41.22%17.69%
andy whitcroftandy whitcroft30.91%17.69%
eric sesterhenneric sesterhenn30.91%17.69%
andrew mortonandrew morton20.61%17.69%
jiang liujiang liu20.61%17.69%
dave hansendave hansen10.30%17.69%
ingo molnaringo molnar10.30%17.69%
Total329100.00%13100.00%

#ifdef CONFIG_MEMORY_HOTPLUG
int arch_add_memory(int nid, u64 start, u64 size, bool for_device) { struct pglist_data *pgdata = NODE_DATA(nid); struct zone *zone = pgdata->node_zones + zone_for_memory(nid, start, size, ZONE_HIGHMEM, for_device); unsigned long start_pfn = start >> PAGE_SHIFT; unsigned long nr_pages = size >> PAGE_SHIFT; return __add_pages(nid, zone, start_pfn, nr_pages); }

Contributors

PersonTokensPropCommitsCommitProp
dave hansendave hansen5267.53%114.29%
wang nanwang nan911.69%114.29%
yasunori gotoyasunori goto810.39%228.57%
dan williamsdan williams56.49%114.29%
gary hadegary hade22.60%114.29%
christoph lameterchristoph lameter11.30%114.29%
Total77100.00%7100.00%

#ifdef CONFIG_MEMORY_HOTREMOVE
int arch_remove_memory(u64 start, u64 size) { unsigned long start_pfn = start >> PAGE_SHIFT; unsigned long nr_pages = size >> PAGE_SHIFT; struct zone *zone; zone = page_zone(pfn_to_page(start_pfn)); return __remove_pages(zone, start_pfn, nr_pages); }

Contributors

PersonTokensPropCommitsCommitProp
wen congyangwen congyang52100.00%1100.00%
Total52100.00%1100.00%

#endif #endif /* * This function cannot be __init, since exceptions don't work in that * section. Put this after the callers, so that it cannot be inlined. */
static noinline int do_test_wp_bit(void) { char tmp_reg; int flag; __asm__ __volatile__( " movb %0, %1 \n" "1: movb %1, %0 \n" " xorl %2, %2 \n" "2: \n" _ASM_EXTABLE(1b,2b) :"=m" (*(char *)fix_to_virt(FIX_WP_TEST)), "=q" (tmp_reg), "=r" (flag) :"2" (1) :"memory"); return flag; }

Contributors

PersonTokensPropCommitsCommitProp
dan alonidan aloni1257.14%116.67%
richard hendersonrichard henderson523.81%116.67%
h. peter anvinh. peter anvin14.76%116.67%
brian gerstbrian gerst14.76%116.67%
linus torvaldslinus torvalds14.76%116.67%
ingo molnaringo molnar14.76%116.67%
Total21100.00%6100.00%

const int rodata_test_data = 0xC3; EXPORT_SYMBOL_GPL(rodata_test_data); int kernel_set_to_readonly __read_mostly;
void set_kernel_text_rw(void) { unsigned long start = PFN_ALIGN(_text); unsigned long size = PFN_ALIGN(_etext) - start; if (!kernel_set_to_readonly) return; pr_debug("Set kernel text: %lx - %lx for read write\n", start, start+size); set_pages_rw(virt_to_page(start), size >> PAGE_SHIFT); }

Contributors

PersonTokensPropCommitsCommitProp
steven rostedtsteven rostedt56100.00%1100.00%
Total56100.00%1100.00%


void set_kernel_text_ro(void) { unsigned long start = PFN_ALIGN(_text); unsigned long size = PFN_ALIGN(_etext) - start; if (!kernel_set_to_readonly) return; pr_debug("Set kernel text: %lx - %lx for read only\n", start, start+size); set_pages_ro(virt_to_page(start), size >> PAGE_SHIFT); }

Contributors

PersonTokensPropCommitsCommitProp
steven rostedtsteven rostedt56100.00%1100.00%
Total56100.00%1100.00%


static void mark_nxdata_nx(void) { /* * When this called, init has already been executed and released, * so everything past _etext should be NX. */ unsigned long start = PFN_ALIGN(_etext); /* * This comes from is_kernel_text upper limit. Also HPAGE where used: */ unsigned long size = (((unsigned long)__init_end + HPAGE_SIZE) & HPAGE_MASK) - start; if (__supported_pte_mask & _PAGE_NX) printk(KERN_INFO "NX-protecting the kernel data: %luk\n", size >> 10); set_pages_nx(virt_to_page(start), size >> PAGE_SHIFT); }

Contributors

PersonTokensPropCommitsCommitProp
matthieu castetmatthieu castet6698.51%150.00%
lucas de marchilucas de marchi11.49%150.00%
Total67100.00%2100.00%


void mark_rodata_ro(void) { unsigned long start = PFN_ALIGN(_text); unsigned long size = PFN_ALIGN(_etext) - start; set_pages_ro(virt_to_page(start), size >> PAGE_SHIFT); printk(KERN_INFO "Write protecting the kernel text: %luk\n", size >> 10); kernel_set_to_readonly = 1; #ifdef CONFIG_CPA_DEBUG printk(KERN_INFO "Testing CPA: Reverting %lx-%lx\n", start, start+size); set_pages_rw(virt_to_page(start), size>>PAGE_SHIFT); printk(KERN_INFO "Testing CPA: write protecting again\n"); set_pages_ro(virt_to_page(start), size>>PAGE_SHIFT); #endif start += size; size = (unsigned long)__end_rodata - start; set_pages_ro(virt_to_page(start), size >> PAGE_SHIFT); printk(KERN_INFO "Write protecting the kernel read-only data: %luk\n", size >> 10); rodata_test(); #ifdef CONFIG_CPA_DEBUG printk(KERN_INFO "Testing CPA: undo %lx-%lx\n", start, start + size); set_pages_rw(virt_to_page(start), size >> PAGE_SHIFT); printk(KERN_INFO "Testing CPA: write protecting again\n"); set_pages_ro(virt_to_page(start), size >> PAGE_SHIFT); #endif mark_nxdata_nx(); if (__supported_pte_mask & _PAGE_NX) debug_checkwx(); }

Contributors

PersonTokensPropCommitsCommitProp
andi kleenandi kleen8643.43%111.11%
jan beulichjan beulich4924.75%111.11%
arjan van de venarjan van de ven4120.71%333.33%
stephen d. smalleystephen d. smalley94.55%111.11%
ingo molnaringo molnar63.03%111.11%
steven rostedtsteven rostedt42.02%111.11%
matthieu castetmatthieu castet31.52%111.11%
Total198100.00%9100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
yinghai luyinghai lu76221.23%2011.56%
pre-gitpre-git38810.81%3419.65%
jeremy fitzhardingejeremy fitzhardinge3589.97%126.94%
jan beulichjan beulich3178.83%42.31%
ingo molnaringo molnar2938.16%84.62%
dan alonidan aloni2095.82%10.58%
andi kleenandi kleen1564.35%42.31%
tejun heotejun heo1474.10%74.05%
steven rostedtsteven rostedt1193.32%10.58%
ian campbellian campbell1093.04%10.58%
suresh siddhasuresh siddha962.67%21.16%
matthieu castetmatthieu castet742.06%10.58%
pekka j enbergpekka j enberg711.98%63.47%
dave hansendave hansen581.62%21.16%
wen congyangwen congyang571.59%10.58%
brian gerstbrian gerst561.56%42.31%
arjan van de venarjan van de ven551.53%42.31%
andrew mortonandrew morton541.50%95.20%
zachary amsdenzachary amsden260.72%10.58%
linus torvaldslinus torvalds250.70%84.62%
richard hendersonrichard henderson170.47%10.58%
attilio raoattilio rao110.31%31.73%
jiang liujiang liu110.31%21.16%
stephen d. smalleystephen d. smalley90.25%10.58%
wang nanwang nan90.25%10.58%
yasunori gotoyasunori goto80.22%21.16%
rusty russellrusty russell70.20%10.58%
minfei huangminfei huang70.20%10.58%
h. peter anvinh. peter anvin60.17%31.73%
adrian bunkadrian bunk60.17%21.16%
ingo bruecklingo brueckl60.17%10.58%
andres salomonandres salomon60.17%10.58%
dan williamsdan williams50.14%10.58%
tang chentang chen50.14%10.58%
stanislaw gruszkastanislaw gruszka40.11%10.58%
borislav petkovborislav petkov40.11%10.58%
li shaohuali shaohua30.08%10.58%
zhao leizhao lei30.08%10.58%
eric sesterhenneric sesterhenn30.08%10.58%
harvey harrisonharvey harrison30.08%10.58%
randy dunlaprandy dunlap30.08%10.58%
huang yinghuang ying30.08%21.16%
andy whitcroftandy whitcroft30.08%10.58%
keith packardkeith packard30.08%10.58%
jaswinder singh rajputjaswinder singh rajput20.06%10.58%
tony lucktony luck20.06%10.58%
gary hadegary hade20.06%10.58%
dmitri vorobievdmitri vorobiev10.03%10.58%
ravikiran g thirumalairavikiran g thirumalai10.03%10.58%
william lee irwin iiiwilliam lee irwin iii10.03%10.58%
lucas de marchilucas de marchi10.03%10.58%
kamezawa hiroyukikamezawa hiroyuki10.03%10.58%
christoph lameterchristoph lameter10.03%10.58%
americo wangamerico wang10.03%10.58%
roman zippelroman zippel10.03%10.58%
Total3589100.00%173100.00%
Directory: arch/x86/mm
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.