HLSLOptions.cpp 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850
  1. //===--- HLSLOptions.cpp - Driver Options Table ---------------------------===//
  2. ///////////////////////////////////////////////////////////////////////////////
  3. // //
  4. // HLSLOptions.cpp //
  5. // Copyright (C) Microsoft Corporation. All rights reserved. //
  6. // This file is distributed under the University of Illinois Open Source //
  7. // License. See LICENSE.TXT for details. //
  8. // //
  9. ///////////////////////////////////////////////////////////////////////////////
  10. #include "llvm/ADT/STLExtras.h"
  11. #include "llvm/Option/OptTable.h"
  12. #include "llvm/Option/Option.h"
  13. #include "llvm/Support/raw_ostream.h"
  14. #include "llvm/Support/Path.h"
  15. #include "llvm/ADT/APInt.h"
  16. #include "dxc/Support/Global.h"
  17. #include "dxc/Support/WinIncludes.h"
  18. #include "dxc/Support/HLSLOptions.h"
  19. #include "dxc/Support/Unicode.h"
  20. #include "dxc/Support/dxcapi.use.h"
  21. #include "dxc/DXIL/DxilShaderModel.h"
  22. using namespace llvm::opt;
  23. using namespace dxc;
  24. using namespace hlsl;
  25. using namespace hlsl::options;
  26. #define PREFIX(NAME, VALUE) static const char *const NAME[] = VALUE;
  27. #include "dxc/Support/HLSLOptions.inc"
  28. #undef PREFIX
  29. static const OptTable::Info HlslInfoTable[] = {
  30. #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
  31. HELPTEXT, METAVAR) \
  32. { PREFIX, NAME, HELPTEXT, METAVAR, OPT_##ID, Option::KIND##Class, PARAM, \
  33. FLAGS, OPT_##GROUP, OPT_##ALIAS, ALIASARGS },
  34. #include "dxc/Support/HLSLOptions.inc"
  35. #undef OPTION
  36. };
  37. namespace {
  38. class HlslOptTable : public OptTable {
  39. public:
  40. HlslOptTable()
  41. : OptTable(HlslInfoTable, llvm::array_lengthof(HlslInfoTable)) {}
  42. };
  43. }
  44. static HlslOptTable *g_HlslOptTable;
  45. std::error_code hlsl::options::initHlslOptTable() {
  46. DXASSERT(g_HlslOptTable == nullptr, "else double-init");
  47. g_HlslOptTable = new (std::nothrow) HlslOptTable();
  48. if (g_HlslOptTable == nullptr)
  49. return std::error_code(E_OUTOFMEMORY, std::system_category());
  50. return std::error_code();
  51. }
  52. void hlsl::options::cleanupHlslOptTable() {
  53. delete g_HlslOptTable;
  54. g_HlslOptTable = nullptr;
  55. }
  56. const OptTable * hlsl::options::getHlslOptTable() {
  57. return g_HlslOptTable;
  58. }
  59. void DxcDefines::push_back(llvm::StringRef value) {
  60. // Skip empty defines.
  61. if (value.size() > 0) {
  62. DefineStrings.push_back(value);
  63. }
  64. }
  65. UINT32 DxcDefines::ComputeNumberOfWCharsNeededForDefines() {
  66. UINT32 wcharSize = 0;
  67. for (llvm::StringRef &S : DefineStrings) {
  68. DXASSERT(S.size() > 0,
  69. "else DxcDefines::push_back should not have added this");
  70. const int utf16Length = ::MultiByteToWideChar(
  71. CP_UTF8, MB_ERR_INVALID_CHARS, S.data(), S.size(), nullptr, 0);
  72. IFTARG(utf16Length != 0);
  73. wcharSize += utf16Length + 1; // adding null terminated character
  74. }
  75. return wcharSize;
  76. }
  77. void DxcDefines::BuildDefines() {
  78. // Calculate and prepare the size of the backing buffer.
  79. DXASSERT(DefineValues == nullptr, "else DxcDefines is already built");
  80. UINT32 wcharSize = ComputeNumberOfWCharsNeededForDefines();
  81. DefineValues = new wchar_t[wcharSize];
  82. DefineVector.resize(DefineStrings.size());
  83. // Build up the define structures while filling in the backing buffer.
  84. UINT32 remaining = wcharSize;
  85. LPWSTR pWriteCursor = DefineValues;
  86. for (size_t i = 0; i < DefineStrings.size(); ++i) {
  87. llvm::StringRef &S = DefineStrings[i];
  88. DxcDefine &D = DefineVector[i];
  89. const int utf16Length =
  90. ::MultiByteToWideChar(CP_UTF8, MB_ERR_INVALID_CHARS, S.data(), S.size(),
  91. pWriteCursor, remaining);
  92. DXASSERT(utf16Length > 0,
  93. "else it should have failed during size calculation");
  94. LPWSTR pDefineEnd = pWriteCursor + utf16Length;
  95. D.Name = pWriteCursor;
  96. LPWSTR pEquals = std::find(pWriteCursor, pDefineEnd, L'=');
  97. if (pEquals == pDefineEnd) {
  98. D.Value = nullptr;
  99. } else {
  100. *pEquals = L'\0';
  101. D.Value = pEquals + 1;
  102. }
  103. // Advance past converted characters and include the null terminator.
  104. pWriteCursor += utf16Length;
  105. *pWriteCursor = L'\0';
  106. ++pWriteCursor;
  107. DXASSERT(pWriteCursor <= DefineValues + wcharSize,
  108. "else this function is calculating this incorrectly");
  109. remaining -= (utf16Length + 1);
  110. }
  111. }
  112. bool DxcOpts::IsRootSignatureProfile() {
  113. return TargetProfile == "rootsig_1_0" ||
  114. TargetProfile == "rootsig_1_1";
  115. }
  116. bool DxcOpts::IsLibraryProfile() {
  117. return TargetProfile.startswith("lib_");
  118. }
  119. bool DxcOpts::IsDebugInfoEnabled() {
  120. return DebugInfo;
  121. }
  122. bool DxcOpts::EmbedDebugInfo() {
  123. return EmbedDebug;
  124. }
  125. bool DxcOpts::EmbedPDBName() {
  126. return IsDebugInfoEnabled() || !DebugFile.empty();
  127. }
  128. bool DxcOpts::DebugFileIsDirectory() {
  129. return !DebugFile.empty() && llvm::sys::path::is_separator(DebugFile[DebugFile.size() - 1]);
  130. }
  131. llvm::StringRef DxcOpts::GetPDBName() {
  132. if (!DebugFileIsDirectory())
  133. return DebugFile;
  134. return llvm::StringRef();
  135. }
  136. MainArgs::MainArgs(int argc, const wchar_t **argv, int skipArgCount) {
  137. if (argc > skipArgCount) {
  138. Utf8StringVector.reserve(argc - skipArgCount);
  139. Utf8CharPtrVector.reserve(argc - skipArgCount);
  140. for (int i = skipArgCount; i < argc; ++i) {
  141. Utf8StringVector.emplace_back(Unicode::UTF16ToUTF8StringOrThrow(argv[i]));
  142. Utf8CharPtrVector.push_back(Utf8StringVector.back().data());
  143. }
  144. }
  145. }
  146. MainArgs::MainArgs(int argc, const char **argv, int skipArgCount) {
  147. if (argc > skipArgCount) {
  148. Utf8StringVector.reserve(argc - skipArgCount);
  149. Utf8CharPtrVector.reserve(argc - skipArgCount);
  150. for (int i = skipArgCount; i < argc; ++i) {
  151. Utf8StringVector.emplace_back(argv[i]);
  152. Utf8CharPtrVector.push_back(Utf8StringVector.back().data());
  153. }
  154. }
  155. }
  156. MainArgs::MainArgs(llvm::ArrayRef<llvm::StringRef> args) {
  157. Utf8StringVector.reserve(args.size());
  158. Utf8CharPtrVector.reserve(args.size());
  159. for (llvm::StringRef str : args) {
  160. Utf8StringVector.emplace_back(str.str());
  161. Utf8CharPtrVector.push_back(Utf8StringVector.back().data());
  162. }
  163. }
  164. MainArgs& MainArgs::operator=(const MainArgs &other) {
  165. Utf8StringVector.clear();
  166. Utf8CharPtrVector.clear();
  167. Utf8StringVector.reserve(other.Utf8StringVector.size());
  168. Utf8CharPtrVector.reserve(other.Utf8StringVector.size());
  169. for (const std::string &str : other.Utf8StringVector) {
  170. Utf8StringVector.emplace_back(str);
  171. Utf8CharPtrVector.push_back(Utf8StringVector.back().data());
  172. }
  173. return *this;
  174. }
  175. StringRefUtf16::StringRefUtf16(llvm::StringRef value) {
  176. if (!value.empty())
  177. m_value = Unicode::UTF8ToUTF16StringOrThrow(value.data());
  178. }
  179. static bool GetTargetVersionFromString(llvm::StringRef ref, unsigned *major, unsigned *minor) {
  180. *major = *minor = -1;
  181. unsigned len = ref.size();
  182. if (len < 6 || len > 11) // length: ps_6_0 to rootsig_1_0
  183. return false;
  184. if (ref[len - 4] != '_' || ref[len - 2] != '_')
  185. return false;
  186. char cMajor = ref[len - 3];
  187. char cMinor = ref[len - 1];
  188. if (cMajor >= '0' && cMajor <= '9')
  189. *major = cMajor - '0';
  190. else
  191. return false;
  192. if (cMinor == 'x')
  193. *minor = 0xF;
  194. else if (cMinor >= '0' && cMinor <= '9')
  195. *minor = cMinor - '0';
  196. else
  197. return false;
  198. return true;
  199. }
  200. // SPIRV Change Starts
  201. #ifdef ENABLE_SPIRV_CODEGEN
  202. /// Checks and collects the arguments for -fvk-{b|s|t|u}-shift into *shifts.
  203. static bool handleVkShiftArgs(const InputArgList &args, OptSpecifier id,
  204. const char *name,
  205. llvm::SmallVectorImpl<int32_t> *shifts,
  206. llvm::raw_ostream &errors) {
  207. const auto values = args.getAllArgValues(id);
  208. if (values.empty())
  209. return true;
  210. if (!args.hasArg(OPT_spirv)) {
  211. errors << "-fvk-" << name << "-shift requires -spirv";
  212. return false;
  213. }
  214. if (!args.getLastArgValue(OPT_fvk_bind_register).empty()) {
  215. errors << "-fvk-" << name
  216. << "-shift cannot be used together with -fvk-bind-register";
  217. return false;
  218. }
  219. shifts->clear();
  220. bool setForAll = false;
  221. for (const auto &val : values) {
  222. int32_t number = 0;
  223. if (val == "all") {
  224. number = -1;
  225. setForAll = true;
  226. } else {
  227. if (llvm::StringRef(val).getAsInteger(10, number)) {
  228. errors << "invalid -fvk-" << name << "-shift argument: " << val;
  229. return false;
  230. }
  231. if (number < 0) {
  232. errors << "negative -fvk-" << name << "-shift argument: " << val;
  233. return false;
  234. }
  235. }
  236. shifts->push_back(number);
  237. }
  238. if (setForAll && shifts->size() > 2) {
  239. errors << "setting all sets via -fvk-" << name
  240. << "-shift argument should be used alone";
  241. return false;
  242. }
  243. return true;
  244. };
  245. #endif
  246. // SPIRV Change Ends
  247. namespace hlsl {
  248. namespace options {
  249. /// Reads all options from the given argument strings, populates opts, and
  250. /// validates reporting errors and warnings.
  251. int ReadDxcOpts(const OptTable *optionTable, unsigned flagsToInclude,
  252. const MainArgs &argStrings, DxcOpts &opts,
  253. llvm::raw_ostream &errors) {
  254. DXASSERT_NOMSG(optionTable != nullptr);
  255. unsigned missingArgIndex = 0, missingArgCount = 0;
  256. InputArgList Args = optionTable->ParseArgs(
  257. argStrings.getArrayRef(), missingArgIndex, missingArgCount, flagsToInclude);
  258. // Verify consistency for external library support.
  259. opts.ExternalLib = Args.getLastArgValue(OPT_external_lib);
  260. opts.ExternalFn = Args.getLastArgValue(OPT_external_fn);
  261. if (opts.ExternalLib.empty()) {
  262. if (!opts.ExternalFn.empty()) {
  263. errors << "External function cannot be specified without an external "
  264. "library name.";
  265. return 1;
  266. }
  267. }
  268. else {
  269. if (opts.ExternalFn.empty()) {
  270. errors << "External library name requires specifying an external "
  271. "function name.";
  272. return 1;
  273. }
  274. }
  275. opts.ShowHelp = Args.hasFlag(OPT_help, OPT_INVALID, false);
  276. opts.ShowHelp |= (opts.ShowHelpHidden = Args.hasFlag(OPT__help_hidden, OPT_INVALID, false));
  277. if (opts.ShowHelp) {
  278. return 0;
  279. }
  280. if (missingArgCount) {
  281. errors << "Argument to '" << Args.getArgString(missingArgIndex)
  282. << "' is missing.";
  283. return 1;
  284. }
  285. if (!Args.hasArg(hlsl::options::OPT_Qunused_arguments)) {
  286. for (const Arg *A : Args.filtered(OPT_UNKNOWN)) {
  287. errors << "Unknown argument: '" << A->getAsString(Args).c_str() << "'";
  288. return 1;
  289. }
  290. }
  291. // Add macros from the command line.
  292. for (const Arg *A : Args.filtered(OPT_D)) {
  293. opts.Defines.push_back(A->getValue());
  294. // If supporting OPT_U and included in filter, handle undefs.
  295. }
  296. opts.Defines.BuildDefines(); // Must be called after all defines are pushed back
  297. DXASSERT(opts.ExternalLib.empty() == opts.ExternalFn.empty(),
  298. "else flow above is incorrect");
  299. // when no-warnings option is present, do not output warnings.
  300. opts.OutputWarnings = Args.hasFlag(OPT_INVALID, OPT_no_warnings, true);
  301. opts.EntryPoint = Args.getLastArgValue(OPT_entrypoint);
  302. // Entry point is required in arguments only for drivers; APIs take this through an argument.
  303. // The value should default to 'main', but we let the caller apply this policy.
  304. if (opts.TargetProfile.empty()) {
  305. opts.TargetProfile = Args.getLastArgValue(OPT_target_profile);
  306. }
  307. if (opts.IsLibraryProfile()) {
  308. if (Args.getLastArg(OPT_entrypoint)) {
  309. errors << "cannot specify entry point for a library";
  310. return 1;
  311. } else {
  312. // Set entry point to impossible name.
  313. opts.EntryPoint = "lib.no::entry";
  314. }
  315. } else {
  316. if (Args.getLastArg(OPT_exports)) {
  317. errors << "library profile required when using -exports option";
  318. return 1;
  319. } else if (Args.hasFlag(OPT_export_shaders_only, OPT_INVALID, false)) {
  320. errors << "library profile required when using -export-shaders-only option";
  321. return 1;
  322. } else if (Args.getLastArg(OPT_default_linkage)) {
  323. errors << "library profile required when using -default-linkage option";
  324. return 1;
  325. }
  326. }
  327. opts.EnableDX9CompatMode = Args.hasFlag(OPT_Gec, OPT_INVALID, false);
  328. llvm::StringRef ver = Args.getLastArgValue(OPT_hlsl_version);
  329. if (ver.empty()) {
  330. if (opts.EnableDX9CompatMode)
  331. opts.HLSLVersion = 2016; // Default to max supported version with /Gec flag
  332. else
  333. opts.HLSLVersion = 2018; // Default to latest version
  334. } else {
  335. try {
  336. opts.HLSLVersion = std::stoul(std::string(ver));
  337. if (opts.HLSLVersion < 2015 || opts.HLSLVersion > 2018) {
  338. errors << "Unknown HLSL version: " << opts.HLSLVersion;
  339. return 1;
  340. }
  341. }
  342. catch (const std::invalid_argument &) {
  343. errors << "Invalid HLSL Version";
  344. return 1;
  345. }
  346. catch (const std::out_of_range &) {
  347. errors << "Invalid HLSL Version";
  348. return 1;
  349. }
  350. }
  351. if (opts.HLSLVersion == 2015 && !(flagsToInclude & HlslFlags::ISenseOption)) {
  352. errors << "HLSL Version 2015 is only supported for language services";
  353. return 1;
  354. }
  355. if (opts.EnableDX9CompatMode && opts.HLSLVersion > 2016) {
  356. errors << "/Gec is not supported with HLSLVersion " << opts.HLSLVersion;
  357. return 1;
  358. }
  359. if (opts.HLSLVersion <= 2016) {
  360. opts.EnableFXCCompatMode = true;
  361. }
  362. // AssemblyCodeHex not supported (Fx)
  363. // OutputLibrary not supported (Fl)
  364. opts.AssemblyCode = Args.getLastArgValue(OPT_Fc);
  365. opts.DebugFile = Args.getLastArgValue(OPT_Fd);
  366. opts.ExtractPrivateFile = Args.getLastArgValue(OPT_getprivate);
  367. opts.Enable16BitTypes = Args.hasFlag(OPT_enable_16bit_types, OPT_INVALID, false);
  368. opts.OutputObject = Args.getLastArgValue(OPT_Fo);
  369. opts.OutputHeader = Args.getLastArgValue(OPT_Fh);
  370. opts.OutputWarningsFile = Args.getLastArgValue(OPT_Fe);
  371. opts.UseColor = Args.hasFlag(OPT_Cc, OPT_INVALID, false);
  372. opts.UseInstructionNumbers = Args.hasFlag(OPT_Ni, OPT_INVALID, false);
  373. opts.UseInstructionByteOffsets = Args.hasFlag(OPT_No, OPT_INVALID, false);
  374. opts.UseHexLiterals = Args.hasFlag(OPT_Lx, OPT_INVALID, false);
  375. opts.Preprocess = Args.getLastArgValue(OPT_P);
  376. opts.AstDump = Args.hasFlag(OPT_ast_dump, OPT_INVALID, false);
  377. opts.CodeGenHighLevel = Args.hasFlag(OPT_fcgl, OPT_INVALID, false);
  378. opts.DebugInfo = Args.hasFlag(OPT__SLASH_Zi, OPT_INVALID, false);
  379. opts.DebugNameForBinary = Args.hasFlag(OPT_Zsb, OPT_INVALID, false);
  380. opts.DebugNameForSource = Args.hasFlag(OPT_Zss, OPT_INVALID, false);
  381. opts.VariableName = Args.getLastArgValue(OPT_Vn);
  382. opts.InputFile = Args.getLastArgValue(OPT_INPUT);
  383. opts.ForceRootSigVer = Args.getLastArgValue(OPT_force_rootsig_ver);
  384. opts.PrivateSource = Args.getLastArgValue(OPT_setprivate);
  385. opts.RootSignatureSource = Args.getLastArgValue(OPT_setrootsignature);
  386. opts.VerifyRootSignatureSource = Args.getLastArgValue(OPT_verifyrootsignature);
  387. opts.RootSignatureDefine = Args.getLastArgValue(OPT_rootsig_define);
  388. if (!opts.ForceRootSigVer.empty() && opts.ForceRootSigVer != "rootsig_1_0" &&
  389. opts.ForceRootSigVer != "rootsig_1_1") {
  390. errors << "Unsupported value '" << opts.ForceRootSigVer
  391. << "' for root signature profile.";
  392. return 1;
  393. }
  394. opts.IEEEStrict = Args.hasFlag(OPT_Gis, OPT_INVALID, false);
  395. opts.IgnoreLineDirectives = Args.hasFlag(OPT_ignore_line_directives, OPT_INVALID, false);
  396. opts.FloatDenormalMode = Args.getLastArgValue(OPT_denorm);
  397. // Check if a given denormalized value is valid
  398. if (!opts.FloatDenormalMode.empty()) {
  399. if (!(opts.FloatDenormalMode.equals_lower("preserve") ||
  400. opts.FloatDenormalMode.equals_lower("ftz") ||
  401. opts.FloatDenormalMode.equals_lower("any"))) {
  402. errors << "Unsupported value '" << opts.FloatDenormalMode
  403. << "' for denorm option.";
  404. return 1;
  405. }
  406. }
  407. llvm::StringRef auto_binding_space = Args.getLastArgValue(OPT_auto_binding_space);
  408. if (!auto_binding_space.empty()) {
  409. if (auto_binding_space.getAsInteger(10, opts.AutoBindingSpace)) {
  410. errors << "Unsupported value '" << auto_binding_space << "' for auto binding space.";
  411. return 1;
  412. }
  413. }
  414. opts.Exports = Args.getAllArgValues(OPT_exports);
  415. opts.DefaultLinkage = Args.getLastArgValue(OPT_default_linkage);
  416. if (!opts.DefaultLinkage.empty()) {
  417. if (!(opts.DefaultLinkage.equals_lower("internal") ||
  418. opts.DefaultLinkage.equals_lower("external"))) {
  419. errors << "Unsupported value '" << opts.DefaultLinkage
  420. << "for -default-linkage option.";
  421. return 1;
  422. }
  423. }
  424. // Check options only allowed in shader model >= 6.2FPDenormalMode
  425. unsigned Major = 0;
  426. unsigned Minor = 0;
  427. if (!opts.TargetProfile.empty()) {
  428. if (!GetTargetVersionFromString(opts.TargetProfile, &Major, &Minor)) {
  429. errors << "unable to parse shader model.";
  430. return 1;
  431. }
  432. }
  433. if (opts.TargetProfile.empty() || Major < 6 || (Major == 6 && Minor < 2)) {
  434. if (!opts.FloatDenormalMode.empty()) {
  435. errors << "denorm option is only allowed for shader model 6.2 and above.";
  436. return 1;
  437. }
  438. }
  439. // /enable-16bit-types only allowed for HLSL 2018 and shader model 6.2
  440. if (opts.Enable16BitTypes) {
  441. if (opts.TargetProfile.empty() || opts.HLSLVersion < 2018
  442. || Major < 6 || (Major == 6 && Minor < 2)) {
  443. errors << "enable-16bit-types is only allowed for shader model >= 6.2 and HLSL Language >= 2018.";
  444. return 1;
  445. }
  446. }
  447. opts.DisableOptimizations = false;
  448. if (Arg *A = Args.getLastArg(OPT_O0, OPT_O1, OPT_O2, OPT_O3, OPT_Od)) {
  449. if (A->getOption().matches(OPT_O0))
  450. opts.OptLevel = 0;
  451. if (A->getOption().matches(OPT_O1))
  452. opts.OptLevel = 1;
  453. if (A->getOption().matches(OPT_O2))
  454. opts.OptLevel = 2;
  455. if (A->getOption().matches(OPT_O3))
  456. opts.OptLevel = 3;
  457. if (A->getOption().matches(OPT_Od)) {
  458. opts.DisableOptimizations = true;
  459. opts.OptLevel = 0;
  460. }
  461. }
  462. else
  463. opts.OptLevel = 3;
  464. opts.OptDump = Args.hasFlag(OPT_Odump, OPT_INVALID, false);
  465. opts.DisableValidation = Args.hasFlag(OPT_VD, OPT_INVALID, false);
  466. opts.AllResourcesBound = Args.hasFlag(OPT_all_resources_bound, OPT_INVALID, false);
  467. opts.ColorCodeAssembly = Args.hasFlag(OPT_Cc, OPT_INVALID, false);
  468. opts.DefaultRowMajor = Args.hasFlag(OPT_Zpr, OPT_INVALID, false);
  469. opts.DefaultColMajor = Args.hasFlag(OPT_Zpc, OPT_INVALID, false);
  470. opts.DumpBin = Args.hasFlag(OPT_dumpbin, OPT_INVALID, false);
  471. opts.NotUseLegacyCBufLoad = Args.hasFlag(OPT_not_use_legacy_cbuf_load, OPT_INVALID, false);
  472. opts.PackPrefixStable = Args.hasFlag(OPT_pack_prefix_stable, OPT_INVALID, false);
  473. opts.PackOptimized = Args.hasFlag(OPT_pack_optimized, OPT_INVALID, false);
  474. opts.DisplayIncludeProcess = Args.hasFlag(OPT_H, OPT_INVALID, false);
  475. opts.WarningAsError = Args.hasFlag(OPT__SLASH_WX, OPT_INVALID, false);
  476. opts.AvoidFlowControl = Args.hasFlag(OPT_Gfa, OPT_INVALID, false);
  477. opts.PreferFlowControl = Args.hasFlag(OPT_Gfp, OPT_INVALID, false);
  478. opts.RecompileFromBinary = Args.hasFlag(OPT_recompile, OPT_INVALID, false);
  479. opts.StripDebug = Args.hasFlag(OPT_Qstrip_debug, OPT_INVALID, false);
  480. opts.EmbedDebug = Args.hasFlag(OPT_Qembed_debug, OPT_INVALID, false);
  481. opts.StripRootSignature = Args.hasFlag(OPT_Qstrip_rootsignature, OPT_INVALID, false);
  482. opts.StripPrivate = Args.hasFlag(OPT_Qstrip_priv, OPT_INVALID, false);
  483. opts.StripReflection = Args.hasFlag(OPT_Qstrip_reflect, OPT_INVALID, false);
  484. opts.KeepReflectionInDxil = Args.hasFlag(OPT_Qkeep_reflect_in_dxil, OPT_INVALID, false);
  485. opts.StripReflectionFromDxil = Args.hasFlag(OPT_Qstrip_reflect_from_dxil, OPT_INVALID, false);
  486. opts.ExtractRootSignature = Args.hasFlag(OPT_extractrootsignature, OPT_INVALID, false);
  487. opts.DisassembleColorCoded = Args.hasFlag(OPT_Cc, OPT_INVALID, false);
  488. opts.DisassembleInstNumbers = Args.hasFlag(OPT_Ni, OPT_INVALID, false);
  489. opts.DisassembleByteOffset = Args.hasFlag(OPT_No, OPT_INVALID, false);
  490. opts.DisaseembleHex = Args.hasFlag(OPT_Lx, OPT_INVALID, false);
  491. opts.LegacyMacroExpansion = Args.hasFlag(OPT_flegacy_macro_expansion, OPT_INVALID, false);
  492. opts.LegacyResourceReservation = Args.hasFlag(OPT_flegacy_resource_reservation, OPT_INVALID, false);
  493. opts.ExportShadersOnly = Args.hasFlag(OPT_export_shaders_only, OPT_INVALID, false);
  494. opts.ResMayAlias = Args.hasFlag(OPT_res_may_alias, OPT_INVALID, false);
  495. if (opts.DefaultColMajor && opts.DefaultRowMajor) {
  496. errors << "Cannot specify /Zpr and /Zpc together, use /? to get usage information";
  497. return 1;
  498. }
  499. if (opts.AvoidFlowControl && opts.PreferFlowControl) {
  500. errors << "Cannot specify /Gfa and /Gfp together, use /? to get usage information";
  501. return 1;
  502. }
  503. if (opts.PackPrefixStable && opts.PackOptimized) {
  504. errors << "Cannot specify /pack_prefix_stable and /pack_optimized together, use /? to get usage information";
  505. return 1;
  506. }
  507. // TODO: more fxc option check.
  508. // ERR_RES_MAY_ALIAS_ONLY_IN_CS_5
  509. // ERR_NOT_ABLE_TO_FLATTEN on if that contain side effects
  510. // TODO: other front-end error.
  511. // ERR_RESOURCE_NOT_IN_TEMPLATE
  512. // ERR_COMPLEX_TEMPLATE_RESOURCE
  513. // ERR_RESOURCE_BIND_CONFLICT
  514. // ERR_TEMPLATE_VAR_CONFLICT
  515. // ERR_ATTRIBUTE_PARAM_SIDE_EFFECT
  516. if ((flagsToInclude & hlsl::options::DriverOption) && opts.InputFile.empty()) {
  517. // Input file is required in arguments only for drivers; APIs take this through an argument.
  518. errors << "Required input file argument is missing. use -help to get more information.";
  519. return 1;
  520. }
  521. if (opts.OutputHeader.empty() && !opts.VariableName.empty()) {
  522. errors << "Cannot specify a header variable name when not writing a header.";
  523. return 1;
  524. }
  525. if (!opts.Preprocess.empty() &&
  526. (!opts.OutputHeader.empty() || !opts.OutputObject.empty() ||
  527. !opts.OutputWarnings || !opts.OutputWarningsFile.empty())) {
  528. errors << "Preprocess cannot be specified with other options.";
  529. return 1;
  530. }
  531. if (opts.DumpBin) {
  532. if (opts.DisplayIncludeProcess || opts.AstDump) {
  533. errors << "Cannot perform actions related to sources from a binary file.";
  534. return 1;
  535. }
  536. if (opts.AllResourcesBound || opts.AvoidFlowControl ||
  537. opts.CodeGenHighLevel || opts.DebugInfo || opts.DefaultColMajor ||
  538. opts.DefaultRowMajor || opts.Defines.size() != 0 ||
  539. opts.DisableOptimizations ||
  540. !opts.EntryPoint.empty() || !opts.ForceRootSigVer.empty() ||
  541. opts.PreferFlowControl || !opts.TargetProfile.empty()) {
  542. errors << "Cannot specify compilation options when reading a binary file.";
  543. return 1;
  544. }
  545. }
  546. if ((flagsToInclude & hlsl::options::DriverOption) &&
  547. opts.TargetProfile.empty() && !opts.DumpBin && opts.Preprocess.empty() && !opts.RecompileFromBinary) {
  548. // Target profile is required in arguments only for drivers when compiling;
  549. // APIs take this through an argument.
  550. errors << "Target profile argument is missing";
  551. return 1;
  552. }
  553. if (opts.EmbedDebug && !opts.DebugInfo) {
  554. errors << "Must enable debug info with /Zi for /Qembed_debug";
  555. return 1;
  556. }
  557. if (opts.DebugInfo && !opts.DebugNameForBinary && !opts.DebugNameForSource) {
  558. opts.DebugNameForBinary = true;
  559. } else if (opts.DebugNameForBinary && opts.DebugNameForSource) {
  560. errors << "Cannot specify both /Zss and /Zsb";
  561. return 1;
  562. }
  563. if (opts.DebugNameForSource && !opts.DebugInfo) {
  564. errors << "/Zss requires debug info (/Zi)";
  565. return 1;
  566. }
  567. llvm::StringRef valVersionStr = Args.getLastArgValue(OPT_validator_version);
  568. if (!valVersionStr.empty()) {
  569. // Parse "major.minor" version string
  570. auto verPair = valVersionStr.split(".");
  571. llvm::APInt major, minor;
  572. if (verPair.first.getAsInteger(0, major) || verPair.second.getAsInteger(0, minor)) {
  573. errors << "Format of validator version is \"<major>.<minor>\" (ex: \"1.4\").";
  574. return 1;
  575. }
  576. uint64_t major64 = major.getLimitedValue();
  577. uint64_t minor64 = minor.getLimitedValue();
  578. if (major64 > DXIL::kDxilMajor ||
  579. (major64 == DXIL::kDxilMajor && minor64 > DXIL::kDxilMinor)) {
  580. errors << "Validator version must be less than or equal to current internal version.";
  581. return 1;
  582. }
  583. if (major64 == 0 && minor64 != 0) {
  584. errors << "If validator major version is 0, minor version must also be 0.";
  585. return 1;
  586. }
  587. opts.ValVerMajor = (unsigned long)major64;
  588. opts.ValVerMinor = (unsigned long)minor64;
  589. }
  590. if (opts.IsLibraryProfile() && Minor == 0xF) {
  591. if (opts.ValVerMajor != UINT_MAX && opts.ValVerMajor != 0) {
  592. errors << "Offline library profile cannot be used with non-zero -validator-version.";
  593. return 1;
  594. }
  595. // Disable validation for offline link only target
  596. opts.DisableValidation = true;
  597. // ValVerMajor == 0 means that the module is not meant to ever be validated.
  598. opts.ValVerMajor = 0;
  599. opts.ValVerMinor = 0;
  600. }
  601. // These targets are only useful as an intermediate step towards linking to matching
  602. // shader targets without going through target downgrading at link time.
  603. // Disable lib_6_1 and lib_6_2 if /Vd is not present
  604. if (opts.IsLibraryProfile() && (Major < 6 || (Major == 6 && Minor < 3))) {
  605. if (!opts.DisableValidation) {
  606. errors << "Must disable validation for unsupported lib_6_1 or lib_6_2 "
  607. "targets.";
  608. return 1;
  609. }
  610. if (opts.ValVerMajor != UINT_MAX && opts.ValVerMajor != 0) {
  611. errors << "non-zero -validator-version cannot be used with library profiles lib_6_1 or lib_6_2.";
  612. return 1;
  613. }
  614. // ValVerMajor == 0 means that the module is not meant to ever be validated.
  615. opts.ValVerMajor = 0;
  616. opts.ValVerMinor = 0;
  617. }
  618. if (opts.KeepReflectionInDxil && opts.StripReflectionFromDxil) {
  619. errors << "-Qstrip_reflect_from_dxil mutually exclusive with -Qkeep_reflect_in_dxil.";
  620. return 1;
  621. }
  622. // SPIRV Change Starts
  623. #ifdef ENABLE_SPIRV_CODEGEN
  624. opts.GenSPIRV = Args.hasFlag(OPT_spirv, OPT_INVALID, false);
  625. opts.SpirvOptions.invertY = Args.hasFlag(OPT_fvk_invert_y, OPT_INVALID, false);
  626. opts.SpirvOptions.invertW = Args.hasFlag(OPT_fvk_use_dx_position_w, OPT_INVALID, false);
  627. opts.SpirvOptions.useGlLayout = Args.hasFlag(OPT_fvk_use_gl_layout, OPT_INVALID, false);
  628. opts.SpirvOptions.useDxLayout = Args.hasFlag(OPT_fvk_use_dx_layout, OPT_INVALID, false);
  629. opts.SpirvOptions.useScalarLayout = Args.hasFlag(OPT_fvk_use_scalar_layout, OPT_INVALID, false);
  630. opts.SpirvOptions.enableReflect = Args.hasFlag(OPT_fspv_reflect, OPT_INVALID, false);
  631. opts.SpirvOptions.noWarnIgnoredFeatures = Args.hasFlag(OPT_Wno_vk_ignored_features, OPT_INVALID, false);
  632. opts.SpirvOptions.noWarnEmulatedFeatures = Args.hasFlag(OPT_Wno_vk_emulated_features, OPT_INVALID, false);
  633. if (!handleVkShiftArgs(Args, OPT_fvk_b_shift, "b", &opts.SpirvOptions.bShift, errors) ||
  634. !handleVkShiftArgs(Args, OPT_fvk_t_shift, "t", &opts.SpirvOptions.tShift, errors) ||
  635. !handleVkShiftArgs(Args, OPT_fvk_s_shift, "s", &opts.SpirvOptions.sShift, errors) ||
  636. !handleVkShiftArgs(Args, OPT_fvk_u_shift, "u", &opts.SpirvOptions.uShift, errors))
  637. return 1;
  638. opts.SpirvOptions.bindRegister = Args.getAllArgValues(OPT_fvk_bind_register);
  639. opts.SpirvOptions.bindGlobals = Args.getAllArgValues(OPT_fvk_bind_globals);
  640. opts.SpirvOptions.stageIoOrder = Args.getLastArgValue(OPT_fvk_stage_io_order_EQ, "decl");
  641. if (opts.SpirvOptions.stageIoOrder != "alpha" && opts.SpirvOptions.stageIoOrder != "decl") {
  642. errors << "unknown Vulkan stage I/O location assignment order: "
  643. << opts.SpirvOptions.stageIoOrder;
  644. return 1;
  645. }
  646. for (const Arg *A : Args.filtered(OPT_fspv_extension_EQ)) {
  647. opts.SpirvOptions.allowedExtensions.push_back(A->getValue());
  648. }
  649. opts.SpirvOptions.debugInfoFile = opts.SpirvOptions.debugInfoSource = false;
  650. opts.SpirvOptions.debugInfoLine = opts.SpirvOptions.debugInfoTool = false;
  651. if (Args.hasArg(OPT_fspv_debug_EQ)) {
  652. opts.DebugInfo = true;
  653. for (const Arg *A : Args.filtered(OPT_fspv_debug_EQ)) {
  654. const llvm::StringRef v = A->getValue();
  655. if (v == "file") {
  656. opts.SpirvOptions.debugInfoFile = true;
  657. } else if (v == "source") {
  658. opts.SpirvOptions.debugInfoFile = true;
  659. opts.SpirvOptions.debugInfoSource = true;
  660. } else if (v == "line") {
  661. opts.SpirvOptions.debugInfoFile = true;
  662. opts.SpirvOptions.debugInfoSource = true;
  663. opts.SpirvOptions.debugInfoLine = true;
  664. } else if (v == "tool") {
  665. opts.SpirvOptions.debugInfoTool = true;
  666. } else {
  667. errors << "unknown SPIR-V debug info control parameter: " << v;
  668. return 1;
  669. }
  670. }
  671. } else if (opts.DebugInfo) {
  672. // By default turn on all categories
  673. opts.SpirvOptions.debugInfoFile = opts.SpirvOptions.debugInfoSource = true;
  674. opts.SpirvOptions.debugInfoLine = opts.SpirvOptions.debugInfoTool = true;
  675. }
  676. opts.SpirvOptions.targetEnv = Args.getLastArgValue(OPT_fspv_target_env_EQ, "vulkan1.0");
  677. // Handle -Oconfig=<comma-separated-list> option.
  678. uint32_t numOconfigs = 0;
  679. for (const Arg *A : Args.filtered(OPT_Oconfig)) {
  680. ++numOconfigs;
  681. if (numOconfigs > 1) {
  682. errors << "-Oconfig should not be specified more than once";
  683. return 1;
  684. }
  685. if (Args.getLastArg(OPT_O0, OPT_O1, OPT_O2, OPT_O3)) {
  686. errors << "-Oconfig should not be used together with -O";
  687. return 1;
  688. }
  689. for (const auto v : A->getValues()) {
  690. opts.SpirvOptions.optConfig.push_back(v);
  691. }
  692. }
  693. #else
  694. if (Args.hasFlag(OPT_spirv, OPT_INVALID, false) ||
  695. Args.hasFlag(OPT_fvk_invert_y, OPT_INVALID, false) ||
  696. Args.hasFlag(OPT_fvk_use_dx_position_w, OPT_INVALID, false) ||
  697. Args.hasFlag(OPT_fvk_use_gl_layout, OPT_INVALID, false) ||
  698. Args.hasFlag(OPT_fvk_use_dx_layout, OPT_INVALID, false) ||
  699. Args.hasFlag(OPT_fvk_use_scalar_layout, OPT_INVALID, false) ||
  700. Args.hasFlag(OPT_fspv_reflect, OPT_INVALID, false) ||
  701. Args.hasFlag(OPT_Wno_vk_ignored_features, OPT_INVALID, false) ||
  702. Args.hasFlag(OPT_Wno_vk_emulated_features, OPT_INVALID, false) ||
  703. !Args.getLastArgValue(OPT_fvk_stage_io_order_EQ).empty() ||
  704. !Args.getLastArgValue(OPT_fspv_debug_EQ).empty() ||
  705. !Args.getLastArgValue(OPT_fspv_extension_EQ).empty() ||
  706. !Args.getLastArgValue(OPT_fspv_target_env_EQ).empty() ||
  707. !Args.getLastArgValue(OPT_Oconfig).empty() ||
  708. !Args.getLastArgValue(OPT_fvk_bind_register).empty() ||
  709. !Args.getLastArgValue(OPT_fvk_bind_globals).empty() ||
  710. !Args.getLastArgValue(OPT_fvk_b_shift).empty() ||
  711. !Args.getLastArgValue(OPT_fvk_t_shift).empty() ||
  712. !Args.getLastArgValue(OPT_fvk_s_shift).empty() ||
  713. !Args.getLastArgValue(OPT_fvk_u_shift).empty()) {
  714. errors << "SPIR-V CodeGen not available. "
  715. "Please recompile with -DENABLE_SPIRV_CODEGEN=ON.";
  716. return 1;
  717. }
  718. #endif // ENABLE_SPIRV_CODEGEN
  719. // SPIRV Change Ends
  720. opts.Args = std::move(Args);
  721. return 0;
  722. }
  723. /// Sets up the specified DxcDllSupport instance as per the given options.
  724. int SetupDxcDllSupport(const DxcOpts &opts, dxc::DxcDllSupport &dxcSupport,
  725. llvm::raw_ostream &errors) {
  726. if (!opts.ExternalLib.empty()) {
  727. DXASSERT(!opts.ExternalFn.empty(), "else ReadDxcOpts should have failed");
  728. StringRefUtf16 externalLib(opts.ExternalLib);
  729. HRESULT hrLoad =
  730. dxcSupport.InitializeForDll(externalLib, opts.ExternalFn.data());
  731. if (DXC_FAILED(hrLoad)) {
  732. errors << "Unable to load support for external DLL " << opts.ExternalLib
  733. << " with function " << opts.ExternalFn << " - error 0x";
  734. errors.write_hex(hrLoad);
  735. return 1;
  736. }
  737. }
  738. return 0;
  739. }
  740. void CopyArgsToWStrings(const InputArgList &inArgs, unsigned flagsToInclude,
  741. std::vector<std::wstring> &outArgs) {
  742. ArgStringList stringList;
  743. for (const Arg *A : inArgs) {
  744. if (A->getOption().hasFlag(flagsToInclude)) {
  745. A->renderAsInput(inArgs, stringList);
  746. }
  747. }
  748. for (const char *argText : stringList) {
  749. outArgs.emplace_back(Unicode::UTF8ToUTF16StringOrThrow(argText));
  750. }
  751. }
  752. } } // hlsl::options