1 /******************************************************************************
 
   3  * Module Name: nswalk - Functions for walking the ACPI namespace
 
   5  *****************************************************************************/
 
   8  * Copyright (C) 2000 - 2008, Intel Corp.
 
  11  * Redistribution and use in source and binary forms, with or without
 
  12  * modification, are permitted provided that the following conditions
 
  14  * 1. Redistributions of source code must retain the above copyright
 
  15  *    notice, this list of conditions, and the following disclaimer,
 
  16  *    without modification.
 
  17  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
 
  18  *    substantially similar to the "NO WARRANTY" disclaimer below
 
  19  *    ("Disclaimer") and any redistribution must be conditioned upon
 
  20  *    including a substantially similar Disclaimer requirement for further
 
  21  *    binary redistribution.
 
  22  * 3. Neither the names of the above-listed copyright holders nor the names
 
  23  *    of any contributors may be used to endorse or promote products derived
 
  24  *    from this software without specific prior written permission.
 
  26  * Alternatively, this software may be distributed under the terms of the
 
  27  * GNU General Public License ("GPL") version 2 as published by the Free
 
  28  * Software Foundation.
 
  31  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 
  32  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 
  33  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
 
  34  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 
  35  * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 
  36  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 
  37  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 
  38  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 
  39  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
 
  40  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 
  41  * POSSIBILITY OF SUCH DAMAGES.
 
  44 #include <acpi/acpi.h>
 
  45 #include <acpi/acnamesp.h>
 
  47 #define _COMPONENT          ACPI_NAMESPACE
 
  48 ACPI_MODULE_NAME("nswalk")
 
  50 /*******************************************************************************
 
  52  * FUNCTION:    acpi_ns_get_next_node
 
  54  * PARAMETERS:  Type                - Type of node to be searched for
 
  55  *              parent_node         - Parent node whose children we are
 
  57  *              child_node          - Previous child that was found.
 
  58  *                                    The NEXT child will be returned
 
  60  * RETURN:      struct acpi_namespace_node - Pointer to the NEXT child or NULL if
 
  63  * DESCRIPTION: Return the next peer node within the namespace.  If Handle
 
  64  *              is valid, Scope is ignored.  Otherwise, the first node
 
  65  *              within Scope is returned.
 
  67  ******************************************************************************/
 
  68 struct acpi_namespace_node *acpi_ns_get_next_node(acpi_object_type type, struct acpi_namespace_node
 
  69                                                   *parent_node, struct acpi_namespace_node
 
  72         struct acpi_namespace_node *next_node = NULL;
 
  74         ACPI_FUNCTION_ENTRY();
 
  78                 /* It's really the parent's _scope_ that we want */
 
  80                 next_node = parent_node->child;
 
  84                 /* Start search at the NEXT node */
 
  86                 next_node = acpi_ns_get_next_valid_node(child_node);
 
  89         /* If any type is OK, we are done */
 
  91         if (type == ACPI_TYPE_ANY) {
 
  93                 /* next_node is NULL if we are at the end-of-list */
 
  98         /* Must search for the node -- but within this scope only */
 
 102                 /* If type matches, we are done */
 
 104                 if (next_node->type == type) {
 
 108                 /* Otherwise, move on to the next node */
 
 110                 next_node = acpi_ns_get_next_valid_node(next_node);
 
 118 /*******************************************************************************
 
 120  * FUNCTION:    acpi_ns_walk_namespace
 
 122  * PARAMETERS:  Type                - acpi_object_type to search for
 
 123  *              start_node          - Handle in namespace where search begins
 
 124  *              max_depth           - Depth to which search is to reach
 
 125  *              Flags               - Whether to unlock the NS before invoking
 
 126  *                                    the callback routine
 
 127  *              user_function       - Called when an object of "Type" is found
 
 128  *              Context             - Passed to user function
 
 129  *              return_value        - from the user_function if terminated early.
 
 130  *                                    Otherwise, returns NULL.
 
 133  * DESCRIPTION: Performs a modified depth-first walk of the namespace tree,
 
 134  *              starting (and ending) at the node specified by start_handle.
 
 135  *              The user_function is called whenever a node that matches
 
 136  *              the type parameter is found.  If the user function returns
 
 137  *              a non-zero value, the search is terminated immediately and this
 
 138  *              value is returned to the caller.
 
 140  *              The point of this procedure is to provide a generic namespace
 
 141  *              walk routine that can be called from multiple places to
 
 142  *              provide multiple services;  the User Function can be tailored
 
 143  *              to each task, whether it is a print function, a compare
 
 146  ******************************************************************************/
 
 149 acpi_ns_walk_namespace(acpi_object_type type,
 
 150                        acpi_handle start_node,
 
 153                        acpi_walk_callback user_function,
 
 154                        void *context, void **return_value)
 
 157         acpi_status mutex_status;
 
 158         struct acpi_namespace_node *child_node;
 
 159         struct acpi_namespace_node *parent_node;
 
 160         acpi_object_type child_type;
 
 163         ACPI_FUNCTION_TRACE(ns_walk_namespace);
 
 165         /* Special case for the namespace Root Node */
 
 167         if (start_node == ACPI_ROOT_OBJECT) {
 
 168                 start_node = acpi_gbl_root_node;
 
 171         /* Null child means "get first node" */
 
 173         parent_node = start_node;
 
 175         child_type = ACPI_TYPE_ANY;
 
 179          * Traverse the tree of nodes until we bubble back up to where we
 
 180          * started. When Level is zero, the loop is done because we have
 
 181          * bubbled up to (and passed) the original parent handle (start_entry)
 
 185                 /* Get the next node in this scope.  Null if not found */
 
 189                     acpi_ns_get_next_node(ACPI_TYPE_ANY, parent_node,
 
 193                         /* Found next child, get the type if we are not searching for ANY */
 
 195                         if (type != ACPI_TYPE_ANY) {
 
 196                                 child_type = child_node->type;
 
 200                          * Ignore all temporary namespace nodes (created during control
 
 201                          * method execution) unless told otherwise. These temporary nodes
 
 202                          * can cause a race condition because they can be deleted during the
 
 203                          * execution of the user function (if the namespace is unlocked before
 
 204                          * invocation of the user function.) Only the debugger namespace dump
 
 205                          * will examine the temporary nodes.
 
 207                         if ((child_node->flags & ANOBJ_TEMPORARY) &&
 
 208                             !(flags & ACPI_NS_WALK_TEMP_NODES)) {
 
 209                                 status = AE_CTRL_DEPTH;
 
 212                         /* Type must match requested type */
 
 214                         else if (child_type == type) {
 
 216                                  * Found a matching node, invoke the user callback function.
 
 217                                  * Unlock the namespace if flag is set.
 
 219                                 if (flags & ACPI_NS_WALK_UNLOCK) {
 
 221                                             acpi_ut_release_mutex
 
 222                                             (ACPI_MTX_NAMESPACE);
 
 223                                         if (ACPI_FAILURE(mutex_status)) {
 
 230                                     user_function(child_node, level, context,
 
 233                                 if (flags & ACPI_NS_WALK_UNLOCK) {
 
 235                                             acpi_ut_acquire_mutex
 
 236                                             (ACPI_MTX_NAMESPACE);
 
 237                                         if (ACPI_FAILURE(mutex_status)) {
 
 247                                         /* Just keep going */
 
 250                                 case AE_CTRL_TERMINATE:
 
 252                                         /* Exit now, with OK status */
 
 254                                         return_ACPI_STATUS(AE_OK);
 
 258                                         /* All others are valid exceptions */
 
 260                                         return_ACPI_STATUS(status);
 
 265                          * Depth first search: Attempt to go down another level in the
 
 266                          * namespace if we are allowed to.  Don't go any further if we have
 
 267                          * reached the caller specified maximum depth or if the user
 
 268                          * function has specified that the maximum depth has been reached.
 
 270                         if ((level < max_depth) && (status != AE_CTRL_DEPTH)) {
 
 271                                 if (acpi_ns_get_next_node
 
 272                                     (ACPI_TYPE_ANY, child_node, NULL)) {
 
 274                                         /* There is at least one child of this node, visit it */
 
 277                                         parent_node = child_node;
 
 283                          * No more children of this node (acpi_ns_get_next_node failed), go
 
 284                          * back upwards in the namespace tree to the node's parent.
 
 287                         child_node = parent_node;
 
 288                         parent_node = acpi_ns_get_parent_node(parent_node);
 
 292         /* Complete walk, not terminated by user function */
 
 294         return_ACPI_STATUS(AE_OK);