//===--- TestClangConfig.h ------------------------------------------------===// // // 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 // //===----------------------------------------------------------------------===// #ifndef LLVM_CLANG_TESTING_TESTCLANGCONFIG_H #define LLVM_CLANG_TESTING_TESTCLANGCONFIG_H #include "clang/Testing/CommandLineArgs.h" #include "llvm/Support/raw_ostream.h" #include #include namespace clang { /// A Clang configuration for end-to-end tests that can be converted to /// command line arguments for the driver. /// /// The configuration is represented as typed, named values, making it easier /// and safer to work with compared to an array of string command line flags. struct TestClangConfig { TestLanguage Language; /// The argument of the `-target` command line flag. std::string Target; bool isC() const { return false #define TESTLANGUAGE_C(lang, version, std_flag, version_index) \ || Language == Lang_##lang##version #include "clang/Testing/TestLanguage.def" ; } bool isC(int Version) const { return false #define TESTLANGUAGE_C(lang, version, std_flag, version_index) \ || (Version == version && Language == Lang_##lang##version) #include "clang/Testing/TestLanguage.def" ; } bool isCOrLater(int MinimumStdVersion) const { const auto MinimumStdVersionIndex = 0 #define TESTLANGUAGE_C(lang, version, std_flag, version_index) \ +(MinimumStdVersion == version ? version_index : 0) #include "clang/Testing/TestLanguage.def" ; switch (Language) { #define TESTLANGUAGE_C(lang, version, std_flag, version_index) \ case Lang_##lang##version: \ return MinimumStdVersionIndex <= version_index; #include "clang/Testing/TestLanguage.def" default: return false; } } bool isC99OrLater() const { return isCOrLater(99); } bool isCOrEarlier(int MaximumStdVersion) const { return isC() && (isC(MaximumStdVersion) || !isCOrLater(MaximumStdVersion)); } bool isCXX() const { return false #define TESTLANGUAGE_CXX(lang, version, std_flag, version_index) \ || Language == Lang_##lang##version #include "clang/Testing/TestLanguage.def" ; } bool isCXX(int Version) const { return false #define TESTLANGUAGE_CXX(lang, version, std_flag, version_index) \ || (Version == version && Language == Lang_##lang##version) #include "clang/Testing/TestLanguage.def" ; } bool isCXXOrLater(int MinimumStdVersion) const { const auto MinimumStdVersionIndex = 0 #define TESTLANGUAGE_CXX(lang, version, std_flag, version_index) \ +(MinimumStdVersion == version ? version_index : 0) #include "clang/Testing/TestLanguage.def" ; switch (Language) { #define TESTLANGUAGE_CXX(lang, version, std_flag, version_index) \ case Lang_##lang##version: \ return MinimumStdVersionIndex <= version_index; #include "clang/Testing/TestLanguage.def" default: return false; } } bool isCXX11OrLater() const { return isCXXOrLater(11); } bool isCXX14OrLater() const { return isCXXOrLater(14); } bool isCXX17OrLater() const { return isCXXOrLater(17); } bool isCXX20OrLater() const { return isCXXOrLater(20); } bool isCXX23OrLater() const { return isCXXOrLater(23); } bool isCXXOrEarlier(int MaximumStdVersion) const { return isCXX() && (isCXX(MaximumStdVersion) || !isCXXOrLater(MaximumStdVersion)); } bool supportsCXXDynamicExceptionSpecification() const { return Language == Lang_CXX03 || Language == Lang_CXX11 || Language == Lang_CXX14; } bool hasDelayedTemplateParsing() const { return Target == "x86_64-pc-win32-msvc"; } std::vector getCommandLineArgs() const { std::vector Result = getCommandLineArgsForTesting(Language); Result.push_back("-target"); Result.push_back(Target); return Result; } std::string toShortString() const { std::string Result; llvm::raw_string_ostream OS(Result); switch (Language) { #define TESTLANGUAGE(lang, version, std_flag, version_index) \ case Lang_##lang##version: \ OS << (#lang #version); \ break; #include "clang/Testing/TestLanguage.def" case Lang_OpenCL: OS << "OpenCL"; break; case Lang_OBJC: OS << "OBJC"; break; case Lang_OBJCXX: OS << "OBJCXX"; break; } OS << (Target.find("win") != std::string::npos ? "_win" : ""); return Result; } std::string toString() const { std::string Result; llvm::raw_string_ostream OS(Result); OS << "{ Language=" << Language << ", Target=" << Target << " }"; return Result; } friend std::ostream &operator<<(std::ostream &OS, const TestClangConfig &ClangConfig) { return OS << ClangConfig.toString(); } }; } // end namespace clang #endif