2005-04-16 15:20:36 -07:00
/*
* arch / parisc / mm / ioremap . c
*
* ( C ) Copyright 1995 1996 Linus Torvalds
2006-03-27 19:52:14 +00:00
* ( C ) Copyright 2001 - 2006 Helge Deller < deller @ gmx . de >
2006-01-10 20:47:52 -05:00
* ( C ) Copyright 2005 Kyle McMartin < kyle @ parisc - linux . org >
2005-04-16 15:20:36 -07:00
*/
# include <linux/vmalloc.h>
# include <linux/errno.h>
# include <linux/module.h>
2006-12-08 02:38:05 -08:00
# include <linux/io.h>
2005-04-16 15:20:36 -07:00
# include <asm/pgalloc.h>
/*
* Generic mapping function ( not visible outside ) :
*/
/*
* Remap an arbitrary physical address space into the kernel virtual
2006-01-10 20:47:52 -05:00
* address space .
2005-04-16 15:20:36 -07:00
*
* NOTE ! We need to allow non - page - aligned mappings too : we will obviously
* have to convert them into an offset in a page - aligned mapping , but the
* caller shouldn ' t need to know that small detail .
*/
void __iomem * __ioremap ( unsigned long phys_addr , unsigned long size , unsigned long flags )
{
2006-11-05 15:24:48 -07:00
void __iomem * addr ;
2006-03-23 00:40:10 -07:00
struct vm_struct * area ;
unsigned long offset , last_addr ;
2006-12-08 02:38:05 -08:00
pgprot_t pgprot ;
2006-03-23 00:40:10 -07:00
2006-03-23 00:32:46 -07:00
# ifdef CONFIG_EISA
2005-04-16 15:20:36 -07:00
unsigned long end = phys_addr + size - 1 ;
/* Support EISA addresses */
2006-03-26 01:54:16 -07:00
if ( ( phys_addr > = 0x00080000 & & end < 0x000fffff ) | |
( phys_addr > = 0x00500000 & & end < 0x03bfffff ) ) {
phys_addr | = F_EXTEND ( 0xfc000000 ) ;
2006-03-27 19:52:14 +00:00
flags | = _PAGE_NO_CACHE ;
2005-04-16 15:20:36 -07:00
}
# endif
/* Don't allow wraparound or zero size */
last_addr = phys_addr + size - 1 ;
if ( ! size | | last_addr < phys_addr )
return NULL ;
/*
* Don ' t allow anybody to remap normal RAM that we ' re using . .
*/
if ( phys_addr < virt_to_phys ( high_memory ) ) {
char * t_addr , * t_end ;
struct page * page ;
t_addr = __va ( phys_addr ) ;
t_end = t_addr + ( size - 1 ) ;
2006-01-10 20:47:52 -05:00
for ( page = virt_to_page ( t_addr ) ;
page < = virt_to_page ( t_end ) ; page + + ) {
2005-04-16 15:20:36 -07:00
if ( ! PageReserved ( page ) )
return NULL ;
2006-01-10 20:47:52 -05:00
}
2005-04-16 15:20:36 -07:00
}
2006-12-08 02:38:05 -08:00
pgprot = __pgprot ( _PAGE_PRESENT | _PAGE_RW | _PAGE_DIRTY |
_PAGE_ACCESSED | flags ) ;
2005-04-16 15:20:36 -07:00
/*
* Mappings have to be page - aligned
*/
offset = phys_addr & ~ PAGE_MASK ;
phys_addr & = PAGE_MASK ;
size = PAGE_ALIGN ( last_addr ) - phys_addr ;
/*
* Ok , go for it . .
*/
area = get_vm_area ( size , VM_IOREMAP ) ;
if ( ! area )
return NULL ;
2006-01-10 20:47:52 -05:00
2006-11-05 15:24:48 -07:00
addr = ( void __iomem * ) area - > addr ;
2006-12-08 02:38:05 -08:00
if ( ioremap_page_range ( ( unsigned long ) addr , ( unsigned long ) addr + size ,
phys_addr , pgprot ) ) {
2005-04-16 15:20:36 -07:00
vfree ( addr ) ;
return NULL ;
}
2006-01-10 20:47:52 -05:00
2006-11-05 15:24:48 -07:00
return ( void __iomem * ) ( offset + ( char __iomem * ) addr ) ;
2005-04-16 15:20:36 -07:00
}
2006-03-29 15:18:32 -07:00
EXPORT_SYMBOL ( __ioremap ) ;
2005-04-16 15:20:36 -07:00
2006-09-19 16:37:01 -06:00
void iounmap ( const volatile void __iomem * addr )
2005-04-16 15:20:36 -07:00
{
if ( addr > high_memory )
return vfree ( ( void * ) ( PAGE_MASK & ( unsigned long __force ) addr ) ) ;
}
2006-03-29 15:18:32 -07:00
EXPORT_SYMBOL ( iounmap ) ;