| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| #include <acpi/acpi.h> |
| #include "accommon.h" |
| #include "acnamesp.h" |
| #include "amlcode.h" |
| |
| #define _COMPONENT ACPI_NAMESPACE |
| ACPI_MODULE_NAME("nsutils") |
| |
| |
| #ifdef ACPI_OBSOLETE_FUNCTIONS |
| acpi_name acpi_ns_find_parent_name(struct acpi_namespace_node *node_to_search); |
| #endif |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| void |
| acpi_ns_print_node_pathname(struct acpi_namespace_node *node, |
| <------><------><------> const char *message) |
| { |
| <------>struct acpi_buffer buffer; |
| <------>acpi_status status; |
| |
| <------>if (!node) { |
| <------><------>acpi_os_printf("[NULL NAME]"); |
| <------><------>return; |
| <------>} |
| |
| <------> |
| |
| <------>buffer.length = ACPI_ALLOCATE_LOCAL_BUFFER; |
| |
| <------>status = acpi_ns_handle_to_pathname(node, &buffer, TRUE); |
| <------>if (ACPI_SUCCESS(status)) { |
| <------><------>if (message) { |
| <------><------><------>acpi_os_printf("%s ", message); |
| <------><------>} |
| |
| <------><------>acpi_os_printf("%s", (char *)buffer.pointer); |
| <------><------>ACPI_FREE(buffer.pointer); |
| <------>} |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| acpi_object_type acpi_ns_get_type(struct acpi_namespace_node * node) |
| { |
| <------>ACPI_FUNCTION_TRACE(ns_get_type); |
| |
| <------>if (!node) { |
| <------><------>ACPI_WARNING((AE_INFO, "Null Node parameter")); |
| <------><------>return_UINT8(ACPI_TYPE_ANY); |
| <------>} |
| |
| <------>return_UINT8(node->type); |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| u32 acpi_ns_local(acpi_object_type type) |
| { |
| <------>ACPI_FUNCTION_TRACE(ns_local); |
| |
| <------>if (!acpi_ut_valid_object_type(type)) { |
| |
| <------><------> |
| |
| <------><------>ACPI_WARNING((AE_INFO, "Invalid Object Type 0x%X", type)); |
| <------><------>return_UINT32(ACPI_NS_NORMAL); |
| <------>} |
| |
| <------>return_UINT32(acpi_gbl_ns_properties[type] & ACPI_NS_LOCAL); |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| void acpi_ns_get_internal_name_length(struct acpi_namestring_info *info) |
| { |
| <------>const char *next_external_char; |
| <------>u32 i; |
| |
| <------>ACPI_FUNCTION_ENTRY(); |
| |
| <------>next_external_char = info->external_name; |
| <------>info->num_carats = 0; |
| <------>info->num_segments = 0; |
| <------>info->fully_qualified = FALSE; |
| |
| <------> |
| <------> * For the internal name, the required length is 4 bytes per segment, |
| <------> * plus 1 each for root_prefix, multi_name_prefix_op, segment count, |
| <------> * trailing null (which is not really needed, but no there's harm in |
| <------> * putting it there) |
| <------> * |
| <------> * strlen() + 1 covers the first name_seg, which has no path separator |
| <------> */ |
| <------>if (ACPI_IS_ROOT_PREFIX(*next_external_char)) { |
| <------><------>info->fully_qualified = TRUE; |
| <------><------>next_external_char++; |
| |
| <------><------> |
| |
| <------><------>while (ACPI_IS_ROOT_PREFIX(*next_external_char)) { |
| <------><------><------>next_external_char++; |
| <------><------>} |
| <------>} else { |
| <------><------> |
| |
| <------><------>while (ACPI_IS_PARENT_PREFIX(*next_external_char)) { |
| <------><------><------>info->num_carats++; |
| <------><------><------>next_external_char++; |
| <------><------>} |
| <------>} |
| |
| <------> |
| <------> * Determine the number of ACPI name "segments" by counting the number of |
| <------> * path separators within the string. Start with one segment since the |
| <------> * segment count is [(# separators) + 1], and zero separators is ok. |
| <------> */ |
| <------>if (*next_external_char) { |
| <------><------>info->num_segments = 1; |
| <------><------>for (i = 0; next_external_char[i]; i++) { |
| <------><------><------>if (ACPI_IS_PATH_SEPARATOR(next_external_char[i])) { |
| <------><------><------><------>info->num_segments++; |
| <------><------><------>} |
| <------><------>} |
| <------>} |
| |
| <------>info->length = (ACPI_NAMESEG_SIZE * info->num_segments) + |
| <------> 4 + info->num_carats; |
| |
| <------>info->next_external_char = next_external_char; |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| acpi_status acpi_ns_build_internal_name(struct acpi_namestring_info *info) |
| { |
| <------>u32 num_segments = info->num_segments; |
| <------>char *internal_name = info->internal_name; |
| <------>const char *external_name = info->next_external_char; |
| <------>char *result = NULL; |
| <------>u32 i; |
| |
| <------>ACPI_FUNCTION_TRACE(ns_build_internal_name); |
| |
| <------> |
| |
| <------>if (info->fully_qualified) { |
| <------><------>internal_name[0] = AML_ROOT_PREFIX; |
| |
| <------><------>if (num_segments <= 1) { |
| <------><------><------>result = &internal_name[1]; |
| <------><------>} else if (num_segments == 2) { |
| <------><------><------>internal_name[1] = AML_DUAL_NAME_PREFIX; |
| <------><------><------>result = &internal_name[2]; |
| <------><------>} else { |
| <------><------><------>internal_name[1] = AML_MULTI_NAME_PREFIX; |
| <------><------><------>internal_name[2] = (char)num_segments; |
| <------><------><------>result = &internal_name[3]; |
| <------><------>} |
| <------>} else { |
| <------><------> |
| <------><------> * Not fully qualified. |
| <------><------> * Handle Carats first, then append the name segments |
| <------><------> */ |
| <------><------>i = 0; |
| <------><------>if (info->num_carats) { |
| <------><------><------>for (i = 0; i < info->num_carats; i++) { |
| <------><------><------><------>internal_name[i] = AML_PARENT_PREFIX; |
| <------><------><------>} |
| <------><------>} |
| |
| <------><------>if (num_segments <= 1) { |
| <------><------><------>result = &internal_name[i]; |
| <------><------>} else if (num_segments == 2) { |
| <------><------><------>internal_name[i] = AML_DUAL_NAME_PREFIX; |
| <------><------><------>result = &internal_name[(acpi_size)i + 1]; |
| <------><------>} else { |
| <------><------><------>internal_name[i] = AML_MULTI_NAME_PREFIX; |
| <------><------><------>internal_name[(acpi_size)i + 1] = (char)num_segments; |
| <------><------><------>result = &internal_name[(acpi_size)i + 2]; |
| <------><------>} |
| <------>} |
| |
| <------> |
| |
| <------>for (; num_segments; num_segments--) { |
| <------><------>for (i = 0; i < ACPI_NAMESEG_SIZE; i++) { |
| <------><------><------>if (ACPI_IS_PATH_SEPARATOR(*external_name) || |
| <------><------><------> (*external_name == 0)) { |
| |
| <------><------><------><------> |
| |
| <------><------><------><------>result[i] = '_'; |
| <------><------><------>} else { |
| <------><------><------><------> |
| |
| <------><------><------><------>result[i] = (char)toupper((int)*external_name); |
| <------><------><------><------>external_name++; |
| <------><------><------>} |
| <------><------>} |
| |
| <------><------> |
| |
| <------><------>if (!ACPI_IS_PATH_SEPARATOR(*external_name) && |
| <------><------> (*external_name != 0)) { |
| <------><------><------>return_ACPI_STATUS(AE_BAD_PATHNAME); |
| <------><------>} |
| |
| <------><------> |
| |
| <------><------>external_name++; |
| <------><------>result += ACPI_NAMESEG_SIZE; |
| <------>} |
| |
| <------> |
| |
| <------>*result = 0; |
| |
| <------>if (info->fully_qualified) { |
| <------><------>ACPI_DEBUG_PRINT((ACPI_DB_EXEC, |
| <------><------><------><------> "Returning [%p] (abs) \"\\%s\"\n", |
| <------><------><------><------> internal_name, internal_name)); |
| <------>} else { |
| <------><------>ACPI_DEBUG_PRINT((ACPI_DB_EXEC, "Returning [%p] (rel) \"%s\"\n", |
| <------><------><------><------> internal_name, internal_name)); |
| <------>} |
| |
| <------>return_ACPI_STATUS(AE_OK); |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| acpi_status |
| acpi_ns_internalize_name(const char *external_name, char **converted_name) |
| { |
| <------>char *internal_name; |
| <------>struct acpi_namestring_info info; |
| <------>acpi_status status; |
| |
| <------>ACPI_FUNCTION_TRACE(ns_internalize_name); |
| |
| <------>if ((!external_name) || (*external_name == 0) || (!converted_name)) { |
| <------><------>return_ACPI_STATUS(AE_BAD_PARAMETER); |
| <------>} |
| |
| <------> |
| |
| <------>info.external_name = external_name; |
| <------>acpi_ns_get_internal_name_length(&info); |
| |
| <------> |
| |
| <------>internal_name = ACPI_ALLOCATE_ZEROED(info.length); |
| <------>if (!internal_name) { |
| <------><------>return_ACPI_STATUS(AE_NO_MEMORY); |
| <------>} |
| |
| <------> |
| |
| <------>info.internal_name = internal_name; |
| <------>status = acpi_ns_build_internal_name(&info); |
| <------>if (ACPI_FAILURE(status)) { |
| <------><------>ACPI_FREE(internal_name); |
| <------><------>return_ACPI_STATUS(status); |
| <------>} |
| |
| <------>*converted_name = internal_name; |
| <------>return_ACPI_STATUS(AE_OK); |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| acpi_status |
| acpi_ns_externalize_name(u32 internal_name_length, |
| <------><------><------> const char *internal_name, |
| <------><------><------> u32 * converted_name_length, char **converted_name) |
| { |
| <------>u32 names_index = 0; |
| <------>u32 num_segments = 0; |
| <------>u32 required_length; |
| <------>u32 prefix_length = 0; |
| <------>u32 i = 0; |
| <------>u32 j = 0; |
| |
| <------>ACPI_FUNCTION_TRACE(ns_externalize_name); |
| |
| <------>if (!internal_name_length || !internal_name || !converted_name) { |
| <------><------>return_ACPI_STATUS(AE_BAD_PARAMETER); |
| <------>} |
| |
| <------> |
| |
| <------>switch (internal_name[0]) { |
| <------>case AML_ROOT_PREFIX: |
| |
| <------><------>prefix_length = 1; |
| <------><------>break; |
| |
| <------>case AML_PARENT_PREFIX: |
| |
| <------><------>for (i = 0; i < internal_name_length; i++) { |
| <------><------><------>if (ACPI_IS_PARENT_PREFIX(internal_name[i])) { |
| <------><------><------><------>prefix_length = i + 1; |
| <------><------><------>} else { |
| <------><------><------><------>break; |
| <------><------><------>} |
| <------><------>} |
| |
| <------><------>if (i == internal_name_length) { |
| <------><------><------>prefix_length = i; |
| <------><------>} |
| |
| <------><------>break; |
| |
| <------>default: |
| |
| <------><------>break; |
| <------>} |
| |
| <------> |
| <------> * Check for object names. Note that there could be 0-255 of these |
| <------> * 4-byte elements. |
| <------> */ |
| <------>if (prefix_length < internal_name_length) { |
| <------><------>switch (internal_name[prefix_length]) { |
| <------><------>case AML_MULTI_NAME_PREFIX: |
| |
| <------><------><------> |
| |
| <------><------><------>names_index = prefix_length + 2; |
| <------><------><------>num_segments = (u8) |
| <------><------><------> internal_name[(acpi_size)prefix_length + 1]; |
| <------><------><------>break; |
| |
| <------><------>case AML_DUAL_NAME_PREFIX: |
| |
| <------><------><------> |
| |
| <------><------><------>names_index = prefix_length + 1; |
| <------><------><------>num_segments = 2; |
| <------><------><------>break; |
| |
| <------><------>case 0: |
| |
| <------><------><------> |
| |
| <------><------><------>names_index = 0; |
| <------><------><------>num_segments = 0; |
| <------><------><------>break; |
| |
| <------><------>default: |
| |
| <------><------><------> |
| |
| <------><------><------>names_index = prefix_length; |
| <------><------><------>num_segments = 1; |
| <------><------><------>break; |
| <------><------>} |
| <------>} |
| |
| <------> |
| <------> * Calculate the length of converted_name, which equals the length |
| <------> * of the prefix, length of all object names, length of any required |
| <------> * punctuation ('.') between object names, plus the NULL terminator. |
| <------> */ |
| <------>required_length = prefix_length + (4 * num_segments) + |
| <------> ((num_segments > 0) ? (num_segments - 1) : 0) + 1; |
| |
| <------> |
| <------> * Check to see if we're still in bounds. If not, there's a problem |
| <------> * with internal_name (invalid format). |
| <------> */ |
| <------>if (required_length > internal_name_length) { |
| <------><------>ACPI_ERROR((AE_INFO, "Invalid internal name")); |
| <------><------>return_ACPI_STATUS(AE_BAD_PATHNAME); |
| <------>} |
| |
| <------> |
| |
| <------>*converted_name = ACPI_ALLOCATE_ZEROED(required_length); |
| <------>if (!(*converted_name)) { |
| <------><------>return_ACPI_STATUS(AE_NO_MEMORY); |
| <------>} |
| |
| <------>j = 0; |
| |
| <------>for (i = 0; i < prefix_length; i++) { |
| <------><------>(*converted_name)[j++] = internal_name[i]; |
| <------>} |
| |
| <------>if (num_segments > 0) { |
| <------><------>for (i = 0; i < num_segments; i++) { |
| <------><------><------>if (i > 0) { |
| <------><------><------><------>(*converted_name)[j++] = '.'; |
| <------><------><------>} |
| |
| <------><------><------> |
| |
| <------><------><------>ACPI_COPY_NAMESEG(&(*converted_name)[j], |
| <------><------><------><------><------> &internal_name[names_index]); |
| <------><------><------>acpi_ut_repair_name(&(*converted_name)[j]); |
| |
| <------><------><------>j += ACPI_NAMESEG_SIZE; |
| <------><------><------>names_index += ACPI_NAMESEG_SIZE; |
| <------><------>} |
| <------>} |
| |
| <------>if (converted_name_length) { |
| <------><------>*converted_name_length = (u32) required_length; |
| <------>} |
| |
| <------>return_ACPI_STATUS(AE_OK); |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| struct acpi_namespace_node *acpi_ns_validate_handle(acpi_handle handle) |
| { |
| |
| <------>ACPI_FUNCTION_ENTRY(); |
| |
| <------> |
| |
| <------>if ((!handle) || (handle == ACPI_ROOT_OBJECT)) { |
| <------><------>return (acpi_gbl_root_node); |
| <------>} |
| |
| <------> |
| |
| <------>if (ACPI_GET_DESCRIPTOR_TYPE(handle) != ACPI_DESC_TYPE_NAMED) { |
| <------><------>return (NULL); |
| <------>} |
| |
| <------>return (ACPI_CAST_PTR(struct acpi_namespace_node, handle)); |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| void acpi_ns_terminate(void) |
| { |
| <------>acpi_status status; |
| |
| <------>ACPI_FUNCTION_TRACE(ns_terminate); |
| |
| <------> |
| <------> * Free the entire namespace -- all nodes and all objects |
| <------> * attached to the nodes |
| <------> */ |
| <------>acpi_ns_delete_namespace_subtree(acpi_gbl_root_node); |
| |
| <------> |
| |
| <------>status = acpi_ut_acquire_mutex(ACPI_MTX_NAMESPACE); |
| <------>if (ACPI_FAILURE(status)) { |
| <------><------>return_VOID; |
| <------>} |
| |
| <------>acpi_ns_delete_node(acpi_gbl_root_node); |
| <------>(void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE); |
| |
| <------>ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Namespace freed\n")); |
| <------>return_VOID; |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| u32 acpi_ns_opens_scope(acpi_object_type type) |
| { |
| <------>ACPI_FUNCTION_ENTRY(); |
| |
| <------>if (type > ACPI_TYPE_LOCAL_MAX) { |
| |
| <------><------> |
| |
| <------><------>ACPI_WARNING((AE_INFO, "Invalid Object Type 0x%X", type)); |
| <------><------>return (ACPI_NS_NORMAL); |
| <------>} |
| |
| <------>return (((u32)acpi_gbl_ns_properties[type]) & ACPI_NS_NEWSCOPE); |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| acpi_status |
| acpi_ns_get_node_unlocked(struct acpi_namespace_node *prefix_node, |
| <------><------><------> const char *pathname, |
| <------><------><------> u32 flags, struct acpi_namespace_node **return_node) |
| { |
| <------>union acpi_generic_state scope_info; |
| <------>acpi_status status; |
| <------>char *internal_path; |
| |
| <------>ACPI_FUNCTION_TRACE_PTR(ns_get_node_unlocked, |
| <------><------><------><------>ACPI_CAST_PTR(char, pathname)); |
| |
| <------> |
| |
| <------>if (!pathname) { |
| <------><------>*return_node = prefix_node; |
| <------><------>if (!prefix_node) { |
| <------><------><------>*return_node = acpi_gbl_root_node; |
| <------><------>} |
| |
| <------><------>return_ACPI_STATUS(AE_OK); |
| <------>} |
| |
| <------> |
| |
| <------>if (ACPI_IS_ROOT_PREFIX(pathname[0]) && (!pathname[1])) { |
| <------><------>*return_node = acpi_gbl_root_node; |
| <------><------>return_ACPI_STATUS(AE_OK); |
| <------>} |
| |
| <------> |
| |
| <------>status = acpi_ns_internalize_name(pathname, &internal_path); |
| <------>if (ACPI_FAILURE(status)) { |
| <------><------>return_ACPI_STATUS(status); |
| <------>} |
| |
| <------> |
| |
| <------>scope_info.scope.node = prefix_node; |
| |
| <------> |
| |
| <------>status = acpi_ns_lookup(&scope_info, internal_path, ACPI_TYPE_ANY, |
| <------><------><------><------>ACPI_IMODE_EXECUTE, |
| <------><------><------><------>(flags | ACPI_NS_DONT_OPEN_SCOPE), NULL, |
| <------><------><------><------>return_node); |
| <------>if (ACPI_FAILURE(status)) { |
| <------><------>ACPI_DEBUG_PRINT((ACPI_DB_EXEC, "%s, %s\n", |
| <------><------><------><------> pathname, acpi_format_exception(status))); |
| <------>} |
| |
| <------>ACPI_FREE(internal_path); |
| <------>return_ACPI_STATUS(status); |
| } |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| acpi_status |
| acpi_ns_get_node(struct acpi_namespace_node *prefix_node, |
| <------><------> const char *pathname, |
| <------><------> u32 flags, struct acpi_namespace_node **return_node) |
| { |
| <------>acpi_status status; |
| |
| <------>ACPI_FUNCTION_TRACE_PTR(ns_get_node, ACPI_CAST_PTR(char, pathname)); |
| |
| <------>status = acpi_ut_acquire_mutex(ACPI_MTX_NAMESPACE); |
| <------>if (ACPI_FAILURE(status)) { |
| <------><------>return_ACPI_STATUS(status); |
| <------>} |
| |
| <------>status = acpi_ns_get_node_unlocked(prefix_node, pathname, |
| <------><------><------><------><------> flags, return_node); |
| |
| <------>(void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE); |
| <------>return_ACPI_STATUS(status); |
| } |
| |