2005-04-16 15:20:36 -07:00
/*
* acpi_tables . c - ACPI Boot - Time Table Parsing
*
* Copyright ( C ) 2001 Paul Diefenbaugh < paul . s . diefenbaugh @ intel . com >
*
* ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*
* ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
*
*/
# include <linux/config.h>
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/sched.h>
# include <linux/smp.h>
# include <linux/string.h>
# include <linux/types.h>
# include <linux/irq.h>
# include <linux/errno.h>
# include <linux/acpi.h>
# include <linux/bootmem.h>
# define PREFIX "ACPI: "
2005-12-30 02:44:59 -05:00
# define ACPI_MAX_TABLES 128
2005-04-16 15:20:36 -07:00
static char * acpi_table_signatures [ ACPI_TABLE_COUNT ] = {
2005-08-05 00:44:28 -04:00
[ ACPI_TABLE_UNKNOWN ] = " ???? " ,
[ ACPI_APIC ] = " APIC " ,
[ ACPI_BOOT ] = " BOOT " ,
[ ACPI_DBGP ] = " DBGP " ,
[ ACPI_DSDT ] = " DSDT " ,
[ ACPI_ECDT ] = " ECDT " ,
[ ACPI_ETDT ] = " ETDT " ,
[ ACPI_FADT ] = " FACP " ,
[ ACPI_FACS ] = " FACS " ,
[ ACPI_OEMX ] = " OEM " ,
[ ACPI_PSDT ] = " PSDT " ,
[ ACPI_SBST ] = " SBST " ,
[ ACPI_SLIT ] = " SLIT " ,
[ ACPI_SPCR ] = " SPCR " ,
[ ACPI_SRAT ] = " SRAT " ,
[ ACPI_SSDT ] = " SSDT " ,
[ ACPI_SPMI ] = " SPMI " ,
[ ACPI_HPET ] = " HPET " ,
[ ACPI_MCFG ] = " MCFG " ,
2005-04-16 15:20:36 -07:00
} ;
static char * mps_inti_flags_polarity [ ] = { " dfl " , " high " , " res " , " low " } ;
static char * mps_inti_flags_trigger [ ] = { " dfl " , " edge " , " res " , " level " } ;
/* System Description Table (RSDT/XSDT) */
struct acpi_table_sdt {
2005-08-05 00:44:28 -04:00
unsigned long pa ;
enum acpi_table_id id ;
unsigned long size ;
2005-04-16 15:20:36 -07:00
} __attribute__ ( ( packed ) ) ;
2005-08-05 00:44:28 -04:00
static unsigned long sdt_pa ; /* Physical Address */
static unsigned long sdt_count ; /* Table count */
2005-04-16 15:20:36 -07:00
2005-12-30 02:44:59 -05:00
static struct acpi_table_sdt sdt_entry [ ACPI_MAX_TABLES ] __initdata ;
2005-04-16 15:20:36 -07:00
2005-08-05 00:44:28 -04:00
void acpi_table_print ( struct acpi_table_header * header , unsigned long phys_addr )
2005-04-16 15:20:36 -07:00
{
2005-08-05 00:44:28 -04:00
char * name = NULL ;
2005-04-16 15:20:36 -07:00
if ( ! header )
return ;
/* Some table signatures aren't good table names */
2005-08-05 00:44:28 -04:00
if ( ! strncmp ( ( char * ) & header - > signature ,
acpi_table_signatures [ ACPI_APIC ] ,
sizeof ( header - > signature ) ) ) {
2005-04-16 15:20:36 -07:00
name = " MADT " ;
2005-08-05 00:44:28 -04:00
} else if ( ! strncmp ( ( char * ) & header - > signature ,
acpi_table_signatures [ ACPI_FADT ] ,
sizeof ( header - > signature ) ) ) {
2005-04-16 15:20:36 -07:00
name = " FADT " ;
2005-08-05 00:44:28 -04:00
} else
2005-04-16 15:20:36 -07:00
name = header - > signature ;
2005-08-05 00:44:28 -04:00
printk ( KERN_DEBUG PREFIX
" %.4s (v%3.3d %6.6s %8.8s 0x%08x %.4s 0x%08x) @ 0x%p \n " , name ,
header - > revision , header - > oem_id , header - > oem_table_id ,
header - > oem_revision , header - > asl_compiler_id ,
header - > asl_compiler_revision , ( void * ) phys_addr ) ;
2005-04-16 15:20:36 -07:00
}
2005-08-05 00:44:28 -04:00
void acpi_table_print_madt_entry ( acpi_table_entry_header * header )
2005-04-16 15:20:36 -07:00
{
if ( ! header )
return ;
switch ( header - > type ) {
case ACPI_MADT_LAPIC :
2005-08-05 00:44:28 -04:00
{
struct acpi_table_lapic * p =
( struct acpi_table_lapic * ) header ;
printk ( KERN_INFO PREFIX
" LAPIC (acpi_id[0x%02x] lapic_id[0x%02x] %s) \n " ,
p - > acpi_id , p - > id ,
p - > flags . enabled ? " enabled " : " disabled " ) ;
}
2005-04-16 15:20:36 -07:00
break ;
case ACPI_MADT_IOAPIC :
2005-08-05 00:44:28 -04:00
{
struct acpi_table_ioapic * p =
( struct acpi_table_ioapic * ) header ;
printk ( KERN_INFO PREFIX
" IOAPIC (id[0x%02x] address[0x%08x] gsi_base[%d]) \n " ,
p - > id , p - > address , p - > global_irq_base ) ;
}
2005-04-16 15:20:36 -07:00
break ;
case ACPI_MADT_INT_SRC_OVR :
2005-08-05 00:44:28 -04:00
{
struct acpi_table_int_src_ovr * p =
( struct acpi_table_int_src_ovr * ) header ;
printk ( KERN_INFO PREFIX
" INT_SRC_OVR (bus %d bus_irq %d global_irq %d %s %s) \n " ,
p - > bus , p - > bus_irq , p - > global_irq ,
mps_inti_flags_polarity [ p - > flags . polarity ] ,
mps_inti_flags_trigger [ p - > flags . trigger ] ) ;
if ( p - > flags . reserved )
printk ( KERN_INFO PREFIX
" INT_SRC_OVR unexpected reserved flags: 0x%x \n " ,
p - > flags . reserved ) ;
2005-04-16 15:20:36 -07:00
2005-08-05 00:44:28 -04:00
}
2005-04-16 15:20:36 -07:00
break ;
case ACPI_MADT_NMI_SRC :
2005-08-05 00:44:28 -04:00
{
struct acpi_table_nmi_src * p =
( struct acpi_table_nmi_src * ) header ;
printk ( KERN_INFO PREFIX
" NMI_SRC (%s %s global_irq %d) \n " ,
mps_inti_flags_polarity [ p - > flags . polarity ] ,
mps_inti_flags_trigger [ p - > flags . trigger ] ,
p - > global_irq ) ;
}
2005-04-16 15:20:36 -07:00
break ;
case ACPI_MADT_LAPIC_NMI :
2005-08-05 00:44:28 -04:00
{
struct acpi_table_lapic_nmi * p =
( struct acpi_table_lapic_nmi * ) header ;
printk ( KERN_INFO PREFIX
" LAPIC_NMI (acpi_id[0x%02x] %s %s lint[0x%x]) \n " ,
p - > acpi_id ,
mps_inti_flags_polarity [ p - > flags . polarity ] ,
mps_inti_flags_trigger [ p - > flags . trigger ] ,
p - > lint ) ;
}
2005-04-16 15:20:36 -07:00
break ;
case ACPI_MADT_LAPIC_ADDR_OVR :
2005-08-05 00:44:28 -04:00
{
struct acpi_table_lapic_addr_ovr * p =
( struct acpi_table_lapic_addr_ovr * ) header ;
printk ( KERN_INFO PREFIX
" LAPIC_ADDR_OVR (address[%p]) \n " ,
( void * ) ( unsigned long ) p - > address ) ;
}
2005-04-16 15:20:36 -07:00
break ;
case ACPI_MADT_IOSAPIC :
2005-08-05 00:44:28 -04:00
{
struct acpi_table_iosapic * p =
( struct acpi_table_iosapic * ) header ;
printk ( KERN_INFO PREFIX
" IOSAPIC (id[0x%x] address[%p] gsi_base[%d]) \n " ,
p - > id , ( void * ) ( unsigned long ) p - > address ,
p - > global_irq_base ) ;
}
2005-04-16 15:20:36 -07:00
break ;
case ACPI_MADT_LSAPIC :
2005-08-05 00:44:28 -04:00
{
struct acpi_table_lsapic * p =
( struct acpi_table_lsapic * ) header ;
printk ( KERN_INFO PREFIX
" LSAPIC (acpi_id[0x%02x] lsapic_id[0x%02x] lsapic_eid[0x%02x] %s) \n " ,
p - > acpi_id , p - > id , p - > eid ,
p - > flags . enabled ? " enabled " : " disabled " ) ;
}
2005-04-16 15:20:36 -07:00
break ;
case ACPI_MADT_PLAT_INT_SRC :
2005-08-05 00:44:28 -04:00
{
struct acpi_table_plat_int_src * p =
( struct acpi_table_plat_int_src * ) header ;
printk ( KERN_INFO PREFIX
" PLAT_INT_SRC (%s %s type[0x%x] id[0x%04x] eid[0x%x] iosapic_vector[0x%x] global_irq[0x%x] \n " ,
mps_inti_flags_polarity [ p - > flags . polarity ] ,
mps_inti_flags_trigger [ p - > flags . trigger ] ,
p - > type , p - > id , p - > eid , p - > iosapic_vector ,
p - > global_irq ) ;
}
2005-04-16 15:20:36 -07:00
break ;
default :
2005-08-05 00:44:28 -04:00
printk ( KERN_WARNING PREFIX
" Found unsupported MADT entry (type = 0x%x) \n " ,
header - > type ) ;
2005-04-16 15:20:36 -07:00
break ;
}
}
static int
2005-08-05 00:44:28 -04:00
acpi_table_compute_checksum ( void * table_pointer , unsigned long length )
2005-04-16 15:20:36 -07:00
{
2005-08-05 00:44:28 -04:00
u8 * p = ( u8 * ) table_pointer ;
unsigned long remains = length ;
unsigned long sum = 0 ;
2005-04-16 15:20:36 -07:00
if ( ! p | | ! length )
return - EINVAL ;
while ( remains - - )
sum + = * p + + ;
return ( sum & 0xFF ) ;
}
/*
* acpi_get_table_header_early ( )
* for acpi_blacklisted ( ) , acpi_table_get_sdt ( )
*/
int __init
2005-08-05 00:44:28 -04:00
acpi_get_table_header_early ( enum acpi_table_id id ,
struct acpi_table_header * * header )
2005-04-16 15:20:36 -07:00
{
unsigned int i ;
enum acpi_table_id temp_id ;
/* DSDT is different from the rest */
if ( id = = ACPI_DSDT )
temp_id = ACPI_FADT ;
else
temp_id = id ;
/* Locate the table. */
for ( i = 0 ; i < sdt_count ; i + + ) {
if ( sdt_entry [ i ] . id ! = temp_id )
continue ;
* header = ( void * )
2005-08-05 00:44:28 -04:00
__acpi_map_table ( sdt_entry [ i ] . pa , sdt_entry [ i ] . size ) ;
2005-04-16 15:20:36 -07:00
if ( ! * header ) {
printk ( KERN_WARNING PREFIX " Unable to map %s \n " ,
acpi_table_signatures [ temp_id ] ) ;
return - ENODEV ;
}
break ;
}
if ( ! * header ) {
printk ( KERN_WARNING PREFIX " %s not present \n " ,
acpi_table_signatures [ id ] ) ;
return - ENODEV ;
}
/* Map the DSDT header via the pointer in the FADT */
if ( id = = ACPI_DSDT ) {
2005-08-05 00:44:28 -04:00
struct fadt_descriptor_rev2 * fadt =
( struct fadt_descriptor_rev2 * ) * header ;
2005-04-16 15:20:36 -07:00
if ( fadt - > revision = = 3 & & fadt - > Xdsdt ) {
2005-08-05 00:44:28 -04:00
* header = ( void * ) __acpi_map_table ( fadt - > Xdsdt ,
sizeof ( struct
acpi_table_header ) ) ;
2005-04-16 15:20:36 -07:00
} else if ( fadt - > V1_dsdt ) {
2005-08-05 00:44:28 -04:00
* header = ( void * ) __acpi_map_table ( fadt - > V1_dsdt ,
sizeof ( struct
acpi_table_header ) ) ;
2005-04-16 15:20:36 -07:00
} else
* header = NULL ;
if ( ! * header ) {
printk ( KERN_WARNING PREFIX " Unable to map DSDT \n " ) ;
return - ENODEV ;
}
}
return 0 ;
}
int __init
2005-08-05 00:44:28 -04:00
acpi_table_parse_madt_family ( enum acpi_table_id id ,
unsigned long madt_size ,
int entry_id ,
acpi_madt_entry_handler handler ,
unsigned int max_entries )
2005-04-16 15:20:36 -07:00
{
2005-08-05 00:44:28 -04:00
void * madt = NULL ;
acpi_table_entry_header * entry ;
unsigned int count = 0 ;
unsigned long madt_end ;
unsigned int i ;
2005-04-16 15:20:36 -07:00
if ( ! handler )
return - EINVAL ;
/* Locate the MADT (if exists). There should only be one. */
for ( i = 0 ; i < sdt_count ; i + + ) {
if ( sdt_entry [ i ] . id ! = id )
continue ;
madt = ( void * )
2005-08-05 00:44:28 -04:00
__acpi_map_table ( sdt_entry [ i ] . pa , sdt_entry [ i ] . size ) ;
2005-04-16 15:20:36 -07:00
if ( ! madt ) {
printk ( KERN_WARNING PREFIX " Unable to map %s \n " ,
acpi_table_signatures [ id ] ) ;
return - ENODEV ;
}
break ;
}
if ( ! madt ) {
printk ( KERN_WARNING PREFIX " %s not present \n " ,
acpi_table_signatures [ id ] ) ;
return - ENODEV ;
}
2005-08-05 00:44:28 -04:00
madt_end = ( unsigned long ) madt + sdt_entry [ i ] . size ;
2005-04-16 15:20:36 -07:00
/* Parse all entries looking for a match. */
entry = ( acpi_table_entry_header * )
2005-08-05 00:44:28 -04:00
( ( unsigned long ) madt + madt_size ) ;
2005-04-16 15:20:36 -07:00
2005-08-05 00:44:28 -04:00
while ( ( ( unsigned long ) entry ) + sizeof ( acpi_table_entry_header ) <
madt_end ) {
if ( entry - > type = = entry_id
& & ( ! max_entries | | count + + < max_entries ) )
2005-04-16 15:20:36 -07:00
if ( handler ( entry , madt_end ) )
return - EINVAL ;
entry = ( acpi_table_entry_header * )
2005-08-05 00:44:28 -04:00
( ( unsigned long ) entry + entry - > length ) ;
2005-04-16 15:20:36 -07:00
}
if ( max_entries & & count > max_entries ) {
printk ( KERN_WARNING PREFIX " [%s:0x%02x] ignored %i entries of "
" %i found \n " , acpi_table_signatures [ id ] , entry_id ,
count - max_entries , count ) ;
}
return count ;
}
int __init
2005-08-05 00:44:28 -04:00
acpi_table_parse_madt ( enum acpi_madt_entry_id id ,
acpi_madt_entry_handler handler , unsigned int max_entries )
2005-04-16 15:20:36 -07:00
{
2005-08-05 00:44:28 -04:00
return acpi_table_parse_madt_family ( ACPI_APIC ,
sizeof ( struct acpi_table_madt ) , id ,
handler , max_entries ) ;
2005-04-16 15:20:36 -07:00
}
2005-08-05 00:44:28 -04:00
int __init acpi_table_parse ( enum acpi_table_id id , acpi_table_handler handler )
2005-04-16 15:20:36 -07:00
{
2005-08-05 00:44:28 -04:00
int count = 0 ;
unsigned int i = 0 ;
2005-04-16 15:20:36 -07:00
if ( ! handler )
return - EINVAL ;
for ( i = 0 ; i < sdt_count ; i + + ) {
if ( sdt_entry [ i ] . id ! = id )
continue ;
count + + ;
if ( count = = 1 )
handler ( sdt_entry [ i ] . pa , sdt_entry [ i ] . size ) ;
else
2005-08-05 00:44:28 -04:00
printk ( KERN_WARNING PREFIX
" %d duplicate %s table ignored. \n " , count ,
acpi_table_signatures [ id ] ) ;
2005-04-16 15:20:36 -07:00
}
return count ;
}
2005-08-05 00:44:28 -04:00
static int __init acpi_table_get_sdt ( struct acpi_table_rsdp * rsdp )
2005-04-16 15:20:36 -07:00
{
struct acpi_table_header * header = NULL ;
2005-08-05 00:44:28 -04:00
unsigned int i , id = 0 ;
2005-04-16 15:20:36 -07:00
if ( ! rsdp )
return - EINVAL ;
/* First check XSDT (but only on ACPI 2.0-compatible systems) */
if ( ( rsdp - > revision > = 2 ) & &
2005-08-05 00:44:28 -04:00
( ( ( struct acpi20_table_rsdp * ) rsdp ) - > xsdt_address ) ) {
struct acpi_table_xsdt * mapped_xsdt = NULL ;
2005-04-16 15:20:36 -07:00
2005-08-05 00:44:28 -04:00
sdt_pa = ( ( struct acpi20_table_rsdp * ) rsdp ) - > xsdt_address ;
2005-04-16 15:20:36 -07:00
/* map in just the header */
header = ( struct acpi_table_header * )
2005-08-05 00:44:28 -04:00
__acpi_map_table ( sdt_pa , sizeof ( struct acpi_table_header ) ) ;
2005-04-16 15:20:36 -07:00
if ( ! header ) {
2005-08-05 00:44:28 -04:00
printk ( KERN_WARNING PREFIX
" Unable to map XSDT header \n " ) ;
2005-04-16 15:20:36 -07:00
return - ENODEV ;
}
/* remap in the entire table before processing */
mapped_xsdt = ( struct acpi_table_xsdt * )
2005-08-05 00:44:28 -04:00
__acpi_map_table ( sdt_pa , header - > length ) ;
2005-04-16 15:20:36 -07:00
if ( ! mapped_xsdt ) {
printk ( KERN_WARNING PREFIX " Unable to map XSDT \n " ) ;
return - ENODEV ;
}
header = & mapped_xsdt - > header ;
if ( strncmp ( header - > signature , " XSDT " , 4 ) ) {
2005-08-05 00:44:28 -04:00
printk ( KERN_WARNING PREFIX
" XSDT signature incorrect \n " ) ;
2005-04-16 15:20:36 -07:00
return - ENODEV ;
}
if ( acpi_table_compute_checksum ( header , header - > length ) ) {
printk ( KERN_WARNING PREFIX " Invalid XSDT checksum \n " ) ;
return - ENODEV ;
}
2005-08-05 00:44:28 -04:00
sdt_count =
( header - > length - sizeof ( struct acpi_table_header ) ) > > 3 ;
2005-04-16 15:20:36 -07:00
if ( sdt_count > ACPI_MAX_TABLES ) {
2005-08-05 00:44:28 -04:00
printk ( KERN_WARNING PREFIX
" Truncated %lu XSDT entries \n " ,
( sdt_count - ACPI_MAX_TABLES ) ) ;
2005-04-16 15:20:36 -07:00
sdt_count = ACPI_MAX_TABLES ;
}
for ( i = 0 ; i < sdt_count ; i + + )
2005-08-05 00:44:28 -04:00
sdt_entry [ i ] . pa = ( unsigned long ) mapped_xsdt - > entry [ i ] ;
2005-04-16 15:20:36 -07:00
}
/* Then check RSDT */
else if ( rsdp - > rsdt_address ) {
2005-08-05 00:44:28 -04:00
struct acpi_table_rsdt * mapped_rsdt = NULL ;
2005-04-16 15:20:36 -07:00
sdt_pa = rsdp - > rsdt_address ;
/* map in just the header */
header = ( struct acpi_table_header * )
2005-08-05 00:44:28 -04:00
__acpi_map_table ( sdt_pa , sizeof ( struct acpi_table_header ) ) ;
2005-04-16 15:20:36 -07:00
if ( ! header ) {
2005-08-05 00:44:28 -04:00
printk ( KERN_WARNING PREFIX
" Unable to map RSDT header \n " ) ;
2005-04-16 15:20:36 -07:00
return - ENODEV ;
}
/* remap in the entire table before processing */
mapped_rsdt = ( struct acpi_table_rsdt * )
2005-08-05 00:44:28 -04:00
__acpi_map_table ( sdt_pa , header - > length ) ;
2005-04-16 15:20:36 -07:00
if ( ! mapped_rsdt ) {
printk ( KERN_WARNING PREFIX " Unable to map RSDT \n " ) ;
return - ENODEV ;
}
header = & mapped_rsdt - > header ;
if ( strncmp ( header - > signature , " RSDT " , 4 ) ) {
2005-08-05 00:44:28 -04:00
printk ( KERN_WARNING PREFIX
" RSDT signature incorrect \n " ) ;
2005-04-16 15:20:36 -07:00
return - ENODEV ;
}
if ( acpi_table_compute_checksum ( header , header - > length ) ) {
printk ( KERN_WARNING PREFIX " Invalid RSDT checksum \n " ) ;
return - ENODEV ;
}
2005-08-05 00:44:28 -04:00
sdt_count =
( header - > length - sizeof ( struct acpi_table_header ) ) > > 2 ;
2005-04-16 15:20:36 -07:00
if ( sdt_count > ACPI_MAX_TABLES ) {
2005-08-05 00:44:28 -04:00
printk ( KERN_WARNING PREFIX
" Truncated %lu RSDT entries \n " ,
( sdt_count - ACPI_MAX_TABLES ) ) ;
2005-04-16 15:20:36 -07:00
sdt_count = ACPI_MAX_TABLES ;
}
for ( i = 0 ; i < sdt_count ; i + + )
2005-08-05 00:44:28 -04:00
sdt_entry [ i ] . pa = ( unsigned long ) mapped_rsdt - > entry [ i ] ;
2005-04-16 15:20:36 -07:00
}
else {
2005-08-05 00:44:28 -04:00
printk ( KERN_WARNING PREFIX
" No System Description Table (RSDT/XSDT) specified in RSDP \n " ) ;
2005-04-16 15:20:36 -07:00
return - ENODEV ;
}
acpi_table_print ( header , sdt_pa ) ;
for ( i = 0 ; i < sdt_count ; i + + ) {
/* map in just the header */
header = ( struct acpi_table_header * )
2005-08-05 00:44:28 -04:00
__acpi_map_table ( sdt_entry [ i ] . pa ,
sizeof ( struct acpi_table_header ) ) ;
2005-04-16 15:20:36 -07:00
if ( ! header )
continue ;
/* remap in the entire table before processing */
header = ( struct acpi_table_header * )
2005-08-05 00:44:28 -04:00
__acpi_map_table ( sdt_entry [ i ] . pa , header - > length ) ;
2005-04-16 15:20:36 -07:00
if ( ! header )
continue ;
2005-08-05 00:44:28 -04:00
2005-04-16 15:20:36 -07:00
acpi_table_print ( header , sdt_entry [ i ] . pa ) ;
if ( acpi_table_compute_checksum ( header , header - > length ) ) {
printk ( KERN_WARNING " >>> ERROR: Invalid checksum \n " ) ;
continue ;
}
sdt_entry [ i ] . size = header - > length ;
for ( id = 0 ; id < ACPI_TABLE_COUNT ; id + + ) {
2005-08-05 00:44:28 -04:00
if ( ! strncmp ( ( char * ) & header - > signature ,
acpi_table_signatures [ id ] ,
sizeof ( header - > signature ) ) ) {
2005-04-16 15:20:36 -07:00
sdt_entry [ i ] . id = id ;
}
}
}
/*
* The DSDT is * not * in the RSDT ( why not ? no idea . ) but we want
* to print its info , because this is what people usually blacklist
* against . Unfortunately , we don ' t know the phys_addr , so just
* print 0. Maybe no one will notice .
*/
2005-08-05 00:44:28 -04:00
if ( ! acpi_get_table_header_early ( ACPI_DSDT , & header ) )
2005-04-16 15:20:36 -07:00
acpi_table_print ( header , 0 ) ;
return 0 ;
}
/*
* acpi_table_init ( )
*
* find RSDP , find and checksum SDT / XSDT .
* checksum all tables , print SDT / XSDT
*
* result : sdt_entry [ ] is initialized
*/
2005-08-05 00:44:28 -04:00
int __init acpi_table_init ( void )
2005-04-16 15:20:36 -07:00
{
2005-08-05 00:44:28 -04:00
struct acpi_table_rsdp * rsdp = NULL ;
unsigned long rsdp_phys = 0 ;
int result = 0 ;
2005-04-16 15:20:36 -07:00
/* Locate and map the Root System Description Table (RSDP) */
rsdp_phys = acpi_find_rsdp ( ) ;
if ( ! rsdp_phys ) {
printk ( KERN_ERR PREFIX " Unable to locate RSDP \n " ) ;
return - ENODEV ;
}
2005-08-05 00:44:28 -04:00
rsdp = ( struct acpi_table_rsdp * ) __va ( rsdp_phys ) ;
2005-04-16 15:20:36 -07:00
if ( ! rsdp ) {
printk ( KERN_WARNING PREFIX " Unable to map RSDP \n " ) ;
return - ENODEV ;
}
2005-08-05 00:44:28 -04:00
printk ( KERN_DEBUG PREFIX
" RSDP (v%3.3d %6.6s ) @ 0x%p \n " ,
rsdp - > revision , rsdp - > oem_id , ( void * ) rsdp_phys ) ;
2005-04-16 15:20:36 -07:00
if ( rsdp - > revision < 2 )
2005-08-05 00:44:28 -04:00
result =
acpi_table_compute_checksum ( rsdp ,
sizeof ( struct acpi_table_rsdp ) ) ;
2005-04-16 15:20:36 -07:00
else
2005-08-05 00:44:28 -04:00
result =
acpi_table_compute_checksum ( rsdp ,
( ( struct acpi20_table_rsdp * )
rsdp ) - > length ) ;
2005-04-16 15:20:36 -07:00
if ( result ) {
printk ( KERN_WARNING " >>> ERROR: Invalid checksum \n " ) ;
return - ENODEV ;
}
/* Locate and map the System Description table (RSDT/XSDT) */
if ( acpi_table_get_sdt ( rsdp ) )
return - ENODEV ;
return 0 ;
}