1 //===-- SymbolVendorMacOSX.cpp --------------------------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #include "SymbolVendorMacOSX.h"
13 #include "Plugins/ObjectFile/Mach-O/ObjectFileMachO.h"
14 #include "lldb/Core/Module.h"
15 #include "lldb/Core/ModuleSpec.h"
16 #include "lldb/Core/PluginManager.h"
17 #include "lldb/Core/Section.h"
18 #include "lldb/Host/Host.h"
19 #include "lldb/Host/XML.h"
20 #include "lldb/Symbol/ObjectFile.h"
21 #include "lldb/Target/Target.h"
22 #include "lldb/Utility/StreamString.h"
23 #include "lldb/Utility/Timer.h"
26 using namespace lldb_private
;
28 LLDB_PLUGIN_DEFINE(SymbolVendorMacOSX
)
30 // SymbolVendorMacOSX constructor
31 SymbolVendorMacOSX::SymbolVendorMacOSX(const lldb::ModuleSP
&module_sp
)
32 : SymbolVendor(module_sp
) {}
34 static bool UUIDsMatch(Module
*module
, ObjectFile
*ofile
,
35 lldb_private::Stream
*feedback_strm
) {
36 if (module
&& ofile
) {
37 // Make sure the UUIDs match
38 lldb_private::UUID dsym_uuid
= ofile
->GetUUID();
41 feedback_strm
->PutCString(
42 "warning: failed to get the uuid for object file: '");
43 ofile
->GetFileSpec().Dump(feedback_strm
->AsRawOstream());
44 feedback_strm
->PutCString("\n");
49 if (dsym_uuid
== module
->GetUUID())
52 // Emit some warning messages since the UUIDs do not match!
54 feedback_strm
->PutCString(
55 "warning: UUID mismatch detected between modules:\n ");
56 module
->GetUUID().Dump(*feedback_strm
);
57 feedback_strm
->PutChar(' ');
58 module
->GetFileSpec().Dump(feedback_strm
->AsRawOstream());
59 feedback_strm
->PutCString("\n ");
60 dsym_uuid
.Dump(*feedback_strm
);
61 feedback_strm
->PutChar(' ');
62 ofile
->GetFileSpec().Dump(feedback_strm
->AsRawOstream());
69 void SymbolVendorMacOSX::Initialize() {
70 PluginManager::RegisterPlugin(GetPluginNameStatic(),
71 GetPluginDescriptionStatic(), CreateInstance
);
74 void SymbolVendorMacOSX::Terminate() {
75 PluginManager::UnregisterPlugin(CreateInstance
);
78 llvm::StringRef
SymbolVendorMacOSX::GetPluginDescriptionStatic() {
79 return "Symbol vendor for MacOSX that looks for dSYM files that match "
85 // Platforms can register a callback to use when creating symbol vendors to
86 // allow for complex debug information file setups, and to also allow for
87 // finding separate debug information files.
89 SymbolVendorMacOSX::CreateInstance(const lldb::ModuleSP
&module_sp
,
90 lldb_private::Stream
*feedback_strm
) {
94 ObjectFile
*obj_file
=
95 llvm::dyn_cast_or_null
<ObjectFileMachO
>(module_sp
->GetObjectFile());
99 static Timer::Category
func_cat(LLVM_PRETTY_FUNCTION
);
100 Timer
scoped_timer(func_cat
,
101 "SymbolVendorMacOSX::CreateInstance (module = %s)",
102 module_sp
->GetFileSpec().GetPath().c_str());
103 SymbolVendorMacOSX
*symbol_vendor
= new SymbolVendorMacOSX(module_sp
);
108 // Try and locate the dSYM file on Mac OS X
109 static Timer::Category
func_cat2(
110 "SymbolVendorMacOSX::CreateInstance() locate dSYM");
113 "SymbolVendorMacOSX::CreateInstance (module = %s) locate dSYM",
114 module_sp
->GetFileSpec().GetPath().c_str());
116 // First check to see if the module has a symbol file in mind already. If
117 // it does, then we MUST use that.
118 FileSpec
dsym_fspec(module_sp
->GetSymbolFileFileSpec());
120 ObjectFileSP dsym_objfile_sp
;
121 // On Darwin, we store the debug information either in object files,
122 // using the debug map to tie them to the executable, or in a dSYM. We
123 // pass through this routine both for binaries and for .o files, but in the
124 // latter case there will never be an external debug file. So we shouldn't
125 // do all the stats needed to find it.
126 if (!dsym_fspec
&& module_sp
->GetObjectFile()->CalculateType() !=
127 ObjectFile::eTypeObjectFile
) {
128 // No symbol file was specified in the module, lets try and find one
130 FileSpec file_spec
= obj_file
->GetFileSpec();
132 file_spec
= module_sp
->GetFileSpec();
134 ModuleSpec
module_spec(file_spec
, module_sp
->GetArchitecture());
135 module_spec
.GetUUID() = module_sp
->GetUUID();
136 FileSpecList search_paths
= Target::GetDefaultDebugFileSearchPaths();
138 PluginManager::LocateExecutableSymbolFile(module_spec
, search_paths
);
139 if (module_spec
.GetSourceMappingList().GetSize())
140 module_sp
->GetSourceMappingList().Append(
141 module_spec
.GetSourceMappingList(), true);
145 // Compute dSYM root.
146 std::string dsym_root
= dsym_fspec
.GetPath();
147 const size_t pos
= dsym_root
.find("/Contents/Resources/");
148 dsym_root
= pos
!= std::string::npos
? dsym_root
.substr(0, pos
) : "";
150 DataBufferSP dsym_file_data_sp
;
151 lldb::offset_t dsym_file_data_offset
= 0;
153 ObjectFile::FindPlugin(module_sp
, &dsym_fspec
, 0,
154 FileSystem::Instance().GetByteSize(dsym_fspec
),
155 dsym_file_data_sp
, dsym_file_data_offset
);
156 // Important to save the dSYM FileSpec so we don't call
157 // PluginManager::LocateExecutableSymbolFile a second time while trying to
158 // add the symbol ObjectFile to this Module.
159 if (dsym_objfile_sp
&& !module_sp
->GetSymbolFileFileSpec()) {
160 module_sp
->SetSymbolFileFileSpec(dsym_fspec
);
162 if (UUIDsMatch(module_sp
.get(), dsym_objfile_sp
.get(), feedback_strm
)) {
163 // We need a XML parser if we hope to parse a plist...
164 if (XMLDocument::XMLEnabled()) {
165 if (module_sp
->GetSourceMappingList().IsEmpty()) {
166 lldb_private::UUID dsym_uuid
= dsym_objfile_sp
->GetUUID();
168 std::string uuid_str
= dsym_uuid
.GetAsString();
169 if (!uuid_str
.empty() && !dsym_root
.empty()) {
170 char dsym_uuid_plist_path
[PATH_MAX
];
171 snprintf(dsym_uuid_plist_path
, sizeof(dsym_uuid_plist_path
),
172 "%s/Contents/Resources/%s.plist", dsym_root
.c_str(),
174 FileSpec
dsym_uuid_plist_spec(dsym_uuid_plist_path
);
175 if (FileSystem::Instance().Exists(dsym_uuid_plist_spec
)) {
176 ApplePropertyList
plist(dsym_uuid_plist_path
);
178 std::string DBGBuildSourcePath
;
179 std::string DBGSourcePath
;
181 // DBGSourcePathRemapping is a dictionary in the plist
182 // with keys which are DBGBuildSourcePath file paths and
183 // values which are DBGSourcePath file paths
185 StructuredData::ObjectSP plist_sp
=
186 plist
.GetStructuredData();
187 if (plist_sp
.get() && plist_sp
->GetAsDictionary() &&
188 plist_sp
->GetAsDictionary()->HasKey(
189 "DBGSourcePathRemapping") &&
190 plist_sp
->GetAsDictionary()
191 ->GetValueForKey("DBGSourcePathRemapping")
192 ->GetAsDictionary()) {
194 // If DBGVersion 1 or DBGVersion missing, ignore
195 // DBGSourcePathRemapping. If DBGVersion 2, strip last two
196 // components of path remappings from
197 // entries to fix an issue with a
198 // specific set of DBGSourcePathRemapping
199 // entries that lldb worked with.
200 // If DBGVersion 3, trust & use the source path remappings
204 bool new_style_source_remapping_dictionary
= false;
205 bool do_truncate_remapping_names
= false;
206 std::string original_DBGSourcePath_value
= DBGSourcePath
;
207 if (plist_sp
->GetAsDictionary()->HasKey("DBGVersion")) {
208 std::string version_string
=
209 std::string(plist_sp
->GetAsDictionary()
210 ->GetValueForKey("DBGVersion")
211 ->GetStringValue(""));
212 if (!version_string
.empty() &&
213 isdigit(version_string
[0])) {
214 int version_number
= atoi(version_string
.c_str());
215 if (version_number
> 1) {
216 new_style_source_remapping_dictionary
= true;
218 if (version_number
== 2) {
219 do_truncate_remapping_names
= true;
224 StructuredData::Dictionary
*remappings_dict
=
225 plist_sp
->GetAsDictionary()
226 ->GetValueForKey("DBGSourcePathRemapping")
228 remappings_dict
->ForEach(
229 [&module_sp
, new_style_source_remapping_dictionary
,
230 original_DBGSourcePath_value
,
231 do_truncate_remapping_names
](
233 StructuredData::Object
*object
) -> bool {
234 if (object
&& object
->GetAsString()) {
236 // key is DBGBuildSourcePath
237 // object is DBGSourcePath
238 std::string DBGSourcePath
=
239 std::string(object
->GetStringValue());
240 if (!new_style_source_remapping_dictionary
&&
241 !original_DBGSourcePath_value
.empty()) {
242 DBGSourcePath
= original_DBGSourcePath_value
;
244 module_sp
->GetSourceMappingList().Append(
245 key
, DBGSourcePath
, true);
246 // With version 2 of DBGSourcePathRemapping, we
247 // can chop off the last two filename parts
248 // from the source remapping and get a more
249 // general source remapping that still works.
250 // Add this as another option in addition to
251 // the full source path remap.
252 if (do_truncate_remapping_names
) {
253 FileSpec
build_path(key
);
254 FileSpec
source_path(DBGSourcePath
.c_str());
255 build_path
.RemoveLastPathComponent();
256 build_path
.RemoveLastPathComponent();
257 source_path
.RemoveLastPathComponent();
258 source_path
.RemoveLastPathComponent();
259 module_sp
->GetSourceMappingList().Append(
260 build_path
.GetPath(), source_path
.GetPath(),
268 // If we have a DBGBuildSourcePath + DBGSourcePath pair,
269 // append those to the source path remappings.
271 plist
.GetValueAsString("DBGBuildSourcePath",
273 plist
.GetValueAsString("DBGSourcePath", DBGSourcePath
);
274 if (!DBGBuildSourcePath
.empty() && !DBGSourcePath
.empty()) {
275 module_sp
->GetSourceMappingList().Append(
276 DBGBuildSourcePath
, DBGSourcePath
, true);
285 symbol_vendor
->AddSymbolFileRepresentation(dsym_objfile_sp
);
286 return symbol_vendor
;
290 // Just create our symbol vendor using the current objfile as this is
291 // either an executable with no dSYM (that we could locate), an executable
292 // with a dSYM that has a UUID that doesn't match.
293 symbol_vendor
->AddSymbolFileRepresentation(obj_file
->shared_from_this());
295 return symbol_vendor
;