SpvTools.cpp 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. //
  2. // Copyright (C) 2014-2016 LunarG, Inc.
  3. // Copyright (C) 2018 Google, Inc.
  4. //
  5. // All rights reserved.
  6. //
  7. // Redistribution and use in source and binary forms, with or without
  8. // modification, are permitted provided that the following conditions
  9. // are met:
  10. //
  11. // Redistributions of source code must retain the above copyright
  12. // notice, this list of conditions and the following disclaimer.
  13. //
  14. // Redistributions in binary form must reproduce the above
  15. // copyright notice, this list of conditions and the following
  16. // disclaimer in the documentation and/or other materials provided
  17. // with the distribution.
  18. //
  19. // Neither the name of 3Dlabs Inc. Ltd. nor the names of its
  20. // contributors may be used to endorse or promote products derived
  21. // from this software without specific prior written permission.
  22. //
  23. // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  24. // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  25. // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
  26. // FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
  27. // COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
  28. // INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
  29. // BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  30. // LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
  31. // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  32. // LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
  33. // ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
  34. // POSSIBILITY OF SUCH DAMAGE.
  35. //
  36. // Call into SPIRV-Tools to disassemble, validate, and optimize.
  37. //
  38. #if ENABLE_OPT
  39. #include <cstdio>
  40. #include <iostream>
  41. #include "SpvTools.h"
  42. #include "spirv-tools/optimizer.hpp"
  43. #include "spirv-tools/libspirv.h"
  44. namespace glslang {
  45. // Translate glslang's view of target versioning to what SPIRV-Tools uses.
  46. spv_target_env MapToSpirvToolsEnv(const SpvVersion& spvVersion, spv::SpvBuildLogger* logger)
  47. {
  48. switch (spvVersion.vulkan) {
  49. case glslang::EShTargetVulkan_1_0: return spv_target_env::SPV_ENV_VULKAN_1_0;
  50. case glslang::EShTargetVulkan_1_1: return spv_target_env::SPV_ENV_VULKAN_1_1;
  51. default:
  52. break;
  53. }
  54. if (spvVersion.openGl > 0)
  55. return spv_target_env::SPV_ENV_OPENGL_4_5;
  56. logger->missingFunctionality("Target version for SPIRV-Tools validator");
  57. return spv_target_env::SPV_ENV_UNIVERSAL_1_0;
  58. }
  59. // Use the SPIRV-Tools disassembler to print SPIR-V.
  60. void SpirvToolsDisassemble(std::ostream& out, const std::vector<unsigned int>& spirv)
  61. {
  62. // disassemble
  63. spv_context context = spvContextCreate(SPV_ENV_UNIVERSAL_1_3);
  64. spv_text text;
  65. spv_diagnostic diagnostic = nullptr;
  66. spvBinaryToText(context, spirv.data(), spirv.size(),
  67. SPV_BINARY_TO_TEXT_OPTION_FRIENDLY_NAMES | SPV_BINARY_TO_TEXT_OPTION_INDENT,
  68. &text, &diagnostic);
  69. // dump
  70. if (diagnostic == nullptr)
  71. out << text->str;
  72. else
  73. spvDiagnosticPrint(diagnostic);
  74. // teardown
  75. spvDiagnosticDestroy(diagnostic);
  76. spvContextDestroy(context);
  77. }
  78. // Apply the SPIRV-Tools validator to generated SPIR-V.
  79. void SpirvToolsValidate(const glslang::TIntermediate& intermediate, std::vector<unsigned int>& spirv,
  80. spv::SpvBuildLogger* logger)
  81. {
  82. // validate
  83. spv_context context = spvContextCreate(MapToSpirvToolsEnv(intermediate.getSpv(), logger));
  84. spv_const_binary_t binary = { spirv.data(), spirv.size() };
  85. spv_diagnostic diagnostic = nullptr;
  86. spv_validator_options options = spvValidatorOptionsCreate();
  87. spvValidatorOptionsSetRelaxBlockLayout(options, intermediate.usingHlslOffsets());
  88. spvValidateWithOptions(context, options, &binary, &diagnostic);
  89. // report
  90. if (diagnostic != nullptr) {
  91. logger->error("SPIRV-Tools Validation Errors");
  92. logger->error(diagnostic->error);
  93. }
  94. // tear down
  95. spvValidatorOptionsDestroy(options);
  96. spvDiagnosticDestroy(diagnostic);
  97. spvContextDestroy(context);
  98. }
  99. // Apply the SPIRV-Tools optimizer to generated SPIR-V, for the purpose of
  100. // legalizing HLSL SPIR-V.
  101. void SpirvToolsLegalize(const glslang::TIntermediate&, std::vector<unsigned int>& spirv,
  102. spv::SpvBuildLogger*, const SpvOptions* options)
  103. {
  104. spv_target_env target_env = SPV_ENV_UNIVERSAL_1_2;
  105. spvtools::Optimizer optimizer(target_env);
  106. optimizer.SetMessageConsumer(
  107. [](spv_message_level_t level, const char *source, const spv_position_t &position, const char *message) {
  108. auto &out = std::cerr;
  109. switch (level)
  110. {
  111. case SPV_MSG_FATAL:
  112. case SPV_MSG_INTERNAL_ERROR:
  113. case SPV_MSG_ERROR:
  114. out << "error: ";
  115. break;
  116. case SPV_MSG_WARNING:
  117. out << "warning: ";
  118. break;
  119. case SPV_MSG_INFO:
  120. case SPV_MSG_DEBUG:
  121. out << "info: ";
  122. break;
  123. default:
  124. break;
  125. }
  126. if (source)
  127. {
  128. out << source << ":";
  129. }
  130. out << position.line << ":" << position.column << ":" << position.index << ":";
  131. if (message)
  132. {
  133. out << " " << message;
  134. }
  135. out << std::endl;
  136. });
  137. // If debug (specifically source line info) is being generated, propagate
  138. // line information into all SPIR-V instructions. This avoids loss of
  139. // information when instructions are deleted or moved. Later, remove
  140. // redundant information to minimize final SPRIR-V size.
  141. if (options->generateDebugInfo) {
  142. optimizer.RegisterPass(spvtools::CreatePropagateLineInfoPass());
  143. }
  144. optimizer.RegisterPass(spvtools::CreateDeadBranchElimPass());
  145. optimizer.RegisterPass(spvtools::CreateMergeReturnPass());
  146. optimizer.RegisterPass(spvtools::CreateInlineExhaustivePass());
  147. optimizer.RegisterPass(spvtools::CreateEliminateDeadFunctionsPass());
  148. optimizer.RegisterPass(spvtools::CreateScalarReplacementPass());
  149. optimizer.RegisterPass(spvtools::CreateLocalAccessChainConvertPass());
  150. optimizer.RegisterPass(spvtools::CreateLocalSingleBlockLoadStoreElimPass());
  151. optimizer.RegisterPass(spvtools::CreateLocalSingleStoreElimPass());
  152. optimizer.RegisterPass(spvtools::CreateSimplificationPass());
  153. optimizer.RegisterPass(spvtools::CreateAggressiveDCEPass());
  154. optimizer.RegisterPass(spvtools::CreateVectorDCEPass());
  155. optimizer.RegisterPass(spvtools::CreateDeadInsertElimPass());
  156. optimizer.RegisterPass(spvtools::CreateAggressiveDCEPass());
  157. optimizer.RegisterPass(spvtools::CreateDeadBranchElimPass());
  158. optimizer.RegisterPass(spvtools::CreateBlockMergePass());
  159. optimizer.RegisterPass(spvtools::CreateLocalMultiStoreElimPass());
  160. optimizer.RegisterPass(spvtools::CreateIfConversionPass());
  161. optimizer.RegisterPass(spvtools::CreateSimplificationPass());
  162. optimizer.RegisterPass(spvtools::CreateAggressiveDCEPass());
  163. optimizer.RegisterPass(spvtools::CreateVectorDCEPass());
  164. optimizer.RegisterPass(spvtools::CreateDeadInsertElimPass());
  165. if (options->optimizeSize) {
  166. optimizer.RegisterPass(spvtools::CreateRedundancyEliminationPass());
  167. // TODO(greg-lunarg): Add this when AMD driver issues are resolved
  168. // optimizer.RegisterPass(CreateCommonUniformElimPass());
  169. }
  170. optimizer.RegisterPass(spvtools::CreateAggressiveDCEPass());
  171. optimizer.RegisterPass(spvtools::CreateCFGCleanupPass());
  172. if (options->generateDebugInfo) {
  173. optimizer.RegisterPass(spvtools::CreateRedundantLineInfoElimPass());
  174. }
  175. spvtools::OptimizerOptions spvOptOptions;
  176. spvOptOptions.set_run_validator(false); // The validator may run as a seperate step later on
  177. optimizer.Run(spirv.data(), spirv.size(), &spirv, spvOptOptions);
  178. }
  179. }; // end namespace glslang
  180. #endif