1 // SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
2 /*******************************************************************************
4 * Module Name: nsxfeval - Public interfaces to the ACPI subsystem
5 * ACPI Object evaluation interfaces
7 ******************************************************************************/
9 #define EXPORT_ACPI_INTERFACES
11 #include <acpi/acpi.h>
16 #define _COMPONENT ACPI_NAMESPACE
17 ACPI_MODULE_NAME("nsxfeval")
19 /* Local prototypes */
20 static void acpi_ns_resolve_references(struct acpi_evaluate_info
*info
);
22 /*******************************************************************************
24 * FUNCTION: acpi_evaluate_object_typed
26 * PARAMETERS: handle - Object handle (optional)
27 * pathname - Object pathname (optional)
28 * external_params - List of parameters to pass to a method,
29 * terminated by NULL. May be NULL
30 * if no parameters are being passed.
31 * return_buffer - Where to put the object's return value (if
32 * any). If NULL, no value is returned.
33 * return_type - Expected type of return object
37 * DESCRIPTION: Find and evaluate the given object, passing the given
38 * parameters if necessary. One of "Handle" or "Pathname" must
41 ******************************************************************************/
44 acpi_evaluate_object_typed(acpi_handle handle
,
46 struct acpi_object_list
*external_params
,
47 struct acpi_buffer
*return_buffer
,
48 acpi_object_type return_type
)
51 u8 free_buffer_on_error
= FALSE
;
52 acpi_handle target_handle
;
55 ACPI_FUNCTION_TRACE(acpi_evaluate_object_typed
);
57 /* Return buffer must be valid */
60 return_ACPI_STATUS(AE_BAD_PARAMETER
);
63 if (return_buffer
->length
== ACPI_ALLOCATE_BUFFER
) {
64 free_buffer_on_error
= TRUE
;
67 /* Get a handle here, in order to build an error message if needed */
69 target_handle
= handle
;
71 status
= acpi_get_handle(handle
, pathname
, &target_handle
);
72 if (ACPI_FAILURE(status
)) {
73 return_ACPI_STATUS(status
);
77 full_pathname
= acpi_ns_get_external_pathname(target_handle
);
79 return_ACPI_STATUS(AE_NO_MEMORY
);
82 /* Evaluate the object */
84 status
= acpi_evaluate_object(target_handle
, NULL
, external_params
,
86 if (ACPI_FAILURE(status
)) {
90 /* Type ANY means "don't care about return value type" */
92 if (return_type
== ACPI_TYPE_ANY
) {
96 if (return_buffer
->length
== 0) {
98 /* Error because caller specifically asked for a return value */
100 ACPI_ERROR((AE_INFO
, "%s did not return any object",
102 status
= AE_NULL_OBJECT
;
106 /* Examine the object type returned from evaluate_object */
108 if (((union acpi_object
*)return_buffer
->pointer
)->type
== return_type
) {
112 /* Return object type does not match requested type */
115 "Incorrect return type from %s - received [%s], requested [%s]",
117 acpi_ut_get_type_name(((union acpi_object
*)return_buffer
->
119 acpi_ut_get_type_name(return_type
)));
121 if (free_buffer_on_error
) {
123 * Free a buffer created via ACPI_ALLOCATE_BUFFER.
124 * Note: We use acpi_os_free here because acpi_os_allocate was used
125 * to allocate the buffer. This purposefully bypasses the
126 * (optionally enabled) allocation tracking mechanism since we
127 * only want to track internal allocations.
129 acpi_os_free(return_buffer
->pointer
);
130 return_buffer
->pointer
= NULL
;
133 return_buffer
->length
= 0;
137 ACPI_FREE(full_pathname
);
138 return_ACPI_STATUS(status
);
141 ACPI_EXPORT_SYMBOL(acpi_evaluate_object_typed
)
143 /*******************************************************************************
145 * FUNCTION: acpi_evaluate_object
147 * PARAMETERS: handle - Object handle (optional)
148 * pathname - Object pathname (optional)
149 * external_params - List of parameters to pass to method,
150 * terminated by NULL. May be NULL
151 * if no parameters are being passed.
152 * return_buffer - Where to put method's return value (if
153 * any). If NULL, no value is returned.
157 * DESCRIPTION: Find and evaluate the given object, passing the given
158 * parameters if necessary. One of "Handle" or "Pathname" must
159 * be valid (non-null)
161 ******************************************************************************/
163 acpi_evaluate_object(acpi_handle handle
,
164 acpi_string pathname
,
165 struct acpi_object_list
*external_params
,
166 struct acpi_buffer
*return_buffer
)
169 struct acpi_evaluate_info
*info
;
170 acpi_size buffer_space_needed
;
173 ACPI_FUNCTION_TRACE(acpi_evaluate_object
);
175 /* Allocate and initialize the evaluation information block */
177 info
= ACPI_ALLOCATE_ZEROED(sizeof(struct acpi_evaluate_info
));
179 return_ACPI_STATUS(AE_NO_MEMORY
);
182 /* Convert and validate the device handle */
184 info
->prefix_node
= acpi_ns_validate_handle(handle
);
185 if (!info
->prefix_node
) {
186 status
= AE_BAD_PARAMETER
;
191 * Get the actual namespace node for the target object.
192 * Handles these cases:
194 * 1) Null node, valid pathname from root (absolute path)
195 * 2) Node and valid pathname (path relative to Node)
196 * 3) Node, Null pathname
198 if ((pathname
) && (ACPI_IS_ROOT_PREFIX(pathname
[0]))) {
200 /* The path is fully qualified, just evaluate by name */
202 info
->prefix_node
= NULL
;
203 } else if (!handle
) {
205 * A handle is optional iff a fully qualified pathname is specified.
206 * Since we've already handled fully qualified names above, this is
210 ACPI_DEBUG_PRINT((ACPI_DB_INFO
,
211 "Both Handle and Pathname are NULL"));
213 ACPI_DEBUG_PRINT((ACPI_DB_INFO
,
214 "Null Handle with relative pathname [%s]",
218 status
= AE_BAD_PARAMETER
;
222 info
->relative_pathname
= pathname
;
225 * Convert all external objects passed as arguments to the
226 * internal version(s).
228 if (external_params
&& external_params
->count
) {
229 info
->param_count
= (u16
)external_params
->count
;
231 /* Warn on impossible argument count */
233 if (info
->param_count
> ACPI_METHOD_NUM_ARGS
) {
234 ACPI_WARN_PREDEFINED((AE_INFO
, pathname
,
236 "Excess arguments (%u) - using only %u",
238 ACPI_METHOD_NUM_ARGS
));
240 info
->param_count
= ACPI_METHOD_NUM_ARGS
;
244 * Allocate a new parameter block for the internal objects
245 * Add 1 to count to allow for null terminated internal list
247 info
->parameters
= ACPI_ALLOCATE_ZEROED(((acpi_size
)info
->
249 1) * sizeof(void *));
250 if (!info
->parameters
) {
251 status
= AE_NO_MEMORY
;
255 /* Convert each external object in the list to an internal object */
257 for (i
= 0; i
< info
->param_count
; i
++) {
259 acpi_ut_copy_eobject_to_iobject(&external_params
->
263 if (ACPI_FAILURE(status
)) {
268 info
->parameters
[info
->param_count
] = NULL
;
271 #ifdef _FUTURE_FEATURE
274 * Begin incoming argument count analysis. Check for too few args
277 switch (acpi_ns_get_type(info
->node
)) {
278 case ACPI_TYPE_METHOD
:
280 /* Check incoming argument count against the method definition */
282 if (info
->obj_desc
->method
.param_count
> info
->param_count
) {
284 "Insufficient arguments (%u) - %u are required",
286 info
->obj_desc
->method
.param_count
));
288 status
= AE_MISSING_ARGUMENTS
;
292 else if (info
->obj_desc
->method
.param_count
< info
->param_count
) {
293 ACPI_WARNING((AE_INFO
,
294 "Excess arguments (%u) - only %u are required",
296 info
->obj_desc
->method
.param_count
));
298 /* Just pass the required number of arguments */
300 info
->param_count
= info
->obj_desc
->method
.param_count
;
304 * Any incoming external objects to be passed as arguments to the
305 * method must be converted to internal objects
307 if (info
->param_count
) {
309 * Allocate a new parameter block for the internal objects
310 * Add 1 to count to allow for null terminated internal list
312 info
->parameters
= ACPI_ALLOCATE_ZEROED(((acpi_size
)
317 if (!info
->parameters
) {
318 status
= AE_NO_MEMORY
;
322 /* Convert each external object in the list to an internal object */
324 for (i
= 0; i
< info
->param_count
; i
++) {
326 acpi_ut_copy_eobject_to_iobject
327 (&external_params
->pointer
[i
],
328 &info
->parameters
[i
]);
329 if (ACPI_FAILURE(status
)) {
334 info
->parameters
[info
->param_count
] = NULL
;
340 /* Warn if arguments passed to an object that is not a method */
342 if (info
->param_count
) {
343 ACPI_WARNING((AE_INFO
,
344 "%u arguments were passed to a non-method ACPI object",
352 /* Now we can evaluate the object */
354 status
= acpi_ns_evaluate(info
);
357 * If we are expecting a return value, and all went well above,
358 * copy the return value to an external object.
360 if (!return_buffer
) {
361 goto cleanup_return_object
;
364 if (!info
->return_object
) {
365 return_buffer
->length
= 0;
369 if (ACPI_GET_DESCRIPTOR_TYPE(info
->return_object
) ==
370 ACPI_DESC_TYPE_NAMED
) {
372 * If we received a NS Node as a return object, this means that
373 * the object we are evaluating has nothing interesting to
374 * return (such as a mutex, etc.) We return an error because
375 * these types are essentially unsupported by this interface.
376 * We don't check up front because this makes it easier to add
377 * support for various types at a later date if necessary.
380 info
->return_object
= NULL
; /* No need to delete a NS Node */
381 return_buffer
->length
= 0;
384 if (ACPI_FAILURE(status
)) {
385 goto cleanup_return_object
;
388 /* Dereference Index and ref_of references */
390 acpi_ns_resolve_references(info
);
392 /* Get the size of the returned object */
394 status
= acpi_ut_get_object_size(info
->return_object
,
395 &buffer_space_needed
);
396 if (ACPI_SUCCESS(status
)) {
398 /* Validate/Allocate/Clear caller buffer */
400 status
= acpi_ut_initialize_buffer(return_buffer
,
401 buffer_space_needed
);
402 if (ACPI_FAILURE(status
)) {
404 * Caller's buffer is too small or a new one can't
407 ACPI_DEBUG_PRINT((ACPI_DB_INFO
,
408 "Needed buffer size %X, %s\n",
409 (u32
)buffer_space_needed
,
410 acpi_format_exception(status
)));
412 /* We have enough space for the object, build it */
415 acpi_ut_copy_iobject_to_eobject(info
->return_object
,
420 cleanup_return_object
:
422 if (info
->return_object
) {
424 * Delete the internal return object. NOTE: Interpreter must be
425 * locked to avoid race condition.
427 acpi_ex_enter_interpreter();
429 /* Remove one reference on the return object (should delete it) */
431 acpi_ut_remove_reference(info
->return_object
);
432 acpi_ex_exit_interpreter();
437 /* Free the input parameter list (if we created one) */
439 if (info
->parameters
) {
441 /* Free the allocated parameter block */
443 acpi_ut_delete_internal_object_list(info
->parameters
);
447 return_ACPI_STATUS(status
);
450 ACPI_EXPORT_SYMBOL(acpi_evaluate_object
)
452 /*******************************************************************************
454 * FUNCTION: acpi_ns_resolve_references
456 * PARAMETERS: info - Evaluation info block
458 * RETURN: Info->return_object is replaced with the dereferenced object
460 * DESCRIPTION: Dereference certain reference objects. Called before an
461 * internal return object is converted to an external union acpi_object.
463 * Performs an automatic dereference of Index and ref_of reference objects.
464 * These reference objects are not supported by the union acpi_object, so this is a
465 * last resort effort to return something useful. Also, provides compatibility
466 * with other ACPI implementations.
468 * NOTE: does not handle references within returned package objects or nested
469 * references, but this support could be added later if found to be necessary.
471 ******************************************************************************/
472 static void acpi_ns_resolve_references(struct acpi_evaluate_info
*info
)
474 union acpi_operand_object
*obj_desc
= NULL
;
475 struct acpi_namespace_node
*node
;
477 /* We are interested in reference objects only */
479 if ((info
->return_object
)->common
.type
!= ACPI_TYPE_LOCAL_REFERENCE
) {
484 * Two types of references are supported - those created by Index and
485 * ref_of operators. A name reference (AML_NAMEPATH_OP) can be converted
486 * to a union acpi_object, so it is not dereferenced here. A ddb_handle
487 * (AML_LOAD_OP) cannot be dereferenced, nor can it be converted to
488 * a union acpi_object.
490 switch (info
->return_object
->reference
.class) {
491 case ACPI_REFCLASS_INDEX
:
493 obj_desc
= *(info
->return_object
->reference
.where
);
496 case ACPI_REFCLASS_REFOF
:
498 node
= info
->return_object
->reference
.object
;
500 obj_desc
= node
->object
;
509 /* Replace the existing reference object */
512 acpi_ut_add_reference(obj_desc
);
513 acpi_ut_remove_reference(info
->return_object
);
514 info
->return_object
= obj_desc
;
520 /*******************************************************************************
522 * FUNCTION: acpi_walk_namespace
524 * PARAMETERS: type - acpi_object_type to search for
525 * start_object - Handle in namespace where search begins
526 * max_depth - Depth to which search is to reach
527 * descending_callback - Called during tree descent
528 * when an object of "Type" is found
529 * ascending_callback - Called during tree ascent
530 * when an object of "Type" is found
531 * context - Passed to user function(s) above
532 * return_value - Location where return value of
533 * user_function is put if terminated early
535 * RETURNS Return value from the user_function if terminated early.
536 * Otherwise, returns NULL.
538 * DESCRIPTION: Performs a modified depth-first walk of the namespace tree,
539 * starting (and ending) at the object specified by start_handle.
540 * The callback function is called whenever an object that matches
541 * the type parameter is found. If the callback function returns
542 * a non-zero value, the search is terminated immediately and this
543 * value is returned to the caller.
545 * The point of this procedure is to provide a generic namespace
546 * walk routine that can be called from multiple places to
547 * provide multiple services; the callback function(s) can be
548 * tailored to each task, whether it is a print function,
549 * a compare function, etc.
551 ******************************************************************************/
554 acpi_walk_namespace(acpi_object_type type
,
555 acpi_handle start_object
,
557 acpi_walk_callback descending_callback
,
558 acpi_walk_callback ascending_callback
,
559 void *context
, void **return_value
)
563 ACPI_FUNCTION_TRACE(acpi_walk_namespace
);
565 /* Parameter validation */
567 if ((type
> ACPI_TYPE_LOCAL_MAX
) ||
568 (!max_depth
) || (!descending_callback
&& !ascending_callback
)) {
569 return_ACPI_STATUS(AE_BAD_PARAMETER
);
573 * Need to acquire the namespace reader lock to prevent interference
574 * with any concurrent table unloads (which causes the deletion of
575 * namespace objects). We cannot allow the deletion of a namespace node
576 * while the user function is using it. The exception to this are the
577 * nodes created and deleted during control method execution -- these
578 * nodes are marked as temporary nodes and are ignored by the namespace
579 * walk. Thus, control methods can be executed while holding the
580 * namespace deletion lock (and the user function can execute control
583 status
= acpi_ut_acquire_read_lock(&acpi_gbl_namespace_rw_lock
);
584 if (ACPI_FAILURE(status
)) {
585 return_ACPI_STATUS(status
);
589 * Lock the namespace around the walk. The namespace will be
590 * unlocked/locked around each call to the user function - since the user
591 * function must be allowed to make ACPICA calls itself (for example, it
592 * will typically execute control methods during device enumeration.)
594 status
= acpi_ut_acquire_mutex(ACPI_MTX_NAMESPACE
);
595 if (ACPI_FAILURE(status
)) {
596 goto unlock_and_exit
;
599 /* Now we can validate the starting node */
601 if (!acpi_ns_validate_handle(start_object
)) {
602 status
= AE_BAD_PARAMETER
;
603 goto unlock_and_exit2
;
606 status
= acpi_ns_walk_namespace(type
, start_object
, max_depth
,
608 descending_callback
, ascending_callback
,
609 context
, return_value
);
612 (void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE
);
615 (void)acpi_ut_release_read_lock(&acpi_gbl_namespace_rw_lock
);
616 return_ACPI_STATUS(status
);
619 ACPI_EXPORT_SYMBOL(acpi_walk_namespace
)
621 /*******************************************************************************
623 * FUNCTION: acpi_ns_get_device_callback
625 * PARAMETERS: Callback from acpi_get_device
629 * DESCRIPTION: Takes callbacks from walk_namespace and filters out all non-
630 * present devices, or if they specified a HID, it filters based
633 ******************************************************************************/
635 acpi_ns_get_device_callback(acpi_handle obj_handle
,
637 void *context
, void **return_value
)
639 struct acpi_get_devices_info
*info
= context
;
641 struct acpi_namespace_node
*node
;
643 struct acpi_pnp_device_id
*hid
;
644 struct acpi_pnp_device_id_list
*cid
;
649 status
= acpi_ut_acquire_mutex(ACPI_MTX_NAMESPACE
);
650 if (ACPI_FAILURE(status
)) {
654 node
= acpi_ns_validate_handle(obj_handle
);
655 status
= acpi_ut_release_mutex(ACPI_MTX_NAMESPACE
);
656 if (ACPI_FAILURE(status
)) {
661 return (AE_BAD_PARAMETER
);
665 * First, filter based on the device HID and CID.
667 * 01/2010: For this case where a specific HID is requested, we don't
668 * want to run _STA until we have an actual HID match. Thus, we will
669 * not unnecessarily execute _STA on devices for which the caller
670 * doesn't care about. Previously, _STA was executed unconditionally
671 * on all devices found here.
673 * A side-effect of this change is that now we will continue to search
674 * for a matching HID even under device trees where the parent device
675 * would have returned a _STA that indicates it is not present or
676 * not functioning (thus aborting the search on that branch).
678 if (info
->hid
!= NULL
) {
679 status
= acpi_ut_execute_HID(node
, &hid
);
680 if (status
== AE_NOT_FOUND
) {
682 } else if (ACPI_FAILURE(status
)) {
683 return (AE_CTRL_DEPTH
);
686 no_match
= strcmp(hid
->string
, info
->hid
);
691 * HID does not match, attempt match within the
692 * list of Compatible IDs (CIDs)
694 status
= acpi_ut_execute_CID(node
, &cid
);
695 if (status
== AE_NOT_FOUND
) {
697 } else if (ACPI_FAILURE(status
)) {
698 return (AE_CTRL_DEPTH
);
701 /* Walk the CID list */
704 for (i
= 0; i
< cid
->count
; i
++) {
705 if (strcmp(cid
->ids
[i
].string
, info
->hid
) == 0) {
707 /* Found a matching CID */
721 /* Run _STA to determine if device is present */
723 status
= acpi_ut_execute_STA(node
, &flags
);
724 if (ACPI_FAILURE(status
)) {
725 return (AE_CTRL_DEPTH
);
728 if (!(flags
& ACPI_STA_DEVICE_PRESENT
) &&
729 !(flags
& ACPI_STA_DEVICE_FUNCTIONING
)) {
731 * Don't examine the children of the device only when the
732 * device is neither present nor functional. See ACPI spec,
733 * description of _STA for more information.
735 return (AE_CTRL_DEPTH
);
738 /* We have a valid device, invoke the user function */
740 status
= info
->user_function(obj_handle
, nesting_level
,
741 info
->context
, return_value
);
745 /*******************************************************************************
747 * FUNCTION: acpi_get_devices
749 * PARAMETERS: HID - HID to search for. Can be NULL.
750 * user_function - Called when a matching object is found
751 * context - Passed to user function
752 * return_value - Location where return value of
753 * user_function is put if terminated early
755 * RETURNS Return value from the user_function if terminated early.
756 * Otherwise, returns NULL.
758 * DESCRIPTION: Performs a modified depth-first walk of the namespace tree,
759 * starting (and ending) at the object specified by start_handle.
760 * The user_function is called whenever an object of type
761 * Device is found. If the user function returns
762 * a non-zero value, the search is terminated immediately and this
763 * value is returned to the caller.
765 * This is a wrapper for walk_namespace, but the callback performs
766 * additional filtering. Please see acpi_ns_get_device_callback.
768 ******************************************************************************/
771 acpi_get_devices(const char *HID
,
772 acpi_walk_callback user_function
,
773 void *context
, void **return_value
)
776 struct acpi_get_devices_info info
;
778 ACPI_FUNCTION_TRACE(acpi_get_devices
);
780 /* Parameter validation */
782 if (!user_function
) {
783 return_ACPI_STATUS(AE_BAD_PARAMETER
);
787 * We're going to call their callback from OUR callback, so we need
788 * to know what it is, and their context parameter.
791 info
.context
= context
;
792 info
.user_function
= user_function
;
795 * Lock the namespace around the walk.
796 * The namespace will be unlocked/locked around each call
797 * to the user function - since this function
798 * must be allowed to make Acpi calls itself.
800 status
= acpi_ut_acquire_mutex(ACPI_MTX_NAMESPACE
);
801 if (ACPI_FAILURE(status
)) {
802 return_ACPI_STATUS(status
);
805 status
= acpi_ns_walk_namespace(ACPI_TYPE_DEVICE
, ACPI_ROOT_OBJECT
,
806 ACPI_UINT32_MAX
, ACPI_NS_WALK_UNLOCK
,
807 acpi_ns_get_device_callback
, NULL
,
808 &info
, return_value
);
810 (void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE
);
811 return_ACPI_STATUS(status
);
814 ACPI_EXPORT_SYMBOL(acpi_get_devices
)
816 /*******************************************************************************
818 * FUNCTION: acpi_attach_data
820 * PARAMETERS: obj_handle - Namespace node
821 * handler - Handler for this attachment
822 * data - Pointer to data to be attached
826 * DESCRIPTION: Attach arbitrary data and handler to a namespace node.
828 ******************************************************************************/
830 acpi_attach_data(acpi_handle obj_handle
,
831 acpi_object_handler handler
, void *data
)
833 struct acpi_namespace_node
*node
;
836 /* Parameter validation */
838 if (!obj_handle
|| !handler
|| !data
) {
839 return (AE_BAD_PARAMETER
);
842 status
= acpi_ut_acquire_mutex(ACPI_MTX_NAMESPACE
);
843 if (ACPI_FAILURE(status
)) {
847 /* Convert and validate the handle */
849 node
= acpi_ns_validate_handle(obj_handle
);
851 status
= AE_BAD_PARAMETER
;
852 goto unlock_and_exit
;
855 status
= acpi_ns_attach_data(node
, handler
, data
);
858 (void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE
);
862 ACPI_EXPORT_SYMBOL(acpi_attach_data
)
864 /*******************************************************************************
866 * FUNCTION: acpi_detach_data
868 * PARAMETERS: obj_handle - Namespace node handle
869 * handler - Handler used in call to acpi_attach_data
873 * DESCRIPTION: Remove data that was previously attached to a node.
875 ******************************************************************************/
877 acpi_detach_data(acpi_handle obj_handle
, acpi_object_handler handler
)
879 struct acpi_namespace_node
*node
;
882 /* Parameter validation */
884 if (!obj_handle
|| !handler
) {
885 return (AE_BAD_PARAMETER
);
888 status
= acpi_ut_acquire_mutex(ACPI_MTX_NAMESPACE
);
889 if (ACPI_FAILURE(status
)) {
893 /* Convert and validate the handle */
895 node
= acpi_ns_validate_handle(obj_handle
);
897 status
= AE_BAD_PARAMETER
;
898 goto unlock_and_exit
;
901 status
= acpi_ns_detach_data(node
, handler
);
904 (void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE
);
908 ACPI_EXPORT_SYMBOL(acpi_detach_data
)
910 /*******************************************************************************
912 * FUNCTION: acpi_get_data_full
914 * PARAMETERS: obj_handle - Namespace node
915 * handler - Handler used in call to attach_data
916 * data - Where the data is returned
917 * callback - function to execute before returning
921 * DESCRIPTION: Retrieve data that was previously attached to a namespace node
922 * and execute a callback before returning.
924 ******************************************************************************/
926 acpi_get_data_full(acpi_handle obj_handle
, acpi_object_handler handler
,
927 void **data
, void (*callback
)(void *))
929 struct acpi_namespace_node
*node
;
932 /* Parameter validation */
934 if (!obj_handle
|| !handler
|| !data
) {
935 return (AE_BAD_PARAMETER
);
938 status
= acpi_ut_acquire_mutex(ACPI_MTX_NAMESPACE
);
939 if (ACPI_FAILURE(status
)) {
943 /* Convert and validate the handle */
945 node
= acpi_ns_validate_handle(obj_handle
);
947 status
= AE_BAD_PARAMETER
;
948 goto unlock_and_exit
;
951 status
= acpi_ns_get_attached_data(node
, handler
, data
);
952 if (ACPI_SUCCESS(status
) && callback
) {
957 (void)acpi_ut_release_mutex(ACPI_MTX_NAMESPACE
);
961 ACPI_EXPORT_SYMBOL(acpi_get_data_full
)
963 /*******************************************************************************
965 * FUNCTION: acpi_get_data
967 * PARAMETERS: obj_handle - Namespace node
968 * handler - Handler used in call to attach_data
969 * data - Where the data is returned
973 * DESCRIPTION: Retrieve data that was previously attached to a namespace node.
975 ******************************************************************************/
977 acpi_get_data(acpi_handle obj_handle
, acpi_object_handler handler
, void **data
)
979 return acpi_get_data_full(obj_handle
, handler
, data
, NULL
);
982 ACPI_EXPORT_SYMBOL(acpi_get_data
)