| //===- unittests/Driver/ToolChainTest.cpp --- ToolChain tests -------------===// |
| // |
| // 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 |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // Unit tests for ToolChains. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "clang/Driver/ToolChain.h" |
| #include "clang/Basic/DiagnosticIDs.h" |
| #include "clang/Basic/DiagnosticOptions.h" |
| #include "clang/Basic/LLVM.h" |
| #include "clang/Driver/Compilation.h" |
| #include "clang/Driver/Driver.h" |
| #include "llvm/ADT/ArrayRef.h" |
| #include "llvm/MC/TargetRegistry.h" |
| #include "llvm/Support/TargetSelect.h" |
| #include "llvm/Support/VirtualFileSystem.h" |
| #include "llvm/Support/raw_ostream.h" |
| #include "gtest/gtest.h" |
| using namespace clang; |
| using namespace clang::driver; |
| |
| namespace { |
| |
| TEST(ToolChainTest, VFSGCCInstallation) { |
| IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions(); |
| |
| IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); |
| struct TestDiagnosticConsumer : public DiagnosticConsumer {}; |
| IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem> InMemoryFileSystem( |
| new llvm::vfs::InMemoryFileSystem); |
| |
| const char *EmptyFiles[] = { |
| "foo.cpp", |
| "/bin/clang", |
| "/usr/lib/gcc/arm-linux-gnueabi/4.6.1/crtbegin.o", |
| "/usr/lib/gcc/arm-linux-gnueabi/4.6.1/crtend.o", |
| "/usr/lib/gcc/arm-linux-gnueabihf/4.6.3/crtbegin.o", |
| "/usr/lib/gcc/arm-linux-gnueabihf/4.6.3/crtend.o", |
| "/usr/lib/arm-linux-gnueabi/crt1.o", |
| "/usr/lib/arm-linux-gnueabi/crti.o", |
| "/usr/lib/arm-linux-gnueabi/crtn.o", |
| "/usr/lib/arm-linux-gnueabihf/crt1.o", |
| "/usr/lib/arm-linux-gnueabihf/crti.o", |
| "/usr/lib/arm-linux-gnueabihf/crtn.o", |
| "/usr/include/arm-linux-gnueabi/.keep", |
| "/usr/include/arm-linux-gnueabihf/.keep", |
| "/lib/arm-linux-gnueabi/.keep", |
| "/lib/arm-linux-gnueabihf/.keep", |
| |
| "/sysroot/usr/lib/gcc/arm-linux-gnueabi/4.5.1/crtbegin.o", |
| "/sysroot/usr/lib/gcc/arm-linux-gnueabi/4.5.1/crtend.o", |
| "/sysroot/usr/lib/gcc/arm-linux-gnueabihf/4.5.3/crtbegin.o", |
| "/sysroot/usr/lib/gcc/arm-linux-gnueabihf/4.5.3/crtend.o", |
| "/sysroot/usr/lib/arm-linux-gnueabi/crt1.o", |
| "/sysroot/usr/lib/arm-linux-gnueabi/crti.o", |
| "/sysroot/usr/lib/arm-linux-gnueabi/crtn.o", |
| "/sysroot/usr/lib/arm-linux-gnueabihf/crt1.o", |
| "/sysroot/usr/lib/arm-linux-gnueabihf/crti.o", |
| "/sysroot/usr/lib/arm-linux-gnueabihf/crtn.o", |
| "/sysroot/usr/include/arm-linux-gnueabi/.keep", |
| "/sysroot/usr/include/arm-linux-gnueabihf/.keep", |
| "/sysroot/lib/arm-linux-gnueabi/.keep", |
| "/sysroot/lib/arm-linux-gnueabihf/.keep", |
| }; |
| |
| for (const char *Path : EmptyFiles) |
| InMemoryFileSystem->addFile(Path, 0, |
| llvm::MemoryBuffer::getMemBuffer("\n")); |
| |
| { |
| DiagnosticsEngine Diags(DiagID, &*DiagOpts, new TestDiagnosticConsumer); |
| Driver TheDriver("/bin/clang", "arm-linux-gnueabihf", Diags, |
| "clang LLVM compiler", InMemoryFileSystem); |
| std::unique_ptr<Compilation> C(TheDriver.BuildCompilation( |
| {"-fsyntax-only", "--gcc-toolchain=", "--sysroot=", "foo.cpp"})); |
| ASSERT_TRUE(C); |
| std::string S; |
| { |
| llvm::raw_string_ostream OS(S); |
| C->getDefaultToolChain().printVerboseInfo(OS); |
| } |
| if (is_style_windows(llvm::sys::path::Style::native)) |
| std::replace(S.begin(), S.end(), '\\', '/'); |
| EXPECT_EQ( |
| "Found candidate GCC installation: " |
| "/usr/lib/gcc/arm-linux-gnueabihf/4.6.3\n" |
| "Selected GCC installation: /usr/lib/gcc/arm-linux-gnueabihf/4.6.3\n" |
| "Candidate multilib: .;@m32\n" |
| "Selected multilib: .;@m32\n", |
| S); |
| } |
| |
| { |
| DiagnosticsEngine Diags(DiagID, &*DiagOpts, new TestDiagnosticConsumer); |
| Driver TheDriver("/bin/clang", "arm-linux-gnueabihf", Diags, |
| "clang LLVM compiler", InMemoryFileSystem); |
| std::unique_ptr<Compilation> C(TheDriver.BuildCompilation( |
| {"-fsyntax-only", "--gcc-toolchain=", "--sysroot=/sysroot", |
| "foo.cpp"})); |
| ASSERT_TRUE(C); |
| std::string S; |
| { |
| llvm::raw_string_ostream OS(S); |
| C->getDefaultToolChain().printVerboseInfo(OS); |
| } |
| if (is_style_windows(llvm::sys::path::Style::native)) |
| std::replace(S.begin(), S.end(), '\\', '/'); |
| // Test that 4.5.3 from --sysroot is not overridden by 4.6.3 (larger |
| // version) from /usr. |
| EXPECT_EQ("Found candidate GCC installation: " |
| "/sysroot/usr/lib/gcc/arm-linux-gnueabihf/4.5.3\n" |
| "Selected GCC installation: " |
| "/sysroot/usr/lib/gcc/arm-linux-gnueabihf/4.5.3\n" |
| "Candidate multilib: .;@m32\n" |
| "Selected multilib: .;@m32\n", |
| S); |
| } |
| } |
| |
| TEST(ToolChainTest, VFSGCCInstallationRelativeDir) { |
| IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions(); |
| |
| IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); |
| struct TestDiagnosticConsumer : public DiagnosticConsumer {}; |
| DiagnosticsEngine Diags(DiagID, &*DiagOpts, new TestDiagnosticConsumer); |
| IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem> InMemoryFileSystem( |
| new llvm::vfs::InMemoryFileSystem); |
| Driver TheDriver("/home/test/bin/clang", "arm-linux-gnueabi", Diags, |
| "clang LLVM compiler", InMemoryFileSystem); |
| |
| const char *EmptyFiles[] = { |
| "foo.cpp", "/home/test/lib/gcc/arm-linux-gnueabi/4.6.1/crtbegin.o", |
| "/home/test/include/arm-linux-gnueabi/.keep"}; |
| |
| for (const char *Path : EmptyFiles) |
| InMemoryFileSystem->addFile(Path, 0, |
| llvm::MemoryBuffer::getMemBuffer("\n")); |
| |
| std::unique_ptr<Compilation> C(TheDriver.BuildCompilation( |
| {"-fsyntax-only", "--gcc-toolchain=", "foo.cpp"})); |
| EXPECT_TRUE(C); |
| |
| std::string S; |
| { |
| llvm::raw_string_ostream OS(S); |
| C->getDefaultToolChain().printVerboseInfo(OS); |
| } |
| if (is_style_windows(llvm::sys::path::Style::windows)) |
| std::replace(S.begin(), S.end(), '\\', '/'); |
| EXPECT_EQ("Found candidate GCC installation: " |
| "/home/test/bin/../lib/gcc/arm-linux-gnueabi/4.6.1\n" |
| "Selected GCC installation: " |
| "/home/test/bin/../lib/gcc/arm-linux-gnueabi/4.6.1\n" |
| "Candidate multilib: .;@m32\n" |
| "Selected multilib: .;@m32\n", |
| S); |
| } |
| |
| TEST(ToolChainTest, DefaultDriverMode) { |
| IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions(); |
| |
| IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); |
| struct TestDiagnosticConsumer : public DiagnosticConsumer {}; |
| DiagnosticsEngine Diags(DiagID, &*DiagOpts, new TestDiagnosticConsumer); |
| IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem> InMemoryFileSystem( |
| new llvm::vfs::InMemoryFileSystem); |
| |
| Driver CCDriver("/home/test/bin/clang", "arm-linux-gnueabi", Diags, |
| "clang LLVM compiler", InMemoryFileSystem); |
| CCDriver.setCheckInputsExist(false); |
| Driver CXXDriver("/home/test/bin/clang++", "arm-linux-gnueabi", Diags, |
| "clang LLVM compiler", InMemoryFileSystem); |
| CXXDriver.setCheckInputsExist(false); |
| Driver CLDriver("/home/test/bin/clang-cl", "arm-linux-gnueabi", Diags, |
| "clang LLVM compiler", InMemoryFileSystem); |
| CLDriver.setCheckInputsExist(false); |
| |
| std::unique_ptr<Compilation> CC(CCDriver.BuildCompilation( |
| { "/home/test/bin/clang", "foo.cpp"})); |
| std::unique_ptr<Compilation> CXX(CXXDriver.BuildCompilation( |
| { "/home/test/bin/clang++", "foo.cpp"})); |
| std::unique_ptr<Compilation> CL(CLDriver.BuildCompilation( |
| { "/home/test/bin/clang-cl", "foo.cpp"})); |
| |
| EXPECT_TRUE(CC); |
| EXPECT_TRUE(CXX); |
| EXPECT_TRUE(CL); |
| EXPECT_TRUE(CCDriver.CCCIsCC()); |
| EXPECT_TRUE(CXXDriver.CCCIsCXX()); |
| EXPECT_TRUE(CLDriver.IsCLMode()); |
| } |
| TEST(ToolChainTest, InvalidArgument) { |
| IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); |
| struct TestDiagnosticConsumer : public DiagnosticConsumer {}; |
| IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions(); |
| DiagnosticsEngine Diags(DiagID, &*DiagOpts, new TestDiagnosticConsumer); |
| Driver TheDriver("/bin/clang", "arm-linux-gnueabihf", Diags); |
| std::unique_ptr<Compilation> C(TheDriver.BuildCompilation( |
| {"-fsyntax-only", "-fan-unknown-option", "foo.cpp"})); |
| EXPECT_TRUE(C); |
| EXPECT_TRUE(C->containsError()); |
| } |
| |
| TEST(ToolChainTest, ParsedClangName) { |
| ParsedClangName Empty; |
| EXPECT_TRUE(Empty.TargetPrefix.empty()); |
| EXPECT_TRUE(Empty.ModeSuffix.empty()); |
| EXPECT_TRUE(Empty.DriverMode == nullptr); |
| EXPECT_FALSE(Empty.TargetIsValid); |
| |
| ParsedClangName DriverOnly("clang", nullptr); |
| EXPECT_TRUE(DriverOnly.TargetPrefix.empty()); |
| EXPECT_TRUE(DriverOnly.ModeSuffix == "clang"); |
| EXPECT_TRUE(DriverOnly.DriverMode == nullptr); |
| EXPECT_FALSE(DriverOnly.TargetIsValid); |
| |
| ParsedClangName DriverOnly2("clang++", "--driver-mode=g++"); |
| EXPECT_TRUE(DriverOnly2.TargetPrefix.empty()); |
| EXPECT_TRUE(DriverOnly2.ModeSuffix == "clang++"); |
| EXPECT_STREQ(DriverOnly2.DriverMode, "--driver-mode=g++"); |
| EXPECT_FALSE(DriverOnly2.TargetIsValid); |
| |
| ParsedClangName TargetAndMode("i386", "clang-g++", "--driver-mode=g++", true); |
| EXPECT_TRUE(TargetAndMode.TargetPrefix == "i386"); |
| EXPECT_TRUE(TargetAndMode.ModeSuffix == "clang-g++"); |
| EXPECT_STREQ(TargetAndMode.DriverMode, "--driver-mode=g++"); |
| EXPECT_TRUE(TargetAndMode.TargetIsValid); |
| } |
| |
| TEST(ToolChainTest, GetTargetAndMode) { |
| llvm::InitializeAllTargets(); |
| std::string IgnoredError; |
| if (!llvm::TargetRegistry::lookupTarget("x86_64", IgnoredError)) |
| return; |
| |
| ParsedClangName Res = ToolChain::getTargetAndModeFromProgramName("clang"); |
| EXPECT_TRUE(Res.TargetPrefix.empty()); |
| EXPECT_TRUE(Res.ModeSuffix == "clang"); |
| EXPECT_TRUE(Res.DriverMode == nullptr); |
| EXPECT_FALSE(Res.TargetIsValid); |
| |
| Res = ToolChain::getTargetAndModeFromProgramName("clang++"); |
| EXPECT_TRUE(Res.TargetPrefix.empty()); |
| EXPECT_TRUE(Res.ModeSuffix == "clang++"); |
| EXPECT_STREQ(Res.DriverMode, "--driver-mode=g++"); |
| EXPECT_FALSE(Res.TargetIsValid); |
| |
| Res = ToolChain::getTargetAndModeFromProgramName("clang++6.0"); |
| EXPECT_TRUE(Res.TargetPrefix.empty()); |
| EXPECT_TRUE(Res.ModeSuffix == "clang++"); |
| EXPECT_STREQ(Res.DriverMode, "--driver-mode=g++"); |
| EXPECT_FALSE(Res.TargetIsValid); |
| |
| Res = ToolChain::getTargetAndModeFromProgramName("clang++-release"); |
| EXPECT_TRUE(Res.TargetPrefix.empty()); |
| EXPECT_TRUE(Res.ModeSuffix == "clang++"); |
| EXPECT_STREQ(Res.DriverMode, "--driver-mode=g++"); |
| EXPECT_FALSE(Res.TargetIsValid); |
| |
| Res = ToolChain::getTargetAndModeFromProgramName("x86_64-clang++"); |
| EXPECT_TRUE(Res.TargetPrefix == "x86_64"); |
| EXPECT_TRUE(Res.ModeSuffix == "clang++"); |
| EXPECT_STREQ(Res.DriverMode, "--driver-mode=g++"); |
| EXPECT_TRUE(Res.TargetIsValid); |
| |
| Res = ToolChain::getTargetAndModeFromProgramName( |
| "x86_64-linux-gnu-clang-c++"); |
| EXPECT_TRUE(Res.TargetPrefix == "x86_64-linux-gnu"); |
| EXPECT_TRUE(Res.ModeSuffix == "clang-c++"); |
| EXPECT_STREQ(Res.DriverMode, "--driver-mode=g++"); |
| EXPECT_TRUE(Res.TargetIsValid); |
| |
| Res = ToolChain::getTargetAndModeFromProgramName( |
| "x86_64-linux-gnu-clang-c++-tot"); |
| EXPECT_TRUE(Res.TargetPrefix == "x86_64-linux-gnu"); |
| EXPECT_TRUE(Res.ModeSuffix == "clang-c++"); |
| EXPECT_STREQ(Res.DriverMode, "--driver-mode=g++"); |
| EXPECT_TRUE(Res.TargetIsValid); |
| |
| Res = ToolChain::getTargetAndModeFromProgramName("qqq"); |
| EXPECT_TRUE(Res.TargetPrefix.empty()); |
| EXPECT_TRUE(Res.ModeSuffix.empty()); |
| EXPECT_TRUE(Res.DriverMode == nullptr); |
| EXPECT_FALSE(Res.TargetIsValid); |
| |
| Res = ToolChain::getTargetAndModeFromProgramName("x86_64-qqq"); |
| EXPECT_TRUE(Res.TargetPrefix.empty()); |
| EXPECT_TRUE(Res.ModeSuffix.empty()); |
| EXPECT_TRUE(Res.DriverMode == nullptr); |
| EXPECT_FALSE(Res.TargetIsValid); |
| |
| Res = ToolChain::getTargetAndModeFromProgramName("qqq-clang-cl"); |
| EXPECT_TRUE(Res.TargetPrefix == "qqq"); |
| EXPECT_TRUE(Res.ModeSuffix == "clang-cl"); |
| EXPECT_STREQ(Res.DriverMode, "--driver-mode=cl"); |
| EXPECT_FALSE(Res.TargetIsValid); |
| } |
| |
| TEST(ToolChainTest, CommandOutput) { |
| IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions(); |
| |
| IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); |
| struct TestDiagnosticConsumer : public DiagnosticConsumer {}; |
| DiagnosticsEngine Diags(DiagID, &*DiagOpts, new TestDiagnosticConsumer); |
| IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem> InMemoryFileSystem( |
| new llvm::vfs::InMemoryFileSystem); |
| |
| Driver CCDriver("/home/test/bin/clang", "arm-linux-gnueabi", Diags, |
| "clang LLVM compiler", InMemoryFileSystem); |
| CCDriver.setCheckInputsExist(false); |
| std::unique_ptr<Compilation> CC( |
| CCDriver.BuildCompilation({"/home/test/bin/clang", "foo.cpp"})); |
| const JobList &Jobs = CC->getJobs(); |
| |
| const auto &CmdCompile = Jobs.getJobs().front(); |
| const auto &InFile = CmdCompile->getInputInfos().front().getFilename(); |
| EXPECT_STREQ(InFile, "foo.cpp"); |
| auto ObjFile = CmdCompile->getOutputFilenames().front(); |
| EXPECT_TRUE(StringRef(ObjFile).endswith(".o")); |
| |
| const auto &CmdLink = Jobs.getJobs().back(); |
| const auto LinkInFile = CmdLink->getInputInfos().front().getFilename(); |
| EXPECT_EQ(ObjFile, LinkInFile); |
| auto ExeFile = CmdLink->getOutputFilenames().front(); |
| EXPECT_EQ("a.out", ExeFile); |
| } |
| |
| TEST(ToolChainTest, PostCallback) { |
| IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions(); |
| IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); |
| struct TestDiagnosticConsumer : public DiagnosticConsumer {}; |
| DiagnosticsEngine Diags(DiagID, &*DiagOpts, new TestDiagnosticConsumer); |
| IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem> InMemoryFileSystem( |
| new llvm::vfs::InMemoryFileSystem); |
| |
| // The executable path must not exist. |
| Driver CCDriver("/home/test/bin/clang", "arm-linux-gnueabi", Diags, |
| "clang LLVM compiler", InMemoryFileSystem); |
| CCDriver.setCheckInputsExist(false); |
| std::unique_ptr<Compilation> CC( |
| CCDriver.BuildCompilation({"/home/test/bin/clang", "foo.cpp"})); |
| bool CallbackHasCalled = false; |
| CC->setPostCallback( |
| [&](const Command &C, int Ret) { CallbackHasCalled = true; }); |
| const JobList &Jobs = CC->getJobs(); |
| auto &CmdCompile = Jobs.getJobs().front(); |
| const Command *FailingCmd = nullptr; |
| CC->ExecuteCommand(*CmdCompile, FailingCmd); |
| EXPECT_TRUE(CallbackHasCalled); |
| } |
| |
| TEST(GetDriverMode, PrefersLastDriverMode) { |
| static constexpr const char *Args[] = {"clang-cl", "--driver-mode=foo", |
| "--driver-mode=bar", "foo.cpp"}; |
| EXPECT_EQ(getDriverMode(Args[0], llvm::makeArrayRef(Args).slice(1)), "bar"); |
| } |
| |
| } // end anonymous namespace. |