1 //===-- SymbolVendor.mm -----------------------------------------*- C++ -*-===// 2 // 3 // The LLVM Compiler Infrastructure 4 // 5 // This file is distributed under the University of Illinois Open Source 6 // License. See LICENSE.TXT for details. 7 // 8 //===----------------------------------------------------------------------===// 9 10 #include "lldb/Symbol/SymbolVendor.h" 11 12 // C Includes 13 // C++ Includes 14 // Other libraries and framework includes 15 // Project includes 16 #include "lldb/Core/Module.h" 17 #include "lldb/Core/PluginManager.h" 18 #include "lldb/Core/Stream.h" 19 #include "lldb/Symbol/CompileUnit.h" 20 #include "lldb/Symbol/ObjectFile.h" 21 #include "lldb/Symbol/SymbolFile.h" 22 23 using namespace lldb; 24 using namespace lldb_private; 25 26 27 //---------------------------------------------------------------------- 28 // FindPlugin 29 // 30 // Platforms can register a callback to use when creating symbol 31 // vendors to allow for complex debug information file setups, and to 32 // also allow for finding separate debug information files. 33 //---------------------------------------------------------------------- 34 SymbolVendor* 35 SymbolVendor::FindPlugin (const lldb::ModuleSP &module_sp, lldb_private::Stream *feedback_strm) 36 { 37 std::auto_ptr<SymbolVendor> instance_ap; 38 //---------------------------------------------------------------------- 39 // We currently only have one debug symbol parser... 40 //---------------------------------------------------------------------- 41 SymbolVendorCreateInstance create_callback; 42 for (size_t idx = 0; (create_callback = PluginManager::GetSymbolVendorCreateCallbackAtIndex(idx)) != NULL; ++idx) 43 { 44 instance_ap.reset(create_callback(module_sp, feedback_strm)); 45 46 if (instance_ap.get()) 47 { 48 // TODO: make sure this symbol vendor is what we want. We 49 // currently are just returning the first one we find, but 50 // we may want to call this function only when we have our 51 // main executable module and then give all symbol vendor 52 // plug-ins a chance to compete for who wins. 53 return instance_ap.release(); 54 } 55 } 56 // The default implementation just tries to create debug information using the 57 // file representation for the module. 58 instance_ap.reset(new SymbolVendor(module_sp)); 59 if (instance_ap.get()) 60 { 61 ObjectFile *objfile = module_sp->GetObjectFile(); 62 if (objfile) 63 instance_ap->AddSymbolFileRepresentation(objfile->shared_from_this()); 64 } 65 return instance_ap.release(); 66 } 67 68 //---------------------------------------------------------------------- 69 // SymbolVendor constructor 70 //---------------------------------------------------------------------- 71 SymbolVendor::SymbolVendor(const lldb::ModuleSP &module_sp) : 72 ModuleChild (module_sp), 73 m_type_list(), 74 m_compile_units(), 75 m_sym_file_ap() 76 { 77 } 78 79 //---------------------------------------------------------------------- 80 // Destructor 81 //---------------------------------------------------------------------- 82 SymbolVendor::~SymbolVendor() 83 { 84 } 85 86 //---------------------------------------------------------------------- 87 // Add a represantion given an object file. 88 //---------------------------------------------------------------------- 89 void 90 SymbolVendor::AddSymbolFileRepresentation(const ObjectFileSP &objfile_sp) 91 { 92 ModuleSP module_sp(GetModule()); 93 if (module_sp) 94 { 95 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 96 if (objfile_sp) 97 { 98 m_objfile_sp = objfile_sp; 99 m_sym_file_ap.reset(SymbolFile::FindPlugin(objfile_sp.get())); 100 } 101 } 102 } 103 104 bool 105 SymbolVendor::SetCompileUnitAtIndex (size_t idx, const CompUnitSP &cu_sp) 106 { 107 ModuleSP module_sp(GetModule()); 108 if (module_sp) 109 { 110 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 111 const size_t num_compile_units = GetNumCompileUnits(); 112 if (idx < num_compile_units) 113 { 114 // Fire off an assertion if this compile unit already exists for now. 115 // The partial parsing should take care of only setting the compile 116 // unit once, so if this assertion fails, we need to make sure that 117 // we don't have a race condition, or have a second parse of the same 118 // compile unit. 119 assert(m_compile_units[idx].get() == NULL); 120 m_compile_units[idx] = cu_sp; 121 return true; 122 } 123 else 124 { 125 // This should NOT happen, and if it does, we want to crash and know 126 // about it 127 assert (idx < num_compile_units); 128 } 129 } 130 return false; 131 } 132 133 size_t 134 SymbolVendor::GetNumCompileUnits() 135 { 136 ModuleSP module_sp(GetModule()); 137 if (module_sp) 138 { 139 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 140 if (m_compile_units.empty()) 141 { 142 if (m_sym_file_ap.get()) 143 { 144 // Resize our array of compile unit shared pointers -- which will 145 // each remain NULL until someone asks for the actual compile unit 146 // information. When this happens, the symbol file will be asked 147 // to parse this compile unit information. 148 m_compile_units.resize(m_sym_file_ap->GetNumCompileUnits()); 149 } 150 } 151 } 152 return m_compile_units.size(); 153 } 154 155 lldb::LanguageType 156 SymbolVendor::ParseCompileUnitLanguage (const SymbolContext& sc) 157 { 158 ModuleSP module_sp(GetModule()); 159 if (module_sp) 160 { 161 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 162 if (m_sym_file_ap.get()) 163 return m_sym_file_ap->ParseCompileUnitLanguage(sc); 164 } 165 return eLanguageTypeUnknown; 166 } 167 168 169 size_t 170 SymbolVendor::ParseCompileUnitFunctions (const SymbolContext &sc) 171 { 172 ModuleSP module_sp(GetModule()); 173 if (module_sp) 174 { 175 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 176 if (m_sym_file_ap.get()) 177 return m_sym_file_ap->ParseCompileUnitFunctions(sc); 178 } 179 return 0; 180 } 181 182 bool 183 SymbolVendor::ParseCompileUnitLineTable (const SymbolContext &sc) 184 { 185 ModuleSP module_sp(GetModule()); 186 if (module_sp) 187 { 188 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 189 if (m_sym_file_ap.get()) 190 return m_sym_file_ap->ParseCompileUnitLineTable(sc); 191 } 192 return false; 193 } 194 195 bool 196 SymbolVendor::ParseCompileUnitSupportFiles (const SymbolContext& sc, FileSpecList& support_files) 197 { 198 ModuleSP module_sp(GetModule()); 199 if (module_sp) 200 { 201 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 202 if (m_sym_file_ap.get()) 203 return m_sym_file_ap->ParseCompileUnitSupportFiles(sc, support_files); 204 } 205 return false; 206 } 207 208 size_t 209 SymbolVendor::ParseFunctionBlocks (const SymbolContext &sc) 210 { 211 ModuleSP module_sp(GetModule()); 212 if (module_sp) 213 { 214 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 215 if (m_sym_file_ap.get()) 216 return m_sym_file_ap->ParseFunctionBlocks(sc); 217 } 218 return 0; 219 } 220 221 size_t 222 SymbolVendor::ParseTypes (const SymbolContext &sc) 223 { 224 ModuleSP module_sp(GetModule()); 225 if (module_sp) 226 { 227 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 228 if (m_sym_file_ap.get()) 229 return m_sym_file_ap->ParseTypes(sc); 230 } 231 return 0; 232 } 233 234 size_t 235 SymbolVendor::ParseVariablesForContext (const SymbolContext& sc) 236 { 237 ModuleSP module_sp(GetModule()); 238 if (module_sp) 239 { 240 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 241 if (m_sym_file_ap.get()) 242 return m_sym_file_ap->ParseVariablesForContext(sc); 243 } 244 return 0; 245 } 246 247 Type* 248 SymbolVendor::ResolveTypeUID(lldb::user_id_t type_uid) 249 { 250 ModuleSP module_sp(GetModule()); 251 if (module_sp) 252 { 253 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 254 if (m_sym_file_ap.get()) 255 return m_sym_file_ap->ResolveTypeUID(type_uid); 256 } 257 return NULL; 258 } 259 260 261 uint32_t 262 SymbolVendor::ResolveSymbolContext (const Address& so_addr, uint32_t resolve_scope, SymbolContext& sc) 263 { 264 ModuleSP module_sp(GetModule()); 265 if (module_sp) 266 { 267 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 268 if (m_sym_file_ap.get()) 269 return m_sym_file_ap->ResolveSymbolContext(so_addr, resolve_scope, sc); 270 } 271 return 0; 272 } 273 274 uint32_t 275 SymbolVendor::ResolveSymbolContext (const FileSpec& file_spec, uint32_t line, bool check_inlines, uint32_t resolve_scope, SymbolContextList& sc_list) 276 { 277 ModuleSP module_sp(GetModule()); 278 if (module_sp) 279 { 280 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 281 if (m_sym_file_ap.get()) 282 return m_sym_file_ap->ResolveSymbolContext(file_spec, line, check_inlines, resolve_scope, sc_list); 283 } 284 return 0; 285 } 286 287 size_t 288 SymbolVendor::FindGlobalVariables (const ConstString &name, const ClangNamespaceDecl *namespace_decl, bool append, size_t max_matches, VariableList& variables) 289 { 290 ModuleSP module_sp(GetModule()); 291 if (module_sp) 292 { 293 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 294 if (m_sym_file_ap.get()) 295 return m_sym_file_ap->FindGlobalVariables(name, namespace_decl, append, max_matches, variables); 296 } 297 return 0; 298 } 299 300 size_t 301 SymbolVendor::FindGlobalVariables (const RegularExpression& regex, bool append, size_t max_matches, VariableList& variables) 302 { 303 ModuleSP module_sp(GetModule()); 304 if (module_sp) 305 { 306 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 307 if (m_sym_file_ap.get()) 308 return m_sym_file_ap->FindGlobalVariables(regex, append, max_matches, variables); 309 } 310 return 0; 311 } 312 313 size_t 314 SymbolVendor::FindFunctions(const ConstString &name, const ClangNamespaceDecl *namespace_decl, uint32_t name_type_mask, bool include_inlines, bool append, SymbolContextList& sc_list) 315 { 316 ModuleSP module_sp(GetModule()); 317 if (module_sp) 318 { 319 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 320 if (m_sym_file_ap.get()) 321 return m_sym_file_ap->FindFunctions(name, namespace_decl, name_type_mask, include_inlines, append, sc_list); 322 } 323 return 0; 324 } 325 326 size_t 327 SymbolVendor::FindFunctions(const RegularExpression& regex, bool include_inlines, bool append, SymbolContextList& sc_list) 328 { 329 ModuleSP module_sp(GetModule()); 330 if (module_sp) 331 { 332 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 333 if (m_sym_file_ap.get()) 334 return m_sym_file_ap->FindFunctions(regex, include_inlines, append, sc_list); 335 } 336 return 0; 337 } 338 339 340 size_t 341 SymbolVendor::FindTypes (const SymbolContext& sc, const ConstString &name, const ClangNamespaceDecl *namespace_decl, bool append, size_t max_matches, TypeList& types) 342 { 343 ModuleSP module_sp(GetModule()); 344 if (module_sp) 345 { 346 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 347 if (m_sym_file_ap.get()) 348 return m_sym_file_ap->FindTypes(sc, name, namespace_decl, append, max_matches, types); 349 } 350 if (!append) 351 types.Clear(); 352 return 0; 353 } 354 355 ClangNamespaceDecl 356 SymbolVendor::FindNamespace(const SymbolContext& sc, const ConstString &name, const ClangNamespaceDecl *parent_namespace_decl) 357 { 358 ClangNamespaceDecl namespace_decl; 359 ModuleSP module_sp(GetModule()); 360 if (module_sp) 361 { 362 lldb_private::Mutex::Locker locker(module_sp->GetMutex()); 363 if (m_sym_file_ap.get()) 364 namespace_decl = m_sym_file_ap->FindNamespace (sc, name, parent_namespace_decl); 365 } 366 return namespace_decl; 367 } 368 369 void 370 SymbolVendor::Dump(Stream *s) 371 { 372 ModuleSP module_sp(GetModule()); 373 if (module_sp) 374 { 375 bool show_context = false; 376 377 s->Printf("%p: ", this); 378 s->Indent(); 379 s->PutCString("SymbolVendor"); 380 if (m_sym_file_ap.get()) 381 { 382 ObjectFile *objfile = m_sym_file_ap->GetObjectFile(); 383 if (objfile) 384 { 385 const FileSpec &objfile_file_spec = objfile->GetFileSpec(); 386 if (objfile_file_spec) 387 { 388 s->PutCString(" ("); 389 objfile_file_spec.Dump(s); 390 s->PutChar(')'); 391 } 392 } 393 } 394 s->EOL(); 395 s->IndentMore(); 396 m_type_list.Dump(s, show_context); 397 398 CompileUnitConstIter cu_pos, cu_end; 399 cu_end = m_compile_units.end(); 400 for (cu_pos = m_compile_units.begin(); cu_pos != cu_end; ++cu_pos) 401 { 402 // We currently only dump the compile units that have been parsed 403 if (cu_pos->get()) 404 (*cu_pos)->Dump(s, show_context); 405 } 406 407 s->IndentLess(); 408 } 409 } 410 411 CompUnitSP 412 SymbolVendor::GetCompileUnitAtIndex(size_t idx) 413 { 414 CompUnitSP cu_sp; 415 ModuleSP module_sp(GetModule()); 416 if (module_sp) 417 { 418 const size_t num_compile_units = GetNumCompileUnits(); 419 if (idx < num_compile_units) 420 { 421 cu_sp = m_compile_units[idx]; 422 if (cu_sp.get() == NULL) 423 { 424 m_compile_units[idx] = m_sym_file_ap->ParseCompileUnitAtIndex(idx); 425 cu_sp = m_compile_units[idx]; 426 } 427 } 428 } 429 return cu_sp; 430 } 431 432 433 //------------------------------------------------------------------ 434 // PluginInterface protocol 435 //------------------------------------------------------------------ 436 const char * 437 SymbolVendor::GetPluginName() 438 { 439 return "SymbolVendor"; 440 } 441 442 const char * 443 SymbolVendor::GetShortPluginName() 444 { 445 return "vendor-default"; 446 } 447 448 uint32_t 449 SymbolVendor::GetPluginVersion() 450 { 451 return 1; 452 } 453 454