LangOptions.cpp 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. //===- LangOptions.cpp - C Language Family Language Options ---------------===//
  2. //
  3. // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
  4. // See https://llvm.org/LICENSE.txt for license information.
  5. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  6. //
  7. //===----------------------------------------------------------------------===//
  8. //
  9. // This file defines the LangOptions class.
  10. //
  11. //===----------------------------------------------------------------------===//
  12. #include "clang/Basic/LangOptions.h"
  13. #include "llvm/ADT/SmallString.h"
  14. #include "llvm/Support/Path.h"
  15. using namespace clang;
  16. LangOptions::LangOptions() : LangStd(LangStandard::lang_unspecified) {
  17. #define LANGOPT(Name, Bits, Default, Description) Name = Default;
  18. #define ENUM_LANGOPT(Name, Type, Bits, Default, Description) set##Name(Default);
  19. #include "clang/Basic/LangOptions.def"
  20. }
  21. void LangOptions::resetNonModularOptions() {
  22. #define LANGOPT(Name, Bits, Default, Description)
  23. #define BENIGN_LANGOPT(Name, Bits, Default, Description) Name = Default;
  24. #define BENIGN_ENUM_LANGOPT(Name, Type, Bits, Default, Description) \
  25. Name = static_cast<unsigned>(Default);
  26. #include "clang/Basic/LangOptions.def"
  27. // These options do not affect AST generation.
  28. NoSanitizeFiles.clear();
  29. XRayAlwaysInstrumentFiles.clear();
  30. XRayNeverInstrumentFiles.clear();
  31. CurrentModule.clear();
  32. IsHeaderFile = false;
  33. }
  34. bool LangOptions::isNoBuiltinFunc(StringRef FuncName) const {
  35. for (unsigned i = 0, e = NoBuiltinFuncs.size(); i != e; ++i)
  36. if (FuncName.equals(NoBuiltinFuncs[i]))
  37. return true;
  38. return false;
  39. }
  40. VersionTuple LangOptions::getOpenCLVersionTuple() const {
  41. const int Ver = OpenCLCPlusPlus ? OpenCLCPlusPlusVersion : OpenCLVersion;
  42. if (OpenCLCPlusPlus && Ver != 100)
  43. return VersionTuple(Ver / 100);
  44. return VersionTuple(Ver / 100, (Ver % 100) / 10);
  45. }
  46. unsigned LangOptions::getOpenCLCompatibleVersion() const {
  47. if (!OpenCLCPlusPlus)
  48. return OpenCLVersion;
  49. if (OpenCLCPlusPlusVersion == 100)
  50. return 200;
  51. if (OpenCLCPlusPlusVersion == 202100)
  52. return 300;
  53. llvm_unreachable("Unknown OpenCL version");
  54. }
  55. void LangOptions::remapPathPrefix(SmallVectorImpl<char> &Path) const {
  56. for (const auto &Entry : MacroPrefixMap)
  57. if (llvm::sys::path::replace_path_prefix(Path, Entry.first, Entry.second))
  58. break;
  59. }
  60. std::string LangOptions::getOpenCLVersionString() const {
  61. std::string Result;
  62. {
  63. llvm::raw_string_ostream Out(Result);
  64. Out << (OpenCLCPlusPlus ? "C++ for OpenCL" : "OpenCL C") << " version "
  65. << getOpenCLVersionTuple().getAsString();
  66. }
  67. return Result;
  68. }
  69. void LangOptions::setLangDefaults(LangOptions &Opts, Language Lang,
  70. const llvm::Triple &T,
  71. std::vector<std::string> &Includes,
  72. LangStandard::Kind LangStd) {
  73. // Set some properties which depend solely on the input kind; it would be nice
  74. // to move these to the language standard, and have the driver resolve the
  75. // input kind + language standard.
  76. //
  77. // FIXME: Perhaps a better model would be for a single source file to have
  78. // multiple language standards (C / C++ std, ObjC std, OpenCL std, OpenMP std)
  79. // simultaneously active?
  80. if (Lang == Language::Asm) {
  81. Opts.AsmPreprocessor = 1;
  82. } else if (Lang == Language::ObjC || Lang == Language::ObjCXX) {
  83. Opts.ObjC = 1;
  84. }
  85. if (LangStd == LangStandard::lang_unspecified)
  86. LangStd = getDefaultLanguageStandard(Lang, T);
  87. const LangStandard &Std = LangStandard::getLangStandardForKind(LangStd);
  88. Opts.LangStd = LangStd;
  89. Opts.LineComment = Std.hasLineComments();
  90. Opts.C99 = Std.isC99();
  91. Opts.C11 = Std.isC11();
  92. Opts.C17 = Std.isC17();
  93. Opts.C2x = Std.isC2x();
  94. Opts.CPlusPlus = Std.isCPlusPlus();
  95. Opts.CPlusPlus11 = Std.isCPlusPlus11();
  96. Opts.CPlusPlus14 = Std.isCPlusPlus14();
  97. Opts.CPlusPlus17 = Std.isCPlusPlus17();
  98. Opts.CPlusPlus20 = Std.isCPlusPlus20();
  99. Opts.CPlusPlus2b = Std.isCPlusPlus2b();
  100. Opts.GNUMode = Std.isGNUMode();
  101. Opts.GNUCVersion = 0;
  102. Opts.HexFloats = Std.hasHexFloats();
  103. Opts.WChar = Std.isCPlusPlus();
  104. Opts.Digraphs = Std.hasDigraphs();
  105. Opts.HLSL = Lang == Language::HLSL;
  106. if (Opts.HLSL && Opts.IncludeDefaultHeader)
  107. Includes.push_back("hlsl.h");
  108. // Set OpenCL Version.
  109. Opts.OpenCL = Std.isOpenCL();
  110. if (LangStd == LangStandard::lang_opencl10)
  111. Opts.OpenCLVersion = 100;
  112. else if (LangStd == LangStandard::lang_opencl11)
  113. Opts.OpenCLVersion = 110;
  114. else if (LangStd == LangStandard::lang_opencl12)
  115. Opts.OpenCLVersion = 120;
  116. else if (LangStd == LangStandard::lang_opencl20)
  117. Opts.OpenCLVersion = 200;
  118. else if (LangStd == LangStandard::lang_opencl30)
  119. Opts.OpenCLVersion = 300;
  120. else if (LangStd == LangStandard::lang_openclcpp10)
  121. Opts.OpenCLCPlusPlusVersion = 100;
  122. else if (LangStd == LangStandard::lang_openclcpp2021)
  123. Opts.OpenCLCPlusPlusVersion = 202100;
  124. else if (LangStd == LangStandard::lang_hlsl2015)
  125. Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2015;
  126. else if (LangStd == LangStandard::lang_hlsl2016)
  127. Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2016;
  128. else if (LangStd == LangStandard::lang_hlsl2017)
  129. Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2017;
  130. else if (LangStd == LangStandard::lang_hlsl2018)
  131. Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2018;
  132. else if (LangStd == LangStandard::lang_hlsl2021)
  133. Opts.HLSLVersion = (unsigned)LangOptions::HLSL_2021;
  134. else if (LangStd == LangStandard::lang_hlsl202x)
  135. Opts.HLSLVersion = (unsigned)LangOptions::HLSL_202x;
  136. // OpenCL has some additional defaults.
  137. if (Opts.OpenCL) {
  138. Opts.AltiVec = 0;
  139. Opts.ZVector = 0;
  140. Opts.setDefaultFPContractMode(LangOptions::FPM_On);
  141. Opts.OpenCLCPlusPlus = Opts.CPlusPlus;
  142. Opts.OpenCLPipes = Opts.getOpenCLCompatibleVersion() == 200;
  143. Opts.OpenCLGenericAddressSpace = Opts.getOpenCLCompatibleVersion() == 200;
  144. // Include default header file for OpenCL.
  145. if (Opts.IncludeDefaultHeader) {
  146. if (Opts.DeclareOpenCLBuiltins) {
  147. // Only include base header file for builtin types and constants.
  148. Includes.push_back("opencl-c-base.h");
  149. } else {
  150. Includes.push_back("opencl-c.h");
  151. }
  152. }
  153. }
  154. Opts.HIP = Lang == Language::HIP;
  155. Opts.CUDA = Lang == Language::CUDA || Opts.HIP;
  156. if (Opts.HIP) {
  157. // HIP toolchain does not support 'Fast' FPOpFusion in backends since it
  158. // fuses multiplication/addition instructions without contract flag from
  159. // device library functions in LLVM bitcode, which causes accuracy loss in
  160. // certain math functions, e.g. tan(-1e20) becomes -0.933 instead of 0.8446.
  161. // For device library functions in bitcode to work, 'Strict' or 'Standard'
  162. // FPOpFusion options in backends is needed. Therefore 'fast-honor-pragmas'
  163. // FP contract option is used to allow fuse across statements in frontend
  164. // whereas respecting contract flag in backend.
  165. Opts.setDefaultFPContractMode(LangOptions::FPM_FastHonorPragmas);
  166. } else if (Opts.CUDA) {
  167. if (T.isSPIRV()) {
  168. // Emit OpenCL version metadata in LLVM IR when targeting SPIR-V.
  169. Opts.OpenCLVersion = 200;
  170. }
  171. // Allow fuse across statements disregarding pragmas.
  172. Opts.setDefaultFPContractMode(LangOptions::FPM_Fast);
  173. }
  174. Opts.RenderScript = Lang == Language::RenderScript;
  175. // OpenCL, C++ and C2x have bool, true, false keywords.
  176. Opts.Bool = Opts.OpenCL || Opts.CPlusPlus || Opts.C2x;
  177. // OpenCL and HLSL have half keyword
  178. Opts.Half = Opts.OpenCL || Opts.HLSL;
  179. }
  180. FPOptions FPOptions::defaultWithoutTrailingStorage(const LangOptions &LO) {
  181. FPOptions result(LO);
  182. return result;
  183. }
  184. FPOptionsOverride FPOptions::getChangesSlow(const FPOptions &Base) const {
  185. FPOptions::storage_type OverrideMask = 0;
  186. #define OPTION(NAME, TYPE, WIDTH, PREVIOUS) \
  187. if (get##NAME() != Base.get##NAME()) \
  188. OverrideMask |= NAME##Mask;
  189. #include "clang/Basic/FPOptions.def"
  190. return FPOptionsOverride(*this, OverrideMask);
  191. }
  192. LLVM_DUMP_METHOD void FPOptions::dump() {
  193. #define OPTION(NAME, TYPE, WIDTH, PREVIOUS) \
  194. llvm::errs() << "\n " #NAME " " << get##NAME();
  195. #include "clang/Basic/FPOptions.def"
  196. llvm::errs() << "\n";
  197. }
  198. LLVM_DUMP_METHOD void FPOptionsOverride::dump() {
  199. #define OPTION(NAME, TYPE, WIDTH, PREVIOUS) \
  200. if (has##NAME##Override()) \
  201. llvm::errs() << "\n " #NAME " Override is " << get##NAME##Override();
  202. #include "clang/Basic/FPOptions.def"
  203. llvm::errs() << "\n";
  204. }