|  | //===-- CppModuleConfigurationTest.cpp ------------------------------------===// | 
|  | // | 
|  | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|  | // See https://llvm.org/LICENSE.txt for license information. | 
|  | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "Plugins/ExpressionParser/Clang/CppModuleConfiguration.h" | 
|  | #include "Plugins/ExpressionParser/Clang/ClangHost.h" | 
|  | #include "TestingSupport/SubsystemRAII.h" | 
|  | #include "lldb/Host/FileSystem.h" | 
|  | #include "lldb/Host/HostInfo.h" | 
|  |  | 
|  | #include "gmock/gmock.h" | 
|  | #include "gtest/gtest.h" | 
|  |  | 
|  | using namespace lldb_private; | 
|  |  | 
|  | namespace { | 
|  | struct CppModuleConfigurationTest : public testing::Test { | 
|  | llvm::MemoryBufferRef m_empty_buffer; | 
|  | llvm::IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem> m_fs; | 
|  |  | 
|  | CppModuleConfigurationTest() | 
|  | : m_empty_buffer("", "<empty buffer>"), | 
|  | m_fs(new llvm::vfs::InMemoryFileSystem()) {} | 
|  |  | 
|  | void SetUp() override { | 
|  | FileSystem::Initialize(m_fs); | 
|  | HostInfo::Initialize(); | 
|  | } | 
|  |  | 
|  | void TearDown() override { | 
|  | HostInfo::Terminate(); | 
|  | FileSystem::Terminate(); | 
|  | } | 
|  |  | 
|  | /// Utility function turning a list of paths into a FileSpecList. | 
|  | FileSpecList makeFiles(llvm::ArrayRef<std::string> paths) { | 
|  | FileSpecList result; | 
|  | for (const std::string &path : paths) { | 
|  | result.Append(FileSpec(path, FileSpec::Style::posix)); | 
|  | if (!m_fs->addFileNoOwn(path, static_cast<time_t>(0), m_empty_buffer)) | 
|  | llvm_unreachable("Invalid test configuration?"); | 
|  | } | 
|  | return result; | 
|  | } | 
|  | }; | 
|  | } // namespace | 
|  |  | 
|  | /// Returns the Clang resource include directory. | 
|  | static std::string ResourceInc() { | 
|  | llvm::SmallString<256> resource_dir; | 
|  | llvm::sys::path::append(resource_dir, GetClangResourceDir().GetPath(), | 
|  | "include"); | 
|  | return std::string(resource_dir); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, Linux) { | 
|  | // Test the average Linux configuration. | 
|  |  | 
|  | std::string usr = "/usr/include"; | 
|  | std::string libcpp = "/usr/include/c++/v1"; | 
|  | std::vector<std::string> files = {// C library | 
|  | usr + "/stdio.h", | 
|  | // C++ library | 
|  | libcpp + "/vector", | 
|  | libcpp + "/module.modulemap"}; | 
|  | CppModuleConfiguration config(makeFiles(files), llvm::Triple()); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre("std")); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), | 
|  | testing::ElementsAre(libcpp, ResourceInc(), usr)); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, LinuxTargetSpecificInclude) { | 
|  | // Test the average Linux configuration. | 
|  |  | 
|  | std::string usr = "/usr/include"; | 
|  | std::string usr_target = "/usr/include/x86_64-linux-gnu"; | 
|  | std::string libcpp = "/usr/include/c++/v1"; | 
|  | std::string libcpp_target = "/usr/include/x86_64-unknown-linux-gnu/c++/v1"; | 
|  | std::vector<std::string> files = { | 
|  | // C library | 
|  | usr + "/stdio.h", usr_target + "/sys/cdefs.h", | 
|  | // C++ library | 
|  | libcpp + "/vector", libcpp + "/module.modulemap"}; | 
|  | CppModuleConfiguration config(makeFiles(files), | 
|  | llvm::Triple("x86_64-unknown-linux-gnu")); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre("std")); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), | 
|  | testing::ElementsAre(libcpp, ResourceInc(), usr, usr_target, | 
|  | libcpp_target)); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, Sysroot) { | 
|  | // Test that having a sysroot for the whole system works fine. | 
|  |  | 
|  | std::string libcpp = "/home/user/sysroot/usr/include/c++/v1"; | 
|  | std::string usr = "/home/user/sysroot/usr/include"; | 
|  | std::vector<std::string> files = {// C library | 
|  | usr + "/stdio.h", | 
|  | // C++ library | 
|  | libcpp + "/vector", | 
|  | libcpp + "/module.modulemap"}; | 
|  | CppModuleConfiguration config(makeFiles(files), llvm::Triple()); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre("std")); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), | 
|  | testing::ElementsAre(libcpp, ResourceInc(), usr)); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, LinuxLocalLibCpp) { | 
|  | // Test that a locally build libc++ is detected. | 
|  |  | 
|  | std::string usr = "/usr/include"; | 
|  | std::string libcpp = "/home/user/llvm-build/include/c++/v1"; | 
|  | std::vector<std::string> files = {// C library | 
|  | usr + "/stdio.h", | 
|  | // C++ library | 
|  | libcpp + "/vector", | 
|  | libcpp + "/module.modulemap"}; | 
|  | CppModuleConfiguration config(makeFiles(files), llvm::Triple()); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre("std")); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), | 
|  | testing::ElementsAre(libcpp, ResourceInc(), usr)); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, UnrelatedLibrary) { | 
|  | // Test that having an unrelated library in /usr/include doesn't break. | 
|  |  | 
|  | std::string usr = "/usr/include"; | 
|  | std::string libcpp = "/home/user/llvm-build/include/c++/v1"; | 
|  | std::vector<std::string> files = {// C library | 
|  | usr + "/stdio.h", | 
|  | // unrelated library | 
|  | usr + "/boost/vector", | 
|  | // C++ library | 
|  | libcpp + "/vector", | 
|  | libcpp + "/module.modulemap"}; | 
|  | CppModuleConfiguration config(makeFiles(files), llvm::Triple()); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre("std")); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), | 
|  | testing::ElementsAre(libcpp, ResourceInc(), usr)); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, UnrelatedLibraryWithTargetSpecificInclude) { | 
|  | // Test that having an unrelated library in /usr/include doesn't break. | 
|  |  | 
|  | std::string usr = "/usr/include"; | 
|  | std::string libcpp = "/home/user/llvm-build/include/c++/v1"; | 
|  | std::string libcpp_target = | 
|  | "/home/user/llvm-build/include/x86_64-unknown-linux-gnu/c++/v1"; | 
|  | std::vector<std::string> files = {// C library | 
|  | usr + "/stdio.h", | 
|  | // unrelated library | 
|  | usr + "/boost/vector", | 
|  | // C++ library | 
|  | libcpp + "/vector", | 
|  | libcpp + "/module.modulemap"}; | 
|  | CppModuleConfiguration config(makeFiles(files), | 
|  | llvm::Triple("x86_64-unknown-linux-gnu")); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre("std")); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), | 
|  | testing::ElementsAre(libcpp, ResourceInc(), usr, libcpp_target)); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, Xcode) { | 
|  | // Test detection of libc++ coming from Xcode with generic platform names. | 
|  |  | 
|  | std::string p = "/Applications/Xcode.app/Contents/Developer/"; | 
|  | std::string libcpp = p + "Toolchains/B.xctoolchain/usr/include/c++/v1"; | 
|  | std::string usr = | 
|  | p + "Platforms/A.platform/Developer/SDKs/OSVers.sdk/usr/include"; | 
|  | std::vector<std::string> files = { | 
|  | // C library | 
|  | usr + "/stdio.h", | 
|  | // C++ library | 
|  | libcpp + "/vector", | 
|  | libcpp + "/module.modulemap", | 
|  | }; | 
|  | CppModuleConfiguration config(makeFiles(files), llvm::Triple()); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre("std")); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), | 
|  | testing::ElementsAre(libcpp, ResourceInc(), usr)); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, LibCppV2) { | 
|  | // Test that a "v2" of libc++ is still correctly detected. | 
|  |  | 
|  | std::string libcpp = "/usr/include/c++/v2"; | 
|  | std::vector<std::string> files = {// C library | 
|  | "/usr/include/stdio.h", | 
|  | // C++ library | 
|  | libcpp + "/vector", | 
|  | libcpp + "/module.modulemap"}; | 
|  | CppModuleConfiguration config(makeFiles(files), llvm::Triple()); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre("std")); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), | 
|  | testing::ElementsAre("/usr/include/c++/v2", ResourceInc(), | 
|  | "/usr/include")); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, UnknownLibCppFile) { | 
|  | // Test that having some unknown file in the libc++ path doesn't break | 
|  | // anything. | 
|  |  | 
|  | std::string libcpp = "/usr/include/c++/v1"; | 
|  | std::vector<std::string> files = {// C library | 
|  | "/usr/include/stdio.h", | 
|  | // C++ library | 
|  | libcpp + "/non_existing_file", | 
|  | libcpp + "/module.modulemap", | 
|  | libcpp + "/vector"}; | 
|  | CppModuleConfiguration config(makeFiles(files), llvm::Triple()); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre("std")); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), | 
|  | testing::ElementsAre("/usr/include/c++/v1", ResourceInc(), | 
|  | "/usr/include")); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, MissingUsrInclude) { | 
|  | // Test that we don't load 'std' if we can't find the C standard library. | 
|  |  | 
|  | std::string libcpp = "/usr/include/c++/v1"; | 
|  | std::vector<std::string> files = {// C++ library | 
|  | libcpp + "/vector", | 
|  | libcpp + "/module.modulemap"}; | 
|  | CppModuleConfiguration config(makeFiles(files), llvm::Triple()); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre()); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), testing::ElementsAre()); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, MissingLibCpp) { | 
|  | // Test that we don't load 'std' if we don't have a libc++. | 
|  |  | 
|  | std::string usr = "/usr/include"; | 
|  | std::vector<std::string> files = { | 
|  | // C library | 
|  | usr + "/stdio.h", | 
|  | }; | 
|  | CppModuleConfiguration config(makeFiles(files), llvm::Triple()); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre()); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), testing::ElementsAre()); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, IgnoreLibStdCpp) { | 
|  | // Test that we don't do anything bad when we encounter libstdc++ paths. | 
|  |  | 
|  | std::string usr = "/usr/include"; | 
|  | std::vector<std::string> files = { | 
|  | // C library | 
|  | usr + "/stdio.h", | 
|  | // C++ library | 
|  | usr + "/c++/8.0.1/vector", | 
|  | }; | 
|  | CppModuleConfiguration config(makeFiles(files), llvm::Triple()); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre()); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), testing::ElementsAre()); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, AmbiguousCLib) { | 
|  | // Test that we don't do anything when we are not sure where the | 
|  | // right C standard library is. | 
|  |  | 
|  | std::string usr1 = "/usr/include"; | 
|  | std::string usr2 = "/usr/include/other/path"; | 
|  | std::string libcpp = usr1 + "c++/v1"; | 
|  | std::vector<std::string> files = { | 
|  | // First C library | 
|  | usr1 + "/stdio.h", | 
|  | // Second C library | 
|  | usr2 + "/stdio.h", | 
|  | // C++ library | 
|  | libcpp + "/vector", | 
|  | libcpp + "/module.modulemap", | 
|  | }; | 
|  | CppModuleConfiguration config(makeFiles(files), llvm::Triple()); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre()); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), testing::ElementsAre()); | 
|  | } | 
|  |  | 
|  | TEST_F(CppModuleConfigurationTest, AmbiguousLibCpp) { | 
|  | // Test that we don't do anything when we are not sure where the | 
|  | // right libc++ is. | 
|  |  | 
|  | std::string usr = "/usr/include"; | 
|  | std::string libcpp1 = usr + "c++/v1"; | 
|  | std::string libcpp2 = usr + "c++/v2"; | 
|  | std::vector<std::string> files = { | 
|  | // C library | 
|  | usr + "/stdio.h", | 
|  | // First C++ library | 
|  | libcpp1 + "/vector", | 
|  | libcpp1 + "/module.modulemap", | 
|  | // Second C++ library | 
|  | libcpp2 + "/vector", | 
|  | libcpp2 + "/module.modulemap", | 
|  | }; | 
|  | CppModuleConfiguration config(makeFiles(files), llvm::Triple()); | 
|  | EXPECT_THAT(config.GetImportedModules(), testing::ElementsAre()); | 
|  | EXPECT_THAT(config.GetIncludeDirs(), testing::ElementsAre()); | 
|  | } |