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 117e72dfb32SGreg Clayton Module::Module(const FileSpec& file_spec, 118e72dfb32SGreg Clayton const ArchSpec& arch, 119e72dfb32SGreg Clayton const ConstString *object_name, 120e72dfb32SGreg Clayton off_t object_offset) : 12130fdc8d8SChris Lattner m_mutex (Mutex::eMutexTypeRecursive), 12230fdc8d8SChris Lattner m_mod_time (file_spec.GetModificationTime()), 12330fdc8d8SChris Lattner m_arch (arch), 12430fdc8d8SChris Lattner m_uuid (), 12530fdc8d8SChris Lattner m_file (file_spec), 12632e0a750SGreg Clayton m_platform_file(), 127e72dfb32SGreg Clayton m_symfile_spec (), 12830fdc8d8SChris Lattner m_object_name (), 1298b82f087SGreg Clayton m_object_offset (object_offset), 130762f7135SGreg Clayton m_objfile_sp (), 131e83e731eSGreg Clayton m_symfile_ap (), 1326beaaa68SGreg Clayton m_ast (), 133e83e731eSGreg Clayton m_did_load_objfile (false), 134e83e731eSGreg Clayton m_did_load_symbol_vendor (false), 135e83e731eSGreg Clayton m_did_parse_uuid (false), 1366beaaa68SGreg Clayton m_did_init_ast (false), 137e38a5eddSGreg Clayton m_is_dynamic_loader_module (false), 138e38a5eddSGreg Clayton m_was_modified (false) 13930fdc8d8SChris Lattner { 14065a03991SGreg Clayton // Scope for locker below... 14165a03991SGreg Clayton { 14265a03991SGreg Clayton Mutex::Locker locker (GetAllocationModuleCollectionMutex()); 14365a03991SGreg Clayton GetModuleCollection().push_back(this); 14465a03991SGreg Clayton } 14565a03991SGreg Clayton 14630fdc8d8SChris Lattner if (object_name) 14730fdc8d8SChris Lattner m_object_name = *object_name; 1482d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_OBJECT)); 14930fdc8d8SChris Lattner if (log) 15030fdc8d8SChris Lattner log->Printf ("%p Module::Module((%s) '%s/%s%s%s%s')", 15130fdc8d8SChris Lattner this, 15264195a2cSGreg Clayton m_arch.GetArchitectureName(), 15330fdc8d8SChris Lattner m_file.GetDirectory().AsCString(""), 15430fdc8d8SChris Lattner m_file.GetFilename().AsCString(""), 15530fdc8d8SChris Lattner m_object_name.IsEmpty() ? "" : "(", 15630fdc8d8SChris Lattner m_object_name.IsEmpty() ? "" : m_object_name.AsCString(""), 15730fdc8d8SChris Lattner m_object_name.IsEmpty() ? "" : ")"); 15830fdc8d8SChris Lattner } 15930fdc8d8SChris Lattner 16030fdc8d8SChris Lattner Module::~Module() 16130fdc8d8SChris Lattner { 16265a03991SGreg Clayton // Scope for locker below... 16365a03991SGreg Clayton { 16465a03991SGreg Clayton Mutex::Locker locker (GetAllocationModuleCollectionMutex()); 16565a03991SGreg Clayton ModuleCollection &modules = GetModuleCollection(); 16665a03991SGreg Clayton ModuleCollection::iterator end = modules.end(); 16765a03991SGreg Clayton ModuleCollection::iterator pos = std::find(modules.begin(), end, this); 16865a03991SGreg Clayton if (pos != end) 16965a03991SGreg Clayton modules.erase(pos); 17065a03991SGreg Clayton } 1712d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_OBJECT)); 17230fdc8d8SChris Lattner if (log) 17330fdc8d8SChris Lattner log->Printf ("%p Module::~Module((%s) '%s/%s%s%s%s')", 17430fdc8d8SChris Lattner this, 17564195a2cSGreg Clayton m_arch.GetArchitectureName(), 17630fdc8d8SChris Lattner m_file.GetDirectory().AsCString(""), 17730fdc8d8SChris Lattner m_file.GetFilename().AsCString(""), 17830fdc8d8SChris Lattner m_object_name.IsEmpty() ? "" : "(", 17930fdc8d8SChris Lattner m_object_name.IsEmpty() ? "" : m_object_name.AsCString(""), 18030fdc8d8SChris Lattner m_object_name.IsEmpty() ? "" : ")"); 1816beaaa68SGreg Clayton // Release any auto pointers before we start tearing down our member 1826beaaa68SGreg Clayton // variables since the object file and symbol files might need to make 1836beaaa68SGreg Clayton // function calls back into this module object. The ordering is important 1846beaaa68SGreg Clayton // here because symbol files can require the module object file. So we tear 1856beaaa68SGreg Clayton // down the symbol file first, then the object file. 1866beaaa68SGreg Clayton m_symfile_ap.reset(); 187762f7135SGreg Clayton m_objfile_sp.reset(); 18830fdc8d8SChris Lattner } 18930fdc8d8SChris Lattner 190*c7f09ccaSGreg Clayton ObjectFile * 191*c7f09ccaSGreg Clayton Module::GetMemoryObjectFile (const lldb::ProcessSP &process_sp, lldb::addr_t header_addr, Error &error) 192*c7f09ccaSGreg Clayton { 193*c7f09ccaSGreg Clayton if (m_objfile_sp) 194*c7f09ccaSGreg Clayton { 195*c7f09ccaSGreg Clayton error.SetErrorString ("object file already exists"); 196*c7f09ccaSGreg Clayton } 197*c7f09ccaSGreg Clayton else 198*c7f09ccaSGreg Clayton { 199*c7f09ccaSGreg Clayton Mutex::Locker locker (m_mutex); 200*c7f09ccaSGreg Clayton if (process_sp) 201*c7f09ccaSGreg Clayton { 202*c7f09ccaSGreg Clayton StreamString s; 203*c7f09ccaSGreg Clayton if (m_file.GetFilename()) 204*c7f09ccaSGreg Clayton s << m_file.GetFilename(); 205*c7f09ccaSGreg Clayton s.Printf("[0x%16.16llx]", header_addr); 206*c7f09ccaSGreg Clayton m_file.GetFilename().SetCString (s.GetData()); 207*c7f09ccaSGreg Clayton m_did_load_objfile = true; 208*c7f09ccaSGreg Clayton std::auto_ptr<DataBufferHeap> data_ap (new DataBufferHeap (512, 0)); 209*c7f09ccaSGreg Clayton Error readmem_error; 210*c7f09ccaSGreg Clayton const size_t bytes_read = process_sp->ReadMemory (header_addr, 211*c7f09ccaSGreg Clayton data_ap->GetBytes(), 212*c7f09ccaSGreg Clayton data_ap->GetByteSize(), 213*c7f09ccaSGreg Clayton readmem_error); 214*c7f09ccaSGreg Clayton if (bytes_read == 512) 215*c7f09ccaSGreg Clayton { 216*c7f09ccaSGreg Clayton DataBufferSP data_sp(data_ap.release()); 217*c7f09ccaSGreg Clayton m_objfile_sp = ObjectFile::FindPlugin(shared_from_this(), process_sp, header_addr, data_sp); 218*c7f09ccaSGreg Clayton if (m_objfile_sp) 219*c7f09ccaSGreg Clayton { 220*c7f09ccaSGreg Clayton // Once we get the object file, update our module with the object file's 221*c7f09ccaSGreg Clayton // architecture since it might differ in vendor/os if some parts were 222*c7f09ccaSGreg Clayton // unknown. 223*c7f09ccaSGreg Clayton m_objfile_sp->GetArchitecture (m_arch); 224*c7f09ccaSGreg Clayton } 225*c7f09ccaSGreg Clayton else 226*c7f09ccaSGreg Clayton { 227*c7f09ccaSGreg Clayton error.SetErrorString ("unable to find suitable object file plug-in"); 228*c7f09ccaSGreg Clayton } 229*c7f09ccaSGreg Clayton } 230*c7f09ccaSGreg Clayton else 231*c7f09ccaSGreg Clayton { 232*c7f09ccaSGreg Clayton error.SetErrorStringWithFormat ("unable to read header from memory: %s", readmem_error.AsCString()); 233*c7f09ccaSGreg Clayton } 234*c7f09ccaSGreg Clayton } 235*c7f09ccaSGreg Clayton else 236*c7f09ccaSGreg Clayton { 237*c7f09ccaSGreg Clayton error.SetErrorString ("invalid process"); 238*c7f09ccaSGreg Clayton } 239*c7f09ccaSGreg Clayton } 240*c7f09ccaSGreg Clayton return m_objfile_sp.get(); 241*c7f09ccaSGreg Clayton } 242*c7f09ccaSGreg Clayton 24330fdc8d8SChris Lattner 24460830268SGreg Clayton const lldb_private::UUID& 24530fdc8d8SChris Lattner Module::GetUUID() 24630fdc8d8SChris Lattner { 24730fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 248e83e731eSGreg Clayton if (m_did_parse_uuid == false) 24930fdc8d8SChris Lattner { 25030fdc8d8SChris Lattner ObjectFile * obj_file = GetObjectFile (); 25130fdc8d8SChris Lattner 25230fdc8d8SChris Lattner if (obj_file != NULL) 25330fdc8d8SChris Lattner { 25430fdc8d8SChris Lattner obj_file->GetUUID(&m_uuid); 255e83e731eSGreg Clayton m_did_parse_uuid = true; 25630fdc8d8SChris Lattner } 25730fdc8d8SChris Lattner } 25830fdc8d8SChris Lattner return m_uuid; 25930fdc8d8SChris Lattner } 26030fdc8d8SChris Lattner 2616beaaa68SGreg Clayton ClangASTContext & 2626beaaa68SGreg Clayton Module::GetClangASTContext () 2636beaaa68SGreg Clayton { 2646beaaa68SGreg Clayton Mutex::Locker locker (m_mutex); 2656beaaa68SGreg Clayton if (m_did_init_ast == false) 2666beaaa68SGreg Clayton { 2676beaaa68SGreg Clayton ObjectFile * objfile = GetObjectFile(); 268514487e8SGreg Clayton ArchSpec object_arch; 269514487e8SGreg Clayton if (objfile && objfile->GetArchitecture(object_arch)) 2706beaaa68SGreg Clayton { 2716beaaa68SGreg Clayton m_did_init_ast = true; 272514487e8SGreg Clayton m_ast.SetArchitecture (object_arch); 2736beaaa68SGreg Clayton } 2746beaaa68SGreg Clayton } 2756beaaa68SGreg Clayton return m_ast; 2766beaaa68SGreg Clayton } 2776beaaa68SGreg Clayton 27830fdc8d8SChris Lattner void 27930fdc8d8SChris Lattner Module::ParseAllDebugSymbols() 28030fdc8d8SChris Lattner { 28130fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 28230fdc8d8SChris Lattner uint32_t num_comp_units = GetNumCompileUnits(); 28330fdc8d8SChris Lattner if (num_comp_units == 0) 28430fdc8d8SChris Lattner return; 28530fdc8d8SChris Lattner 286a2eee184SGreg Clayton SymbolContext sc; 287e1cd1be6SGreg Clayton sc.module_sp = shared_from_this(); 28830fdc8d8SChris Lattner uint32_t cu_idx; 28930fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 29030fdc8d8SChris Lattner 29130fdc8d8SChris Lattner for (cu_idx = 0; cu_idx < num_comp_units; cu_idx++) 29230fdc8d8SChris Lattner { 29330fdc8d8SChris Lattner sc.comp_unit = symbols->GetCompileUnitAtIndex(cu_idx).get(); 29430fdc8d8SChris Lattner if (sc.comp_unit) 29530fdc8d8SChris Lattner { 29630fdc8d8SChris Lattner sc.function = NULL; 29730fdc8d8SChris Lattner symbols->ParseVariablesForContext(sc); 29830fdc8d8SChris Lattner 29930fdc8d8SChris Lattner symbols->ParseCompileUnitFunctions(sc); 30030fdc8d8SChris Lattner 30130fdc8d8SChris Lattner uint32_t func_idx; 30230fdc8d8SChris Lattner for (func_idx = 0; (sc.function = sc.comp_unit->GetFunctionAtIndex(func_idx).get()) != NULL; ++func_idx) 30330fdc8d8SChris Lattner { 30430fdc8d8SChris Lattner symbols->ParseFunctionBlocks(sc); 30530fdc8d8SChris Lattner 30630fdc8d8SChris Lattner // Parse the variables for this function and all its blocks 30730fdc8d8SChris Lattner symbols->ParseVariablesForContext(sc); 30830fdc8d8SChris Lattner } 30930fdc8d8SChris Lattner 31030fdc8d8SChris Lattner 31130fdc8d8SChris Lattner // Parse all types for this compile unit 31230fdc8d8SChris Lattner sc.function = NULL; 31330fdc8d8SChris Lattner symbols->ParseTypes(sc); 31430fdc8d8SChris Lattner } 31530fdc8d8SChris Lattner } 31630fdc8d8SChris Lattner } 31730fdc8d8SChris Lattner 31830fdc8d8SChris Lattner void 31930fdc8d8SChris Lattner Module::CalculateSymbolContext(SymbolContext* sc) 32030fdc8d8SChris Lattner { 321e1cd1be6SGreg Clayton sc->module_sp = shared_from_this(); 32230fdc8d8SChris Lattner } 32330fdc8d8SChris Lattner 324e72dfb32SGreg Clayton ModuleSP 3257e9b1fd0SGreg Clayton Module::CalculateSymbolContextModule () 3267e9b1fd0SGreg Clayton { 327e72dfb32SGreg Clayton return shared_from_this(); 3287e9b1fd0SGreg Clayton } 3297e9b1fd0SGreg Clayton 33030fdc8d8SChris Lattner void 33130fdc8d8SChris Lattner Module::DumpSymbolContext(Stream *s) 33230fdc8d8SChris Lattner { 333fd54b368SJason Molenda s->Printf(", Module{%p}", this); 33430fdc8d8SChris Lattner } 33530fdc8d8SChris Lattner 33630fdc8d8SChris Lattner uint32_t 33730fdc8d8SChris Lattner Module::GetNumCompileUnits() 33830fdc8d8SChris Lattner { 33930fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 34030fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, "Module::GetNumCompileUnits (module = %p)", this); 34130fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 34230fdc8d8SChris Lattner if (symbols) 34330fdc8d8SChris Lattner return symbols->GetNumCompileUnits(); 34430fdc8d8SChris Lattner return 0; 34530fdc8d8SChris Lattner } 34630fdc8d8SChris Lattner 34730fdc8d8SChris Lattner CompUnitSP 34830fdc8d8SChris Lattner Module::GetCompileUnitAtIndex (uint32_t index) 34930fdc8d8SChris Lattner { 35030fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 35130fdc8d8SChris Lattner uint32_t num_comp_units = GetNumCompileUnits (); 35230fdc8d8SChris Lattner CompUnitSP cu_sp; 35330fdc8d8SChris Lattner 35430fdc8d8SChris Lattner if (index < num_comp_units) 35530fdc8d8SChris Lattner { 35630fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 35730fdc8d8SChris Lattner if (symbols) 35830fdc8d8SChris Lattner cu_sp = symbols->GetCompileUnitAtIndex(index); 35930fdc8d8SChris Lattner } 36030fdc8d8SChris Lattner return cu_sp; 36130fdc8d8SChris Lattner } 36230fdc8d8SChris Lattner 36330fdc8d8SChris Lattner bool 36430fdc8d8SChris Lattner Module::ResolveFileAddress (lldb::addr_t vm_addr, Address& so_addr) 36530fdc8d8SChris Lattner { 36630fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 36730fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, "Module::ResolveFileAddress (vm_addr = 0x%llx)", vm_addr); 36830fdc8d8SChris Lattner ObjectFile* ofile = GetObjectFile(); 36930fdc8d8SChris Lattner if (ofile) 37030fdc8d8SChris Lattner return so_addr.ResolveAddressUsingFileSections(vm_addr, ofile->GetSectionList()); 37130fdc8d8SChris Lattner return false; 37230fdc8d8SChris Lattner } 37330fdc8d8SChris Lattner 37430fdc8d8SChris Lattner uint32_t 37530fdc8d8SChris Lattner Module::ResolveSymbolContextForAddress (const Address& so_addr, uint32_t resolve_scope, SymbolContext& sc) 37630fdc8d8SChris Lattner { 37730fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 37830fdc8d8SChris Lattner uint32_t resolved_flags = 0; 37930fdc8d8SChris Lattner 38030fdc8d8SChris Lattner // Clear the result symbol context in case we don't find anything 38130fdc8d8SChris Lattner sc.Clear(); 38230fdc8d8SChris Lattner 38330fdc8d8SChris Lattner // Get the section from the section/offset address. 384e72dfb32SGreg Clayton SectionSP section_sp (so_addr.GetSection()); 38530fdc8d8SChris Lattner 38630fdc8d8SChris Lattner // Make sure the section matches this module before we try and match anything 387e72dfb32SGreg Clayton if (section_sp && section_sp->GetModule().get() == this) 38830fdc8d8SChris Lattner { 38930fdc8d8SChris Lattner // If the section offset based address resolved itself, then this 39030fdc8d8SChris Lattner // is the right module. 391e1cd1be6SGreg Clayton sc.module_sp = shared_from_this(); 39230fdc8d8SChris Lattner resolved_flags |= eSymbolContextModule; 39330fdc8d8SChris Lattner 39430fdc8d8SChris Lattner // Resolve the compile unit, function, block, line table or line 39530fdc8d8SChris Lattner // entry if requested. 39630fdc8d8SChris Lattner if (resolve_scope & eSymbolContextCompUnit || 39730fdc8d8SChris Lattner resolve_scope & eSymbolContextFunction || 39830fdc8d8SChris Lattner resolve_scope & eSymbolContextBlock || 39930fdc8d8SChris Lattner resolve_scope & eSymbolContextLineEntry ) 40030fdc8d8SChris Lattner { 40130fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 40230fdc8d8SChris Lattner if (symbols) 40330fdc8d8SChris Lattner resolved_flags |= symbols->ResolveSymbolContext (so_addr, resolve_scope, sc); 40430fdc8d8SChris Lattner } 40530fdc8d8SChris Lattner 406680e1778SJim Ingham // Resolve the symbol if requested, but don't re-look it up if we've already found it. 407680e1778SJim Ingham if (resolve_scope & eSymbolContextSymbol && !(resolved_flags & eSymbolContextSymbol)) 40830fdc8d8SChris Lattner { 40930fdc8d8SChris Lattner ObjectFile* ofile = GetObjectFile(); 41030fdc8d8SChris Lattner if (ofile) 41130fdc8d8SChris Lattner { 41230fdc8d8SChris Lattner Symtab *symtab = ofile->GetSymtab(); 41330fdc8d8SChris Lattner if (symtab) 41430fdc8d8SChris Lattner { 41530fdc8d8SChris Lattner if (so_addr.IsSectionOffset()) 41630fdc8d8SChris Lattner { 41730fdc8d8SChris Lattner sc.symbol = symtab->FindSymbolContainingFileAddress(so_addr.GetFileAddress()); 41830fdc8d8SChris Lattner if (sc.symbol) 41930fdc8d8SChris Lattner resolved_flags |= eSymbolContextSymbol; 42030fdc8d8SChris Lattner } 42130fdc8d8SChris Lattner } 42230fdc8d8SChris Lattner } 42330fdc8d8SChris Lattner } 42430fdc8d8SChris Lattner } 42530fdc8d8SChris Lattner return resolved_flags; 42630fdc8d8SChris Lattner } 42730fdc8d8SChris Lattner 42830fdc8d8SChris Lattner uint32_t 429274060b6SGreg Clayton Module::ResolveSymbolContextForFilePath 430274060b6SGreg Clayton ( 431274060b6SGreg Clayton const char *file_path, 432274060b6SGreg Clayton uint32_t line, 433274060b6SGreg Clayton bool check_inlines, 434274060b6SGreg Clayton uint32_t resolve_scope, 435274060b6SGreg Clayton SymbolContextList& sc_list 436274060b6SGreg Clayton ) 43730fdc8d8SChris Lattner { 438274060b6SGreg Clayton FileSpec file_spec(file_path, false); 43930fdc8d8SChris Lattner return ResolveSymbolContextsForFileSpec (file_spec, line, check_inlines, resolve_scope, sc_list); 44030fdc8d8SChris Lattner } 44130fdc8d8SChris Lattner 44230fdc8d8SChris Lattner uint32_t 44330fdc8d8SChris Lattner Module::ResolveSymbolContextsForFileSpec (const FileSpec &file_spec, uint32_t line, bool check_inlines, uint32_t resolve_scope, SymbolContextList& sc_list) 44430fdc8d8SChris Lattner { 44530fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 44630fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, 44730fdc8d8SChris Lattner "Module::ResolveSymbolContextForFilePath (%s%s%s:%u, check_inlines = %s, resolve_scope = 0x%8.8x)", 44830fdc8d8SChris Lattner file_spec.GetDirectory().AsCString(""), 44930fdc8d8SChris Lattner file_spec.GetDirectory() ? "/" : "", 45030fdc8d8SChris Lattner file_spec.GetFilename().AsCString(""), 45130fdc8d8SChris Lattner line, 45230fdc8d8SChris Lattner check_inlines ? "yes" : "no", 45330fdc8d8SChris Lattner resolve_scope); 45430fdc8d8SChris Lattner 45530fdc8d8SChris Lattner const uint32_t initial_count = sc_list.GetSize(); 45630fdc8d8SChris Lattner 45730fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 45830fdc8d8SChris Lattner if (symbols) 45930fdc8d8SChris Lattner symbols->ResolveSymbolContext (file_spec, line, check_inlines, resolve_scope, sc_list); 46030fdc8d8SChris Lattner 46130fdc8d8SChris Lattner return sc_list.GetSize() - initial_count; 46230fdc8d8SChris Lattner } 46330fdc8d8SChris Lattner 46430fdc8d8SChris Lattner 46530fdc8d8SChris Lattner uint32_t 466b6d70ebcSSean Callanan Module::FindGlobalVariables(const ConstString &name, const ClangNamespaceDecl *namespace_decl, bool append, uint32_t max_matches, VariableList& variables) 46730fdc8d8SChris Lattner { 46830fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 46930fdc8d8SChris Lattner if (symbols) 470213fdb8bSSean Callanan return symbols->FindGlobalVariables(name, namespace_decl, append, max_matches, variables); 47130fdc8d8SChris Lattner return 0; 47230fdc8d8SChris Lattner } 47330fdc8d8SChris Lattner uint32_t 47430fdc8d8SChris Lattner Module::FindGlobalVariables(const RegularExpression& regex, bool append, uint32_t max_matches, VariableList& variables) 47530fdc8d8SChris Lattner { 47630fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 47730fdc8d8SChris Lattner if (symbols) 47830fdc8d8SChris Lattner return symbols->FindGlobalVariables(regex, append, max_matches, variables); 47930fdc8d8SChris Lattner return 0; 48030fdc8d8SChris Lattner } 48130fdc8d8SChris Lattner 48230fdc8d8SChris Lattner uint32_t 483644247c1SGreg Clayton Module::FindCompileUnits (const FileSpec &path, 484644247c1SGreg Clayton bool append, 485644247c1SGreg Clayton SymbolContextList &sc_list) 486644247c1SGreg Clayton { 487644247c1SGreg Clayton if (!append) 488644247c1SGreg Clayton sc_list.Clear(); 489644247c1SGreg Clayton 490644247c1SGreg Clayton const uint32_t start_size = sc_list.GetSize(); 491644247c1SGreg Clayton const uint32_t num_compile_units = GetNumCompileUnits(); 492644247c1SGreg Clayton SymbolContext sc; 493e1cd1be6SGreg Clayton sc.module_sp = shared_from_this(); 494644247c1SGreg Clayton const bool compare_directory = path.GetDirectory(); 495644247c1SGreg Clayton for (uint32_t i=0; i<num_compile_units; ++i) 496644247c1SGreg Clayton { 497644247c1SGreg Clayton sc.comp_unit = GetCompileUnitAtIndex(i).get(); 498644247c1SGreg Clayton if (FileSpec::Equal (*sc.comp_unit, path, compare_directory)) 499644247c1SGreg Clayton sc_list.Append(sc); 500644247c1SGreg Clayton } 501644247c1SGreg Clayton return sc_list.GetSize() - start_size; 502644247c1SGreg Clayton } 503644247c1SGreg Clayton 504644247c1SGreg Clayton uint32_t 505931180e6SGreg Clayton Module::FindFunctions (const ConstString &name, 506b6d70ebcSSean Callanan const ClangNamespaceDecl *namespace_decl, 507931180e6SGreg Clayton uint32_t name_type_mask, 508931180e6SGreg Clayton bool include_symbols, 5099df05fbbSSean Callanan bool include_inlines, 510931180e6SGreg Clayton bool append, 511931180e6SGreg Clayton SymbolContextList& sc_list) 51230fdc8d8SChris Lattner { 513931180e6SGreg Clayton if (!append) 514931180e6SGreg Clayton sc_list.Clear(); 515931180e6SGreg Clayton 516931180e6SGreg Clayton const uint32_t start_size = sc_list.GetSize(); 517931180e6SGreg Clayton 518931180e6SGreg Clayton // Find all the functions (not symbols, but debug information functions... 51930fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 52030fdc8d8SChris Lattner if (symbols) 5219df05fbbSSean Callanan symbols->FindFunctions(name, namespace_decl, name_type_mask, include_inlines, append, sc_list); 522931180e6SGreg Clayton 523931180e6SGreg Clayton // Now check our symbol table for symbols that are code symbols if requested 524931180e6SGreg Clayton if (include_symbols) 525931180e6SGreg Clayton { 526931180e6SGreg Clayton ObjectFile *objfile = GetObjectFile(); 527931180e6SGreg Clayton if (objfile) 528931180e6SGreg Clayton { 529931180e6SGreg Clayton Symtab *symtab = objfile->GetSymtab(); 530931180e6SGreg Clayton if (symtab) 531931180e6SGreg Clayton { 532931180e6SGreg Clayton std::vector<uint32_t> symbol_indexes; 533931180e6SGreg Clayton symtab->FindAllSymbolsWithNameAndType (name, eSymbolTypeCode, Symtab::eDebugAny, Symtab::eVisibilityAny, symbol_indexes); 534931180e6SGreg Clayton const uint32_t num_matches = symbol_indexes.size(); 535931180e6SGreg Clayton if (num_matches) 536931180e6SGreg Clayton { 537357132ebSGreg Clayton const bool merge_symbol_into_function = true; 538931180e6SGreg Clayton SymbolContext sc(this); 539931180e6SGreg Clayton for (uint32_t i=0; i<num_matches; i++) 540931180e6SGreg Clayton { 541931180e6SGreg Clayton sc.symbol = symtab->SymbolAtIndex(symbol_indexes[i]); 542357132ebSGreg Clayton sc_list.AppendIfUnique (sc, merge_symbol_into_function); 543931180e6SGreg Clayton } 544931180e6SGreg Clayton } 545931180e6SGreg Clayton } 546931180e6SGreg Clayton } 547931180e6SGreg Clayton } 548931180e6SGreg Clayton return sc_list.GetSize() - start_size; 54930fdc8d8SChris Lattner } 55030fdc8d8SChris Lattner 55130fdc8d8SChris Lattner uint32_t 552931180e6SGreg Clayton Module::FindFunctions (const RegularExpression& regex, 553931180e6SGreg Clayton bool include_symbols, 5549df05fbbSSean Callanan bool include_inlines, 555931180e6SGreg Clayton bool append, 556931180e6SGreg Clayton SymbolContextList& sc_list) 55730fdc8d8SChris Lattner { 558931180e6SGreg Clayton if (!append) 559931180e6SGreg Clayton sc_list.Clear(); 560931180e6SGreg Clayton 561931180e6SGreg Clayton const uint32_t start_size = sc_list.GetSize(); 562931180e6SGreg Clayton 56330fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 56430fdc8d8SChris Lattner if (symbols) 5659df05fbbSSean Callanan symbols->FindFunctions(regex, include_inlines, append, sc_list); 566931180e6SGreg Clayton // Now check our symbol table for symbols that are code symbols if requested 567931180e6SGreg Clayton if (include_symbols) 568931180e6SGreg Clayton { 569931180e6SGreg Clayton ObjectFile *objfile = GetObjectFile(); 570931180e6SGreg Clayton if (objfile) 571931180e6SGreg Clayton { 572931180e6SGreg Clayton Symtab *symtab = objfile->GetSymtab(); 573931180e6SGreg Clayton if (symtab) 574931180e6SGreg Clayton { 575931180e6SGreg Clayton std::vector<uint32_t> symbol_indexes; 576931180e6SGreg Clayton symtab->AppendSymbolIndexesMatchingRegExAndType (regex, eSymbolTypeCode, Symtab::eDebugAny, Symtab::eVisibilityAny, symbol_indexes); 577931180e6SGreg Clayton const uint32_t num_matches = symbol_indexes.size(); 578931180e6SGreg Clayton if (num_matches) 579931180e6SGreg Clayton { 580357132ebSGreg Clayton const bool merge_symbol_into_function = true; 581931180e6SGreg Clayton SymbolContext sc(this); 582931180e6SGreg Clayton for (uint32_t i=0; i<num_matches; i++) 583931180e6SGreg Clayton { 584931180e6SGreg Clayton sc.symbol = symtab->SymbolAtIndex(symbol_indexes[i]); 585357132ebSGreg Clayton sc_list.AppendIfUnique (sc, merge_symbol_into_function); 586931180e6SGreg Clayton } 587931180e6SGreg Clayton } 588931180e6SGreg Clayton } 589931180e6SGreg Clayton } 590931180e6SGreg Clayton } 591931180e6SGreg Clayton return sc_list.GetSize() - start_size; 59230fdc8d8SChris Lattner } 59330fdc8d8SChris Lattner 5943504eee8SGreg Clayton uint32_t 595213fdb8bSSean Callanan Module::FindTypes_Impl (const SymbolContext& sc, const ConstString &name, const ClangNamespaceDecl *namespace_decl, bool append, uint32_t max_matches, TypeList& types) 5963504eee8SGreg Clayton { 5973504eee8SGreg Clayton Timer scoped_timer(__PRETTY_FUNCTION__, __PRETTY_FUNCTION__); 5983504eee8SGreg Clayton if (sc.module_sp.get() == NULL || sc.module_sp.get() == this) 5993504eee8SGreg Clayton { 6003504eee8SGreg Clayton SymbolVendor *symbols = GetSymbolVendor (); 6013504eee8SGreg Clayton if (symbols) 602213fdb8bSSean Callanan return symbols->FindTypes(sc, name, namespace_decl, append, max_matches, types); 6033504eee8SGreg Clayton } 6043504eee8SGreg Clayton return 0; 6053504eee8SGreg Clayton } 6063504eee8SGreg Clayton 6076f3533fbSEnrico Granata // depending on implementation details, type lookup might fail because of 6086f3533fbSEnrico Granata // embedded spurious namespace:: prefixes. this call strips them, paying 6096f3533fbSEnrico Granata // attention to the fact that a type might have namespace'd type names as 6106f3533fbSEnrico Granata // arguments to templates, and those must not be stripped off 6116f3533fbSEnrico Granata static const char* 6126f3533fbSEnrico Granata StripTypeName(const char* name_cstr) 6136f3533fbSEnrico Granata { 614c6770763SJohnny Chen // Protect against null c string. 615c6770763SJohnny Chen if (!name_cstr) 616c6770763SJohnny Chen return name_cstr; 6176f3533fbSEnrico Granata const char* skip_namespace = strstr(name_cstr, "::"); 6186f3533fbSEnrico Granata const char* template_arg_char = strchr(name_cstr, '<'); 6196f3533fbSEnrico Granata while (skip_namespace != NULL) 6206f3533fbSEnrico Granata { 6216f3533fbSEnrico Granata if (template_arg_char != NULL && 6226f3533fbSEnrico Granata skip_namespace > template_arg_char) // but namespace'd template arguments are still good to go 6236f3533fbSEnrico Granata break; 6246f3533fbSEnrico Granata name_cstr = skip_namespace+2; 6256f3533fbSEnrico Granata skip_namespace = strstr(name_cstr, "::"); 6266f3533fbSEnrico Granata } 6276f3533fbSEnrico Granata return name_cstr; 6286f3533fbSEnrico Granata } 6296f3533fbSEnrico Granata 6306f3533fbSEnrico Granata uint32_t 631b6d70ebcSSean Callanan Module::FindTypes (const SymbolContext& sc, const ConstString &name, const ClangNamespaceDecl *namespace_decl, bool append, uint32_t max_matches, TypeList& types) 6326f3533fbSEnrico Granata { 633213fdb8bSSean Callanan uint32_t retval = FindTypes_Impl(sc, name, namespace_decl, append, max_matches, types); 6346f3533fbSEnrico Granata 6356f3533fbSEnrico Granata if (retval == 0) 6366f3533fbSEnrico Granata { 63750b3d507SJim Ingham const char *orig_name = name.GetCString(); 63850b3d507SJim Ingham const char *stripped = StripTypeName(orig_name); 63950b3d507SJim Ingham // Only do this lookup if StripTypeName has stripped the name: 64050b3d507SJim Ingham if (stripped != orig_name) 641213fdb8bSSean Callanan return FindTypes_Impl(sc, ConstString(stripped), namespace_decl, append, max_matches, types); 64250b3d507SJim Ingham else 64350b3d507SJim Ingham return 0; 6446f3533fbSEnrico Granata } 6456f3533fbSEnrico Granata else 6466f3533fbSEnrico Granata return retval; 6476f3533fbSEnrico Granata 6486f3533fbSEnrico Granata } 6496f3533fbSEnrico Granata 65030fdc8d8SChris Lattner //uint32_t 65130fdc8d8SChris Lattner //Module::FindTypes(const SymbolContext& sc, const RegularExpression& regex, bool append, uint32_t max_matches, Type::Encoding encoding, const char *udt_name, TypeList& types) 65230fdc8d8SChris Lattner //{ 65330fdc8d8SChris Lattner // Timer scoped_timer(__PRETTY_FUNCTION__); 65430fdc8d8SChris Lattner // SymbolVendor *symbols = GetSymbolVendor (); 65530fdc8d8SChris Lattner // if (symbols) 65630fdc8d8SChris Lattner // return symbols->FindTypes(sc, regex, append, max_matches, encoding, udt_name, types); 65730fdc8d8SChris Lattner // return 0; 65830fdc8d8SChris Lattner // 65930fdc8d8SChris Lattner //} 66030fdc8d8SChris Lattner 66130fdc8d8SChris Lattner SymbolVendor* 66230fdc8d8SChris Lattner Module::GetSymbolVendor (bool can_create) 66330fdc8d8SChris Lattner { 66430fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 665e83e731eSGreg Clayton if (m_did_load_symbol_vendor == false && can_create) 66630fdc8d8SChris Lattner { 66730fdc8d8SChris Lattner ObjectFile *obj_file = GetObjectFile (); 66830fdc8d8SChris Lattner if (obj_file != NULL) 66930fdc8d8SChris Lattner { 67030fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, __PRETTY_FUNCTION__); 671e72dfb32SGreg Clayton m_symfile_ap.reset(SymbolVendor::FindPlugin(shared_from_this())); 672e83e731eSGreg Clayton m_did_load_symbol_vendor = true; 67330fdc8d8SChris Lattner } 67430fdc8d8SChris Lattner } 67530fdc8d8SChris Lattner return m_symfile_ap.get(); 67630fdc8d8SChris Lattner } 67730fdc8d8SChris Lattner 67830fdc8d8SChris Lattner void 67930fdc8d8SChris Lattner Module::SetFileSpecAndObjectName (const FileSpec &file, const ConstString &object_name) 68030fdc8d8SChris Lattner { 68130fdc8d8SChris Lattner // Container objects whose paths do not specify a file directly can call 68230fdc8d8SChris Lattner // this function to correct the file and object names. 68330fdc8d8SChris Lattner m_file = file; 68430fdc8d8SChris Lattner m_mod_time = file.GetModificationTime(); 68530fdc8d8SChris Lattner m_object_name = object_name; 68630fdc8d8SChris Lattner } 68730fdc8d8SChris Lattner 68830fdc8d8SChris Lattner const ArchSpec& 68930fdc8d8SChris Lattner Module::GetArchitecture () const 69030fdc8d8SChris Lattner { 69130fdc8d8SChris Lattner return m_arch; 69230fdc8d8SChris Lattner } 69330fdc8d8SChris Lattner 69430fdc8d8SChris Lattner void 695c982b3d6SGreg Clayton Module::GetDescription (Stream *s, lldb::DescriptionLevel level) 696ceb6b139SCaroline Tice { 697ceb6b139SCaroline Tice Mutex::Locker locker (m_mutex); 698ceb6b139SCaroline Tice 699c982b3d6SGreg Clayton if (level >= eDescriptionLevelFull) 700c982b3d6SGreg Clayton { 701cfd1acedSGreg Clayton if (m_arch.IsValid()) 70264195a2cSGreg Clayton s->Printf("(%s) ", m_arch.GetArchitectureName()); 703c982b3d6SGreg Clayton } 704ceb6b139SCaroline Tice 705c982b3d6SGreg Clayton if (level == eDescriptionLevelBrief) 706c982b3d6SGreg Clayton { 707c982b3d6SGreg Clayton const char *filename = m_file.GetFilename().GetCString(); 708c982b3d6SGreg Clayton if (filename) 709c982b3d6SGreg Clayton s->PutCString (filename); 710c982b3d6SGreg Clayton } 711c982b3d6SGreg Clayton else 712c982b3d6SGreg Clayton { 713cfd1acedSGreg Clayton char path[PATH_MAX]; 714cfd1acedSGreg Clayton if (m_file.GetPath(path, sizeof(path))) 715cfd1acedSGreg Clayton s->PutCString(path); 716c982b3d6SGreg Clayton } 717cfd1acedSGreg Clayton 718cfd1acedSGreg Clayton const char *object_name = m_object_name.GetCString(); 719cfd1acedSGreg Clayton if (object_name) 720cfd1acedSGreg Clayton s->Printf("(%s)", object_name); 721ceb6b139SCaroline Tice } 722ceb6b139SCaroline Tice 723ceb6b139SCaroline Tice void 724c982b3d6SGreg Clayton Module::ReportError (const char *format, ...) 725c982b3d6SGreg Clayton { 726e38a5eddSGreg Clayton if (format && format[0]) 727e38a5eddSGreg Clayton { 728e38a5eddSGreg Clayton StreamString strm; 729e38a5eddSGreg Clayton strm.PutCString("error: "); 730e38a5eddSGreg Clayton GetDescription(&strm, lldb::eDescriptionLevelBrief); 7318b35334eSGreg Clayton strm.PutChar (' '); 732c982b3d6SGreg Clayton va_list args; 733c982b3d6SGreg Clayton va_start (args, format); 734e38a5eddSGreg Clayton strm.PrintfVarArg(format, args); 735c982b3d6SGreg Clayton va_end (args); 736e38a5eddSGreg Clayton 737e38a5eddSGreg Clayton const int format_len = strlen(format); 738e38a5eddSGreg Clayton if (format_len > 0) 739e38a5eddSGreg Clayton { 740e38a5eddSGreg Clayton const char last_char = format[format_len-1]; 741e38a5eddSGreg Clayton if (last_char != '\n' || last_char != '\r') 742e38a5eddSGreg Clayton strm.EOL(); 743e38a5eddSGreg Clayton } 744e38a5eddSGreg Clayton Host::SystemLog (Host::eSystemLogError, "%s", strm.GetString().c_str()); 745e38a5eddSGreg Clayton 746e38a5eddSGreg Clayton } 747e38a5eddSGreg Clayton } 748e38a5eddSGreg Clayton 749e38a5eddSGreg Clayton void 750e38a5eddSGreg Clayton Module::ReportErrorIfModifyDetected (const char *format, ...) 751e38a5eddSGreg Clayton { 752e38a5eddSGreg Clayton if (!GetModified(true) && GetModified(false)) 753e38a5eddSGreg Clayton { 754e38a5eddSGreg Clayton if (format) 755e38a5eddSGreg Clayton { 756e38a5eddSGreg Clayton StreamString strm; 757e38a5eddSGreg Clayton strm.PutCString("error: the object file "); 758e38a5eddSGreg Clayton GetDescription(&strm, lldb::eDescriptionLevelFull); 759e38a5eddSGreg Clayton strm.PutCString (" has been modified\n"); 760e38a5eddSGreg Clayton 761e38a5eddSGreg Clayton va_list args; 762e38a5eddSGreg Clayton va_start (args, format); 763e38a5eddSGreg Clayton strm.PrintfVarArg(format, args); 764e38a5eddSGreg Clayton va_end (args); 765e38a5eddSGreg Clayton 766e38a5eddSGreg Clayton const int format_len = strlen(format); 767e38a5eddSGreg Clayton if (format_len > 0) 768e38a5eddSGreg Clayton { 769e38a5eddSGreg Clayton const char last_char = format[format_len-1]; 770e38a5eddSGreg Clayton if (last_char != '\n' || last_char != '\r') 771e38a5eddSGreg Clayton strm.EOL(); 772e38a5eddSGreg Clayton } 773e38a5eddSGreg Clayton strm.PutCString("The debug session should be aborted as the original debug information has been overwritten.\n"); 774e38a5eddSGreg Clayton Host::SystemLog (Host::eSystemLogError, "%s", strm.GetString().c_str()); 775e38a5eddSGreg Clayton } 776e38a5eddSGreg Clayton } 777c982b3d6SGreg Clayton } 778c982b3d6SGreg Clayton 779c982b3d6SGreg Clayton void 780c982b3d6SGreg Clayton Module::ReportWarning (const char *format, ...) 781c982b3d6SGreg Clayton { 782e38a5eddSGreg Clayton if (format && format[0]) 783e38a5eddSGreg Clayton { 784e38a5eddSGreg Clayton StreamString strm; 785e38a5eddSGreg Clayton strm.PutCString("warning: "); 7868b35334eSGreg Clayton GetDescription(&strm, lldb::eDescriptionLevelFull); 7878b35334eSGreg Clayton strm.PutChar (' '); 788c982b3d6SGreg Clayton 789c982b3d6SGreg Clayton va_list args; 790c982b3d6SGreg Clayton va_start (args, format); 791e38a5eddSGreg Clayton strm.PrintfVarArg(format, args); 792c982b3d6SGreg Clayton va_end (args); 793e38a5eddSGreg Clayton 794e38a5eddSGreg Clayton const int format_len = strlen(format); 795e38a5eddSGreg Clayton if (format_len > 0) 796e38a5eddSGreg Clayton { 797e38a5eddSGreg Clayton const char last_char = format[format_len-1]; 798e38a5eddSGreg Clayton if (last_char != '\n' || last_char != '\r') 799e38a5eddSGreg Clayton strm.EOL(); 800e38a5eddSGreg Clayton } 801e38a5eddSGreg Clayton Host::SystemLog (Host::eSystemLogWarning, "%s", strm.GetString().c_str()); 802e38a5eddSGreg Clayton } 803c982b3d6SGreg Clayton } 804c982b3d6SGreg Clayton 805c982b3d6SGreg Clayton void 806c982b3d6SGreg Clayton Module::LogMessage (Log *log, const char *format, ...) 807c982b3d6SGreg Clayton { 808c982b3d6SGreg Clayton if (log) 809c982b3d6SGreg Clayton { 810c982b3d6SGreg Clayton StreamString log_message; 8118b35334eSGreg Clayton GetDescription(&log_message, lldb::eDescriptionLevelFull); 812c982b3d6SGreg Clayton log_message.PutCString (": "); 813c982b3d6SGreg Clayton va_list args; 814c982b3d6SGreg Clayton va_start (args, format); 815c982b3d6SGreg Clayton log_message.PrintfVarArg (format, args); 816c982b3d6SGreg Clayton va_end (args); 817c982b3d6SGreg Clayton log->PutCString(log_message.GetString().c_str()); 818c982b3d6SGreg Clayton } 819c982b3d6SGreg Clayton } 820c982b3d6SGreg Clayton 821e38a5eddSGreg Clayton bool 822e38a5eddSGreg Clayton Module::GetModified (bool use_cached_only) 823e38a5eddSGreg Clayton { 824e38a5eddSGreg Clayton if (m_was_modified == false && use_cached_only == false) 825e38a5eddSGreg Clayton { 826e38a5eddSGreg Clayton TimeValue curr_mod_time (m_file.GetModificationTime()); 827e38a5eddSGreg Clayton m_was_modified = curr_mod_time != m_mod_time; 828e38a5eddSGreg Clayton } 829e38a5eddSGreg Clayton return m_was_modified; 830e38a5eddSGreg Clayton } 831e38a5eddSGreg Clayton 832e38a5eddSGreg Clayton bool 833e38a5eddSGreg Clayton Module::SetModified (bool b) 834e38a5eddSGreg Clayton { 835e38a5eddSGreg Clayton const bool prev_value = m_was_modified; 836e38a5eddSGreg Clayton m_was_modified = b; 837e38a5eddSGreg Clayton return prev_value; 838e38a5eddSGreg Clayton } 839e38a5eddSGreg Clayton 840e38a5eddSGreg Clayton 841c982b3d6SGreg Clayton void 84230fdc8d8SChris Lattner Module::Dump(Stream *s) 84330fdc8d8SChris Lattner { 84430fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 8458941142aSGreg Clayton //s->Printf("%.*p: ", (int)sizeof(void*) * 2, this); 84630fdc8d8SChris Lattner s->Indent(); 84730fdc8d8SChris Lattner s->Printf("Module %s/%s%s%s%s\n", 84830fdc8d8SChris Lattner m_file.GetDirectory().AsCString(), 84930fdc8d8SChris Lattner m_file.GetFilename().AsCString(), 85030fdc8d8SChris Lattner m_object_name ? "(" : "", 85130fdc8d8SChris Lattner m_object_name ? m_object_name.GetCString() : "", 85230fdc8d8SChris Lattner m_object_name ? ")" : ""); 85330fdc8d8SChris Lattner 85430fdc8d8SChris Lattner s->IndentMore(); 85530fdc8d8SChris Lattner ObjectFile *objfile = GetObjectFile (); 85630fdc8d8SChris Lattner 85730fdc8d8SChris Lattner if (objfile) 85830fdc8d8SChris Lattner objfile->Dump(s); 85930fdc8d8SChris Lattner 86030fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 86130fdc8d8SChris Lattner 86230fdc8d8SChris Lattner if (symbols) 86330fdc8d8SChris Lattner symbols->Dump(s); 86430fdc8d8SChris Lattner 86530fdc8d8SChris Lattner s->IndentLess(); 86630fdc8d8SChris Lattner } 86730fdc8d8SChris Lattner 86830fdc8d8SChris Lattner 86930fdc8d8SChris Lattner TypeList* 87030fdc8d8SChris Lattner Module::GetTypeList () 87130fdc8d8SChris Lattner { 87230fdc8d8SChris Lattner SymbolVendor *symbols = GetSymbolVendor (); 87330fdc8d8SChris Lattner if (symbols) 87430fdc8d8SChris Lattner return &symbols->GetTypeList(); 87530fdc8d8SChris Lattner return NULL; 87630fdc8d8SChris Lattner } 87730fdc8d8SChris Lattner 87830fdc8d8SChris Lattner const ConstString & 87930fdc8d8SChris Lattner Module::GetObjectName() const 88030fdc8d8SChris Lattner { 88130fdc8d8SChris Lattner return m_object_name; 88230fdc8d8SChris Lattner } 88330fdc8d8SChris Lattner 88430fdc8d8SChris Lattner ObjectFile * 88530fdc8d8SChris Lattner Module::GetObjectFile() 88630fdc8d8SChris Lattner { 88730fdc8d8SChris Lattner Mutex::Locker locker (m_mutex); 888e83e731eSGreg Clayton if (m_did_load_objfile == false) 88930fdc8d8SChris Lattner { 890e83e731eSGreg Clayton m_did_load_objfile = true; 89130fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, 89230fdc8d8SChris Lattner "Module::GetObjectFile () module = %s", GetFileSpec().GetFilename().AsCString("")); 89344435ed0SGreg Clayton DataBufferSP file_data_sp; 894e72dfb32SGreg Clayton m_objfile_sp = ObjectFile::FindPlugin (shared_from_this(), 895e72dfb32SGreg Clayton &m_file, 896e72dfb32SGreg Clayton m_object_offset, 897e72dfb32SGreg Clayton m_file.GetByteSize(), 898e72dfb32SGreg Clayton file_data_sp); 899593577a1SGreg Clayton if (m_objfile_sp) 900593577a1SGreg Clayton { 901593577a1SGreg Clayton // Once we get the object file, update our module with the object file's 902593577a1SGreg Clayton // architecture since it might differ in vendor/os if some parts were 903593577a1SGreg Clayton // unknown. 904593577a1SGreg Clayton m_objfile_sp->GetArchitecture (m_arch); 905593577a1SGreg Clayton } 90630fdc8d8SChris Lattner } 907762f7135SGreg Clayton return m_objfile_sp.get(); 90830fdc8d8SChris Lattner } 90930fdc8d8SChris Lattner 91030fdc8d8SChris Lattner 91130fdc8d8SChris Lattner const Symbol * 91230fdc8d8SChris Lattner Module::FindFirstSymbolWithNameAndType (const ConstString &name, SymbolType symbol_type) 91330fdc8d8SChris Lattner { 91430fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, 91530fdc8d8SChris Lattner "Module::FindFirstSymbolWithNameAndType (name = %s, type = %i)", 91630fdc8d8SChris Lattner name.AsCString(), 91730fdc8d8SChris Lattner symbol_type); 91830fdc8d8SChris Lattner ObjectFile *objfile = GetObjectFile(); 91930fdc8d8SChris Lattner if (objfile) 92030fdc8d8SChris Lattner { 92130fdc8d8SChris Lattner Symtab *symtab = objfile->GetSymtab(); 92230fdc8d8SChris Lattner if (symtab) 923bcf2cfbdSGreg Clayton return symtab->FindFirstSymbolWithNameAndType (name, symbol_type, Symtab::eDebugAny, Symtab::eVisibilityAny); 92430fdc8d8SChris Lattner } 92530fdc8d8SChris Lattner return NULL; 92630fdc8d8SChris Lattner } 92730fdc8d8SChris Lattner void 92830fdc8d8SChris Lattner Module::SymbolIndicesToSymbolContextList (Symtab *symtab, std::vector<uint32_t> &symbol_indexes, SymbolContextList &sc_list) 92930fdc8d8SChris Lattner { 93030fdc8d8SChris Lattner // No need to protect this call using m_mutex all other method calls are 93130fdc8d8SChris Lattner // already thread safe. 93230fdc8d8SChris Lattner 93330fdc8d8SChris Lattner size_t num_indices = symbol_indexes.size(); 93430fdc8d8SChris Lattner if (num_indices > 0) 93530fdc8d8SChris Lattner { 93630fdc8d8SChris Lattner SymbolContext sc; 93730fdc8d8SChris Lattner CalculateSymbolContext (&sc); 93830fdc8d8SChris Lattner for (size_t i = 0; i < num_indices; i++) 93930fdc8d8SChris Lattner { 94030fdc8d8SChris Lattner sc.symbol = symtab->SymbolAtIndex (symbol_indexes[i]); 94130fdc8d8SChris Lattner if (sc.symbol) 94230fdc8d8SChris Lattner sc_list.Append (sc); 94330fdc8d8SChris Lattner } 94430fdc8d8SChris Lattner } 94530fdc8d8SChris Lattner } 94630fdc8d8SChris Lattner 94730fdc8d8SChris Lattner size_t 948b96ff33bSSean Callanan Module::FindSymbolsWithNameAndType (const ConstString &name, SymbolType symbol_type, SymbolContextList &sc_list) 94930fdc8d8SChris Lattner { 95030fdc8d8SChris Lattner // No need to protect this call using m_mutex all other method calls are 95130fdc8d8SChris Lattner // already thread safe. 95230fdc8d8SChris Lattner 95330fdc8d8SChris Lattner 95430fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, 95530fdc8d8SChris Lattner "Module::FindSymbolsWithNameAndType (name = %s, type = %i)", 95630fdc8d8SChris Lattner name.AsCString(), 95730fdc8d8SChris Lattner symbol_type); 95830fdc8d8SChris Lattner const size_t initial_size = sc_list.GetSize(); 95930fdc8d8SChris Lattner ObjectFile *objfile = GetObjectFile (); 96030fdc8d8SChris Lattner if (objfile) 96130fdc8d8SChris Lattner { 96230fdc8d8SChris Lattner Symtab *symtab = objfile->GetSymtab(); 96330fdc8d8SChris Lattner if (symtab) 96430fdc8d8SChris Lattner { 96530fdc8d8SChris Lattner std::vector<uint32_t> symbol_indexes; 96630fdc8d8SChris Lattner symtab->FindAllSymbolsWithNameAndType (name, symbol_type, symbol_indexes); 96730fdc8d8SChris Lattner SymbolIndicesToSymbolContextList (symtab, symbol_indexes, sc_list); 96830fdc8d8SChris Lattner } 96930fdc8d8SChris Lattner } 97030fdc8d8SChris Lattner return sc_list.GetSize() - initial_size; 97130fdc8d8SChris Lattner } 97230fdc8d8SChris Lattner 97330fdc8d8SChris Lattner size_t 97430fdc8d8SChris Lattner Module::FindSymbolsMatchingRegExAndType (const RegularExpression ®ex, SymbolType symbol_type, SymbolContextList &sc_list) 97530fdc8d8SChris Lattner { 97630fdc8d8SChris Lattner // No need to protect this call using m_mutex all other method calls are 97730fdc8d8SChris Lattner // already thread safe. 97830fdc8d8SChris Lattner 97930fdc8d8SChris Lattner Timer scoped_timer(__PRETTY_FUNCTION__, 98030fdc8d8SChris Lattner "Module::FindSymbolsMatchingRegExAndType (regex = %s, type = %i)", 98130fdc8d8SChris Lattner regex.GetText(), 98230fdc8d8SChris Lattner symbol_type); 98330fdc8d8SChris Lattner const size_t initial_size = sc_list.GetSize(); 98430fdc8d8SChris Lattner ObjectFile *objfile = GetObjectFile (); 98530fdc8d8SChris Lattner if (objfile) 98630fdc8d8SChris Lattner { 98730fdc8d8SChris Lattner Symtab *symtab = objfile->GetSymtab(); 98830fdc8d8SChris Lattner if (symtab) 98930fdc8d8SChris Lattner { 99030fdc8d8SChris Lattner std::vector<uint32_t> symbol_indexes; 991bcf2cfbdSGreg Clayton symtab->FindAllSymbolsMatchingRexExAndType (regex, symbol_type, Symtab::eDebugAny, Symtab::eVisibilityAny, symbol_indexes); 99230fdc8d8SChris Lattner SymbolIndicesToSymbolContextList (symtab, symbol_indexes, sc_list); 99330fdc8d8SChris Lattner } 99430fdc8d8SChris Lattner } 99530fdc8d8SChris Lattner return sc_list.GetSize() - initial_size; 99630fdc8d8SChris Lattner } 99730fdc8d8SChris Lattner 99830fdc8d8SChris Lattner const TimeValue & 99930fdc8d8SChris Lattner Module::GetModificationTime () const 100030fdc8d8SChris Lattner { 100130fdc8d8SChris Lattner return m_mod_time; 100230fdc8d8SChris Lattner } 10035aee162fSJim Ingham 10045aee162fSJim Ingham bool 10055aee162fSJim Ingham Module::IsExecutable () 10065aee162fSJim Ingham { 10075aee162fSJim Ingham if (GetObjectFile() == NULL) 10085aee162fSJim Ingham return false; 10095aee162fSJim Ingham else 10105aee162fSJim Ingham return GetObjectFile()->IsExecutable(); 10115aee162fSJim Ingham } 10125aee162fSJim Ingham 10135aee162fSJim Ingham bool 1014b53cb271SJim Ingham Module::IsLoadedInTarget (Target *target) 1015b53cb271SJim Ingham { 1016b53cb271SJim Ingham ObjectFile *obj_file = GetObjectFile(); 1017b53cb271SJim Ingham if (obj_file) 1018b53cb271SJim Ingham { 1019b53cb271SJim Ingham SectionList *sections = obj_file->GetSectionList(); 1020b53cb271SJim Ingham if (sections != NULL) 1021b53cb271SJim Ingham { 1022b53cb271SJim Ingham size_t num_sections = sections->GetSize(); 1023b53cb271SJim Ingham for (size_t sect_idx = 0; sect_idx < num_sections; sect_idx++) 1024b53cb271SJim Ingham { 1025b53cb271SJim Ingham SectionSP section_sp = sections->GetSectionAtIndex(sect_idx); 1026b53cb271SJim Ingham if (section_sp->GetLoadBaseAddress(target) != LLDB_INVALID_ADDRESS) 1027b53cb271SJim Ingham { 1028b53cb271SJim Ingham return true; 1029b53cb271SJim Ingham } 1030b53cb271SJim Ingham } 1031b53cb271SJim Ingham } 1032b53cb271SJim Ingham } 1033b53cb271SJim Ingham return false; 1034b53cb271SJim Ingham } 1035b53cb271SJim Ingham bool 10365aee162fSJim Ingham Module::SetArchitecture (const ArchSpec &new_arch) 10375aee162fSJim Ingham { 103864195a2cSGreg Clayton if (!m_arch.IsValid()) 10395aee162fSJim Ingham { 10405aee162fSJim Ingham m_arch = new_arch; 10415aee162fSJim Ingham return true; 10425aee162fSJim Ingham } 104364195a2cSGreg Clayton return m_arch == new_arch; 10445aee162fSJim Ingham } 10455aee162fSJim Ingham 1046c9660546SGreg Clayton bool 1047c9660546SGreg Clayton Module::SetLoadAddress (Target &target, lldb::addr_t offset, bool &changed) 1048c9660546SGreg Clayton { 1049c9660546SGreg Clayton changed = false; 1050c9660546SGreg Clayton ObjectFile *image_object_file = GetObjectFile(); 1051c9660546SGreg Clayton if (image_object_file) 1052c9660546SGreg Clayton { 1053c9660546SGreg Clayton SectionList *section_list = image_object_file->GetSectionList (); 1054c9660546SGreg Clayton if (section_list) 1055c9660546SGreg Clayton { 1056c9660546SGreg Clayton const size_t num_sections = section_list->GetSize(); 1057c9660546SGreg Clayton size_t sect_idx = 0; 1058c9660546SGreg Clayton for (sect_idx = 0; sect_idx < num_sections; ++sect_idx) 1059c9660546SGreg Clayton { 1060c9660546SGreg Clayton // Iterate through the object file sections to find the 1061c9660546SGreg Clayton // first section that starts of file offset zero and that 1062c9660546SGreg Clayton // has bytes in the file... 1063c9660546SGreg Clayton Section *section = section_list->GetSectionAtIndex (sect_idx).get(); 1064c9660546SGreg Clayton if (section) 1065c9660546SGreg Clayton { 1066c9660546SGreg Clayton if (target.GetSectionLoadList().SetSectionLoadAddress (section, section->GetFileAddress() + offset)) 1067c9660546SGreg Clayton changed = true; 1068c9660546SGreg Clayton } 1069c9660546SGreg Clayton } 1070c9660546SGreg Clayton return sect_idx > 0; 1071c9660546SGreg Clayton } 1072c9660546SGreg Clayton } 1073c9660546SGreg Clayton return false; 1074c9660546SGreg Clayton } 1075c9660546SGreg Clayton 1076