1 //===-- ClangHost.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 //===----------------------------------------------------------------------===//
11 #include "clang/Basic/Version.h"
12 #include "clang/Config/config.h"
13 #include "clang/Driver/Driver.h"
15 #include "llvm/ADT/StringRef.h"
16 #include "llvm/ADT/Twine.h"
17 #include "llvm/Support/FileSystem.h"
18 #include "llvm/Support/Threading.h"
20 #include "lldb/Host/Config.h"
21 #include "lldb/Host/FileSystem.h"
22 #include "lldb/Host/HostInfo.h"
23 #include "lldb/Utility/FileSpec.h"
24 #include "lldb/Utility/LLDBLog.h"
25 #include "lldb/Utility/Log.h"
29 using namespace lldb_private
;
31 static bool VerifyClangPath(const llvm::Twine
&clang_path
) {
32 if (FileSystem::Instance().IsDirectory(clang_path
))
34 Log
*log
= GetLog(LLDBLog::Host
);
37 "failed to stat clang resource directory at \"%s\"",
38 clang_path
.str().c_str());
43 /// This will compute the clang resource directory assuming that clang was
44 /// installed with the same prefix as lldb.
46 /// If verify is true, the first candidate resource directory will be returned.
47 /// This mode is only used for testing.
49 static bool DefaultComputeClangResourceDirectory(FileSpec
&lldb_shlib_spec
,
52 Log
*log
= GetLog(LLDBLog::Host
);
53 std::string raw_path
= lldb_shlib_spec
.GetPath();
54 llvm::StringRef parent_dir
= llvm::sys::path::parent_path(raw_path
);
55 static const std::string clang_resource_path
=
56 clang::driver::Driver::GetResourcesPath("bin/lldb");
58 static const llvm::StringRef kResourceDirSuffixes
[] = {
59 // LLVM.org's build of LLDB uses the clang resource directory placed
60 // in $install_dir/lib{,64}/clang/$clang_version or
61 // $install_dir/bin/$CLANG_RESOURCE_DIR
63 // swift-lldb uses the clang resource directory copied from swift, which
64 // by default is placed in $install_dir/lib{,64}/lldb/clang. LLDB places
65 // it there, so we use LLDB_INSTALL_LIBDIR_BASENAME.
66 LLDB_INSTALL_LIBDIR_BASENAME
"/lldb/clang",
69 for (const auto &Suffix
: kResourceDirSuffixes
) {
70 llvm::SmallString
<256> clang_dir(parent_dir
);
71 llvm::SmallString
<32> relative_path(Suffix
);
72 llvm::sys::path::native(relative_path
);
73 llvm::sys::path::append(clang_dir
, relative_path
);
74 if (!verify
|| VerifyClangPath(clang_dir
)) {
76 "DefaultComputeClangResourceDir: Setting ClangResourceDir "
77 "to \"{0}\", verify = {1}",
78 clang_dir
.str(), verify
? "true" : "false");
79 file_spec
.SetDirectory(clang_dir
);
80 FileSystem::Instance().Resolve(file_spec
);
88 bool lldb_private::ComputeClangResourceDirectory(FileSpec
&lldb_shlib_spec
,
91 #if !defined(__APPLE__)
92 return DefaultComputeClangResourceDirectory(lldb_shlib_spec
, file_spec
,
95 std::string raw_path
= lldb_shlib_spec
.GetPath();
97 auto rev_it
= llvm::sys::path::rbegin(raw_path
);
98 auto r_end
= llvm::sys::path::rend(raw_path
);
100 // Check for a Posix-style build of LLDB.
101 while (rev_it
!= r_end
) {
102 if (*rev_it
== "LLDB.framework")
107 // We found a non-framework build of LLDB
109 return DefaultComputeClangResourceDirectory(lldb_shlib_spec
, file_spec
,
112 // Inside Xcode and in Xcode toolchains LLDB is always in lockstep
113 // with the Swift compiler, so it can reuse its Clang resource
114 // directory. This allows LLDB and the Swift compiler to share the
115 // same Clang module cache.
116 llvm::SmallString
<256> clang_path
;
117 const char *swift_clang_resource_dir
= "usr/lib/swift/clang";
118 auto parent
= std::next(rev_it
);
119 if (parent
!= r_end
&& *parent
== "SharedFrameworks") {
120 // This is the top-level LLDB in the Xcode.app bundle.
121 // E.g., "Xcode.app/Contents/SharedFrameworks/LLDB.framework/Versions/A"
122 raw_path
.resize(parent
- r_end
);
123 llvm::sys::path::append(clang_path
, raw_path
,
124 "Developer/Toolchains/XcodeDefault.xctoolchain",
125 swift_clang_resource_dir
);
126 if (!verify
|| VerifyClangPath(clang_path
)) {
127 file_spec
.SetDirectory(clang_path
);
128 FileSystem::Instance().Resolve(file_spec
);
131 } else if (parent
!= r_end
&& *parent
== "PrivateFrameworks" &&
132 std::distance(parent
, r_end
) > 2) {
135 if (*parent
== "System") {
136 // This is LLDB inside an Xcode toolchain.
137 // E.g., "Xcode.app/Contents/Developer/Toolchains/" \
138 // "My.xctoolchain/System/Library/PrivateFrameworks/LLDB.framework"
139 raw_path
.resize(parent
- r_end
);
140 llvm::sys::path::append(clang_path
, raw_path
, swift_clang_resource_dir
);
141 if (!verify
|| VerifyClangPath(clang_path
)) {
142 file_spec
.SetDirectory(clang_path
);
143 FileSystem::Instance().Resolve(file_spec
);
149 // Fall back to the Clang resource directory inside the framework.
150 raw_path
= lldb_shlib_spec
.GetPath();
151 raw_path
.resize(rev_it
- r_end
);
152 raw_path
.append("LLDB.framework/Resources/Clang");
153 file_spec
.SetDirectory(raw_path
);
154 FileSystem::Instance().Resolve(file_spec
);
159 FileSpec
lldb_private::GetClangResourceDir() {
160 static FileSpec g_cached_resource_dir
;
161 static llvm::once_flag g_once_flag
;
162 llvm::call_once(g_once_flag
, []() {
163 if (FileSpec lldb_file_spec
= HostInfo::GetShlibDir())
164 ComputeClangResourceDirectory(lldb_file_spec
, g_cached_resource_dir
,
166 Log
*log
= GetLog(LLDBLog::Host
);
167 LLDB_LOGF(log
, "GetClangResourceDir() => '%s'",
168 g_cached_resource_dir
.GetPath().c_str());
170 return g_cached_resource_dir
;