diff --git a/lldb/source/API/SBTarget.cpp b/lldb/source/API/SBTarget.cpp
--- a/lldb/source/API/SBTarget.cpp
+++ b/lldb/source/API/SBTarget.cpp
@@ -7,6 +7,7 @@
//===----------------------------------------------------------------------===//
#include "lldb/API/SBTarget.h"
+#include "lldb/Symbol/LocateSymbolFile.h"
#include "lldb/Utility/Instrumentation.h"
#include "lldb/Utility/LLDBLog.h"
#include "lldb/lldb-public.h"
@@ -1477,28 +1478,29 @@
const char *uuid_cstr, const char *symfile) {
LLDB_INSTRUMENT_VA(this, path, triple, uuid_cstr, symfile);
- lldb::SBModule sb_module;
TargetSP target_sp(GetSP());
- if (target_sp) {
- ModuleSpec module_spec;
- if (path)
- module_spec.GetFileSpec().SetFile(path, FileSpec::Style::native);
+ if (!target_sp)
+ return {};
- if (uuid_cstr)
- module_spec.GetUUID().SetFromStringRef(uuid_cstr);
+ ModuleSpec module_spec;
+ if (path)
+ module_spec.GetFileSpec().SetFile(path, FileSpec::Style::native);
- if (triple)
- module_spec.GetArchitecture() = Platform::GetAugmentedArchSpec(
- target_sp->GetPlatform().get(), triple);
- else
- module_spec.GetArchitecture() = target_sp->GetArchitecture();
+ if (uuid_cstr)
+ module_spec.GetUUID().SetFromStringRef(uuid_cstr);
- if (symfile)
- module_spec.GetSymbolFileSpec().SetFile(symfile, FileSpec::Style::native);
+ if (triple)
+ module_spec.GetArchitecture() =
+ Platform::GetAugmentedArchSpec(target_sp->GetPlatform().get(), triple);
+ else
+ module_spec.GetArchitecture() = target_sp->GetArchitecture();
- sb_module.SetSP(target_sp->GetOrCreateModule(module_spec, true /* notify */));
- }
- return sb_module;
+ if (symfile)
+ module_spec.GetSymbolFileSpec().SetFile(symfile, FileSpec::Style::native);
+
+ SBModuleSpec sb_modulespec(module_spec);
+
+ return AddModule(sb_modulespec);
}
lldb::SBModule SBTarget::AddModule(const SBModuleSpec &module_spec) {
@@ -1506,9 +1508,26 @@
lldb::SBModule sb_module;
TargetSP target_sp(GetSP());
- if (target_sp)
+ if (target_sp) {
sb_module.SetSP(target_sp->GetOrCreateModule(*module_spec.m_opaque_up,
true /* notify */));
+ if (!sb_module.IsValid() && module_spec.m_opaque_up->GetUUID().IsValid()) {
+ Status error;
+ if (Symbols::DownloadObjectAndSymbolFile(*module_spec.m_opaque_up, error,
+ /* force_lookup */ true)) {
+ if (FileSystem::Instance().Exists(
+ module_spec.m_opaque_up->GetFileSpec())) {
+ sb_module.SetSP(target_sp->GetOrCreateModule(*module_spec.m_opaque_up,
+ true /* notify */));
+ }
+ }
+ }
+ }
+ // If the target hasn't initialized any architecture yet, use the
+ // binary's architecture.
+ if (sb_module.IsValid() && !target_sp->GetArchitecture().IsValid() &&
+ sb_module.GetSP()->GetArchitecture().IsValid())
+ target_sp->SetArchitecture(sb_module.GetSP()->GetArchitecture());
return sb_module;
}
diff --git a/lldb/test/API/python_api/target-arch-from-module/Makefile b/lldb/test/API/python_api/target-arch-from-module/Makefile
new file mode 100644
--- /dev/null
+++ b/lldb/test/API/python_api/target-arch-from-module/Makefile
@@ -0,0 +1,3 @@
+C_SOURCES := main.c
+
+include Makefile.rules
diff --git a/lldb/test/API/python_api/target-arch-from-module/TestTargetArchFromModule.py b/lldb/test/API/python_api/target-arch-from-module/TestTargetArchFromModule.py
new file mode 100644
--- /dev/null
+++ b/lldb/test/API/python_api/target-arch-from-module/TestTargetArchFromModule.py
@@ -0,0 +1,107 @@
+"""
+An SBTarget with no arch, call AddModule, SBTarget's arch should be set.
+"""
+
+import os
+import subprocess
+import re
+import lldb
+from lldbsuite.test.decorators import *
+from lldbsuite.test.lldbtest import *
+from lldbsuite.test import lldbutil
+
+
+class TargetArchFromModule(TestBase):
+ @skipIf(
+ debug_info=no_match(["dsym"]),
+ bugnumber="This test is looking explicitly for a dSYM",
+ )
+ @skipUnlessDarwin
+ @skipIfRemote
+ def test_target_arch_init(self):
+ self.build()
+ aout_exe = self.getBuildArtifact("a.out")
+ aout_dsym = self.getBuildArtifact("a.out.dSYM")
+ hidden_dir = self.getBuildArtifact("hide.noindex")
+ hidden_aout_exe = self.getBuildArtifact("hide.noindex/a.out")
+ hidden_aout_dsym = self.getBuildArtifact("hide.noindex/a.out.dSYM")
+ dsym_for_uuid = self.getBuildArtifact("dsym-for-uuid.sh")
+
+ # We can hook in our dsym-for-uuid shell script to lldb with
+ # this env var instead of requiring a defaults write.
+ os.environ["LLDB_APPLE_DSYMFORUUID_EXECUTABLE"] = dsym_for_uuid
+ self.addTearDownHook(
+ lambda: os.environ.pop("LLDB_APPLE_DSYMFORUUID_EXECUTABLE", None)
+ )
+
+ dwarfdump_uuid_regex = re.compile("UUID: ([-0-9a-fA-F]+) \(([^\(]+)\) .*")
+ dwarfdump_cmd_output = subprocess.check_output(
+ ('/usr/bin/dwarfdump --uuid "%s"' % aout_exe), shell=True
+ ).decode("utf-8")
+ aout_uuid = None
+ for line in dwarfdump_cmd_output.splitlines():
+ match = dwarfdump_uuid_regex.search(line)
+ if match:
+ aout_uuid = match.group(1)
+ self.assertNotEqual(aout_uuid, None, "Could not get uuid of built a.out")
+
+ ### Create our dsym-for-uuid shell script which returns self.hidden_aout_exe.
+ shell_cmds = [
+ "#! /bin/sh",
+ "# the last argument is the uuid",
+ "while [ $# -gt 1 ]",
+ "do",
+ " shift",
+ "done",
+ "ret=0",
+ 'echo ""',
+ 'echo ""',
+ 'echo ""',
+ "",
+ 'if [ "$1" = "%s" ]' % aout_uuid,
+ "then",
+ " uuid=%s" % aout_uuid,
+ " bin=%s" % hidden_aout_exe,
+ " dsym=%s.dSYM/Contents/Resources/DWARF/%s"
+ % (hidden_aout_exe, os.path.basename(hidden_aout_exe)),
+ "fi",
+ 'echo " "',
+ 'echo " $1"',
+ 'echo " "',
+ 'if [ -z "$uuid" -o -z "$bin" -o ! -f "$bin" ]',
+ "then",
+ ' echo " DBGError"',
+ ' echo " not found by $0"',
+ ' echo " "',
+ ' echo " "',
+ ' echo ""',
+ " exit 0",
+ "fi",
+ "",
+ 'echo "DBGDSYMPath$dsym"',
+ 'echo "DBGSymbolRichExecutable$bin"',
+ 'echo ""',
+ "exit $ret",
+ ]
+
+ with open(dsym_for_uuid, "w") as writer:
+ for l in shell_cmds:
+ writer.write(l + "\n")
+
+ os.chmod(dsym_for_uuid, 0o755)
+
+ # Move the main binary and its dSYM into the hide.noindex
+ # directory. Now the only way lldb can find them is with
+ # the LLDB_APPLE_DSYMFORUUID_EXECUTABLE shell script -
+ # so we're testing that this dSYM discovery method works.
+ lldbutil.mkdir_p(hidden_dir)
+ os.rename(aout_exe, hidden_aout_exe)
+ os.rename(aout_dsym, hidden_aout_dsym)
+
+ target = self.dbg.CreateTarget("")
+ self.assertTrue(target.IsValid())
+ self.expect("target list", matching=False, substrs=["arch="])
+
+ m = target.AddModule(None, None, aout_uuid)
+ self.assertTrue(m.IsValid())
+ self.expect("target list", substrs=["arch="])
diff --git a/lldb/test/API/python_api/target-arch-from-module/main.c b/lldb/test/API/python_api/target-arch-from-module/main.c
new file mode 100644
--- /dev/null
+++ b/lldb/test/API/python_api/target-arch-from-module/main.c
@@ -0,0 +1,5 @@
+#include
+int main() {
+ puts("HI i am a program");
+ return 0;
+}