206 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			206 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
 | |
| /******************************************************************************
 | |
|  *
 | |
|  * Module Name: dsdebug - Parser/Interpreter interface - debugging
 | |
|  *
 | |
|  * Copyright (C) 2000 - 2023, Intel Corp.
 | |
|  *
 | |
|  *****************************************************************************/
 | |
| 
 | |
| #include <acpi/acpi.h>
 | |
| #include "accommon.h"
 | |
| #include "acdispat.h"
 | |
| #include "acnamesp.h"
 | |
| #ifdef ACPI_DISASSEMBLER
 | |
| #include "acdisasm.h"
 | |
| #endif
 | |
| #include "acinterp.h"
 | |
| 
 | |
| #define _COMPONENT          ACPI_DISPATCHER
 | |
| ACPI_MODULE_NAME("dsdebug")
 | |
| 
 | |
| #if defined(ACPI_DEBUG_OUTPUT) || defined(ACPI_DEBUGGER)
 | |
| /* Local prototypes */
 | |
| static void
 | |
| acpi_ds_print_node_pathname(struct acpi_namespace_node *node,
 | |
| 			    const char *message);
 | |
| 
 | |
| /*******************************************************************************
 | |
|  *
 | |
|  * FUNCTION:    acpi_ds_print_node_pathname
 | |
|  *
 | |
|  * PARAMETERS:  node            - Object
 | |
|  *              message         - Prefix message
 | |
|  *
 | |
|  * DESCRIPTION: Print an object's full namespace pathname
 | |
|  *              Manages allocation/freeing of a pathname buffer
 | |
|  *
 | |
|  ******************************************************************************/
 | |
| 
 | |
| static void
 | |
| acpi_ds_print_node_pathname(struct acpi_namespace_node *node,
 | |
| 			    const char *message)
 | |
| {
 | |
| 	struct acpi_buffer buffer;
 | |
| 	acpi_status status;
 | |
| 
 | |
| 	ACPI_FUNCTION_TRACE(ds_print_node_pathname);
 | |
| 
 | |
| 	if (!node) {
 | |
| 		ACPI_DEBUG_PRINT_RAW((ACPI_DB_DISPATCH, "[NULL NAME]"));
 | |
| 		return_VOID;
 | |
| 	}
 | |
| 
 | |
| 	/* Convert handle to full pathname and print it (with supplied message) */
 | |
| 
 | |
| 	buffer.length = ACPI_ALLOCATE_LOCAL_BUFFER;
 | |
| 
 | |
| 	status = acpi_ns_handle_to_pathname(node, &buffer, TRUE);
 | |
| 	if (ACPI_SUCCESS(status)) {
 | |
| 		if (message) {
 | |
| 			ACPI_DEBUG_PRINT_RAW((ACPI_DB_DISPATCH, "%s ",
 | |
| 					      message));
 | |
| 		}
 | |
| 
 | |
| 		ACPI_DEBUG_PRINT_RAW((ACPI_DB_DISPATCH, "[%s] (Node %p)",
 | |
| 				      (char *)buffer.pointer, node));
 | |
| 		ACPI_FREE(buffer.pointer);
 | |
| 	}
 | |
| 
 | |
| 	return_VOID;
 | |
| }
 | |
| 
 | |
| /*******************************************************************************
 | |
|  *
 | |
|  * FUNCTION:    acpi_ds_dump_method_stack
 | |
|  *
 | |
|  * PARAMETERS:  status          - Method execution status
 | |
|  *              walk_state      - Current state of the parse tree walk
 | |
|  *              op              - Executing parse op
 | |
|  *
 | |
|  * RETURN:      None
 | |
|  *
 | |
|  * DESCRIPTION: Called when a method has been aborted because of an error.
 | |
|  *              Dumps the method execution stack.
 | |
|  *
 | |
|  ******************************************************************************/
 | |
| 
 | |
| void
 | |
| acpi_ds_dump_method_stack(acpi_status status,
 | |
| 			  struct acpi_walk_state *walk_state,
 | |
| 			  union acpi_parse_object *op)
 | |
| {
 | |
| 	union acpi_parse_object *next;
 | |
| 	struct acpi_thread_state *thread;
 | |
| 	struct acpi_walk_state *next_walk_state;
 | |
| 	struct acpi_namespace_node *previous_method = NULL;
 | |
| 	union acpi_operand_object *method_desc;
 | |
| 
 | |
| 	ACPI_FUNCTION_TRACE(ds_dump_method_stack);
 | |
| 
 | |
| 	/* Ignore control codes, they are not errors */
 | |
| 
 | |
| 	if (ACPI_CNTL_EXCEPTION(status)) {
 | |
| 		return_VOID;
 | |
| 	}
 | |
| 
 | |
| 	/* We may be executing a deferred opcode */
 | |
| 
 | |
| 	if (walk_state->deferred_node) {
 | |
| 		ACPI_DEBUG_PRINT((ACPI_DB_DISPATCH,
 | |
| 				  "Executing subtree for Buffer/Package/Region\n"));
 | |
| 		return_VOID;
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 * If there is no Thread, we are not actually executing a method.
 | |
| 	 * This can happen when the iASL compiler calls the interpreter
 | |
| 	 * to perform constant folding.
 | |
| 	 */
 | |
| 	thread = walk_state->thread;
 | |
| 	if (!thread) {
 | |
| 		return_VOID;
 | |
| 	}
 | |
| 
 | |
| 	/* Display exception and method name */
 | |
| 
 | |
| 	ACPI_DEBUG_PRINT((ACPI_DB_DISPATCH,
 | |
| 			  "\n**** Exception %s during execution of method ",
 | |
| 			  acpi_format_exception(status)));
 | |
| 
 | |
| 	acpi_ds_print_node_pathname(walk_state->method_node, NULL);
 | |
| 
 | |
| 	/* Display stack of executing methods */
 | |
| 
 | |
| 	ACPI_DEBUG_PRINT_RAW((ACPI_DB_DISPATCH,
 | |
| 			      "\n\nMethod Execution Stack:\n"));
 | |
| 	next_walk_state = thread->walk_state_list;
 | |
| 
 | |
| 	/* Walk list of linked walk states */
 | |
| 
 | |
| 	while (next_walk_state) {
 | |
| 		method_desc = next_walk_state->method_desc;
 | |
| 		if (method_desc) {
 | |
| 			acpi_ex_stop_trace_method((struct acpi_namespace_node *)
 | |
| 						  method_desc->method.node,
 | |
| 						  method_desc, walk_state);
 | |
| 		}
 | |
| 
 | |
| 		ACPI_DEBUG_PRINT((ACPI_DB_DISPATCH,
 | |
| 				  "    Method [%4.4s] executing: ",
 | |
| 				  acpi_ut_get_node_name(next_walk_state->
 | |
| 							method_node)));
 | |
| 
 | |
| 		/* First method is the currently executing method */
 | |
| 
 | |
| 		if (next_walk_state == walk_state) {
 | |
| 			if (op) {
 | |
| 
 | |
| 				/* Display currently executing ASL statement */
 | |
| 
 | |
| 				next = op->common.next;
 | |
| 				op->common.next = NULL;
 | |
| 
 | |
| #ifdef ACPI_DISASSEMBLER
 | |
| 				if (walk_state->method_node !=
 | |
| 				    acpi_gbl_root_node) {
 | |
| 
 | |
| 					/* More verbose if not module-level code */
 | |
| 
 | |
| 					acpi_os_printf("Failed at ");
 | |
| 					acpi_dm_disassemble(next_walk_state, op,
 | |
| 							    ACPI_UINT32_MAX);
 | |
| 				}
 | |
| #endif
 | |
| 				op->common.next = next;
 | |
| 			}
 | |
| 		} else {
 | |
| 			/*
 | |
| 			 * This method has called another method
 | |
| 			 * NOTE: the method call parse subtree is already deleted at
 | |
| 			 * this point, so we cannot disassemble the method invocation.
 | |
| 			 */
 | |
| 			ACPI_DEBUG_PRINT_RAW((ACPI_DB_DISPATCH,
 | |
| 					      "Call to method "));
 | |
| 			acpi_ds_print_node_pathname(previous_method, NULL);
 | |
| 		}
 | |
| 
 | |
| 		previous_method = next_walk_state->method_node;
 | |
| 		next_walk_state = next_walk_state->next;
 | |
| 		ACPI_DEBUG_PRINT_RAW((ACPI_DB_DISPATCH, "\n"));
 | |
| 	}
 | |
| 
 | |
| 	return_VOID;
 | |
| }
 | |
| 
 | |
| #else
 | |
| void
 | |
| acpi_ds_dump_method_stack(acpi_status status,
 | |
| 			  struct acpi_walk_state *walk_state,
 | |
| 			  union acpi_parse_object *op)
 | |
| {
 | |
| 	return;
 | |
| }
 | |
| 
 | |
| #endif
 |