Linux Kernel  3.7.1
 All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Groups Pages
mem.c
Go to the documentation of this file.
1 /*
2  * Copyright (C) 2000 - 2007 Jeff Dike (jdike@{addtoit,linux.intel}.com)
3  * Licensed under the GPL
4  */
5 
6 #include <linux/stddef.h>
7 #include <linux/module.h>
8 #include <linux/bootmem.h>
9 #include <linux/highmem.h>
10 #include <linux/mm.h>
11 #include <linux/swap.h>
12 #include <linux/slab.h>
13 #include <asm/fixmap.h>
14 #include <asm/page.h>
15 #include <as-layout.h>
16 #include <init.h>
17 #include <kern.h>
18 #include <kern_util.h>
19 #include <mem_user.h>
20 #include <os.h>
21 
22 /* allocated in paging_init, zeroed in mem_init, and unchanged thereafter */
23 unsigned long *empty_zero_page = NULL;
25 /* allocated in paging_init and unchanged thereafter */
26 static unsigned long *empty_bad_page = NULL;
27 
28 /*
29  * Initialized during boot, and readonly for initializing page tables
30  * afterwards
31  */
33 
34 /* Initialized at boot time, and readonly after that */
35 unsigned long long highmem;
36 int kmalloc_ok = 0;
37 
38 /* Used during early boot */
39 static unsigned long brk_end;
40 
41 #ifdef CONFIG_HIGHMEM
42 static void setup_highmem(unsigned long highmem_start,
43  unsigned long highmem_len)
44 {
45  struct page *page;
46  unsigned long highmem_pfn;
47  int i;
48 
49  highmem_pfn = __pa(highmem_start) >> PAGE_SHIFT;
50  for (i = 0; i < highmem_len >> PAGE_SHIFT; i++) {
51  page = &mem_map[highmem_pfn + i];
52  ClearPageReserved(page);
53  init_page_count(page);
54  __free_page(page);
55  }
56 }
57 #endif
58 
59 void __init mem_init(void)
60 {
61  /* clear the zero-page */
63 
64  /* Map in the area just after the brk now that kmalloc is about
65  * to be turned on.
66  */
67  brk_end = (unsigned long) UML_ROUND_UP(sbrk(0));
68  map_memory(brk_end, __pa(brk_end), uml_reserved - brk_end, 1, 1, 0);
69  free_bootmem(__pa(brk_end), uml_reserved - brk_end);
70  uml_reserved = brk_end;
71 
72  /* this will put all low memory onto the freelists */
73  totalram_pages = free_all_bootmem();
74  max_low_pfn = totalram_pages;
75 #ifdef CONFIG_HIGHMEM
77  totalram_pages += totalhigh_pages;
78 #endif
79  num_physpages = totalram_pages;
80  max_pfn = totalram_pages;
81  printk(KERN_INFO "Memory: %luk available\n",
82  nr_free_pages() << (PAGE_SHIFT-10));
83  kmalloc_ok = 1;
84 
85 #ifdef CONFIG_HIGHMEM
86  setup_highmem(end_iomem, highmem);
87 #endif
88 }
89 
90 /*
91  * Create a page table and place a pointer to it in a middle page
92  * directory entry.
93  */
94 static void __init one_page_table_init(pmd_t *pmd)
95 {
96  if (pmd_none(*pmd)) {
99  (unsigned long) __pa(pte)));
100  if (pte != pte_offset_kernel(pmd, 0))
101  BUG();
102  }
103 }
104 
105 static void __init one_md_table_init(pud_t *pud)
106 {
107 #ifdef CONFIG_3_LEVEL_PGTABLES
108  pmd_t *pmd_table = (pmd_t *) alloc_bootmem_low_pages(PAGE_SIZE);
109  set_pud(pud, __pud(_KERNPG_TABLE + (unsigned long) __pa(pmd_table)));
110  if (pmd_table != pmd_offset(pud, 0))
111  BUG();
112 #endif
113 }
114 
115 static void __init fixrange_init(unsigned long start, unsigned long end,
116  pgd_t *pgd_base)
117 {
118  pgd_t *pgd;
119  pud_t *pud;
120  pmd_t *pmd;
121  int i, j;
122  unsigned long vaddr;
123 
124  vaddr = start;
125  i = pgd_index(vaddr);
126  j = pmd_index(vaddr);
127  pgd = pgd_base + i;
128 
129  for ( ; (i < PTRS_PER_PGD) && (vaddr < end); pgd++, i++) {
130  pud = pud_offset(pgd, vaddr);
131  if (pud_none(*pud))
132  one_md_table_init(pud);
133  pmd = pmd_offset(pud, vaddr);
134  for (; (j < PTRS_PER_PMD) && (vaddr < end); pmd++, j++) {
135  one_page_table_init(pmd);
136  vaddr += PMD_SIZE;
137  }
138  j = 0;
139  }
140 }
141 
142 #ifdef CONFIG_HIGHMEM
143 pte_t *kmap_pte;
145 
146 #define kmap_get_fixmap_pte(vaddr) \
147  pte_offset_kernel(pmd_offset(pud_offset(pgd_offset_k(vaddr), (vaddr)),\
148  (vaddr)), (vaddr))
149 
150 static void __init kmap_init(void)
151 {
152  unsigned long kmap_vstart;
153 
154  /* cache the first kmap pte */
155  kmap_vstart = __fix_to_virt(FIX_KMAP_BEGIN);
156  kmap_pte = kmap_get_fixmap_pte(kmap_vstart);
157 
158  kmap_prot = PAGE_KERNEL;
159 }
160 
161 static void __init init_highmem(void)
162 {
163  pgd_t *pgd;
164  pud_t *pud;
165  pmd_t *pmd;
166  pte_t *pte;
167  unsigned long vaddr;
168 
169  /*
170  * Permanent kmaps:
171  */
172  vaddr = PKMAP_BASE;
173  fixrange_init(vaddr, vaddr + PAGE_SIZE*LAST_PKMAP, swapper_pg_dir);
174 
175  pgd = swapper_pg_dir + pgd_index(vaddr);
176  pud = pud_offset(pgd, vaddr);
177  pmd = pmd_offset(pud, vaddr);
178  pte = pte_offset_kernel(pmd, vaddr);
180 
181  kmap_init();
182 }
183 #endif /* CONFIG_HIGHMEM */
184 
185 static void __init fixaddr_user_init( void)
186 {
187 #ifdef CONFIG_ARCH_REUSE_HOST_VSYSCALL_AREA
189  pgd_t *pgd;
190  pud_t *pud;
191  pmd_t *pmd;
192  pte_t *pte;
193  phys_t p;
194  unsigned long v, vaddr = FIXADDR_USER_START;
195 
196  if (!size)
197  return;
198 
199  fixrange_init( FIXADDR_USER_START, FIXADDR_USER_END, swapper_pg_dir);
200  v = (unsigned long) alloc_bootmem_low_pages(size);
201  memcpy((void *) v , (void *) FIXADDR_USER_START, size);
202  p = __pa(v);
203  for ( ; size > 0; size -= PAGE_SIZE, vaddr += PAGE_SIZE,
204  p += PAGE_SIZE) {
205  pgd = swapper_pg_dir + pgd_index(vaddr);
206  pud = pud_offset(pgd, vaddr);
207  pmd = pmd_offset(pud, vaddr);
208  pte = pte_offset_kernel(pmd, vaddr);
209  pte_set_val(*pte, p, PAGE_READONLY);
210  }
211 #endif
212 }
213 
214 void __init paging_init(void)
215 {
216  unsigned long zones_size[MAX_NR_ZONES], vaddr;
217  int i;
218 
220  empty_bad_page = (unsigned long *) alloc_bootmem_low_pages(PAGE_SIZE);
221  for (i = 0; i < ARRAY_SIZE(zones_size); i++)
222  zones_size[i] = 0;
223 
224  zones_size[ZONE_NORMAL] = (end_iomem >> PAGE_SHIFT) -
225  (uml_physmem >> PAGE_SHIFT);
226 #ifdef CONFIG_HIGHMEM
227  zones_size[ZONE_HIGHMEM] = highmem >> PAGE_SHIFT;
228 #endif
229  free_area_init(zones_size);
230 
231  /*
232  * Fixed mappings, only the page table structure has to be
233  * created - mappings will be set by set_fixmap():
234  */
236  fixrange_init(vaddr, FIXADDR_TOP, swapper_pg_dir);
237 
238  fixaddr_user_init();
239 
240 #ifdef CONFIG_HIGHMEM
241  init_highmem();
242 #endif
243 }
244 
245 /*
246  * This can't do anything because nothing in the kernel image can be freed
247  * since it's not in kernel physical memory.
248  */
249 
250 void free_initmem(void)
251 {
252 }
253 
254 #ifdef CONFIG_BLK_DEV_INITRD
255 void free_initrd_mem(unsigned long start, unsigned long end)
256 {
257  if (start < end)
258  printk(KERN_INFO "Freeing initrd memory: %ldk freed\n",
259  (end - start) >> 10);
260  for (; start < end; start += PAGE_SIZE) {
261  ClearPageReserved(virt_to_page(start));
262  init_page_count(virt_to_page(start));
263  free_page(start);
264  totalram_pages++;
265  }
266 }
267 #endif
268 
269 /* Allocate and free page tables. */
270 
272 {
274 
275  if (pgd) {
276  memset(pgd, 0, USER_PTRS_PER_PGD * sizeof(pgd_t));
278  swapper_pg_dir + USER_PTRS_PER_PGD,
279  (PTRS_PER_PGD - USER_PTRS_PER_PGD) * sizeof(pgd_t));
280  }
281  return pgd;
282 }
283 
284 void pgd_free(struct mm_struct *mm, pgd_t *pgd)
285 {
286  free_page((unsigned long) pgd);
287 }
288 
289 pte_t *pte_alloc_one_kernel(struct mm_struct *mm, unsigned long address)
290 {
291  pte_t *pte;
292 
294  return pte;
295 }
296 
297 pgtable_t pte_alloc_one(struct mm_struct *mm, unsigned long address)
298 {
299  struct page *pte;
300 
302  if (pte)
303  pgtable_page_ctor(pte);
304  return pte;
305 }
306 
307 #ifdef CONFIG_3_LEVEL_PGTABLES
308 pmd_t *pmd_alloc_one(struct mm_struct *mm, unsigned long address)
309 {
311 
312  if (pmd)
313  memset(pmd, 0, PAGE_SIZE);
314 
315  return pmd;
316 }
317 #endif
318 
319 void *uml_kmalloc(int size, int flags)
320 {
321  return kmalloc(size, flags);
322 }