2005-04-17 06:20:36 +08:00
/******************************************************************************
*
* Module Name : nsinit - namespace initialization
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/*
2006-01-14 05:22:00 +08:00
* Copyright ( C ) 2000 - 2006 , R . Byron Moore
2005-04-17 06:20:36 +08:00
* All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
* 1. Redistributions of source code must retain the above copyright
* notice , this list of conditions , and the following disclaimer ,
* without modification .
* 2. Redistributions in binary form must reproduce at minimum a disclaimer
* substantially similar to the " NO WARRANTY " disclaimer below
* ( " Disclaimer " ) and any redistribution must be conditioned upon
* including a substantially similar Disclaimer requirement for further
* binary redistribution .
* 3. Neither the names of the above - listed copyright holders nor the names
* of any contributors may be used to endorse or promote products derived
* from this software without specific prior written permission .
*
* Alternatively , this software may be distributed under the terms of the
* GNU General Public License ( " GPL " ) version 2 as published by the Free
* Software Foundation .
*
* NO WARRANTY
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* " AS IS " AND ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT
* LIMITED TO , THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT
* HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL , EXEMPLARY , OR CONSEQUENTIAL
* DAMAGES ( INCLUDING , BUT NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES ; LOSS OF USE , DATA , OR PROFITS ; OR BUSINESS INTERRUPTION )
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY , WHETHER IN CONTRACT ,
* STRICT LIABILITY , OR TORT ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING
* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGES .
*/
# include <acpi/acpi.h>
# include <acpi/acnamesp.h>
# include <acpi/acdispat.h>
# include <acpi/acinterp.h>
# define _COMPONENT ACPI_NAMESPACE
2005-08-05 12:44:28 +08:00
ACPI_MODULE_NAME ( " nsinit " )
2005-04-17 06:20:36 +08:00
2005-04-19 10:49:35 +08:00
/* Local prototypes */
static acpi_status
2005-08-05 12:44:28 +08:00
acpi_ns_init_one_object ( acpi_handle obj_handle ,
u32 level , void * context , void * * return_value ) ;
2005-04-19 10:49:35 +08:00
static acpi_status
2005-08-05 12:44:28 +08:00
acpi_ns_init_one_device ( acpi_handle obj_handle ,
u32 nesting_level , void * context , void * * return_value ) ;
2005-04-17 06:20:36 +08:00
/*******************************************************************************
*
* FUNCTION : acpi_ns_initialize_objects
*
* PARAMETERS : None
*
* RETURN : Status
*
* DESCRIPTION : Walk the entire namespace and perform any necessary
* initialization on the objects found therein
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-08-05 12:44:28 +08:00
acpi_status acpi_ns_initialize_objects ( void )
2005-04-17 06:20:36 +08:00
{
2005-08-05 12:44:28 +08:00
acpi_status status ;
struct acpi_init_walk_info info ;
2005-04-17 06:20:36 +08:00
2005-08-05 12:44:28 +08:00
ACPI_FUNCTION_TRACE ( " ns_initialize_objects " ) ;
2005-04-17 06:20:36 +08:00
2005-08-05 12:44:28 +08:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_DISPATCH ,
" **** Starting initialization of namespace objects **** \n " ) ) ;
ACPI_DEBUG_PRINT_RAW ( ( ACPI_DB_INIT ,
" Completing Region/Field/Buffer/Package initialization: " ) ) ;
2005-04-17 06:20:36 +08:00
/* Set all init info to zero */
2005-08-05 12:44:28 +08:00
ACPI_MEMSET ( & info , 0 , sizeof ( struct acpi_init_walk_info ) ) ;
2005-04-17 06:20:36 +08:00
/* Walk entire namespace from the supplied root */
2005-08-05 12:44:28 +08:00
status = acpi_walk_namespace ( ACPI_TYPE_ANY , ACPI_ROOT_OBJECT ,
ACPI_UINT32_MAX , acpi_ns_init_one_object ,
& info , NULL ) ;
if ( ACPI_FAILURE ( status ) ) {
2006-01-14 05:22:00 +08:00
ACPI_REPORT_ERROR ( ( " walk_namespace failed! %s \n " ,
acpi_format_exception ( status ) ) ) ;
2005-04-17 06:20:36 +08:00
}
2005-08-05 12:44:28 +08:00
ACPI_DEBUG_PRINT_RAW ( ( ACPI_DB_INIT ,
" \n Initialized %hd/%hd Regions %hd/%hd Fields %hd/%hd Buffers %hd/%hd Packages (%hd nodes) \n " ,
info . op_region_init , info . op_region_count ,
info . field_init , info . field_count ,
info . buffer_init , info . buffer_count ,
info . package_init , info . package_count ,
info . object_count ) ) ;
2005-04-17 06:20:36 +08:00
2005-08-05 12:44:28 +08:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_DISPATCH ,
" %hd Control Methods found \n " , info . method_count ) ) ;
ACPI_DEBUG_PRINT ( ( ACPI_DB_DISPATCH ,
" %hd Op Regions found \n " , info . op_region_count ) ) ;
2005-04-17 06:20:36 +08:00
2005-08-05 12:44:28 +08:00
return_ACPI_STATUS ( AE_OK ) ;
2005-04-17 06:20:36 +08:00
}
/*******************************************************************************
*
* FUNCTION : acpi_ns_initialize_devices
*
* PARAMETERS : None
*
* RETURN : acpi_status
*
* DESCRIPTION : Walk the entire namespace and initialize all ACPI devices .
* This means running _INI on all present devices .
*
* Note : We install PCI config space handler on region access ,
* not here .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-08-05 12:44:28 +08:00
acpi_status acpi_ns_initialize_devices ( void )
2005-04-17 06:20:36 +08:00
{
2005-08-05 12:44:28 +08:00
acpi_status status ;
struct acpi_device_walk_info info ;
2005-04-17 06:20:36 +08:00
2005-08-05 12:44:28 +08:00
ACPI_FUNCTION_TRACE ( " ns_initialize_devices " ) ;
2005-04-17 06:20:36 +08:00
/* Init counters */
info . device_count = 0 ;
info . num_STA = 0 ;
info . num_INI = 0 ;
2005-08-05 12:44:28 +08:00
ACPI_DEBUG_PRINT_RAW ( ( ACPI_DB_INIT ,
" Executing all Device _STA and_INI methods: " ) ) ;
2005-04-17 06:20:36 +08:00
2005-08-05 12:44:28 +08:00
status = acpi_ut_acquire_mutex ( ACPI_MTX_NAMESPACE ) ;
if ( ACPI_FAILURE ( status ) ) {
return_ACPI_STATUS ( status ) ;
2005-04-17 06:20:36 +08:00
}
/* Walk namespace for all objects */
2005-08-05 12:44:28 +08:00
status = acpi_ns_walk_namespace ( ACPI_TYPE_ANY , ACPI_ROOT_OBJECT ,
ACPI_UINT32_MAX , TRUE ,
acpi_ns_init_one_device , & info , NULL ) ;
2005-04-17 06:20:36 +08:00
2005-08-05 12:44:28 +08:00
( void ) acpi_ut_release_mutex ( ACPI_MTX_NAMESPACE ) ;
2005-04-17 06:20:36 +08:00
2005-08-05 12:44:28 +08:00
if ( ACPI_FAILURE ( status ) ) {
2006-01-14 05:22:00 +08:00
ACPI_REPORT_ERROR ( ( " walk_namespace failed! %s \n " ,
acpi_format_exception ( status ) ) ) ;
2005-04-17 06:20:36 +08:00
}
2005-08-05 12:44:28 +08:00
ACPI_DEBUG_PRINT_RAW ( ( ACPI_DB_INIT ,
" \n %hd Devices found containing: %hd _STA, %hd _INI methods \n " ,
info . device_count , info . num_STA , info . num_INI ) ) ;
2005-04-17 06:20:36 +08:00
2005-08-05 12:44:28 +08:00
return_ACPI_STATUS ( status ) ;
2005-04-17 06:20:36 +08:00
}
/*******************************************************************************
*
* FUNCTION : acpi_ns_init_one_object
*
* PARAMETERS : obj_handle - Node
* Level - Current nesting level
* Context - Points to a init info struct
* return_value - Not used
*
* RETURN : Status
*
* DESCRIPTION : Callback from acpi_walk_namespace . Invoked for every object
* within the namespace .
*
* Currently , the only objects that require initialization are :
* 1 ) Methods
* 2 ) Op Regions
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-04-19 10:49:35 +08:00
static acpi_status
2005-08-05 12:44:28 +08:00
acpi_ns_init_one_object ( acpi_handle obj_handle ,
u32 level , void * context , void * * return_value )
2005-04-17 06:20:36 +08:00
{
2005-08-05 12:44:28 +08:00
acpi_object_type type ;
acpi_status status ;
struct acpi_init_walk_info * info =
( struct acpi_init_walk_info * ) context ;
struct acpi_namespace_node * node =
( struct acpi_namespace_node * ) obj_handle ;
union acpi_operand_object * obj_desc ;
2005-04-17 06:20:36 +08:00
2005-08-05 12:44:28 +08:00
ACPI_FUNCTION_NAME ( " ns_init_one_object " ) ;
2005-04-17 06:20:36 +08:00
info - > object_count + + ;
/* And even then, we are only interested in a few object types */
2005-08-05 12:44:28 +08:00
type = acpi_ns_get_type ( obj_handle ) ;
obj_desc = acpi_ns_get_attached_object ( node ) ;
2005-04-17 06:20:36 +08:00
if ( ! obj_desc ) {
return ( AE_OK ) ;
}
/* Increment counters for object types we are looking for */
switch ( type ) {
case ACPI_TYPE_REGION :
info - > op_region_count + + ;
break ;
case ACPI_TYPE_BUFFER_FIELD :
info - > field_count + + ;
break ;
case ACPI_TYPE_BUFFER :
info - > buffer_count + + ;
break ;
case ACPI_TYPE_PACKAGE :
info - > package_count + + ;
break ;
default :
/* No init required, just exit now */
return ( AE_OK ) ;
}
/*
* If the object is already initialized , nothing else to do
*/
if ( obj_desc - > common . flags & AOPOBJ_DATA_VALID ) {
return ( AE_OK ) ;
}
/*
* Must lock the interpreter before executing AML code
*/
2005-08-05 12:44:28 +08:00
status = acpi_ex_enter_interpreter ( ) ;
if ( ACPI_FAILURE ( status ) ) {
2005-04-17 06:20:36 +08:00
return ( status ) ;
}
/*
* Each of these types can contain executable AML code within the
* declaration .
*/
switch ( type ) {
case ACPI_TYPE_REGION :
info - > op_region_init + + ;
2005-08-05 12:44:28 +08:00
status = acpi_ds_get_region_arguments ( obj_desc ) ;
2005-04-17 06:20:36 +08:00
break ;
case ACPI_TYPE_BUFFER_FIELD :
info - > field_init + + ;
2005-08-05 12:44:28 +08:00
status = acpi_ds_get_buffer_field_arguments ( obj_desc ) ;
2005-04-17 06:20:36 +08:00
break ;
case ACPI_TYPE_BUFFER :
info - > buffer_init + + ;
2005-08-05 12:44:28 +08:00
status = acpi_ds_get_buffer_arguments ( obj_desc ) ;
2005-04-17 06:20:36 +08:00
break ;
case ACPI_TYPE_PACKAGE :
info - > package_init + + ;
2005-08-05 12:44:28 +08:00
status = acpi_ds_get_package_arguments ( obj_desc ) ;
2005-04-17 06:20:36 +08:00
break ;
default :
/* No other types can get here */
break ;
}
2005-08-05 12:44:28 +08:00
if ( ACPI_FAILURE ( status ) ) {
2006-01-14 05:22:00 +08:00
ACPI_REPORT_ERROR ( ( " \n Could not execute arguments for [%4.4s] (%s), %s \n " , acpi_ut_get_node_name ( node ) , acpi_ut_get_type_name ( type ) , acpi_format_exception ( status ) ) ) ;
2005-04-17 06:20:36 +08:00
}
/*
* Print a dot for each object unless we are going to print the entire
* pathname
*/
if ( ! ( acpi_dbg_level & ACPI_LV_INIT_NAMES ) ) {
2005-08-05 12:44:28 +08:00
ACPI_DEBUG_PRINT_RAW ( ( ACPI_DB_INIT , " . " ) ) ;
2005-04-17 06:20:36 +08:00
}
/*
* We ignore errors from above , and always return OK , since we don ' t want
* to abort the walk on any single error .
*/
2005-08-05 12:44:28 +08:00
acpi_ex_exit_interpreter ( ) ;
2005-04-17 06:20:36 +08:00
return ( AE_OK ) ;
}
/*******************************************************************************
*
* FUNCTION : acpi_ns_init_one_device
*
* PARAMETERS : acpi_walk_callback
*
* RETURN : acpi_status
*
* DESCRIPTION : This is called once per device soon after ACPI is enabled
* to initialize each device . It determines if the device is
* present , and if so , calls _INI .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-04-19 10:49:35 +08:00
static acpi_status
2005-08-05 12:44:28 +08:00
acpi_ns_init_one_device ( acpi_handle obj_handle ,
u32 nesting_level , void * context , void * * return_value )
2005-04-17 06:20:36 +08:00
{
2005-08-05 12:44:28 +08:00
struct acpi_device_walk_info * info =
( struct acpi_device_walk_info * ) context ;
struct acpi_parameter_info pinfo ;
u32 flags ;
acpi_status status ;
2005-12-17 06:05:00 +08:00
struct acpi_namespace_node * ini_node ;
struct acpi_namespace_node * device_node ;
2005-04-17 06:20:36 +08:00
2005-08-05 12:44:28 +08:00
ACPI_FUNCTION_TRACE ( " ns_init_one_device " ) ;
2005-04-17 06:20:36 +08:00
2005-12-17 06:05:00 +08:00
device_node = acpi_ns_map_handle_to_node ( obj_handle ) ;
if ( ! device_node ) {
2005-08-05 12:44:28 +08:00
return_ACPI_STATUS ( AE_BAD_PARAMETER ) ;
2005-04-17 06:20:36 +08:00
}
/*
* We will run _STA / _INI on Devices , Processors and thermal_zones only
*/
2005-12-17 06:05:00 +08:00
if ( ( device_node - > type ! = ACPI_TYPE_DEVICE ) & &
( device_node - > type ! = ACPI_TYPE_PROCESSOR ) & &
( device_node - > type ! = ACPI_TYPE_THERMAL ) ) {
2005-08-05 12:44:28 +08:00
return_ACPI_STATUS ( AE_OK ) ;
2005-04-17 06:20:36 +08:00
}
if ( ( acpi_dbg_level < = ACPI_LV_ALL_EXCEPTIONS ) & &
2005-08-05 12:44:28 +08:00
( ! ( acpi_dbg_level & ACPI_LV_INFO ) ) ) {
ACPI_DEBUG_PRINT_RAW ( ( ACPI_DB_INIT , " . " ) ) ;
2005-04-17 06:20:36 +08:00
}
info - > device_count + + ;
/*
2005-12-17 06:05:00 +08:00
* Check if the _INI method exists for this device -
* if _INI does not exist , there is no need to run _STA
* No _INI means device requires no initialization
*/
status = acpi_ns_search_node ( * ACPI_CAST_PTR ( u32 , METHOD_NAME__INI ) ,
device_node , ACPI_TYPE_METHOD , & ini_node ) ;
if ( ACPI_FAILURE ( status ) ) {
/* No _INI method found - move on to next device */
return_ACPI_STATUS ( AE_OK ) ;
}
/*
* Run _STA to determine if we can run _INI on the device -
* the device must be present before _INI can be run .
* However , _STA is not required - assume device present if no _STA
2005-04-17 06:20:36 +08:00
*/
2005-08-05 12:44:28 +08:00
ACPI_DEBUG_EXEC ( acpi_ut_display_init_pathname ( ACPI_TYPE_METHOD ,
2005-12-17 06:05:00 +08:00
device_node ,
2005-08-05 12:44:28 +08:00
METHOD_NAME__STA ) ) ;
2005-04-17 06:20:36 +08:00
2005-12-17 06:05:00 +08:00
pinfo . node = device_node ;
pinfo . parameters = NULL ;
pinfo . parameter_type = ACPI_PARAM_ARGS ;
status = acpi_ut_execute_STA ( pinfo . node , & flags ) ;
2005-08-05 12:44:28 +08:00
if ( ACPI_FAILURE ( status ) ) {
2005-12-17 06:05:00 +08:00
/* Ignore error and move on to next device */
2005-04-17 06:20:36 +08:00
2005-12-17 06:05:00 +08:00
return_ACPI_STATUS ( AE_OK ) ;
}
2005-04-17 06:20:36 +08:00
2005-12-17 06:05:00 +08:00
if ( flags ! = ACPI_UINT32_MAX ) {
2005-04-17 06:20:36 +08:00
info - > num_STA + + ;
2005-12-17 06:05:00 +08:00
}
2005-04-17 06:20:36 +08:00
2005-12-17 06:05:00 +08:00
if ( ! ( flags & ACPI_STA_DEVICE_PRESENT ) ) {
/* Don't look at children of a not present device */
2005-04-17 06:20:36 +08:00
2005-12-17 06:05:00 +08:00
return_ACPI_STATUS ( AE_CTRL_DEPTH ) ;
2005-04-17 06:20:36 +08:00
}
/*
2005-12-17 06:05:00 +08:00
* The device is present and _INI exists . Run the _INI method .
* ( We already have the _INI node from above )
2005-04-17 06:20:36 +08:00
*/
2005-08-05 12:44:28 +08:00
ACPI_DEBUG_EXEC ( acpi_ut_display_init_pathname ( ACPI_TYPE_METHOD ,
pinfo . node ,
METHOD_NAME__INI ) ) ;
2005-04-17 06:20:36 +08:00
2005-12-17 06:05:00 +08:00
pinfo . node = ini_node ;
status = acpi_ns_evaluate_by_handle ( & pinfo ) ;
if ( ACPI_FAILURE ( status ) ) {
/* Ignore error and move on to next device */
2005-04-17 06:20:36 +08:00
# ifdef ACPI_DEBUG_OUTPUT
2005-12-17 06:05:00 +08:00
char * scope_name = acpi_ns_get_external_pathname ( ini_node ) ;
2005-04-17 06:20:36 +08:00
2006-01-14 05:22:00 +08:00
ACPI_REPORT_WARNING ( ( " %s._INI failed: %s \n " ,
scope_name ,
acpi_format_exception ( status ) ) ) ;
2005-04-17 06:20:36 +08:00
2005-12-17 06:05:00 +08:00
ACPI_MEM_FREE ( scope_name ) ;
2005-04-17 06:20:36 +08:00
# endif
2005-08-05 12:44:28 +08:00
} else {
2005-04-17 06:20:36 +08:00
/* Delete any return object (especially if implicit_return is enabled) */
if ( pinfo . return_object ) {
2005-08-05 12:44:28 +08:00
acpi_ut_remove_reference ( pinfo . return_object ) ;
2005-04-17 06:20:36 +08:00
}
/* Count of successful INIs */
info - > num_INI + + ;
}
if ( acpi_gbl_init_handler ) {
/* External initialization handler is present, call it */
2005-08-05 12:44:28 +08:00
status =
acpi_gbl_init_handler ( pinfo . node , ACPI_INIT_DEVICE_INI ) ;
2005-04-17 06:20:36 +08:00
}
2005-12-17 06:05:00 +08:00
return_ACPI_STATUS ( AE_OK ) ;
2005-04-17 06:20:36 +08:00
}