2 * Copyright (C) 2000 - 2003 Jeff Dike (jdike@addtoit.com)
3 * Licensed under the GPL
7 #include "linux/rbtree.h"
8 #include "linux/slab.h"
9 #include "linux/vmalloc.h"
10 #include "linux/bootmem.h"
11 #include "linux/module.h"
12 #include "asm/types.h"
13 #include "asm/pgtable.h"
14 #include "kern_util.h"
15 #include "user_util.h"
16 #include "mode_kern.h"
29 struct list_head list
;
32 static struct rb_root phys_mappings
= RB_ROOT
;
34 static struct rb_node
**find_rb(void *virt
)
36 struct rb_node
**n
= &phys_mappings
.rb_node
;
40 d
= rb_entry(*n
, struct phys_desc
, rb
);
53 static struct phys_desc
*find_phys_mapping(void *virt
)
55 struct rb_node
**n
= find_rb(virt
);
60 return(rb_entry(*n
, struct phys_desc
, rb
));
63 static void insert_phys_mapping(struct phys_desc
*desc
)
65 struct rb_node
**n
= find_rb(desc
->virt
);
68 panic("Physical remapping for %p already present",
71 rb_link_node(&desc
->rb
, (*n
)->rb_parent
, n
);
72 rb_insert_color(&desc
->rb
, &phys_mappings
);
75 LIST_HEAD(descriptor_mappings
);
79 struct list_head list
;
80 struct list_head pages
;
83 static struct desc_mapping
*find_mapping(int fd
)
85 struct desc_mapping
*desc
;
86 struct list_head
*ele
;
88 list_for_each(ele
, &descriptor_mappings
){
89 desc
= list_entry(ele
, struct desc_mapping
, list
);
97 static struct desc_mapping
*descriptor_mapping(int fd
)
99 struct desc_mapping
*desc
;
101 desc
= find_mapping(fd
);
105 desc
= kmalloc(sizeof(*desc
), GFP_ATOMIC
);
109 *desc
= ((struct desc_mapping
)
111 .list
= LIST_HEAD_INIT(desc
->list
),
112 .pages
= LIST_HEAD_INIT(desc
->pages
) });
113 list_add(&desc
->list
, &descriptor_mappings
);
118 int physmem_subst_mapping(void *virt
, int fd
, __u64 offset
, int w
)
120 struct desc_mapping
*fd_maps
;
121 struct phys_desc
*desc
;
125 fd_maps
= descriptor_mapping(fd
);
130 desc
= find_phys_mapping(virt
);
132 panic("Address 0x%p is already substituted\n", virt
);
135 desc
= kmalloc(sizeof(*desc
), GFP_ATOMIC
);
139 *desc
= ((struct phys_desc
)
144 .list
= LIST_HEAD_INIT(desc
->list
) });
145 insert_phys_mapping(desc
);
147 list_add(&desc
->list
, &fd_maps
->pages
);
149 virt
= (void *) ((unsigned long) virt
& PAGE_MASK
);
150 err
= os_map_memory(virt
, fd
, offset
, PAGE_SIZE
, 1, w
, 0);
154 rb_erase(&desc
->rb
, &phys_mappings
);
160 static int physmem_fd
= -1;
162 static void remove_mapping(struct phys_desc
*desc
)
164 void *virt
= desc
->virt
;
167 rb_erase(&desc
->rb
, &phys_mappings
);
168 list_del(&desc
->list
);
171 err
= os_map_memory(virt
, physmem_fd
, __pa(virt
), PAGE_SIZE
, 1, 1, 0);
173 panic("Failed to unmap block device page from physical memory, "
177 int physmem_remove_mapping(void *virt
)
179 struct phys_desc
*desc
;
181 virt
= (void *) ((unsigned long) virt
& PAGE_MASK
);
182 desc
= find_phys_mapping(virt
);
186 remove_mapping(desc
);
190 void physmem_forget_descriptor(int fd
)
192 struct desc_mapping
*desc
;
193 struct phys_desc
*page
;
194 struct list_head
*ele
, *next
;
199 desc
= find_mapping(fd
);
203 list_for_each_safe(ele
, next
, &desc
->pages
){
204 page
= list_entry(ele
, struct phys_desc
, list
);
205 offset
= page
->offset
;
207 remove_mapping(page
);
208 err
= os_seek_file(fd
, offset
);
210 panic("physmem_forget_descriptor - failed to seek "
211 "to %lld in fd %d, error = %d\n",
213 err
= os_read_file(fd
, addr
, PAGE_SIZE
);
215 panic("physmem_forget_descriptor - failed to read "
216 "from fd %d to 0x%p, error = %d\n",
220 list_del(&desc
->list
);
224 EXPORT_SYMBOL(physmem_forget_descriptor
);
225 EXPORT_SYMBOL(physmem_remove_mapping
);
226 EXPORT_SYMBOL(physmem_subst_mapping
);
228 void arch_free_page(struct page
*page
, int order
)
233 for(i
= 0; i
< (1 << order
); i
++){
234 virt
= __va(page_to_phys(page
+ i
));
235 physmem_remove_mapping(virt
);
239 int is_remapped(void *virt
)
241 struct phys_desc
*desc
= find_phys_mapping(virt
);
243 return(desc
!= NULL
);
246 /* Changed during early boot */
247 unsigned long high_physmem
;
249 extern unsigned long physmem_size
;
251 int init_maps(unsigned long physmem
, unsigned long iomem
, unsigned long highmem
)
253 struct page
*p
, *map
;
254 unsigned long phys_len
, phys_pages
, highmem_len
, highmem_pages
;
255 unsigned long iomem_len
, iomem_pages
, total_len
, total_pages
;
258 phys_pages
= physmem
>> PAGE_SHIFT
;
259 phys_len
= phys_pages
* sizeof(struct page
);
261 iomem_pages
= iomem
>> PAGE_SHIFT
;
262 iomem_len
= iomem_pages
* sizeof(struct page
);
264 highmem_pages
= highmem
>> PAGE_SHIFT
;
265 highmem_len
= highmem_pages
* sizeof(struct page
);
267 total_pages
= phys_pages
+ iomem_pages
+ highmem_pages
;
268 total_len
= phys_len
+ iomem_pages
+ highmem_len
;
271 map
= kmalloc(total_len
, GFP_KERNEL
);
273 map
= vmalloc(total_len
);
275 else map
= alloc_bootmem_low_pages(total_len
);
280 for(i
= 0; i
< total_pages
; i
++){
282 set_page_count(p
, 0);
284 INIT_LIST_HEAD(&p
->lru
);
287 max_mapnr
= total_pages
;
291 /* Changed during early boot */
292 static unsigned long kmem_top
= 0;
294 unsigned long get_kmem_end(void)
297 kmem_top
= CHOOSE_MODE(kmem_end_tt
, kmem_end_skas
);
301 void map_memory(unsigned long virt
, unsigned long phys
, unsigned long len
,
307 fd
= phys_mapping(phys
, &offset
);
308 err
= os_map_memory((void *) virt
, fd
, offset
, len
, r
, w
, x
);
311 printk("try increasing the host's "
312 "/proc/sys/vm/max_map_count to <physical "
313 "memory size>/4096\n");
314 panic("map_memory(0x%lx, %d, 0x%llx, %ld, %d, %d, %d) failed, "
315 "err = %d\n", virt
, fd
, offset
, len
, r
, w
, x
, err
);
319 #define PFN_UP(x) (((x) + PAGE_SIZE-1) >> PAGE_SHIFT)
321 extern int __syscall_stub_start
, __binary_start
;
323 void setup_physmem(unsigned long start
, unsigned long reserve_end
,
324 unsigned long len
, unsigned long highmem
)
326 unsigned long reserve
= reserve_end
- start
;
327 int pfn
= PFN_UP(__pa(reserve_end
));
328 int delta
= (len
- reserve
) >> PAGE_SHIFT
;
329 int err
, offset
, bootmap_size
;
331 physmem_fd
= create_mem_file(len
+ highmem
);
333 offset
= uml_reserved
- uml_physmem
;
334 err
= os_map_memory((void *) uml_reserved
, physmem_fd
, offset
,
335 len
- offset
, 1, 1, 0);
337 os_print_error(err
, "Mapping memory");
341 /* Special kludge - This page will be mapped in to userspace processes
342 * from physmem_fd, so it needs to be written out there.
344 os_seek_file(physmem_fd
, __pa(&__syscall_stub_start
));
345 os_write_file(physmem_fd
, &__syscall_stub_start
, PAGE_SIZE
);
347 bootmap_size
= init_bootmem(pfn
, pfn
+ delta
);
348 free_bootmem(__pa(reserve_end
) + bootmap_size
,
349 len
- bootmap_size
- reserve
);
352 int phys_mapping(unsigned long phys
, __u64
*offset_out
)
354 struct phys_desc
*desc
= find_phys_mapping(__va(phys
& PAGE_MASK
));
359 *offset_out
= desc
->offset
;
361 else if(phys
< physmem_size
){
365 else if(phys
< __pa(end_iomem
)){
366 struct iomem_region
*region
= iomem_regions
;
368 while(region
!= NULL
){
369 if((phys
>= region
->phys
) &&
370 (phys
< region
->phys
+ region
->size
)){
372 *offset_out
= phys
- region
->phys
;
375 region
= region
->next
;
378 else if(phys
< __pa(end_iomem
) + highmem
){
380 *offset_out
= phys
- iomem_size
;
386 static int __init
uml_mem_setup(char *line
, int *add
)
389 physmem_size
= memparse(line
,&retptr
);
392 __uml_setup("mem=", uml_mem_setup
,
393 "mem=<Amount of desired ram>\n"
394 " This controls how much \"physical\" memory the kernel allocates\n"
395 " for the system. The size is specified as a number followed by\n"
396 " one of 'k', 'K', 'm', 'M', which have the obvious meanings.\n"
397 " This is not related to the amount of memory in the host. It can\n"
398 " be more, and the excess, if it's ever used, will just be swapped out.\n"
399 " Example: mem=64M\n\n"
402 unsigned long find_iomem(char *driver
, unsigned long *len_out
)
404 struct iomem_region
*region
= iomem_regions
;
406 while(region
!= NULL
){
407 if(!strcmp(region
->driver
, driver
)){
408 *len_out
= region
->size
;
409 return(region
->virt
);
416 int setup_iomem(void)
418 struct iomem_region
*region
= iomem_regions
;
419 unsigned long iomem_start
= high_physmem
+ PAGE_SIZE
;
422 while(region
!= NULL
){
423 err
= os_map_memory((void *) iomem_start
, region
->fd
, 0,
424 region
->size
, 1, 1, 0);
426 printk("Mapping iomem region for driver '%s' failed, "
427 "errno = %d\n", region
->driver
, -err
);
429 region
->virt
= iomem_start
;
430 region
->phys
= __pa(region
->virt
);
433 iomem_start
+= region
->size
+ PAGE_SIZE
;
434 region
= region
->next
;
440 __initcall(setup_iomem
);
443 * Overrides for Emacs so that we follow Linus's tabbing style.
444 * Emacs will notice this stuff at the end of the file and automatically
445 * adjust the settings for this buffer only. This must remain at the end
447 * ---------------------------------------------------------------------------
449 * c-file-style: "linux"