10b57cec5SDimitry Andric //===- LangOptions.cpp - C Language Family Language Options ---------------===//
20b57cec5SDimitry Andric //
30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric //
70b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
80b57cec5SDimitry Andric //
90b57cec5SDimitry Andric //  This file defines the LangOptions class.
100b57cec5SDimitry Andric //
110b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
120b57cec5SDimitry Andric 
130b57cec5SDimitry Andric #include "clang/Basic/LangOptions.h"
146e75b2fbSDimitry Andric #include "llvm/ADT/SmallString.h"
156e75b2fbSDimitry Andric #include "llvm/Support/Path.h"
160b57cec5SDimitry Andric 
170b57cec5SDimitry Andric using namespace clang;
180b57cec5SDimitry Andric 
LangOptions()19e8d8bef9SDimitry Andric LangOptions::LangOptions() : LangStd(LangStandard::lang_unspecified) {
200b57cec5SDimitry Andric #define LANGOPT(Name, Bits, Default, Description) Name = Default;
210b57cec5SDimitry Andric #define ENUM_LANGOPT(Name, Type, Bits, Default, Description) set##Name(Default);
220b57cec5SDimitry Andric #include "clang/Basic/LangOptions.def"
230b57cec5SDimitry Andric }
240b57cec5SDimitry Andric 
resetNonModularOptions()250b57cec5SDimitry Andric void LangOptions::resetNonModularOptions() {
260b57cec5SDimitry Andric #define LANGOPT(Name, Bits, Default, Description)
270b57cec5SDimitry Andric #define BENIGN_LANGOPT(Name, Bits, Default, Description) Name = Default;
280b57cec5SDimitry Andric #define BENIGN_ENUM_LANGOPT(Name, Type, Bits, Default, Description) \
295ffd83dbSDimitry Andric   Name = static_cast<unsigned>(Default);
300b57cec5SDimitry Andric #include "clang/Basic/LangOptions.def"
310b57cec5SDimitry Andric 
3206c3fb27SDimitry Andric   // Reset "benign" options with implied values (Options.td ImpliedBy relations)
3306c3fb27SDimitry Andric   // rather than their defaults. This avoids unexpected combinations and
3406c3fb27SDimitry Andric   // invocations that cannot be round-tripped to arguments.
3506c3fb27SDimitry Andric   // FIXME: we should derive this automatically from ImpliedBy in tablegen.
3606c3fb27SDimitry Andric   AllowFPReassoc = UnsafeFPMath;
3706c3fb27SDimitry Andric   NoHonorNaNs = FiniteMathOnly;
3806c3fb27SDimitry Andric   NoHonorInfs = FiniteMathOnly;
3906c3fb27SDimitry Andric 
400b57cec5SDimitry Andric   // These options do not affect AST generation.
41fe6060f1SDimitry Andric   NoSanitizeFiles.clear();
420b57cec5SDimitry Andric   XRayAlwaysInstrumentFiles.clear();
430b57cec5SDimitry Andric   XRayNeverInstrumentFiles.clear();
440b57cec5SDimitry Andric 
450b57cec5SDimitry Andric   CurrentModule.clear();
460b57cec5SDimitry Andric   IsHeaderFile = false;
470b57cec5SDimitry Andric }
480b57cec5SDimitry Andric 
isNoBuiltinFunc(StringRef FuncName) const490b57cec5SDimitry Andric bool LangOptions::isNoBuiltinFunc(StringRef FuncName) const {
500b57cec5SDimitry Andric   for (unsigned i = 0, e = NoBuiltinFuncs.size(); i != e; ++i)
510b57cec5SDimitry Andric     if (FuncName.equals(NoBuiltinFuncs[i]))
520b57cec5SDimitry Andric       return true;
530b57cec5SDimitry Andric   return false;
540b57cec5SDimitry Andric }
550b57cec5SDimitry Andric 
getOpenCLVersionTuple() const560b57cec5SDimitry Andric VersionTuple LangOptions::getOpenCLVersionTuple() const {
570b57cec5SDimitry Andric   const int Ver = OpenCLCPlusPlus ? OpenCLCPlusPlusVersion : OpenCLVersion;
58349cc55cSDimitry Andric   if (OpenCLCPlusPlus && Ver != 100)
59349cc55cSDimitry Andric     return VersionTuple(Ver / 100);
600b57cec5SDimitry Andric   return VersionTuple(Ver / 100, (Ver % 100) / 10);
610b57cec5SDimitry Andric }
625ffd83dbSDimitry Andric 
getOpenCLCompatibleVersion() const63349cc55cSDimitry Andric unsigned LangOptions::getOpenCLCompatibleVersion() const {
64349cc55cSDimitry Andric   if (!OpenCLCPlusPlus)
65349cc55cSDimitry Andric     return OpenCLVersion;
66349cc55cSDimitry Andric   if (OpenCLCPlusPlusVersion == 100)
67349cc55cSDimitry Andric     return 200;
68349cc55cSDimitry Andric   if (OpenCLCPlusPlusVersion == 202100)
69349cc55cSDimitry Andric     return 300;
70349cc55cSDimitry Andric   llvm_unreachable("Unknown OpenCL version");
71349cc55cSDimitry Andric }
72349cc55cSDimitry Andric 
remapPathPrefix(SmallVectorImpl<char> & Path) const7381ad6265SDimitry Andric void LangOptions::remapPathPrefix(SmallVectorImpl<char> &Path) const {
746e75b2fbSDimitry Andric   for (const auto &Entry : MacroPrefixMap)
756e75b2fbSDimitry Andric     if (llvm::sys::path::replace_path_prefix(Path, Entry.first, Entry.second))
766e75b2fbSDimitry Andric       break;
776e75b2fbSDimitry Andric }
786e75b2fbSDimitry Andric 
getOpenCLVersionString() const79349cc55cSDimitry Andric std::string LangOptions::getOpenCLVersionString() const {
80349cc55cSDimitry Andric   std::string Result;
81349cc55cSDimitry Andric   {
82349cc55cSDimitry Andric     llvm::raw_string_ostream Out(Result);
83349cc55cSDimitry Andric     Out << (OpenCLCPlusPlus ? "C++ for OpenCL" : "OpenCL C") << " version "
84349cc55cSDimitry Andric         << getOpenCLVersionTuple().getAsString();
85349cc55cSDimitry Andric   }
86349cc55cSDimitry Andric   return Result;
87349cc55cSDimitry Andric }
88349cc55cSDimitry Andric 
setLangDefaults(LangOptions & Opts,Language Lang,const llvm::Triple & T,std::vector<std::string> & Includes,LangStandard::Kind LangStd)8981ad6265SDimitry Andric void LangOptions::setLangDefaults(LangOptions &Opts, Language Lang,
9081ad6265SDimitry Andric                                   const llvm::Triple &T,
9181ad6265SDimitry Andric                                   std::vector<std::string> &Includes,
9281ad6265SDimitry Andric                                   LangStandard::Kind LangStd) {
9381ad6265SDimitry Andric   // Set some properties which depend solely on the input kind; it would be nice
9481ad6265SDimitry Andric   // to move these to the language standard, and have the driver resolve the
9581ad6265SDimitry Andric   // input kind + language standard.
9681ad6265SDimitry Andric   //
9781ad6265SDimitry Andric   // FIXME: Perhaps a better model would be for a single source file to have
9881ad6265SDimitry Andric   // multiple language standards (C / C++ std, ObjC std, OpenCL std, OpenMP std)
9981ad6265SDimitry Andric   // simultaneously active?
10081ad6265SDimitry Andric   if (Lang == Language::Asm) {
10181ad6265SDimitry Andric     Opts.AsmPreprocessor = 1;
10281ad6265SDimitry Andric   } else if (Lang == Language::ObjC || Lang == Language::ObjCXX) {
10381ad6265SDimitry Andric     Opts.ObjC = 1;
10481ad6265SDimitry Andric   }
10581ad6265SDimitry Andric 
10681ad6265SDimitry Andric   if (LangStd == LangStandard::lang_unspecified)
10781ad6265SDimitry Andric     LangStd = getDefaultLanguageStandard(Lang, T);
10881ad6265SDimitry Andric   const LangStandard &Std = LangStandard::getLangStandardForKind(LangStd);
10981ad6265SDimitry Andric   Opts.LangStd = LangStd;
11081ad6265SDimitry Andric   Opts.LineComment = Std.hasLineComments();
11181ad6265SDimitry Andric   Opts.C99 = Std.isC99();
11281ad6265SDimitry Andric   Opts.C11 = Std.isC11();
11381ad6265SDimitry Andric   Opts.C17 = Std.isC17();
1145f757f3fSDimitry Andric   Opts.C23 = Std.isC23();
11581ad6265SDimitry Andric   Opts.CPlusPlus = Std.isCPlusPlus();
11681ad6265SDimitry Andric   Opts.CPlusPlus11 = Std.isCPlusPlus11();
11781ad6265SDimitry Andric   Opts.CPlusPlus14 = Std.isCPlusPlus14();
11881ad6265SDimitry Andric   Opts.CPlusPlus17 = Std.isCPlusPlus17();
11981ad6265SDimitry Andric   Opts.CPlusPlus20 = Std.isCPlusPlus20();
12006c3fb27SDimitry Andric   Opts.CPlusPlus23 = Std.isCPlusPlus23();
12106c3fb27SDimitry Andric   Opts.CPlusPlus26 = Std.isCPlusPlus26();
12281ad6265SDimitry Andric   Opts.GNUMode = Std.isGNUMode();
12381ad6265SDimitry Andric   Opts.GNUCVersion = 0;
12481ad6265SDimitry Andric   Opts.HexFloats = Std.hasHexFloats();
12581ad6265SDimitry Andric   Opts.WChar = Std.isCPlusPlus();
12681ad6265SDimitry Andric   Opts.Digraphs = Std.hasDigraphs();
12781ad6265SDimitry Andric 
12881ad6265SDimitry Andric   Opts.HLSL = Lang == Language::HLSL;
12981ad6265SDimitry Andric   if (Opts.HLSL && Opts.IncludeDefaultHeader)
13081ad6265SDimitry Andric     Includes.push_back("hlsl.h");
13181ad6265SDimitry Andric 
13281ad6265SDimitry Andric   // Set OpenCL Version.
13381ad6265SDimitry Andric   Opts.OpenCL = Std.isOpenCL();
13481ad6265SDimitry Andric   if (LangStd == LangStandard::lang_opencl10)
13581ad6265SDimitry Andric     Opts.OpenCLVersion = 100;
13681ad6265SDimitry Andric   else if (LangStd == LangStandard::lang_opencl11)
13781ad6265SDimitry Andric     Opts.OpenCLVersion = 110;
13881ad6265SDimitry Andric   else if (LangStd == LangStandard::lang_opencl12)
13981ad6265SDimitry Andric     Opts.OpenCLVersion = 120;
14081ad6265SDimitry Andric   else if (LangStd == LangStandard::lang_opencl20)
14181ad6265SDimitry Andric     Opts.OpenCLVersion = 200;
14281ad6265SDimitry Andric   else if (LangStd == LangStandard::lang_opencl30)
14381ad6265SDimitry Andric     Opts.OpenCLVersion = 300;
14481ad6265SDimitry Andric   else if (LangStd == LangStandard::lang_openclcpp10)
14581ad6265SDimitry Andric     Opts.OpenCLCPlusPlusVersion = 100;
14681ad6265SDimitry Andric   else if (LangStd == LangStandard::lang_openclcpp2021)
14781ad6265SDimitry Andric     Opts.OpenCLCPlusPlusVersion = 202100;
14881ad6265SDimitry Andric   else if (LangStd == LangStandard::lang_hlsl2015)
14981ad6265SDimitry Andric     Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2015;
15081ad6265SDimitry Andric   else if (LangStd == LangStandard::lang_hlsl2016)
15181ad6265SDimitry Andric     Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2016;
15281ad6265SDimitry Andric   else if (LangStd == LangStandard::lang_hlsl2017)
15381ad6265SDimitry Andric     Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2017;
15481ad6265SDimitry Andric   else if (LangStd == LangStandard::lang_hlsl2018)
15581ad6265SDimitry Andric     Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2018;
15681ad6265SDimitry Andric   else if (LangStd == LangStandard::lang_hlsl2021)
15781ad6265SDimitry Andric     Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2021;
15881ad6265SDimitry Andric   else if (LangStd == LangStandard::lang_hlsl202x)
15981ad6265SDimitry Andric     Opts.HLSLVersion = (unsigned)LangOptions::HLSL_202x;
16081ad6265SDimitry Andric 
16181ad6265SDimitry Andric   // OpenCL has some additional defaults.
16281ad6265SDimitry Andric   if (Opts.OpenCL) {
16381ad6265SDimitry Andric     Opts.AltiVec = 0;
16481ad6265SDimitry Andric     Opts.ZVector = 0;
16581ad6265SDimitry Andric     Opts.setDefaultFPContractMode(LangOptions::FPM_On);
16681ad6265SDimitry Andric     Opts.OpenCLCPlusPlus = Opts.CPlusPlus;
16781ad6265SDimitry Andric     Opts.OpenCLPipes = Opts.getOpenCLCompatibleVersion() == 200;
16881ad6265SDimitry Andric     Opts.OpenCLGenericAddressSpace = Opts.getOpenCLCompatibleVersion() == 200;
16981ad6265SDimitry Andric 
17081ad6265SDimitry Andric     // Include default header file for OpenCL.
17181ad6265SDimitry Andric     if (Opts.IncludeDefaultHeader) {
17281ad6265SDimitry Andric       if (Opts.DeclareOpenCLBuiltins) {
17381ad6265SDimitry Andric         // Only include base header file for builtin types and constants.
17481ad6265SDimitry Andric         Includes.push_back("opencl-c-base.h");
17581ad6265SDimitry Andric       } else {
17681ad6265SDimitry Andric         Includes.push_back("opencl-c.h");
17781ad6265SDimitry Andric       }
17881ad6265SDimitry Andric     }
17981ad6265SDimitry Andric   }
18081ad6265SDimitry Andric 
18181ad6265SDimitry Andric   Opts.HIP = Lang == Language::HIP;
18281ad6265SDimitry Andric   Opts.CUDA = Lang == Language::CUDA || Opts.HIP;
18381ad6265SDimitry Andric   if (Opts.HIP) {
18481ad6265SDimitry Andric     // HIP toolchain does not support 'Fast' FPOpFusion in backends since it
18581ad6265SDimitry Andric     // fuses multiplication/addition instructions without contract flag from
18681ad6265SDimitry Andric     // device library functions in LLVM bitcode, which causes accuracy loss in
18781ad6265SDimitry Andric     // certain math functions, e.g. tan(-1e20) becomes -0.933 instead of 0.8446.
18881ad6265SDimitry Andric     // For device library functions in bitcode to work, 'Strict' or 'Standard'
18981ad6265SDimitry Andric     // FPOpFusion options in backends is needed. Therefore 'fast-honor-pragmas'
19081ad6265SDimitry Andric     // FP contract option is used to allow fuse across statements in frontend
19181ad6265SDimitry Andric     // whereas respecting contract flag in backend.
19281ad6265SDimitry Andric     Opts.setDefaultFPContractMode(LangOptions::FPM_FastHonorPragmas);
19381ad6265SDimitry Andric   } else if (Opts.CUDA) {
19481ad6265SDimitry Andric     if (T.isSPIRV()) {
19581ad6265SDimitry Andric       // Emit OpenCL version metadata in LLVM IR when targeting SPIR-V.
19681ad6265SDimitry Andric       Opts.OpenCLVersion = 200;
19781ad6265SDimitry Andric     }
19881ad6265SDimitry Andric     // Allow fuse across statements disregarding pragmas.
19981ad6265SDimitry Andric     Opts.setDefaultFPContractMode(LangOptions::FPM_Fast);
20081ad6265SDimitry Andric   }
20181ad6265SDimitry Andric 
20281ad6265SDimitry Andric   Opts.RenderScript = Lang == Language::RenderScript;
20381ad6265SDimitry Andric 
2045f757f3fSDimitry Andric   // OpenCL, C++ and C23 have bool, true, false keywords.
2055f757f3fSDimitry Andric   Opts.Bool = Opts.OpenCL || Opts.CPlusPlus || Opts.C23;
20681ad6265SDimitry Andric 
20781ad6265SDimitry Andric   // OpenCL and HLSL have half keyword
20881ad6265SDimitry Andric   Opts.Half = Opts.OpenCL || Opts.HLSL;
20981ad6265SDimitry Andric }
21081ad6265SDimitry Andric 
defaultWithoutTrailingStorage(const LangOptions & LO)2115ffd83dbSDimitry Andric FPOptions FPOptions::defaultWithoutTrailingStorage(const LangOptions &LO) {
2125ffd83dbSDimitry Andric   FPOptions result(LO);
2135ffd83dbSDimitry Andric   return result;
2145ffd83dbSDimitry Andric }
2155ffd83dbSDimitry Andric 
getChangesSlow(const FPOptions & Base) const21681ad6265SDimitry Andric FPOptionsOverride FPOptions::getChangesSlow(const FPOptions &Base) const {
21781ad6265SDimitry Andric   FPOptions::storage_type OverrideMask = 0;
21881ad6265SDimitry Andric #define OPTION(NAME, TYPE, WIDTH, PREVIOUS)                                    \
21981ad6265SDimitry Andric   if (get##NAME() != Base.get##NAME())                                         \
22081ad6265SDimitry Andric     OverrideMask |= NAME##Mask;
22181ad6265SDimitry Andric #include "clang/Basic/FPOptions.def"
22281ad6265SDimitry Andric   return FPOptionsOverride(*this, OverrideMask);
22381ad6265SDimitry Andric }
22481ad6265SDimitry Andric 
dump()2255ffd83dbSDimitry Andric LLVM_DUMP_METHOD void FPOptions::dump() {
2265ffd83dbSDimitry Andric #define OPTION(NAME, TYPE, WIDTH, PREVIOUS)                                    \
2275ffd83dbSDimitry Andric   llvm::errs() << "\n " #NAME " " << get##NAME();
2285ffd83dbSDimitry Andric #include "clang/Basic/FPOptions.def"
2295ffd83dbSDimitry Andric   llvm::errs() << "\n";
2305ffd83dbSDimitry Andric }
2315ffd83dbSDimitry Andric 
dump()2325ffd83dbSDimitry Andric LLVM_DUMP_METHOD void FPOptionsOverride::dump() {
2335ffd83dbSDimitry Andric #define OPTION(NAME, TYPE, WIDTH, PREVIOUS)                                    \
2345ffd83dbSDimitry Andric   if (has##NAME##Override())                                                   \
2355ffd83dbSDimitry Andric     llvm::errs() << "\n " #NAME " Override is " << get##NAME##Override();
2365ffd83dbSDimitry Andric #include "clang/Basic/FPOptions.def"
2375ffd83dbSDimitry Andric   llvm::errs() << "\n";
2385ffd83dbSDimitry Andric }
239