156 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			156 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			C++
		
	
	
	
//===-- SymbolVendorELF.cpp -----------------------------------------------===//
 | 
						|
//
 | 
						|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
 | 
						|
// See https://llvm.org/LICENSE.txt for license information.
 | 
						|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#include "SymbolVendorELF.h"
 | 
						|
 | 
						|
#include <string.h>
 | 
						|
 | 
						|
#include "Plugins/ObjectFile/ELF/ObjectFileELF.h"
 | 
						|
#include "lldb/Core/Module.h"
 | 
						|
#include "lldb/Core/ModuleSpec.h"
 | 
						|
#include "lldb/Core/PluginManager.h"
 | 
						|
#include "lldb/Core/Section.h"
 | 
						|
#include "lldb/Host/Host.h"
 | 
						|
#include "lldb/Symbol/LocateSymbolFile.h"
 | 
						|
#include "lldb/Symbol/ObjectFile.h"
 | 
						|
#include "lldb/Target/Target.h"
 | 
						|
#include "lldb/Utility/StreamString.h"
 | 
						|
#include "lldb/Utility/Timer.h"
 | 
						|
 | 
						|
using namespace lldb;
 | 
						|
using namespace lldb_private;
 | 
						|
 | 
						|
LLDB_PLUGIN_DEFINE(SymbolVendorELF)
 | 
						|
 | 
						|
// SymbolVendorELF constructor
 | 
						|
SymbolVendorELF::SymbolVendorELF(const lldb::ModuleSP &module_sp)
 | 
						|
    : SymbolVendor(module_sp) {}
 | 
						|
 | 
						|
// Destructor
 | 
						|
SymbolVendorELF::~SymbolVendorELF() {}
 | 
						|
 | 
						|
void SymbolVendorELF::Initialize() {
 | 
						|
  PluginManager::RegisterPlugin(GetPluginNameStatic(),
 | 
						|
                                GetPluginDescriptionStatic(), CreateInstance);
 | 
						|
}
 | 
						|
 | 
						|
void SymbolVendorELF::Terminate() {
 | 
						|
  PluginManager::UnregisterPlugin(CreateInstance);
 | 
						|
}
 | 
						|
 | 
						|
lldb_private::ConstString SymbolVendorELF::GetPluginNameStatic() {
 | 
						|
  static ConstString g_name("ELF");
 | 
						|
  return g_name;
 | 
						|
}
 | 
						|
 | 
						|
const char *SymbolVendorELF::GetPluginDescriptionStatic() {
 | 
						|
  return "Symbol vendor for ELF that looks for dSYM files that match "
 | 
						|
         "executables.";
 | 
						|
}
 | 
						|
 | 
						|
// CreateInstance
 | 
						|
//
 | 
						|
// Platforms can register a callback to use when creating symbol vendors to
 | 
						|
// allow for complex debug information file setups, and to also allow for
 | 
						|
// finding separate debug information files.
 | 
						|
SymbolVendor *
 | 
						|
SymbolVendorELF::CreateInstance(const lldb::ModuleSP &module_sp,
 | 
						|
                                lldb_private::Stream *feedback_strm) {
 | 
						|
  if (!module_sp)
 | 
						|
    return nullptr;
 | 
						|
 | 
						|
  ObjectFileELF *obj_file =
 | 
						|
      llvm::dyn_cast_or_null<ObjectFileELF>(module_sp->GetObjectFile());
 | 
						|
  if (!obj_file)
 | 
						|
    return nullptr;
 | 
						|
 | 
						|
  lldb_private::UUID uuid = obj_file->GetUUID();
 | 
						|
  if (!uuid)
 | 
						|
    return nullptr;
 | 
						|
 | 
						|
  // If the main object file already contains debug info, then we are done.
 | 
						|
  if (obj_file->GetSectionList()->FindSectionByType(
 | 
						|
          lldb::eSectionTypeDWARFDebugInfo, true))
 | 
						|
    return nullptr;
 | 
						|
 | 
						|
  // If the module specified a filespec, use that.
 | 
						|
  FileSpec fspec = module_sp->GetSymbolFileFileSpec();
 | 
						|
  // Otherwise, try gnu_debuglink, if one exists.
 | 
						|
  if (!fspec)
 | 
						|
    fspec = obj_file->GetDebugLink().getValueOr(FileSpec());
 | 
						|
 | 
						|
  static Timer::Category func_cat(LLVM_PRETTY_FUNCTION);
 | 
						|
  Timer scoped_timer(func_cat, "SymbolVendorELF::CreateInstance (module = %s)",
 | 
						|
                     module_sp->GetFileSpec().GetPath().c_str());
 | 
						|
 | 
						|
  ModuleSpec module_spec;
 | 
						|
 | 
						|
  module_spec.GetFileSpec() = obj_file->GetFileSpec();
 | 
						|
  FileSystem::Instance().Resolve(module_spec.GetFileSpec());
 | 
						|
  module_spec.GetSymbolFileSpec() = fspec;
 | 
						|
  module_spec.GetUUID() = uuid;
 | 
						|
  FileSpecList search_paths = Target::GetDefaultDebugFileSearchPaths();
 | 
						|
  FileSpec dsym_fspec =
 | 
						|
      Symbols::LocateExecutableSymbolFile(module_spec, search_paths);
 | 
						|
  if (!dsym_fspec)
 | 
						|
    return nullptr;
 | 
						|
 | 
						|
  DataBufferSP dsym_file_data_sp;
 | 
						|
  lldb::offset_t dsym_file_data_offset = 0;
 | 
						|
  ObjectFileSP dsym_objfile_sp = ObjectFile::FindPlugin(
 | 
						|
      module_sp, &dsym_fspec, 0, FileSystem::Instance().GetByteSize(dsym_fspec),
 | 
						|
      dsym_file_data_sp, dsym_file_data_offset);
 | 
						|
  if (!dsym_objfile_sp)
 | 
						|
    return nullptr;
 | 
						|
 | 
						|
  // This objfile is for debugging purposes. Sadly, ObjectFileELF won't
 | 
						|
  // be able to figure this out consistently as the symbol file may not
 | 
						|
  // have stripped the code sections, etc.
 | 
						|
  dsym_objfile_sp->SetType(ObjectFile::eTypeDebugInfo);
 | 
						|
 | 
						|
  SymbolVendorELF *symbol_vendor = new SymbolVendorELF(module_sp);
 | 
						|
 | 
						|
  // Get the module unified section list and add our debug sections to
 | 
						|
  // that.
 | 
						|
  SectionList *module_section_list = module_sp->GetSectionList();
 | 
						|
  SectionList *objfile_section_list = dsym_objfile_sp->GetSectionList();
 | 
						|
 | 
						|
  static const SectionType g_sections[] = {
 | 
						|
      eSectionTypeDWARFDebugAbbrev,     eSectionTypeDWARFDebugAddr,
 | 
						|
      eSectionTypeDWARFDebugAranges,    eSectionTypeDWARFDebugCuIndex,
 | 
						|
      eSectionTypeDWARFDebugFrame,      eSectionTypeDWARFDebugInfo,
 | 
						|
      eSectionTypeDWARFDebugLine,       eSectionTypeDWARFDebugLineStr,
 | 
						|
      eSectionTypeDWARFDebugLoc,        eSectionTypeDWARFDebugLocLists,
 | 
						|
      eSectionTypeDWARFDebugMacInfo,    eSectionTypeDWARFDebugMacro,
 | 
						|
      eSectionTypeDWARFDebugNames,      eSectionTypeDWARFDebugPubNames,
 | 
						|
      eSectionTypeDWARFDebugPubTypes,   eSectionTypeDWARFDebugRanges,
 | 
						|
      eSectionTypeDWARFDebugRngLists,   eSectionTypeDWARFDebugStr,
 | 
						|
      eSectionTypeDWARFDebugStrOffsets, eSectionTypeDWARFDebugTypes,
 | 
						|
      eSectionTypeELFSymbolTable,       eSectionTypeDWARFGNUDebugAltLink,
 | 
						|
  };
 | 
						|
  for (SectionType section_type : g_sections) {
 | 
						|
    if (SectionSP section_sp =
 | 
						|
            objfile_section_list->FindSectionByType(section_type, true)) {
 | 
						|
      if (SectionSP module_section_sp =
 | 
						|
              module_section_list->FindSectionByType(section_type, true))
 | 
						|
        module_section_list->ReplaceSection(module_section_sp->GetID(),
 | 
						|
                                            section_sp);
 | 
						|
      else
 | 
						|
        module_section_list->AddSection(section_sp);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  symbol_vendor->AddSymbolFileRepresentation(dsym_objfile_sp);
 | 
						|
  return symbol_vendor;
 | 
						|
}
 | 
						|
 | 
						|
// PluginInterface protocol
 | 
						|
ConstString SymbolVendorELF::GetPluginName() { return GetPluginNameStatic(); }
 | 
						|
 | 
						|
uint32_t SymbolVendorELF::GetPluginVersion() { return 1; }
 |