From 0c628976d0fbfd460559bc5654289f27107a4271 Mon Sep 17 00:00:00 2001 From: Christoph Lameter Date: Tue, 6 Nov 2007 11:33:49 -0800 Subject: [PATCH] cpu alloc: x86_64 support Set up a cpu area that allows the use of up 256MB for each processor. Cpu memory use can grow rapidly. F.e. if we assume that a page struct occupies 64 bytes of memory and we have 3 zones per node then we need 3 * 1k * 16k = 50 million pagesets or 3096 pagesets per processor. This results in a total of 3.2 GB of page structs. So each cpu needs around 200k of cpu storage for the page allocator alone. For the UP and SMP case map the area using 4k ptes. Typical use of per cpu data is around 16k for UP and SMP configurations. Allocating a 2M segment would be overkill. There is enough reserve around to run a few 100000 ip tunnels if one wants to and other lots of other frills. For NUMA map the area using 2M PMDs. A large NUMA system may use lots of cpu data for the page allocator data alone. We typically have large amounts of memory around. Using a 2M page size reduces TLB pressure. Some numbers for envisioned maximum configurations of NUMA systems: 4k cpu configurations with 256 nodes: 4096 * 256MB = 1TB of virtual space. Maximum theoretical configuration 16384 processors 1k nodes: 16384 * 256MB = 4TB of virtual space. Both fit within the established limits established. Signed-off-by: Christoph Lameter --- arch/x86/Kconfig.x86_64 | 13 +++++++++++++ arch/x86/mm/init_64.c | 39 +++++++++++++++++++++++++++++++++++++++ include/asm-x86/pgtable_64.h | 1 + 3 files changed, 53 insertions(+) Index: linux-2.6.24-rc2-mm1/arch/x86/Kconfig.x86_64 =================================================================== --- linux-2.6.24-rc2-mm1.orig/arch/x86/Kconfig.x86_64 2007-11-14 11:08:31.464511611 -0800 +++ linux-2.6.24-rc2-mm1/arch/x86/Kconfig.x86_64 2007-11-14 16:43:09.966070546 -0800 @@ -137,6 +137,19 @@ config ARCH_HAS_ILOG2_U64 bool default n +config CPU_AREA_VIRTUAL + bool + default y + +config CPU_AREA_ORDER + int + default "16" + +config CPU_AREA_ALLOC_ORDER + int + default "9" if NUMA + default "0" if !NUMA + source "init/Kconfig" Index: linux-2.6.24-rc2-mm1/arch/x86/mm/init_64.c =================================================================== --- linux-2.6.24-rc2-mm1.orig/arch/x86/mm/init_64.c 2007-11-14 11:08:31.908011496 -0800 +++ linux-2.6.24-rc2-mm1/arch/x86/mm/init_64.c 2007-11-14 16:43:09.966070546 -0800 @@ -783,3 +784,41 @@ int __meminit vmemmap_populate(struct pa return 0; } #endif + +#ifdef CONFIG_NUMA +int __meminit cpu_area_populate(void *start, unsigned long size, + gfp_t flags, int node) +{ + unsigned long addr = (unsigned long)start; + unsigned long end = addr + size; + unsigned long next; + pgd_t *pgd; + pud_t *pud; + pmd_t *pmd; + + for (; addr < end; addr = next) { + next = pmd_addr_end(addr, end); + + pgd = cpu_area_pgd_populate(addr, flags, node); + if (!pgd) + return -ENOMEM; + pud = cpu_area_pud_populate(pgd, addr, flags, node); + if (!pud) + return -ENOMEM; + + pmd = pmd_offset(pud, addr); + if (pmd_none(*pmd)) { + pte_t entry; + void *p = cpu_area_alloc_block(PMD_SIZE, flags, node); + if (!p) + return -ENOMEM; + + entry = pfn_pte(__pa(p) >> PAGE_SHIFT, PAGE_KERNEL); + mk_pte_huge(entry); + set_pmd(pmd, __pmd(pte_val(entry))); + } + } + + return 0; +} +#endif Index: linux-2.6.24-rc2-mm1/include/asm-x86/pgtable_64.h =================================================================== --- linux-2.6.24-rc2-mm1.orig/include/asm-x86/pgtable_64.h 2007-11-14 11:09:18.272011376 -0800 +++ linux-2.6.24-rc2-mm1/include/asm-x86/pgtable_64.h 2007-11-14 16:43:09.966070546 -0800 @@ -138,6 +138,7 @@ static inline pte_t ptep_get_and_clear_f #define VMALLOC_START _AC(0xffffc20000000000, UL) #define VMALLOC_END _AC(0xffffe1ffffffffff, UL) #define VMEMMAP_START _AC(0xffffe20000000000, UL) +#define CPU_AREA_BASE _AC(0xfffff20000000000, UL) #define MODULES_VADDR _AC(0xffffffff88000000, UL) #define MODULES_END _AC(0xfffffffffff00000, UL) #define MODULES_LEN (MODULES_END - MODULES_VADDR)