1 //===-- DynamicLoaderHexagonDYLD.h ------------------------------*- C++ -*-===// 2 // 3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4 // See https://llvm.org/LICENSE.txt for license information. 5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6 // 7 //===----------------------------------------------------------------------===// 8 9 #ifndef LLDB_SOURCE_PLUGINS_DYNAMICLOADER_HEXAGON_DYLD_DYNAMICLOADERHEXAGONDYLD_H 10 #define LLDB_SOURCE_PLUGINS_DYNAMICLOADER_HEXAGON_DYLD_DYNAMICLOADERHEXAGONDYLD_H 11 12 #include "lldb/Breakpoint/StoppointCallbackContext.h" 13 #include "lldb/Target/DynamicLoader.h" 14 15 #include "HexagonDYLDRendezvous.h" 16 17 class DynamicLoaderHexagonDYLD : public lldb_private::DynamicLoader { 18 public: 19 DynamicLoaderHexagonDYLD(lldb_private::Process *process); 20 21 ~DynamicLoaderHexagonDYLD() override; 22 23 static void Initialize(); 24 25 static void Terminate(); 26 27 static llvm::StringRef GetPluginNameStatic() { return "hexagon-dyld"; } 28 29 static llvm::StringRef GetPluginDescriptionStatic(); 30 31 static lldb_private::DynamicLoader * 32 CreateInstance(lldb_private::Process *process, bool force); 33 34 // DynamicLoader protocol 35 36 void DidAttach() override; 37 38 void DidLaunch() override; 39 40 lldb::ThreadPlanSP GetStepThroughTrampolinePlan(lldb_private::Thread &thread, 41 bool stop_others) override; 42 43 lldb_private::Status CanLoadImage() override; 44 45 lldb::addr_t GetThreadLocalData(const lldb::ModuleSP module, 46 const lldb::ThreadSP thread, 47 lldb::addr_t tls_file_addr) override; 48 49 // PluginInterface protocol 50 llvm::StringRef GetPluginName() override { return GetPluginNameStatic(); } 51 52 protected: 53 /// Runtime linker rendezvous structure. 54 HexagonDYLDRendezvous m_rendezvous; 55 56 /// Virtual load address of the inferior process. 57 lldb::addr_t m_load_offset; 58 59 /// Virtual entry address of the inferior process. 60 lldb::addr_t m_entry_point; 61 62 /// Rendezvous breakpoint. 63 lldb::break_id_t m_dyld_bid; 64 65 /// Loaded module list. (link map for each module) 66 std::map<lldb::ModuleWP, lldb::addr_t, std::owner_less<lldb::ModuleWP>> 67 m_loaded_modules; 68 69 /// Enables a breakpoint on a function called by the runtime 70 /// linker each time a module is loaded or unloaded. 71 bool SetRendezvousBreakpoint(); 72 73 /// Callback routine which updates the current list of loaded modules based 74 /// on the information supplied by the runtime linker. 75 static bool RendezvousBreakpointHit( 76 void *baton, lldb_private::StoppointCallbackContext *context, 77 lldb::user_id_t break_id, lldb::user_id_t break_loc_id); 78 79 /// Helper method for RendezvousBreakpointHit. Updates LLDB's current set 80 /// of loaded modules. 81 void RefreshModules(); 82 83 /// Updates the load address of every allocatable section in \p module. 84 /// 85 /// \param module The module to traverse. 86 /// 87 /// \param link_map_addr The virtual address of the link map for the @p 88 /// module. 89 /// 90 /// \param base_addr The virtual base address \p module is loaded at. 91 void UpdateLoadedSections(lldb::ModuleSP module, lldb::addr_t link_map_addr, 92 lldb::addr_t base_addr, 93 bool base_addr_is_offset) override; 94 95 /// Removes the loaded sections from the target in \p module. 96 /// 97 /// \param module The module to traverse. 98 void UnloadSections(const lldb::ModuleSP module) override; 99 100 /// Callback routine invoked when we hit the breakpoint on process entry. 101 /// 102 /// This routine is responsible for resolving the load addresses of all 103 /// dependent modules required by the inferior and setting up the rendezvous 104 /// breakpoint. 105 static bool 106 EntryBreakpointHit(void *baton, 107 lldb_private::StoppointCallbackContext *context, 108 lldb::user_id_t break_id, lldb::user_id_t break_loc_id); 109 110 /// Helper for the entry breakpoint callback. Resolves the load addresses 111 /// of all dependent modules. 112 void LoadAllCurrentModules(); 113 114 /// Computes a value for m_load_offset returning the computed address on 115 /// success and LLDB_INVALID_ADDRESS on failure. 116 lldb::addr_t ComputeLoadOffset(); 117 118 /// Computes a value for m_entry_point returning the computed address on 119 /// success and LLDB_INVALID_ADDRESS on failure. 120 lldb::addr_t GetEntryPoint(); 121 122 /// Checks to see if the target module has changed, updates the target 123 /// accordingly and returns the target executable module. 124 lldb::ModuleSP GetTargetExecutable(); 125 126 /// return the address of the Rendezvous breakpoint 127 lldb::addr_t FindRendezvousBreakpointAddress(); 128 129 private: 130 const lldb_private::SectionList * 131 GetSectionListFromModule(const lldb::ModuleSP module) const; 132 }; 133 134 #endif // LLDB_SOURCE_PLUGINS_DYNAMICLOADER_HEXAGON_DYLD_DYNAMICLOADERHEXAGONDYLD_H 135