URL
https://opencores.org/ocsvn/or1k/or1k/trunk
Subversion Repositories or1k
[/] [or1k/] [trunk/] [rc203soc/] [sw/] [uClinux/] [arch/] [m68k/] [mm/] [init.c] - Rev 1765
Compare with Previous | Blame | View Log
/* * linux/arch/m68k/mm/init.c * * Copyright (C) 1995 Hamish Macdonald */ #include <linux/config.h> #include <linux/signal.h> #include <linux/sched.h> #include <linux/mm.h> #include <linux/swap.h> #include <linux/kernel.h> #include <linux/string.h> #include <linux/types.h> #ifdef CONFIG_BLK_DEV_RAM #include <linux/blk.h> #endif #include <asm/segment.h> #include <asm/page.h> #include <asm/pgtable.h> #include <asm/system.h> #include <asm/bootinfo.h> #include <asm/machdep.h> extern void die_if_kernel(char *,struct pt_regs *,long); extern void init_kpointer_table(void); extern void show_net_buffers(void); extern unsigned long mm_phys_to_virt (unsigned long addr); extern char *rd_start; extern int rd_doload; unsigned long ramdisk_length; /* * BAD_PAGE is the page that is used for page faults when linux * is out-of-memory. Older versions of linux just did a * do_exit(), but using this instead means there is less risk * for a process dying in kernel mode, possibly leaving a inode * unused etc.. * * BAD_PAGETABLE is the accompanying page-table: it is initialized * to point to BAD_PAGE entries. * * ZERO_PAGE is a special page that is used for zero-initialized * data and COW. */ static unsigned long empty_bad_page_table; pte_t *__bad_pagetable(void) { memset((void *)empty_bad_page_table, 0, PAGE_SIZE); return (pte_t *)empty_bad_page_table; } static unsigned long empty_bad_page; pte_t __bad_page(void) { memset ((void *)empty_bad_page, 0, PAGE_SIZE); return pte_mkdirty(mk_pte(empty_bad_page, PAGE_SHARED)); } unsigned long empty_zero_page; void show_mem(void) { unsigned long i; int free = 0, total = 0, reserved = 0, nonshared = 0, shared = 0; printk("\nMem-info:\n"); show_free_areas(); printk("Free swap: %6dkB\n",nr_swap_pages<<(PAGE_SHIFT-10)); i = high_memory >> PAGE_SHIFT; while (i-- > 0) { total++; if (PageReserved(mem_map+i)) reserved++; else if (!mem_map[i].count) free++; else if (mem_map[i].count == 1) nonshared++; else shared += mem_map[i].count-1; } printk("%d pages of RAM\n",total); printk("%d free pages\n",free); printk("%d reserved pages\n",reserved); printk("%d pages nonshared\n",nonshared); printk("%d pages shared\n",shared); show_buffers(); #ifdef CONFIG_NET show_net_buffers(); #endif } #if 0 /* The 68030 doesn't care about reserved bits. */ /* * Bits to add to page descriptors for "normal" caching mode. * For 68020/030 this is 0. * For 68040, this is _PAGE_CACHE040 (cachable, copyback) */ unsigned long mm_cachebits; #endif pte_t *kernel_page_table (unsigned long *memavailp) { pte_t *ptablep; ptablep = (pte_t *)*memavailp; *memavailp += PAGE_SIZE; nocache_page ((unsigned long)ptablep); return ptablep; } static unsigned long map_chunk (unsigned long addr, unsigned long size, unsigned long *memavailp) { #define ONEMEG (1024*1024) #define L3TREESIZE (256*1024) int is040 = m68k_is040or060; static unsigned long mem_mapped = 0; static unsigned long virtaddr = 0; static pte_t *ktablep = NULL; unsigned long *kpointerp; unsigned long physaddr; extern pte_t *kpt; int pindex; /* index into pointer table */ pgd_t *page_dir = pgd_offset_k (virtaddr); if (!pgd_present (*page_dir)) { /* we need a new pointer table */ kpointerp = (unsigned long *) get_kpointer_table (); pgd_set (page_dir, (pmd_t *) kpointerp); memset (kpointerp, 0, PTRS_PER_PMD * sizeof (pmd_t)); } else kpointerp = (unsigned long *) pgd_page (*page_dir); /* * pindex is the offset into the pointer table for the * descriptors for the current virtual address being mapped. */ pindex = (virtaddr >> 18) & 0x7f; #ifdef DEBUG printk ("mm=%ld, kernel_pg_dir=%p, kpointerp=%p, pindex=%d\n", mem_mapped, kernel_pg_dir, kpointerp, pindex); #endif /* * if this is running on an '040, we already allocated a page * table for the first 4M. The address is stored in kpt by * arch/head.S * */ if (is040 && mem_mapped == 0) ktablep = kpt; for (physaddr = addr; physaddr < addr + size; mem_mapped += L3TREESIZE, virtaddr += L3TREESIZE) { #ifdef DEBUG printk ("pa=%#lx va=%#lx ", physaddr, virtaddr); #endif if (pindex > 127 && mem_mapped >= 32*ONEMEG) { /* we need a new pointer table every 32M */ #ifdef DEBUG printk ("[new pointer]"); #endif kpointerp = (unsigned long *)get_kpointer_table (); pgd_set(pgd_offset_k(virtaddr), (pmd_t *)kpointerp); pindex = 0; } if (is040) { int i; unsigned long ktable; /* Don't map the first 4 MB again. The pagetables * for this range have already been initialized * in boot/head.S. Otherwise the pages used for * tables would be reinitialized to copyback mode. */ if (mem_mapped < 4 * ONEMEG) { #ifdef DEBUG printk ("Already initialized\n"); #endif physaddr += L3TREESIZE; pindex++; continue; } #ifdef DEBUG printk ("[setup table]"); #endif /* * 68040, use page tables pointed to by the * kernel pointer table. */ if ((pindex & 15) == 0) { /* Need new page table every 4M on the '040 */ #ifdef DEBUG printk ("[new table]"); #endif ktablep = kernel_page_table (memavailp); } ktable = VTOP(ktablep); /* * initialize section of the page table mapping * this 256K portion. */ for (i = 0; i < 64; i++) { pte_val(ktablep[i]) = physaddr | _PAGE_PRESENT | _PAGE_CACHE040 | _PAGE_GLOBAL040; physaddr += PAGE_SIZE; } ktablep += 64; /* * make the kernel pointer table point to the * kernel page table. Each entries point to a * 64 entry section of the page table. */ kpointerp[pindex++] = ktable | _PAGE_TABLE; } else { /* * 68030, use early termination page descriptors. * Each one points to 64 pages (256K). */ #ifdef DEBUG printk ("[early term] "); #endif if (virtaddr == 0UL) { /* map the first 256K using a 64 entry * 3rd level page table. * UNMAP the first entry to trap * zero page (NULL pointer) references */ int i; unsigned long *tbl; tbl = (unsigned long *)get_kpointer_table(); kpointerp[pindex++] = VTOP(tbl) | _PAGE_TABLE; for (i = 0; i < 64; i++, physaddr += PAGE_SIZE) tbl[i] = physaddr | _PAGE_PRESENT; /* unmap the zero page */ tbl[0] = 0; } else { /* not the first 256K */ kpointerp[pindex++] = physaddr | _PAGE_PRESENT; #ifdef DEBUG printk ("%lx=%lx ", VTOP(&kpointerp[pindex-1]), kpointerp[pindex-1]); #endif physaddr += 64 * PAGE_SIZE; } } #ifdef DEBUG printk ("\n"); #endif } return mem_mapped; } extern unsigned long free_area_init(unsigned long, unsigned long); extern pgd_t swapper_pg_dir[PTRS_PER_PGD]; /* * paging_init() continues the virtual memory environment setup which * was begun by the code in arch/head.S. * The parameters are pointers to where to stick the starting and ending * addresses of available kernel virtual memory. */ unsigned long paging_init(unsigned long start_mem, unsigned long end_mem) { int chunk; unsigned long mem_avail = 0; /* pointer to page table for kernel stacks */ extern unsigned long availmem; #ifdef DEBUG { extern pte_t *kpt; printk ("start of paging_init (%p, %p, %lx, %lx, %lx)\n", kernel_pg_dir, kpt, availmem, start_mem, end_mem); } #endif init_kpointer_table(); #if 0 /* * Setup cache bits */ mm_cachebits = m68k_is040or060 ? _PAGE_CACHE040 : 0; /* Initialize protection map. */ protection_map[0] = PAGE_READONLY; protection_map[1] = PAGE_READONLY; protection_map[2] = PAGE_COPY; protection_map[3] = PAGE_COPY; protection_map[4] = PAGE_READONLY; protection_map[5] = PAGE_READONLY; protection_map[6] = PAGE_COPY; protection_map[7] = PAGE_COPY; protection_map[8] = PAGE_READONLY; protection_map[9] = PAGE_READONLY; protection_map[10] = PAGE_SHARED; protection_map[11] = PAGE_SHARED; protection_map[12] = PAGE_READONLY; protection_map[13] = PAGE_READONLY; protection_map[14] = PAGE_SHARED; protection_map[15] = PAGE_SHARED; #endif /* * Map the physical memory available into the kernel virtual * address space. It may allocate some memory for page * tables and thus modify availmem. */ for (chunk = 0; chunk < boot_info.num_memory; chunk++) { mem_avail = map_chunk (boot_info.memory[chunk].addr, boot_info.memory[chunk].size, &availmem); } flush_tlb_all(); #ifdef DEBUG printk ("memory available is %ldKB\n", mem_avail >> 10); #endif /* * virtual address after end of kernel * "availmem" is setup by the code in head.S. */ start_mem = availmem; #ifdef DEBUG printk ("start_mem is %#lx\nvirtual_end is %#lx\n", start_mem, end_mem); #endif /* * initialize the bad page table and bad page to point * to a couple of allocated pages */ empty_bad_page_table = start_mem; start_mem += PAGE_SIZE; empty_bad_page = start_mem; start_mem += PAGE_SIZE; empty_zero_page = start_mem; start_mem += PAGE_SIZE; memset((void *)empty_zero_page, 0, PAGE_SIZE); #if 0 /* * allocate the "swapper" page directory and * record in task 0 (swapper) tss */ swapper_pg_dir = (pgd_t *)get_kpointer_table(); init_mm.pgd = swapper_pg_dir; #endif memset (swapper_pg_dir, 0, sizeof(pgd_t)*PTRS_PER_PGD); task[0]->tss.pagedir_v = (unsigned long *)swapper_pg_dir; task[0]->tss.pagedir_p = VTOP (swapper_pg_dir); #ifdef DEBUG printk ("task 0 pagedir at %p virt, %#lx phys\n", task[0]->tss.pagedir_v, task[0]->tss.pagedir_p); #endif /* setup CPU root pointer for swapper task */ task[0]->tss.crp[0] = 0x80000000 | _PAGE_SHORT; task[0]->tss.crp[1] = task[0]->tss.pagedir_p; if (m68k_is040or060) asm ("movel %0,%/d0\n\t" ".long 0x4e7b0806" /* movec d0,urp */ : /* no outputs */ : "g" (task[0]->tss.crp[1]) : "d0"); else asm ("pmove %0@,%/crp" : /* no outputs */ : "a" (task[0]->tss.crp)); #ifdef DEBUG printk ("set crp\n"); #endif /* * Set up SFC/DFC registers (user data space) */ set_fs (USER_DS); #ifdef DEBUG printk ("before free_area_init\n"); #endif #ifdef CONFIG_BLK_DEV_RAM #ifndef CONFIG_BLK_DEV_INITRD /* * Since the initialization of the ramdisk's has been changed * so it fits the new driver initialization scheme, we have to * make room for our preloaded image here, instead of doing it * in rd_init() as we cannot kmalloc() a block large enough * for the image. */ ramdisk_length = boot_info.ramdisk_size * 1024; if ((ramdisk_length > 0) && (ROOT_DEV == 0)) { char *rdp; /* current location of ramdisk */ rd_start = (char *) start_mem; /* get current address of ramdisk */ rdp = (char *)mm_phys_to_virt (boot_info.ramdisk_addr); /* copy the ram disk image */ memcpy (rd_start, rdp, ramdisk_length); start_mem += ramdisk_length; rd_doload = 1; /* tell rd_load to load this thing */ } #endif #endif return free_area_init (start_mem, end_mem); } void mem_init(unsigned long start_mem, unsigned long end_mem) { int codepages = 0; int datapages = 0; unsigned long tmp; extern int _etext; end_mem &= PAGE_MASK; high_memory = end_mem; start_mem = PAGE_ALIGN(start_mem); while (start_mem < high_memory) { clear_bit(PG_reserved, &mem_map[MAP_NR(start_mem)].flags); start_mem += PAGE_SIZE; } #ifdef CONFIG_ATARI if (MACH_IS_ATARI) { /* If the page with physical address 0 isn't the first kernel * code page, it has to be reserved because the first 2 KB of * ST-Ram can only be accessed from supervisor mode by * hardware. */ unsigned long virt0 = PTOV( 0 ), adr; extern unsigned long rsvd_stram_beg, rsvd_stram_end; if (virt0 != 0) { set_bit(PG_reserved, &mem_map[MAP_NR(virt0)].flags); /* Also, reserve all pages that have been marked by * stram_alloc() (e.g. for the screen memory). (This may * treat the first ST-Ram page a second time, but that * doesn't hurt...) */ rsvd_stram_end += PAGE_SIZE - 1; rsvd_stram_end &= PAGE_MASK; rsvd_stram_beg &= PAGE_MASK; for( adr = rsvd_stram_beg; adr < rsvd_stram_end; adr += PAGE_SIZE ) set_bit(PG_reserved, &mem_map[MAP_NR(adr)].flags); } } #endif #ifdef DEBUG printk ("task[0] root table is %p\n", task[0]->tss.pagedir_v); #endif for (tmp = 0 ; tmp < end_mem ; tmp += PAGE_SIZE) { if (VTOP (tmp) >= mach_max_dma_address) clear_bit(PG_DMA, &mem_map[MAP_NR(tmp)].flags); if (PageReserved(mem_map+MAP_NR(tmp))) { if (tmp < (unsigned long)&_etext) codepages++; else datapages++; continue; } mem_map[MAP_NR(tmp)].count = 1; #ifdef CONFIG_BLK_DEV_INITRD if (!initrd_start || (tmp < initrd_start || tmp >= initrd_end)) #endif free_page(tmp); } tmp = nr_free_pages << PAGE_SHIFT; printk("Memory: %luk/%luk available (%dk kernel code, %dk data)\n", tmp >> 10, high_memory >> 10, codepages << (PAGE_SHIFT-10), datapages << (PAGE_SHIFT-10)); } void si_meminfo(struct sysinfo *val) { unsigned long i; i = high_memory >> PAGE_SHIFT; val->totalram = 0; val->sharedram = 0; val->freeram = nr_free_pages << PAGE_SHIFT; val->bufferram = buffermem; while (i-- > 0) { if (PageReserved(mem_map+i)) continue; val->totalram++; if (!mem_map[i].count) continue; val->sharedram += mem_map[i].count-1; } val->totalram <<= PAGE_SHIFT; val->sharedram <<= PAGE_SHIFT; return; }