You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			132 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			132 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //===-- ArchitectureArm.cpp -------------------------------------*- C++ -*-===//
 | ||
|  | //
 | ||
|  | //                     The LLVM Compiler Infrastructure
 | ||
|  | //
 | ||
|  | // This file is distributed under the University of Illinois Open Source
 | ||
|  | // License. See LICENSE.TXT for details.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | 
 | ||
|  | #include "Plugins/Architecture/Arm/ArchitectureArm.h"
 | ||
|  | #include "Plugins/Process/Utility/ARMDefines.h"
 | ||
|  | #include "Plugins/Process/Utility/InstructionUtils.h"
 | ||
|  | #include "lldb/Core/PluginManager.h"
 | ||
|  | #include "lldb/Target/RegisterContext.h"
 | ||
|  | #include "lldb/Target/Thread.h"
 | ||
|  | #include "lldb/Utility/ArchSpec.h"
 | ||
|  | 
 | ||
|  | using namespace lldb_private; | ||
|  | using namespace lldb; | ||
|  | 
 | ||
|  | ConstString ArchitectureArm::GetPluginNameStatic() { | ||
|  |   return ConstString("arm"); | ||
|  | } | ||
|  | 
 | ||
|  | void ArchitectureArm::Initialize() { | ||
|  |   PluginManager::RegisterPlugin(GetPluginNameStatic(), | ||
|  |                                 "Arm-specific algorithms", | ||
|  |                                 &ArchitectureArm::Create); | ||
|  | } | ||
|  | 
 | ||
|  | void ArchitectureArm::Terminate() { | ||
|  |   PluginManager::UnregisterPlugin(&ArchitectureArm::Create); | ||
|  | } | ||
|  | 
 | ||
|  | std::unique_ptr<Architecture> ArchitectureArm::Create(const ArchSpec &arch) { | ||
|  |   if (arch.GetMachine() != llvm::Triple::arm) | ||
|  |     return nullptr; | ||
|  |   return std::unique_ptr<Architecture>(new ArchitectureArm()); | ||
|  | } | ||
|  | 
 | ||
|  | ConstString ArchitectureArm::GetPluginName() { return GetPluginNameStatic(); } | ||
|  | uint32_t ArchitectureArm::GetPluginVersion() { return 1; } | ||
|  | 
 | ||
|  | void ArchitectureArm::OverrideStopInfo(Thread &thread) { | ||
|  |   // We need to check if we are stopped in Thumb mode in a IT instruction
 | ||
|  |   // and detect if the condition doesn't pass. If this is the case it means
 | ||
|  |   // we won't actually execute this instruction. If this happens we need to
 | ||
|  |   // clear the stop reason to no thread plans think we are stopped for a
 | ||
|  |   // reason and the plans should keep going.
 | ||
|  |   //
 | ||
|  |   // We do this because when single stepping many ARM processes, debuggers
 | ||
|  |   // often use the BVR/BCR registers that says "stop when the PC is not
 | ||
|  |   // equal to its current value". This method of stepping means we can end
 | ||
|  |   // up stopping on instructions inside an if/then block that wouldn't get
 | ||
|  |   // executed. By fixing this we can stop the debugger from seeming like
 | ||
|  |   // you stepped through both the "if" _and_ the "else" clause when source
 | ||
|  |   // level stepping because the debugger stops regardless due to the BVR/BCR
 | ||
|  |   // triggering a stop.
 | ||
|  |   //
 | ||
|  |   // It also means we can set breakpoints on instructions inside an an
 | ||
|  |   // if/then block and correctly skip them if we use the BKPT instruction.
 | ||
|  |   // The ARM and Thumb BKPT instructions are unconditional even when executed
 | ||
|  |   // in a Thumb IT block.
 | ||
|  |   //
 | ||
|  |   // If your debugger inserts software traps in ARM/Thumb code, it will
 | ||
|  |   // need to use 16 and 32 bit instruction for 16 and 32 bit thumb
 | ||
|  |   // instructions respectively. If your debugger inserts a 16 bit thumb
 | ||
|  |   // trap on top of a 32 bit thumb instruction for an opcode that is inside
 | ||
|  |   // an if/then, it will change the it/then to conditionally execute your
 | ||
|  |   // 16 bit trap and then cause your program to crash if it executes the
 | ||
|  |   // trailing 16 bits (the second half of the 32 bit thumb instruction you
 | ||
|  |   // partially overwrote).
 | ||
|  | 
 | ||
|  |   RegisterContextSP reg_ctx_sp(thread.GetRegisterContext()); | ||
|  |   if (!reg_ctx_sp) | ||
|  |     return; | ||
|  | 
 | ||
|  |   const uint32_t cpsr = reg_ctx_sp->GetFlags(0); | ||
|  |   if (cpsr == 0) | ||
|  |     return; | ||
|  | 
 | ||
|  |   // Read the J and T bits to get the ISETSTATE
 | ||
|  |   const uint32_t J = Bit32(cpsr, 24); | ||
|  |   const uint32_t T = Bit32(cpsr, 5); | ||
|  |   const uint32_t ISETSTATE = J << 1 | T; | ||
|  |   if (ISETSTATE == 0) { | ||
|  | // NOTE: I am pretty sure we want to enable the code below
 | ||
|  | // that detects when we stop on an instruction in ARM mode
 | ||
|  | // that is conditional and the condition doesn't pass. This
 | ||
|  | // can happen if you set a breakpoint on an instruction that
 | ||
|  | // is conditional. We currently will _always_ stop on the
 | ||
|  | // instruction which is bad. You can also run into this while
 | ||
|  | // single stepping and you could appear to run code in the "if"
 | ||
|  | // and in the "else" clause because it would stop at all of the
 | ||
|  | // conditional instructions in both.
 | ||
|  | // In such cases, we really don't want to stop at this location.
 | ||
|  | // I will check with the lldb-dev list first before I enable this.
 | ||
|  | #if 0
 | ||
|  |     // ARM mode: check for condition on intsruction
 | ||
|  |     const addr_t pc = reg_ctx_sp->GetPC(); | ||
|  |     Status error; | ||
|  |     // If we fail to read the opcode we will get UINT64_MAX as the
 | ||
|  |     // result in "opcode" which we can use to detect if we read a
 | ||
|  |     // valid opcode.
 | ||
|  |     const uint64_t opcode = thread.GetProcess()->ReadUnsignedIntegerFromMemory(pc, 4, UINT64_MAX, error); | ||
|  |     if (opcode <= UINT32_MAX) | ||
|  |     { | ||
|  |         const uint32_t condition = Bits32((uint32_t)opcode, 31, 28); | ||
|  |         if (!ARMConditionPassed(condition, cpsr)) | ||
|  |         { | ||
|  |             // We ARE stopped on an ARM instruction whose condition doesn't
 | ||
|  |             // pass so this instruction won't get executed.
 | ||
|  |             // Regardless of why it stopped, we need to clear the stop info
 | ||
|  |             thread.SetStopInfo (StopInfoSP()); | ||
|  |         } | ||
|  |     } | ||
|  | #endif
 | ||
|  |   } else if (ISETSTATE == 1) { | ||
|  |     // Thumb mode
 | ||
|  |     const uint32_t ITSTATE = Bits32(cpsr, 15, 10) << 2 | Bits32(cpsr, 26, 25); | ||
|  |     if (ITSTATE != 0) { | ||
|  |       const uint32_t condition = Bits32(ITSTATE, 7, 4); | ||
|  |       if (!ARMConditionPassed(condition, cpsr)) { | ||
|  |         // We ARE stopped in a Thumb IT instruction on an instruction whose
 | ||
|  |         // condition doesn't pass so this instruction won't get executed.
 | ||
|  |         // Regardless of why it stopped, we need to clear the stop info
 | ||
|  |         thread.SetStopInfo(StopInfoSP()); | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | } |