diff options
Diffstat (limited to 'arch/s390/mm/init.c')
| -rw-r--r-- | arch/s390/mm/init.c | 52 |
1 files changed, 46 insertions, 6 deletions
diff --git a/arch/s390/mm/init.c b/arch/s390/mm/init.c index 94b8ba2ec857..bb409332a484 100644 --- a/arch/s390/mm/init.c +++ b/arch/s390/mm/init.c | |||
| @@ -38,13 +38,54 @@ | |||
| 38 | #include <asm/tlbflush.h> | 38 | #include <asm/tlbflush.h> |
| 39 | #include <asm/sections.h> | 39 | #include <asm/sections.h> |
| 40 | 40 | ||
| 41 | DEFINE_PER_CPU(struct mmu_gather, mmu_gathers); | ||
| 42 | |||
| 43 | pgd_t swapper_pg_dir[PTRS_PER_PGD] __attribute__((__aligned__(PAGE_SIZE))); | 41 | pgd_t swapper_pg_dir[PTRS_PER_PGD] __attribute__((__aligned__(PAGE_SIZE))); |
| 44 | 42 | ||
| 45 | char empty_zero_page[PAGE_SIZE] __attribute__((__aligned__(PAGE_SIZE))); | 43 | unsigned long empty_zero_page, zero_page_mask; |
| 46 | EXPORT_SYMBOL(empty_zero_page); | 44 | EXPORT_SYMBOL(empty_zero_page); |
| 47 | 45 | ||
| 46 | static unsigned long setup_zero_pages(void) | ||
| 47 | { | ||
| 48 | struct cpuid cpu_id; | ||
| 49 | unsigned int order; | ||
| 50 | unsigned long size; | ||
| 51 | struct page *page; | ||
| 52 | int i; | ||
| 53 | |||
| 54 | get_cpu_id(&cpu_id); | ||
| 55 | switch (cpu_id.machine) { | ||
| 56 | case 0x9672: /* g5 */ | ||
| 57 | case 0x2064: /* z900 */ | ||
| 58 | case 0x2066: /* z900 */ | ||
| 59 | case 0x2084: /* z990 */ | ||
| 60 | case 0x2086: /* z990 */ | ||
| 61 | case 0x2094: /* z9-109 */ | ||
| 62 | case 0x2096: /* z9-109 */ | ||
| 63 | order = 0; | ||
| 64 | break; | ||
| 65 | case 0x2097: /* z10 */ | ||
| 66 | case 0x2098: /* z10 */ | ||
| 67 | default: | ||
| 68 | order = 2; | ||
| 69 | break; | ||
| 70 | } | ||
| 71 | |||
| 72 | empty_zero_page = __get_free_pages(GFP_KERNEL | __GFP_ZERO, order); | ||
| 73 | if (!empty_zero_page) | ||
| 74 | panic("Out of memory in setup_zero_pages"); | ||
| 75 | |||
| 76 | page = virt_to_page((void *) empty_zero_page); | ||
| 77 | split_page(page, order); | ||
| 78 | for (i = 1 << order; i > 0; i--) { | ||
| 79 | SetPageReserved(page); | ||
| 80 | page++; | ||
| 81 | } | ||
| 82 | |||
| 83 | size = PAGE_SIZE << order; | ||
| 84 | zero_page_mask = (size - 1) & PAGE_MASK; | ||
| 85 | |||
| 86 | return 1UL << order; | ||
| 87 | } | ||
| 88 | |||
| 48 | /* | 89 | /* |
| 49 | * paging_init() sets up the page tables | 90 | * paging_init() sets up the page tables |
| 50 | */ | 91 | */ |
| @@ -83,6 +124,7 @@ void __init paging_init(void) | |||
| 83 | #endif | 124 | #endif |
| 84 | max_zone_pfns[ZONE_NORMAL] = max_low_pfn; | 125 | max_zone_pfns[ZONE_NORMAL] = max_low_pfn; |
| 85 | free_area_init_nodes(max_zone_pfns); | 126 | free_area_init_nodes(max_zone_pfns); |
| 127 | fault_init(); | ||
| 86 | } | 128 | } |
| 87 | 129 | ||
| 88 | void __init mem_init(void) | 130 | void __init mem_init(void) |
| @@ -92,14 +134,12 @@ void __init mem_init(void) | |||
| 92 | max_mapnr = num_physpages = max_low_pfn; | 134 | max_mapnr = num_physpages = max_low_pfn; |
| 93 | high_memory = (void *) __va(max_low_pfn * PAGE_SIZE); | 135 | high_memory = (void *) __va(max_low_pfn * PAGE_SIZE); |
| 94 | 136 | ||
| 95 | /* clear the zero-page */ | ||
| 96 | memset(empty_zero_page, 0, PAGE_SIZE); | ||
| 97 | |||
| 98 | /* Setup guest page hinting */ | 137 | /* Setup guest page hinting */ |
| 99 | cmma_init(); | 138 | cmma_init(); |
| 100 | 139 | ||
| 101 | /* this will put all low memory onto the freelists */ | 140 | /* this will put all low memory onto the freelists */ |
| 102 | totalram_pages += free_all_bootmem(); | 141 | totalram_pages += free_all_bootmem(); |
| 142 | totalram_pages -= setup_zero_pages(); /* Setup zeroed pages. */ | ||
| 103 | 143 | ||
| 104 | reservedpages = 0; | 144 | reservedpages = 0; |
| 105 | 145 | ||
