2005-04-17 06:20:36 +08:00
|
|
|
/*******************************************************************************
|
|
|
|
*
|
|
|
|
* Module Name: nsxfeval - Public interfaces to the ACPI subsystem
|
|
|
|
* ACPI Object evaluation interfaces
|
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
|
|
|
|
/*
|
2017-02-08 11:00:08 +08:00
|
|
|
* Copyright (C) 2000 - 2017, Intel Corp.
|
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.
|
|
|
|
*/
|
|
|
|
|
2013-10-29 09:29:51 +08:00
|
|
|
#define EXPORT_ACPI_INTERFACES
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
#include <acpi/acpi.h>
|
2009-01-09 13:30:03 +08:00
|
|
|
#include "accommon.h"
|
|
|
|
#include "acnamesp.h"
|
|
|
|
#include "acinterp.h"
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#define _COMPONENT ACPI_NAMESPACE
|
2005-08-05 12:44:28 +08:00
|
|
|
ACPI_MODULE_NAME("nsxfeval")
|
2008-08-04 13:22:10 +08:00
|
|
|
|
|
|
|
/* Local prototypes */
|
|
|
|
static void acpi_ns_resolve_references(struct acpi_evaluate_info *info);
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*******************************************************************************
|
|
|
|
*
|
|
|
|
* FUNCTION: acpi_evaluate_object_typed
|
|
|
|
*
|
2012-07-12 09:40:10 +08:00
|
|
|
* PARAMETERS: handle - Object handle (optional)
|
|
|
|
* pathname - Object pathname (optional)
|
2018-01-04 07:06:22 +08:00
|
|
|
* external_params - List of parameters to pass to a method,
|
2012-10-31 10:26:55 +08:00
|
|
|
* terminated by NULL. May be NULL
|
2005-04-17 06:20:36 +08:00
|
|
|
* if no parameters are being passed.
|
2018-01-04 07:06:22 +08:00
|
|
|
* return_buffer - Where to put the object's return value (if
|
2012-10-31 10:26:55 +08:00
|
|
|
* any). If NULL, no value is returned.
|
2005-04-17 06:20:36 +08:00
|
|
|
* return_type - Expected type of return object
|
|
|
|
*
|
|
|
|
* RETURN: Status
|
|
|
|
*
|
|
|
|
* DESCRIPTION: Find and evaluate the given object, passing the given
|
2012-10-31 10:26:55 +08:00
|
|
|
* parameters if necessary. One of "Handle" or "Pathname" must
|
2005-04-17 06:20:36 +08:00
|
|
|
* be valid (non-null)
|
|
|
|
*
|
|
|
|
******************************************************************************/
|
2008-08-04 13:22:10 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
acpi_status
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_evaluate_object_typed(acpi_handle handle,
|
|
|
|
acpi_string pathname,
|
2007-05-10 11:34:35 +08:00
|
|
|
struct acpi_object_list *external_params,
|
|
|
|
struct acpi_buffer *return_buffer,
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_object_type return_type)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_status status;
|
2014-01-08 13:43:23 +08:00
|
|
|
u8 free_buffer_on_error = FALSE;
|
2017-06-05 16:41:23 +08:00
|
|
|
acpi_handle target_handle;
|
|
|
|
char *full_pathname;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
ACPI: ACPICA 20060421
Removed a device initialization optimization introduced in
20051216 where the _STA method was not run unless an _INI
was also present for the same device. This optimization
could cause problems because it could allow _INI methods
to be run within a not-present device subtree (If a
not-present device had no _INI, _STA would not be run,
the not-present status would not be discovered, and the
children of the device would be incorrectly traversed.)
Implemented a new _STA optimization where namespace
subtrees that do not contain _INI are identified and
ignored during device initialization. Selectively running
_STA can significantly improve boot time on large machines
(with assistance from Len Brown.)
Implemented support for the device initialization case
where the returned _STA flags indicate a device not-present
but functioning. In this case, _INI is not run, but the
device children are examined for presence, as per the
ACPI specification.
Implemented an additional change to the IndexField support
in order to conform to MS behavior. The value written to
the Index Register is not simply a byte offset, it is a
byte offset in units of the access width of the parent
Index Field. (Fiodor Suietov)
Defined and deployed a new OSL interface,
acpi_os_validate_address(). This interface is called during
the creation of all AML operation regions, and allows
the host OS to exert control over what addresses it will
allow the AML code to access. Operation Regions whose
addresses are disallowed will cause a runtime exception
when they are actually accessed (will not affect or abort
table loading.)
Defined and deployed a new OSL interface,
acpi_os_validate_interface(). This interface allows the host OS
to match the various "optional" interface/behavior strings
for the _OSI predefined control method as appropriate
(with assistance from Bjorn Helgaas.)
Restructured and corrected various problems in the
exception handling code paths within DsCallControlMethod
and DsTerminateControlMethod in dsmethod (with assistance
from Takayoshi Kochi.)
Modified the Linux source converter to ignore quoted string
literals while converting identifiers from mixed to lower
case. This will correct problems with the disassembler
and other areas where such strings must not be modified.
The ACPI_FUNCTION_* macros no longer require quotes around
the function name. This allows the Linux source converter
to convert the names, now that the converter ignores
quoted strings.
Signed-off-by: Bob Moore <robert.moore@intel.com>
Signed-off-by: Len Brown <len.brown@intel.com>
2006-04-22 05:15:00 +08:00
|
|
|
ACPI_FUNCTION_TRACE(acpi_evaluate_object_typed);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* Return buffer must be valid */
|
|
|
|
|
|
|
|
if (!return_buffer) {
|
2005-08-05 12:44:28 +08:00
|
|
|
return_ACPI_STATUS(AE_BAD_PARAMETER);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (return_buffer->length == ACPI_ALLOCATE_BUFFER) {
|
2014-01-08 13:43:23 +08:00
|
|
|
free_buffer_on_error = TRUE;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2018-01-04 07:06:22 +08:00
|
|
|
/* Get a handle here, in order to build an error message if needed */
|
|
|
|
|
|
|
|
target_handle = handle;
|
2017-08-23 04:44:13 +08:00
|
|
|
if (pathname) {
|
|
|
|
status = acpi_get_handle(handle, pathname, &target_handle);
|
|
|
|
if (ACPI_FAILURE(status)) {
|
|
|
|
return_ACPI_STATUS(status);
|
|
|
|
}
|
2017-06-05 16:41:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
full_pathname = acpi_ns_get_external_pathname(target_handle);
|
|
|
|
if (!full_pathname) {
|
|
|
|
return_ACPI_STATUS(AE_NO_MEMORY);
|
|
|
|
}
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/* Evaluate the object */
|
|
|
|
|
2017-06-05 16:41:23 +08:00
|
|
|
status = acpi_evaluate_object(target_handle, NULL, external_params,
|
|
|
|
return_buffer);
|
2005-08-05 12:44:28 +08:00
|
|
|
if (ACPI_FAILURE(status)) {
|
2017-06-05 16:41:23 +08:00
|
|
|
goto exit;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2017-06-05 16:41:23 +08:00
|
|
|
/* Type ANY means "don't care about return value type" */
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
if (return_type == ACPI_TYPE_ANY) {
|
2017-06-05 16:41:23 +08:00
|
|
|
goto exit;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (return_buffer->length == 0) {
|
2006-10-02 12:00:00 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/* Error because caller specifically asked for a return value */
|
|
|
|
|
2017-06-05 16:41:23 +08:00
|
|
|
ACPI_ERROR((AE_INFO, "%s did not return any object",
|
|
|
|
full_pathname));
|
|
|
|
status = AE_NULL_OBJECT;
|
|
|
|
goto exit;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Examine the object type returned from evaluate_object */
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
if (((union acpi_object *)return_buffer->pointer)->type == return_type) {
|
2017-06-05 16:41:23 +08:00
|
|
|
goto exit;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Return object type does not match requested type */
|
|
|
|
|
2006-01-28 05:43:00 +08:00
|
|
|
ACPI_ERROR((AE_INFO,
|
2017-06-05 16:41:23 +08:00
|
|
|
"Incorrect return type from %s - received [%s], requested [%s]",
|
|
|
|
full_pathname,
|
2006-01-28 05:43:00 +08:00
|
|
|
acpi_ut_get_type_name(((union acpi_object *)return_buffer->
|
|
|
|
pointer)->type),
|
|
|
|
acpi_ut_get_type_name(return_type)));
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2014-01-08 13:43:23 +08:00
|
|
|
if (free_buffer_on_error) {
|
|
|
|
/*
|
|
|
|
* Free a buffer created via ACPI_ALLOCATE_BUFFER.
|
|
|
|
* Note: We use acpi_os_free here because acpi_os_allocate was used
|
|
|
|
* to allocate the buffer. This purposefully bypasses the
|
|
|
|
* (optionally enabled) allocation tracking mechanism since we
|
|
|
|
* only want to track internal allocations.
|
|
|
|
*/
|
|
|
|
acpi_os_free(return_buffer->pointer);
|
2005-04-17 06:20:36 +08:00
|
|
|
return_buffer->pointer = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return_buffer->length = 0;
|
2017-06-05 16:41:23 +08:00
|
|
|
status = AE_TYPE;
|
|
|
|
|
|
|
|
exit:
|
|
|
|
ACPI_FREE(full_pathname);
|
|
|
|
return_ACPI_STATUS(status);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2006-10-03 12:00:00 +08:00
|
|
|
|
|
|
|
ACPI_EXPORT_SYMBOL(acpi_evaluate_object_typed)
|
2009-04-07 21:32:59 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*******************************************************************************
|
|
|
|
*
|
|
|
|
* FUNCTION: acpi_evaluate_object
|
|
|
|
*
|
2012-07-12 09:40:10 +08:00
|
|
|
* PARAMETERS: handle - Object handle (optional)
|
|
|
|
* pathname - Object pathname (optional)
|
2005-04-17 06:20:36 +08:00
|
|
|
* external_params - List of parameters to pass to method,
|
2012-10-31 10:26:55 +08:00
|
|
|
* terminated by NULL. May be NULL
|
2005-04-17 06:20:36 +08:00
|
|
|
* if no parameters are being passed.
|
|
|
|
* return_buffer - Where to put method's return value (if
|
2012-10-31 10:26:55 +08:00
|
|
|
* any). If NULL, no value is returned.
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
|
|
|
* RETURN: Status
|
|
|
|
*
|
|
|
|
* DESCRIPTION: Find and evaluate the given object, passing the given
|
2012-10-31 10:26:55 +08:00
|
|
|
* parameters if necessary. One of "Handle" or "Pathname" must
|
2005-04-17 06:20:36 +08:00
|
|
|
* be valid (non-null)
|
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
acpi_status
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_evaluate_object(acpi_handle handle,
|
|
|
|
acpi_string pathname,
|
|
|
|
struct acpi_object_list *external_params,
|
|
|
|
struct acpi_buffer *return_buffer)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_status status;
|
2006-05-27 04:36:00 +08:00
|
|
|
struct acpi_evaluate_info *info;
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_size buffer_space_needed;
|
|
|
|
u32 i;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
ACPI: ACPICA 20060421
Removed a device initialization optimization introduced in
20051216 where the _STA method was not run unless an _INI
was also present for the same device. This optimization
could cause problems because it could allow _INI methods
to be run within a not-present device subtree (If a
not-present device had no _INI, _STA would not be run,
the not-present status would not be discovered, and the
children of the device would be incorrectly traversed.)
Implemented a new _STA optimization where namespace
subtrees that do not contain _INI are identified and
ignored during device initialization. Selectively running
_STA can significantly improve boot time on large machines
(with assistance from Len Brown.)
Implemented support for the device initialization case
where the returned _STA flags indicate a device not-present
but functioning. In this case, _INI is not run, but the
device children are examined for presence, as per the
ACPI specification.
Implemented an additional change to the IndexField support
in order to conform to MS behavior. The value written to
the Index Register is not simply a byte offset, it is a
byte offset in units of the access width of the parent
Index Field. (Fiodor Suietov)
Defined and deployed a new OSL interface,
acpi_os_validate_address(). This interface is called during
the creation of all AML operation regions, and allows
the host OS to exert control over what addresses it will
allow the AML code to access. Operation Regions whose
addresses are disallowed will cause a runtime exception
when they are actually accessed (will not affect or abort
table loading.)
Defined and deployed a new OSL interface,
acpi_os_validate_interface(). This interface allows the host OS
to match the various "optional" interface/behavior strings
for the _OSI predefined control method as appropriate
(with assistance from Bjorn Helgaas.)
Restructured and corrected various problems in the
exception handling code paths within DsCallControlMethod
and DsTerminateControlMethod in dsmethod (with assistance
from Takayoshi Kochi.)
Modified the Linux source converter to ignore quoted string
literals while converting identifiers from mixed to lower
case. This will correct problems with the disassembler
and other areas where such strings must not be modified.
The ACPI_FUNCTION_* macros no longer require quotes around
the function name. This allows the Linux source converter
to convert the names, now that the converter ignores
quoted strings.
Signed-off-by: Bob Moore <robert.moore@intel.com>
Signed-off-by: Len Brown <len.brown@intel.com>
2006-04-22 05:15:00 +08:00
|
|
|
ACPI_FUNCTION_TRACE(acpi_evaluate_object);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2006-05-27 04:36:00 +08:00
|
|
|
/* Allocate and initialize the evaluation information block */
|
|
|
|
|
|
|
|
info = ACPI_ALLOCATE_ZEROED(sizeof(struct acpi_evaluate_info));
|
|
|
|
if (!info) {
|
|
|
|
return_ACPI_STATUS(AE_NO_MEMORY);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Convert and validate the device handle */
|
|
|
|
|
2009-12-11 14:57:00 +08:00
|
|
|
info->prefix_node = acpi_ns_validate_handle(handle);
|
2006-05-27 04:36:00 +08:00
|
|
|
if (!info->prefix_node) {
|
|
|
|
status = AE_BAD_PARAMETER;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/*
|
2013-05-30 10:00:01 +08:00
|
|
|
* Get the actual namespace node for the target object.
|
|
|
|
* Handles these cases:
|
|
|
|
*
|
|
|
|
* 1) Null node, valid pathname from root (absolute path)
|
|
|
|
* 2) Node and valid pathname (path relative to Node)
|
|
|
|
* 3) Node, Null pathname
|
|
|
|
*/
|
|
|
|
if ((pathname) && (ACPI_IS_ROOT_PREFIX(pathname[0]))) {
|
|
|
|
|
|
|
|
/* The path is fully qualified, just evaluate by name */
|
|
|
|
|
|
|
|
info->prefix_node = NULL;
|
|
|
|
} else if (!handle) {
|
|
|
|
/*
|
|
|
|
* A handle is optional iff a fully qualified pathname is specified.
|
|
|
|
* Since we've already handled fully qualified names above, this is
|
|
|
|
* an error.
|
|
|
|
*/
|
|
|
|
if (!pathname) {
|
|
|
|
ACPI_DEBUG_PRINT((ACPI_DB_INFO,
|
|
|
|
"Both Handle and Pathname are NULL"));
|
|
|
|
} else {
|
|
|
|
ACPI_DEBUG_PRINT((ACPI_DB_INFO,
|
|
|
|
"Null Handle with relative pathname [%s]",
|
|
|
|
pathname));
|
|
|
|
}
|
|
|
|
|
|
|
|
status = AE_BAD_PARAMETER;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
info->relative_pathname = pathname;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Convert all external objects passed as arguments to the
|
|
|
|
* internal version(s).
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
|
|
|
if (external_params && external_params->count) {
|
2013-05-30 10:00:01 +08:00
|
|
|
info->param_count = (u16)external_params->count;
|
|
|
|
|
|
|
|
/* Warn on impossible argument count */
|
|
|
|
|
|
|
|
if (info->param_count > ACPI_METHOD_NUM_ARGS) {
|
|
|
|
ACPI_WARN_PREDEFINED((AE_INFO, pathname,
|
|
|
|
ACPI_WARN_ALWAYS,
|
|
|
|
"Excess arguments (%u) - using only %u",
|
|
|
|
info->param_count,
|
|
|
|
ACPI_METHOD_NUM_ARGS));
|
|
|
|
|
|
|
|
info->param_count = ACPI_METHOD_NUM_ARGS;
|
|
|
|
}
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* Allocate a new parameter block for the internal objects
|
|
|
|
* Add 1 to count to allow for null terminated internal list
|
|
|
|
*/
|
2016-05-05 12:57:53 +08:00
|
|
|
info->parameters = ACPI_ALLOCATE_ZEROED(((acpi_size)info->
|
2013-05-30 10:00:01 +08:00
|
|
|
param_count +
|
2006-05-27 04:36:00 +08:00
|
|
|
1) * sizeof(void *));
|
|
|
|
if (!info->parameters) {
|
|
|
|
status = AE_NO_MEMORY;
|
|
|
|
goto cleanup;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2006-05-27 04:36:00 +08:00
|
|
|
/* Convert each external object in the list to an internal object */
|
|
|
|
|
2013-05-30 10:00:01 +08:00
|
|
|
for (i = 0; i < info->param_count; i++) {
|
2005-08-05 12:44:28 +08:00
|
|
|
status =
|
|
|
|
acpi_ut_copy_eobject_to_iobject(&external_params->
|
|
|
|
pointer[i],
|
2006-05-27 04:36:00 +08:00
|
|
|
&info->
|
2005-08-05 12:44:28 +08:00
|
|
|
parameters[i]);
|
|
|
|
if (ACPI_FAILURE(status)) {
|
2006-05-27 04:36:00 +08:00
|
|
|
goto cleanup;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
}
|
2013-05-30 10:00:01 +08:00
|
|
|
|
|
|
|
info->parameters[info->param_count] = NULL;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2016-05-05 12:58:00 +08:00
|
|
|
#ifdef _FUTURE_FEATURE
|
2013-05-30 10:00:01 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
2013-05-30 10:00:01 +08:00
|
|
|
* Begin incoming argument count analysis. Check for too few args
|
|
|
|
* and too many args.
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
2013-05-30 10:00:01 +08:00
|
|
|
switch (acpi_ns_get_type(info->node)) {
|
|
|
|
case ACPI_TYPE_METHOD:
|
|
|
|
|
|
|
|
/* Check incoming argument count against the method definition */
|
|
|
|
|
|
|
|
if (info->obj_desc->method.param_count > info->param_count) {
|
|
|
|
ACPI_ERROR((AE_INFO,
|
|
|
|
"Insufficient arguments (%u) - %u are required",
|
|
|
|
info->param_count,
|
|
|
|
info->obj_desc->method.param_count));
|
|
|
|
|
|
|
|
status = AE_MISSING_ARGUMENTS;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (info->obj_desc->method.param_count < info->param_count) {
|
|
|
|
ACPI_WARNING((AE_INFO,
|
|
|
|
"Excess arguments (%u) - only %u are required",
|
|
|
|
info->param_count,
|
|
|
|
info->obj_desc->method.param_count));
|
|
|
|
|
|
|
|
/* Just pass the required number of arguments */
|
|
|
|
|
|
|
|
info->param_count = info->obj_desc->method.param_count;
|
|
|
|
}
|
2006-10-02 12:00:00 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
2013-05-30 10:00:01 +08:00
|
|
|
* Any incoming external objects to be passed as arguments to the
|
|
|
|
* method must be converted to internal objects
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
2013-05-30 10:00:01 +08:00
|
|
|
if (info->param_count) {
|
|
|
|
/*
|
|
|
|
* Allocate a new parameter block for the internal objects
|
|
|
|
* Add 1 to count to allow for null terminated internal list
|
|
|
|
*/
|
|
|
|
info->parameters = ACPI_ALLOCATE_ZEROED(((acpi_size)
|
|
|
|
info->
|
|
|
|
param_count +
|
|
|
|
1) *
|
|
|
|
sizeof(void *));
|
|
|
|
if (!info->parameters) {
|
|
|
|
status = AE_NO_MEMORY;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Convert each external object in the list to an internal object */
|
|
|
|
|
|
|
|
for (i = 0; i < info->param_count; i++) {
|
|
|
|
status =
|
|
|
|
acpi_ut_copy_eobject_to_iobject
|
|
|
|
(&external_params->pointer[i],
|
|
|
|
&info->parameters[i]);
|
|
|
|
if (ACPI_FAILURE(status)) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
info->parameters[info->param_count] = NULL;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
2013-05-30 10:00:01 +08:00
|
|
|
break;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2013-05-30 10:00:01 +08:00
|
|
|
default:
|
|
|
|
|
|
|
|
/* Warn if arguments passed to an object that is not a method */
|
2006-10-02 12:00:00 +08:00
|
|
|
|
2013-05-30 10:00:01 +08:00
|
|
|
if (info->param_count) {
|
|
|
|
ACPI_WARNING((AE_INFO,
|
|
|
|
"%u arguments were passed to a non-method ACPI object",
|
|
|
|
info->param_count));
|
|
|
|
}
|
|
|
|
break;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2013-05-30 10:00:01 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Now we can evaluate the object */
|
|
|
|
|
|
|
|
status = acpi_ns_evaluate(info);
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* If we are expecting a return value, and all went well above,
|
|
|
|
* copy the return value to an external object.
|
|
|
|
*/
|
2016-05-05 12:58:00 +08:00
|
|
|
if (!return_buffer) {
|
|
|
|
goto cleanup_return_object;
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2016-05-05 12:58:00 +08:00
|
|
|
if (!info->return_object) {
|
|
|
|
return_buffer->length = 0;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2006-05-27 04:36:00 +08:00
|
|
|
|
2016-05-05 12:58:00 +08:00
|
|
|
if (ACPI_GET_DESCRIPTOR_TYPE(info->return_object) ==
|
|
|
|
ACPI_DESC_TYPE_NAMED) {
|
|
|
|
/*
|
|
|
|
* If we received a NS Node as a return object, this means that
|
|
|
|
* the object we are evaluating has nothing interesting to
|
|
|
|
* return (such as a mutex, etc.) We return an error because
|
|
|
|
* these types are essentially unsupported by this interface.
|
|
|
|
* We don't check up front because this makes it easier to add
|
|
|
|
* support for various types at a later date if necessary.
|
|
|
|
*/
|
|
|
|
status = AE_TYPE;
|
|
|
|
info->return_object = NULL; /* No need to delete a NS Node */
|
|
|
|
return_buffer->length = 0;
|
|
|
|
}
|
2008-08-04 13:22:10 +08:00
|
|
|
|
2016-05-05 12:58:00 +08:00
|
|
|
if (ACPI_FAILURE(status)) {
|
|
|
|
goto cleanup_return_object;
|
|
|
|
}
|
2008-08-04 13:22:10 +08:00
|
|
|
|
2016-05-05 12:58:00 +08:00
|
|
|
/* Dereference Index and ref_of references */
|
2006-05-27 04:36:00 +08:00
|
|
|
|
2016-05-05 12:58:00 +08:00
|
|
|
acpi_ns_resolve_references(info);
|
|
|
|
|
|
|
|
/* Get the size of the returned object */
|
|
|
|
|
|
|
|
status = acpi_ut_get_object_size(info->return_object,
|
|
|
|
&buffer_space_needed);
|
|
|
|
if (ACPI_SUCCESS(status)) {
|
|
|
|
|
|
|
|
/* Validate/Allocate/Clear caller buffer */
|
|
|
|
|
|
|
|
status = acpi_ut_initialize_buffer(return_buffer,
|
|
|
|
buffer_space_needed);
|
|
|
|
if (ACPI_FAILURE(status)) {
|
|
|
|
/*
|
|
|
|
* Caller's buffer is too small or a new one can't
|
|
|
|
* be allocated
|
|
|
|
*/
|
|
|
|
ACPI_DEBUG_PRINT((ACPI_DB_INFO,
|
|
|
|
"Needed buffer size %X, %s\n",
|
|
|
|
(u32)buffer_space_needed,
|
|
|
|
acpi_format_exception(status)));
|
|
|
|
} else {
|
|
|
|
/* We have enough space for the object, build it */
|
|
|
|
|
|
|
|
status =
|
|
|
|
acpi_ut_copy_iobject_to_eobject(info->return_object,
|
|
|
|
return_buffer);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-05 12:58:00 +08:00
|
|
|
cleanup_return_object:
|
|
|
|
|
2006-05-27 04:36:00 +08:00
|
|
|
if (info->return_object) {
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
2006-05-27 04:36:00 +08:00
|
|
|
* Delete the internal return object. NOTE: Interpreter must be
|
|
|
|
* locked to avoid race condition.
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
2007-05-10 10:56:38 +08:00
|
|
|
acpi_ex_enter_interpreter();
|
2006-05-27 04:36:00 +08:00
|
|
|
|
2007-05-10 10:56:38 +08:00
|
|
|
/* Remove one reference on the return object (should delete it) */
|
2006-05-27 04:36:00 +08:00
|
|
|
|
2007-05-10 10:56:38 +08:00
|
|
|
acpi_ut_remove_reference(info->return_object);
|
|
|
|
acpi_ex_exit_interpreter();
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2013-10-29 09:30:02 +08:00
|
|
|
cleanup:
|
2006-05-27 04:36:00 +08:00
|
|
|
|
2006-10-02 12:00:00 +08:00
|
|
|
/* Free the input parameter list (if we created one) */
|
|
|
|
|
2006-05-27 04:36:00 +08:00
|
|
|
if (info->parameters) {
|
2006-10-02 12:00:00 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/* Free the allocated parameter block */
|
|
|
|
|
2006-05-27 04:36:00 +08:00
|
|
|
acpi_ut_delete_internal_object_list(info->parameters);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2006-05-27 04:36:00 +08:00
|
|
|
ACPI_FREE(info);
|
2005-08-05 12:44:28 +08:00
|
|
|
return_ACPI_STATUS(status);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2006-10-03 12:00:00 +08:00
|
|
|
ACPI_EXPORT_SYMBOL(acpi_evaluate_object)
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2008-08-04 13:22:10 +08:00
|
|
|
/*******************************************************************************
|
|
|
|
*
|
|
|
|
* FUNCTION: acpi_ns_resolve_references
|
|
|
|
*
|
2012-07-12 09:40:10 +08:00
|
|
|
* PARAMETERS: info - Evaluation info block
|
2008-08-04 13:22:10 +08:00
|
|
|
*
|
|
|
|
* RETURN: Info->return_object is replaced with the dereferenced object
|
|
|
|
*
|
|
|
|
* DESCRIPTION: Dereference certain reference objects. Called before an
|
|
|
|
* internal return object is converted to an external union acpi_object.
|
|
|
|
*
|
|
|
|
* Performs an automatic dereference of Index and ref_of reference objects.
|
|
|
|
* These reference objects are not supported by the union acpi_object, so this is a
|
|
|
|
* last resort effort to return something useful. Also, provides compatibility
|
|
|
|
* with other ACPI implementations.
|
|
|
|
*
|
|
|
|
* NOTE: does not handle references within returned package objects or nested
|
|
|
|
* references, but this support could be added later if found to be necessary.
|
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
static void acpi_ns_resolve_references(struct acpi_evaluate_info *info)
|
|
|
|
{
|
|
|
|
union acpi_operand_object *obj_desc = NULL;
|
|
|
|
struct acpi_namespace_node *node;
|
|
|
|
|
|
|
|
/* We are interested in reference objects only */
|
|
|
|
|
2009-02-18 14:44:03 +08:00
|
|
|
if ((info->return_object)->common.type != ACPI_TYPE_LOCAL_REFERENCE) {
|
2008-08-04 13:22:10 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Two types of references are supported - those created by Index and
|
|
|
|
* ref_of operators. A name reference (AML_NAMEPATH_OP) can be converted
|
2017-02-28 06:28:49 +08:00
|
|
|
* to a union acpi_object, so it is not dereferenced here. A ddb_handle
|
2008-08-04 13:22:10 +08:00
|
|
|
* (AML_LOAD_OP) cannot be dereferenced, nor can it be converted to
|
2017-02-28 06:28:49 +08:00
|
|
|
* a union acpi_object.
|
2008-08-04 13:22:10 +08:00
|
|
|
*/
|
2008-09-27 11:08:41 +08:00
|
|
|
switch (info->return_object->reference.class) {
|
|
|
|
case ACPI_REFCLASS_INDEX:
|
2008-08-04 13:22:10 +08:00
|
|
|
|
|
|
|
obj_desc = *(info->return_object->reference.where);
|
|
|
|
break;
|
|
|
|
|
2008-09-27 11:08:41 +08:00
|
|
|
case ACPI_REFCLASS_REFOF:
|
2008-08-04 13:22:10 +08:00
|
|
|
|
|
|
|
node = info->return_object->reference.object;
|
|
|
|
if (node) {
|
|
|
|
obj_desc = node->object;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2013-06-08 08:58:14 +08:00
|
|
|
|
2008-08-04 13:22:10 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Replace the existing reference object */
|
|
|
|
|
|
|
|
if (obj_desc) {
|
|
|
|
acpi_ut_add_reference(obj_desc);
|
|
|
|
acpi_ut_remove_reference(info->return_object);
|
|
|
|
info->return_object = obj_desc;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*******************************************************************************
|
|
|
|
*
|
|
|
|
* FUNCTION: acpi_walk_namespace
|
|
|
|
*
|
2012-07-12 09:40:10 +08:00
|
|
|
* PARAMETERS: type - acpi_object_type to search for
|
2005-04-17 06:20:36 +08:00
|
|
|
* start_object - Handle in namespace where search begins
|
|
|
|
* max_depth - Depth to which search is to reach
|
2013-08-08 15:30:05 +08:00
|
|
|
* descending_callback - Called during tree descent
|
2009-11-13 10:06:08 +08:00
|
|
|
* when an object of "Type" is found
|
2013-08-08 15:30:05 +08:00
|
|
|
* ascending_callback - Called during tree ascent
|
2009-11-13 10:06:08 +08:00
|
|
|
* when an object of "Type" is found
|
2012-07-12 09:40:10 +08:00
|
|
|
* context - Passed to user function(s) above
|
2005-04-17 06:20:36 +08:00
|
|
|
* return_value - Location where return value of
|
|
|
|
* user_function is put if terminated early
|
|
|
|
*
|
|
|
|
* RETURNS Return value from the user_function if terminated early.
|
|
|
|
* Otherwise, returns NULL.
|
|
|
|
*
|
|
|
|
* DESCRIPTION: Performs a modified depth-first walk of the namespace tree,
|
|
|
|
* starting (and ending) at the object specified by start_handle.
|
2009-11-13 10:06:08 +08:00
|
|
|
* The callback function is called whenever an object that matches
|
|
|
|
* the type parameter is found. If the callback function returns
|
2005-04-17 06:20:36 +08:00
|
|
|
* a non-zero value, the search is terminated immediately and this
|
|
|
|
* value is returned to the caller.
|
|
|
|
*
|
|
|
|
* The point of this procedure is to provide a generic namespace
|
|
|
|
* walk routine that can be called from multiple places to
|
2009-11-13 10:06:08 +08:00
|
|
|
* provide multiple services; the callback function(s) can be
|
|
|
|
* tailored to each task, whether it is a print function,
|
|
|
|
* a compare function, etc.
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
|
|
|
******************************************************************************/
|
2008-08-04 13:22:10 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
acpi_status
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_walk_namespace(acpi_object_type type,
|
|
|
|
acpi_handle start_object,
|
|
|
|
u32 max_depth,
|
2013-08-08 15:30:05 +08:00
|
|
|
acpi_walk_callback descending_callback,
|
|
|
|
acpi_walk_callback ascending_callback,
|
2005-08-05 12:44:28 +08:00
|
|
|
void *context, void **return_value)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_status status;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
ACPI: ACPICA 20060421
Removed a device initialization optimization introduced in
20051216 where the _STA method was not run unless an _INI
was also present for the same device. This optimization
could cause problems because it could allow _INI methods
to be run within a not-present device subtree (If a
not-present device had no _INI, _STA would not be run,
the not-present status would not be discovered, and the
children of the device would be incorrectly traversed.)
Implemented a new _STA optimization where namespace
subtrees that do not contain _INI are identified and
ignored during device initialization. Selectively running
_STA can significantly improve boot time on large machines
(with assistance from Len Brown.)
Implemented support for the device initialization case
where the returned _STA flags indicate a device not-present
but functioning. In this case, _INI is not run, but the
device children are examined for presence, as per the
ACPI specification.
Implemented an additional change to the IndexField support
in order to conform to MS behavior. The value written to
the Index Register is not simply a byte offset, it is a
byte offset in units of the access width of the parent
Index Field. (Fiodor Suietov)
Defined and deployed a new OSL interface,
acpi_os_validate_address(). This interface is called during
the creation of all AML operation regions, and allows
the host OS to exert control over what addresses it will
allow the AML code to access. Operation Regions whose
addresses are disallowed will cause a runtime exception
when they are actually accessed (will not affect or abort
table loading.)
Defined and deployed a new OSL interface,
acpi_os_validate_interface(). This interface allows the host OS
to match the various "optional" interface/behavior strings
for the _OSI predefined control method as appropriate
(with assistance from Bjorn Helgaas.)
Restructured and corrected various problems in the
exception handling code paths within DsCallControlMethod
and DsTerminateControlMethod in dsmethod (with assistance
from Takayoshi Kochi.)
Modified the Linux source converter to ignore quoted string
literals while converting identifiers from mixed to lower
case. This will correct problems with the disassembler
and other areas where such strings must not be modified.
The ACPI_FUNCTION_* macros no longer require quotes around
the function name. This allows the Linux source converter
to convert the names, now that the converter ignores
quoted strings.
Signed-off-by: Bob Moore <robert.moore@intel.com>
Signed-off-by: Len Brown <len.brown@intel.com>
2006-04-22 05:15:00 +08:00
|
|
|
ACPI_FUNCTION_TRACE(acpi_walk_namespace);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* Parameter validation */
|
|
|
|
|
2009-11-13 10:06:08 +08:00
|
|
|
if ((type > ACPI_TYPE_LOCAL_MAX) ||
|
2013-08-08 15:30:05 +08:00
|
|
|
(!max_depth) || (!descending_callback && !ascending_callback)) {
|
2005-08-05 12:44:28 +08:00
|
|
|
return_ACPI_STATUS(AE_BAD_PARAMETER);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2009-03-09 16:31:04 +08:00
|
|
|
* Need to acquire the namespace reader lock to prevent interference
|
|
|
|
* with any concurrent table unloads (which causes the deletion of
|
|
|
|
* namespace objects). We cannot allow the deletion of a namespace node
|
|
|
|
* while the user function is using it. The exception to this are the
|
|
|
|
* nodes created and deleted during control method execution -- these
|
|
|
|
* nodes are marked as temporary nodes and are ignored by the namespace
|
|
|
|
* walk. Thus, control methods can be executed while holding the
|
|
|
|
* namespace deletion lock (and the user function can execute control
|
|
|
|
* methods.)
|
|
|
|
*/
|
|
|
|
status = acpi_ut_acquire_read_lock(&acpi_gbl_namespace_rw_lock);
|
|
|
|
if (ACPI_FAILURE(status)) {
|
2012-12-19 13:37:29 +08:00
|
|
|
return_ACPI_STATUS(status);
|
2009-03-09 16:31:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Lock the namespace around the walk. The namespace will be
|
|
|
|
* unlocked/locked around each call to the user function - since the user
|
|
|
|
* function must be allowed to make ACPICA calls itself (for example, it
|
|
|
|
* will typically execute control methods during device enumeration.)
|
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)) {
|
2009-03-09 16:31:04 +08:00
|
|
|
goto unlock_and_exit;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2013-09-23 09:51:58 +08:00
|
|
|
/* Now we can validate the starting node */
|
|
|
|
|
|
|
|
if (!acpi_ns_validate_handle(start_object)) {
|
|
|
|
status = AE_BAD_PARAMETER;
|
|
|
|
goto unlock_and_exit2;
|
|
|
|
}
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
status = acpi_ns_walk_namespace(type, start_object, max_depth,
|
2013-08-08 15:30:05 +08:00
|
|
|
ACPI_NS_WALK_UNLOCK,
|
|
|
|
descending_callback, ascending_callback,
|
|
|
|
context, return_value);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2013-10-29 09:30:02 +08:00
|
|
|
unlock_and_exit2:
|
2005-08-05 12:44:28 +08:00
|
|
|
(void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE);
|
2009-03-09 16:31:04 +08:00
|
|
|
|
2013-10-29 09:30:02 +08:00
|
|
|
unlock_and_exit:
|
2009-03-09 16:31:04 +08:00
|
|
|
(void)acpi_ut_release_read_lock(&acpi_gbl_namespace_rw_lock);
|
2005-08-05 12:44:28 +08:00
|
|
|
return_ACPI_STATUS(status);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2006-10-03 12:00:00 +08:00
|
|
|
ACPI_EXPORT_SYMBOL(acpi_walk_namespace)
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
*
|
|
|
|
* FUNCTION: acpi_ns_get_device_callback
|
|
|
|
*
|
|
|
|
* PARAMETERS: Callback from acpi_get_device
|
|
|
|
*
|
|
|
|
* RETURN: Status
|
|
|
|
*
|
|
|
|
* DESCRIPTION: Takes callbacks from walk_namespace and filters out all non-
|
|
|
|
* present devices, or if they specified a HID, it filters based
|
|
|
|
* on that.
|
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
static acpi_status
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_ns_get_device_callback(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_get_devices_info *info = context;
|
|
|
|
acpi_status status;
|
|
|
|
struct acpi_namespace_node *node;
|
|
|
|
u32 flags;
|
2012-10-31 10:25:24 +08:00
|
|
|
struct acpi_pnp_device_id *hid;
|
|
|
|
struct acpi_pnp_device_id_list *cid;
|
2008-06-10 13:42:13 +08:00
|
|
|
u32 i;
|
2009-06-29 13:39:29 +08:00
|
|
|
u8 found;
|
|
|
|
int no_match;
|
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)) {
|
2012-12-19 13:37:29 +08:00
|
|
|
return (status);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2009-12-11 14:57:00 +08:00
|
|
|
node = acpi_ns_validate_handle(obj_handle);
|
2005-08-05 12:44:28 +08:00
|
|
|
status = acpi_ut_release_mutex(ACPI_MTX_NAMESPACE);
|
|
|
|
if (ACPI_FAILURE(status)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
return (status);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!node) {
|
|
|
|
return (AE_BAD_PARAMETER);
|
|
|
|
}
|
|
|
|
|
2010-01-21 09:15:20 +08:00
|
|
|
/*
|
|
|
|
* First, filter based on the device HID and CID.
|
|
|
|
*
|
|
|
|
* 01/2010: For this case where a specific HID is requested, we don't
|
|
|
|
* want to run _STA until we have an actual HID match. Thus, we will
|
|
|
|
* not unnecessarily execute _STA on devices for which the caller
|
|
|
|
* doesn't care about. Previously, _STA was executed unconditionally
|
|
|
|
* on all devices found here.
|
|
|
|
*
|
|
|
|
* A side-effect of this change is that now we will continue to search
|
|
|
|
* for a matching HID even under device trees where the parent device
|
|
|
|
* would have returned a _STA that indicates it is not present or
|
|
|
|
* not functioning (thus aborting the search on that branch).
|
|
|
|
*/
|
2005-04-17 06:20:36 +08:00
|
|
|
if (info->hid != NULL) {
|
2005-08-05 12:44:28 +08:00
|
|
|
status = acpi_ut_execute_HID(node, &hid);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (status == AE_NOT_FOUND) {
|
|
|
|
return (AE_OK);
|
2005-08-05 12:44:28 +08:00
|
|
|
} else if (ACPI_FAILURE(status)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
return (AE_CTRL_DEPTH);
|
|
|
|
}
|
|
|
|
|
2015-07-01 14:45:11 +08:00
|
|
|
no_match = strcmp(hid->string, info->hid);
|
2009-06-29 13:39:29 +08:00
|
|
|
ACPI_FREE(hid);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2009-06-29 13:39:29 +08:00
|
|
|
if (no_match) {
|
|
|
|
/*
|
|
|
|
* HID does not match, attempt match within the
|
|
|
|
* list of Compatible IDs (CIDs)
|
|
|
|
*/
|
2005-08-05 12:44:28 +08:00
|
|
|
status = acpi_ut_execute_CID(node, &cid);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (status == AE_NOT_FOUND) {
|
|
|
|
return (AE_OK);
|
2005-08-05 12:44:28 +08:00
|
|
|
} else if (ACPI_FAILURE(status)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
return (AE_CTRL_DEPTH);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Walk the CID list */
|
|
|
|
|
2012-12-20 09:07:26 +08:00
|
|
|
found = FALSE;
|
2005-04-17 06:20:36 +08:00
|
|
|
for (i = 0; i < cid->count; i++) {
|
2015-07-01 14:45:11 +08:00
|
|
|
if (strcmp(cid->ids[i].string, info->hid) == 0) {
|
2012-12-19 13:37:15 +08:00
|
|
|
|
2012-12-19 13:36:49 +08:00
|
|
|
/* Found a matching CID */
|
2012-12-19 13:37:15 +08:00
|
|
|
|
2012-12-20 09:07:26 +08:00
|
|
|
found = TRUE;
|
2008-01-23 08:18:22 +08:00
|
|
|
break;
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
}
|
2012-12-20 09:07:26 +08:00
|
|
|
|
2006-10-03 12:00:00 +08:00
|
|
|
ACPI_FREE(cid);
|
2012-12-20 09:07:26 +08:00
|
|
|
if (!found) {
|
2008-01-23 08:18:22 +08:00
|
|
|
return (AE_OK);
|
2012-12-20 09:07:26 +08:00
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-01-21 09:15:20 +08:00
|
|
|
/* Run _STA to determine if device is present */
|
|
|
|
|
|
|
|
status = acpi_ut_execute_STA(node, &flags);
|
|
|
|
if (ACPI_FAILURE(status)) {
|
|
|
|
return (AE_CTRL_DEPTH);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(flags & ACPI_STA_DEVICE_PRESENT) &&
|
|
|
|
!(flags & ACPI_STA_DEVICE_FUNCTIONING)) {
|
|
|
|
/*
|
|
|
|
* Don't examine the children of the device only when the
|
|
|
|
* device is neither present nor functional. See ACPI spec,
|
|
|
|
* description of _STA for more information.
|
|
|
|
*/
|
|
|
|
return (AE_CTRL_DEPTH);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We have a valid device, invoke the user function */
|
|
|
|
|
2015-12-29 13:54:36 +08:00
|
|
|
status = info->user_function(obj_handle, nesting_level,
|
|
|
|
info->context, return_value);
|
2005-04-17 06:20:36 +08:00
|
|
|
return (status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
*
|
|
|
|
* FUNCTION: acpi_get_devices
|
|
|
|
*
|
|
|
|
* PARAMETERS: HID - HID to search for. Can be NULL.
|
|
|
|
* user_function - Called when a matching object is found
|
2012-07-12 09:40:10 +08:00
|
|
|
* context - Passed to user function
|
2005-04-17 06:20:36 +08:00
|
|
|
* return_value - Location where return value of
|
|
|
|
* user_function is put if terminated early
|
|
|
|
*
|
|
|
|
* RETURNS Return value from the user_function if terminated early.
|
|
|
|
* Otherwise, returns NULL.
|
|
|
|
*
|
|
|
|
* DESCRIPTION: Performs a modified depth-first walk of the namespace tree,
|
|
|
|
* starting (and ending) at the object specified by start_handle.
|
|
|
|
* The user_function is called whenever an object of type
|
2012-10-31 10:26:55 +08:00
|
|
|
* Device is found. If the user function returns
|
2005-04-17 06:20:36 +08:00
|
|
|
* a non-zero value, the search is terminated immediately and this
|
|
|
|
* value is returned to the caller.
|
|
|
|
*
|
|
|
|
* This is a wrapper for walk_namespace, but the callback performs
|
2008-04-10 23:06:41 +08:00
|
|
|
* additional filtering. Please see acpi_ns_get_device_callback.
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
|
|
|
|
acpi_status
|
2007-08-21 23:18:20 +08:00
|
|
|
acpi_get_devices(const char *HID,
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_walk_callback user_function,
|
|
|
|
void *context, void **return_value)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_status status;
|
|
|
|
struct acpi_get_devices_info info;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
ACPI: ACPICA 20060421
Removed a device initialization optimization introduced in
20051216 where the _STA method was not run unless an _INI
was also present for the same device. This optimization
could cause problems because it could allow _INI methods
to be run within a not-present device subtree (If a
not-present device had no _INI, _STA would not be run,
the not-present status would not be discovered, and the
children of the device would be incorrectly traversed.)
Implemented a new _STA optimization where namespace
subtrees that do not contain _INI are identified and
ignored during device initialization. Selectively running
_STA can significantly improve boot time on large machines
(with assistance from Len Brown.)
Implemented support for the device initialization case
where the returned _STA flags indicate a device not-present
but functioning. In this case, _INI is not run, but the
device children are examined for presence, as per the
ACPI specification.
Implemented an additional change to the IndexField support
in order to conform to MS behavior. The value written to
the Index Register is not simply a byte offset, it is a
byte offset in units of the access width of the parent
Index Field. (Fiodor Suietov)
Defined and deployed a new OSL interface,
acpi_os_validate_address(). This interface is called during
the creation of all AML operation regions, and allows
the host OS to exert control over what addresses it will
allow the AML code to access. Operation Regions whose
addresses are disallowed will cause a runtime exception
when they are actually accessed (will not affect or abort
table loading.)
Defined and deployed a new OSL interface,
acpi_os_validate_interface(). This interface allows the host OS
to match the various "optional" interface/behavior strings
for the _OSI predefined control method as appropriate
(with assistance from Bjorn Helgaas.)
Restructured and corrected various problems in the
exception handling code paths within DsCallControlMethod
and DsTerminateControlMethod in dsmethod (with assistance
from Takayoshi Kochi.)
Modified the Linux source converter to ignore quoted string
literals while converting identifiers from mixed to lower
case. This will correct problems with the disassembler
and other areas where such strings must not be modified.
The ACPI_FUNCTION_* macros no longer require quotes around
the function name. This allows the Linux source converter
to convert the names, now that the converter ignores
quoted strings.
Signed-off-by: Bob Moore <robert.moore@intel.com>
Signed-off-by: Len Brown <len.brown@intel.com>
2006-04-22 05:15:00 +08:00
|
|
|
ACPI_FUNCTION_TRACE(acpi_get_devices);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* Parameter validation */
|
|
|
|
|
|
|
|
if (!user_function) {
|
2005-08-05 12:44:28 +08:00
|
|
|
return_ACPI_STATUS(AE_BAD_PARAMETER);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We're going to call their callback from OUR callback, so we need
|
|
|
|
* to know what it is, and their context parameter.
|
|
|
|
*/
|
2006-10-02 12:00:00 +08:00
|
|
|
info.hid = HID;
|
2005-08-05 12:44:28 +08:00
|
|
|
info.context = context;
|
2005-04-17 06:20:36 +08:00
|
|
|
info.user_function = user_function;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Lock the namespace around the walk.
|
|
|
|
* The namespace will be unlocked/locked around each call
|
|
|
|
* to the user function - since this function
|
|
|
|
* must be allowed to make Acpi calls itself.
|
|
|
|
*/
|
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
|
|
|
}
|
|
|
|
|
2006-10-02 12:00:00 +08:00
|
|
|
status = acpi_ns_walk_namespace(ACPI_TYPE_DEVICE, ACPI_ROOT_OBJECT,
|
|
|
|
ACPI_UINT32_MAX, ACPI_NS_WALK_UNLOCK,
|
2009-11-13 10:06:08 +08:00
|
|
|
acpi_ns_get_device_callback, NULL,
|
|
|
|
&info, return_value);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
(void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE);
|
|
|
|
return_ACPI_STATUS(status);
|
2005-04-17 06:20:36 +08:00
|
|
|
}
|
|
|
|
|
2006-10-03 12:00:00 +08:00
|
|
|
ACPI_EXPORT_SYMBOL(acpi_get_devices)
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
*
|
|
|
|
* FUNCTION: acpi_attach_data
|
|
|
|
*
|
|
|
|
* PARAMETERS: obj_handle - Namespace node
|
2012-07-12 09:40:10 +08:00
|
|
|
* handler - Handler for this attachment
|
|
|
|
* data - Pointer to data to be attached
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
|
|
|
* RETURN: Status
|
|
|
|
*
|
|
|
|
* DESCRIPTION: Attach arbitrary data and handler to a namespace node.
|
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
acpi_status
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_attach_data(acpi_handle obj_handle,
|
|
|
|
acpi_object_handler handler, void *data)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_namespace_node *node;
|
|
|
|
acpi_status status;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* Parameter validation */
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
if (!obj_handle || !handler || !data) {
|
2005-04-17 06:20:36 +08:00
|
|
|
return (AE_BAD_PARAMETER);
|
|
|
|
}
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
status = acpi_ut_acquire_mutex(ACPI_MTX_NAMESPACE);
|
|
|
|
if (ACPI_FAILURE(status)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
return (status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Convert and validate the handle */
|
|
|
|
|
2009-12-11 14:57:00 +08:00
|
|
|
node = acpi_ns_validate_handle(obj_handle);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!node) {
|
|
|
|
status = AE_BAD_PARAMETER;
|
|
|
|
goto unlock_and_exit;
|
|
|
|
}
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
status = acpi_ns_attach_data(node, handler, data);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2013-10-29 09:30:02 +08:00
|
|
|
unlock_and_exit:
|
2005-08-05 12:44:28 +08:00
|
|
|
(void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE);
|
2005-04-17 06:20:36 +08:00
|
|
|
return (status);
|
|
|
|
}
|
|
|
|
|
2006-10-03 12:00:00 +08:00
|
|
|
ACPI_EXPORT_SYMBOL(acpi_attach_data)
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*******************************************************************************
|
|
|
|
*
|
|
|
|
* FUNCTION: acpi_detach_data
|
|
|
|
*
|
|
|
|
* PARAMETERS: obj_handle - Namespace node handle
|
2012-07-12 09:40:10 +08:00
|
|
|
* handler - Handler used in call to acpi_attach_data
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
|
|
|
* RETURN: Status
|
|
|
|
*
|
|
|
|
* DESCRIPTION: Remove data that was previously attached to a node.
|
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
acpi_status
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_detach_data(acpi_handle obj_handle, acpi_object_handler handler)
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_namespace_node *node;
|
|
|
|
acpi_status status;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* Parameter validation */
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
if (!obj_handle || !handler) {
|
2005-04-17 06:20:36 +08:00
|
|
|
return (AE_BAD_PARAMETER);
|
|
|
|
}
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
status = acpi_ut_acquire_mutex(ACPI_MTX_NAMESPACE);
|
|
|
|
if (ACPI_FAILURE(status)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
return (status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Convert and validate the handle */
|
|
|
|
|
2009-12-11 14:57:00 +08:00
|
|
|
node = acpi_ns_validate_handle(obj_handle);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!node) {
|
|
|
|
status = AE_BAD_PARAMETER;
|
|
|
|
goto unlock_and_exit;
|
|
|
|
}
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
status = acpi_ns_detach_data(node, handler);
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2013-10-29 09:30:02 +08:00
|
|
|
unlock_and_exit:
|
2005-08-05 12:44:28 +08:00
|
|
|
(void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE);
|
2005-04-17 06:20:36 +08:00
|
|
|
return (status);
|
|
|
|
}
|
|
|
|
|
2006-10-03 12:00:00 +08:00
|
|
|
ACPI_EXPORT_SYMBOL(acpi_detach_data)
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*******************************************************************************
|
|
|
|
*
|
2014-02-04 07:42:46 +08:00
|
|
|
* FUNCTION: acpi_get_data_full
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
|
|
|
* PARAMETERS: obj_handle - Namespace node
|
2012-07-12 09:40:10 +08:00
|
|
|
* handler - Handler used in call to attach_data
|
|
|
|
* data - Where the data is returned
|
2014-02-04 07:42:46 +08:00
|
|
|
* callback - function to execute before returning
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
|
|
|
* RETURN: Status
|
|
|
|
*
|
2014-02-04 07:42:46 +08:00
|
|
|
* DESCRIPTION: Retrieve data that was previously attached to a namespace node
|
|
|
|
* and execute a callback before returning.
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
acpi_status
|
2014-02-04 07:42:46 +08:00
|
|
|
acpi_get_data_full(acpi_handle obj_handle, acpi_object_handler handler,
|
|
|
|
void **data, void (*callback)(void *))
|
2005-04-17 06:20:36 +08:00
|
|
|
{
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_namespace_node *node;
|
|
|
|
acpi_status status;
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/* Parameter validation */
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
if (!obj_handle || !handler || !data) {
|
2005-04-17 06:20:36 +08:00
|
|
|
return (AE_BAD_PARAMETER);
|
|
|
|
}
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
status = acpi_ut_acquire_mutex(ACPI_MTX_NAMESPACE);
|
|
|
|
if (ACPI_FAILURE(status)) {
|
2005-04-17 06:20:36 +08:00
|
|
|
return (status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Convert and validate the handle */
|
|
|
|
|
2009-12-11 14:57:00 +08:00
|
|
|
node = acpi_ns_validate_handle(obj_handle);
|
2005-04-17 06:20:36 +08:00
|
|
|
if (!node) {
|
|
|
|
status = AE_BAD_PARAMETER;
|
|
|
|
goto unlock_and_exit;
|
|
|
|
}
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
status = acpi_ns_get_attached_data(node, handler, data);
|
2014-02-04 07:42:46 +08:00
|
|
|
if (ACPI_SUCCESS(status) && callback) {
|
|
|
|
callback(*data);
|
|
|
|
}
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2013-10-29 09:30:02 +08:00
|
|
|
unlock_and_exit:
|
2005-08-05 12:44:28 +08:00
|
|
|
(void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE);
|
2005-04-17 06:20:36 +08:00
|
|
|
return (status);
|
|
|
|
}
|
2006-10-03 12:00:00 +08:00
|
|
|
|
2014-02-04 07:42:46 +08:00
|
|
|
ACPI_EXPORT_SYMBOL(acpi_get_data_full)
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
*
|
|
|
|
* FUNCTION: acpi_get_data
|
|
|
|
*
|
|
|
|
* PARAMETERS: obj_handle - Namespace node
|
|
|
|
* handler - Handler used in call to attach_data
|
|
|
|
* data - Where the data is returned
|
|
|
|
*
|
|
|
|
* RETURN: Status
|
|
|
|
*
|
|
|
|
* DESCRIPTION: Retrieve data that was previously attached to a namespace node.
|
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
acpi_status
|
|
|
|
acpi_get_data(acpi_handle obj_handle, acpi_object_handler handler, void **data)
|
|
|
|
{
|
|
|
|
return acpi_get_data_full(obj_handle, handler, data, NULL);
|
|
|
|
}
|
|
|
|
|
2006-10-03 12:00:00 +08:00
|
|
|
ACPI_EXPORT_SYMBOL(acpi_get_data)
|