//===- unittests/Driver/MultilibBuilderTest.cpp --- MultilibBuilder 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 MultilibBuilder and MultilibSetBuilder // //===----------------------------------------------------------------------===// #include "clang/Driver/MultilibBuilder.h" #include "../../lib/Driver/ToolChains/CommonArgs.h" #include "SimpleDiagnosticConsumer.h" #include "clang/Basic/LLVM.h" #include "llvm/ADT/ArrayRef.h" #include "llvm/ADT/StringRef.h" #include "llvm/ADT/StringSwitch.h" #include "gtest/gtest.h" using llvm::is_contained; using namespace clang; using namespace driver; TEST(MultilibBuilderTest, MultilibValidity) { ASSERT_TRUE(MultilibBuilder().isValid()) << "Empty multilib is not valid"; ASSERT_TRUE(MultilibBuilder().flag("-foo").isValid()) << "Single indicative flag is not valid"; ASSERT_TRUE(MultilibBuilder().flag("-foo", /*Disallow=*/true).isValid()) << "Single contraindicative flag is not valid"; ASSERT_FALSE( MultilibBuilder().flag("-foo").flag("-foo", /*Disallow=*/true).isValid()) << "Conflicting flags should invalidate the Multilib"; ASSERT_TRUE(MultilibBuilder().flag("-foo").flag("-foo").isValid()) << "Multilib should be valid even if it has the same flag " "twice"; ASSERT_TRUE(MultilibBuilder() .flag("-foo") .flag("-foobar", /*Disallow=*/true) .isValid()) << "Seemingly conflicting prefixes shouldn't actually conflict"; } TEST(MultilibBuilderTest, Construction1) { MultilibBuilder M("gcc64", "os64", "inc64"); ASSERT_TRUE(M.gccSuffix() == "/gcc64"); ASSERT_TRUE(M.osSuffix() == "/os64"); ASSERT_TRUE(M.includeSuffix() == "/inc64"); } TEST(MultilibBuilderTest, Construction3) { MultilibBuilder M = MultilibBuilder().flag("-f1").flag("-f2").flag("-f3", /*Disallow=*/true); for (const std::string &A : M.flags()) { ASSERT_TRUE(llvm::StringSwitch(A) .Cases("-f1", "-f2", "!f3", true) .Default(false)); } } TEST(MultilibBuilderTest, SetConstruction1) { // Single maybe MultilibSet MS = MultilibSetBuilder() .Maybe(MultilibBuilder("64").flag("-m64")) .makeMultilibSet(); ASSERT_TRUE(MS.size() == 2); for (MultilibSet::const_iterator I = MS.begin(), E = MS.end(); I != E; ++I) { if (I->gccSuffix() == "/64") ASSERT_TRUE(*I->flags().begin() == "-m64"); else if (I->gccSuffix() == "") ASSERT_TRUE(*I->flags().begin() == "!m64"); else FAIL() << "Unrecognized gccSufix: " << I->gccSuffix(); } } TEST(MultilibBuilderTest, SetConstruction2) { // Double maybe MultilibSet MS = MultilibSetBuilder() .Maybe(MultilibBuilder("sof").flag("-sof")) .Maybe(MultilibBuilder("el").flag("-EL")) .makeMultilibSet(); ASSERT_TRUE(MS.size() == 4); for (MultilibSet::const_iterator I = MS.begin(), E = MS.end(); I != E; ++I) { ASSERT_TRUE(llvm::StringSwitch(I->gccSuffix()) .Cases("", "/sof", "/el", "/sof/el", true) .Default(false)) << "Multilib " << *I << " wasn't expected"; ASSERT_TRUE(llvm::StringSwitch(I->gccSuffix()) .Case("", is_contained(I->flags(), "!sof")) .Case("/sof", is_contained(I->flags(), "-sof")) .Case("/el", is_contained(I->flags(), "!sof")) .Case("/sof/el", is_contained(I->flags(), "-sof")) .Default(false)) << "Multilib " << *I << " didn't have the appropriate {-,!}sof flag"; ASSERT_TRUE(llvm::StringSwitch(I->gccSuffix()) .Case("", is_contained(I->flags(), "!EL")) .Case("/sof", is_contained(I->flags(), "!EL")) .Case("/el", is_contained(I->flags(), "-EL")) .Case("/sof/el", is_contained(I->flags(), "-EL")) .Default(false)) << "Multilib " << *I << " didn't have the appropriate {-,!}EL flag"; } } TEST(MultilibBuilderTest, SetRegexFilter) { MultilibSetBuilder MB; MB.Maybe(MultilibBuilder("one")) .Maybe(MultilibBuilder("two")) .Maybe(MultilibBuilder("three")) .makeMultilibSet(); MultilibSet MS = MB.makeMultilibSet(); ASSERT_EQ(MS.size(), (unsigned)2 * 2 * 2) << "Size before filter was incorrect. Contents:\n" << MS; MB.FilterOut("/one/two/three"); MS = MB.makeMultilibSet(); ASSERT_EQ(MS.size(), (unsigned)2 * 2 * 2 - 1) << "Size after filter was incorrect. Contents:\n" << MS; for (MultilibSet::const_iterator I = MS.begin(), E = MS.end(); I != E; ++I) { ASSERT_TRUE(I->gccSuffix() != "/one/two/three") << "The filter should have removed " << *I; } } TEST(MultilibBuilderTest, SetFilterObject) { MultilibSet MS = MultilibSetBuilder() .Maybe(MultilibBuilder("orange")) .Maybe(MultilibBuilder("pear")) .Maybe(MultilibBuilder("plum")) .makeMultilibSet(); ASSERT_EQ((int)MS.size(), 1 /* Default */ + 1 /* pear */ + 1 /* plum */ + 1 /* pear/plum */ + 1 /* orange */ + 1 /* orange/pear */ + 1 /* orange/plum */ + 1 /* orange/pear/plum */) << "Size before filter was incorrect. Contents:\n" << MS; MS.FilterOut([](const Multilib &M) { return StringRef(M.gccSuffix()).starts_with("/p"); }); ASSERT_EQ((int)MS.size(), 1 /* Default */ + 1 /* orange */ + 1 /* orange/pear */ + 1 /* orange/plum */ + 1 /* orange/pear/plum */) << "Size after filter was incorrect. Contents:\n" << MS; for (MultilibSet::const_iterator I = MS.begin(), E = MS.end(); I != E; ++I) { ASSERT_FALSE(StringRef(I->gccSuffix()).starts_with("/p")) << "The filter should have removed " << *I; } } TEST(MultilibBuilderTest, SetSelection1) { MultilibSet MS1 = MultilibSetBuilder() .Maybe(MultilibBuilder("64").flag("-m64")) .makeMultilibSet(); Multilib::flags_list FlagM64 = {"-m64"}; llvm::SmallVector SelectionM64; Driver TheDriver = diagnostic_test_driver(); ASSERT_TRUE(MS1.select(TheDriver, FlagM64, SelectionM64)) << "Flag set was {\"-m64\"}, but selection not found"; ASSERT_TRUE(SelectionM64.back().gccSuffix() == "/64") << "Selection picked " << SelectionM64.back() << " which was not expected"; Multilib::flags_list FlagNoM64 = {"!m64"}; llvm::SmallVector SelectionNoM64; ASSERT_TRUE(MS1.select(TheDriver, FlagNoM64, SelectionNoM64)) << "Flag set was {\"!m64\"}, but selection not found"; ASSERT_TRUE(SelectionNoM64.back().gccSuffix() == "") << "Selection picked " << SelectionNoM64.back() << " which was not expected"; } TEST(MultilibBuilderTest, SetSelection2) { MultilibSet MS2 = MultilibSetBuilder() .Maybe(MultilibBuilder("el").flag("-EL")) .Maybe(MultilibBuilder("sf").flag("-SF")) .makeMultilibSet(); for (unsigned I = 0; I < 4; ++I) { bool IsEL = I & 0x1; bool IsSF = I & 0x2; Multilib::flags_list Flags; if (IsEL) Flags.push_back("-EL"); else Flags.push_back("!EL"); if (IsSF) Flags.push_back("-SF"); else Flags.push_back("!SF"); llvm::SmallVector Selection; Driver TheDriver = diagnostic_test_driver(); ASSERT_TRUE(MS2.select(TheDriver, Flags, Selection)) << "Selection failed for " << (IsEL ? "-EL" : "!EL") << " " << (IsSF ? "-SF" : "!SF"); std::string Suffix; if (IsEL) Suffix += "/el"; if (IsSF) Suffix += "/sf"; ASSERT_EQ(Selection.back().gccSuffix(), Suffix) << "Selection picked " << Selection.back() << " which was not expected "; } }