transformation.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. // Copyright (c) 2019 Google LLC
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. #include "source/fuzz/transformation.h"
  15. #include <cassert>
  16. #include "source/fuzz/fuzzer_util.h"
  17. #include "source/fuzz/transformation_access_chain.h"
  18. #include "source/fuzz/transformation_add_constant_boolean.h"
  19. #include "source/fuzz/transformation_add_constant_composite.h"
  20. #include "source/fuzz/transformation_add_constant_null.h"
  21. #include "source/fuzz/transformation_add_constant_scalar.h"
  22. #include "source/fuzz/transformation_add_copy_memory.h"
  23. #include "source/fuzz/transformation_add_dead_block.h"
  24. #include "source/fuzz/transformation_add_dead_break.h"
  25. #include "source/fuzz/transformation_add_dead_continue.h"
  26. #include "source/fuzz/transformation_add_function.h"
  27. #include "source/fuzz/transformation_add_global_undef.h"
  28. #include "source/fuzz/transformation_add_global_variable.h"
  29. #include "source/fuzz/transformation_add_image_sample_unused_components.h"
  30. #include "source/fuzz/transformation_add_local_variable.h"
  31. #include "source/fuzz/transformation_add_no_contraction_decoration.h"
  32. #include "source/fuzz/transformation_add_parameter.h"
  33. #include "source/fuzz/transformation_add_spec_constant_op.h"
  34. #include "source/fuzz/transformation_add_type_array.h"
  35. #include "source/fuzz/transformation_add_type_boolean.h"
  36. #include "source/fuzz/transformation_add_type_float.h"
  37. #include "source/fuzz/transformation_add_type_function.h"
  38. #include "source/fuzz/transformation_add_type_int.h"
  39. #include "source/fuzz/transformation_add_type_matrix.h"
  40. #include "source/fuzz/transformation_add_type_pointer.h"
  41. #include "source/fuzz/transformation_add_type_struct.h"
  42. #include "source/fuzz/transformation_add_type_vector.h"
  43. #include "source/fuzz/transformation_adjust_branch_weights.h"
  44. #include "source/fuzz/transformation_composite_construct.h"
  45. #include "source/fuzz/transformation_composite_extract.h"
  46. #include "source/fuzz/transformation_compute_data_synonym_fact_closure.h"
  47. #include "source/fuzz/transformation_copy_object.h"
  48. #include "source/fuzz/transformation_equation_instruction.h"
  49. #include "source/fuzz/transformation_function_call.h"
  50. #include "source/fuzz/transformation_invert_comparison_operator.h"
  51. #include "source/fuzz/transformation_load.h"
  52. #include "source/fuzz/transformation_merge_blocks.h"
  53. #include "source/fuzz/transformation_move_block_down.h"
  54. #include "source/fuzz/transformation_outline_function.h"
  55. #include "source/fuzz/transformation_permute_function_parameters.h"
  56. #include "source/fuzz/transformation_permute_phi_operands.h"
  57. #include "source/fuzz/transformation_push_id_through_variable.h"
  58. #include "source/fuzz/transformation_record_synonymous_constants.h"
  59. #include "source/fuzz/transformation_replace_boolean_constant_with_constant_binary.h"
  60. #include "source/fuzz/transformation_replace_constant_with_uniform.h"
  61. #include "source/fuzz/transformation_replace_id_with_synonym.h"
  62. #include "source/fuzz/transformation_replace_linear_algebra_instruction.h"
  63. #include "source/fuzz/transformation_replace_parameter_with_global.h"
  64. #include "source/fuzz/transformation_set_function_control.h"
  65. #include "source/fuzz/transformation_set_loop_control.h"
  66. #include "source/fuzz/transformation_set_memory_operands_mask.h"
  67. #include "source/fuzz/transformation_set_selection_control.h"
  68. #include "source/fuzz/transformation_split_block.h"
  69. #include "source/fuzz/transformation_store.h"
  70. #include "source/fuzz/transformation_swap_commutable_operands.h"
  71. #include "source/fuzz/transformation_swap_conditional_branch_operands.h"
  72. #include "source/fuzz/transformation_toggle_access_chain_instruction.h"
  73. #include "source/fuzz/transformation_vector_shuffle.h"
  74. #include "source/util/make_unique.h"
  75. namespace spvtools {
  76. namespace fuzz {
  77. Transformation::~Transformation() = default;
  78. std::unique_ptr<Transformation> Transformation::FromMessage(
  79. const protobufs::Transformation& message) {
  80. switch (message.transformation_case()) {
  81. case protobufs::Transformation::TransformationCase::kAccessChain:
  82. return MakeUnique<TransformationAccessChain>(message.access_chain());
  83. case protobufs::Transformation::TransformationCase::kAddConstantBoolean:
  84. return MakeUnique<TransformationAddConstantBoolean>(
  85. message.add_constant_boolean());
  86. case protobufs::Transformation::TransformationCase::kAddConstantComposite:
  87. return MakeUnique<TransformationAddConstantComposite>(
  88. message.add_constant_composite());
  89. case protobufs::Transformation::TransformationCase::kAddConstantNull:
  90. return MakeUnique<TransformationAddConstantNull>(
  91. message.add_constant_null());
  92. case protobufs::Transformation::TransformationCase::kAddConstantScalar:
  93. return MakeUnique<TransformationAddConstantScalar>(
  94. message.add_constant_scalar());
  95. case protobufs::Transformation::TransformationCase::kAddCopyMemory:
  96. return MakeUnique<TransformationAddCopyMemory>(message.add_copy_memory());
  97. case protobufs::Transformation::TransformationCase::kAddDeadBlock:
  98. return MakeUnique<TransformationAddDeadBlock>(message.add_dead_block());
  99. case protobufs::Transformation::TransformationCase::kAddDeadBreak:
  100. return MakeUnique<TransformationAddDeadBreak>(message.add_dead_break());
  101. case protobufs::Transformation::TransformationCase::kAddDeadContinue:
  102. return MakeUnique<TransformationAddDeadContinue>(
  103. message.add_dead_continue());
  104. case protobufs::Transformation::TransformationCase::kAddFunction:
  105. return MakeUnique<TransformationAddFunction>(message.add_function());
  106. case protobufs::Transformation::TransformationCase::kAddGlobalUndef:
  107. return MakeUnique<TransformationAddGlobalUndef>(
  108. message.add_global_undef());
  109. case protobufs::Transformation::TransformationCase::kAddGlobalVariable:
  110. return MakeUnique<TransformationAddGlobalVariable>(
  111. message.add_global_variable());
  112. case protobufs::Transformation::TransformationCase::
  113. kAddImageSampleUnusedComponents:
  114. return MakeUnique<TransformationAddImageSampleUnusedComponents>(
  115. message.add_image_sample_unused_components());
  116. case protobufs::Transformation::TransformationCase::kAddLocalVariable:
  117. return MakeUnique<TransformationAddLocalVariable>(
  118. message.add_local_variable());
  119. case protobufs::Transformation::TransformationCase::
  120. kAddNoContractionDecoration:
  121. return MakeUnique<TransformationAddNoContractionDecoration>(
  122. message.add_no_contraction_decoration());
  123. case protobufs::Transformation::TransformationCase::kAddParameter:
  124. return MakeUnique<TransformationAddParameter>(message.add_parameter());
  125. case protobufs::Transformation::TransformationCase::kAddSpecConstantOp:
  126. return MakeUnique<TransformationAddSpecConstantOp>(
  127. message.add_spec_constant_op());
  128. case protobufs::Transformation::TransformationCase::kAddTypeArray:
  129. return MakeUnique<TransformationAddTypeArray>(message.add_type_array());
  130. case protobufs::Transformation::TransformationCase::kAddTypeBoolean:
  131. return MakeUnique<TransformationAddTypeBoolean>(
  132. message.add_type_boolean());
  133. case protobufs::Transformation::TransformationCase::kAddTypeFloat:
  134. return MakeUnique<TransformationAddTypeFloat>(message.add_type_float());
  135. case protobufs::Transformation::TransformationCase::kAddTypeFunction:
  136. return MakeUnique<TransformationAddTypeFunction>(
  137. message.add_type_function());
  138. case protobufs::Transformation::TransformationCase::kAddTypeInt:
  139. return MakeUnique<TransformationAddTypeInt>(message.add_type_int());
  140. case protobufs::Transformation::TransformationCase::kAddTypeMatrix:
  141. return MakeUnique<TransformationAddTypeMatrix>(message.add_type_matrix());
  142. case protobufs::Transformation::TransformationCase::kAddTypePointer:
  143. return MakeUnique<TransformationAddTypePointer>(
  144. message.add_type_pointer());
  145. case protobufs::Transformation::TransformationCase::kAddTypeStruct:
  146. return MakeUnique<TransformationAddTypeStruct>(message.add_type_struct());
  147. case protobufs::Transformation::TransformationCase::kAddTypeVector:
  148. return MakeUnique<TransformationAddTypeVector>(message.add_type_vector());
  149. case protobufs::Transformation::TransformationCase::kAdjustBranchWeights:
  150. return MakeUnique<TransformationAdjustBranchWeights>(
  151. message.adjust_branch_weights());
  152. case protobufs::Transformation::TransformationCase::kCompositeConstruct:
  153. return MakeUnique<TransformationCompositeConstruct>(
  154. message.composite_construct());
  155. case protobufs::Transformation::TransformationCase::kCompositeExtract:
  156. return MakeUnique<TransformationCompositeExtract>(
  157. message.composite_extract());
  158. case protobufs::Transformation::TransformationCase::
  159. kComputeDataSynonymFactClosure:
  160. return MakeUnique<TransformationComputeDataSynonymFactClosure>(
  161. message.compute_data_synonym_fact_closure());
  162. case protobufs::Transformation::TransformationCase::kCopyObject:
  163. return MakeUnique<TransformationCopyObject>(message.copy_object());
  164. case protobufs::Transformation::TransformationCase::kEquationInstruction:
  165. return MakeUnique<TransformationEquationInstruction>(
  166. message.equation_instruction());
  167. case protobufs::Transformation::TransformationCase::kFunctionCall:
  168. return MakeUnique<TransformationFunctionCall>(message.function_call());
  169. case protobufs::Transformation::TransformationCase::
  170. kInvertComparisonOperator:
  171. return MakeUnique<TransformationInvertComparisonOperator>(
  172. message.invert_comparison_operator());
  173. case protobufs::Transformation::TransformationCase::kLoad:
  174. return MakeUnique<TransformationLoad>(message.load());
  175. case protobufs::Transformation::TransformationCase::kMergeBlocks:
  176. return MakeUnique<TransformationMergeBlocks>(message.merge_blocks());
  177. case protobufs::Transformation::TransformationCase::kMoveBlockDown:
  178. return MakeUnique<TransformationMoveBlockDown>(message.move_block_down());
  179. case protobufs::Transformation::TransformationCase::kOutlineFunction:
  180. return MakeUnique<TransformationOutlineFunction>(
  181. message.outline_function());
  182. case protobufs::Transformation::TransformationCase::
  183. kPermuteFunctionParameters:
  184. return MakeUnique<TransformationPermuteFunctionParameters>(
  185. message.permute_function_parameters());
  186. case protobufs::Transformation::TransformationCase::kPermutePhiOperands:
  187. return MakeUnique<TransformationPermutePhiOperands>(
  188. message.permute_phi_operands());
  189. case protobufs::Transformation::TransformationCase::kPushIdThroughVariable:
  190. return MakeUnique<TransformationPushIdThroughVariable>(
  191. message.push_id_through_variable());
  192. case protobufs::Transformation::TransformationCase::
  193. kRecordSynonymousConstants:
  194. return MakeUnique<TransformationRecordSynonymousConstants>(
  195. message.record_synonymous_constants());
  196. case protobufs::Transformation::TransformationCase::
  197. kReplaceParameterWithGlobal:
  198. return MakeUnique<TransformationReplaceParameterWithGlobal>(
  199. message.replace_parameter_with_global());
  200. case protobufs::Transformation::TransformationCase::
  201. kReplaceBooleanConstantWithConstantBinary:
  202. return MakeUnique<TransformationReplaceBooleanConstantWithConstantBinary>(
  203. message.replace_boolean_constant_with_constant_binary());
  204. case protobufs::Transformation::TransformationCase::
  205. kReplaceConstantWithUniform:
  206. return MakeUnique<TransformationReplaceConstantWithUniform>(
  207. message.replace_constant_with_uniform());
  208. case protobufs::Transformation::TransformationCase::kReplaceIdWithSynonym:
  209. return MakeUnique<TransformationReplaceIdWithSynonym>(
  210. message.replace_id_with_synonym());
  211. case protobufs::Transformation::TransformationCase::
  212. kReplaceLinearAlgebraInstruction:
  213. return MakeUnique<TransformationReplaceLinearAlgebraInstruction>(
  214. message.replace_linear_algebra_instruction());
  215. case protobufs::Transformation::TransformationCase::kSetFunctionControl:
  216. return MakeUnique<TransformationSetFunctionControl>(
  217. message.set_function_control());
  218. case protobufs::Transformation::TransformationCase::kSetLoopControl:
  219. return MakeUnique<TransformationSetLoopControl>(
  220. message.set_loop_control());
  221. case protobufs::Transformation::TransformationCase::kSetMemoryOperandsMask:
  222. return MakeUnique<TransformationSetMemoryOperandsMask>(
  223. message.set_memory_operands_mask());
  224. case protobufs::Transformation::TransformationCase::kSetSelectionControl:
  225. return MakeUnique<TransformationSetSelectionControl>(
  226. message.set_selection_control());
  227. case protobufs::Transformation::TransformationCase::kSplitBlock:
  228. return MakeUnique<TransformationSplitBlock>(message.split_block());
  229. case protobufs::Transformation::TransformationCase::kStore:
  230. return MakeUnique<TransformationStore>(message.store());
  231. case protobufs::Transformation::TransformationCase::kSwapCommutableOperands:
  232. return MakeUnique<TransformationSwapCommutableOperands>(
  233. message.swap_commutable_operands());
  234. case protobufs::Transformation::TransformationCase::
  235. kSwapConditionalBranchOperands:
  236. return MakeUnique<TransformationSwapConditionalBranchOperands>(
  237. message.swap_conditional_branch_operands());
  238. case protobufs::Transformation::TransformationCase::
  239. kToggleAccessChainInstruction:
  240. return MakeUnique<TransformationToggleAccessChainInstruction>(
  241. message.toggle_access_chain_instruction());
  242. case protobufs::Transformation::TransformationCase::kVectorShuffle:
  243. return MakeUnique<TransformationVectorShuffle>(message.vector_shuffle());
  244. case protobufs::Transformation::TRANSFORMATION_NOT_SET:
  245. assert(false && "An unset transformation was encountered.");
  246. return nullptr;
  247. }
  248. assert(false && "Should be unreachable as all cases must be handled above.");
  249. return nullptr;
  250. }
  251. bool Transformation::CheckIdIsFreshAndNotUsedByThisTransformation(
  252. uint32_t id, opt::IRContext* ir_context,
  253. std::set<uint32_t>* ids_used_by_this_transformation) {
  254. if (!fuzzerutil::IsFreshId(ir_context, id)) {
  255. return false;
  256. }
  257. if (ids_used_by_this_transformation->count(id) != 0) {
  258. return false;
  259. }
  260. ids_used_by_this_transformation->insert(id);
  261. return true;
  262. }
  263. } // namespace fuzz
  264. } // namespace spvtools