| 1 | |
| 2 | |
| 3 | |
| 4 | |
| 5 | |
| 6 | |
| 7 | |
| 8 | |
| 9 | |
| 10 | |
| 11 | |
| 12 | |
| 13 | #include "clang/Driver/ToolChain.h" |
| 14 | #include "clang/Basic/DiagnosticIDs.h" |
| 15 | #include "clang/Basic/DiagnosticOptions.h" |
| 16 | #include "clang/Basic/LLVM.h" |
| 17 | #include "clang/Driver/Compilation.h" |
| 18 | #include "clang/Driver/Driver.h" |
| 19 | #include "llvm/Support/TargetRegistry.h" |
| 20 | #include "llvm/Support/TargetSelect.h" |
| 21 | #include "llvm/Support/VirtualFileSystem.h" |
| 22 | #include "llvm/Support/raw_ostream.h" |
| 23 | #include "gtest/gtest.h" |
| 24 | using namespace clang; |
| 25 | using namespace clang::driver; |
| 26 | |
| 27 | namespace { |
| 28 | |
| 29 | TEST(ToolChainTest, VFSGCCInstallation) { |
| 30 | IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions(); |
| 31 | |
| 32 | IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); |
| 33 | struct TestDiagnosticConsumer : public DiagnosticConsumer {}; |
| 34 | DiagnosticsEngine Diags(DiagID, &*DiagOpts, new TestDiagnosticConsumer); |
| 35 | IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem> InMemoryFileSystem( |
| 36 | new llvm::vfs::InMemoryFileSystem); |
| 37 | Driver TheDriver("/bin/clang", "arm-linux-gnueabihf", Diags, |
| 38 | InMemoryFileSystem); |
| 39 | |
| 40 | const char *EmptyFiles[] = { |
| 41 | "foo.cpp", |
| 42 | "/bin/clang", |
| 43 | "/usr/lib/gcc/arm-linux-gnueabi/4.6.1/crtbegin.o", |
| 44 | "/usr/lib/gcc/arm-linux-gnueabi/4.6.1/crtend.o", |
| 45 | "/usr/lib/gcc/arm-linux-gnueabihf/4.6.3/crtbegin.o", |
| 46 | "/usr/lib/gcc/arm-linux-gnueabihf/4.6.3/crtend.o", |
| 47 | "/usr/lib/arm-linux-gnueabi/crt1.o", |
| 48 | "/usr/lib/arm-linux-gnueabi/crti.o", |
| 49 | "/usr/lib/arm-linux-gnueabi/crtn.o", |
| 50 | "/usr/lib/arm-linux-gnueabihf/crt1.o", |
| 51 | "/usr/lib/arm-linux-gnueabihf/crti.o", |
| 52 | "/usr/lib/arm-linux-gnueabihf/crtn.o", |
| 53 | "/usr/include/arm-linux-gnueabi/.keep", |
| 54 | "/usr/include/arm-linux-gnueabihf/.keep", |
| 55 | "/lib/arm-linux-gnueabi/.keep", |
| 56 | "/lib/arm-linux-gnueabihf/.keep"}; |
| 57 | |
| 58 | for (const char *Path : EmptyFiles) |
| 59 | InMemoryFileSystem->addFile(Path, 0, |
| 60 | llvm::MemoryBuffer::getMemBuffer("\n")); |
| 61 | |
| 62 | std::unique_ptr<Compilation> C(TheDriver.BuildCompilation( |
| 63 | {"-fsyntax-only", "--gcc-toolchain=", "foo.cpp"})); |
| 64 | EXPECT_TRUE(C); |
| 65 | |
| 66 | std::string S; |
| 67 | { |
| 68 | llvm::raw_string_ostream OS(S); |
| 69 | C->getDefaultToolChain().printVerboseInfo(OS); |
| 70 | } |
| 71 | #if _WIN32 |
| 72 | std::replace(S.begin(), S.end(), '\\', '/'); |
| 73 | #endif |
| 74 | EXPECT_EQ( |
| 75 | "Found candidate GCC installation: " |
| 76 | "/usr/lib/gcc/arm-linux-gnueabihf/4.6.3\n" |
| 77 | "Selected GCC installation: /usr/lib/gcc/arm-linux-gnueabihf/4.6.3\n" |
| 78 | "Candidate multilib: .;@m32\n" |
| 79 | "Selected multilib: .;@m32\n", |
| 80 | S); |
| 81 | } |
| 82 | |
| 83 | TEST(ToolChainTest, VFSGCCInstallationRelativeDir) { |
| 84 | IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions(); |
| 85 | |
| 86 | IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); |
| 87 | struct TestDiagnosticConsumer : public DiagnosticConsumer {}; |
| 88 | DiagnosticsEngine Diags(DiagID, &*DiagOpts, new TestDiagnosticConsumer); |
| 89 | IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem> InMemoryFileSystem( |
| 90 | new llvm::vfs::InMemoryFileSystem); |
| 91 | Driver TheDriver("/home/test/bin/clang", "arm-linux-gnueabi", Diags, |
| 92 | InMemoryFileSystem); |
| 93 | |
| 94 | const char *EmptyFiles[] = { |
| 95 | "foo.cpp", "/home/test/lib/gcc/arm-linux-gnueabi/4.6.1/crtbegin.o", |
| 96 | "/home/test/include/arm-linux-gnueabi/.keep"}; |
| 97 | |
| 98 | for (const char *Path : EmptyFiles) |
| 99 | InMemoryFileSystem->addFile(Path, 0, |
| 100 | llvm::MemoryBuffer::getMemBuffer("\n")); |
| 101 | |
| 102 | std::unique_ptr<Compilation> C(TheDriver.BuildCompilation( |
| 103 | {"-fsyntax-only", "--gcc-toolchain=", "foo.cpp"})); |
| 104 | EXPECT_TRUE(C); |
| 105 | |
| 106 | std::string S; |
| 107 | { |
| 108 | llvm::raw_string_ostream OS(S); |
| 109 | C->getDefaultToolChain().printVerboseInfo(OS); |
| 110 | } |
| 111 | #if _WIN32 |
| 112 | std::replace(S.begin(), S.end(), '\\', '/'); |
| 113 | #endif |
| 114 | EXPECT_EQ("Found candidate GCC installation: " |
| 115 | "/home/test/bin/../lib/gcc/arm-linux-gnueabi/4.6.1\n" |
| 116 | "Selected GCC installation: " |
| 117 | "/home/test/bin/../lib/gcc/arm-linux-gnueabi/4.6.1\n" |
| 118 | "Candidate multilib: .;@m32\n" |
| 119 | "Selected multilib: .;@m32\n", |
| 120 | S); |
| 121 | } |
| 122 | |
| 123 | TEST(ToolChainTest, DefaultDriverMode) { |
| 124 | IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions(); |
| 125 | |
| 126 | IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); |
| 127 | struct TestDiagnosticConsumer : public DiagnosticConsumer {}; |
| 128 | DiagnosticsEngine Diags(DiagID, &*DiagOpts, new TestDiagnosticConsumer); |
| 129 | IntrusiveRefCntPtr<llvm::vfs::InMemoryFileSystem> InMemoryFileSystem( |
| 130 | new llvm::vfs::InMemoryFileSystem); |
| 131 | |
| 132 | Driver CCDriver("/home/test/bin/clang", "arm-linux-gnueabi", Diags, |
| 133 | InMemoryFileSystem); |
| 134 | CCDriver.setCheckInputsExist(false); |
| 135 | Driver CXXDriver("/home/test/bin/clang++", "arm-linux-gnueabi", Diags, |
| 136 | InMemoryFileSystem); |
| 137 | CXXDriver.setCheckInputsExist(false); |
| 138 | Driver CLDriver("/home/test/bin/clang-cl", "arm-linux-gnueabi", Diags, |
| 139 | InMemoryFileSystem); |
| 140 | CLDriver.setCheckInputsExist(false); |
| 141 | |
| 142 | std::unique_ptr<Compilation> CC(CCDriver.BuildCompilation( |
| 143 | { "/home/test/bin/clang", "foo.cpp"})); |
| 144 | std::unique_ptr<Compilation> CXX(CXXDriver.BuildCompilation( |
| 145 | { "/home/test/bin/clang++", "foo.cpp"})); |
| 146 | std::unique_ptr<Compilation> CL(CLDriver.BuildCompilation( |
| 147 | { "/home/test/bin/clang-cl", "foo.cpp"})); |
| 148 | |
| 149 | EXPECT_TRUE(CC); |
| 150 | EXPECT_TRUE(CXX); |
| 151 | EXPECT_TRUE(CL); |
| 152 | EXPECT_TRUE(CCDriver.CCCIsCC()); |
| 153 | EXPECT_TRUE(CXXDriver.CCCIsCXX()); |
| 154 | EXPECT_TRUE(CLDriver.IsCLMode()); |
| 155 | } |
| 156 | TEST(ToolChainTest, InvalidArgument) { |
| 157 | IntrusiveRefCntPtr<DiagnosticIDs> DiagID(new DiagnosticIDs()); |
| 158 | struct TestDiagnosticConsumer : public DiagnosticConsumer {}; |
| 159 | IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts = new DiagnosticOptions(); |
| 160 | DiagnosticsEngine Diags(DiagID, &*DiagOpts, new TestDiagnosticConsumer); |
| 161 | Driver TheDriver("/bin/clang", "arm-linux-gnueabihf", Diags); |
| 162 | std::unique_ptr<Compilation> C(TheDriver.BuildCompilation( |
| 163 | {"-fsyntax-only", "-fan-unknown-option", "foo.cpp"})); |
| 164 | EXPECT_TRUE(C); |
| 165 | EXPECT_TRUE(C->containsError()); |
| 166 | } |
| 167 | |
| 168 | TEST(ToolChainTest, ParsedClangName) { |
| 169 | ParsedClangName Empty; |
| 170 | EXPECT_TRUE(Empty.TargetPrefix.empty()); |
| 171 | EXPECT_TRUE(Empty.ModeSuffix.empty()); |
| 172 | EXPECT_TRUE(Empty.DriverMode == nullptr); |
| 173 | EXPECT_FALSE(Empty.TargetIsValid); |
| 174 | |
| 175 | ParsedClangName DriverOnly("clang", nullptr); |
| 176 | EXPECT_TRUE(DriverOnly.TargetPrefix.empty()); |
| 177 | EXPECT_TRUE(DriverOnly.ModeSuffix == "clang"); |
| 178 | EXPECT_TRUE(DriverOnly.DriverMode == nullptr); |
| 179 | EXPECT_FALSE(DriverOnly.TargetIsValid); |
| 180 | |
| 181 | ParsedClangName DriverOnly2("clang++", "--driver-mode=g++"); |
| 182 | EXPECT_TRUE(DriverOnly2.TargetPrefix.empty()); |
| 183 | EXPECT_TRUE(DriverOnly2.ModeSuffix == "clang++"); |
| 184 | EXPECT_STREQ(DriverOnly2.DriverMode, "--driver-mode=g++"); |
| 185 | EXPECT_FALSE(DriverOnly2.TargetIsValid); |
| 186 | |
| 187 | ParsedClangName TargetAndMode("i386", "clang-g++", "--driver-mode=g++", true); |
| 188 | EXPECT_TRUE(TargetAndMode.TargetPrefix == "i386"); |
| 189 | EXPECT_TRUE(TargetAndMode.ModeSuffix == "clang-g++"); |
| 190 | EXPECT_STREQ(TargetAndMode.DriverMode, "--driver-mode=g++"); |
| 191 | EXPECT_TRUE(TargetAndMode.TargetIsValid); |
| 192 | } |
| 193 | |
| 194 | TEST(ToolChainTest, GetTargetAndMode) { |
| 195 | llvm::InitializeAllTargets(); |
| 196 | std::string IgnoredError; |
| 197 | if (!llvm::TargetRegistry::lookupTarget("x86_64", IgnoredError)) |
| 198 | return; |
| 199 | |
| 200 | ParsedClangName Res = ToolChain::getTargetAndModeFromProgramName("clang"); |
| 201 | EXPECT_TRUE(Res.TargetPrefix.empty()); |
| 202 | EXPECT_TRUE(Res.ModeSuffix == "clang"); |
| 203 | EXPECT_TRUE(Res.DriverMode == nullptr); |
| 204 | EXPECT_FALSE(Res.TargetIsValid); |
| 205 | |
| 206 | Res = ToolChain::getTargetAndModeFromProgramName("clang++"); |
| 207 | EXPECT_TRUE(Res.TargetPrefix.empty()); |
| 208 | EXPECT_TRUE(Res.ModeSuffix == "clang++"); |
| 209 | EXPECT_STREQ(Res.DriverMode, "--driver-mode=g++"); |
| 210 | EXPECT_FALSE(Res.TargetIsValid); |
| 211 | |
| 212 | Res = ToolChain::getTargetAndModeFromProgramName("clang++6.0"); |
| 213 | EXPECT_TRUE(Res.TargetPrefix.empty()); |
| 214 | EXPECT_TRUE(Res.ModeSuffix == "clang++"); |
| 215 | EXPECT_STREQ(Res.DriverMode, "--driver-mode=g++"); |
| 216 | EXPECT_FALSE(Res.TargetIsValid); |
| 217 | |
| 218 | Res = ToolChain::getTargetAndModeFromProgramName("clang++-release"); |
| 219 | EXPECT_TRUE(Res.TargetPrefix.empty()); |
| 220 | EXPECT_TRUE(Res.ModeSuffix == "clang++"); |
| 221 | EXPECT_STREQ(Res.DriverMode, "--driver-mode=g++"); |
| 222 | EXPECT_FALSE(Res.TargetIsValid); |
| 223 | |
| 224 | Res = ToolChain::getTargetAndModeFromProgramName("x86_64-clang++"); |
| 225 | EXPECT_TRUE(Res.TargetPrefix == "x86_64"); |
| 226 | EXPECT_TRUE(Res.ModeSuffix == "clang++"); |
| 227 | EXPECT_STREQ(Res.DriverMode, "--driver-mode=g++"); |
| 228 | EXPECT_TRUE(Res.TargetIsValid); |
| 229 | |
| 230 | Res = ToolChain::getTargetAndModeFromProgramName( |
| 231 | "x86_64-linux-gnu-clang-c++"); |
| 232 | EXPECT_TRUE(Res.TargetPrefix == "x86_64-linux-gnu"); |
| 233 | EXPECT_TRUE(Res.ModeSuffix == "clang-c++"); |
| 234 | EXPECT_STREQ(Res.DriverMode, "--driver-mode=g++"); |
| 235 | EXPECT_TRUE(Res.TargetIsValid); |
| 236 | |
| 237 | Res = ToolChain::getTargetAndModeFromProgramName( |
| 238 | "x86_64-linux-gnu-clang-c++-tot"); |
| 239 | EXPECT_TRUE(Res.TargetPrefix == "x86_64-linux-gnu"); |
| 240 | EXPECT_TRUE(Res.ModeSuffix == "clang-c++"); |
| 241 | EXPECT_STREQ(Res.DriverMode, "--driver-mode=g++"); |
| 242 | EXPECT_TRUE(Res.TargetIsValid); |
| 243 | |
| 244 | Res = ToolChain::getTargetAndModeFromProgramName("qqq"); |
| 245 | EXPECT_TRUE(Res.TargetPrefix.empty()); |
| 246 | EXPECT_TRUE(Res.ModeSuffix.empty()); |
| 247 | EXPECT_TRUE(Res.DriverMode == nullptr); |
| 248 | EXPECT_FALSE(Res.TargetIsValid); |
| 249 | |
| 250 | Res = ToolChain::getTargetAndModeFromProgramName("x86_64-qqq"); |
| 251 | EXPECT_TRUE(Res.TargetPrefix.empty()); |
| 252 | EXPECT_TRUE(Res.ModeSuffix.empty()); |
| 253 | EXPECT_TRUE(Res.DriverMode == nullptr); |
| 254 | EXPECT_FALSE(Res.TargetIsValid); |
| 255 | |
| 256 | Res = ToolChain::getTargetAndModeFromProgramName("qqq-clang-cl"); |
| 257 | EXPECT_TRUE(Res.TargetPrefix == "qqq"); |
| 258 | EXPECT_TRUE(Res.ModeSuffix == "clang-cl"); |
| 259 | EXPECT_STREQ(Res.DriverMode, "--driver-mode=cl"); |
| 260 | EXPECT_FALSE(Res.TargetIsValid); |
| 261 | } |
| 262 | } |
| 263 | |