Path: blob/main/contrib/llvm-project/clang/lib/Basic/LangOptions.cpp
35233 views
//===- LangOptions.cpp - C Language Family Language Options ---------------===//1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===//7//8// This file defines the LangOptions class.9//10//===----------------------------------------------------------------------===//1112#include "clang/Basic/LangOptions.h"13#include "llvm/ADT/SmallString.h"14#include "llvm/Support/Path.h"1516using namespace clang;1718LangOptions::LangOptions() : LangStd(LangStandard::lang_unspecified) {19#define LANGOPT(Name, Bits, Default, Description) Name = Default;20#define ENUM_LANGOPT(Name, Type, Bits, Default, Description) set##Name(Default);21#include "clang/Basic/LangOptions.def"22}2324void LangOptions::resetNonModularOptions() {25#define LANGOPT(Name, Bits, Default, Description)26#define BENIGN_LANGOPT(Name, Bits, Default, Description) Name = Default;27#define BENIGN_ENUM_LANGOPT(Name, Type, Bits, Default, Description) \28Name = static_cast<unsigned>(Default);29#include "clang/Basic/LangOptions.def"3031// Reset "benign" options with implied values (Options.td ImpliedBy relations)32// rather than their defaults. This avoids unexpected combinations and33// invocations that cannot be round-tripped to arguments.34// FIXME: we should derive this automatically from ImpliedBy in tablegen.35AllowFPReassoc = UnsafeFPMath;36NoHonorNaNs = FiniteMathOnly;37NoHonorInfs = FiniteMathOnly;3839// These options do not affect AST generation.40NoSanitizeFiles.clear();41XRayAlwaysInstrumentFiles.clear();42XRayNeverInstrumentFiles.clear();4344CurrentModule.clear();45IsHeaderFile = false;46}4748bool LangOptions::isNoBuiltinFunc(StringRef FuncName) const {49for (unsigned i = 0, e = NoBuiltinFuncs.size(); i != e; ++i)50if (FuncName == NoBuiltinFuncs[i])51return true;52return false;53}5455VersionTuple LangOptions::getOpenCLVersionTuple() const {56const int Ver = OpenCLCPlusPlus ? OpenCLCPlusPlusVersion : OpenCLVersion;57if (OpenCLCPlusPlus && Ver != 100)58return VersionTuple(Ver / 100);59return VersionTuple(Ver / 100, (Ver % 100) / 10);60}6162unsigned LangOptions::getOpenCLCompatibleVersion() const {63if (!OpenCLCPlusPlus)64return OpenCLVersion;65if (OpenCLCPlusPlusVersion == 100)66return 200;67if (OpenCLCPlusPlusVersion == 202100)68return 300;69llvm_unreachable("Unknown OpenCL version");70}7172void LangOptions::remapPathPrefix(SmallVectorImpl<char> &Path) const {73for (const auto &Entry : MacroPrefixMap)74if (llvm::sys::path::replace_path_prefix(Path, Entry.first, Entry.second))75break;76}7778std::string LangOptions::getOpenCLVersionString() const {79std::string Result;80{81llvm::raw_string_ostream Out(Result);82Out << (OpenCLCPlusPlus ? "C++ for OpenCL" : "OpenCL C") << " version "83<< getOpenCLVersionTuple().getAsString();84}85return Result;86}8788void LangOptions::setLangDefaults(LangOptions &Opts, Language Lang,89const llvm::Triple &T,90std::vector<std::string> &Includes,91LangStandard::Kind LangStd) {92// Set some properties which depend solely on the input kind; it would be nice93// to move these to the language standard, and have the driver resolve the94// input kind + language standard.95//96// FIXME: Perhaps a better model would be for a single source file to have97// multiple language standards (C / C++ std, ObjC std, OpenCL std, OpenMP std)98// simultaneously active?99if (Lang == Language::Asm) {100Opts.AsmPreprocessor = 1;101} else if (Lang == Language::ObjC || Lang == Language::ObjCXX) {102Opts.ObjC = 1;103}104105if (LangStd == LangStandard::lang_unspecified)106LangStd = getDefaultLanguageStandard(Lang, T);107const LangStandard &Std = LangStandard::getLangStandardForKind(LangStd);108Opts.LangStd = LangStd;109Opts.LineComment = Std.hasLineComments();110Opts.C99 = Std.isC99();111Opts.C11 = Std.isC11();112Opts.C17 = Std.isC17();113Opts.C23 = Std.isC23();114Opts.C2y = Std.isC2y();115Opts.CPlusPlus = Std.isCPlusPlus();116Opts.CPlusPlus11 = Std.isCPlusPlus11();117Opts.CPlusPlus14 = Std.isCPlusPlus14();118Opts.CPlusPlus17 = Std.isCPlusPlus17();119Opts.CPlusPlus20 = Std.isCPlusPlus20();120Opts.CPlusPlus23 = Std.isCPlusPlus23();121Opts.CPlusPlus26 = Std.isCPlusPlus26();122Opts.GNUMode = Std.isGNUMode();123Opts.GNUCVersion = 0;124Opts.HexFloats = Std.hasHexFloats();125Opts.WChar = Std.isCPlusPlus();126Opts.Digraphs = Std.hasDigraphs();127Opts.RawStringLiterals = Std.hasRawStringLiterals();128129Opts.HLSL = Lang == Language::HLSL;130if (Opts.HLSL && Opts.IncludeDefaultHeader)131Includes.push_back("hlsl.h");132133// Set OpenCL Version.134Opts.OpenCL = Std.isOpenCL();135if (LangStd == LangStandard::lang_opencl10)136Opts.OpenCLVersion = 100;137else if (LangStd == LangStandard::lang_opencl11)138Opts.OpenCLVersion = 110;139else if (LangStd == LangStandard::lang_opencl12)140Opts.OpenCLVersion = 120;141else if (LangStd == LangStandard::lang_opencl20)142Opts.OpenCLVersion = 200;143else if (LangStd == LangStandard::lang_opencl30)144Opts.OpenCLVersion = 300;145else if (LangStd == LangStandard::lang_openclcpp10)146Opts.OpenCLCPlusPlusVersion = 100;147else if (LangStd == LangStandard::lang_openclcpp2021)148Opts.OpenCLCPlusPlusVersion = 202100;149else if (LangStd == LangStandard::lang_hlsl2015)150Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2015;151else if (LangStd == LangStandard::lang_hlsl2016)152Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2016;153else if (LangStd == LangStandard::lang_hlsl2017)154Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2017;155else if (LangStd == LangStandard::lang_hlsl2018)156Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2018;157else if (LangStd == LangStandard::lang_hlsl2021)158Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2021;159else if (LangStd == LangStandard::lang_hlsl202x)160Opts.HLSLVersion = (unsigned)LangOptions::HLSL_202x;161162// OpenCL has some additional defaults.163if (Opts.OpenCL) {164Opts.AltiVec = 0;165Opts.ZVector = 0;166Opts.setDefaultFPContractMode(LangOptions::FPM_On);167Opts.OpenCLCPlusPlus = Opts.CPlusPlus;168Opts.OpenCLPipes = Opts.getOpenCLCompatibleVersion() == 200;169Opts.OpenCLGenericAddressSpace = Opts.getOpenCLCompatibleVersion() == 200;170171// Include default header file for OpenCL.172if (Opts.IncludeDefaultHeader) {173if (Opts.DeclareOpenCLBuiltins) {174// Only include base header file for builtin types and constants.175Includes.push_back("opencl-c-base.h");176} else {177Includes.push_back("opencl-c.h");178}179}180}181182Opts.HIP = Lang == Language::HIP;183Opts.CUDA = Lang == Language::CUDA || Opts.HIP;184if (Opts.HIP) {185// HIP toolchain does not support 'Fast' FPOpFusion in backends since it186// fuses multiplication/addition instructions without contract flag from187// device library functions in LLVM bitcode, which causes accuracy loss in188// certain math functions, e.g. tan(-1e20) becomes -0.933 instead of 0.8446.189// For device library functions in bitcode to work, 'Strict' or 'Standard'190// FPOpFusion options in backends is needed. Therefore 'fast-honor-pragmas'191// FP contract option is used to allow fuse across statements in frontend192// whereas respecting contract flag in backend.193Opts.setDefaultFPContractMode(LangOptions::FPM_FastHonorPragmas);194} else if (Opts.CUDA) {195if (T.isSPIRV()) {196// Emit OpenCL version metadata in LLVM IR when targeting SPIR-V.197Opts.OpenCLVersion = 200;198}199// Allow fuse across statements disregarding pragmas.200Opts.setDefaultFPContractMode(LangOptions::FPM_Fast);201}202203Opts.RenderScript = Lang == Language::RenderScript;204205// OpenCL, C++ and C23 have bool, true, false keywords.206Opts.Bool = Opts.OpenCL || Opts.CPlusPlus || Opts.C23;207208// OpenCL and HLSL have half keyword209Opts.Half = Opts.OpenCL || Opts.HLSL;210}211212FPOptions FPOptions::defaultWithoutTrailingStorage(const LangOptions &LO) {213FPOptions result(LO);214return result;215}216217FPOptionsOverride FPOptions::getChangesSlow(const FPOptions &Base) const {218FPOptions::storage_type OverrideMask = 0;219#define OPTION(NAME, TYPE, WIDTH, PREVIOUS) \220if (get##NAME() != Base.get##NAME()) \221OverrideMask |= NAME##Mask;222#include "clang/Basic/FPOptions.def"223return FPOptionsOverride(*this, OverrideMask);224}225226LLVM_DUMP_METHOD void FPOptions::dump() {227#define OPTION(NAME, TYPE, WIDTH, PREVIOUS) \228llvm::errs() << "\n " #NAME " " << get##NAME();229#include "clang/Basic/FPOptions.def"230llvm::errs() << "\n";231}232233LLVM_DUMP_METHOD void FPOptionsOverride::dump() {234#define OPTION(NAME, TYPE, WIDTH, PREVIOUS) \235if (has##NAME##Override()) \236llvm::errs() << "\n " #NAME " Override is " << get##NAME##Override();237#include "clang/Basic/FPOptions.def"238llvm::errs() << "\n";239}240241242