2005-04-17 02:20:36 +04:00
/*
* Extensible Firmware Interface
*
2008-02-05 02:31:49 +03:00
* Based on Extensible Firmware Interface Specification version 0.9
* April 30 , 1999
2005-04-17 02:20:36 +04:00
*
* Copyright ( C ) 1999 VA Linux Systems
* Copyright ( C ) 1999 Walt Drummond < drummond @ valinux . com >
* Copyright ( C ) 1999 - 2003 Hewlett - Packard Co .
* David Mosberger - Tang < davidm @ hpl . hp . com >
* Stephane Eranian < eranian @ hpl . hp . com >
2006-05-06 03:19:50 +04:00
* ( c ) Copyright 2006 Hewlett - Packard Development Company , L . P .
* Bjorn Helgaas < bjorn . helgaas @ hp . com >
2005-04-17 02:20:36 +04:00
*
* All EFI Runtime Services are not implemented yet as EFI only
* supports physical mode addressing on SoftSDV . This is to be fixed
* in a future version . - - drummond 1999 - 07 - 20
*
* Implemented EFI runtime services and virtual mode calls . - - davidm
*
* Goutham Rao : < goutham . rao @ intel . com >
* Skip non - WB memory and ignore empty memory ranges .
*/
# include <linux/module.h>
2007-03-06 13:34:21 +03:00
# include <linux/bootmem.h>
2005-04-17 02:20:36 +04:00
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/types.h>
# include <linux/time.h>
# include <linux/efi.h>
2006-12-07 20:51:35 +03:00
# include <linux/kexec.h>
2007-07-17 15:03:14 +04:00
# include <linux/mm.h>
2005-04-17 02:20:36 +04:00
# include <asm/io.h>
# include <asm/kregs.h>
# include <asm/meminit.h>
# include <asm/pgtable.h>
# include <asm/processor.h>
# include <asm/mca.h>
# define EFI_DEBUG 0
extern efi_status_t efi_call_phys ( void * , . . . ) ;
struct efi efi ;
EXPORT_SYMBOL ( efi ) ;
static efi_runtime_services_t * runtime ;
2006-12-07 20:51:35 +03:00
static unsigned long mem_limit = ~ 0UL , max_addr = ~ 0UL , min_addr = 0UL ;
2005-04-17 02:20:36 +04:00
# define efi_call_virt(f, args...) (*(f))(args)
2008-02-05 02:31:49 +03:00
# define STUB_GET_TIME(prefix, adjust_arg) \
static efi_status_t \
prefix # # _get_time ( efi_time_t * tm , efi_time_cap_t * tc ) \
{ \
struct ia64_fpreg fr [ 6 ] ; \
efi_time_cap_t * atc = NULL ; \
efi_status_t ret ; \
\
if ( tc ) \
atc = adjust_arg ( tc ) ; \
ia64_save_scratch_fpregs ( fr ) ; \
ret = efi_call_ # # prefix ( ( efi_get_time_t * ) __va ( runtime - > get_time ) , \
adjust_arg ( tm ) , atc ) ; \
ia64_load_scratch_fpregs ( fr ) ; \
return ret ; \
2005-04-17 02:20:36 +04:00
}
2008-02-05 02:31:49 +03:00
# define STUB_SET_TIME(prefix, adjust_arg) \
static efi_status_t \
prefix # # _set_time ( efi_time_t * tm ) \
{ \
struct ia64_fpreg fr [ 6 ] ; \
efi_status_t ret ; \
\
ia64_save_scratch_fpregs ( fr ) ; \
ret = efi_call_ # # prefix ( ( efi_set_time_t * ) __va ( runtime - > set_time ) , \
adjust_arg ( tm ) ) ; \
ia64_load_scratch_fpregs ( fr ) ; \
return ret ; \
2005-04-17 02:20:36 +04:00
}
2008-02-05 02:31:49 +03:00
# define STUB_GET_WAKEUP_TIME(prefix, adjust_arg) \
static efi_status_t \
prefix # # _get_wakeup_time ( efi_bool_t * enabled , efi_bool_t * pending , \
efi_time_t * tm ) \
{ \
struct ia64_fpreg fr [ 6 ] ; \
efi_status_t ret ; \
\
ia64_save_scratch_fpregs ( fr ) ; \
ret = efi_call_ # # prefix ( \
( efi_get_wakeup_time_t * ) __va ( runtime - > get_wakeup_time ) , \
adjust_arg ( enabled ) , adjust_arg ( pending ) , adjust_arg ( tm ) ) ; \
ia64_load_scratch_fpregs ( fr ) ; \
return ret ; \
2005-04-17 02:20:36 +04:00
}
2008-02-05 02:31:49 +03:00
# define STUB_SET_WAKEUP_TIME(prefix, adjust_arg) \
static efi_status_t \
prefix # # _set_wakeup_time ( efi_bool_t enabled , efi_time_t * tm ) \
{ \
struct ia64_fpreg fr [ 6 ] ; \
efi_time_t * atm = NULL ; \
efi_status_t ret ; \
\
if ( tm ) \
atm = adjust_arg ( tm ) ; \
ia64_save_scratch_fpregs ( fr ) ; \
ret = efi_call_ # # prefix ( \
( efi_set_wakeup_time_t * ) __va ( runtime - > set_wakeup_time ) , \
enabled , atm ) ; \
ia64_load_scratch_fpregs ( fr ) ; \
return ret ; \
2005-04-17 02:20:36 +04:00
}
2008-02-05 02:31:49 +03:00
# define STUB_GET_VARIABLE(prefix, adjust_arg) \
static efi_status_t \
prefix # # _get_variable ( efi_char16_t * name , efi_guid_t * vendor , u32 * attr , \
unsigned long * data_size , void * data ) \
{ \
struct ia64_fpreg fr [ 6 ] ; \
u32 * aattr = NULL ; \
efi_status_t ret ; \
\
if ( attr ) \
aattr = adjust_arg ( attr ) ; \
ia64_save_scratch_fpregs ( fr ) ; \
ret = efi_call_ # # prefix ( \
( efi_get_variable_t * ) __va ( runtime - > get_variable ) , \
adjust_arg ( name ) , adjust_arg ( vendor ) , aattr , \
adjust_arg ( data_size ) , adjust_arg ( data ) ) ; \
ia64_load_scratch_fpregs ( fr ) ; \
return ret ; \
2005-04-17 02:20:36 +04:00
}
2008-02-05 02:31:49 +03:00
# define STUB_GET_NEXT_VARIABLE(prefix, adjust_arg) \
static efi_status_t \
prefix # # _get_next_variable ( unsigned long * name_size , efi_char16_t * name , \
efi_guid_t * vendor ) \
{ \
struct ia64_fpreg fr [ 6 ] ; \
efi_status_t ret ; \
\
ia64_save_scratch_fpregs ( fr ) ; \
ret = efi_call_ # # prefix ( \
( efi_get_next_variable_t * ) __va ( runtime - > get_next_variable ) , \
adjust_arg ( name_size ) , adjust_arg ( name ) , adjust_arg ( vendor ) ) ; \
ia64_load_scratch_fpregs ( fr ) ; \
return ret ; \
2005-04-17 02:20:36 +04:00
}
2008-02-05 02:31:49 +03:00
# define STUB_SET_VARIABLE(prefix, adjust_arg) \
static efi_status_t \
prefix # # _set_variable ( efi_char16_t * name , efi_guid_t * vendor , \
unsigned long attr , unsigned long data_size , \
void * data ) \
{ \
struct ia64_fpreg fr [ 6 ] ; \
efi_status_t ret ; \
\
ia64_save_scratch_fpregs ( fr ) ; \
ret = efi_call_ # # prefix ( \
( efi_set_variable_t * ) __va ( runtime - > set_variable ) , \
adjust_arg ( name ) , adjust_arg ( vendor ) , attr , data_size , \
adjust_arg ( data ) ) ; \
ia64_load_scratch_fpregs ( fr ) ; \
return ret ; \
2005-04-17 02:20:36 +04:00
}
2008-02-05 02:31:49 +03:00
# define STUB_GET_NEXT_HIGH_MONO_COUNT(prefix, adjust_arg) \
static efi_status_t \
prefix # # _get_next_high_mono_count ( u32 * count ) \
{ \
struct ia64_fpreg fr [ 6 ] ; \
efi_status_t ret ; \
\
ia64_save_scratch_fpregs ( fr ) ; \
ret = efi_call_ # # prefix ( ( efi_get_next_high_mono_count_t * ) \
__va ( runtime - > get_next_high_mono_count ) , \
adjust_arg ( count ) ) ; \
ia64_load_scratch_fpregs ( fr ) ; \
return ret ; \
2005-04-17 02:20:36 +04:00
}
2008-02-05 02:31:49 +03:00
# define STUB_RESET_SYSTEM(prefix, adjust_arg) \
static void \
prefix # # _reset_system ( int reset_type , efi_status_t status , \
unsigned long data_size , efi_char16_t * data ) \
{ \
struct ia64_fpreg fr [ 6 ] ; \
efi_char16_t * adata = NULL ; \
\
if ( data ) \
adata = adjust_arg ( data ) ; \
\
ia64_save_scratch_fpregs ( fr ) ; \
efi_call_ # # prefix ( \
( efi_reset_system_t * ) __va ( runtime - > reset_system ) , \
reset_type , status , data_size , adata ) ; \
/* should not return, but just in case... */ \
ia64_load_scratch_fpregs ( fr ) ; \
2005-04-17 02:20:36 +04:00
}
# define phys_ptr(arg) ((__typeof__(arg)) ia64_tpa(arg))
STUB_GET_TIME ( phys , phys_ptr )
STUB_SET_TIME ( phys , phys_ptr )
STUB_GET_WAKEUP_TIME ( phys , phys_ptr )
STUB_SET_WAKEUP_TIME ( phys , phys_ptr )
STUB_GET_VARIABLE ( phys , phys_ptr )
STUB_GET_NEXT_VARIABLE ( phys , phys_ptr )
STUB_SET_VARIABLE ( phys , phys_ptr )
STUB_GET_NEXT_HIGH_MONO_COUNT ( phys , phys_ptr )
STUB_RESET_SYSTEM ( phys , phys_ptr )
# define id(arg) arg
STUB_GET_TIME ( virt , id )
STUB_SET_TIME ( virt , id )
STUB_GET_WAKEUP_TIME ( virt , id )
STUB_SET_WAKEUP_TIME ( virt , id )
STUB_GET_VARIABLE ( virt , id )
STUB_GET_NEXT_VARIABLE ( virt , id )
STUB_SET_VARIABLE ( virt , id )
STUB_GET_NEXT_HIGH_MONO_COUNT ( virt , id )
STUB_RESET_SYSTEM ( virt , id )
void
efi_gettimeofday ( struct timespec * ts )
{
efi_time_t tm ;
2007-11-07 00:40:39 +03:00
if ( ( * efi . get_time ) ( & tm , NULL ) ! = EFI_SUCCESS ) {
memset ( ts , 0 , sizeof ( * ts ) ) ;
2005-04-17 02:20:36 +04:00
return ;
2007-11-07 00:40:39 +03:00
}
2005-04-17 02:20:36 +04:00
2008-02-05 02:31:49 +03:00
ts - > tv_sec = mktime ( tm . year , tm . month , tm . day ,
tm . hour , tm . minute , tm . second ) ;
2005-04-17 02:20:36 +04:00
ts - > tv_nsec = tm . nanosecond ;
}
static int
2006-12-05 01:58:35 +03:00
is_memory_available ( efi_memory_desc_t * md )
2005-04-17 02:20:36 +04:00
{
if ( ! ( md - > attribute & EFI_MEMORY_WB ) )
return 0 ;
switch ( md - > type ) {
case EFI_LOADER_CODE :
case EFI_LOADER_DATA :
case EFI_BOOT_SERVICES_CODE :
case EFI_BOOT_SERVICES_DATA :
case EFI_CONVENTIONAL_MEMORY :
return 1 ;
}
return 0 ;
}
2005-09-08 23:39:59 +04:00
typedef struct kern_memdesc {
u64 attribute ;
u64 start ;
u64 num_pages ;
} kern_memdesc_t ;
2005-04-17 02:20:36 +04:00
2005-09-08 23:39:59 +04:00
static kern_memdesc_t * kern_memmap ;
2005-04-17 02:20:36 +04:00
2006-01-08 12:04:13 +03:00
# define efi_md_size(md) (md->num_pages << EFI_PAGE_SHIFT)
static inline u64
kmd_end ( kern_memdesc_t * kmd )
{
return ( kmd - > start + ( kmd - > num_pages < < EFI_PAGE_SHIFT ) ) ;
}
static inline u64
efi_md_end ( efi_memory_desc_t * md )
{
return ( md - > phys_addr + efi_md_size ( md ) ) ;
}
static inline int
efi_wb ( efi_memory_desc_t * md )
{
return ( md - > attribute & EFI_MEMORY_WB ) ;
}
static inline int
efi_uc ( efi_memory_desc_t * md )
{
return ( md - > attribute & EFI_MEMORY_UC ) ;
}
2005-04-17 02:20:36 +04:00
static void
2005-09-08 23:39:59 +04:00
walk ( efi_freemem_callback_t callback , void * arg , u64 attr )
2005-04-17 02:20:36 +04:00
{
2005-09-08 23:39:59 +04:00
kern_memdesc_t * k ;
u64 start , end , voff ;
2005-04-17 02:20:36 +04:00
2005-09-08 23:39:59 +04:00
voff = ( attr = = EFI_MEMORY_WB ) ? PAGE_OFFSET : __IA64_UNCACHED_OFFSET ;
for ( k = kern_memmap ; k - > start ! = ~ 0UL ; k + + ) {
if ( k - > attribute ! = attr )
continue ;
start = PAGE_ALIGN ( k - > start ) ;
end = ( k - > start + ( k - > num_pages < < EFI_PAGE_SHIFT ) ) & PAGE_MASK ;
if ( start < end )
if ( ( * callback ) ( start + voff , end + voff , arg ) < 0 )
return ;
}
2005-04-17 02:20:36 +04:00
}
/*
2008-01-09 06:29:37 +03:00
* Walk the EFI memory map and call CALLBACK once for each EFI memory
2008-02-05 02:31:49 +03:00
* descriptor that has memory that is available for OS use .
2005-04-17 02:20:36 +04:00
*/
void
efi_memmap_walk ( efi_freemem_callback_t callback , void * arg )
{
2005-09-08 23:39:59 +04:00
walk ( callback , arg , EFI_MEMORY_WB ) ;
2005-04-17 02:20:36 +04:00
}
2005-06-22 04:15:02 +04:00
/*
2008-01-09 06:29:37 +03:00
* Walk the EFI memory map and call CALLBACK once for each EFI memory
2008-02-05 02:31:49 +03:00
* descriptor that has memory that is available for uncached allocator .
2005-06-22 04:15:02 +04:00
*/
2005-09-08 23:39:59 +04:00
void
efi_memmap_walk_uc ( efi_freemem_callback_t callback , void * arg )
2005-06-22 04:15:02 +04:00
{
2005-09-08 23:39:59 +04:00
walk ( callback , arg , EFI_MEMORY_UC ) ;
2005-06-22 04:15:02 +04:00
}
2005-04-17 02:20:36 +04:00
/*
2008-01-09 06:29:37 +03:00
* Look for the PAL_CODE region reported by EFI and map it using an
2005-04-17 02:20:36 +04:00
* ITR to enable safe PAL calls in virtual mode . See IA - 64 Processor
* Abstraction Layer chapter 11 in ADAG
*/
void *
efi_get_pal_addr ( void )
{
void * efi_map_start , * efi_map_end , * p ;
efi_memory_desc_t * md ;
u64 efi_desc_size ;
int pal_code_count = 0 ;
u64 vaddr , mask ;
efi_map_start = __va ( ia64_boot_param - > efi_memmap ) ;
efi_map_end = efi_map_start + ia64_boot_param - > efi_memmap_size ;
efi_desc_size = ia64_boot_param - > efi_memdesc_size ;
for ( p = efi_map_start ; p < efi_map_end ; p + = efi_desc_size ) {
md = p ;
if ( md - > type ! = EFI_PAL_CODE )
continue ;
if ( + + pal_code_count > 1 ) {
2008-02-05 02:31:49 +03:00
printk ( KERN_ERR " Too many EFI Pal Code memory ranges, "
" dropped @ %lx \n " , md - > phys_addr ) ;
2005-04-17 02:20:36 +04:00
continue ;
}
/*
2008-02-05 02:31:49 +03:00
* The only ITLB entry in region 7 that is used is the one
* installed by __start ( ) . That entry covers a 64 MB range .
2005-04-17 02:20:36 +04:00
*/
mask = ~ ( ( 1 < < KERNEL_TR_PAGE_SHIFT ) - 1 ) ;
vaddr = PAGE_OFFSET + md - > phys_addr ;
/*
2008-02-05 02:31:49 +03:00
* We must check that the PAL mapping won ' t overlap with the
* kernel mapping .
2005-04-17 02:20:36 +04:00
*
2008-02-05 02:31:49 +03:00
* PAL code is guaranteed to be aligned on a power of 2 between
* 4 k and 256 KB and that only one ITR is needed to map it . This
* implies that the PAL code is always aligned on its size ,
* i . e . , the closest matching page size supported by the TLB .
* Therefore PAL code is guaranteed never to cross a 64 MB unless
* it is bigger than 64 MB ( very unlikely ! ) . So for now the
* following test is enough to determine whether or not we need
* a dedicated ITR for the PAL code .
2005-04-17 02:20:36 +04:00
*/
if ( ( vaddr & mask ) = = ( KERNEL_START & mask ) ) {
2008-02-05 02:31:49 +03:00
printk ( KERN_INFO " %s: no need to install ITR for "
" PAL code \n " , __FUNCTION__ ) ;
2005-04-17 02:20:36 +04:00
continue ;
}
2007-11-22 01:58:26 +03:00
if ( efi_md_size ( md ) > IA64_GRANULE_SIZE )
2008-01-09 06:29:37 +03:00
panic ( " Whoa! PAL code size bigger than a granule! " ) ;
2005-04-17 02:20:36 +04:00
# if EFI_DEBUG
mask = ~ ( ( 1 < < IA64_GRANULE_SHIFT ) - 1 ) ;
2008-02-05 02:31:49 +03:00
printk ( KERN_INFO " CPU %d: mapping PAL code "
" [0x%lx-0x%lx) into [0x%lx-0x%lx) \n " ,
smp_processor_id ( ) , md - > phys_addr ,
md - > phys_addr + efi_md_size ( md ) ,
vaddr & mask , ( vaddr & mask ) + IA64_GRANULE_SIZE ) ;
2005-04-17 02:20:36 +04:00
# endif
return __va ( md - > phys_addr ) ;
}
2007-02-05 04:17:38 +03:00
printk ( KERN_WARNING " %s: no PAL-code memory-descriptor found \n " ,
2005-04-17 02:20:36 +04:00
__FUNCTION__ ) ;
return NULL ;
}
void
efi_map_pal_code ( void )
{
void * pal_vaddr = efi_get_pal_addr ( ) ;
u64 psr ;
if ( ! pal_vaddr )
return ;
/*
* Cannot write to CRx with PSR . ic = 1
*/
psr = ia64_clear_ic ( ) ;
2008-02-05 02:31:49 +03:00
ia64_itr ( 0x1 , IA64_TR_PALCODE ,
GRANULEROUNDDOWN ( ( unsigned long ) pal_vaddr ) ,
2005-04-17 02:20:36 +04:00
pte_val ( pfn_pte ( __pa ( pal_vaddr ) > > PAGE_SHIFT , PAGE_KERNEL ) ) ,
IA64_GRANULE_SHIFT ) ;
ia64_set_psr ( psr ) ; /* restore psr */
}
void __init
efi_init ( void )
{
void * efi_map_start , * efi_map_end ;
efi_config_table_t * config_tables ;
efi_char16_t * c16 ;
u64 efi_desc_size ;
2006-02-07 06:35:46 +03:00
char * cp , vendor [ 100 ] = " unknown " ;
2005-04-17 02:20:36 +04:00
int i ;
2008-02-05 02:31:49 +03:00
/*
2008-01-09 06:29:37 +03:00
* It ' s too early to be able to use the standard kernel command line
2008-02-05 02:31:49 +03:00
* support . . .
*/
2007-02-12 11:54:12 +03:00
for ( cp = boot_command_line ; * cp ; ) {
2005-04-17 02:20:36 +04:00
if ( memcmp ( cp , " mem= " , 4 ) = = 0 ) {
2006-02-07 06:35:46 +03:00
mem_limit = memparse ( cp + 4 , & cp ) ;
2005-04-17 02:20:36 +04:00
} else if ( memcmp ( cp , " max_addr= " , 9 ) = = 0 ) {
2006-02-07 06:35:46 +03:00
max_addr = GRANULEROUNDDOWN ( memparse ( cp + 9 , & cp ) ) ;
2006-12-07 20:51:35 +03:00
} else if ( memcmp ( cp , " min_addr= " , 9 ) = = 0 ) {
min_addr = GRANULEROUNDDOWN ( memparse ( cp + 9 , & cp ) ) ;
2005-04-17 02:20:36 +04:00
} else {
while ( * cp ! = ' ' & & * cp )
+ + cp ;
while ( * cp = = ' ' )
+ + cp ;
}
}
2006-12-07 20:51:35 +03:00
if ( min_addr ! = 0UL )
2008-02-05 02:31:49 +03:00
printk ( KERN_INFO " Ignoring memory below %luMB \n " ,
min_addr > > 20 ) ;
2005-04-17 02:20:36 +04:00
if ( max_addr ! = ~ 0UL )
2008-02-05 02:31:49 +03:00
printk ( KERN_INFO " Ignoring memory above %luMB \n " ,
max_addr > > 20 ) ;
2005-04-17 02:20:36 +04:00
efi . systab = __va ( ia64_boot_param - > efi_systab ) ;
/*
* Verify the EFI Table
*/
if ( efi . systab = = NULL )
2008-01-09 06:29:37 +03:00
panic ( " Whoa! Can't find EFI system table. \n " ) ;
2005-04-17 02:20:36 +04:00
if ( efi . systab - > hdr . signature ! = EFI_SYSTEM_TABLE_SIGNATURE )
2008-01-09 06:29:37 +03:00
panic ( " Whoa! EFI system table signature incorrect \n " ) ;
2007-05-08 11:29:57 +04:00
if ( ( efi . systab - > hdr . revision > > 16 ) = = 0 )
printk ( KERN_WARNING " Warning: EFI system table version "
" %d.%02d, expected 1.00 or greater \n " ,
efi . systab - > hdr . revision > > 16 ,
efi . systab - > hdr . revision & 0xffff ) ;
2005-04-17 02:20:36 +04:00
config_tables = __va ( efi . systab - > tables ) ;
/* Show what we know for posterity */
c16 = __va ( efi . systab - > fw_vendor ) ;
if ( c16 ) {
2006-02-07 06:25:55 +03:00
for ( i = 0 ; i < ( int ) sizeof ( vendor ) - 1 & & * c16 ; + + i )
2005-04-17 02:20:36 +04:00
vendor [ i ] = * c16 + + ;
vendor [ i ] = ' \0 ' ;
}
printk ( KERN_INFO " EFI v%u.%.02u by %s: " ,
2008-02-05 02:31:49 +03:00
efi . systab - > hdr . revision > > 16 ,
efi . systab - > hdr . revision & 0xffff , vendor ) ;
2005-04-17 02:20:36 +04:00
2006-03-26 13:37:08 +04:00
efi . mps = EFI_INVALID_TABLE_ADDR ;
efi . acpi = EFI_INVALID_TABLE_ADDR ;
efi . acpi20 = EFI_INVALID_TABLE_ADDR ;
efi . smbios = EFI_INVALID_TABLE_ADDR ;
efi . sal_systab = EFI_INVALID_TABLE_ADDR ;
efi . boot_info = EFI_INVALID_TABLE_ADDR ;
efi . hcdp = EFI_INVALID_TABLE_ADDR ;
efi . uga = EFI_INVALID_TABLE_ADDR ;
2005-04-17 02:20:36 +04:00
for ( i = 0 ; i < ( int ) efi . systab - > nr_tables ; i + + ) {
if ( efi_guidcmp ( config_tables [ i ] . guid , MPS_TABLE_GUID ) = = 0 ) {
2006-03-26 13:37:08 +04:00
efi . mps = config_tables [ i ] . table ;
2005-04-17 02:20:36 +04:00
printk ( " MPS=0x%lx " , config_tables [ i ] . table ) ;
} else if ( efi_guidcmp ( config_tables [ i ] . guid , ACPI_20_TABLE_GUID ) = = 0 ) {
2006-03-26 13:37:08 +04:00
efi . acpi20 = config_tables [ i ] . table ;
2005-04-17 02:20:36 +04:00
printk ( " ACPI 2.0=0x%lx " , config_tables [ i ] . table ) ;
} else if ( efi_guidcmp ( config_tables [ i ] . guid , ACPI_TABLE_GUID ) = = 0 ) {
2006-03-26 13:37:08 +04:00
efi . acpi = config_tables [ i ] . table ;
2005-04-17 02:20:36 +04:00
printk ( " ACPI=0x%lx " , config_tables [ i ] . table ) ;
} else if ( efi_guidcmp ( config_tables [ i ] . guid , SMBIOS_TABLE_GUID ) = = 0 ) {
2006-03-26 13:37:08 +04:00
efi . smbios = config_tables [ i ] . table ;
2005-04-17 02:20:36 +04:00
printk ( " SMBIOS=0x%lx " , config_tables [ i ] . table ) ;
} else if ( efi_guidcmp ( config_tables [ i ] . guid , SAL_SYSTEM_TABLE_GUID ) = = 0 ) {
2006-03-26 13:37:08 +04:00
efi . sal_systab = config_tables [ i ] . table ;
2005-04-17 02:20:36 +04:00
printk ( " SALsystab=0x%lx " , config_tables [ i ] . table ) ;
} else if ( efi_guidcmp ( config_tables [ i ] . guid , HCDP_TABLE_GUID ) = = 0 ) {
2006-03-26 13:37:08 +04:00
efi . hcdp = config_tables [ i ] . table ;
2005-04-17 02:20:36 +04:00
printk ( " HCDP=0x%lx " , config_tables [ i ] . table ) ;
}
}
printk ( " \n " ) ;
runtime = __va ( efi . systab - > runtime ) ;
efi . get_time = phys_get_time ;
efi . set_time = phys_set_time ;
efi . get_wakeup_time = phys_get_wakeup_time ;
efi . set_wakeup_time = phys_set_wakeup_time ;
efi . get_variable = phys_get_variable ;
efi . get_next_variable = phys_get_next_variable ;
efi . set_variable = phys_set_variable ;
efi . get_next_high_mono_count = phys_get_next_high_mono_count ;
efi . reset_system = phys_reset_system ;
efi_map_start = __va ( ia64_boot_param - > efi_memmap ) ;
efi_map_end = efi_map_start + ia64_boot_param - > efi_memmap_size ;
efi_desc_size = ia64_boot_param - > efi_memdesc_size ;
# if EFI_DEBUG
/* print EFI memory map: */
{
efi_memory_desc_t * md ;
void * p ;
2008-02-05 02:31:49 +03:00
for ( i = 0 , p = efi_map_start ; p < efi_map_end ;
+ + i , p + = efi_desc_size )
{
2005-04-17 02:20:36 +04:00
md = p ;
2008-02-05 02:31:49 +03:00
printk ( " mem%02u: type=%u, attr=0x%lx, "
" range=[0x%016lx-0x%016lx) (%luMB) \n " ,
2005-04-17 02:20:36 +04:00
i , md - > type , md - > attribute , md - > phys_addr ,
2007-11-22 01:58:26 +03:00
md - > phys_addr + efi_md_size ( md ) ,
2005-04-17 02:20:36 +04:00
md - > num_pages > > ( 20 - EFI_PAGE_SHIFT ) ) ;
}
}
# endif
efi_map_pal_code ( ) ;
efi_enter_virtual_mode ( ) ;
}
void
efi_enter_virtual_mode ( void )
{
void * efi_map_start , * efi_map_end , * p ;
efi_memory_desc_t * md ;
efi_status_t status ;
u64 efi_desc_size ;
efi_map_start = __va ( ia64_boot_param - > efi_memmap ) ;
efi_map_end = efi_map_start + ia64_boot_param - > efi_memmap_size ;
efi_desc_size = ia64_boot_param - > efi_memdesc_size ;
for ( p = efi_map_start ; p < efi_map_end ; p + = efi_desc_size ) {
md = p ;
if ( md - > attribute & EFI_MEMORY_RUNTIME ) {
/*
2008-02-05 02:31:49 +03:00
* Some descriptors have multiple bits set , so the
* order of the tests is relevant .
2005-04-17 02:20:36 +04:00
*/
if ( md - > attribute & EFI_MEMORY_WB ) {
md - > virt_addr = ( u64 ) __va ( md - > phys_addr ) ;
} else if ( md - > attribute & EFI_MEMORY_UC ) {
md - > virt_addr = ( u64 ) ioremap ( md - > phys_addr , 0 ) ;
} else if ( md - > attribute & EFI_MEMORY_WC ) {
#if 0
2008-02-05 02:31:49 +03:00
md - > virt_addr = ia64_remap ( md - > phys_addr ,
( _PAGE_A |
_PAGE_P |
_PAGE_D |
_PAGE_MA_WC |
_PAGE_PL_0 |
_PAGE_AR_RW ) ) ;
2005-04-17 02:20:36 +04:00
# else
printk ( KERN_INFO " EFI_MEMORY_WC mapping \n " ) ;
md - > virt_addr = ( u64 ) ioremap ( md - > phys_addr , 0 ) ;
# endif
} else if ( md - > attribute & EFI_MEMORY_WT ) {
#if 0
2008-02-05 02:31:49 +03:00
md - > virt_addr = ia64_remap ( md - > phys_addr ,
( _PAGE_A |
_PAGE_P |
_PAGE_D |
_PAGE_MA_WT |
_PAGE_PL_0 |
_PAGE_AR_RW ) ) ;
2005-04-17 02:20:36 +04:00
# else
printk ( KERN_INFO " EFI_MEMORY_WT mapping \n " ) ;
md - > virt_addr = ( u64 ) ioremap ( md - > phys_addr , 0 ) ;
# endif
}
}
}
status = efi_call_phys ( __va ( runtime - > set_virtual_address_map ) ,
ia64_boot_param - > efi_memmap_size ,
2008-02-05 02:31:49 +03:00
efi_desc_size ,
ia64_boot_param - > efi_memdesc_version ,
2005-04-17 02:20:36 +04:00
ia64_boot_param - > efi_memmap ) ;
if ( status ! = EFI_SUCCESS ) {
2008-02-05 02:31:49 +03:00
printk ( KERN_WARNING " warning: unable to switch EFI into "
" virtual mode (status=%lu) \n " , status ) ;
2005-04-17 02:20:36 +04:00
return ;
}
/*
2008-02-05 02:31:49 +03:00
* Now that EFI is in virtual mode , we call the EFI functions more
* efficiently :
2005-04-17 02:20:36 +04:00
*/
efi . get_time = virt_get_time ;
efi . set_time = virt_set_time ;
efi . get_wakeup_time = virt_get_wakeup_time ;
efi . set_wakeup_time = virt_set_wakeup_time ;
efi . get_variable = virt_get_variable ;
efi . get_next_variable = virt_get_next_variable ;
efi . set_variable = virt_set_variable ;
efi . get_next_high_mono_count = virt_get_next_high_mono_count ;
efi . reset_system = virt_reset_system ;
}
/*
2008-02-05 02:31:49 +03:00
* Walk the EFI memory map looking for the I / O port range . There can only be
* one entry of this type , other I / O port ranges should be described via ACPI .
2005-04-17 02:20:36 +04:00
*/
u64
efi_get_iobase ( void )
{
void * efi_map_start , * efi_map_end , * p ;
efi_memory_desc_t * md ;
u64 efi_desc_size ;
efi_map_start = __va ( ia64_boot_param - > efi_memmap ) ;
efi_map_end = efi_map_start + ia64_boot_param - > efi_memmap_size ;
efi_desc_size = ia64_boot_param - > efi_memdesc_size ;
for ( p = efi_map_start ; p < efi_map_end ; p + = efi_desc_size ) {
md = p ;
if ( md - > type = = EFI_MEMORY_MAPPED_IO_PORT_SPACE ) {
if ( md - > attribute & EFI_MEMORY_UC )
return md - > phys_addr ;
}
}
return 0 ;
}
2006-05-06 03:19:50 +04:00
static struct kern_memdesc *
kern_memory_descriptor ( unsigned long phys_addr )
2005-04-17 02:20:36 +04:00
{
2006-05-06 03:19:50 +04:00
struct kern_memdesc * md ;
2005-04-17 02:20:36 +04:00
2006-05-06 03:19:50 +04:00
for ( md = kern_memmap ; md - > start ! = ~ 0UL ; md + + ) {
if ( phys_addr - md - > start < ( md - > num_pages < < EFI_PAGE_SHIFT ) )
2006-01-08 12:04:13 +03:00
return md ;
2005-04-17 02:20:36 +04:00
}
2006-07-17 09:41:59 +04:00
return NULL ;
2005-04-17 02:20:36 +04:00
}
2006-05-06 03:19:50 +04:00
static efi_memory_desc_t *
efi_memory_descriptor ( unsigned long phys_addr )
2005-04-17 02:20:36 +04:00
{
void * efi_map_start , * efi_map_end , * p ;
efi_memory_desc_t * md ;
u64 efi_desc_size ;
efi_map_start = __va ( ia64_boot_param - > efi_memmap ) ;
efi_map_end = efi_map_start + ia64_boot_param - > efi_memmap_size ;
efi_desc_size = ia64_boot_param - > efi_memdesc_size ;
for ( p = efi_map_start ; p < efi_map_end ; p + = efi_desc_size ) {
md = p ;
2007-11-22 01:58:26 +03:00
if ( phys_addr - md - > phys_addr < efi_md_size ( md ) )
2006-05-06 03:19:50 +04:00
return md ;
2005-04-17 02:20:36 +04:00
}
2006-07-17 09:41:59 +04:00
return NULL ;
2005-04-17 02:20:36 +04:00
}
2006-01-08 12:04:13 +03:00
2007-03-30 20:35:43 +04:00
static int
efi_memmap_intersects ( unsigned long phys_addr , unsigned long size )
{
void * efi_map_start , * efi_map_end , * p ;
efi_memory_desc_t * md ;
u64 efi_desc_size ;
unsigned long end ;
efi_map_start = __va ( ia64_boot_param - > efi_memmap ) ;
efi_map_end = efi_map_start + ia64_boot_param - > efi_memmap_size ;
efi_desc_size = ia64_boot_param - > efi_memdesc_size ;
end = phys_addr + size ;
for ( p = efi_map_start ; p < efi_map_end ; p + = efi_desc_size ) {
md = p ;
if ( md - > phys_addr < end & & efi_md_end ( md ) > phys_addr )
return 1 ;
}
return 0 ;
}
2006-01-08 12:04:13 +03:00
u32
efi_mem_type ( unsigned long phys_addr )
{
efi_memory_desc_t * md = efi_memory_descriptor ( phys_addr ) ;
if ( md )
return md - > type ;
return 0 ;
}
u64
efi_mem_attributes ( unsigned long phys_addr )
{
efi_memory_desc_t * md = efi_memory_descriptor ( phys_addr ) ;
if ( md )
return md - > attribute ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
EXPORT_SYMBOL ( efi_mem_attributes ) ;
2006-05-06 03:19:50 +04:00
u64
efi_mem_attribute ( unsigned long phys_addr , unsigned long size )
2006-01-08 12:04:13 +03:00
{
2006-03-26 13:37:05 +04:00
unsigned long end = phys_addr + size ;
2006-01-08 12:04:13 +03:00
efi_memory_desc_t * md = efi_memory_descriptor ( phys_addr ) ;
2006-05-06 03:19:50 +04:00
u64 attr ;
if ( ! md )
return 0 ;
/*
* EFI_MEMORY_RUNTIME is not a memory attribute ; it just tells
* the kernel that firmware needs this region mapped .
*/
attr = md - > attribute & ~ EFI_MEMORY_RUNTIME ;
do {
unsigned long md_end = efi_md_end ( md ) ;
if ( end < = md_end )
return attr ;
md = efi_memory_descriptor ( md_end ) ;
if ( ! md | | ( md - > attribute & ~ EFI_MEMORY_RUNTIME ) ! = attr )
return 0 ;
} while ( md ) ;
2008-01-09 06:29:38 +03:00
return 0 ; /* never reached */
2006-05-06 03:19:50 +04:00
}
u64
kern_mem_attribute ( unsigned long phys_addr , unsigned long size )
{
unsigned long end = phys_addr + size ;
struct kern_memdesc * md ;
u64 attr ;
2006-01-08 12:04:13 +03:00
2006-03-26 13:37:05 +04:00
/*
2006-05-06 03:19:50 +04:00
* This is a hack for ioremap calls before we set up kern_memmap .
* Maybe we should do efi_memmap_init ( ) earlier instead .
2006-03-26 13:37:05 +04:00
*/
2006-05-06 03:19:50 +04:00
if ( ! kern_memmap ) {
attr = efi_mem_attribute ( phys_addr , size ) ;
if ( attr & EFI_MEMORY_WB )
return EFI_MEMORY_WB ;
2006-01-08 12:04:13 +03:00
return 0 ;
2006-03-26 13:37:05 +04:00
}
2006-01-08 12:04:13 +03:00
2006-05-06 03:19:50 +04:00
md = kern_memory_descriptor ( phys_addr ) ;
if ( ! md )
return 0 ;
attr = md - > attribute ;
2006-01-08 12:04:13 +03:00
do {
2006-05-06 03:19:50 +04:00
unsigned long md_end = kmd_end ( md ) ;
2006-03-26 13:37:05 +04:00
if ( end < = md_end )
2006-05-06 03:19:50 +04:00
return attr ;
2006-01-08 12:04:13 +03:00
2006-05-06 03:19:50 +04:00
md = kern_memory_descriptor ( md_end ) ;
if ( ! md | | md - > attribute ! = attr )
2006-03-26 13:37:05 +04:00
return 0 ;
2006-01-08 12:04:13 +03:00
} while ( md ) ;
2008-01-09 06:29:38 +03:00
return 0 ; /* never reached */
2006-01-08 12:04:13 +03:00
}
2006-05-06 03:19:50 +04:00
EXPORT_SYMBOL ( kern_mem_attribute ) ;
2006-01-08 12:04:13 +03:00
2005-04-17 02:20:36 +04:00
int
2006-03-26 13:37:05 +04:00
valid_phys_addr_range ( unsigned long phys_addr , unsigned long size )
2005-04-17 02:20:36 +04:00
{
2006-05-06 03:19:50 +04:00
u64 attr ;
/*
* / dev / mem reads and writes use copy_to_user ( ) , which implicitly
* uses a granule - sized kernel identity mapping . It ' s really
* only safe to do this for regions in kern_memmap . For more
* details , see Documentation / ia64 / aliasing . txt .
*/
attr = kern_mem_attribute ( phys_addr , size ) ;
if ( attr & EFI_MEMORY_WB | | attr & EFI_MEMORY_UC )
return 1 ;
return 0 ;
2006-01-08 12:04:13 +03:00
}
2005-04-17 02:20:36 +04:00
2006-01-08 12:04:13 +03:00
int
2006-07-10 15:45:27 +04:00
valid_mmap_phys_addr_range ( unsigned long pfn , unsigned long size )
2006-01-08 12:04:13 +03:00
{
2007-03-30 20:35:43 +04:00
unsigned long phys_addr = pfn < < PAGE_SHIFT ;
u64 attr ;
attr = efi_mem_attribute ( phys_addr , size ) ;
2006-05-06 03:19:50 +04:00
/*
2007-03-30 20:35:43 +04:00
* / dev / mem mmap uses normal user pages , so we don ' t need the entire
* granule , but the entire region we ' re mapping must support the same
* attribute .
2006-05-06 03:19:50 +04:00
*/
2007-03-30 20:35:43 +04:00
if ( attr & EFI_MEMORY_WB | | attr & EFI_MEMORY_UC )
return 1 ;
/*
* Intel firmware doesn ' t tell us about all the MMIO regions , so
* in general we have to allow mmap requests . But if EFI * does *
* tell us about anything inside this region , we should deny it .
* The user can always map a smaller region to avoid the overlap .
*/
if ( efi_memmap_intersects ( phys_addr , size ) )
return 0 ;
2006-05-06 03:19:50 +04:00
return 1 ;
}
2005-04-17 02:20:36 +04:00
2006-05-06 03:19:50 +04:00
pgprot_t
phys_mem_access_prot ( struct file * file , unsigned long pfn , unsigned long size ,
pgprot_t vma_prot )
{
unsigned long phys_addr = pfn < < PAGE_SHIFT ;
u64 attr ;
2005-04-17 02:20:36 +04:00
2006-05-06 03:19:50 +04:00
/*
* For / dev / mem mmap , we use user mappings , but if the region is
* in kern_memmap ( and hence may be covered by a kernel mapping ) ,
* we must use the same attribute as the kernel mapping .
*/
attr = kern_mem_attribute ( phys_addr , size ) ;
if ( attr & EFI_MEMORY_WB )
return pgprot_cacheable ( vma_prot ) ;
else if ( attr & EFI_MEMORY_UC )
return pgprot_noncached ( vma_prot ) ;
/*
* Some chipsets don ' t support UC access to memory . If
* WB is supported , we prefer that .
*/
if ( efi_mem_attribute ( phys_addr , size ) & EFI_MEMORY_WB )
return pgprot_cacheable ( vma_prot ) ;
return pgprot_noncached ( vma_prot ) ;
2005-04-17 02:20:36 +04:00
}
int __init
efi_uart_console_only ( void )
{
efi_status_t status ;
char * s , name [ ] = " ConOut " ;
efi_guid_t guid = EFI_GLOBAL_VARIABLE_GUID ;
efi_char16_t * utf16 , name_utf16 [ 32 ] ;
unsigned char data [ 1024 ] ;
unsigned long size = sizeof ( data ) ;
struct efi_generic_dev_path * hdr , * end_addr ;
int uart = 0 ;
/* Convert to UTF-16 */
utf16 = name_utf16 ;
s = name ;
while ( * s )
* utf16 + + = * s + + & 0x7f ;
* utf16 = 0 ;
status = efi . get_variable ( name_utf16 , & guid , NULL , & size , data ) ;
if ( status ! = EFI_SUCCESS ) {
printk ( KERN_ERR " No EFI %s variable? \n " , name ) ;
return 0 ;
}
hdr = ( struct efi_generic_dev_path * ) data ;
end_addr = ( struct efi_generic_dev_path * ) ( ( u8 * ) data + size ) ;
while ( hdr < end_addr ) {
if ( hdr - > type = = EFI_DEV_MSG & &
hdr - > sub_type = = EFI_DEV_MSG_UART )
uart = 1 ;
else if ( hdr - > type = = EFI_DEV_END_PATH | |
hdr - > type = = EFI_DEV_END_PATH2 ) {
if ( ! uart )
return 0 ;
if ( hdr - > sub_type = = EFI_DEV_END_ENTIRE )
return 1 ;
uart = 0 ;
}
2008-02-05 02:31:49 +03:00
hdr = ( struct efi_generic_dev_path * ) ( ( u8 * ) hdr + hdr - > length ) ;
2005-04-17 02:20:36 +04:00
}
printk ( KERN_ERR " Malformed %s value \n " , name ) ;
return 0 ;
}
2005-09-08 23:39:59 +04:00
/*
* Look for the first granule aligned memory descriptor memory
* that is big enough to hold EFI memory map . Make sure this
* descriptor is atleast granule sized so it does not get trimmed
*/
struct kern_memdesc *
find_memmap_space ( void )
{
u64 contig_low = 0 , contig_high = 0 ;
u64 as = 0 , ae ;
void * efi_map_start , * efi_map_end , * p , * q ;
efi_memory_desc_t * md , * pmd = NULL , * check_md ;
u64 space_needed , efi_desc_size ;
unsigned long total_mem = 0 ;
efi_map_start = __va ( ia64_boot_param - > efi_memmap ) ;
efi_map_end = efi_map_start + ia64_boot_param - > efi_memmap_size ;
efi_desc_size = ia64_boot_param - > efi_memdesc_size ;
/*
* Worst case : we need 3 kernel descriptors for each efi descriptor
* ( if every entry has a WB part in the middle , and UC head and tail ) ,
* plus one for the end marker .
*/
space_needed = sizeof ( kern_memdesc_t ) *
( 3 * ( ia64_boot_param - > efi_memmap_size / efi_desc_size ) + 1 ) ;
for ( p = efi_map_start ; p < efi_map_end ; pmd = md , p + = efi_desc_size ) {
md = p ;
if ( ! efi_wb ( md ) ) {
continue ;
}
2008-02-05 02:31:49 +03:00
if ( pmd = = NULL | | ! efi_wb ( pmd ) | |
efi_md_end ( pmd ) ! = md - > phys_addr ) {
2005-09-08 23:39:59 +04:00
contig_low = GRANULEROUNDUP ( md - > phys_addr ) ;
contig_high = efi_md_end ( md ) ;
2008-02-05 02:31:49 +03:00
for ( q = p + efi_desc_size ; q < efi_map_end ;
q + = efi_desc_size ) {
2005-09-08 23:39:59 +04:00
check_md = q ;
if ( ! efi_wb ( check_md ) )
break ;
if ( contig_high ! = check_md - > phys_addr )
break ;
contig_high = efi_md_end ( check_md ) ;
}
contig_high = GRANULEROUNDDOWN ( contig_high ) ;
}
2006-12-05 01:58:35 +03:00
if ( ! is_memory_available ( md ) | | md - > type = = EFI_LOADER_DATA )
2005-09-08 23:39:59 +04:00
continue ;
/* Round ends inward to granule boundaries */
as = max ( contig_low , md - > phys_addr ) ;
ae = min ( contig_high , efi_md_end ( md ) ) ;
2006-12-07 20:51:35 +03:00
/* keep within max_addr= and min_addr= command line arg */
as = max ( as , min_addr ) ;
2005-09-08 23:39:59 +04:00
ae = min ( ae , max_addr ) ;
if ( ae < = as )
continue ;
/* avoid going over mem= command line arg */
if ( total_mem + ( ae - as ) > mem_limit )
ae - = total_mem + ( ae - as ) - mem_limit ;
if ( ae < = as )
continue ;
if ( ae - as > space_needed )
break ;
}
if ( p > = efi_map_end )
panic ( " Can't allocate space for kernel memory descriptors " ) ;
return __va ( as ) ;
}
/*
* Walk the EFI memory map and gather all memory available for kernel
* to use . We can allocate partial granules only if the unavailable
* parts exist , and are WB .
*/
2007-10-19 10:41:00 +04:00
unsigned long
2005-09-08 23:39:59 +04:00
efi_memmap_init ( unsigned long * s , unsigned long * e )
{
2006-07-17 09:41:59 +04:00
struct kern_memdesc * k , * prev = NULL ;
2005-09-08 23:39:59 +04:00
u64 contig_low = 0 , contig_high = 0 ;
u64 as , ae , lim ;
void * efi_map_start , * efi_map_end , * p , * q ;
efi_memory_desc_t * md , * pmd = NULL , * check_md ;
u64 efi_desc_size ;
unsigned long total_mem = 0 ;
k = kern_memmap = find_memmap_space ( ) ;
efi_map_start = __va ( ia64_boot_param - > efi_memmap ) ;
efi_map_end = efi_map_start + ia64_boot_param - > efi_memmap_size ;
efi_desc_size = ia64_boot_param - > efi_memdesc_size ;
for ( p = efi_map_start ; p < efi_map_end ; pmd = md , p + = efi_desc_size ) {
md = p ;
if ( ! efi_wb ( md ) ) {
2008-02-05 02:31:49 +03:00
if ( efi_uc ( md ) & &
( md - > type = = EFI_CONVENTIONAL_MEMORY | |
md - > type = = EFI_BOOT_SERVICES_DATA ) ) {
2005-09-08 23:39:59 +04:00
k - > attribute = EFI_MEMORY_UC ;
k - > start = md - > phys_addr ;
k - > num_pages = md - > num_pages ;
k + + ;
}
continue ;
}
2008-02-05 02:31:49 +03:00
if ( pmd = = NULL | | ! efi_wb ( pmd ) | |
efi_md_end ( pmd ) ! = md - > phys_addr ) {
2005-09-08 23:39:59 +04:00
contig_low = GRANULEROUNDUP ( md - > phys_addr ) ;
contig_high = efi_md_end ( md ) ;
2008-02-05 02:31:49 +03:00
for ( q = p + efi_desc_size ; q < efi_map_end ;
q + = efi_desc_size ) {
2005-09-08 23:39:59 +04:00
check_md = q ;
if ( ! efi_wb ( check_md ) )
break ;
if ( contig_high ! = check_md - > phys_addr )
break ;
contig_high = efi_md_end ( check_md ) ;
}
contig_high = GRANULEROUNDDOWN ( contig_high ) ;
}
2006-12-05 01:58:35 +03:00
if ( ! is_memory_available ( md ) )
2005-09-08 23:39:59 +04:00
continue ;
2007-03-08 03:13:25 +03:00
# ifdef CONFIG_CRASH_DUMP
/* saved_max_pfn should ignore max_addr= command line arg */
if ( saved_max_pfn < ( efi_md_end ( md ) > > PAGE_SHIFT ) )
saved_max_pfn = ( efi_md_end ( md ) > > PAGE_SHIFT ) ;
# endif
2005-09-08 23:39:59 +04:00
/*
* Round ends inward to granule boundaries
* Give trimmings to uncached allocator
*/
if ( md - > phys_addr < contig_low ) {
lim = min ( efi_md_end ( md ) , contig_low ) ;
if ( efi_uc ( md ) ) {
2008-02-05 02:31:49 +03:00
if ( k > kern_memmap & &
( k - 1 ) - > attribute = = EFI_MEMORY_UC & &
2005-09-08 23:39:59 +04:00
kmd_end ( k - 1 ) = = md - > phys_addr ) {
2008-02-05 02:31:49 +03:00
( k - 1 ) - > num_pages + =
( lim - md - > phys_addr )
> > EFI_PAGE_SHIFT ;
2005-09-08 23:39:59 +04:00
} else {
k - > attribute = EFI_MEMORY_UC ;
k - > start = md - > phys_addr ;
2008-02-05 02:31:49 +03:00
k - > num_pages = ( lim - md - > phys_addr )
> > EFI_PAGE_SHIFT ;
2005-09-08 23:39:59 +04:00
k + + ;
}
}
as = contig_low ;
} else
as = md - > phys_addr ;
if ( efi_md_end ( md ) > contig_high ) {
lim = max ( md - > phys_addr , contig_high ) ;
if ( efi_uc ( md ) ) {
if ( lim = = md - > phys_addr & & k > kern_memmap & &
( k - 1 ) - > attribute = = EFI_MEMORY_UC & &
kmd_end ( k - 1 ) = = md - > phys_addr ) {
( k - 1 ) - > num_pages + = md - > num_pages ;
} else {
k - > attribute = EFI_MEMORY_UC ;
k - > start = lim ;
2008-02-05 02:31:49 +03:00
k - > num_pages = ( efi_md_end ( md ) - lim )
> > EFI_PAGE_SHIFT ;
2005-09-08 23:39:59 +04:00
k + + ;
}
}
ae = contig_high ;
} else
ae = efi_md_end ( md ) ;
2006-12-07 20:51:35 +03:00
/* keep within max_addr= and min_addr= command line arg */
as = max ( as , min_addr ) ;
2005-09-08 23:39:59 +04:00
ae = min ( ae , max_addr ) ;
if ( ae < = as )
continue ;
/* avoid going over mem= command line arg */
if ( total_mem + ( ae - as ) > mem_limit )
ae - = total_mem + ( ae - as ) - mem_limit ;
if ( ae < = as )
continue ;
if ( prev & & kmd_end ( prev ) = = md - > phys_addr ) {
prev - > num_pages + = ( ae - as ) > > EFI_PAGE_SHIFT ;
total_mem + = ae - as ;
continue ;
}
k - > attribute = EFI_MEMORY_WB ;
k - > start = as ;
k - > num_pages = ( ae - as ) > > EFI_PAGE_SHIFT ;
total_mem + = ae - as ;
prev = k + + ;
}
k - > start = ~ 0L ; /* end-marker */
/* reserve the memory we are using for kern_memmap */
* s = ( u64 ) kern_memmap ;
* e = ( u64 ) + + k ;
2007-10-19 10:41:00 +04:00
return total_mem ;
2005-09-08 23:39:59 +04:00
}
2005-09-20 02:42:36 +04:00
void
efi_initialize_iomem_resources ( struct resource * code_resource ,
2007-10-22 03:42:01 +04:00
struct resource * data_resource ,
struct resource * bss_resource )
2005-09-20 02:42:36 +04:00
{
struct resource * res ;
void * efi_map_start , * efi_map_end , * p ;
efi_memory_desc_t * md ;
u64 efi_desc_size ;
char * name ;
unsigned long flags ;
efi_map_start = __va ( ia64_boot_param - > efi_memmap ) ;
efi_map_end = efi_map_start + ia64_boot_param - > efi_memmap_size ;
efi_desc_size = ia64_boot_param - > efi_memdesc_size ;
res = NULL ;
for ( p = efi_map_start ; p < efi_map_end ; p + = efi_desc_size ) {
md = p ;
if ( md - > num_pages = = 0 ) /* should not happen */
continue ;
2007-11-15 03:59:20 +03:00
flags = IORESOURCE_MEM | IORESOURCE_BUSY ;
2005-09-20 02:42:36 +04:00
switch ( md - > type ) {
case EFI_MEMORY_MAPPED_IO :
case EFI_MEMORY_MAPPED_IO_PORT_SPACE :
continue ;
case EFI_LOADER_CODE :
case EFI_LOADER_DATA :
case EFI_BOOT_SERVICES_DATA :
case EFI_BOOT_SERVICES_CODE :
case EFI_CONVENTIONAL_MEMORY :
if ( md - > attribute & EFI_MEMORY_WP ) {
name = " System ROM " ;
flags | = IORESOURCE_READONLY ;
} else {
name = " System RAM " ;
}
break ;
case EFI_ACPI_MEMORY_NVS :
name = " ACPI Non-volatile Storage " ;
break ;
case EFI_UNUSABLE_MEMORY :
name = " reserved " ;
2007-11-15 03:59:20 +03:00
flags | = IORESOURCE_DISABLED ;
2005-09-20 02:42:36 +04:00
break ;
case EFI_RESERVED_TYPE :
case EFI_RUNTIME_SERVICES_CODE :
case EFI_RUNTIME_SERVICES_DATA :
case EFI_ACPI_RECLAIM_MEMORY :
default :
name = " reserved " ;
break ;
}
2008-02-05 02:31:49 +03:00
if ( ( res = kzalloc ( sizeof ( struct resource ) ,
GFP_KERNEL ) ) = = NULL ) {
printk ( KERN_ERR
2008-01-09 06:29:37 +03:00
" failed to allocate resource for iomem \n " ) ;
2005-09-20 02:42:36 +04:00
return ;
}
res - > name = name ;
res - > start = md - > phys_addr ;
2007-11-22 01:58:26 +03:00
res - > end = md - > phys_addr + efi_md_size ( md ) - 1 ;
2005-09-20 02:42:36 +04:00
res - > flags = flags ;
if ( insert_resource ( & iomem_resource , res ) < 0 )
kfree ( res ) ;
else {
/*
* We don ' t know which region contains
* kernel data so we try it repeatedly and
* let the resource manager test it .
*/
insert_resource ( res , code_resource ) ;
insert_resource ( res , data_resource ) ;
2007-10-22 03:42:01 +04:00
insert_resource ( res , bss_resource ) ;
2006-12-07 20:51:35 +03:00
# ifdef CONFIG_KEXEC
insert_resource ( res , & efi_memmap_res ) ;
insert_resource ( res , & boot_param_res ) ;
if ( crashk_res . end > crashk_res . start )
insert_resource ( res , & crashk_res ) ;
# endif
2005-09-20 02:42:36 +04:00
}
}
}
2006-12-07 20:51:35 +03:00
# ifdef CONFIG_KEXEC
/* find a block of memory aligned to 64M exclude reserved regions
rsvd_regions are sorted
*/
2007-03-06 10:28:50 +03:00
unsigned long __init
2008-02-05 02:31:49 +03:00
kdump_find_rsvd_region ( unsigned long size , struct rsvd_region * r , int n )
2006-12-07 20:51:35 +03:00
{
2008-02-05 02:31:49 +03:00
int i ;
u64 start , end ;
u64 alignment = 1UL < < _PAGE_SIZE_64M ;
void * efi_map_start , * efi_map_end , * p ;
efi_memory_desc_t * md ;
u64 efi_desc_size ;
efi_map_start = __va ( ia64_boot_param - > efi_memmap ) ;
efi_map_end = efi_map_start + ia64_boot_param - > efi_memmap_size ;
efi_desc_size = ia64_boot_param - > efi_memdesc_size ;
for ( p = efi_map_start ; p < efi_map_end ; p + = efi_desc_size ) {
md = p ;
if ( ! efi_wb ( md ) )
continue ;
start = ALIGN ( md - > phys_addr , alignment ) ;
end = efi_md_end ( md ) ;
for ( i = 0 ; i < n ; i + + ) {
if ( __pa ( r [ i ] . start ) > = start & & __pa ( r [ i ] . end ) < end ) {
if ( __pa ( r [ i ] . start ) > start + size )
return start ;
start = ALIGN ( __pa ( r [ i ] . end ) , alignment ) ;
if ( i < n - 1 & &
__pa ( r [ i + 1 ] . start ) < start + size )
continue ;
else
break ;
}
2006-12-07 20:51:35 +03:00
}
2008-02-05 02:31:49 +03:00
if ( end > start + size )
return start ;
}
printk ( KERN_WARNING
" Cannot reserve 0x%lx byte of memory for crashdump \n " , size ) ;
return ~ 0UL ;
2006-12-07 20:51:35 +03:00
}
# endif
2007-03-06 13:34:26 +03:00
# ifdef CONFIG_PROC_VMCORE
/* locate the size find a the descriptor at a certain address */
2007-10-29 11:10:29 +03:00
unsigned long __init
2007-03-06 13:34:26 +03:00
vmcore_find_descriptor_size ( unsigned long address )
{
void * efi_map_start , * efi_map_end , * p ;
efi_memory_desc_t * md ;
u64 efi_desc_size ;
unsigned long ret = 0 ;
efi_map_start = __va ( ia64_boot_param - > efi_memmap ) ;
efi_map_end = efi_map_start + ia64_boot_param - > efi_memmap_size ;
efi_desc_size = ia64_boot_param - > efi_memdesc_size ;
for ( p = efi_map_start ; p < efi_map_end ; p + = efi_desc_size ) {
md = p ;
if ( efi_wb ( md ) & & md - > type = = EFI_LOADER_DATA
& & md - > phys_addr = = address ) {
ret = efi_md_size ( md ) ;
break ;
}
}
if ( ret = = 0 )
printk ( KERN_WARNING " Cannot locate EFI vmcore descriptor \n " ) ;
return ret ;
}
# endif