forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			162 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			162 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===-- UnwindAssemblyInstEmulation.h ---------------------------*- C++ -*-===//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is distributed under the University of Illinois Open Source
 | |
| // License. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #ifndef liblldb_UnwindAssemblyInstEmulation_h_
 | |
| #define liblldb_UnwindAssemblyInstEmulation_h_
 | |
| 
 | |
| // C Includes
 | |
| // C++ Includes
 | |
| // Other libraries and framework includes
 | |
| // Project includes
 | |
| #include "lldb/Core/EmulateInstruction.h"
 | |
| #include "lldb/Core/RegisterValue.h"
 | |
| #include "lldb/Symbol/UnwindPlan.h"
 | |
| #include "lldb/Target/UnwindAssembly.h"
 | |
| #include "lldb/lldb-private.h"
 | |
| 
 | |
| class UnwindAssemblyInstEmulation : public lldb_private::UnwindAssembly {
 | |
| public:
 | |
|   ~UnwindAssemblyInstEmulation() override = default;
 | |
| 
 | |
|   bool GetNonCallSiteUnwindPlanFromAssembly(
 | |
|       lldb_private::AddressRange &func, lldb_private::Thread &thread,
 | |
|       lldb_private::UnwindPlan &unwind_plan) override;
 | |
| 
 | |
|   bool
 | |
|   GetNonCallSiteUnwindPlanFromAssembly(lldb_private::AddressRange &func,
 | |
|                                        uint8_t *opcode_data, size_t opcode_size,
 | |
|                                        lldb_private::UnwindPlan &unwind_plan);
 | |
| 
 | |
|   bool
 | |
|   AugmentUnwindPlanFromCallSite(lldb_private::AddressRange &func,
 | |
|                                 lldb_private::Thread &thread,
 | |
|                                 lldb_private::UnwindPlan &unwind_plan) override;
 | |
| 
 | |
|   bool GetFastUnwindPlan(lldb_private::AddressRange &func,
 | |
|                          lldb_private::Thread &thread,
 | |
|                          lldb_private::UnwindPlan &unwind_plan) override;
 | |
| 
 | |
|   // thread may be NULL in which case we only use the Target (e.g. if this is
 | |
|   // called pre-process-launch).
 | |
|   bool
 | |
|   FirstNonPrologueInsn(lldb_private::AddressRange &func,
 | |
|                        const lldb_private::ExecutionContext &exe_ctx,
 | |
|                        lldb_private::Address &first_non_prologue_insn) override;
 | |
| 
 | |
|   static lldb_private::UnwindAssembly *
 | |
|   CreateInstance(const lldb_private::ArchSpec &arch);
 | |
| 
 | |
|   //------------------------------------------------------------------
 | |
|   // PluginInterface protocol
 | |
|   //------------------------------------------------------------------
 | |
|   static void Initialize();
 | |
| 
 | |
|   static void Terminate();
 | |
| 
 | |
|   static lldb_private::ConstString GetPluginNameStatic();
 | |
| 
 | |
|   static const char *GetPluginDescriptionStatic();
 | |
| 
 | |
|   lldb_private::ConstString GetPluginName() override;
 | |
| 
 | |
|   uint32_t GetPluginVersion() override;
 | |
| 
 | |
| private:
 | |
|   // Call CreateInstance to get an instance of this class
 | |
|   UnwindAssemblyInstEmulation(const lldb_private::ArchSpec &arch,
 | |
|                               lldb_private::EmulateInstruction *inst_emulator)
 | |
|       : UnwindAssembly(arch), m_inst_emulator_ap(inst_emulator),
 | |
|         m_range_ptr(NULL), m_unwind_plan_ptr(NULL), m_curr_row(),
 | |
|         m_cfa_reg_info(), m_fp_is_cfa(false), m_register_values(),
 | |
|         m_pushed_regs(), m_curr_row_modified(false),
 | |
|         m_forward_branch_offset(0) {
 | |
|     if (m_inst_emulator_ap.get()) {
 | |
|       m_inst_emulator_ap->SetBaton(this);
 | |
|       m_inst_emulator_ap->SetCallbacks(ReadMemory, WriteMemory, ReadRegister,
 | |
|                                        WriteRegister);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   static size_t
 | |
|   ReadMemory(lldb_private::EmulateInstruction *instruction, void *baton,
 | |
|              const lldb_private::EmulateInstruction::Context &context,
 | |
|              lldb::addr_t addr, void *dst, size_t length);
 | |
| 
 | |
|   static size_t
 | |
|   WriteMemory(lldb_private::EmulateInstruction *instruction, void *baton,
 | |
|               const lldb_private::EmulateInstruction::Context &context,
 | |
|               lldb::addr_t addr, const void *dst, size_t length);
 | |
| 
 | |
|   static bool ReadRegister(lldb_private::EmulateInstruction *instruction,
 | |
|                            void *baton,
 | |
|                            const lldb_private::RegisterInfo *reg_info,
 | |
|                            lldb_private::RegisterValue ®_value);
 | |
| 
 | |
|   static bool
 | |
|   WriteRegister(lldb_private::EmulateInstruction *instruction, void *baton,
 | |
|                 const lldb_private::EmulateInstruction::Context &context,
 | |
|                 const lldb_private::RegisterInfo *reg_info,
 | |
|                 const lldb_private::RegisterValue ®_value);
 | |
| 
 | |
|   //    size_t
 | |
|   //    ReadMemory (lldb_private::EmulateInstruction *instruction,
 | |
|   //                const lldb_private::EmulateInstruction::Context &context,
 | |
|   //                lldb::addr_t addr,
 | |
|   //                void *dst,
 | |
|   //                size_t length);
 | |
| 
 | |
|   size_t WriteMemory(lldb_private::EmulateInstruction *instruction,
 | |
|                      const lldb_private::EmulateInstruction::Context &context,
 | |
|                      lldb::addr_t addr, const void *dst, size_t length);
 | |
| 
 | |
|   bool ReadRegister(lldb_private::EmulateInstruction *instruction,
 | |
|                     const lldb_private::RegisterInfo *reg_info,
 | |
|                     lldb_private::RegisterValue ®_value);
 | |
| 
 | |
|   bool WriteRegister(lldb_private::EmulateInstruction *instruction,
 | |
|                      const lldb_private::EmulateInstruction::Context &context,
 | |
|                      const lldb_private::RegisterInfo *reg_info,
 | |
|                      const lldb_private::RegisterValue ®_value);
 | |
| 
 | |
|   static uint64_t
 | |
|   MakeRegisterKindValuePair(const lldb_private::RegisterInfo ®_info);
 | |
| 
 | |
|   void SetRegisterValue(const lldb_private::RegisterInfo ®_info,
 | |
|                         const lldb_private::RegisterValue ®_value);
 | |
| 
 | |
|   bool GetRegisterValue(const lldb_private::RegisterInfo ®_info,
 | |
|                         lldb_private::RegisterValue ®_value);
 | |
| 
 | |
|   std::unique_ptr<lldb_private::EmulateInstruction> m_inst_emulator_ap;
 | |
|   lldb_private::AddressRange *m_range_ptr;
 | |
|   lldb_private::UnwindPlan *m_unwind_plan_ptr;
 | |
|   lldb_private::UnwindPlan::RowSP m_curr_row;
 | |
|   typedef std::map<uint64_t, uint64_t> PushedRegisterToAddrMap;
 | |
|   uint64_t m_initial_sp;
 | |
|   lldb_private::RegisterInfo m_cfa_reg_info;
 | |
|   bool m_fp_is_cfa;
 | |
|   typedef std::map<uint64_t, lldb_private::RegisterValue> RegisterValueMap;
 | |
|   RegisterValueMap m_register_values;
 | |
|   PushedRegisterToAddrMap m_pushed_regs;
 | |
| 
 | |
|   // While processing the instruction stream, we need to communicate some state
 | |
|   // change
 | |
|   // information up to the higher level loop that makes decisions about how to
 | |
|   // push
 | |
|   // the unwind instructions for the UnwindPlan we're constructing.
 | |
| 
 | |
|   // The instruction we're processing updated the UnwindPlan::Row contents
 | |
|   bool m_curr_row_modified;
 | |
|   // The instruction is branching forward with the given offset. 0 value means
 | |
|   // no branching.
 | |
|   uint32_t m_forward_branch_offset;
 | |
| };
 | |
| 
 | |
| #endif // liblldb_UnwindAssemblyInstEmulation_h_
 |