1*385cc6b4SJerry Jelinek /****************************************************************************** 2*385cc6b4SJerry Jelinek * 3*385cc6b4SJerry Jelinek * Module Name: dsdebug - Parser/Interpreter interface - debugging 4*385cc6b4SJerry Jelinek * 5*385cc6b4SJerry Jelinek *****************************************************************************/ 6*385cc6b4SJerry Jelinek 7*385cc6b4SJerry Jelinek /* 8*385cc6b4SJerry Jelinek * Copyright (C) 2000 - 2016, Intel Corp. 9*385cc6b4SJerry Jelinek * All rights reserved. 10*385cc6b4SJerry Jelinek * 11*385cc6b4SJerry Jelinek * Redistribution and use in source and binary forms, with or without 12*385cc6b4SJerry Jelinek * modification, are permitted provided that the following conditions 13*385cc6b4SJerry Jelinek * are met: 14*385cc6b4SJerry Jelinek * 1. Redistributions of source code must retain the above copyright 15*385cc6b4SJerry Jelinek * notice, this list of conditions, and the following disclaimer, 16*385cc6b4SJerry Jelinek * without modification. 17*385cc6b4SJerry Jelinek * 2. Redistributions in binary form must reproduce at minimum a disclaimer 18*385cc6b4SJerry Jelinek * substantially similar to the "NO WARRANTY" disclaimer below 19*385cc6b4SJerry Jelinek * ("Disclaimer") and any redistribution must be conditioned upon 20*385cc6b4SJerry Jelinek * including a substantially similar Disclaimer requirement for further 21*385cc6b4SJerry Jelinek * binary redistribution. 22*385cc6b4SJerry Jelinek * 3. Neither the names of the above-listed copyright holders nor the names 23*385cc6b4SJerry Jelinek * of any contributors may be used to endorse or promote products derived 24*385cc6b4SJerry Jelinek * from this software without specific prior written permission. 25*385cc6b4SJerry Jelinek * 26*385cc6b4SJerry Jelinek * Alternatively, this software may be distributed under the terms of the 27*385cc6b4SJerry Jelinek * GNU General Public License ("GPL") version 2 as published by the Free 28*385cc6b4SJerry Jelinek * Software Foundation. 29*385cc6b4SJerry Jelinek * 30*385cc6b4SJerry Jelinek * NO WARRANTY 31*385cc6b4SJerry Jelinek * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 32*385cc6b4SJerry Jelinek * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 33*385cc6b4SJerry Jelinek * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR 34*385cc6b4SJerry Jelinek * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 35*385cc6b4SJerry Jelinek * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 36*385cc6b4SJerry Jelinek * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 37*385cc6b4SJerry Jelinek * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 38*385cc6b4SJerry Jelinek * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, 39*385cc6b4SJerry Jelinek * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING 40*385cc6b4SJerry Jelinek * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 41*385cc6b4SJerry Jelinek * POSSIBILITY OF SUCH DAMAGES. 42*385cc6b4SJerry Jelinek */ 43*385cc6b4SJerry Jelinek 44*385cc6b4SJerry Jelinek #include "acpi.h" 45*385cc6b4SJerry Jelinek #include "accommon.h" 46*385cc6b4SJerry Jelinek #include "acdispat.h" 47*385cc6b4SJerry Jelinek #include "acnamesp.h" 48*385cc6b4SJerry Jelinek #include "acdisasm.h" 49*385cc6b4SJerry Jelinek #include "acinterp.h" 50*385cc6b4SJerry Jelinek 51*385cc6b4SJerry Jelinek 52*385cc6b4SJerry Jelinek #define _COMPONENT ACPI_DISPATCHER 53*385cc6b4SJerry Jelinek ACPI_MODULE_NAME ("dsdebug") 54*385cc6b4SJerry Jelinek 55*385cc6b4SJerry Jelinek 56*385cc6b4SJerry Jelinek #if defined(ACPI_DEBUG_OUTPUT) || defined(ACPI_DEBUGGER) 57*385cc6b4SJerry Jelinek 58*385cc6b4SJerry Jelinek /* Local prototypes */ 59*385cc6b4SJerry Jelinek 60*385cc6b4SJerry Jelinek static void 61*385cc6b4SJerry Jelinek AcpiDsPrintNodePathname ( 62*385cc6b4SJerry Jelinek ACPI_NAMESPACE_NODE *Node, 63*385cc6b4SJerry Jelinek const char *Message); 64*385cc6b4SJerry Jelinek 65*385cc6b4SJerry Jelinek 66*385cc6b4SJerry Jelinek /******************************************************************************* 67*385cc6b4SJerry Jelinek * 68*385cc6b4SJerry Jelinek * FUNCTION: AcpiDsPrintNodePathname 69*385cc6b4SJerry Jelinek * 70*385cc6b4SJerry Jelinek * PARAMETERS: Node - Object 71*385cc6b4SJerry Jelinek * Message - Prefix message 72*385cc6b4SJerry Jelinek * 73*385cc6b4SJerry Jelinek * DESCRIPTION: Print an object's full namespace pathname 74*385cc6b4SJerry Jelinek * Manages allocation/freeing of a pathname buffer 75*385cc6b4SJerry Jelinek * 76*385cc6b4SJerry Jelinek ******************************************************************************/ 77*385cc6b4SJerry Jelinek 78*385cc6b4SJerry Jelinek static void 79*385cc6b4SJerry Jelinek AcpiDsPrintNodePathname ( 80*385cc6b4SJerry Jelinek ACPI_NAMESPACE_NODE *Node, 81*385cc6b4SJerry Jelinek const char *Message) 82*385cc6b4SJerry Jelinek { 83*385cc6b4SJerry Jelinek ACPI_BUFFER Buffer; 84*385cc6b4SJerry Jelinek ACPI_STATUS Status; 85*385cc6b4SJerry Jelinek 86*385cc6b4SJerry Jelinek 87*385cc6b4SJerry Jelinek ACPI_FUNCTION_TRACE (DsPrintNodePathname); 88*385cc6b4SJerry Jelinek 89*385cc6b4SJerry Jelinek if (!Node) 90*385cc6b4SJerry Jelinek { 91*385cc6b4SJerry Jelinek ACPI_DEBUG_PRINT_RAW ((ACPI_DB_DISPATCH, "[NULL NAME]")); 92*385cc6b4SJerry Jelinek return_VOID; 93*385cc6b4SJerry Jelinek } 94*385cc6b4SJerry Jelinek 95*385cc6b4SJerry Jelinek /* Convert handle to full pathname and print it (with supplied message) */ 96*385cc6b4SJerry Jelinek 97*385cc6b4SJerry Jelinek Buffer.Length = ACPI_ALLOCATE_LOCAL_BUFFER; 98*385cc6b4SJerry Jelinek 99*385cc6b4SJerry Jelinek Status = AcpiNsHandleToPathname (Node, &Buffer, TRUE); 100*385cc6b4SJerry Jelinek if (ACPI_SUCCESS (Status)) 101*385cc6b4SJerry Jelinek { 102*385cc6b4SJerry Jelinek if (Message) 103*385cc6b4SJerry Jelinek { 104*385cc6b4SJerry Jelinek ACPI_DEBUG_PRINT_RAW ((ACPI_DB_DISPATCH, "%s ", Message)); 105*385cc6b4SJerry Jelinek } 106*385cc6b4SJerry Jelinek 107*385cc6b4SJerry Jelinek ACPI_DEBUG_PRINT_RAW ((ACPI_DB_DISPATCH, "[%s] (Node %p)", 108*385cc6b4SJerry Jelinek (char *) Buffer.Pointer, Node)); 109*385cc6b4SJerry Jelinek ACPI_FREE (Buffer.Pointer); 110*385cc6b4SJerry Jelinek } 111*385cc6b4SJerry Jelinek 112*385cc6b4SJerry Jelinek return_VOID; 113*385cc6b4SJerry Jelinek } 114*385cc6b4SJerry Jelinek 115*385cc6b4SJerry Jelinek 116*385cc6b4SJerry Jelinek /******************************************************************************* 117*385cc6b4SJerry Jelinek * 118*385cc6b4SJerry Jelinek * FUNCTION: AcpiDsDumpMethodStack 119*385cc6b4SJerry Jelinek * 120*385cc6b4SJerry Jelinek * PARAMETERS: Status - Method execution status 121*385cc6b4SJerry Jelinek * WalkState - Current state of the parse tree walk 122*385cc6b4SJerry Jelinek * Op - Executing parse op 123*385cc6b4SJerry Jelinek * 124*385cc6b4SJerry Jelinek * RETURN: None 125*385cc6b4SJerry Jelinek * 126*385cc6b4SJerry Jelinek * DESCRIPTION: Called when a method has been aborted because of an error. 127*385cc6b4SJerry Jelinek * Dumps the method execution stack. 128*385cc6b4SJerry Jelinek * 129*385cc6b4SJerry Jelinek ******************************************************************************/ 130*385cc6b4SJerry Jelinek 131*385cc6b4SJerry Jelinek void 132*385cc6b4SJerry Jelinek AcpiDsDumpMethodStack ( 133*385cc6b4SJerry Jelinek ACPI_STATUS Status, 134*385cc6b4SJerry Jelinek ACPI_WALK_STATE *WalkState, 135*385cc6b4SJerry Jelinek ACPI_PARSE_OBJECT *Op) 136*385cc6b4SJerry Jelinek { 137*385cc6b4SJerry Jelinek ACPI_PARSE_OBJECT *Next; 138*385cc6b4SJerry Jelinek ACPI_THREAD_STATE *Thread; 139*385cc6b4SJerry Jelinek ACPI_WALK_STATE *NextWalkState; 140*385cc6b4SJerry Jelinek ACPI_NAMESPACE_NODE *PreviousMethod = NULL; 141*385cc6b4SJerry Jelinek ACPI_OPERAND_OBJECT *MethodDesc; 142*385cc6b4SJerry Jelinek 143*385cc6b4SJerry Jelinek 144*385cc6b4SJerry Jelinek ACPI_FUNCTION_TRACE (DsDumpMethodStack); 145*385cc6b4SJerry Jelinek 146*385cc6b4SJerry Jelinek /* Ignore control codes, they are not errors */ 147*385cc6b4SJerry Jelinek 148*385cc6b4SJerry Jelinek if ((Status & AE_CODE_MASK) == AE_CODE_CONTROL) 149*385cc6b4SJerry Jelinek { 150*385cc6b4SJerry Jelinek return_VOID; 151*385cc6b4SJerry Jelinek } 152*385cc6b4SJerry Jelinek 153*385cc6b4SJerry Jelinek /* We may be executing a deferred opcode */ 154*385cc6b4SJerry Jelinek 155*385cc6b4SJerry Jelinek if (WalkState->DeferredNode) 156*385cc6b4SJerry Jelinek { 157*385cc6b4SJerry Jelinek ACPI_DEBUG_PRINT ((ACPI_DB_DISPATCH, 158*385cc6b4SJerry Jelinek "Executing subtree for Buffer/Package/Region\n")); 159*385cc6b4SJerry Jelinek return_VOID; 160*385cc6b4SJerry Jelinek } 161*385cc6b4SJerry Jelinek 162*385cc6b4SJerry Jelinek /* 163*385cc6b4SJerry Jelinek * If there is no Thread, we are not actually executing a method. 164*385cc6b4SJerry Jelinek * This can happen when the iASL compiler calls the interpreter 165*385cc6b4SJerry Jelinek * to perform constant folding. 166*385cc6b4SJerry Jelinek */ 167*385cc6b4SJerry Jelinek Thread = WalkState->Thread; 168*385cc6b4SJerry Jelinek if (!Thread) 169*385cc6b4SJerry Jelinek { 170*385cc6b4SJerry Jelinek return_VOID; 171*385cc6b4SJerry Jelinek } 172*385cc6b4SJerry Jelinek 173*385cc6b4SJerry Jelinek /* Display exception and method name */ 174*385cc6b4SJerry Jelinek 175*385cc6b4SJerry Jelinek ACPI_DEBUG_PRINT ((ACPI_DB_DISPATCH, 176*385cc6b4SJerry Jelinek "\n**** Exception %s during execution of method ", 177*385cc6b4SJerry Jelinek AcpiFormatException (Status))); 178*385cc6b4SJerry Jelinek 179*385cc6b4SJerry Jelinek AcpiDsPrintNodePathname (WalkState->MethodNode, NULL); 180*385cc6b4SJerry Jelinek 181*385cc6b4SJerry Jelinek /* Display stack of executing methods */ 182*385cc6b4SJerry Jelinek 183*385cc6b4SJerry Jelinek ACPI_DEBUG_PRINT_RAW ((ACPI_DB_DISPATCH, 184*385cc6b4SJerry Jelinek "\n\nMethod Execution Stack:\n")); 185*385cc6b4SJerry Jelinek NextWalkState = Thread->WalkStateList; 186*385cc6b4SJerry Jelinek 187*385cc6b4SJerry Jelinek /* Walk list of linked walk states */ 188*385cc6b4SJerry Jelinek 189*385cc6b4SJerry Jelinek while (NextWalkState) 190*385cc6b4SJerry Jelinek { 191*385cc6b4SJerry Jelinek MethodDesc = NextWalkState->MethodDesc; 192*385cc6b4SJerry Jelinek if (MethodDesc) 193*385cc6b4SJerry Jelinek { 194*385cc6b4SJerry Jelinek AcpiExStopTraceMethod ( 195*385cc6b4SJerry Jelinek (ACPI_NAMESPACE_NODE *) MethodDesc->Method.Node, 196*385cc6b4SJerry Jelinek MethodDesc, WalkState); 197*385cc6b4SJerry Jelinek } 198*385cc6b4SJerry Jelinek 199*385cc6b4SJerry Jelinek ACPI_DEBUG_PRINT ((ACPI_DB_DISPATCH, 200*385cc6b4SJerry Jelinek " Method [%4.4s] executing: ", 201*385cc6b4SJerry Jelinek AcpiUtGetNodeName (NextWalkState->MethodNode))); 202*385cc6b4SJerry Jelinek 203*385cc6b4SJerry Jelinek /* First method is the currently executing method */ 204*385cc6b4SJerry Jelinek 205*385cc6b4SJerry Jelinek if (NextWalkState == WalkState) 206*385cc6b4SJerry Jelinek { 207*385cc6b4SJerry Jelinek if (Op) 208*385cc6b4SJerry Jelinek { 209*385cc6b4SJerry Jelinek /* Display currently executing ASL statement */ 210*385cc6b4SJerry Jelinek 211*385cc6b4SJerry Jelinek Next = Op->Common.Next; 212*385cc6b4SJerry Jelinek Op->Common.Next = NULL; 213*385cc6b4SJerry Jelinek 214*385cc6b4SJerry Jelinek #ifdef ACPI_DISASSEMBLER 215*385cc6b4SJerry Jelinek AcpiDmDisassemble (NextWalkState, Op, ACPI_UINT32_MAX); 216*385cc6b4SJerry Jelinek #endif 217*385cc6b4SJerry Jelinek Op->Common.Next = Next; 218*385cc6b4SJerry Jelinek } 219*385cc6b4SJerry Jelinek } 220*385cc6b4SJerry Jelinek else 221*385cc6b4SJerry Jelinek { 222*385cc6b4SJerry Jelinek /* 223*385cc6b4SJerry Jelinek * This method has called another method 224*385cc6b4SJerry Jelinek * NOTE: the method call parse subtree is already deleted at 225*385cc6b4SJerry Jelinek * this point, so we cannot disassemble the method invocation. 226*385cc6b4SJerry Jelinek */ 227*385cc6b4SJerry Jelinek ACPI_DEBUG_PRINT_RAW ((ACPI_DB_DISPATCH, "Call to method ")); 228*385cc6b4SJerry Jelinek AcpiDsPrintNodePathname (PreviousMethod, NULL); 229*385cc6b4SJerry Jelinek } 230*385cc6b4SJerry Jelinek 231*385cc6b4SJerry Jelinek PreviousMethod = NextWalkState->MethodNode; 232*385cc6b4SJerry Jelinek NextWalkState = NextWalkState->Next; 233*385cc6b4SJerry Jelinek ACPI_DEBUG_PRINT_RAW ((ACPI_DB_DISPATCH, "\n")); 234*385cc6b4SJerry Jelinek } 235*385cc6b4SJerry Jelinek 236*385cc6b4SJerry Jelinek return_VOID; 237*385cc6b4SJerry Jelinek } 238*385cc6b4SJerry Jelinek 239*385cc6b4SJerry Jelinek #else 240*385cc6b4SJerry Jelinek 241*385cc6b4SJerry Jelinek void 242*385cc6b4SJerry Jelinek AcpiDsDumpMethodStack ( 243*385cc6b4SJerry Jelinek ACPI_STATUS Status, 244*385cc6b4SJerry Jelinek ACPI_WALK_STATE *WalkState, 245*385cc6b4SJerry Jelinek ACPI_PARSE_OBJECT *Op) 246*385cc6b4SJerry Jelinek { 247*385cc6b4SJerry Jelinek return; 248*385cc6b4SJerry Jelinek } 249*385cc6b4SJerry Jelinek 250*385cc6b4SJerry Jelinek #endif 251