2005-04-16 15:20:36 -07:00
/*
* acpi_numa . c - ACPI NUMA support
*
* Copyright ( C ) 2002 Takayoshi Kochi < t - kochi @ bq . jp . nec . 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/module.h>
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/types.h>
# include <linux/errno.h>
# include <linux/acpi.h>
# include <acpi/acpi_bus.h>
# include <acpi/acmacros.h>
# define ACPI_NUMA 0x80000000
# define _COMPONENT ACPI_NUMA
2005-08-05 00:44:28 -04:00
ACPI_MODULE_NAME ( " numa " )
2005-04-16 15:20:36 -07:00
2006-06-23 02:03:19 -07:00
static nodemask_t nodes_found_map = NODE_MASK_NONE ;
# define PXM_INVAL -1
# define NID_INVAL -1
/* maps to convert between proximity domain and logical node ID */
int __cpuinitdata pxm_to_node_map [ MAX_PXM_DOMAINS ]
= { [ 0 . . . MAX_PXM_DOMAINS - 1 ] = NID_INVAL } ;
int __cpuinitdata node_to_pxm_map [ MAX_NUMNODES ]
= { [ 0 . . . MAX_NUMNODES - 1 ] = PXM_INVAL } ;
2005-08-05 00:44:28 -04:00
extern int __init 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
2006-06-23 02:03:19 -07:00
int __cpuinit pxm_to_node ( int pxm )
{
if ( pxm < 0 )
return NID_INVAL ;
return pxm_to_node_map [ pxm ] ;
}
int __cpuinit node_to_pxm ( int node )
{
if ( node < 0 )
return PXM_INVAL ;
return node_to_pxm_map [ node ] ;
}
int __cpuinit acpi_map_pxm_to_node ( int pxm )
{
int node = pxm_to_node_map [ pxm ] ;
if ( node < 0 ) {
if ( nodes_weight ( nodes_found_map ) > = MAX_NUMNODES )
return NID_INVAL ;
node = first_unset_node ( nodes_found_map ) ;
pxm_to_node_map [ pxm ] = node ;
node_to_pxm_map [ node ] = pxm ;
node_set ( node , nodes_found_map ) ;
}
return node ;
}
void __cpuinit acpi_unmap_pxm_to_node ( int node )
{
int pxm = node_to_pxm_map [ node ] ;
pxm_to_node_map [ pxm ] = NID_INVAL ;
node_to_pxm_map [ node ] = PXM_INVAL ;
node_clear ( node , nodes_found_map ) ;
}
2005-08-05 00:44:28 -04:00
void __init acpi_table_print_srat_entry ( acpi_table_entry_header * header )
2005-04-16 15:20:36 -07:00
{
2005-08-05 00:44:28 -04:00
ACPI_FUNCTION_NAME ( " acpi_table_print_srat_entry " ) ;
2005-04-16 15:20:36 -07:00
if ( ! header )
return ;
switch ( header - > type ) {
case ACPI_SRAT_PROCESSOR_AFFINITY :
# ifdef ACPI_DEBUG_OUTPUT
2005-08-05 00:44:28 -04:00
{
struct acpi_table_processor_affinity * p =
( struct acpi_table_processor_affinity * ) header ;
ACPI_DEBUG_PRINT ( ( ACPI_DB_INFO ,
" SRAT Processor (id[0x%02x] eid[0x%02x]) in proximity domain %d %s \n " ,
p - > apic_id , p - > lsapic_eid ,
p - > proximity_domain ,
p - > flags .
enabled ? " enabled " : " disabled " ) ) ;
}
# endif /* ACPI_DEBUG_OUTPUT */
2005-04-16 15:20:36 -07:00
break ;
case ACPI_SRAT_MEMORY_AFFINITY :
# ifdef ACPI_DEBUG_OUTPUT
2005-08-05 00:44:28 -04:00
{
struct acpi_table_memory_affinity * p =
( struct acpi_table_memory_affinity * ) header ;
ACPI_DEBUG_PRINT ( ( ACPI_DB_INFO ,
" SRAT Memory (0x%08x%08x length 0x%08x%08x type 0x%x) in proximity domain %d %s%s \n " ,
p - > base_addr_hi , p - > base_addr_lo ,
p - > length_hi , p - > length_lo ,
p - > memory_type , p - > proximity_domain ,
p - > flags .
enabled ? " enabled " : " disabled " ,
p - > flags .
hot_pluggable ? " hot-pluggable " :
" " ) ) ;
}
# endif /* ACPI_DEBUG_OUTPUT */
2005-04-16 15:20:36 -07:00
break ;
default :
2005-08-05 00:44:28 -04:00
printk ( KERN_WARNING PREFIX
" Found unsupported SRAT entry (type = 0x%x) \n " ,
header - > type ) ;
2005-04-16 15:20:36 -07:00
break ;
}
}
2005-08-05 00:44:28 -04:00
static int __init acpi_parse_slit ( unsigned long phys_addr , unsigned long size )
2005-04-16 15:20:36 -07:00
{
2005-08-05 00:44:28 -04:00
struct acpi_table_slit * slit ;
u32 localities ;
2005-04-16 15:20:36 -07:00
if ( ! phys_addr | | ! size )
return - EINVAL ;
2005-08-05 00:44:28 -04:00
slit = ( struct acpi_table_slit * ) __va ( phys_addr ) ;
2005-04-16 15:20:36 -07:00
/* downcast just for %llu vs %lu for i386/ia64 */
localities = ( u32 ) slit - > localities ;
acpi_numa_slit_init ( slit ) ;
return 0 ;
}
static int __init
2005-08-05 00:44:28 -04:00
acpi_parse_processor_affinity ( acpi_table_entry_header * header ,
const unsigned long end )
2005-04-16 15:20:36 -07:00
{
struct acpi_table_processor_affinity * processor_affinity ;
2005-08-05 00:44:28 -04:00
processor_affinity = ( struct acpi_table_processor_affinity * ) header ;
2005-04-16 15:20:36 -07:00
if ( ! processor_affinity )
return - EINVAL ;
acpi_table_print_srat_entry ( header ) ;
/* let architecture-dependent part to do it */
acpi_numa_processor_affinity_init ( processor_affinity ) ;
return 0 ;
}
static int __init
2005-08-05 00:44:28 -04:00
acpi_parse_memory_affinity ( acpi_table_entry_header * header ,
const unsigned long end )
2005-04-16 15:20:36 -07:00
{
struct acpi_table_memory_affinity * memory_affinity ;
2005-08-05 00:44:28 -04:00
memory_affinity = ( struct acpi_table_memory_affinity * ) header ;
2005-04-16 15:20:36 -07:00
if ( ! memory_affinity )
return - EINVAL ;
acpi_table_print_srat_entry ( header ) ;
/* let architecture-dependent part to do it */
acpi_numa_memory_affinity_init ( memory_affinity ) ;
return 0 ;
}
2005-08-05 00:44:28 -04:00
static int __init acpi_parse_srat ( unsigned long phys_addr , unsigned long size )
2005-04-16 15:20:36 -07:00
{
2005-08-05 00:44:28 -04:00
struct acpi_table_srat * srat ;
2005-04-16 15:20:36 -07:00
if ( ! phys_addr | | ! size )
return - EINVAL ;
2005-08-05 00:44:28 -04:00
srat = ( struct acpi_table_srat * ) __va ( phys_addr ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
int __init
2005-08-05 00:44:28 -04:00
acpi_table_parse_srat ( enum acpi_srat_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_SRAT ,
sizeof ( struct acpi_table_srat ) , id ,
handler , max_entries ) ;
2005-04-16 15:20:36 -07:00
}
2005-08-05 00:44:28 -04:00
int __init acpi_numa_init ( void )
2005-04-16 15:20:36 -07:00
{
2005-08-05 00:44:28 -04:00
int result ;
2005-04-16 15:20:36 -07:00
/* SRAT: Static Resource Affinity Table */
result = acpi_table_parse ( ACPI_SRAT , acpi_parse_srat ) ;
if ( result > 0 ) {
result = acpi_table_parse_srat ( ACPI_SRAT_PROCESSOR_AFFINITY ,
acpi_parse_processor_affinity ,
NR_CPUS ) ;
2005-08-05 00:44:28 -04:00
result = acpi_table_parse_srat ( ACPI_SRAT_MEMORY_AFFINITY , acpi_parse_memory_affinity , NR_NODE_MEMBLKS ) ; // IA64 specific
2005-04-16 15:20:36 -07:00
}
/* SLIT: System Locality Information Table */
result = acpi_table_parse ( ACPI_SLIT , acpi_parse_slit ) ;
acpi_numa_arch_fixup ( ) ;
return 0 ;
}
2005-08-05 00:44:28 -04:00
int acpi_get_pxm ( acpi_handle h )
2005-04-16 15:20:36 -07:00
{
unsigned long pxm ;
acpi_status status ;
acpi_handle handle ;
acpi_handle phandle = h ;
do {
handle = phandle ;
status = acpi_evaluate_integer ( handle , " _PXM " , NULL , & pxm ) ;
if ( ACPI_SUCCESS ( status ) )
2006-10-01 00:28:50 +02:00
return pxm ;
2005-04-16 15:20:36 -07:00
status = acpi_get_parent ( handle , & phandle ) ;
2005-08-05 00:44:28 -04:00
} while ( ACPI_SUCCESS ( status ) ) ;
2005-04-16 15:20:36 -07:00
return - 1 ;
}
EXPORT_SYMBOL ( acpi_get_pxm ) ;
2006-06-27 02:53:31 -07:00
int acpi_get_node ( acpi_handle * handle )
{
int pxm , node = - 1 ;
pxm = acpi_get_pxm ( handle ) ;
if ( pxm > = 0 )
node = acpi_map_pxm_to_node ( pxm ) ;
2006-07-01 16:48:23 -04:00
return node ;
2006-06-27 02:53:31 -07:00
}
EXPORT_SYMBOL ( acpi_get_node ) ;