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