2005-04-16 15:20:36 -07:00
# include <linux/sched.h>
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/mm.h>
2007-10-17 18:04:34 +02:00
# include <linux/nmi.h>
2005-04-16 15:20:36 -07:00
# include <linux/swap.h>
# include <linux/smp.h>
# include <linux/highmem.h>
# include <linux/slab.h>
# include <linux/pagemap.h>
# include <linux/spinlock.h>
2006-09-25 23:32:25 -07:00
# include <linux/module.h>
2007-05-12 11:15:24 -07:00
# include <linux/quicklist.h>
2005-04-16 15:20:36 -07:00
# include <asm/system.h>
# include <asm/pgtable.h>
# include <asm/pgalloc.h>
# include <asm/fixmap.h>
# include <asm/e820.h>
# include <asm/tlb.h>
# include <asm/tlbflush.h>
2009-03-03 12:55:04 +02:00
unsigned int __VMALLOC_RESERVE = 128 < < 20 ;
2005-04-16 15:20:36 -07:00
/*
* Associate a virtual page frame with a given physical page frame
* and protection flags for that frame .
*/
2008-06-17 11:41:59 -07:00
void set_pte_vaddr ( unsigned long vaddr , pte_t pteval )
2005-04-16 15:20:36 -07:00
{
pgd_t * pgd ;
pud_t * pud ;
pmd_t * pmd ;
pte_t * pte ;
pgd = swapper_pg_dir + pgd_index ( vaddr ) ;
if ( pgd_none ( * pgd ) ) {
BUG ( ) ;
return ;
}
pud = pud_offset ( pgd , vaddr ) ;
if ( pud_none ( * pud ) ) {
BUG ( ) ;
return ;
}
pmd = pmd_offset ( pud , vaddr ) ;
if ( pmd_none ( * pmd ) ) {
BUG ( ) ;
return ;
}
pte = pte_offset_kernel ( pmd , vaddr ) ;
2008-06-17 11:41:59 -07:00
if ( pte_val ( pteval ) )
2009-03-18 13:03:32 -07:00
set_pte_at ( & init_mm , vaddr , pte , pteval ) ;
2006-12-07 02:14:09 +01:00
else
pte_clear ( & init_mm , vaddr , pte ) ;
2005-04-16 15:20:36 -07:00
/*
* It ' s enough to flush this one mapping .
* ( PGE mappings get flushed as well )
*/
__flush_tlb_one ( vaddr ) ;
}
/*
* Associate a large virtual page frame with a given physical page frame
* and protection flags for that frame . pfn is for the base of the page ,
* vaddr is what the page gets mapped to - both must be properly aligned .
* The pmd must already be instantiated . Assumes PAE mode .
*/
void set_pmd_pfn ( unsigned long vaddr , unsigned long pfn , pgprot_t flags )
{
pgd_t * pgd ;
pud_t * pud ;
pmd_t * pmd ;
if ( vaddr & ( PMD_SIZE - 1 ) ) { /* vaddr is misaligned */
2005-06-25 14:59:24 -07:00
printk ( KERN_WARNING " set_pmd_pfn: vaddr misaligned \n " ) ;
2005-04-16 15:20:36 -07:00
return ; /* BUG(); */
}
if ( pfn & ( PTRS_PER_PTE - 1 ) ) { /* pfn is misaligned */
2005-06-25 14:59:24 -07:00
printk ( KERN_WARNING " set_pmd_pfn: pfn misaligned \n " ) ;
2005-04-16 15:20:36 -07:00
return ; /* BUG(); */
}
pgd = swapper_pg_dir + pgd_index ( vaddr ) ;
if ( pgd_none ( * pgd ) ) {
2005-06-25 14:59:24 -07:00
printk ( KERN_WARNING " set_pmd_pfn: pgd_none \n " ) ;
2005-04-16 15:20:36 -07:00
return ; /* BUG(); */
}
pud = pud_offset ( pgd , vaddr ) ;
pmd = pmd_offset ( pud , vaddr ) ;
set_pmd ( pmd , pfn_pmd ( pfn , flags ) ) ;
/*
* It ' s enough to flush this one mapping .
* ( PGE mappings get flushed as well )
*/
__flush_tlb_one ( vaddr ) ;
}
2006-09-25 23:32:25 -07:00
unsigned long __FIXADDR_TOP = 0xfffff000 ;
EXPORT_SYMBOL ( __FIXADDR_TOP ) ;
2008-06-22 17:40:10 -07:00
/*
* vmalloc = size forces the vmalloc area to be exactly ' size '
* bytes . This can be used to increase ( or decrease ) the
* vmalloc area - the default is 128 m .
*/
static int __init parse_vmalloc ( char * arg )
{
if ( ! arg )
return - EINVAL ;
2008-08-20 16:43:03 -07:00
/* Add VMALLOC_OFFSET to the parsed value due to vm area guard hole*/
__VMALLOC_RESERVE = memparse ( arg , & arg ) + VMALLOC_OFFSET ;
2008-06-22 17:40:10 -07:00
return 0 ;
}
early_param ( " vmalloc " , parse_vmalloc ) ;
/*
* reservetop = size reserves a hole at the top of the kernel address space which
* a hypervisor can load into later . Needed for dynamically loaded hypervisors ,
* so relocating the fixmap can be done before paging initialization .
*/
static int __init parse_reservetop ( char * arg )
{
unsigned long address ;
if ( ! arg )
return - EINVAL ;
address = memparse ( arg , & arg ) ;
reserve_top_address ( address ) ;
return 0 ;
}
early_param ( " reservetop " , parse_reservetop ) ;