130fdc8d8SChris Lattner //===-- Module.cpp ----------------------------------------------*- C++ -*-===// 230fdc8d8SChris Lattner // 330fdc8d8SChris Lattner // The LLVM Compiler Infrastructure 430fdc8d8SChris Lattner // 530fdc8d8SChris Lattner // This file is distributed under the University of Illinois Open Source 630fdc8d8SChris Lattner // License. See LICENSE.TXT for details. 730fdc8d8SChris Lattner // 830fdc8d8SChris Lattner //===----------------------------------------------------------------------===// 930fdc8d8SChris Lattner 1030fdc8d8SChris Lattner #include "lldb/Core/Module.h" 11c9660546SGreg Clayton #include "lldb/Core/DataBuffer.h" 12c9660546SGreg Clayton #include "lldb/Core/DataBufferHeap.h" 1330fdc8d8SChris Lattner #include "lldb/Core/Log.h" 1430fdc8d8SChris Lattner #include "lldb/Core/ModuleList.h" 1530fdc8d8SChris Lattner #include "lldb/Core/RegularExpression.h" 16c982b3d6SGreg Clayton #include "lldb/Core/StreamString.h" 1730fdc8d8SChris Lattner #include "lldb/Core/Timer.h" 18e38a5eddSGreg Clayton #include "lldb/Host/Host.h" 1930fdc8d8SChris Lattner #include "lldb/lldb-private-log.h" 2030fdc8d8SChris Lattner #include "lldb/Symbol/ObjectFile.h" 2130fdc8d8SChris Lattner #include "lldb/Symbol/SymbolContext.h" 2230fdc8d8SChris Lattner #include "lldb/Symbol/SymbolVendor.h" 23c9660546SGreg Clayton #include "lldb/Target/Process.h" 24c9660546SGreg Clayton #include "lldb/Target/Target.h" 2530fdc8d8SChris Lattner 2630fdc8d8SChris Lattner using namespace lldb; 2730fdc8d8SChris Lattner using namespace lldb_private; 2830fdc8d8SChris Lattner 2965a03991SGreg Clayton // Shared pointers to modules track module lifetimes in 3065a03991SGreg Clayton // targets and in the global module, but this collection 3165a03991SGreg Clayton // will track all module objects that are still alive 3265a03991SGreg Clayton typedef std::vector<Module *> ModuleCollection; 3365a03991SGreg Clayton 3465a03991SGreg Clayton static ModuleCollection & 3565a03991SGreg Clayton GetModuleCollection() 3665a03991SGreg Clayton { 37549f7374SJim Ingham // This module collection needs to live past any module, so we could either make it a 38549f7374SJim Ingham // shared pointer in each module or just leak is. Since it is only an empty vector by 39549f7374SJim Ingham // the time all the modules have gone away, we just leak it for now. If we decide this 40549f7374SJim Ingham // is a big problem we can introduce a Finalize method that will tear everything down in 41549f7374SJim Ingham // a predictable order. 42549f7374SJim Ingham 43549f7374SJim Ingham static ModuleCollection *g_module_collection = NULL; 44549f7374SJim Ingham if (g_module_collection == NULL) 45549f7374SJim Ingham g_module_collection = new ModuleCollection(); 46549f7374SJim Ingham 47549f7374SJim Ingham return *g_module_collection; 4865a03991SGreg Clayton } 4965a03991SGreg Clayton 50b26e6bebSGreg Clayton Mutex * 5165a03991SGreg Clayton Module::GetAllocationModuleCollectionMutex() 5265a03991SGreg Clayton { 53b26e6bebSGreg Clayton // NOTE: The mutex below must be leaked since the global module list in 54b26e6bebSGreg Clayton // the ModuleList class will get torn at some point, and we can't know 55b26e6bebSGreg Clayton // if it will tear itself down before the "g_module_collection_mutex" below 56b26e6bebSGreg Clayton // will. So we leak a Mutex object below to safeguard against that 57b26e6bebSGreg Clayton 58b26e6bebSGreg Clayton static Mutex *g_module_collection_mutex = NULL; 59b26e6bebSGreg Clayton if (g_module_collection_mutex == NULL) 60b26e6bebSGreg Clayton g_module_collection_mutex = new Mutex (Mutex::eMutexTypeRecursive); // NOTE: known leak 6165a03991SGreg Clayton return g_module_collection_mutex; 6265a03991SGreg Clayton } 6365a03991SGreg Clayton 6465a03991SGreg Clayton size_t 6565a03991SGreg Clayton Module::GetNumberAllocatedModules () 6665a03991SGreg Clayton { 6765a03991SGreg Clayton Mutex::Locker locker (GetAllocationModuleCollectionMutex()); 6865a03991SGreg Clayton return GetModuleCollection().size(); 6965a03991SGreg Clayton } 7065a03991SGreg Clayton 7165a03991SGreg Clayton Module * 7265a03991SGreg Clayton Module::GetAllocatedModuleAtIndex (size_t idx) 7365a03991SGreg Clayton { 7465a03991SGreg Clayton Mutex::Locker locker (GetAllocationModuleCollectionMutex()); 7565a03991SGreg Clayton ModuleCollection &modules = GetModuleCollection(); 7665a03991SGreg Clayton if (idx < modules.size()) 7765a03991SGreg Clayton return modules[idx]; 7865a03991SGreg Clayton return NULL; 7965a03991SGreg Clayton } 8029ad7b91SGreg Clayton #if 0 8165a03991SGreg Clayton 8229ad7b91SGreg Clayton // These functions help us to determine if modules are still loaded, yet don't require that 8329ad7b91SGreg Clayton // you have a command interpreter and can easily be called from an external debugger. 8429ad7b91SGreg Clayton namespace lldb { 8565a03991SGreg Clayton 8629ad7b91SGreg Clayton void 8729ad7b91SGreg Clayton ClearModuleInfo (void) 8829ad7b91SGreg Clayton { 8929ad7b91SGreg Clayton ModuleList::RemoveOrphanSharedModules(); 9029ad7b91SGreg Clayton } 9129ad7b91SGreg Clayton 9229ad7b91SGreg Clayton void 9329ad7b91SGreg Clayton DumpModuleInfo (void) 9429ad7b91SGreg Clayton { 9529ad7b91SGreg Clayton Mutex::Locker locker (Module::GetAllocationModuleCollectionMutex()); 9629ad7b91SGreg Clayton ModuleCollection &modules = GetModuleCollection(); 9729ad7b91SGreg Clayton const size_t count = modules.size(); 9829ad7b91SGreg Clayton printf ("%s: %zu modules:\n", __PRETTY_FUNCTION__, count); 9929ad7b91SGreg Clayton for (size_t i=0; i<count; ++i) 10029ad7b91SGreg Clayton { 10129ad7b91SGreg Clayton 10229ad7b91SGreg Clayton StreamString strm; 10329ad7b91SGreg Clayton Module *module = modules[i]; 10429ad7b91SGreg Clayton const bool in_shared_module_list = ModuleList::ModuleIsInCache (module); 10529ad7b91SGreg Clayton module->GetDescription(&strm, eDescriptionLevelFull); 10629ad7b91SGreg Clayton printf ("%p: shared = %i, ref_count = %3u, module = %s\n", 10729ad7b91SGreg Clayton module, 10829ad7b91SGreg Clayton in_shared_module_list, 10929ad7b91SGreg Clayton (uint32_t)module->use_count(), 11029ad7b91SGreg Clayton strm.GetString().c_str()); 11129ad7b91SGreg Clayton } 11229ad7b91SGreg Clayton } 11329ad7b91SGreg Clayton } 11429ad7b91SGreg Clayton 11529ad7b91SGreg Clayton #endif 11665a03991SGreg Clayton 11765a03991SGreg Clayton 118c9660546SGreg Clayton Module::Module(const FileSpec& file_spec, const lldb::ProcessSP &process_sp, lldb::addr_t header_addr) : 119c9660546SGreg Clayton m_mutex (Mutex::eMutexTypeRecursive), 120c9660546SGreg Clayton m_mod_time (), 121c9660546SGreg Clayton m_arch (), 122c9660546SGreg Clayton m_uuid (), 123c9660546SGreg Clayton m_file (file_spec), 124c9660546SGreg Clayton m_platform_file(), 125c9660546SGreg Clayton m_object_name (), 126c9660546SGreg Clayton m_object_offset (), 127c9660546SGreg Clayton m_objfile_sp (), 128c9660546SGreg Clayton m_symfile_ap (), 129c9660546SGreg Clayton m_ast (), 130c9660546SGreg Clayton m_did_load_objfile (false), 131c9660546SGreg Clayton m_did_load_symbol_vendor (false), 132c9660546SGreg Clayton m_did_parse_uuid (false), 133c9660546SGreg Clayton m_did_init_ast (false), 134c9660546SGreg Clayton m_is_dynamic_loader_module (false), 135c9660546SGreg Clayton m_was_modified (false) 136c9660546SGreg Clayton { 137c9660546SGreg Clayton // Scope for locker below... 138c9660546SGreg Clayton { 139c9660546SGreg Clayton Mutex::Locker locker (GetAllocationModuleCollectionMutex()); 140c9660546SGreg Clayton GetModuleCollection().push_back(this); 141c9660546SGreg Clayton } 142c9660546SGreg Clayton StreamString s; 143c9660546SGreg Clayton if (m_file.GetFilename()) 144c9660546SGreg Clayton s << m_file.GetFilename(); 145c9660546SGreg Clayton s.Printf("[0x%16.16llx]", header_addr); 146c9660546SGreg Clayton m_file.GetFilename().SetCString (s.GetData()); 147c9660546SGreg Clayton Mutex::Locker locker (m_mutex); 148c9660546SGreg Clayton DataBufferSP data_sp; 149c9660546SGreg Clayton if (process_sp) 150c9660546SGreg Clayton { 151c9660546SGreg Clayton m_did_load_objfile = true; 152c9660546SGreg Clayton std::auto_ptr<DataBufferHeap> data_ap (new DataBufferHeap (512, 0)); 153c9660546SGreg Clayton Error error; 154c9660546SGreg Clayton const size_t bytes_read = process_sp->ReadMemory (header_addr, 155c9660546SGreg Clayton data_ap->GetBytes(), 156c9660546SGreg Clayton data_ap->GetByteSize(), 157c9660546SGreg Clayton error); 158c9660546SGreg Clayton if (bytes_read == 512) 159c9660546SGreg Clayton { 160c9660546SGreg Clayton data_sp.reset (data_ap.release()); 161c9660546SGreg Clayton m_objfile_sp = ObjectFile::FindPlugin(this, process_sp, header_addr, data_sp); 162c9660546SGreg Clayton if (m_objfile_sp) 163c9660546SGreg Clayton { 164c9660546SGreg Clayton // Once we get the object file, update our module with the object file's 165c9660546SGreg Clayton // architecture since it might differ in vendor/os if some parts were 166c9660546SGreg Clayton // unknown. 167c9660546SGreg Clayton m_objfile_sp->GetArchitecture (m_arch); 168c9660546SGreg Clayton } 169c9660546SGreg Clayton } 170c9660546SGreg Clayton } 171c9660546SGreg Clayton } 172c9660546SGreg Clayton 17330fdc8d8SChris Lattner Module::Module(const FileSpec& file_spec, const ArchSpec& arch, const ConstString *object_name, off_t object_offset) : 17430fdc8d8SChris Lattner m_mutex (Mutex::eMutexTypeRecursive), 17530fdc8d8SChris Lattner m_mod_time (file_spec.GetModificationTime()), 17630fdc8d8SChris Lattner m_arch (arch), 17730fdc8d8SChris Lattner m_uuid (), 17830fdc8d8SChris Lattner m_file (file_spec), 17932e0a750SGreg Clayton m_platform_file(), 18030fdc8d8SChris Lattner m_object_name (), 1818b82f087SGreg Clayton m_object_offset (object_offset), 182762f7135SGreg Clayton m_objfile_sp (), 183e83e731eSGreg Clayton m_symfile_ap (), 1846beaaa68SGreg Clayton m_ast (), 185e83e731eSGreg Clayton m_did_load_objfile (false), 186e83e731eSGreg Clayton m_did_load_symbol_vendor (false), 187e83e731eSGreg Clayton m_did_parse_uuid (false), 1886beaaa68SGreg Clayton m_did_init_ast (false), 189e38a5eddSGreg Clayton m_is_dynamic_loader_module (false), 190e38a5eddSGreg Clayton m_was_modified (false) 19130fdc8d8SChris Lattner { 19265a03991SGreg Clayton // Scope for locker below... 19365a03991SGreg Clayton { 19465a03991SGreg Clayton Mutex::Locker locker (GetAllocationModuleCollectionMutex()); 19565a03991SGreg Clayton GetModuleCollection().push_back(this); 19665a03991SGreg Clayton } 19765a03991SGreg Clayton 19830fdc8d8SChris Lattner if (object_name) 19930fdc8d8SChris Lattner m_object_name = *object_name; 2002d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_OBJECT)); 20130fdc8d8SChris Lattner if (log) 20230fdc8d8SChris Lattner log->Printf ("%p Module::Module((%s) '%s/%s%s%s%s')", 20330fdc8d8SChris Lattner this, 20464195a2cSGreg Clayton m_arch.GetArchitectureName(), 20530fdc8d8SChris Lattner m_file.GetDirectory().AsCString(""), 20630fdc8d8SChris Lattner m_file.GetFilename().AsCString(""), 20730fdc8d8SChris Lattner m_object_name.IsEmpty() ? "" : "(", 20830fdc8d8SChris Lattner m_object_name.IsEmpty() ? "" : m_object_name.AsCString(""), 20930fdc8d8SChris Lattner m_object_name.IsEmpty() ? "" : ")"); 21030fdc8d8SChris Lattner } 21130fdc8d8SChris Lattner 21230fdc8d8SChris Lattner Module::~Module() 21330fdc8d8SChris Lattner { 21465a03991SGreg Clayton // Scope for locker below... 21565a03991SGreg Clayton { 21665a03991SGreg Clayton Mutex::Locker locker (GetAllocationModuleCollectionMutex()); 21765a03991SGreg Clayton ModuleCollection &modules = GetModuleCollection(); 21865a03991SGreg Clayton ModuleCollection::iterator end = modules.end(); 21965a03991SGreg Clayton ModuleCollection::iterator pos = std::find(modules.begin(), end, this); 22065a03991SGreg Clayton if (pos != end) 22165a03991SGreg Clayton modules.erase(pos); 22265a03991SGreg Clayton } 2232d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_OBJECT)); 22430fdc8d8SChris Lattner if (log) 22530fdc8d8SChris Lattner log->Printf ("%p Module::~Module((%s) '%s/%s%s%s%s')", 22630fdc8d8SChris Lattner this, 22764195a2cSGreg Clayton m_arch.GetArchitectureName(), 22830fdc8d8SChris Lattner m_file.GetDirectory().AsCString(""), 22930fdc8d8SChris Lattner m_file.GetFilename().AsCString(""), 23030fdc8d8SChris Lattner m_object_name.IsEmpty() ? "" : "(", 23130fdc8d8SChris Lattner m_object_name.IsEmpty() ? "" : m_object_name.AsCString(""), 23230fdc8d8SChris Lattner m_object_name.IsEmpty() ? "" : ")"); 2336beaaa68SGreg Clayton // Release any auto pointers before we start tearing down our member 2346beaaa68SGreg Clayton // variables since the object file and symbol files might need to make 2356beaaa68SGreg Clayton // function calls back into this module object. The ordering is important 2366beaaa68SGreg Clayton // here because symbol files can require the module object file. So we tear 2376beaaa68SGreg Clayton // down the symbol file first, then the object file. 2386beaaa68SGreg Clayton m_symfile_ap.reset(); 239762f7135SGreg Clayton m_objfile_sp.reset(); 24030fdc8d8SChris Lattner } 24130fdc8d8SChris Lattner 24230fdc8d8SChris Lattner 24360830268SGreg Clayton const lldb_private::UUID& 24430fdc8d8SChris Lattner Module::GetUUID() 24530fdc8d8SChris Lattner { 24630fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 247e83e731eSGreg Clayton if (m_did_parse_uuid == false) 24830fdc8d8SChris Lattner { 24930fdc8d8SChris Lattner ObjectFile * obj_file = GetObjectFile (); 25030fdc8d8SChris Lattner 25130fdc8d8SChris Lattner if (obj_file != NULL) 25230fdc8d8SChris Lattner { 25330fdc8d8SChris Lattner obj_file->GetUUID(&m_uuid); 254e83e731eSGreg Clayton m_did_parse_uuid = true; 25530fdc8d8SChris Lattner } 25630fdc8d8SChris Lattner } 25730fdc8d8SChris Lattner return m_uuid; 25830fdc8d8SChris Lattner } 25930fdc8d8SChris Lattner 2606beaaa68SGreg Clayton ClangASTContext & 2616beaaa68SGreg Clayton Module::GetClangASTContext () 2626beaaa68SGreg Clayton { 2636beaaa68SGreg Clayton Mutex::Locker locker (m_mutex); 2646beaaa68SGreg Clayton if (m_did_init_ast == false) 2656beaaa68SGreg Clayton { 2666beaaa68SGreg Clayton ObjectFile * objfile = GetObjectFile(); 267514487e8SGreg Clayton ArchSpec object_arch; 268514487e8SGreg Clayton if (objfile && objfile->GetArchitecture(object_arch)) 2696beaaa68SGreg Clayton { 2706beaaa68SGreg Clayton m_did_init_ast = true; 271514487e8SGreg Clayton m_ast.SetArchitecture (object_arch); 2726beaaa68SGreg Clayton } 2736beaaa68SGreg Clayton } 2746beaaa68SGreg Clayton return m_ast; 2756beaaa68SGreg Clayton } 2766beaaa68SGreg Clayton 27730fdc8d8SChris Lattner void 27830fdc8d8SChris Lattner Module::ParseAllDebugSymbols() 27930fdc8d8SChris Lattner { 28030fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 28130fdc8d8SChris Lattner uint32_t num_comp_units = GetNumCompileUnits(); 28230fdc8d8SChris Lattner if (num_comp_units == 0) 28330fdc8d8SChris Lattner return; 28430fdc8d8SChris Lattner 285a2eee184SGreg Clayton SymbolContext sc; 286e1cd1be6SGreg Clayton sc.module_sp = shared_from_this(); 28730fdc8d8SChris Lattner uint32_t cu_idx; 28830fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 28930fdc8d8SChris Lattner 29030fdc8d8SChris Lattner for (cu_idx = 0; cu_idx < num_comp_units; cu_idx++) 29130fdc8d8SChris Lattner { 29230fdc8d8SChris Lattner sc.comp_unit = symbols->GetCompileUnitAtIndex(cu_idx).get(); 29330fdc8d8SChris Lattner if (sc.comp_unit) 29430fdc8d8SChris Lattner { 29530fdc8d8SChris Lattner sc.function = NULL; 29630fdc8d8SChris Lattner symbols->ParseVariablesForContext(sc); 29730fdc8d8SChris Lattner 29830fdc8d8SChris Lattner symbols->ParseCompileUnitFunctions(sc); 29930fdc8d8SChris Lattner 30030fdc8d8SChris Lattner uint32_t func_idx; 30130fdc8d8SChris Lattner for (func_idx = 0; (sc.function = sc.comp_unit->GetFunctionAtIndex(func_idx).get()) != NULL; ++func_idx) 30230fdc8d8SChris Lattner { 30330fdc8d8SChris Lattner symbols->ParseFunctionBlocks(sc); 30430fdc8d8SChris Lattner 30530fdc8d8SChris Lattner // Parse the variables for this function and all its blocks 30630fdc8d8SChris Lattner symbols->ParseVariablesForContext(sc); 30730fdc8d8SChris Lattner } 30830fdc8d8SChris Lattner 30930fdc8d8SChris Lattner 31030fdc8d8SChris Lattner // Parse all types for this compile unit 31130fdc8d8SChris Lattner sc.function = NULL; 31230fdc8d8SChris Lattner symbols->ParseTypes(sc); 31330fdc8d8SChris Lattner } 31430fdc8d8SChris Lattner } 31530fdc8d8SChris Lattner } 31630fdc8d8SChris Lattner 31730fdc8d8SChris Lattner void 31830fdc8d8SChris Lattner Module::CalculateSymbolContext(SymbolContext* sc) 31930fdc8d8SChris Lattner { 320e1cd1be6SGreg Clayton sc->module_sp = shared_from_this(); 32130fdc8d8SChris Lattner } 32230fdc8d8SChris Lattner 3237e9b1fd0SGreg Clayton Module * 3247e9b1fd0SGreg Clayton Module::CalculateSymbolContextModule () 3257e9b1fd0SGreg Clayton { 3267e9b1fd0SGreg Clayton return this; 3277e9b1fd0SGreg Clayton } 3287e9b1fd0SGreg Clayton 32930fdc8d8SChris Lattner void 33030fdc8d8SChris Lattner Module::DumpSymbolContext(Stream *s) 33130fdc8d8SChris Lattner { 332fd54b368SJason Molenda s->Printf(", Module{%p}", this); 33330fdc8d8SChris Lattner } 33430fdc8d8SChris Lattner 33530fdc8d8SChris Lattner uint32_t 33630fdc8d8SChris Lattner Module::GetNumCompileUnits() 33730fdc8d8SChris Lattner { 33830fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 33930fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, "Module::GetNumCompileUnits (module = %p)", this); 34030fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 34130fdc8d8SChris Lattner if (symbols) 34230fdc8d8SChris Lattner return symbols->GetNumCompileUnits(); 34330fdc8d8SChris Lattner return 0; 34430fdc8d8SChris Lattner } 34530fdc8d8SChris Lattner 34630fdc8d8SChris Lattner CompUnitSP 34730fdc8d8SChris Lattner Module::GetCompileUnitAtIndex (uint32_t index) 34830fdc8d8SChris Lattner { 34930fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 35030fdc8d8SChris Lattner uint32_t num_comp_units = GetNumCompileUnits (); 35130fdc8d8SChris Lattner CompUnitSP cu_sp; 35230fdc8d8SChris Lattner 35330fdc8d8SChris Lattner if (index < num_comp_units) 35430fdc8d8SChris Lattner { 35530fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 35630fdc8d8SChris Lattner if (symbols) 35730fdc8d8SChris Lattner cu_sp = symbols->GetCompileUnitAtIndex(index); 35830fdc8d8SChris Lattner } 35930fdc8d8SChris Lattner return cu_sp; 36030fdc8d8SChris Lattner } 36130fdc8d8SChris Lattner 36230fdc8d8SChris Lattner bool 36330fdc8d8SChris Lattner Module::ResolveFileAddress (lldb::addr_t vm_addr, Address& so_addr) 36430fdc8d8SChris Lattner { 36530fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 36630fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, "Module::ResolveFileAddress (vm_addr = 0x%llx)", vm_addr); 36730fdc8d8SChris Lattner ObjectFile* ofile = GetObjectFile(); 36830fdc8d8SChris Lattner if (ofile) 36930fdc8d8SChris Lattner return so_addr.ResolveAddressUsingFileSections(vm_addr, ofile->GetSectionList()); 37030fdc8d8SChris Lattner return false; 37130fdc8d8SChris Lattner } 37230fdc8d8SChris Lattner 37330fdc8d8SChris Lattner uint32_t 37430fdc8d8SChris Lattner Module::ResolveSymbolContextForAddress (const Address& so_addr, uint32_t resolve_scope, SymbolContext& sc) 37530fdc8d8SChris Lattner { 37630fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 37730fdc8d8SChris Lattner uint32_t resolved_flags = 0; 37830fdc8d8SChris Lattner 37930fdc8d8SChris Lattner // Clear the result symbol context in case we don't find anything 38030fdc8d8SChris Lattner sc.Clear(); 38130fdc8d8SChris Lattner 38230fdc8d8SChris Lattner // Get the section from the section/offset address. 38330fdc8d8SChris Lattner const Section *section = so_addr.GetSection(); 38430fdc8d8SChris Lattner 38530fdc8d8SChris Lattner // Make sure the section matches this module before we try and match anything 38630fdc8d8SChris Lattner if (section && section->GetModule() == this) 38730fdc8d8SChris Lattner { 38830fdc8d8SChris Lattner // If the section offset based address resolved itself, then this 38930fdc8d8SChris Lattner // is the right module. 390e1cd1be6SGreg Clayton sc.module_sp = shared_from_this(); 39130fdc8d8SChris Lattner resolved_flags |= eSymbolContextModule; 39230fdc8d8SChris Lattner 39330fdc8d8SChris Lattner // Resolve the compile unit, function, block, line table or line 39430fdc8d8SChris Lattner // entry if requested. 39530fdc8d8SChris Lattner if (resolve_scope & eSymbolContextCompUnit || 39630fdc8d8SChris Lattner resolve_scope & eSymbolContextFunction || 39730fdc8d8SChris Lattner resolve_scope & eSymbolContextBlock || 39830fdc8d8SChris Lattner resolve_scope & eSymbolContextLineEntry ) 39930fdc8d8SChris Lattner { 40030fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 40130fdc8d8SChris Lattner if (symbols) 40230fdc8d8SChris Lattner resolved_flags |= symbols->ResolveSymbolContext (so_addr, resolve_scope, sc); 40330fdc8d8SChris Lattner } 40430fdc8d8SChris Lattner 405680e1778SJim Ingham // Resolve the symbol if requested, but don't re-look it up if we've already found it. 406680e1778SJim Ingham if (resolve_scope & eSymbolContextSymbol && !(resolved_flags & eSymbolContextSymbol)) 40730fdc8d8SChris Lattner { 40830fdc8d8SChris Lattner ObjectFile* ofile = GetObjectFile(); 40930fdc8d8SChris Lattner if (ofile) 41030fdc8d8SChris Lattner { 41130fdc8d8SChris Lattner Symtab *symtab = ofile->GetSymtab(); 41230fdc8d8SChris Lattner if (symtab) 41330fdc8d8SChris Lattner { 41430fdc8d8SChris Lattner if (so_addr.IsSectionOffset()) 41530fdc8d8SChris Lattner { 41630fdc8d8SChris Lattner sc.symbol = symtab->FindSymbolContainingFileAddress(so_addr.GetFileAddress()); 41730fdc8d8SChris Lattner if (sc.symbol) 41830fdc8d8SChris Lattner resolved_flags |= eSymbolContextSymbol; 41930fdc8d8SChris Lattner } 42030fdc8d8SChris Lattner } 42130fdc8d8SChris Lattner } 42230fdc8d8SChris Lattner } 42330fdc8d8SChris Lattner } 42430fdc8d8SChris Lattner return resolved_flags; 42530fdc8d8SChris Lattner } 42630fdc8d8SChris Lattner 42730fdc8d8SChris Lattner uint32_t 428274060b6SGreg Clayton Module::ResolveSymbolContextForFilePath 429274060b6SGreg Clayton ( 430274060b6SGreg Clayton const char *file_path, 431274060b6SGreg Clayton uint32_t line, 432274060b6SGreg Clayton bool check_inlines, 433274060b6SGreg Clayton uint32_t resolve_scope, 434274060b6SGreg Clayton SymbolContextList& sc_list 435274060b6SGreg Clayton ) 43630fdc8d8SChris Lattner { 437274060b6SGreg Clayton FileSpec file_spec(file_path, false); 43830fdc8d8SChris Lattner return ResolveSymbolContextsForFileSpec (file_spec, line, check_inlines, resolve_scope, sc_list); 43930fdc8d8SChris Lattner } 44030fdc8d8SChris Lattner 44130fdc8d8SChris Lattner uint32_t 44230fdc8d8SChris Lattner Module::ResolveSymbolContextsForFileSpec (const FileSpec &file_spec, uint32_t line, bool check_inlines, uint32_t resolve_scope, SymbolContextList& sc_list) 44330fdc8d8SChris Lattner { 44430fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 44530fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, 44630fdc8d8SChris Lattner "Module::ResolveSymbolContextForFilePath (%s%s%s:%u, check_inlines = %s, resolve_scope = 0x%8.8x)", 44730fdc8d8SChris Lattner file_spec.GetDirectory().AsCString(""), 44830fdc8d8SChris Lattner file_spec.GetDirectory() ? "/" : "", 44930fdc8d8SChris Lattner file_spec.GetFilename().AsCString(""), 45030fdc8d8SChris Lattner line, 45130fdc8d8SChris Lattner check_inlines ? "yes" : "no", 45230fdc8d8SChris Lattner resolve_scope); 45330fdc8d8SChris Lattner 45430fdc8d8SChris Lattner const uint32_t initial_count = sc_list.GetSize(); 45530fdc8d8SChris Lattner 45630fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 45730fdc8d8SChris Lattner if (symbols) 45830fdc8d8SChris Lattner symbols->ResolveSymbolContext (file_spec, line, check_inlines, resolve_scope, sc_list); 45930fdc8d8SChris Lattner 46030fdc8d8SChris Lattner return sc_list.GetSize() - initial_count; 46130fdc8d8SChris Lattner } 46230fdc8d8SChris Lattner 46330fdc8d8SChris Lattner 46430fdc8d8SChris Lattner uint32_t 465b6d70ebcSSean Callanan Module::FindGlobalVariables(const ConstString &name, const ClangNamespaceDecl *namespace_decl, bool append, uint32_t max_matches, VariableList& variables) 46630fdc8d8SChris Lattner { 46730fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 46830fdc8d8SChris Lattner if (symbols) 469213fdb8bSSean Callanan return symbols->FindGlobalVariables(name, namespace_decl, append, max_matches, variables); 47030fdc8d8SChris Lattner return 0; 47130fdc8d8SChris Lattner } 47230fdc8d8SChris Lattner uint32_t 47330fdc8d8SChris Lattner Module::FindGlobalVariables(const RegularExpression& regex, bool append, uint32_t max_matches, VariableList& variables) 47430fdc8d8SChris Lattner { 47530fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 47630fdc8d8SChris Lattner if (symbols) 47730fdc8d8SChris Lattner return symbols->FindGlobalVariables(regex, append, max_matches, variables); 47830fdc8d8SChris Lattner return 0; 47930fdc8d8SChris Lattner } 48030fdc8d8SChris Lattner 48130fdc8d8SChris Lattner uint32_t 482644247c1SGreg Clayton Module::FindCompileUnits (const FileSpec &path, 483644247c1SGreg Clayton bool append, 484644247c1SGreg Clayton SymbolContextList &sc_list) 485644247c1SGreg Clayton { 486644247c1SGreg Clayton if (!append) 487644247c1SGreg Clayton sc_list.Clear(); 488644247c1SGreg Clayton 489644247c1SGreg Clayton const uint32_t start_size = sc_list.GetSize(); 490644247c1SGreg Clayton const uint32_t num_compile_units = GetNumCompileUnits(); 491644247c1SGreg Clayton SymbolContext sc; 492e1cd1be6SGreg Clayton sc.module_sp = shared_from_this(); 493644247c1SGreg Clayton const bool compare_directory = path.GetDirectory(); 494644247c1SGreg Clayton for (uint32_t i=0; i<num_compile_units; ++i) 495644247c1SGreg Clayton { 496644247c1SGreg Clayton sc.comp_unit = GetCompileUnitAtIndex(i).get(); 497644247c1SGreg Clayton if (FileSpec::Equal (*sc.comp_unit, path, compare_directory)) 498644247c1SGreg Clayton sc_list.Append(sc); 499644247c1SGreg Clayton } 500644247c1SGreg Clayton return sc_list.GetSize() - start_size; 501644247c1SGreg Clayton } 502644247c1SGreg Clayton 503644247c1SGreg Clayton uint32_t 504931180e6SGreg Clayton Module::FindFunctions (const ConstString &name, 505b6d70ebcSSean Callanan const ClangNamespaceDecl *namespace_decl, 506931180e6SGreg Clayton uint32_t name_type_mask, 507931180e6SGreg Clayton bool include_symbols, 508*9df05fbbSSean Callanan bool include_inlines, 509931180e6SGreg Clayton bool append, 510931180e6SGreg Clayton SymbolContextList& sc_list) 51130fdc8d8SChris Lattner { 512931180e6SGreg Clayton if (!append) 513931180e6SGreg Clayton sc_list.Clear(); 514931180e6SGreg Clayton 515931180e6SGreg Clayton const uint32_t start_size = sc_list.GetSize(); 516931180e6SGreg Clayton 517931180e6SGreg Clayton // Find all the functions (not symbols, but debug information functions... 51830fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 51930fdc8d8SChris Lattner if (symbols) 520*9df05fbbSSean Callanan symbols->FindFunctions(name, namespace_decl, name_type_mask, include_inlines, append, sc_list); 521931180e6SGreg Clayton 522931180e6SGreg Clayton // Now check our symbol table for symbols that are code symbols if requested 523931180e6SGreg Clayton if (include_symbols) 524931180e6SGreg Clayton { 525931180e6SGreg Clayton ObjectFile *objfile = GetObjectFile(); 526931180e6SGreg Clayton if (objfile) 527931180e6SGreg Clayton { 528931180e6SGreg Clayton Symtab *symtab = objfile->GetSymtab(); 529931180e6SGreg Clayton if (symtab) 530931180e6SGreg Clayton { 531931180e6SGreg Clayton std::vector<uint32_t> symbol_indexes; 532931180e6SGreg Clayton symtab->FindAllSymbolsWithNameAndType (name, eSymbolTypeCode, Symtab::eDebugAny, Symtab::eVisibilityAny, symbol_indexes); 533931180e6SGreg Clayton const uint32_t num_matches = symbol_indexes.size(); 534931180e6SGreg Clayton if (num_matches) 535931180e6SGreg Clayton { 536357132ebSGreg Clayton const bool merge_symbol_into_function = true; 537931180e6SGreg Clayton SymbolContext sc(this); 538931180e6SGreg Clayton for (uint32_t i=0; i<num_matches; i++) 539931180e6SGreg Clayton { 540931180e6SGreg Clayton sc.symbol = symtab->SymbolAtIndex(symbol_indexes[i]); 541357132ebSGreg Clayton sc_list.AppendIfUnique (sc, merge_symbol_into_function); 542931180e6SGreg Clayton } 543931180e6SGreg Clayton } 544931180e6SGreg Clayton } 545931180e6SGreg Clayton } 546931180e6SGreg Clayton } 547931180e6SGreg Clayton return sc_list.GetSize() - start_size; 54830fdc8d8SChris Lattner } 54930fdc8d8SChris Lattner 55030fdc8d8SChris Lattner uint32_t 551931180e6SGreg Clayton Module::FindFunctions (const RegularExpression& regex, 552931180e6SGreg Clayton bool include_symbols, 553*9df05fbbSSean Callanan bool include_inlines, 554931180e6SGreg Clayton bool append, 555931180e6SGreg Clayton SymbolContextList& sc_list) 55630fdc8d8SChris Lattner { 557931180e6SGreg Clayton if (!append) 558931180e6SGreg Clayton sc_list.Clear(); 559931180e6SGreg Clayton 560931180e6SGreg Clayton const uint32_t start_size = sc_list.GetSize(); 561931180e6SGreg Clayton 56230fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 56330fdc8d8SChris Lattner if (symbols) 564*9df05fbbSSean Callanan symbols->FindFunctions(regex, include_inlines, append, sc_list); 565931180e6SGreg Clayton // Now check our symbol table for symbols that are code symbols if requested 566931180e6SGreg Clayton if (include_symbols) 567931180e6SGreg Clayton { 568931180e6SGreg Clayton ObjectFile *objfile = GetObjectFile(); 569931180e6SGreg Clayton if (objfile) 570931180e6SGreg Clayton { 571931180e6SGreg Clayton Symtab *symtab = objfile->GetSymtab(); 572931180e6SGreg Clayton if (symtab) 573931180e6SGreg Clayton { 574931180e6SGreg Clayton std::vector<uint32_t> symbol_indexes; 575931180e6SGreg Clayton symtab->AppendSymbolIndexesMatchingRegExAndType (regex, eSymbolTypeCode, Symtab::eDebugAny, Symtab::eVisibilityAny, symbol_indexes); 576931180e6SGreg Clayton const uint32_t num_matches = symbol_indexes.size(); 577931180e6SGreg Clayton if (num_matches) 578931180e6SGreg Clayton { 579357132ebSGreg Clayton const bool merge_symbol_into_function = true; 580931180e6SGreg Clayton SymbolContext sc(this); 581931180e6SGreg Clayton for (uint32_t i=0; i<num_matches; i++) 582931180e6SGreg Clayton { 583931180e6SGreg Clayton sc.symbol = symtab->SymbolAtIndex(symbol_indexes[i]); 584357132ebSGreg Clayton sc_list.AppendIfUnique (sc, merge_symbol_into_function); 585931180e6SGreg Clayton } 586931180e6SGreg Clayton } 587931180e6SGreg Clayton } 588931180e6SGreg Clayton } 589931180e6SGreg Clayton } 590931180e6SGreg Clayton return sc_list.GetSize() - start_size; 59130fdc8d8SChris Lattner } 59230fdc8d8SChris Lattner 5933504eee8SGreg Clayton uint32_t 594213fdb8bSSean Callanan Module::FindTypes_Impl (const SymbolContext& sc, const ConstString &name, const ClangNamespaceDecl *namespace_decl, bool append, uint32_t max_matches, TypeList& types) 5953504eee8SGreg Clayton { 5963504eee8SGreg Clayton Timer scoped_timer(__PRETTY_FUNCTION__, __PRETTY_FUNCTION__); 5973504eee8SGreg Clayton if (sc.module_sp.get() == NULL || sc.module_sp.get() == this) 5983504eee8SGreg Clayton { 5993504eee8SGreg Clayton SymbolVendor *symbols = GetSymbolVendor (); 6003504eee8SGreg Clayton if (symbols) 601213fdb8bSSean Callanan return symbols->FindTypes(sc, name, namespace_decl, append, max_matches, types); 6023504eee8SGreg Clayton } 6033504eee8SGreg Clayton return 0; 6043504eee8SGreg Clayton } 6053504eee8SGreg Clayton 6066f3533fbSEnrico Granata // depending on implementation details, type lookup might fail because of 6076f3533fbSEnrico Granata // embedded spurious namespace:: prefixes. this call strips them, paying 6086f3533fbSEnrico Granata // attention to the fact that a type might have namespace'd type names as 6096f3533fbSEnrico Granata // arguments to templates, and those must not be stripped off 6106f3533fbSEnrico Granata static const char* 6116f3533fbSEnrico Granata StripTypeName(const char* name_cstr) 6126f3533fbSEnrico Granata { 613c6770763SJohnny Chen // Protect against null c string. 614c6770763SJohnny Chen if (!name_cstr) 615c6770763SJohnny Chen return name_cstr; 6166f3533fbSEnrico Granata const char* skip_namespace = strstr(name_cstr, "::"); 6176f3533fbSEnrico Granata const char* template_arg_char = strchr(name_cstr, '<'); 6186f3533fbSEnrico Granata while (skip_namespace != NULL) 6196f3533fbSEnrico Granata { 6206f3533fbSEnrico Granata if (template_arg_char != NULL && 6216f3533fbSEnrico Granata skip_namespace > template_arg_char) // but namespace'd template arguments are still good to go 6226f3533fbSEnrico Granata break; 6236f3533fbSEnrico Granata name_cstr = skip_namespace+2; 6246f3533fbSEnrico Granata skip_namespace = strstr(name_cstr, "::"); 6256f3533fbSEnrico Granata } 6266f3533fbSEnrico Granata return name_cstr; 6276f3533fbSEnrico Granata } 6286f3533fbSEnrico Granata 6296f3533fbSEnrico Granata uint32_t 630b6d70ebcSSean Callanan Module::FindTypes (const SymbolContext& sc, const ConstString &name, const ClangNamespaceDecl *namespace_decl, bool append, uint32_t max_matches, TypeList& types) 6316f3533fbSEnrico Granata { 632213fdb8bSSean Callanan uint32_t retval = FindTypes_Impl(sc, name, namespace_decl, append, max_matches, types); 6336f3533fbSEnrico Granata 6346f3533fbSEnrico Granata if (retval == 0) 6356f3533fbSEnrico Granata { 63650b3d507SJim Ingham const char *orig_name = name.GetCString(); 63750b3d507SJim Ingham const char *stripped = StripTypeName(orig_name); 63850b3d507SJim Ingham // Only do this lookup if StripTypeName has stripped the name: 63950b3d507SJim Ingham if (stripped != orig_name) 640213fdb8bSSean Callanan return FindTypes_Impl(sc, ConstString(stripped), namespace_decl, append, max_matches, types); 64150b3d507SJim Ingham else 64250b3d507SJim Ingham return 0; 6436f3533fbSEnrico Granata } 6446f3533fbSEnrico Granata else 6456f3533fbSEnrico Granata return retval; 6466f3533fbSEnrico Granata 6476f3533fbSEnrico Granata } 6486f3533fbSEnrico Granata 64930fdc8d8SChris Lattner //uint32_t 65030fdc8d8SChris Lattner //Module::FindTypes(const SymbolContext& sc, const RegularExpression& regex, bool append, uint32_t max_matches, Type::Encoding encoding, const char *udt_name, TypeList& types) 65130fdc8d8SChris Lattner //{ 65230fdc8d8SChris Lattner // Timer scoped_timer(__PRETTY_FUNCTION__); 65330fdc8d8SChris Lattner // SymbolVendor *symbols = GetSymbolVendor (); 65430fdc8d8SChris Lattner // if (symbols) 65530fdc8d8SChris Lattner // return symbols->FindTypes(sc, regex, append, max_matches, encoding, udt_name, types); 65630fdc8d8SChris Lattner // return 0; 65730fdc8d8SChris Lattner // 65830fdc8d8SChris Lattner //} 65930fdc8d8SChris Lattner 66030fdc8d8SChris Lattner SymbolVendor* 66130fdc8d8SChris Lattner Module::GetSymbolVendor (bool can_create) 66230fdc8d8SChris Lattner { 66330fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 664e83e731eSGreg Clayton if (m_did_load_symbol_vendor == false && can_create) 66530fdc8d8SChris Lattner { 66630fdc8d8SChris Lattner ObjectFile *obj_file = GetObjectFile (); 66730fdc8d8SChris Lattner if (obj_file != NULL) 66830fdc8d8SChris Lattner { 66930fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, __PRETTY_FUNCTION__); 67030fdc8d8SChris Lattner m_symfile_ap.reset(SymbolVendor::FindPlugin(this)); 671e83e731eSGreg Clayton m_did_load_symbol_vendor = true; 67230fdc8d8SChris Lattner } 67330fdc8d8SChris Lattner } 67430fdc8d8SChris Lattner return m_symfile_ap.get(); 67530fdc8d8SChris Lattner } 67630fdc8d8SChris Lattner 67730fdc8d8SChris Lattner void 67830fdc8d8SChris Lattner Module::SetFileSpecAndObjectName (const FileSpec &file, const ConstString &object_name) 67930fdc8d8SChris Lattner { 68030fdc8d8SChris Lattner // Container objects whose paths do not specify a file directly can call 68130fdc8d8SChris Lattner // this function to correct the file and object names. 68230fdc8d8SChris Lattner m_file = file; 68330fdc8d8SChris Lattner m_mod_time = file.GetModificationTime(); 68430fdc8d8SChris Lattner m_object_name = object_name; 68530fdc8d8SChris Lattner } 68630fdc8d8SChris Lattner 68730fdc8d8SChris Lattner const ArchSpec& 68830fdc8d8SChris Lattner Module::GetArchitecture () const 68930fdc8d8SChris Lattner { 69030fdc8d8SChris Lattner return m_arch; 69130fdc8d8SChris Lattner } 69230fdc8d8SChris Lattner 69330fdc8d8SChris Lattner void 694c982b3d6SGreg Clayton Module::GetDescription (Stream *s, lldb::DescriptionLevel level) 695ceb6b139SCaroline Tice { 696ceb6b139SCaroline Tice Mutex::Locker locker (m_mutex); 697ceb6b139SCaroline Tice 698c982b3d6SGreg Clayton if (level >= eDescriptionLevelFull) 699c982b3d6SGreg Clayton { 700cfd1acedSGreg Clayton if (m_arch.IsValid()) 70164195a2cSGreg Clayton s->Printf("(%s) ", m_arch.GetArchitectureName()); 702c982b3d6SGreg Clayton } 703ceb6b139SCaroline Tice 704c982b3d6SGreg Clayton if (level == eDescriptionLevelBrief) 705c982b3d6SGreg Clayton { 706c982b3d6SGreg Clayton const char *filename = m_file.GetFilename().GetCString(); 707c982b3d6SGreg Clayton if (filename) 708c982b3d6SGreg Clayton s->PutCString (filename); 709c982b3d6SGreg Clayton } 710c982b3d6SGreg Clayton else 711c982b3d6SGreg Clayton { 712cfd1acedSGreg Clayton char path[PATH_MAX]; 713cfd1acedSGreg Clayton if (m_file.GetPath(path, sizeof(path))) 714cfd1acedSGreg Clayton s->PutCString(path); 715c982b3d6SGreg Clayton } 716cfd1acedSGreg Clayton 717cfd1acedSGreg Clayton const char *object_name = m_object_name.GetCString(); 718cfd1acedSGreg Clayton if (object_name) 719cfd1acedSGreg Clayton s->Printf("(%s)", object_name); 720ceb6b139SCaroline Tice } 721ceb6b139SCaroline Tice 722ceb6b139SCaroline Tice void 723c982b3d6SGreg Clayton Module::ReportError (const char *format, ...) 724c982b3d6SGreg Clayton { 725e38a5eddSGreg Clayton if (format && format[0]) 726e38a5eddSGreg Clayton { 727e38a5eddSGreg Clayton StreamString strm; 728e38a5eddSGreg Clayton strm.PutCString("error: "); 729e38a5eddSGreg Clayton GetDescription(&strm, lldb::eDescriptionLevelBrief); 7308b35334eSGreg Clayton strm.PutChar (' '); 731c982b3d6SGreg Clayton va_list args; 732c982b3d6SGreg Clayton va_start (args, format); 733e38a5eddSGreg Clayton strm.PrintfVarArg(format, args); 734c982b3d6SGreg Clayton va_end (args); 735e38a5eddSGreg Clayton 736e38a5eddSGreg Clayton const int format_len = strlen(format); 737e38a5eddSGreg Clayton if (format_len > 0) 738e38a5eddSGreg Clayton { 739e38a5eddSGreg Clayton const char last_char = format[format_len-1]; 740e38a5eddSGreg Clayton if (last_char != '\n' || last_char != '\r') 741e38a5eddSGreg Clayton strm.EOL(); 742e38a5eddSGreg Clayton } 743e38a5eddSGreg Clayton Host::SystemLog (Host::eSystemLogError, "%s", strm.GetString().c_str()); 744e38a5eddSGreg Clayton 745e38a5eddSGreg Clayton } 746e38a5eddSGreg Clayton } 747e38a5eddSGreg Clayton 748e38a5eddSGreg Clayton void 749e38a5eddSGreg Clayton Module::ReportErrorIfModifyDetected (const char *format, ...) 750e38a5eddSGreg Clayton { 751e38a5eddSGreg Clayton if (!GetModified(true) && GetModified(false)) 752e38a5eddSGreg Clayton { 753e38a5eddSGreg Clayton if (format) 754e38a5eddSGreg Clayton { 755e38a5eddSGreg Clayton StreamString strm; 756e38a5eddSGreg Clayton strm.PutCString("error: the object file "); 757e38a5eddSGreg Clayton GetDescription(&strm, lldb::eDescriptionLevelFull); 758e38a5eddSGreg Clayton strm.PutCString (" has been modified\n"); 759e38a5eddSGreg Clayton 760e38a5eddSGreg Clayton va_list args; 761e38a5eddSGreg Clayton va_start (args, format); 762e38a5eddSGreg Clayton strm.PrintfVarArg(format, args); 763e38a5eddSGreg Clayton va_end (args); 764e38a5eddSGreg Clayton 765e38a5eddSGreg Clayton const int format_len = strlen(format); 766e38a5eddSGreg Clayton if (format_len > 0) 767e38a5eddSGreg Clayton { 768e38a5eddSGreg Clayton const char last_char = format[format_len-1]; 769e38a5eddSGreg Clayton if (last_char != '\n' || last_char != '\r') 770e38a5eddSGreg Clayton strm.EOL(); 771e38a5eddSGreg Clayton } 772e38a5eddSGreg Clayton strm.PutCString("The debug session should be aborted as the original debug information has been overwritten.\n"); 773e38a5eddSGreg Clayton Host::SystemLog (Host::eSystemLogError, "%s", strm.GetString().c_str()); 774e38a5eddSGreg Clayton } 775e38a5eddSGreg Clayton } 776c982b3d6SGreg Clayton } 777c982b3d6SGreg Clayton 778c982b3d6SGreg Clayton void 779c982b3d6SGreg Clayton Module::ReportWarning (const char *format, ...) 780c982b3d6SGreg Clayton { 781e38a5eddSGreg Clayton if (format && format[0]) 782e38a5eddSGreg Clayton { 783e38a5eddSGreg Clayton StreamString strm; 784e38a5eddSGreg Clayton strm.PutCString("warning: "); 7858b35334eSGreg Clayton GetDescription(&strm, lldb::eDescriptionLevelFull); 7868b35334eSGreg Clayton strm.PutChar (' '); 787c982b3d6SGreg Clayton 788c982b3d6SGreg Clayton va_list args; 789c982b3d6SGreg Clayton va_start (args, format); 790e38a5eddSGreg Clayton strm.PrintfVarArg(format, args); 791c982b3d6SGreg Clayton va_end (args); 792e38a5eddSGreg Clayton 793e38a5eddSGreg Clayton const int format_len = strlen(format); 794e38a5eddSGreg Clayton if (format_len > 0) 795e38a5eddSGreg Clayton { 796e38a5eddSGreg Clayton const char last_char = format[format_len-1]; 797e38a5eddSGreg Clayton if (last_char != '\n' || last_char != '\r') 798e38a5eddSGreg Clayton strm.EOL(); 799e38a5eddSGreg Clayton } 800e38a5eddSGreg Clayton Host::SystemLog (Host::eSystemLogWarning, "%s", strm.GetString().c_str()); 801e38a5eddSGreg Clayton } 802c982b3d6SGreg Clayton } 803c982b3d6SGreg Clayton 804c982b3d6SGreg Clayton void 805c982b3d6SGreg Clayton Module::LogMessage (Log *log, const char *format, ...) 806c982b3d6SGreg Clayton { 807c982b3d6SGreg Clayton if (log) 808c982b3d6SGreg Clayton { 809c982b3d6SGreg Clayton StreamString log_message; 8108b35334eSGreg Clayton GetDescription(&log_message, lldb::eDescriptionLevelFull); 811c982b3d6SGreg Clayton log_message.PutCString (": "); 812c982b3d6SGreg Clayton va_list args; 813c982b3d6SGreg Clayton va_start (args, format); 814c982b3d6SGreg Clayton log_message.PrintfVarArg (format, args); 815c982b3d6SGreg Clayton va_end (args); 816c982b3d6SGreg Clayton log->PutCString(log_message.GetString().c_str()); 817c982b3d6SGreg Clayton } 818c982b3d6SGreg Clayton } 819c982b3d6SGreg Clayton 820e38a5eddSGreg Clayton bool 821e38a5eddSGreg Clayton Module::GetModified (bool use_cached_only) 822e38a5eddSGreg Clayton { 823e38a5eddSGreg Clayton if (m_was_modified == false && use_cached_only == false) 824e38a5eddSGreg Clayton { 825e38a5eddSGreg Clayton TimeValue curr_mod_time (m_file.GetModificationTime()); 826e38a5eddSGreg Clayton m_was_modified = curr_mod_time != m_mod_time; 827e38a5eddSGreg Clayton } 828e38a5eddSGreg Clayton return m_was_modified; 829e38a5eddSGreg Clayton } 830e38a5eddSGreg Clayton 831e38a5eddSGreg Clayton bool 832e38a5eddSGreg Clayton Module::SetModified (bool b) 833e38a5eddSGreg Clayton { 834e38a5eddSGreg Clayton const bool prev_value = m_was_modified; 835e38a5eddSGreg Clayton m_was_modified = b; 836e38a5eddSGreg Clayton return prev_value; 837e38a5eddSGreg Clayton } 838e38a5eddSGreg Clayton 839e38a5eddSGreg Clayton 840c982b3d6SGreg Clayton void 84130fdc8d8SChris Lattner Module::Dump(Stream *s) 84230fdc8d8SChris Lattner { 84330fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 8448941142aSGreg Clayton //s->Printf("%.*p: ", (int)sizeof(void*) * 2, this); 84530fdc8d8SChris Lattner s->Indent(); 84630fdc8d8SChris Lattner s->Printf("Module %s/%s%s%s%s\n", 84730fdc8d8SChris Lattner m_file.GetDirectory().AsCString(), 84830fdc8d8SChris Lattner m_file.GetFilename().AsCString(), 84930fdc8d8SChris Lattner m_object_name ? "(" : "", 85030fdc8d8SChris Lattner m_object_name ? m_object_name.GetCString() : "", 85130fdc8d8SChris Lattner m_object_name ? ")" : ""); 85230fdc8d8SChris Lattner 85330fdc8d8SChris Lattner s->IndentMore(); 85430fdc8d8SChris Lattner ObjectFile *objfile = GetObjectFile (); 85530fdc8d8SChris Lattner 85630fdc8d8SChris Lattner if (objfile) 85730fdc8d8SChris Lattner objfile->Dump(s); 85830fdc8d8SChris Lattner 85930fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 86030fdc8d8SChris Lattner 86130fdc8d8SChris Lattner if (symbols) 86230fdc8d8SChris Lattner symbols->Dump(s); 86330fdc8d8SChris Lattner 86430fdc8d8SChris Lattner s->IndentLess(); 86530fdc8d8SChris Lattner } 86630fdc8d8SChris Lattner 86730fdc8d8SChris Lattner 86830fdc8d8SChris Lattner TypeList* 86930fdc8d8SChris Lattner Module::GetTypeList () 87030fdc8d8SChris Lattner { 87130fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 87230fdc8d8SChris Lattner if (symbols) 87330fdc8d8SChris Lattner return &symbols->GetTypeList(); 87430fdc8d8SChris Lattner return NULL; 87530fdc8d8SChris Lattner } 87630fdc8d8SChris Lattner 87730fdc8d8SChris Lattner const ConstString & 87830fdc8d8SChris Lattner Module::GetObjectName() const 87930fdc8d8SChris Lattner { 88030fdc8d8SChris Lattner return m_object_name; 88130fdc8d8SChris Lattner } 88230fdc8d8SChris Lattner 88330fdc8d8SChris Lattner ObjectFile * 88430fdc8d8SChris Lattner Module::GetObjectFile() 88530fdc8d8SChris Lattner { 88630fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 887e83e731eSGreg Clayton if (m_did_load_objfile == false) 88830fdc8d8SChris Lattner { 889e83e731eSGreg Clayton m_did_load_objfile = true; 89030fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, 89130fdc8d8SChris Lattner "Module::GetObjectFile () module = %s", GetFileSpec().GetFilename().AsCString("")); 89244435ed0SGreg Clayton DataBufferSP file_data_sp; 89344435ed0SGreg Clayton m_objfile_sp = ObjectFile::FindPlugin(this, &m_file, m_object_offset, m_file.GetByteSize(), file_data_sp); 894593577a1SGreg Clayton if (m_objfile_sp) 895593577a1SGreg Clayton { 896593577a1SGreg Clayton // Once we get the object file, update our module with the object file's 897593577a1SGreg Clayton // architecture since it might differ in vendor/os if some parts were 898593577a1SGreg Clayton // unknown. 899593577a1SGreg Clayton m_objfile_sp->GetArchitecture (m_arch); 900593577a1SGreg Clayton } 90130fdc8d8SChris Lattner } 902762f7135SGreg Clayton return m_objfile_sp.get(); 90330fdc8d8SChris Lattner } 90430fdc8d8SChris Lattner 90530fdc8d8SChris Lattner 90630fdc8d8SChris Lattner const Symbol * 90730fdc8d8SChris Lattner Module::FindFirstSymbolWithNameAndType (const ConstString &name, SymbolType symbol_type) 90830fdc8d8SChris Lattner { 90930fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, 91030fdc8d8SChris Lattner "Module::FindFirstSymbolWithNameAndType (name = %s, type = %i)", 91130fdc8d8SChris Lattner name.AsCString(), 91230fdc8d8SChris Lattner symbol_type); 91330fdc8d8SChris Lattner ObjectFile *objfile = GetObjectFile(); 91430fdc8d8SChris Lattner if (objfile) 91530fdc8d8SChris Lattner { 91630fdc8d8SChris Lattner Symtab *symtab = objfile->GetSymtab(); 91730fdc8d8SChris Lattner if (symtab) 918bcf2cfbdSGreg Clayton return symtab->FindFirstSymbolWithNameAndType (name, symbol_type, Symtab::eDebugAny, Symtab::eVisibilityAny); 91930fdc8d8SChris Lattner } 92030fdc8d8SChris Lattner return NULL; 92130fdc8d8SChris Lattner } 92230fdc8d8SChris Lattner void 92330fdc8d8SChris Lattner Module::SymbolIndicesToSymbolContextList (Symtab *symtab, std::vector<uint32_t> &symbol_indexes, SymbolContextList &sc_list) 92430fdc8d8SChris Lattner { 92530fdc8d8SChris Lattner // No need to protect this call using m_mutex all other method calls are 92630fdc8d8SChris Lattner // already thread safe. 92730fdc8d8SChris Lattner 92830fdc8d8SChris Lattner size_t num_indices = symbol_indexes.size(); 92930fdc8d8SChris Lattner if (num_indices > 0) 93030fdc8d8SChris Lattner { 93130fdc8d8SChris Lattner SymbolContext sc; 93230fdc8d8SChris Lattner CalculateSymbolContext (&sc); 93330fdc8d8SChris Lattner for (size_t i = 0; i < num_indices; i++) 93430fdc8d8SChris Lattner { 93530fdc8d8SChris Lattner sc.symbol = symtab->SymbolAtIndex (symbol_indexes[i]); 93630fdc8d8SChris Lattner if (sc.symbol) 93730fdc8d8SChris Lattner sc_list.Append (sc); 93830fdc8d8SChris Lattner } 93930fdc8d8SChris Lattner } 94030fdc8d8SChris Lattner } 94130fdc8d8SChris Lattner 94230fdc8d8SChris Lattner size_t 943b96ff33bSSean Callanan Module::FindSymbolsWithNameAndType (const ConstString &name, SymbolType symbol_type, SymbolContextList &sc_list) 94430fdc8d8SChris Lattner { 94530fdc8d8SChris Lattner // No need to protect this call using m_mutex all other method calls are 94630fdc8d8SChris Lattner // already thread safe. 94730fdc8d8SChris Lattner 94830fdc8d8SChris Lattner 94930fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, 95030fdc8d8SChris Lattner "Module::FindSymbolsWithNameAndType (name = %s, type = %i)", 95130fdc8d8SChris Lattner name.AsCString(), 95230fdc8d8SChris Lattner symbol_type); 95330fdc8d8SChris Lattner const size_t initial_size = sc_list.GetSize(); 95430fdc8d8SChris Lattner ObjectFile *objfile = GetObjectFile (); 95530fdc8d8SChris Lattner if (objfile) 95630fdc8d8SChris Lattner { 95730fdc8d8SChris Lattner Symtab *symtab = objfile->GetSymtab(); 95830fdc8d8SChris Lattner if (symtab) 95930fdc8d8SChris Lattner { 96030fdc8d8SChris Lattner std::vector<uint32_t> symbol_indexes; 96130fdc8d8SChris Lattner symtab->FindAllSymbolsWithNameAndType (name, symbol_type, symbol_indexes); 96230fdc8d8SChris Lattner SymbolIndicesToSymbolContextList (symtab, symbol_indexes, sc_list); 96330fdc8d8SChris Lattner } 96430fdc8d8SChris Lattner } 96530fdc8d8SChris Lattner return sc_list.GetSize() - initial_size; 96630fdc8d8SChris Lattner } 96730fdc8d8SChris Lattner 96830fdc8d8SChris Lattner size_t 96930fdc8d8SChris Lattner Module::FindSymbolsMatchingRegExAndType (const RegularExpression ®ex, SymbolType symbol_type, SymbolContextList &sc_list) 97030fdc8d8SChris Lattner { 97130fdc8d8SChris Lattner // No need to protect this call using m_mutex all other method calls are 97230fdc8d8SChris Lattner // already thread safe. 97330fdc8d8SChris Lattner 97430fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, 97530fdc8d8SChris Lattner "Module::FindSymbolsMatchingRegExAndType (regex = %s, type = %i)", 97630fdc8d8SChris Lattner regex.GetText(), 97730fdc8d8SChris Lattner symbol_type); 97830fdc8d8SChris Lattner const size_t initial_size = sc_list.GetSize(); 97930fdc8d8SChris Lattner ObjectFile *objfile = GetObjectFile (); 98030fdc8d8SChris Lattner if (objfile) 98130fdc8d8SChris Lattner { 98230fdc8d8SChris Lattner Symtab *symtab = objfile->GetSymtab(); 98330fdc8d8SChris Lattner if (symtab) 98430fdc8d8SChris Lattner { 98530fdc8d8SChris Lattner std::vector<uint32_t> symbol_indexes; 986bcf2cfbdSGreg Clayton symtab->FindAllSymbolsMatchingRexExAndType (regex, symbol_type, Symtab::eDebugAny, Symtab::eVisibilityAny, symbol_indexes); 98730fdc8d8SChris Lattner SymbolIndicesToSymbolContextList (symtab, symbol_indexes, sc_list); 98830fdc8d8SChris Lattner } 98930fdc8d8SChris Lattner } 99030fdc8d8SChris Lattner return sc_list.GetSize() - initial_size; 99130fdc8d8SChris Lattner } 99230fdc8d8SChris Lattner 99330fdc8d8SChris Lattner const TimeValue & 99430fdc8d8SChris Lattner Module::GetModificationTime () const 99530fdc8d8SChris Lattner { 99630fdc8d8SChris Lattner return m_mod_time; 99730fdc8d8SChris Lattner } 9985aee162fSJim Ingham 9995aee162fSJim Ingham bool 10005aee162fSJim Ingham Module::IsExecutable () 10015aee162fSJim Ingham { 10025aee162fSJim Ingham if (GetObjectFile() == NULL) 10035aee162fSJim Ingham return false; 10045aee162fSJim Ingham else 10055aee162fSJim Ingham return GetObjectFile()->IsExecutable(); 10065aee162fSJim Ingham } 10075aee162fSJim Ingham 10085aee162fSJim Ingham bool 1009b53cb271SJim Ingham Module::IsLoadedInTarget (Target *target) 1010b53cb271SJim Ingham { 1011b53cb271SJim Ingham ObjectFile *obj_file = GetObjectFile(); 1012b53cb271SJim Ingham if (obj_file) 1013b53cb271SJim Ingham { 1014b53cb271SJim Ingham SectionList *sections = obj_file->GetSectionList(); 1015b53cb271SJim Ingham if (sections != NULL) 1016b53cb271SJim Ingham { 1017b53cb271SJim Ingham size_t num_sections = sections->GetSize(); 1018b53cb271SJim Ingham for (size_t sect_idx = 0; sect_idx < num_sections; sect_idx++) 1019b53cb271SJim Ingham { 1020b53cb271SJim Ingham SectionSP section_sp = sections->GetSectionAtIndex(sect_idx); 1021b53cb271SJim Ingham if (section_sp->GetLoadBaseAddress(target) != LLDB_INVALID_ADDRESS) 1022b53cb271SJim Ingham { 1023b53cb271SJim Ingham return true; 1024b53cb271SJim Ingham } 1025b53cb271SJim Ingham } 1026b53cb271SJim Ingham } 1027b53cb271SJim Ingham } 1028b53cb271SJim Ingham return false; 1029b53cb271SJim Ingham } 1030b53cb271SJim Ingham bool 10315aee162fSJim Ingham Module::SetArchitecture (const ArchSpec &new_arch) 10325aee162fSJim Ingham { 103364195a2cSGreg Clayton if (!m_arch.IsValid()) 10345aee162fSJim Ingham { 10355aee162fSJim Ingham m_arch = new_arch; 10365aee162fSJim Ingham return true; 10375aee162fSJim Ingham } 103864195a2cSGreg Clayton return m_arch == new_arch; 10395aee162fSJim Ingham } 10405aee162fSJim Ingham 1041c9660546SGreg Clayton bool 1042c9660546SGreg Clayton Module::SetLoadAddress (Target &target, lldb::addr_t offset, bool &changed) 1043c9660546SGreg Clayton { 1044c9660546SGreg Clayton changed = false; 1045c9660546SGreg Clayton ObjectFile *image_object_file = GetObjectFile(); 1046c9660546SGreg Clayton if (image_object_file) 1047c9660546SGreg Clayton { 1048c9660546SGreg Clayton SectionList *section_list = image_object_file->GetSectionList (); 1049c9660546SGreg Clayton if (section_list) 1050c9660546SGreg Clayton { 1051c9660546SGreg Clayton const size_t num_sections = section_list->GetSize(); 1052c9660546SGreg Clayton size_t sect_idx = 0; 1053c9660546SGreg Clayton for (sect_idx = 0; sect_idx < num_sections; ++sect_idx) 1054c9660546SGreg Clayton { 1055c9660546SGreg Clayton // Iterate through the object file sections to find the 1056c9660546SGreg Clayton // first section that starts of file offset zero and that 1057c9660546SGreg Clayton // has bytes in the file... 1058c9660546SGreg Clayton Section *section = section_list->GetSectionAtIndex (sect_idx).get(); 1059c9660546SGreg Clayton if (section) 1060c9660546SGreg Clayton { 1061c9660546SGreg Clayton if (target.GetSectionLoadList().SetSectionLoadAddress (section, section->GetFileAddress() + offset)) 1062c9660546SGreg Clayton changed = true; 1063c9660546SGreg Clayton } 1064c9660546SGreg Clayton } 1065c9660546SGreg Clayton return sect_idx > 0; 1066c9660546SGreg Clayton } 1067c9660546SGreg Clayton } 1068c9660546SGreg Clayton return false; 1069c9660546SGreg Clayton } 1070c9660546SGreg Clayton 1071