[NFC][AArch64] Explicitly define undefined bits for instructions (#122129)
[llvm-project.git] / lldb / source / Plugins / Platform / MacOSX / PlatformDarwin.cpp
blob2a36f95c94d0ce42fb595e9d8e923b2e3d80b505
1 //===-- PlatformDarwin.cpp ------------------------------------------------===//
2 //
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
6 //
7 //===----------------------------------------------------------------------===//
9 #include "PlatformDarwin.h"
11 #include <cstring>
13 #include <algorithm>
14 #include <memory>
15 #include <mutex>
16 #include <optional>
18 #include "lldb/Breakpoint/BreakpointLocation.h"
19 #include "lldb/Breakpoint/BreakpointSite.h"
20 #include "lldb/Core/Debugger.h"
21 #include "lldb/Core/Module.h"
22 #include "lldb/Core/ModuleSpec.h"
23 #include "lldb/Core/PluginManager.h"
24 #include "lldb/Core/Section.h"
25 #include "lldb/Host/Host.h"
26 #include "lldb/Host/HostInfo.h"
27 #include "lldb/Host/XML.h"
28 #include "lldb/Interpreter/CommandInterpreter.h"
29 #include "lldb/Interpreter/OptionValueProperties.h"
30 #include "lldb/Interpreter/OptionValueString.h"
31 #include "lldb/Interpreter/Options.h"
32 #include "lldb/Symbol/CompileUnit.h"
33 #include "lldb/Symbol/ObjectFile.h"
34 #include "lldb/Symbol/SymbolFile.h"
35 #include "lldb/Symbol/SymbolVendor.h"
36 #include "lldb/Target/Platform.h"
37 #include "lldb/Target/Process.h"
38 #include "lldb/Target/Target.h"
39 #include "lldb/Utility/LLDBLog.h"
40 #include "lldb/Utility/Log.h"
41 #include "lldb/Utility/ProcessInfo.h"
42 #include "lldb/Utility/Status.h"
43 #include "lldb/Utility/Timer.h"
44 #include "llvm/ADT/STLExtras.h"
45 #include "llvm/Support/Error.h"
46 #include "llvm/Support/FileSystem.h"
47 #include "llvm/Support/Threading.h"
48 #include "llvm/Support/VersionTuple.h"
50 #if defined(__APPLE__)
51 #include <TargetConditionals.h>
52 #endif
54 using namespace lldb;
55 using namespace lldb_private;
57 #define OPTTABLE_STR_TABLE_CODE
58 #include "clang/Driver/Options.inc"
59 #undef OPTTABLE_STR_TABLE_CODE
61 static Status ExceptionMaskValidator(const char *string, void *unused) {
62 Status error;
63 llvm::StringRef str_ref(string);
64 llvm::SmallVector<llvm::StringRef> candidates;
65 str_ref.split(candidates, '|');
66 for (auto candidate : candidates) {
67 if (!(candidate == "EXC_BAD_ACCESS"
68 || candidate == "EXC_BAD_INSTRUCTION"
69 || candidate == "EXC_ARITHMETIC"
70 || candidate == "EXC_RESOURCE"
71 || candidate == "EXC_GUARD"
72 || candidate == "EXC_SYSCALL")) {
73 error = Status::FromErrorStringWithFormat("invalid exception type: '%s'",
74 candidate.str().c_str());
75 return error;
78 return {};
81 /// Destructor.
82 ///
83 /// The destructor is virtual since this class is designed to be
84 /// inherited from by the plug-in instance.
85 PlatformDarwin::~PlatformDarwin() = default;
87 // Static Variables
88 static uint32_t g_initialize_count = 0;
90 void PlatformDarwin::Initialize() {
91 Platform::Initialize();
93 if (g_initialize_count++ == 0) {
94 PluginManager::RegisterPlugin(PlatformDarwin::GetPluginNameStatic(),
95 PlatformDarwin::GetDescriptionStatic(),
96 PlatformDarwin::CreateInstance,
97 PlatformDarwin::DebuggerInitialize);
101 void PlatformDarwin::Terminate() {
102 if (g_initialize_count > 0) {
103 if (--g_initialize_count == 0) {
104 PluginManager::UnregisterPlugin(PlatformDarwin::CreateInstance);
108 Platform::Terminate();
111 llvm::StringRef PlatformDarwin::GetDescriptionStatic() {
112 return "Darwin platform plug-in.";
115 PlatformSP PlatformDarwin::CreateInstance(bool force, const ArchSpec *arch) {
116 // We only create subclasses of the PlatformDarwin plugin.
117 return PlatformSP();
120 #define LLDB_PROPERTIES_platformdarwin
121 #include "PlatformMacOSXProperties.inc"
123 #define LLDB_PROPERTIES_platformdarwin
124 enum {
125 #include "PlatformMacOSXPropertiesEnum.inc"
128 class PlatformDarwinProperties : public Properties {
129 public:
130 static llvm::StringRef GetSettingName() {
131 static constexpr llvm::StringLiteral g_setting_name("darwin");
132 return g_setting_name;
135 PlatformDarwinProperties() : Properties() {
136 m_collection_sp = std::make_shared<OptionValueProperties>(GetSettingName());
137 m_collection_sp->Initialize(g_platformdarwin_properties);
140 ~PlatformDarwinProperties() override = default;
142 const char *GetIgnoredExceptions() const {
143 const uint32_t idx = ePropertyIgnoredExceptions;
144 const OptionValueString *option_value =
145 m_collection_sp->GetPropertyAtIndexAsOptionValueString(idx);
146 assert(option_value);
147 return option_value->GetCurrentValue();
150 OptionValueString *GetIgnoredExceptionValue() {
151 const uint32_t idx = ePropertyIgnoredExceptions;
152 OptionValueString *option_value =
153 m_collection_sp->GetPropertyAtIndexAsOptionValueString(idx);
154 assert(option_value);
155 return option_value;
159 static PlatformDarwinProperties &GetGlobalProperties() {
160 static PlatformDarwinProperties g_settings;
161 return g_settings;
164 void PlatformDarwin::DebuggerInitialize(
165 lldb_private::Debugger &debugger) {
166 if (!PluginManager::GetSettingForPlatformPlugin(
167 debugger, PlatformDarwinProperties::GetSettingName())) {
168 const bool is_global_setting = false;
169 PluginManager::CreateSettingForPlatformPlugin(
170 debugger, GetGlobalProperties().GetValueProperties(),
171 "Properties for the Darwin platform plug-in.", is_global_setting);
172 OptionValueString *value = GetGlobalProperties().GetIgnoredExceptionValue();
173 value->SetValidator(ExceptionMaskValidator);
177 Args
178 PlatformDarwin::GetExtraStartupCommands() {
179 std::string ignored_exceptions
180 = GetGlobalProperties().GetIgnoredExceptions();
181 if (ignored_exceptions.empty())
182 return {};
183 Args ret_args;
184 std::string packet = "QSetIgnoredExceptions:";
185 packet.append(ignored_exceptions);
186 ret_args.AppendArgument(packet);
187 return ret_args;
190 lldb_private::Status
191 PlatformDarwin::PutFile(const lldb_private::FileSpec &source,
192 const lldb_private::FileSpec &destination, uint32_t uid,
193 uint32_t gid) {
194 // Unconditionally unlink the destination. If it is an executable,
195 // simply opening it and truncating its contents would invalidate
196 // its cached code signature.
197 Unlink(destination);
198 return PlatformPOSIX::PutFile(source, destination, uid, gid);
201 FileSpecList PlatformDarwin::LocateExecutableScriptingResources(
202 Target *target, Module &module, Stream &feedback_stream) {
203 FileSpecList file_list;
204 if (target &&
205 target->GetDebugger().GetScriptLanguage() == eScriptLanguagePython) {
206 // NB some extensions might be meaningful and should not be stripped -
207 // "this.binary.file"
208 // should not lose ".file" but GetFileNameStrippingExtension() will do
209 // precisely that. Ideally, we should have a per-platform list of
210 // extensions (".exe", ".app", ".dSYM", ".framework") which should be
211 // stripped while leaving "this.binary.file" as-is.
213 FileSpec module_spec = module.GetFileSpec();
215 if (module_spec) {
216 if (SymbolFile *symfile = module.GetSymbolFile()) {
217 ObjectFile *objfile = symfile->GetObjectFile();
218 if (objfile) {
219 FileSpec symfile_spec(objfile->GetFileSpec());
220 if (symfile_spec &&
221 llvm::StringRef(symfile_spec.GetPath())
222 .contains_insensitive(".dSYM/Contents/Resources/DWARF") &&
223 FileSystem::Instance().Exists(symfile_spec)) {
224 while (module_spec.GetFilename()) {
225 std::string module_basename(
226 module_spec.GetFilename().GetCString());
227 std::string original_module_basename(module_basename);
229 bool was_keyword = false;
231 // FIXME: for Python, we cannot allow certain characters in
232 // module
233 // filenames we import. Theoretically, different scripting
234 // languages may have different sets of forbidden tokens in
235 // filenames, and that should be dealt with by each
236 // ScriptInterpreter. For now, we just replace dots with
237 // underscores, but if we ever support anything other than
238 // Python we will need to rework this
239 std::replace(module_basename.begin(), module_basename.end(), '.',
240 '_');
241 std::replace(module_basename.begin(), module_basename.end(), ' ',
242 '_');
243 std::replace(module_basename.begin(), module_basename.end(), '-',
244 '_');
245 ScriptInterpreter *script_interpreter =
246 target->GetDebugger().GetScriptInterpreter();
247 if (script_interpreter &&
248 script_interpreter->IsReservedWord(module_basename.c_str())) {
249 module_basename.insert(module_basename.begin(), '_');
250 was_keyword = true;
253 StreamString path_string;
254 StreamString original_path_string;
255 // for OSX we are going to be in
256 // .dSYM/Contents/Resources/DWARF/<basename> let us go to
257 // .dSYM/Contents/Resources/Python/<basename>.py and see if the
258 // file exists
259 path_string.Printf("%s/../Python/%s.py",
260 symfile_spec.GetDirectory().GetCString(),
261 module_basename.c_str());
262 original_path_string.Printf(
263 "%s/../Python/%s.py",
264 symfile_spec.GetDirectory().GetCString(),
265 original_module_basename.c_str());
266 FileSpec script_fspec(path_string.GetString());
267 FileSystem::Instance().Resolve(script_fspec);
268 FileSpec orig_script_fspec(original_path_string.GetString());
269 FileSystem::Instance().Resolve(orig_script_fspec);
271 // if we did some replacements of reserved characters, and a
272 // file with the untampered name exists, then warn the user
273 // that the file as-is shall not be loaded
274 if (module_basename != original_module_basename &&
275 FileSystem::Instance().Exists(orig_script_fspec)) {
276 const char *reason_for_complaint =
277 was_keyword ? "conflicts with a keyword"
278 : "contains reserved characters";
279 if (FileSystem::Instance().Exists(script_fspec))
280 feedback_stream.Printf(
281 "warning: the symbol file '%s' contains a debug "
282 "script. However, its name"
283 " '%s' %s and as such cannot be loaded. LLDB will"
284 " load '%s' instead. Consider removing the file with "
285 "the malformed name to"
286 " eliminate this warning.\n",
287 symfile_spec.GetPath().c_str(),
288 original_path_string.GetData(), reason_for_complaint,
289 path_string.GetData());
290 else
291 feedback_stream.Printf(
292 "warning: the symbol file '%s' contains a debug "
293 "script. However, its name"
294 " %s and as such cannot be loaded. If you intend"
295 " to have this script loaded, please rename '%s' to "
296 "'%s' and retry.\n",
297 symfile_spec.GetPath().c_str(), reason_for_complaint,
298 original_path_string.GetData(), path_string.GetData());
301 if (FileSystem::Instance().Exists(script_fspec)) {
302 file_list.Append(script_fspec);
303 break;
306 // If we didn't find the python file, then keep stripping the
307 // extensions and try again
308 ConstString filename_no_extension(
309 module_spec.GetFileNameStrippingExtension());
310 if (module_spec.GetFilename() == filename_no_extension)
311 break;
313 module_spec.SetFilename(filename_no_extension);
320 return file_list;
323 Status PlatformDarwin::ResolveSymbolFile(Target &target,
324 const ModuleSpec &sym_spec,
325 FileSpec &sym_file) {
326 sym_file = sym_spec.GetSymbolFileSpec();
327 if (FileSystem::Instance().IsDirectory(sym_file)) {
328 sym_file = PluginManager::FindSymbolFileInBundle(
329 sym_file, sym_spec.GetUUIDPtr(), sym_spec.GetArchitecturePtr());
331 return {};
334 Status PlatformDarwin::GetSharedModule(
335 const ModuleSpec &module_spec, Process *process, ModuleSP &module_sp,
336 const FileSpecList *module_search_paths_ptr,
337 llvm::SmallVectorImpl<ModuleSP> *old_modules, bool *did_create_ptr) {
338 Status error;
339 module_sp.reset();
341 if (IsRemote()) {
342 // If we have a remote platform always, let it try and locate the shared
343 // module first.
344 if (m_remote_platform_sp) {
345 error = m_remote_platform_sp->GetSharedModule(
346 module_spec, process, module_sp, module_search_paths_ptr, old_modules,
347 did_create_ptr);
351 if (!module_sp) {
352 // Fall back to the local platform and find the file locally
353 error = Platform::GetSharedModule(module_spec, process, module_sp,
354 module_search_paths_ptr, old_modules,
355 did_create_ptr);
357 const FileSpec &platform_file = module_spec.GetFileSpec();
358 if (!module_sp && module_search_paths_ptr && platform_file) {
359 // We can try to pull off part of the file path up to the bundle
360 // directory level and try any module search paths...
361 FileSpec bundle_directory;
362 if (Host::GetBundleDirectory(platform_file, bundle_directory)) {
363 if (platform_file == bundle_directory) {
364 ModuleSpec new_module_spec(module_spec);
365 new_module_spec.GetFileSpec() = bundle_directory;
366 if (Host::ResolveExecutableInBundle(new_module_spec.GetFileSpec())) {
367 Status new_error(Platform::GetSharedModule(
368 new_module_spec, process, module_sp, nullptr, old_modules,
369 did_create_ptr));
371 if (module_sp)
372 return new_error;
374 } else {
375 char platform_path[PATH_MAX];
376 char bundle_dir[PATH_MAX];
377 platform_file.GetPath(platform_path, sizeof(platform_path));
378 const size_t bundle_directory_len =
379 bundle_directory.GetPath(bundle_dir, sizeof(bundle_dir));
380 char new_path[PATH_MAX];
381 size_t num_module_search_paths = module_search_paths_ptr->GetSize();
382 for (size_t i = 0; i < num_module_search_paths; ++i) {
383 const size_t search_path_len =
384 module_search_paths_ptr->GetFileSpecAtIndex(i).GetPath(
385 new_path, sizeof(new_path));
386 if (search_path_len < sizeof(new_path)) {
387 snprintf(new_path + search_path_len,
388 sizeof(new_path) - search_path_len, "/%s",
389 platform_path + bundle_directory_len);
390 FileSpec new_file_spec(new_path);
391 if (FileSystem::Instance().Exists(new_file_spec)) {
392 ModuleSpec new_module_spec(module_spec);
393 new_module_spec.GetFileSpec() = new_file_spec;
394 Status new_error(Platform::GetSharedModule(
395 new_module_spec, process, module_sp, nullptr, old_modules,
396 did_create_ptr));
398 if (module_sp) {
399 module_sp->SetPlatformFileSpec(new_file_spec);
400 return new_error;
409 if (module_sp)
410 module_sp->SetPlatformFileSpec(module_spec.GetFileSpec());
411 return error;
414 size_t
415 PlatformDarwin::GetSoftwareBreakpointTrapOpcode(Target &target,
416 BreakpointSite *bp_site) {
417 const uint8_t *trap_opcode = nullptr;
418 uint32_t trap_opcode_size = 0;
419 bool bp_is_thumb = false;
421 llvm::Triple::ArchType machine = target.GetArchitecture().GetMachine();
422 switch (machine) {
423 case llvm::Triple::aarch64_32:
424 case llvm::Triple::aarch64: {
425 // 'brk #0' or 0xd4200000 in BE byte order
426 static const uint8_t g_arm64_breakpoint_opcode[] = {0x00, 0x00, 0x20, 0xD4};
427 trap_opcode = g_arm64_breakpoint_opcode;
428 trap_opcode_size = sizeof(g_arm64_breakpoint_opcode);
429 } break;
431 case llvm::Triple::thumb:
432 bp_is_thumb = true;
433 [[fallthrough]];
434 case llvm::Triple::arm: {
435 static const uint8_t g_arm_breakpoint_opcode[] = {0xFE, 0xDE, 0xFF, 0xE7};
436 static const uint8_t g_thumb_breakpooint_opcode[] = {0xFE, 0xDE};
438 // Auto detect arm/thumb if it wasn't explicitly specified
439 if (!bp_is_thumb) {
440 lldb::BreakpointLocationSP bp_loc_sp(bp_site->GetConstituentAtIndex(0));
441 if (bp_loc_sp)
442 bp_is_thumb = bp_loc_sp->GetAddress().GetAddressClass() ==
443 AddressClass::eCodeAlternateISA;
445 if (bp_is_thumb) {
446 trap_opcode = g_thumb_breakpooint_opcode;
447 trap_opcode_size = sizeof(g_thumb_breakpooint_opcode);
448 break;
450 trap_opcode = g_arm_breakpoint_opcode;
451 trap_opcode_size = sizeof(g_arm_breakpoint_opcode);
452 } break;
454 case llvm::Triple::ppc:
455 case llvm::Triple::ppc64: {
456 static const uint8_t g_ppc_breakpoint_opcode[] = {0x7F, 0xC0, 0x00, 0x08};
457 trap_opcode = g_ppc_breakpoint_opcode;
458 trap_opcode_size = sizeof(g_ppc_breakpoint_opcode);
459 } break;
461 default:
462 return Platform::GetSoftwareBreakpointTrapOpcode(target, bp_site);
465 if (trap_opcode && trap_opcode_size) {
466 if (bp_site->SetTrapOpcode(trap_opcode, trap_opcode_size))
467 return trap_opcode_size;
469 return 0;
472 bool PlatformDarwin::ModuleIsExcludedForUnconstrainedSearches(
473 lldb_private::Target &target, const lldb::ModuleSP &module_sp) {
474 if (!module_sp)
475 return false;
477 ObjectFile *obj_file = module_sp->GetObjectFile();
478 if (!obj_file)
479 return false;
481 ObjectFile::Type obj_type = obj_file->GetType();
482 return obj_type == ObjectFile::eTypeDynamicLinker;
485 void PlatformDarwin::x86GetSupportedArchitectures(
486 std::vector<ArchSpec> &archs) {
487 ArchSpec host_arch = HostInfo::GetArchitecture(HostInfo::eArchKindDefault);
488 archs.push_back(host_arch);
490 if (host_arch.GetCore() == ArchSpec::eCore_x86_64_x86_64h) {
491 archs.push_back(ArchSpec("x86_64-apple-macosx"));
492 archs.push_back(HostInfo::GetArchitecture(HostInfo::eArchKind32));
493 } else {
494 ArchSpec host_arch64 = HostInfo::GetArchitecture(HostInfo::eArchKind64);
495 if (host_arch.IsExactMatch(host_arch64))
496 archs.push_back(HostInfo::GetArchitecture(HostInfo::eArchKind32));
500 static llvm::ArrayRef<const char *> GetCompatibleArchs(ArchSpec::Core core) {
501 switch (core) {
502 default:
503 [[fallthrough]];
504 case ArchSpec::eCore_arm_arm64e: {
505 static const char *g_arm64e_compatible_archs[] = {
506 "arm64e", "arm64", "armv7", "armv7f", "armv7k", "armv7s",
507 "armv7m", "armv7em", "armv6m", "armv6", "armv5", "armv4",
508 "arm", "thumbv7", "thumbv7f", "thumbv7k", "thumbv7s", "thumbv7m",
509 "thumbv7em", "thumbv6m", "thumbv6", "thumbv5", "thumbv4t", "thumb",
511 return {g_arm64e_compatible_archs};
513 case ArchSpec::eCore_arm_arm64: {
514 static const char *g_arm64_compatible_archs[] = {
515 "arm64", "armv7", "armv7f", "armv7k", "armv7s", "armv7m",
516 "armv7em", "armv6m", "armv6", "armv5", "armv4", "arm",
517 "thumbv7", "thumbv7f", "thumbv7k", "thumbv7s", "thumbv7m", "thumbv7em",
518 "thumbv6m", "thumbv6", "thumbv5", "thumbv4t", "thumb",
520 return {g_arm64_compatible_archs};
522 case ArchSpec::eCore_arm_armv7: {
523 static const char *g_armv7_compatible_archs[] = {
524 "armv7", "armv6m", "armv6", "armv5", "armv4", "arm",
525 "thumbv7", "thumbv6m", "thumbv6", "thumbv5", "thumbv4t", "thumb",
527 return {g_armv7_compatible_archs};
529 case ArchSpec::eCore_arm_armv7f: {
530 static const char *g_armv7f_compatible_archs[] = {
531 "armv7f", "armv7", "armv6m", "armv6", "armv5",
532 "armv4", "arm", "thumbv7f", "thumbv7", "thumbv6m",
533 "thumbv6", "thumbv5", "thumbv4t", "thumb",
535 return {g_armv7f_compatible_archs};
537 case ArchSpec::eCore_arm_armv7k: {
538 static const char *g_armv7k_compatible_archs[] = {
539 "armv7k", "armv7", "armv6m", "armv6", "armv5",
540 "armv4", "arm", "thumbv7k", "thumbv7", "thumbv6m",
541 "thumbv6", "thumbv5", "thumbv4t", "thumb",
543 return {g_armv7k_compatible_archs};
545 case ArchSpec::eCore_arm_armv7s: {
546 static const char *g_armv7s_compatible_archs[] = {
547 "armv7s", "armv7", "armv6m", "armv6", "armv5",
548 "armv4", "arm", "thumbv7s", "thumbv7", "thumbv6m",
549 "thumbv6", "thumbv5", "thumbv4t", "thumb",
551 return {g_armv7s_compatible_archs};
553 case ArchSpec::eCore_arm_armv7m: {
554 static const char *g_armv7m_compatible_archs[] = {
555 "armv7m", "armv7", "armv6m", "armv6", "armv5",
556 "armv4", "arm", "thumbv7m", "thumbv7", "thumbv6m",
557 "thumbv6", "thumbv5", "thumbv4t", "thumb",
559 return {g_armv7m_compatible_archs};
561 case ArchSpec::eCore_arm_armv7em: {
562 static const char *g_armv7em_compatible_archs[] = {
563 "armv7em", "armv7", "armv6m", "armv6", "armv5",
564 "armv4", "arm", "thumbv7em", "thumbv7", "thumbv6m",
565 "thumbv6", "thumbv5", "thumbv4t", "thumb",
567 return {g_armv7em_compatible_archs};
569 case ArchSpec::eCore_arm_armv6m: {
570 static const char *g_armv6m_compatible_archs[] = {
571 "armv6m", "armv6", "armv5", "armv4", "arm",
572 "thumbv6m", "thumbv6", "thumbv5", "thumbv4t", "thumb",
574 return {g_armv6m_compatible_archs};
576 case ArchSpec::eCore_arm_armv6: {
577 static const char *g_armv6_compatible_archs[] = {
578 "armv6", "armv5", "armv4", "arm",
579 "thumbv6", "thumbv5", "thumbv4t", "thumb",
581 return {g_armv6_compatible_archs};
583 case ArchSpec::eCore_arm_armv5: {
584 static const char *g_armv5_compatible_archs[] = {
585 "armv5", "armv4", "arm", "thumbv5", "thumbv4t", "thumb",
587 return {g_armv5_compatible_archs};
589 case ArchSpec::eCore_arm_armv4: {
590 static const char *g_armv4_compatible_archs[] = {
591 "armv4",
592 "arm",
593 "thumbv4t",
594 "thumb",
596 return {g_armv4_compatible_archs};
599 return {};
602 /// The architecture selection rules for arm processors These cpu subtypes have
603 /// distinct names (e.g. armv7f) but armv7 binaries run fine on an armv7f
604 /// processor.
605 void PlatformDarwin::ARMGetSupportedArchitectures(
606 std::vector<ArchSpec> &archs, std::optional<llvm::Triple::OSType> os) {
607 const ArchSpec system_arch = GetSystemArchitecture();
608 const ArchSpec::Core system_core = system_arch.GetCore();
609 for (const char *arch : GetCompatibleArchs(system_core)) {
610 llvm::Triple triple;
611 triple.setArchName(arch);
612 triple.setVendor(llvm::Triple::VendorType::Apple);
613 if (os)
614 triple.setOS(*os);
615 archs.push_back(ArchSpec(triple));
619 static FileSpec GetXcodeSelectPath() {
620 static FileSpec g_xcode_select_filespec;
622 if (!g_xcode_select_filespec) {
623 FileSpec xcode_select_cmd("/usr/bin/xcode-select");
624 if (FileSystem::Instance().Exists(xcode_select_cmd)) {
625 int exit_status = -1;
626 int signo = -1;
627 std::string command_output;
628 Status status =
629 Host::RunShellCommand("/usr/bin/xcode-select --print-path",
630 FileSpec(), // current working directory
631 &exit_status, &signo, &command_output,
632 std::chrono::seconds(2), // short timeout
633 false); // don't run in a shell
634 if (status.Success() && exit_status == 0 && !command_output.empty()) {
635 size_t first_non_newline = command_output.find_last_not_of("\r\n");
636 if (first_non_newline != std::string::npos) {
637 command_output.erase(first_non_newline + 1);
639 g_xcode_select_filespec = FileSpec(command_output);
644 return g_xcode_select_filespec;
647 BreakpointSP PlatformDarwin::SetThreadCreationBreakpoint(Target &target) {
648 BreakpointSP bp_sp;
649 static const char *g_bp_names[] = {
650 "start_wqthread", "_pthread_wqthread", "_pthread_start",
653 static const char *g_bp_modules[] = {"libsystem_c.dylib",
654 "libSystem.B.dylib"};
656 FileSpecList bp_modules;
657 for (size_t i = 0; i < std::size(g_bp_modules); i++) {
658 const char *bp_module = g_bp_modules[i];
659 bp_modules.EmplaceBack(bp_module);
662 bool internal = true;
663 bool hardware = false;
664 LazyBool skip_prologue = eLazyBoolNo;
665 bp_sp = target.CreateBreakpoint(&bp_modules, nullptr, g_bp_names,
666 std::size(g_bp_names), eFunctionNameTypeFull,
667 eLanguageTypeUnknown, 0, skip_prologue,
668 internal, hardware);
669 bp_sp->SetBreakpointKind("thread-creation");
671 return bp_sp;
674 uint32_t
675 PlatformDarwin::GetResumeCountForLaunchInfo(ProcessLaunchInfo &launch_info) {
676 const FileSpec &shell = launch_info.GetShell();
677 if (!shell)
678 return 1;
680 std::string shell_string = shell.GetPath();
681 const char *shell_name = strrchr(shell_string.c_str(), '/');
682 if (shell_name == nullptr)
683 shell_name = shell_string.c_str();
684 else
685 shell_name++;
687 if (strcmp(shell_name, "sh") == 0) {
688 // /bin/sh re-exec's itself as /bin/bash requiring another resume. But it
689 // only does this if the COMMAND_MODE environment variable is set to
690 // "legacy".
691 if (launch_info.GetEnvironment().lookup("COMMAND_MODE") == "legacy")
692 return 2;
693 return 1;
694 } else if (strcmp(shell_name, "csh") == 0 ||
695 strcmp(shell_name, "tcsh") == 0 ||
696 strcmp(shell_name, "zsh") == 0) {
697 // csh and tcsh always seem to re-exec themselves.
698 return 2;
699 } else
700 return 1;
703 lldb::ProcessSP PlatformDarwin::DebugProcess(ProcessLaunchInfo &launch_info,
704 Debugger &debugger, Target &target,
705 Status &error) {
706 ProcessSP process_sp;
708 if (IsHost()) {
709 // We are going to hand this process off to debugserver which will be in
710 // charge of setting the exit status. However, we still need to reap it
711 // from lldb. So, make sure we use a exit callback which does not set exit
712 // status.
713 launch_info.SetMonitorProcessCallback(
714 &ProcessLaunchInfo::NoOpMonitorCallback);
715 process_sp = Platform::DebugProcess(launch_info, debugger, target, error);
716 } else {
717 if (m_remote_platform_sp)
718 process_sp = m_remote_platform_sp->DebugProcess(launch_info, debugger,
719 target, error);
720 else
721 error =
722 Status::FromErrorString("the platform is not currently connected");
724 return process_sp;
727 void PlatformDarwin::CalculateTrapHandlerSymbolNames() {
728 m_trap_handlers.push_back(ConstString("_sigtramp"));
731 static FileSpec GetCommandLineToolsLibraryPath() {
732 static FileSpec g_command_line_tools_filespec;
734 if (!g_command_line_tools_filespec) {
735 FileSpec command_line_tools_path(GetXcodeSelectPath());
736 command_line_tools_path.AppendPathComponent("Library");
737 if (FileSystem::Instance().Exists(command_line_tools_path)) {
738 g_command_line_tools_filespec = command_line_tools_path;
742 return g_command_line_tools_filespec;
745 FileSystem::EnumerateDirectoryResult PlatformDarwin::DirectoryEnumerator(
746 void *baton, llvm::sys::fs::file_type file_type, llvm::StringRef path) {
747 SDKEnumeratorInfo *enumerator_info = static_cast<SDKEnumeratorInfo *>(baton);
749 FileSpec spec(path);
750 if (XcodeSDK::SDKSupportsModules(enumerator_info->sdk_type, spec)) {
751 enumerator_info->found_path = spec;
752 return FileSystem::EnumerateDirectoryResult::eEnumerateDirectoryResultNext;
755 return FileSystem::EnumerateDirectoryResult::eEnumerateDirectoryResultNext;
758 FileSpec PlatformDarwin::FindSDKInXcodeForModules(XcodeSDK::Type sdk_type,
759 const FileSpec &sdks_spec) {
760 // Look inside Xcode for the required installed iOS SDK version
762 if (!FileSystem::Instance().IsDirectory(sdks_spec)) {
763 return FileSpec();
766 const bool find_directories = true;
767 const bool find_files = false;
768 const bool find_other = true; // include symlinks
770 SDKEnumeratorInfo enumerator_info;
772 enumerator_info.sdk_type = sdk_type;
774 FileSystem::Instance().EnumerateDirectory(
775 sdks_spec.GetPath(), find_directories, find_files, find_other,
776 DirectoryEnumerator, &enumerator_info);
778 if (FileSystem::Instance().IsDirectory(enumerator_info.found_path))
779 return enumerator_info.found_path;
780 else
781 return FileSpec();
784 FileSpec PlatformDarwin::GetSDKDirectoryForModules(XcodeSDK::Type sdk_type) {
785 FileSpec sdks_spec = HostInfo::GetXcodeContentsDirectory();
786 sdks_spec.AppendPathComponent("Developer");
787 sdks_spec.AppendPathComponent("Platforms");
789 switch (sdk_type) {
790 case XcodeSDK::Type::MacOSX:
791 sdks_spec.AppendPathComponent("MacOSX.platform");
792 break;
793 case XcodeSDK::Type::iPhoneSimulator:
794 sdks_spec.AppendPathComponent("iPhoneSimulator.platform");
795 break;
796 case XcodeSDK::Type::iPhoneOS:
797 sdks_spec.AppendPathComponent("iPhoneOS.platform");
798 break;
799 case XcodeSDK::Type::WatchSimulator:
800 sdks_spec.AppendPathComponent("WatchSimulator.platform");
801 break;
802 case XcodeSDK::Type::AppleTVSimulator:
803 sdks_spec.AppendPathComponent("AppleTVSimulator.platform");
804 break;
805 case XcodeSDK::Type::XRSimulator:
806 sdks_spec.AppendPathComponent("XRSimulator.platform");
807 break;
808 default:
809 llvm_unreachable("unsupported sdk");
812 sdks_spec.AppendPathComponent("Developer");
813 sdks_spec.AppendPathComponent("SDKs");
815 if (sdk_type == XcodeSDK::Type::MacOSX) {
816 llvm::VersionTuple version = HostInfo::GetOSVersion();
818 if (!version.empty()) {
819 if (XcodeSDK::SDKSupportsModules(XcodeSDK::Type::MacOSX, version)) {
820 // If the Xcode SDKs are not available then try to use the
821 // Command Line Tools one which is only for MacOSX.
822 if (!FileSystem::Instance().Exists(sdks_spec)) {
823 sdks_spec = GetCommandLineToolsLibraryPath();
824 sdks_spec.AppendPathComponent("SDKs");
827 // We slightly prefer the exact SDK for this machine. See if it is
828 // there.
830 FileSpec native_sdk_spec = sdks_spec;
831 StreamString native_sdk_name;
832 native_sdk_name.Printf("MacOSX%u.%u.sdk", version.getMajor(),
833 version.getMinor().value_or(0));
834 native_sdk_spec.AppendPathComponent(native_sdk_name.GetString());
836 if (FileSystem::Instance().Exists(native_sdk_spec)) {
837 return native_sdk_spec;
843 return FindSDKInXcodeForModules(sdk_type, sdks_spec);
846 std::tuple<llvm::VersionTuple, llvm::StringRef>
847 PlatformDarwin::ParseVersionBuildDir(llvm::StringRef dir) {
848 llvm::StringRef build;
849 llvm::StringRef version_str;
850 llvm::StringRef build_str;
851 std::tie(version_str, build_str) = dir.split(' ');
852 llvm::VersionTuple version;
853 if (!version.tryParse(version_str) ||
854 build_str.empty()) {
855 if (build_str.consume_front("(")) {
856 size_t pos = build_str.find(')');
857 build = build_str.slice(0, pos);
861 return std::make_tuple(version, build);
864 llvm::Expected<StructuredData::DictionarySP>
865 PlatformDarwin::FetchExtendedCrashInformation(Process &process) {
866 static constexpr llvm::StringLiteral crash_info_key("Crash-Info Annotations");
867 static constexpr llvm::StringLiteral asi_info_key(
868 "Application Specific Information");
870 // We cache the information we find in the process extended info dict:
871 StructuredData::DictionarySP process_dict_sp =
872 process.GetExtendedCrashInfoDict();
873 StructuredData::Array *annotations = nullptr;
874 StructuredData::ArraySP new_annotations_sp;
875 if (!process_dict_sp->GetValueForKeyAsArray(crash_info_key, annotations)) {
876 new_annotations_sp = ExtractCrashInfoAnnotations(process);
877 if (new_annotations_sp && new_annotations_sp->GetSize()) {
878 process_dict_sp->AddItem(crash_info_key, new_annotations_sp);
879 annotations = new_annotations_sp.get();
883 StructuredData::Dictionary *app_specific_info;
884 StructuredData::DictionarySP new_app_specific_info_sp;
885 if (!process_dict_sp->GetValueForKeyAsDictionary(asi_info_key,
886 app_specific_info)) {
887 new_app_specific_info_sp = ExtractAppSpecificInfo(process);
888 if (new_app_specific_info_sp && new_app_specific_info_sp->GetSize()) {
889 process_dict_sp->AddItem(asi_info_key, new_app_specific_info_sp);
890 app_specific_info = new_app_specific_info_sp.get();
894 // Now get anything else that was in the process info dict, and add it to the
895 // return here:
896 return process_dict_sp->GetSize() ? process_dict_sp : nullptr;
899 StructuredData::ArraySP
900 PlatformDarwin::ExtractCrashInfoAnnotations(Process &process) {
901 Log *log = GetLog(LLDBLog::Process);
903 ConstString section_name("__crash_info");
904 Target &target = process.GetTarget();
905 StructuredData::ArraySP array_sp = std::make_shared<StructuredData::Array>();
907 for (ModuleSP module : target.GetImages().Modules()) {
908 SectionList *sections = module->GetSectionList();
910 std::string module_name = module->GetSpecificationDescription();
912 // The DYDL module is skipped since it's always loaded when running the
913 // binary.
914 if (module_name == "/usr/lib/dyld")
915 continue;
917 if (!sections) {
918 LLDB_LOG(log, "Module {0} doesn't have any section!", module_name);
919 continue;
922 SectionSP crash_info = sections->FindSectionByName(section_name);
923 if (!crash_info) {
924 LLDB_LOG(log, "Module {0} doesn't have section {1}!", module_name,
925 section_name);
926 continue;
929 addr_t load_addr = crash_info->GetLoadBaseAddress(&target);
931 if (load_addr == LLDB_INVALID_ADDRESS) {
932 LLDB_LOG(log, "Module {0} has an invalid '{1}' section load address: {2}",
933 module_name, section_name, load_addr);
934 continue;
937 Status error;
938 CrashInfoAnnotations annotations;
939 size_t expected_size = sizeof(CrashInfoAnnotations);
940 size_t bytes_read = process.ReadMemoryFromInferior(load_addr, &annotations,
941 expected_size, error);
943 if (expected_size != bytes_read || error.Fail()) {
944 LLDB_LOG(log, "Failed to read {0} section from memory in module {1}: {2}",
945 section_name, module_name, error);
946 continue;
949 // initial support added for version 5
950 if (annotations.version < 5) {
951 LLDB_LOG(log,
952 "Annotation version lower than 5 unsupported! Module {0} has "
953 "version {1} instead.",
954 module_name, annotations.version);
955 continue;
958 if (!annotations.message) {
959 LLDB_LOG(log, "No message available for module {0}.", module_name);
960 continue;
963 std::string message;
964 bytes_read =
965 process.ReadCStringFromMemory(annotations.message, message, error);
967 if (message.empty() || bytes_read != message.size() || error.Fail()) {
968 LLDB_LOG(log, "Failed to read the message from memory in module {0}: {1}",
969 module_name, error);
970 continue;
973 // Remove trailing newline from message
974 if (message.back() == '\n')
975 message.pop_back();
977 if (!annotations.message2)
978 LLDB_LOG(log, "No message2 available for module {0}.", module_name);
980 std::string message2;
981 bytes_read =
982 process.ReadCStringFromMemory(annotations.message2, message2, error);
984 if (!message2.empty() && bytes_read == message2.size() && error.Success())
985 if (message2.back() == '\n')
986 message2.pop_back();
988 StructuredData::DictionarySP entry_sp =
989 std::make_shared<StructuredData::Dictionary>();
991 entry_sp->AddStringItem("image", module->GetFileSpec().GetPath(false));
992 entry_sp->AddStringItem("uuid", module->GetUUID().GetAsString());
993 entry_sp->AddStringItem("message", message);
994 entry_sp->AddStringItem("message2", message2);
995 entry_sp->AddIntegerItem("abort-cause", annotations.abort_cause);
997 array_sp->AddItem(entry_sp);
1000 return array_sp;
1003 StructuredData::DictionarySP
1004 PlatformDarwin::ExtractAppSpecificInfo(Process &process) {
1005 StructuredData::DictionarySP metadata_sp = process.GetMetadata();
1007 if (!metadata_sp || !metadata_sp->GetSize() || !metadata_sp->HasKey("asi"))
1008 return {};
1010 StructuredData::Dictionary *asi;
1011 if (!metadata_sp->GetValueForKeyAsDictionary("asi", asi))
1012 return {};
1014 StructuredData::DictionarySP dict_sp =
1015 std::make_shared<StructuredData::Dictionary>();
1017 auto flatten_asi_dict = [&dict_sp](llvm::StringRef key,
1018 StructuredData::Object *val) -> bool {
1019 if (!val)
1020 return false;
1022 StructuredData::Array *arr = val->GetAsArray();
1023 if (!arr || !arr->GetSize())
1024 return false;
1026 dict_sp->AddItem(key, arr->GetItemAtIndex(0));
1027 return true;
1030 asi->ForEach(flatten_asi_dict);
1032 return dict_sp;
1035 void PlatformDarwin::AddClangModuleCompilationOptionsForSDKType(
1036 Target *target, std::vector<std::string> &options, XcodeSDK::Type sdk_type) {
1037 const std::vector<std::string> apple_arguments = {
1038 "-x", "objective-c++", "-fobjc-arc",
1039 "-fblocks", "-D_ISO646_H", "-D__ISO646_H",
1040 "-fgnuc-version=4.2.1"};
1042 options.insert(options.end(), apple_arguments.begin(), apple_arguments.end());
1044 StreamString minimum_version_option;
1045 bool use_current_os_version = false;
1046 // If the SDK type is for the host OS, use its version number.
1047 auto get_host_os = []() { return HostInfo::GetTargetTriple().getOS(); };
1048 switch (sdk_type) {
1049 case XcodeSDK::Type::MacOSX:
1050 use_current_os_version = get_host_os() == llvm::Triple::MacOSX;
1051 break;
1052 case XcodeSDK::Type::iPhoneOS:
1053 use_current_os_version = get_host_os() == llvm::Triple::IOS;
1054 break;
1055 case XcodeSDK::Type::AppleTVOS:
1056 use_current_os_version = get_host_os() == llvm::Triple::TvOS;
1057 break;
1058 case XcodeSDK::Type::watchOS:
1059 use_current_os_version = get_host_os() == llvm::Triple::WatchOS;
1060 break;
1061 case XcodeSDK::Type::XROS:
1062 use_current_os_version = get_host_os() == llvm::Triple::XROS;
1063 break;
1064 default:
1065 break;
1068 llvm::VersionTuple version;
1069 if (use_current_os_version)
1070 version = GetOSVersion();
1071 else if (target) {
1072 // Our OS doesn't match our executable so we need to get the min OS version
1073 // from the object file
1074 ModuleSP exe_module_sp = target->GetExecutableModule();
1075 if (exe_module_sp) {
1076 ObjectFile *object_file = exe_module_sp->GetObjectFile();
1077 if (object_file)
1078 version = object_file->GetMinimumOSVersion();
1081 // Only add the version-min options if we got a version from somewhere.
1082 // clang has no version-min clang flag for XROS.
1083 if (!version.empty() && sdk_type != XcodeSDK::Type::Linux &&
1084 sdk_type != XcodeSDK::Type::XROS) {
1085 #define OPTION(PREFIX_OFFSET, NAME_OFFSET, VAR, ...) \
1086 llvm::StringRef opt_##VAR = &OptionStrTable[NAME_OFFSET]; \
1087 (void)opt_##VAR;
1088 #include "clang/Driver/Options.inc"
1089 #undef OPTION
1090 minimum_version_option << '-';
1091 switch (sdk_type) {
1092 case XcodeSDK::Type::MacOSX:
1093 minimum_version_option << opt_mmacos_version_min_EQ;
1094 break;
1095 case XcodeSDK::Type::iPhoneSimulator:
1096 minimum_version_option << opt_mios_simulator_version_min_EQ;
1097 break;
1098 case XcodeSDK::Type::iPhoneOS:
1099 minimum_version_option << opt_mios_version_min_EQ;
1100 break;
1101 case XcodeSDK::Type::AppleTVSimulator:
1102 minimum_version_option << opt_mtvos_simulator_version_min_EQ;
1103 break;
1104 case XcodeSDK::Type::AppleTVOS:
1105 minimum_version_option << opt_mtvos_version_min_EQ;
1106 break;
1107 case XcodeSDK::Type::WatchSimulator:
1108 minimum_version_option << opt_mwatchos_simulator_version_min_EQ;
1109 break;
1110 case XcodeSDK::Type::watchOS:
1111 minimum_version_option << opt_mwatchos_version_min_EQ;
1112 break;
1113 case XcodeSDK::Type::XRSimulator:
1114 case XcodeSDK::Type::XROS:
1115 // FIXME: Pass the right argument once it exists.
1116 case XcodeSDK::Type::bridgeOS:
1117 case XcodeSDK::Type::Linux:
1118 case XcodeSDK::Type::unknown:
1119 if (Log *log = GetLog(LLDBLog::Host)) {
1120 XcodeSDK::Info info;
1121 info.type = sdk_type;
1122 LLDB_LOGF(log, "Clang modules on %s are not supported",
1123 XcodeSDK::GetCanonicalName(info).c_str());
1125 return;
1127 minimum_version_option << version.getAsString();
1128 options.emplace_back(std::string(minimum_version_option.GetString()));
1131 FileSpec sysroot_spec;
1133 if (target) {
1134 if (ModuleSP exe_module_sp = target->GetExecutableModule()) {
1135 auto path_or_err = ResolveSDKPathFromDebugInfo(*exe_module_sp);
1136 if (path_or_err) {
1137 sysroot_spec = FileSpec(*path_or_err);
1138 } else {
1139 LLDB_LOG_ERROR(GetLog(LLDBLog::Types | LLDBLog::Host),
1140 path_or_err.takeError(),
1141 "Failed to resolve SDK path: {0}");
1146 if (!FileSystem::Instance().IsDirectory(sysroot_spec.GetPath())) {
1147 std::lock_guard<std::mutex> guard(m_mutex);
1148 sysroot_spec = GetSDKDirectoryForModules(sdk_type);
1151 if (FileSystem::Instance().IsDirectory(sysroot_spec.GetPath())) {
1152 options.push_back("-isysroot");
1153 options.push_back(sysroot_spec.GetPath());
1157 ConstString PlatformDarwin::GetFullNameForDylib(ConstString basename) {
1158 if (basename.IsEmpty())
1159 return basename;
1161 StreamString stream;
1162 stream.Printf("lib%s.dylib", basename.GetCString());
1163 return ConstString(stream.GetString());
1166 llvm::VersionTuple PlatformDarwin::GetOSVersion(Process *process) {
1167 if (process && GetPluginName().contains("-simulator")) {
1168 lldb_private::ProcessInstanceInfo proc_info;
1169 if (Host::GetProcessInfo(process->GetID(), proc_info)) {
1170 const Environment &env = proc_info.GetEnvironment();
1172 llvm::VersionTuple result;
1173 if (!result.tryParse(env.lookup("SIMULATOR_RUNTIME_VERSION")))
1174 return result;
1176 std::string dyld_root_path = env.lookup("DYLD_ROOT_PATH");
1177 if (!dyld_root_path.empty()) {
1178 dyld_root_path += "/System/Library/CoreServices/SystemVersion.plist";
1179 ApplePropertyList system_version_plist(dyld_root_path.c_str());
1180 std::string product_version;
1181 if (system_version_plist.GetValueAsString("ProductVersion",
1182 product_version)) {
1183 if (!result.tryParse(product_version))
1184 return result;
1188 // For simulator platforms, do NOT call back through
1189 // Platform::GetOSVersion() as it might call Process::GetHostOSVersion()
1190 // which we don't want as it will be incorrect
1191 return llvm::VersionTuple();
1194 return Platform::GetOSVersion(process);
1197 lldb_private::FileSpec PlatformDarwin::LocateExecutable(const char *basename) {
1198 // A collection of SBFileSpec whose SBFileSpec.m_directory members are filled
1199 // in with any executable directories that should be searched.
1200 static std::vector<FileSpec> g_executable_dirs;
1202 // Find the global list of directories that we will search for executables
1203 // once so we don't keep doing the work over and over.
1204 static llvm::once_flag g_once_flag;
1205 llvm::call_once(g_once_flag, []() {
1207 // When locating executables, trust the DEVELOPER_DIR first if it is set
1208 FileSpec xcode_contents_dir = HostInfo::GetXcodeContentsDirectory();
1209 if (xcode_contents_dir) {
1210 FileSpec xcode_lldb_resources = xcode_contents_dir;
1211 xcode_lldb_resources.AppendPathComponent("SharedFrameworks");
1212 xcode_lldb_resources.AppendPathComponent("LLDB.framework");
1213 xcode_lldb_resources.AppendPathComponent("Resources");
1214 if (FileSystem::Instance().Exists(xcode_lldb_resources)) {
1215 FileSpec dir;
1216 dir.SetDirectory(xcode_lldb_resources.GetPathAsConstString());
1217 g_executable_dirs.push_back(dir);
1220 // Xcode might not be installed so we also check for the Command Line Tools.
1221 FileSpec command_line_tools_dir = GetCommandLineToolsLibraryPath();
1222 if (command_line_tools_dir) {
1223 FileSpec cmd_line_lldb_resources = command_line_tools_dir;
1224 cmd_line_lldb_resources.AppendPathComponent("PrivateFrameworks");
1225 cmd_line_lldb_resources.AppendPathComponent("LLDB.framework");
1226 cmd_line_lldb_resources.AppendPathComponent("Resources");
1227 if (FileSystem::Instance().Exists(cmd_line_lldb_resources)) {
1228 FileSpec dir;
1229 dir.SetDirectory(cmd_line_lldb_resources.GetPathAsConstString());
1230 g_executable_dirs.push_back(dir);
1235 // Now search the global list of executable directories for the executable we
1236 // are looking for
1237 for (const auto &executable_dir : g_executable_dirs) {
1238 FileSpec executable_file;
1239 executable_file.SetDirectory(executable_dir.GetDirectory());
1240 executable_file.SetFilename(basename);
1241 if (FileSystem::Instance().Exists(executable_file))
1242 return executable_file;
1245 return FileSpec();
1248 lldb_private::Status
1249 PlatformDarwin::LaunchProcess(lldb_private::ProcessLaunchInfo &launch_info) {
1250 // Starting in Fall 2016 OSes, NSLog messages only get mirrored to stderr if
1251 // the OS_ACTIVITY_DT_MODE environment variable is set. (It doesn't require
1252 // any specific value; rather, it just needs to exist). We will set it here
1253 // as long as the IDE_DISABLED_OS_ACTIVITY_DT_MODE flag is not set. Xcode
1254 // makes use of IDE_DISABLED_OS_ACTIVITY_DT_MODE to tell
1255 // LLDB *not* to muck with the OS_ACTIVITY_DT_MODE flag when they
1256 // specifically want it unset.
1257 const char *disable_env_var = "IDE_DISABLED_OS_ACTIVITY_DT_MODE";
1258 auto &env_vars = launch_info.GetEnvironment();
1259 if (!env_vars.count(disable_env_var)) {
1260 // We want to make sure that OS_ACTIVITY_DT_MODE is set so that we get
1261 // os_log and NSLog messages mirrored to the target process stderr.
1262 env_vars.try_emplace("OS_ACTIVITY_DT_MODE", "enable");
1265 // Let our parent class do the real launching.
1266 return PlatformPOSIX::LaunchProcess(launch_info);
1269 lldb_private::Status PlatformDarwin::FindBundleBinaryInExecSearchPaths(
1270 const ModuleSpec &module_spec, Process *process, ModuleSP &module_sp,
1271 const FileSpecList *module_search_paths_ptr,
1272 llvm::SmallVectorImpl<ModuleSP> *old_modules, bool *did_create_ptr) {
1273 const FileSpec &platform_file = module_spec.GetFileSpec();
1274 // See if the file is present in any of the module_search_paths_ptr
1275 // directories.
1276 if (!module_sp && module_search_paths_ptr && platform_file) {
1277 // create a vector of all the file / directory names in platform_file e.g.
1278 // this might be
1279 // /System/Library/PrivateFrameworks/UIFoundation.framework/UIFoundation
1281 // We'll need to look in the module_search_paths_ptr directories for both
1282 // "UIFoundation" and "UIFoundation.framework" -- most likely the latter
1283 // will be the one we find there.
1285 std::vector<llvm::StringRef> path_parts = platform_file.GetComponents();
1286 // We want the components in reverse order.
1287 std::reverse(path_parts.begin(), path_parts.end());
1288 const size_t path_parts_size = path_parts.size();
1290 size_t num_module_search_paths = module_search_paths_ptr->GetSize();
1291 for (size_t i = 0; i < num_module_search_paths; ++i) {
1292 Log *log_verbose = GetLog(LLDBLog::Host);
1293 LLDB_LOGF(
1294 log_verbose,
1295 "PlatformRemoteDarwinDevice::GetSharedModule searching for binary in "
1296 "search-path %s",
1297 module_search_paths_ptr->GetFileSpecAtIndex(i).GetPath().c_str());
1298 // Create a new FileSpec with this module_search_paths_ptr plus just the
1299 // filename ("UIFoundation"), then the parent dir plus filename
1300 // ("UIFoundation.framework/UIFoundation") etc - up to four names (to
1301 // handle "Foo.framework/Contents/MacOS/Foo")
1303 for (size_t j = 0; j < 4 && j < path_parts_size - 1; ++j) {
1304 FileSpec path_to_try(module_search_paths_ptr->GetFileSpecAtIndex(i));
1306 // Add the components backwards. For
1307 // .../PrivateFrameworks/UIFoundation.framework/UIFoundation path_parts
1308 // is
1309 // [0] UIFoundation
1310 // [1] UIFoundation.framework
1311 // [2] PrivateFrameworks
1313 // and if 'j' is 2, we want to append path_parts[1] and then
1314 // path_parts[0], aka 'UIFoundation.framework/UIFoundation', to the
1315 // module_search_paths_ptr path.
1317 for (int k = j; k >= 0; --k) {
1318 path_to_try.AppendPathComponent(path_parts[k]);
1321 if (FileSystem::Instance().Exists(path_to_try)) {
1322 ModuleSpec new_module_spec(module_spec);
1323 new_module_spec.GetFileSpec() = path_to_try;
1324 Status new_error(
1325 Platform::GetSharedModule(new_module_spec, process, module_sp,
1326 nullptr, old_modules, did_create_ptr));
1328 if (module_sp) {
1329 module_sp->SetPlatformFileSpec(path_to_try);
1330 return new_error;
1336 return Status();
1339 std::string PlatformDarwin::FindComponentInPath(llvm::StringRef path,
1340 llvm::StringRef component) {
1341 auto begin = llvm::sys::path::begin(path);
1342 auto end = llvm::sys::path::end(path);
1343 for (auto it = begin; it != end; ++it) {
1344 if (it->contains(component)) {
1345 llvm::SmallString<128> buffer;
1346 llvm::sys::path::append(buffer, begin, ++it,
1347 llvm::sys::path::Style::posix);
1348 return buffer.str().str();
1351 return {};
1354 FileSpec PlatformDarwin::GetCurrentToolchainDirectory() {
1355 if (FileSpec fspec = HostInfo::GetShlibDir())
1356 return FileSpec(FindComponentInPath(fspec.GetPath(), ".xctoolchain"));
1357 return {};
1360 FileSpec PlatformDarwin::GetCurrentCommandLineToolsDirectory() {
1361 if (FileSpec fspec = HostInfo::GetShlibDir())
1362 return FileSpec(FindComponentInPath(fspec.GetPath(), "CommandLineTools"));
1363 return {};
1366 llvm::Triple::OSType PlatformDarwin::GetHostOSType() {
1367 #if !defined(__APPLE__)
1368 return llvm::Triple::MacOSX;
1369 #else
1370 #if TARGET_OS_OSX
1371 return llvm::Triple::MacOSX;
1372 #elif TARGET_OS_IOS
1373 return llvm::Triple::IOS;
1374 #elif TARGET_OS_WATCH
1375 return llvm::Triple::WatchOS;
1376 #elif TARGET_OS_TV
1377 return llvm::Triple::TvOS;
1378 #elif TARGET_OS_BRIDGE
1379 return llvm::Triple::BridgeOS;
1380 #elif TARGET_OS_XR
1381 return llvm::Triple::XROS;
1382 #else
1383 #error "LLDB being compiled for an unrecognized Darwin OS"
1384 #endif
1385 #endif // __APPLE__
1388 llvm::Expected<std::pair<XcodeSDK, bool>>
1389 PlatformDarwin::GetSDKPathFromDebugInfo(Module &module) {
1390 SymbolFile *sym_file = module.GetSymbolFile();
1391 if (!sym_file)
1392 return llvm::createStringError(
1393 llvm::inconvertibleErrorCode(),
1394 llvm::formatv("No symbol file available for module '{0}'",
1395 module.GetFileSpec().GetFilename().AsCString("")));
1397 bool found_public_sdk = false;
1398 bool found_internal_sdk = false;
1399 XcodeSDK merged_sdk;
1400 for (unsigned i = 0; i < sym_file->GetNumCompileUnits(); ++i) {
1401 if (auto cu_sp = sym_file->GetCompileUnitAtIndex(i)) {
1402 auto cu_sdk = sym_file->ParseXcodeSDK(*cu_sp);
1403 bool is_internal_sdk = cu_sdk.IsAppleInternalSDK();
1404 found_public_sdk |= !is_internal_sdk;
1405 found_internal_sdk |= is_internal_sdk;
1407 merged_sdk.Merge(cu_sdk);
1411 const bool found_mismatch = found_internal_sdk && found_public_sdk;
1413 return std::pair{std::move(merged_sdk), found_mismatch};
1416 llvm::Expected<std::string>
1417 PlatformDarwin::ResolveSDKPathFromDebugInfo(Module &module) {
1418 auto sdk_or_err = GetSDKPathFromDebugInfo(module);
1419 if (!sdk_or_err)
1420 return llvm::createStringError(
1421 llvm::inconvertibleErrorCode(),
1422 llvm::formatv("Failed to parse SDK path from debug-info: {0}",
1423 llvm::toString(sdk_or_err.takeError())));
1425 auto [sdk, _] = std::move(*sdk_or_err);
1427 auto path_or_err = HostInfo::GetSDKRoot(HostInfo::SDKOptions{sdk});
1428 if (!path_or_err)
1429 return llvm::createStringError(
1430 llvm::inconvertibleErrorCode(),
1431 llvm::formatv("Error while searching for SDK (XcodeSDK '{0}'): {1}",
1432 sdk.GetString(),
1433 llvm::toString(path_or_err.takeError())));
1435 return path_or_err->str();
1438 llvm::Expected<XcodeSDK>
1439 PlatformDarwin::GetSDKPathFromDebugInfo(CompileUnit &unit) {
1440 ModuleSP module_sp = unit.CalculateSymbolContextModule();
1441 if (!module_sp)
1442 return llvm::createStringError("compile unit has no module");
1443 SymbolFile *sym_file = module_sp->GetSymbolFile();
1444 if (!sym_file)
1445 return llvm::createStringError(
1446 llvm::formatv("No symbol file available for module '{0}'",
1447 module_sp->GetFileSpec().GetFilename()));
1449 return sym_file->ParseXcodeSDK(unit);
1452 llvm::Expected<std::string>
1453 PlatformDarwin::ResolveSDKPathFromDebugInfo(CompileUnit &unit) {
1454 auto sdk_or_err = GetSDKPathFromDebugInfo(unit);
1455 if (!sdk_or_err)
1456 return llvm::createStringError(
1457 llvm::inconvertibleErrorCode(),
1458 llvm::formatv("Failed to parse SDK path from debug-info: {0}",
1459 llvm::toString(sdk_or_err.takeError())));
1461 auto sdk = std::move(*sdk_or_err);
1463 auto path_or_err = HostInfo::GetSDKRoot(HostInfo::SDKOptions{sdk});
1464 if (!path_or_err)
1465 return llvm::createStringError(
1466 llvm::inconvertibleErrorCode(),
1467 llvm::formatv("Error while searching for SDK (XcodeSDK '{0}'): {1}",
1468 sdk.GetString(),
1469 llvm::toString(path_or_err.takeError())));
1471 return path_or_err->str();