125cae5a2SEugene Zelenko //===- LangOptions.cpp - C Language Family Language Options ---------------===//
279a91418SDouglas Gregor //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
679a91418SDouglas Gregor //
779a91418SDouglas Gregor //===----------------------------------------------------------------------===//
879a91418SDouglas Gregor //
979a91418SDouglas Gregor //  This file defines the LangOptions class.
1079a91418SDouglas Gregor //
1179a91418SDouglas Gregor //===----------------------------------------------------------------------===//
1225cae5a2SEugene Zelenko 
1379a91418SDouglas Gregor #include "clang/Basic/LangOptions.h"
147df405e0SPavel Asyutchenko #include "llvm/ADT/SmallString.h"
157df405e0SPavel Asyutchenko #include "llvm/Support/Path.h"
1679a91418SDouglas Gregor 
1779a91418SDouglas Gregor using namespace clang;
1879a91418SDouglas Gregor 
LangOptions()192154cffdSJan Svoboda LangOptions::LangOptions() : LangStd(LangStandard::lang_unspecified) {
2079a91418SDouglas Gregor #define LANGOPT(Name, Bits, Default, Description) Name = Default;
2179a91418SDouglas Gregor #define ENUM_LANGOPT(Name, Type, Bits, Default, Description) set##Name(Default);
2279a91418SDouglas Gregor #include "clang/Basic/LangOptions.def"
2379a91418SDouglas Gregor }
24f1312a82SDouglas Gregor 
resetNonModularOptions()25f1312a82SDouglas Gregor void LangOptions::resetNonModularOptions() {
26f1312a82SDouglas Gregor #define LANGOPT(Name, Bits, Default, Description)
27f1312a82SDouglas Gregor #define BENIGN_LANGOPT(Name, Bits, Default, Description) Name = Default;
288455e76fSDouglas Gregor #define BENIGN_ENUM_LANGOPT(Name, Type, Bits, Default, Description) \
29f4aaed3bSMelanie Blower   Name = static_cast<unsigned>(Default);
308455e76fSDouglas Gregor #include "clang/Basic/LangOptions.def"
317d106e42SDouglas Gregor 
3285a83c2cSVedant Kumar   // These options do not affect AST generation.
33e64fcdf8SMelanie Blower   NoSanitizeFiles.clear();
34835832d3SDean Michael Berris   XRayAlwaysInstrumentFiles.clear();
35835832d3SDean Michael Berris   XRayNeverInstrumentFiles.clear();
36f54319c8SWill Dietz 
377d106e42SDouglas Gregor   CurrentModule.clear();
38e0bde755SErik Verbruggen   IsHeaderFile = false;
39f1312a82SDouglas Gregor }
40f1312a82SDouglas Gregor 
isNoBuiltinFunc(StringRef FuncName) const41ce2da5e1SMehdi Amini bool LangOptions::isNoBuiltinFunc(StringRef FuncName) const {
427dbc9cf8SChad Rosier   for (unsigned i = 0, e = NoBuiltinFuncs.size(); i != e; ++i)
437dbc9cf8SChad Rosier     if (FuncName.equals(NoBuiltinFuncs[i]))
447dbc9cf8SChad Rosier       return true;
457dbc9cf8SChad Rosier   return false;
467dbc9cf8SChad Rosier }
4735b61397SSven van Haastregt 
getOpenCLVersionTuple() const4835b61397SSven van Haastregt VersionTuple LangOptions::getOpenCLVersionTuple() const {
4935b61397SSven van Haastregt   const int Ver = OpenCLCPlusPlus ? OpenCLCPlusPlusVersion : OpenCLVersion;
509ae92628SJustas Janickas   if (OpenCLCPlusPlus && Ver != 100)
519ae92628SJustas Janickas     return VersionTuple(Ver / 100);
5235b61397SSven van Haastregt   return VersionTuple(Ver / 100, (Ver % 100) / 10);
5335b61397SSven van Haastregt }
542ba4e3a4SMelanie Blower 
getOpenCLCompatibleVersion() const55f9bc1b3bSJustas Janickas unsigned LangOptions::getOpenCLCompatibleVersion() const {
56f9bc1b3bSJustas Janickas   if (!OpenCLCPlusPlus)
57f9bc1b3bSJustas Janickas     return OpenCLVersion;
58f9bc1b3bSJustas Janickas   if (OpenCLCPlusPlusVersion == 100)
59f9bc1b3bSJustas Janickas     return 200;
60f9bc1b3bSJustas Janickas   if (OpenCLCPlusPlusVersion == 202100)
61f9bc1b3bSJustas Janickas     return 300;
62f9bc1b3bSJustas Janickas   llvm_unreachable("Unknown OpenCL version");
63f9bc1b3bSJustas Janickas }
64f9bc1b3bSJustas Janickas 
remapPathPrefix(SmallVectorImpl<char> & Path) const656398f3f2SAlan Zhao void LangOptions::remapPathPrefix(SmallVectorImpl<char> &Path) const {
667df405e0SPavel Asyutchenko   for (const auto &Entry : MacroPrefixMap)
677df405e0SPavel Asyutchenko     if (llvm::sys::path::replace_path_prefix(Path, Entry.first, Entry.second))
687df405e0SPavel Asyutchenko       break;
697df405e0SPavel Asyutchenko }
707df405e0SPavel Asyutchenko 
getOpenCLVersionString() const71cfdfb75cSJustas Janickas std::string LangOptions::getOpenCLVersionString() const {
72cfdfb75cSJustas Janickas   std::string Result;
73cfdfb75cSJustas Janickas   {
74cfdfb75cSJustas Janickas     llvm::raw_string_ostream Out(Result);
75cfdfb75cSJustas Janickas     Out << (OpenCLCPlusPlus ? "C++ for OpenCL" : "OpenCL C") << " version "
76cfdfb75cSJustas Janickas         << getOpenCLVersionTuple().getAsString();
77cfdfb75cSJustas Janickas   }
78cfdfb75cSJustas Janickas   return Result;
79cfdfb75cSJustas Janickas }
80cfdfb75cSJustas Janickas 
setLangDefaults(LangOptions & Opts,Language Lang,const llvm::Triple & T,std::vector<std::string> & Includes,LangStandard::Kind LangStd)8193471e65SHaojian Wu void LangOptions::setLangDefaults(LangOptions &Opts, Language Lang,
8293471e65SHaojian Wu                                   const llvm::Triple &T,
8393471e65SHaojian Wu                                   std::vector<std::string> &Includes,
8493471e65SHaojian Wu                                   LangStandard::Kind LangStd) {
8593471e65SHaojian Wu   // Set some properties which depend solely on the input kind; it would be nice
8693471e65SHaojian Wu   // to move these to the language standard, and have the driver resolve the
8793471e65SHaojian Wu   // input kind + language standard.
8893471e65SHaojian Wu   //
8993471e65SHaojian Wu   // FIXME: Perhaps a better model would be for a single source file to have
9093471e65SHaojian Wu   // multiple language standards (C / C++ std, ObjC std, OpenCL std, OpenMP std)
9193471e65SHaojian Wu   // simultaneously active?
9293471e65SHaojian Wu   if (Lang == Language::Asm) {
9393471e65SHaojian Wu     Opts.AsmPreprocessor = 1;
9493471e65SHaojian Wu   } else if (Lang == Language::ObjC || Lang == Language::ObjCXX) {
9593471e65SHaojian Wu     Opts.ObjC = 1;
9693471e65SHaojian Wu   }
9793471e65SHaojian Wu 
9893471e65SHaojian Wu   if (LangStd == LangStandard::lang_unspecified)
9993471e65SHaojian Wu     LangStd = getDefaultLanguageStandard(Lang, T);
10093471e65SHaojian Wu   const LangStandard &Std = LangStandard::getLangStandardForKind(LangStd);
10193471e65SHaojian Wu   Opts.LangStd = LangStd;
10293471e65SHaojian Wu   Opts.LineComment = Std.hasLineComments();
10393471e65SHaojian Wu   Opts.C99 = Std.isC99();
10493471e65SHaojian Wu   Opts.C11 = Std.isC11();
10593471e65SHaojian Wu   Opts.C17 = Std.isC17();
10693471e65SHaojian Wu   Opts.C2x = Std.isC2x();
10793471e65SHaojian Wu   Opts.CPlusPlus = Std.isCPlusPlus();
10893471e65SHaojian Wu   Opts.CPlusPlus11 = Std.isCPlusPlus11();
10993471e65SHaojian Wu   Opts.CPlusPlus14 = Std.isCPlusPlus14();
11093471e65SHaojian Wu   Opts.CPlusPlus17 = Std.isCPlusPlus17();
11193471e65SHaojian Wu   Opts.CPlusPlus20 = Std.isCPlusPlus20();
11293471e65SHaojian Wu   Opts.CPlusPlus2b = Std.isCPlusPlus2b();
11393471e65SHaojian Wu   Opts.GNUMode = Std.isGNUMode();
11493471e65SHaojian Wu   Opts.GNUCVersion = 0;
11593471e65SHaojian Wu   Opts.HexFloats = Std.hasHexFloats();
11693471e65SHaojian Wu   Opts.WChar = Std.isCPlusPlus();
11793471e65SHaojian Wu   Opts.Digraphs = Std.hasDigraphs();
11893471e65SHaojian Wu 
11993471e65SHaojian Wu   Opts.HLSL = Lang == Language::HLSL;
12013e1a653SXiang Li   if (Opts.HLSL && Opts.IncludeDefaultHeader)
12113e1a653SXiang Li     Includes.push_back("hlsl.h");
12293471e65SHaojian Wu 
12393471e65SHaojian Wu   // Set OpenCL Version.
12493471e65SHaojian Wu   Opts.OpenCL = Std.isOpenCL();
12593471e65SHaojian Wu   if (LangStd == LangStandard::lang_opencl10)
12693471e65SHaojian Wu     Opts.OpenCLVersion = 100;
12793471e65SHaojian Wu   else if (LangStd == LangStandard::lang_opencl11)
12893471e65SHaojian Wu     Opts.OpenCLVersion = 110;
12993471e65SHaojian Wu   else if (LangStd == LangStandard::lang_opencl12)
13093471e65SHaojian Wu     Opts.OpenCLVersion = 120;
13193471e65SHaojian Wu   else if (LangStd == LangStandard::lang_opencl20)
13293471e65SHaojian Wu     Opts.OpenCLVersion = 200;
13393471e65SHaojian Wu   else if (LangStd == LangStandard::lang_opencl30)
13493471e65SHaojian Wu     Opts.OpenCLVersion = 300;
13593471e65SHaojian Wu   else if (LangStd == LangStandard::lang_openclcpp10)
13693471e65SHaojian Wu     Opts.OpenCLCPlusPlusVersion = 100;
13793471e65SHaojian Wu   else if (LangStd == LangStandard::lang_openclcpp2021)
13893471e65SHaojian Wu     Opts.OpenCLCPlusPlusVersion = 202100;
13993471e65SHaojian Wu   else if (LangStd == LangStandard::lang_hlsl2015)
14093471e65SHaojian Wu     Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2015;
14193471e65SHaojian Wu   else if (LangStd == LangStandard::lang_hlsl2016)
14293471e65SHaojian Wu     Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2016;
14393471e65SHaojian Wu   else if (LangStd == LangStandard::lang_hlsl2017)
14493471e65SHaojian Wu     Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2017;
14593471e65SHaojian Wu   else if (LangStd == LangStandard::lang_hlsl2018)
14693471e65SHaojian Wu     Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2018;
14793471e65SHaojian Wu   else if (LangStd == LangStandard::lang_hlsl2021)
14893471e65SHaojian Wu     Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2021;
14993471e65SHaojian Wu   else if (LangStd == LangStandard::lang_hlsl202x)
15093471e65SHaojian Wu     Opts.HLSLVersion = (unsigned)LangOptions::HLSL_202x;
15193471e65SHaojian Wu 
15293471e65SHaojian Wu   // OpenCL has some additional defaults.
15393471e65SHaojian Wu   if (Opts.OpenCL) {
15493471e65SHaojian Wu     Opts.AltiVec = 0;
15593471e65SHaojian Wu     Opts.ZVector = 0;
15693471e65SHaojian Wu     Opts.setDefaultFPContractMode(LangOptions::FPM_On);
15793471e65SHaojian Wu     Opts.OpenCLCPlusPlus = Opts.CPlusPlus;
15893471e65SHaojian Wu     Opts.OpenCLPipes = Opts.getOpenCLCompatibleVersion() == 200;
15993471e65SHaojian Wu     Opts.OpenCLGenericAddressSpace = Opts.getOpenCLCompatibleVersion() == 200;
16093471e65SHaojian Wu 
16193471e65SHaojian Wu     // Include default header file for OpenCL.
16293471e65SHaojian Wu     if (Opts.IncludeDefaultHeader) {
16393471e65SHaojian Wu       if (Opts.DeclareOpenCLBuiltins) {
16493471e65SHaojian Wu         // Only include base header file for builtin types and constants.
16593471e65SHaojian Wu         Includes.push_back("opencl-c-base.h");
16693471e65SHaojian Wu       } else {
16793471e65SHaojian Wu         Includes.push_back("opencl-c.h");
16893471e65SHaojian Wu       }
16993471e65SHaojian Wu     }
17093471e65SHaojian Wu   }
17193471e65SHaojian Wu 
17293471e65SHaojian Wu   Opts.HIP = Lang == Language::HIP;
17393471e65SHaojian Wu   Opts.CUDA = Lang == Language::CUDA || Opts.HIP;
17493471e65SHaojian Wu   if (Opts.HIP) {
17593471e65SHaojian Wu     // HIP toolchain does not support 'Fast' FPOpFusion in backends since it
17693471e65SHaojian Wu     // fuses multiplication/addition instructions without contract flag from
17793471e65SHaojian Wu     // device library functions in LLVM bitcode, which causes accuracy loss in
17893471e65SHaojian Wu     // certain math functions, e.g. tan(-1e20) becomes -0.933 instead of 0.8446.
17993471e65SHaojian Wu     // For device library functions in bitcode to work, 'Strict' or 'Standard'
18093471e65SHaojian Wu     // FPOpFusion options in backends is needed. Therefore 'fast-honor-pragmas'
18193471e65SHaojian Wu     // FP contract option is used to allow fuse across statements in frontend
18293471e65SHaojian Wu     // whereas respecting contract flag in backend.
18393471e65SHaojian Wu     Opts.setDefaultFPContractMode(LangOptions::FPM_FastHonorPragmas);
18493471e65SHaojian Wu   } else if (Opts.CUDA) {
18593471e65SHaojian Wu     if (T.isSPIRV()) {
18693471e65SHaojian Wu       // Emit OpenCL version metadata in LLVM IR when targeting SPIR-V.
18793471e65SHaojian Wu       Opts.OpenCLVersion = 200;
18893471e65SHaojian Wu     }
18993471e65SHaojian Wu     // Allow fuse across statements disregarding pragmas.
19093471e65SHaojian Wu     Opts.setDefaultFPContractMode(LangOptions::FPM_Fast);
19193471e65SHaojian Wu   }
19293471e65SHaojian Wu 
19393471e65SHaojian Wu   Opts.RenderScript = Lang == Language::RenderScript;
19493471e65SHaojian Wu 
19593471e65SHaojian Wu   // OpenCL, C++ and C2x have bool, true, false keywords.
19693471e65SHaojian Wu   Opts.Bool = Opts.OpenCL || Opts.CPlusPlus || Opts.C2x;
19793471e65SHaojian Wu 
19877f72ac1SXiang Li   // OpenCL and HLSL have half keyword
19977f72ac1SXiang Li   Opts.Half = Opts.OpenCL || Opts.HLSL;
20093471e65SHaojian Wu }
20193471e65SHaojian Wu 
defaultWithoutTrailingStorage(const LangOptions & LO)2023ee1ec0bSBenjamin Kramer FPOptions FPOptions::defaultWithoutTrailingStorage(const LangOptions &LO) {
2033ee1ec0bSBenjamin Kramer   FPOptions result(LO);
2042ba4e3a4SMelanie Blower   return result;
2052ba4e3a4SMelanie Blower }
2062ba4e3a4SMelanie Blower 
getChangesSlow(const FPOptions & Base) const207*f7819ce1SSerge Pavlov FPOptionsOverride FPOptions::getChangesSlow(const FPOptions &Base) const {
208*f7819ce1SSerge Pavlov   FPOptions::storage_type OverrideMask = 0;
209*f7819ce1SSerge Pavlov #define OPTION(NAME, TYPE, WIDTH, PREVIOUS)                                    \
210*f7819ce1SSerge Pavlov   if (get##NAME() != Base.get##NAME())                                         \
211*f7819ce1SSerge Pavlov     OverrideMask |= NAME##Mask;
212*f7819ce1SSerge Pavlov #include "clang/Basic/FPOptions.def"
213*f7819ce1SSerge Pavlov   return FPOptionsOverride(*this, OverrideMask);
214*f7819ce1SSerge Pavlov }
215*f7819ce1SSerge Pavlov 
dump()216f4aaed3bSMelanie Blower LLVM_DUMP_METHOD void FPOptions::dump() {
217f4aaed3bSMelanie Blower #define OPTION(NAME, TYPE, WIDTH, PREVIOUS)                                    \
218f4aaed3bSMelanie Blower   llvm::errs() << "\n " #NAME " " << get##NAME();
219f4aaed3bSMelanie Blower #include "clang/Basic/FPOptions.def"
220f4aaed3bSMelanie Blower   llvm::errs() << "\n";
221f4aaed3bSMelanie Blower }
222f4aaed3bSMelanie Blower 
dump()223f4aaed3bSMelanie Blower LLVM_DUMP_METHOD void FPOptionsOverride::dump() {
224f4aaed3bSMelanie Blower #define OPTION(NAME, TYPE, WIDTH, PREVIOUS)                                    \
225f4aaed3bSMelanie Blower   if (has##NAME##Override())                                                   \
226f4aaed3bSMelanie Blower     llvm::errs() << "\n " #NAME " Override is " << get##NAME##Override();
227f4aaed3bSMelanie Blower #include "clang/Basic/FPOptions.def"
228f4aaed3bSMelanie Blower   llvm::errs() << "\n";
2292ba4e3a4SMelanie Blower }
230