1 /******************************************************************************* 2 * 3 * Module Name: nsalloc - Namespace allocation and deletion utilities 4 * 5 ******************************************************************************/ 6 7 /* 8 * Copyright (C) 2000 - 2018, Intel Corp. 9 * All rights reserved. 10 * 11 * Redistribution and use in source and binary forms, with or without 12 * modification, are permitted provided that the following conditions 13 * are met: 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. 25 * 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. 29 * 30 * NO WARRANTY 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. 42 */ 43 44 #include <acpi/acpi.h> 45 #include "accommon.h" 46 #include "acnamesp.h" 47 48 #define _COMPONENT ACPI_NAMESPACE 49 ACPI_MODULE_NAME("nsalloc") 50 51 /******************************************************************************* 52 * 53 * FUNCTION: acpi_ns_create_node 54 * 55 * PARAMETERS: name - Name of the new node (4 char ACPI name) 56 * 57 * RETURN: New namespace node (Null on failure) 58 * 59 * DESCRIPTION: Create a namespace node 60 * 61 ******************************************************************************/ 62 struct acpi_namespace_node *acpi_ns_create_node(u32 name) 63 { 64 struct acpi_namespace_node *node; 65 #ifdef ACPI_DBG_TRACK_ALLOCATIONS 66 u32 temp; 67 #endif 68 69 ACPI_FUNCTION_TRACE(ns_create_node); 70 71 node = acpi_os_acquire_object(acpi_gbl_namespace_cache); 72 if (!node) { 73 return_PTR(NULL); 74 } 75 76 ACPI_MEM_TRACKING(acpi_gbl_ns_node_list->total_allocated++); 77 78 #ifdef ACPI_DBG_TRACK_ALLOCATIONS 79 temp = acpi_gbl_ns_node_list->total_allocated - 80 acpi_gbl_ns_node_list->total_freed; 81 if (temp > acpi_gbl_ns_node_list->max_occupied) { 82 acpi_gbl_ns_node_list->max_occupied = temp; 83 } 84 #endif 85 86 node->name.integer = name; 87 ACPI_SET_DESCRIPTOR_TYPE(node, ACPI_DESC_TYPE_NAMED); 88 return_PTR(node); 89 } 90 91 /******************************************************************************* 92 * 93 * FUNCTION: acpi_ns_delete_node 94 * 95 * PARAMETERS: node - Node to be deleted 96 * 97 * RETURN: None 98 * 99 * DESCRIPTION: Delete a namespace node. All node deletions must come through 100 * here. Detaches any attached objects, including any attached 101 * data. If a handler is associated with attached data, it is 102 * invoked before the node is deleted. 103 * 104 ******************************************************************************/ 105 106 void acpi_ns_delete_node(struct acpi_namespace_node *node) 107 { 108 union acpi_operand_object *obj_desc; 109 union acpi_operand_object *next_desc; 110 111 ACPI_FUNCTION_NAME(ns_delete_node); 112 113 /* Detach an object if there is one */ 114 115 acpi_ns_detach_object(node); 116 117 /* 118 * Delete an attached data object list if present (objects that were 119 * attached via acpi_attach_data). Note: After any normal object is 120 * detached above, the only possible remaining object(s) are data 121 * objects, in a linked list. 122 */ 123 obj_desc = node->object; 124 while (obj_desc && (obj_desc->common.type == ACPI_TYPE_LOCAL_DATA)) { 125 126 /* Invoke the attached data deletion handler if present */ 127 128 if (obj_desc->data.handler) { 129 obj_desc->data.handler(node, obj_desc->data.pointer); 130 } 131 132 next_desc = obj_desc->common.next_object; 133 acpi_ut_remove_reference(obj_desc); 134 obj_desc = next_desc; 135 } 136 137 /* Special case for the statically allocated root node */ 138 139 if (node == acpi_gbl_root_node) { 140 return; 141 } 142 143 /* Now we can delete the node */ 144 145 (void)acpi_os_release_object(acpi_gbl_namespace_cache, node); 146 147 ACPI_MEM_TRACKING(acpi_gbl_ns_node_list->total_freed++); 148 ACPI_DEBUG_PRINT((ACPI_DB_ALLOCATIONS, "Node %p, Remaining %X\n", 149 node, acpi_gbl_current_node_count)); 150 } 151 152 /******************************************************************************* 153 * 154 * FUNCTION: acpi_ns_remove_node 155 * 156 * PARAMETERS: node - Node to be removed/deleted 157 * 158 * RETURN: None 159 * 160 * DESCRIPTION: Remove (unlink) and delete a namespace node 161 * 162 ******************************************************************************/ 163 164 void acpi_ns_remove_node(struct acpi_namespace_node *node) 165 { 166 struct acpi_namespace_node *parent_node; 167 struct acpi_namespace_node *prev_node; 168 struct acpi_namespace_node *next_node; 169 170 ACPI_FUNCTION_TRACE_PTR(ns_remove_node, node); 171 172 parent_node = node->parent; 173 174 prev_node = NULL; 175 next_node = parent_node->child; 176 177 /* Find the node that is the previous peer in the parent's child list */ 178 179 while (next_node != node) { 180 prev_node = next_node; 181 next_node = next_node->peer; 182 } 183 184 if (prev_node) { 185 186 /* Node is not first child, unlink it */ 187 188 prev_node->peer = node->peer; 189 } else { 190 /* 191 * Node is first child (has no previous peer). 192 * Link peer list to parent 193 */ 194 parent_node->child = node->peer; 195 } 196 197 /* Delete the node and any attached objects */ 198 199 acpi_ns_delete_node(node); 200 return_VOID; 201 } 202 203 /******************************************************************************* 204 * 205 * FUNCTION: acpi_ns_install_node 206 * 207 * PARAMETERS: walk_state - Current state of the walk 208 * parent_node - The parent of the new Node 209 * node - The new Node to install 210 * type - ACPI object type of the new Node 211 * 212 * RETURN: None 213 * 214 * DESCRIPTION: Initialize a new namespace node and install it amongst 215 * its peers. 216 * 217 * Note: Current namespace lookup is linear search. This appears 218 * to be sufficient as namespace searches consume only a small 219 * fraction of the execution time of the ACPI subsystem. 220 * 221 ******************************************************************************/ 222 223 void acpi_ns_install_node(struct acpi_walk_state *walk_state, struct acpi_namespace_node *parent_node, /* Parent */ 224 struct acpi_namespace_node *node, /* New Child */ 225 acpi_object_type type) 226 { 227 acpi_owner_id owner_id = 0; 228 struct acpi_namespace_node *child_node; 229 230 ACPI_FUNCTION_TRACE(ns_install_node); 231 232 if (walk_state) { 233 /* 234 * Get the owner ID from the Walk state. The owner ID is used to 235 * track table deletion and deletion of objects created by methods. 236 */ 237 owner_id = walk_state->owner_id; 238 239 if ((walk_state->method_desc) && 240 (parent_node != walk_state->method_node)) { 241 /* 242 * A method is creating a new node that is not a child of the 243 * method (it is non-local). Mark the executing method as having 244 * modified the namespace. This is used for cleanup when the 245 * method exits. 246 */ 247 walk_state->method_desc->method.info_flags |= 248 ACPI_METHOD_MODIFIED_NAMESPACE; 249 } 250 } 251 252 /* Link the new entry into the parent and existing children */ 253 254 node->peer = NULL; 255 node->parent = parent_node; 256 child_node = parent_node->child; 257 258 if (!child_node) { 259 parent_node->child = node; 260 } else { 261 /* Add node to the end of the peer list */ 262 263 while (child_node->peer) { 264 child_node = child_node->peer; 265 } 266 267 child_node->peer = node; 268 } 269 270 /* Init the new entry */ 271 272 node->owner_id = owner_id; 273 node->type = (u8) type; 274 275 ACPI_DEBUG_PRINT((ACPI_DB_NAMES, 276 "%4.4s (%s) [Node %p Owner %X] added to %4.4s (%s) [Node %p]\n", 277 acpi_ut_get_node_name(node), 278 acpi_ut_get_type_name(node->type), node, owner_id, 279 acpi_ut_get_node_name(parent_node), 280 acpi_ut_get_type_name(parent_node->type), 281 parent_node)); 282 283 return_VOID; 284 } 285 286 /******************************************************************************* 287 * 288 * FUNCTION: acpi_ns_delete_children 289 * 290 * PARAMETERS: parent_node - Delete this objects children 291 * 292 * RETURN: None. 293 * 294 * DESCRIPTION: Delete all children of the parent object. In other words, 295 * deletes a "scope". 296 * 297 ******************************************************************************/ 298 299 void acpi_ns_delete_children(struct acpi_namespace_node *parent_node) 300 { 301 struct acpi_namespace_node *next_node; 302 struct acpi_namespace_node *node_to_delete; 303 304 ACPI_FUNCTION_TRACE_PTR(ns_delete_children, parent_node); 305 306 if (!parent_node) { 307 return_VOID; 308 } 309 310 /* Deallocate all children at this level */ 311 312 next_node = parent_node->child; 313 while (next_node) { 314 315 /* Grandchildren should have all been deleted already */ 316 317 if (next_node->child) { 318 ACPI_ERROR((AE_INFO, "Found a grandchild! P=%p C=%p", 319 parent_node, next_node)); 320 } 321 322 /* 323 * Delete this child node and move on to the next child in the list. 324 * No need to unlink the node since we are deleting the entire branch. 325 */ 326 node_to_delete = next_node; 327 next_node = next_node->peer; 328 acpi_ns_delete_node(node_to_delete); 329 }; 330 331 /* Clear the parent's child pointer */ 332 333 parent_node->child = NULL; 334 return_VOID; 335 } 336 337 /******************************************************************************* 338 * 339 * FUNCTION: acpi_ns_delete_namespace_subtree 340 * 341 * PARAMETERS: parent_node - Root of the subtree to be deleted 342 * 343 * RETURN: None. 344 * 345 * DESCRIPTION: Delete a subtree of the namespace. This includes all objects 346 * stored within the subtree. 347 * 348 ******************************************************************************/ 349 350 void acpi_ns_delete_namespace_subtree(struct acpi_namespace_node *parent_node) 351 { 352 struct acpi_namespace_node *child_node = NULL; 353 u32 level = 1; 354 acpi_status status; 355 356 ACPI_FUNCTION_TRACE(ns_delete_namespace_subtree); 357 358 if (!parent_node) { 359 return_VOID; 360 } 361 362 /* Lock namespace for possible update */ 363 364 status = acpi_ut_acquire_mutex(ACPI_MTX_NAMESPACE); 365 if (ACPI_FAILURE(status)) { 366 return_VOID; 367 } 368 369 /* 370 * Traverse the tree of objects until we bubble back up 371 * to where we started. 372 */ 373 while (level > 0) { 374 375 /* Get the next node in this scope (NULL if none) */ 376 377 child_node = acpi_ns_get_next_node(parent_node, child_node); 378 if (child_node) { 379 380 /* Found a child node - detach any attached object */ 381 382 acpi_ns_detach_object(child_node); 383 384 /* Check if this node has any children */ 385 386 if (child_node->child) { 387 /* 388 * There is at least one child of this node, 389 * visit the node 390 */ 391 level++; 392 parent_node = child_node; 393 child_node = NULL; 394 } 395 } else { 396 /* 397 * No more children of this parent node. 398 * Move up to the grandparent. 399 */ 400 level--; 401 402 /* 403 * Now delete all of the children of this parent 404 * all at the same time. 405 */ 406 acpi_ns_delete_children(parent_node); 407 408 /* New "last child" is this parent node */ 409 410 child_node = parent_node; 411 412 /* Move up the tree to the grandparent */ 413 414 parent_node = parent_node->parent; 415 } 416 } 417 418 (void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE); 419 return_VOID; 420 } 421 422 /******************************************************************************* 423 * 424 * FUNCTION: acpi_ns_delete_namespace_by_owner 425 * 426 * PARAMETERS: owner_id - All nodes with this owner will be deleted 427 * 428 * RETURN: Status 429 * 430 * DESCRIPTION: Delete entries within the namespace that are owned by a 431 * specific ID. Used to delete entire ACPI tables. All 432 * reference counts are updated. 433 * 434 * MUTEX: Locks namespace during deletion walk. 435 * 436 ******************************************************************************/ 437 438 void acpi_ns_delete_namespace_by_owner(acpi_owner_id owner_id) 439 { 440 struct acpi_namespace_node *child_node; 441 struct acpi_namespace_node *deletion_node; 442 struct acpi_namespace_node *parent_node; 443 u32 level; 444 acpi_status status; 445 446 ACPI_FUNCTION_TRACE_U32(ns_delete_namespace_by_owner, owner_id); 447 448 if (owner_id == 0) { 449 return_VOID; 450 } 451 452 /* Lock namespace for possible update */ 453 454 status = acpi_ut_acquire_mutex(ACPI_MTX_NAMESPACE); 455 if (ACPI_FAILURE(status)) { 456 return_VOID; 457 } 458 459 deletion_node = NULL; 460 parent_node = acpi_gbl_root_node; 461 child_node = NULL; 462 level = 1; 463 464 /* 465 * Traverse the tree of nodes until we bubble back up 466 * to where we started. 467 */ 468 while (level > 0) { 469 /* 470 * Get the next child of this parent node. When child_node is NULL, 471 * the first child of the parent is returned 472 */ 473 child_node = acpi_ns_get_next_node(parent_node, child_node); 474 475 if (deletion_node) { 476 acpi_ns_delete_children(deletion_node); 477 acpi_ns_remove_node(deletion_node); 478 deletion_node = NULL; 479 } 480 481 if (child_node) { 482 if (child_node->owner_id == owner_id) { 483 484 /* Found a matching child node - detach any attached object */ 485 486 acpi_ns_detach_object(child_node); 487 } 488 489 /* Check if this node has any children */ 490 491 if (child_node->child) { 492 /* 493 * There is at least one child of this node, 494 * visit the node 495 */ 496 level++; 497 parent_node = child_node; 498 child_node = NULL; 499 } else if (child_node->owner_id == owner_id) { 500 deletion_node = child_node; 501 } 502 } else { 503 /* 504 * No more children of this parent node. 505 * Move up to the grandparent. 506 */ 507 level--; 508 if (level != 0) { 509 if (parent_node->owner_id == owner_id) { 510 deletion_node = parent_node; 511 } 512 } 513 514 /* New "last child" is this parent node */ 515 516 child_node = parent_node; 517 518 /* Move up the tree to the grandparent */ 519 520 parent_node = parent_node->parent; 521 } 522 } 523 524 (void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE); 525 return_VOID; 526 } 527