xref: /freebsd-src/contrib/llvm-project/lldb/source/Plugins/ExpressionParser/Clang/ClangHost.cpp (revision 06c3fb2749bda94cb5201f81ffdb8fa6c3161b2e)
15ffd83dbSDimitry Andric //===-- ClangHost.cpp -----------------------------------------------------===//
20b57cec5SDimitry Andric //
30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric //
70b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
80b57cec5SDimitry Andric 
90b57cec5SDimitry Andric #include "ClangHost.h"
100b57cec5SDimitry Andric 
110b57cec5SDimitry Andric #include "clang/Basic/Version.h"
120b57cec5SDimitry Andric #include "clang/Config/config.h"
13*06c3fb27SDimitry Andric #include "clang/Driver/Driver.h"
140b57cec5SDimitry Andric 
150b57cec5SDimitry Andric #include "llvm/ADT/StringRef.h"
160b57cec5SDimitry Andric #include "llvm/ADT/Twine.h"
170b57cec5SDimitry Andric #include "llvm/Support/FileSystem.h"
180b57cec5SDimitry Andric #include "llvm/Support/Threading.h"
190b57cec5SDimitry Andric 
200b57cec5SDimitry Andric #include "lldb/Host/Config.h"
210b57cec5SDimitry Andric #include "lldb/Host/FileSystem.h"
220b57cec5SDimitry Andric #include "lldb/Host/HostInfo.h"
230b57cec5SDimitry Andric #include "lldb/Utility/FileSpec.h"
2481ad6265SDimitry Andric #include "lldb/Utility/LLDBLog.h"
250b57cec5SDimitry Andric #include "lldb/Utility/Log.h"
260b57cec5SDimitry Andric 
270b57cec5SDimitry Andric #include <string>
280b57cec5SDimitry Andric 
290b57cec5SDimitry Andric using namespace lldb_private;
300b57cec5SDimitry Andric 
VerifyClangPath(const llvm::Twine & clang_path)310b57cec5SDimitry Andric static bool VerifyClangPath(const llvm::Twine &clang_path) {
320b57cec5SDimitry Andric   if (FileSystem::Instance().IsDirectory(clang_path))
330b57cec5SDimitry Andric     return true;
3481ad6265SDimitry Andric   Log *log = GetLog(LLDBLog::Host);
359dba64beSDimitry Andric   LLDB_LOGF(log,
369dba64beSDimitry Andric             "VerifyClangPath(): "
370b57cec5SDimitry Andric             "failed to stat clang resource directory at \"%s\"",
380b57cec5SDimitry Andric             clang_path.str().c_str());
390b57cec5SDimitry Andric   return false;
400b57cec5SDimitry Andric }
410b57cec5SDimitry Andric 
420b57cec5SDimitry Andric ///
430b57cec5SDimitry Andric /// This will compute the clang resource directory assuming that clang was
440b57cec5SDimitry Andric /// installed with the same prefix as lldb.
450b57cec5SDimitry Andric ///
460b57cec5SDimitry Andric /// If verify is true, the first candidate resource directory will be returned.
470b57cec5SDimitry Andric /// This mode is only used for testing.
480b57cec5SDimitry Andric ///
DefaultComputeClangResourceDirectory(FileSpec & lldb_shlib_spec,FileSpec & file_spec,bool verify)490b57cec5SDimitry Andric static bool DefaultComputeClangResourceDirectory(FileSpec &lldb_shlib_spec,
500b57cec5SDimitry Andric                                                  FileSpec &file_spec,
510b57cec5SDimitry Andric                                                  bool verify) {
5281ad6265SDimitry Andric   Log *log = GetLog(LLDBLog::Host);
530b57cec5SDimitry Andric   std::string raw_path = lldb_shlib_spec.GetPath();
540b57cec5SDimitry Andric   llvm::StringRef parent_dir = llvm::sys::path::parent_path(raw_path);
55*06c3fb27SDimitry Andric   static const std::string clang_resource_path =
56*06c3fb27SDimitry Andric       clang::driver::Driver::GetResourcesPath("bin/lldb", CLANG_RESOURCE_DIR);
570b57cec5SDimitry Andric 
580b57cec5SDimitry Andric   static const llvm::StringRef kResourceDirSuffixes[] = {
590b57cec5SDimitry Andric       // LLVM.org's build of LLDB uses the clang resource directory placed
60*06c3fb27SDimitry Andric       // in $install_dir/lib{,64}/clang/$clang_version or
61*06c3fb27SDimitry Andric       // $install_dir/bin/$CLANG_RESOURCE_DIR
62*06c3fb27SDimitry Andric       clang_resource_path,
630b57cec5SDimitry Andric       // swift-lldb uses the clang resource directory copied from swift, which
640b57cec5SDimitry Andric       // by default is placed in $install_dir/lib{,64}/lldb/clang. LLDB places
65bdd1243dSDimitry Andric       // it there, so we use LLDB_INSTALL_LIBDIR_BASENAME.
66bdd1243dSDimitry Andric       LLDB_INSTALL_LIBDIR_BASENAME "/lldb/clang",
670b57cec5SDimitry Andric   };
680b57cec5SDimitry Andric 
690b57cec5SDimitry Andric   for (const auto &Suffix : kResourceDirSuffixes) {
700b57cec5SDimitry Andric     llvm::SmallString<256> clang_dir(parent_dir);
710b57cec5SDimitry Andric     llvm::SmallString<32> relative_path(Suffix);
720b57cec5SDimitry Andric     llvm::sys::path::native(relative_path);
730b57cec5SDimitry Andric     llvm::sys::path::append(clang_dir, relative_path);
740b57cec5SDimitry Andric     if (!verify || VerifyClangPath(clang_dir)) {
755ffd83dbSDimitry Andric       LLDB_LOG(log,
769dba64beSDimitry Andric                "DefaultComputeClangResourceDir: Setting ClangResourceDir "
775ffd83dbSDimitry Andric                "to \"{0}\", verify = {1}",
785ffd83dbSDimitry Andric                clang_dir.str(), verify ? "true" : "false");
79bdd1243dSDimitry Andric       file_spec.SetDirectory(clang_dir);
800b57cec5SDimitry Andric       FileSystem::Instance().Resolve(file_spec);
810b57cec5SDimitry Andric       return true;
820b57cec5SDimitry Andric     }
830b57cec5SDimitry Andric   }
840b57cec5SDimitry Andric 
850b57cec5SDimitry Andric   return false;
860b57cec5SDimitry Andric }
870b57cec5SDimitry Andric 
ComputeClangResourceDirectory(FileSpec & lldb_shlib_spec,FileSpec & file_spec,bool verify)880b57cec5SDimitry Andric bool lldb_private::ComputeClangResourceDirectory(FileSpec &lldb_shlib_spec,
89*06c3fb27SDimitry Andric                                                  FileSpec &file_spec,
90*06c3fb27SDimitry Andric                                                  bool verify) {
910b57cec5SDimitry Andric #if !defined(__APPLE__)
920b57cec5SDimitry Andric   return DefaultComputeClangResourceDirectory(lldb_shlib_spec, file_spec,
930b57cec5SDimitry Andric                                               verify);
940b57cec5SDimitry Andric #else
950b57cec5SDimitry Andric   std::string raw_path = lldb_shlib_spec.GetPath();
960b57cec5SDimitry Andric 
970b57cec5SDimitry Andric   auto rev_it = llvm::sys::path::rbegin(raw_path);
980b57cec5SDimitry Andric   auto r_end = llvm::sys::path::rend(raw_path);
990b57cec5SDimitry Andric 
1000b57cec5SDimitry Andric   // Check for a Posix-style build of LLDB.
1010b57cec5SDimitry Andric   while (rev_it != r_end) {
1020b57cec5SDimitry Andric     if (*rev_it == "LLDB.framework")
1030b57cec5SDimitry Andric       break;
1040b57cec5SDimitry Andric     ++rev_it;
1050b57cec5SDimitry Andric   }
1060b57cec5SDimitry Andric 
1070b57cec5SDimitry Andric   // We found a non-framework build of LLDB
1080b57cec5SDimitry Andric   if (rev_it == r_end)
1090b57cec5SDimitry Andric     return DefaultComputeClangResourceDirectory(lldb_shlib_spec, file_spec,
1100b57cec5SDimitry Andric                                                 verify);
1110b57cec5SDimitry Andric 
1120b57cec5SDimitry Andric   // Inside Xcode and in Xcode toolchains LLDB is always in lockstep
1130b57cec5SDimitry Andric   // with the Swift compiler, so it can reuse its Clang resource
1140b57cec5SDimitry Andric   // directory. This allows LLDB and the Swift compiler to share the
1150b57cec5SDimitry Andric   // same Clang module cache.
1160b57cec5SDimitry Andric   llvm::SmallString<256> clang_path;
1170b57cec5SDimitry Andric   const char *swift_clang_resource_dir = "usr/lib/swift/clang";
1180b57cec5SDimitry Andric   auto parent = std::next(rev_it);
1190b57cec5SDimitry Andric   if (parent != r_end && *parent == "SharedFrameworks") {
1200b57cec5SDimitry Andric     // This is the top-level LLDB in the Xcode.app bundle.
1210b57cec5SDimitry Andric     // E.g., "Xcode.app/Contents/SharedFrameworks/LLDB.framework/Versions/A"
1220b57cec5SDimitry Andric     raw_path.resize(parent - r_end);
1230b57cec5SDimitry Andric     llvm::sys::path::append(clang_path, raw_path,
1240b57cec5SDimitry Andric                             "Developer/Toolchains/XcodeDefault.xctoolchain",
1250b57cec5SDimitry Andric                             swift_clang_resource_dir);
1260b57cec5SDimitry Andric     if (!verify || VerifyClangPath(clang_path)) {
127bdd1243dSDimitry Andric       file_spec.SetDirectory(clang_path);
1280b57cec5SDimitry Andric       FileSystem::Instance().Resolve(file_spec);
1290b57cec5SDimitry Andric       return true;
1300b57cec5SDimitry Andric     }
1310b57cec5SDimitry Andric   } else if (parent != r_end && *parent == "PrivateFrameworks" &&
1320b57cec5SDimitry Andric              std::distance(parent, r_end) > 2) {
1330b57cec5SDimitry Andric     ++parent;
1340b57cec5SDimitry Andric     ++parent;
1350b57cec5SDimitry Andric     if (*parent == "System") {
1360b57cec5SDimitry Andric       // This is LLDB inside an Xcode toolchain.
1370b57cec5SDimitry Andric       // E.g., "Xcode.app/Contents/Developer/Toolchains/"               \
1380b57cec5SDimitry Andric       //       "My.xctoolchain/System/Library/PrivateFrameworks/LLDB.framework"
1390b57cec5SDimitry Andric       raw_path.resize(parent - r_end);
1400b57cec5SDimitry Andric       llvm::sys::path::append(clang_path, raw_path, swift_clang_resource_dir);
1410b57cec5SDimitry Andric       if (!verify || VerifyClangPath(clang_path)) {
142bdd1243dSDimitry Andric         file_spec.SetDirectory(clang_path);
1430b57cec5SDimitry Andric         FileSystem::Instance().Resolve(file_spec);
1440b57cec5SDimitry Andric         return true;
1450b57cec5SDimitry Andric       }
1460b57cec5SDimitry Andric     }
1470b57cec5SDimitry Andric   }
1480b57cec5SDimitry Andric 
1490b57cec5SDimitry Andric   // Fall back to the Clang resource directory inside the framework.
150e8d8bef9SDimitry Andric   raw_path = lldb_shlib_spec.GetPath();
151e8d8bef9SDimitry Andric   raw_path.resize(rev_it - r_end);
1520b57cec5SDimitry Andric   raw_path.append("LLDB.framework/Resources/Clang");
153bdd1243dSDimitry Andric   file_spec.SetDirectory(raw_path);
1540b57cec5SDimitry Andric   FileSystem::Instance().Resolve(file_spec);
1550b57cec5SDimitry Andric   return true;
1560b57cec5SDimitry Andric #endif // __APPLE__
1570b57cec5SDimitry Andric }
1580b57cec5SDimitry Andric 
GetClangResourceDir()1590b57cec5SDimitry Andric FileSpec lldb_private::GetClangResourceDir() {
1600b57cec5SDimitry Andric   static FileSpec g_cached_resource_dir;
1610b57cec5SDimitry Andric   static llvm::once_flag g_once_flag;
1620b57cec5SDimitry Andric   llvm::call_once(g_once_flag, []() {
1630b57cec5SDimitry Andric     if (FileSpec lldb_file_spec = HostInfo::GetShlibDir())
1640b57cec5SDimitry Andric       ComputeClangResourceDirectory(lldb_file_spec, g_cached_resource_dir,
1650b57cec5SDimitry Andric                                     true);
16681ad6265SDimitry Andric     Log *log = GetLog(LLDBLog::Host);
1679dba64beSDimitry Andric     LLDB_LOGF(log, "GetClangResourceDir() => '%s'",
1680b57cec5SDimitry Andric               g_cached_resource_dir.GetPath().c_str());
1690b57cec5SDimitry Andric   });
1700b57cec5SDimitry Andric   return g_cached_resource_dir;
1710b57cec5SDimitry Andric }
172