DeltaManager.cpp 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. //===- DeltaManager.cpp - Runs Delta Passes to reduce Input ---------------===//
  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 calls each specialized Delta pass in order to reduce the input IR
  10. // file.
  11. //
  12. //===----------------------------------------------------------------------===//
  13. #include "DeltaManager.h"
  14. #include "ReducerWorkItem.h"
  15. #include "TestRunner.h"
  16. #include "deltas/Delta.h"
  17. #include "deltas/ReduceAliases.h"
  18. #include "deltas/ReduceArguments.h"
  19. #include "deltas/ReduceAttributes.h"
  20. #include "deltas/ReduceBasicBlocks.h"
  21. #include "deltas/ReduceDIMetadata.h"
  22. #include "deltas/ReduceFunctionBodies.h"
  23. #include "deltas/ReduceFunctions.h"
  24. #include "deltas/ReduceGlobalObjects.h"
  25. #include "deltas/ReduceGlobalValues.h"
  26. #include "deltas/ReduceGlobalVarInitializers.h"
  27. #include "deltas/ReduceGlobalVars.h"
  28. #include "deltas/ReduceIRReferences.h"
  29. #include "deltas/ReduceInstructionFlags.h"
  30. #include "deltas/ReduceInstructionFlagsMIR.h"
  31. #include "deltas/ReduceInstructions.h"
  32. #include "deltas/ReduceInstructionsMIR.h"
  33. #include "deltas/ReduceInvokes.h"
  34. #include "deltas/ReduceMemoryOperations.h"
  35. #include "deltas/ReduceMetadata.h"
  36. #include "deltas/ReduceModuleData.h"
  37. #include "deltas/ReduceOpcodes.h"
  38. #include "deltas/ReduceOperandBundles.h"
  39. #include "deltas/ReduceOperands.h"
  40. #include "deltas/ReduceOperandsSkip.h"
  41. #include "deltas/ReduceOperandsToArgs.h"
  42. #include "deltas/ReduceRegisterDefs.h"
  43. #include "deltas/ReduceRegisterMasks.h"
  44. #include "deltas/ReduceRegisterUses.h"
  45. #include "deltas/ReduceSpecialGlobals.h"
  46. #include "deltas/ReduceUsingSimplifyCFG.h"
  47. #include "deltas/ReduceVirtualRegisters.h"
  48. #include "deltas/RunIRPasses.h"
  49. #include "deltas/SimplifyInstructions.h"
  50. #include "deltas/StripDebugInfo.h"
  51. #include "llvm/ADT/SmallSet.h"
  52. #include "llvm/Support/CommandLine.h"
  53. using namespace llvm;
  54. using SmallStringSet = SmallSet<StringRef, 8>;
  55. extern cl::OptionCategory LLVMReduceOptions;
  56. static cl::list<std::string>
  57. DeltaPasses("delta-passes",
  58. cl::desc("Delta passes to run, separated by commas. By "
  59. "default, run all delta passes."),
  60. cl::cat(LLVMReduceOptions), cl::CommaSeparated);
  61. static cl::list<std::string>
  62. SkipDeltaPasses("skip-delta-passes",
  63. cl::desc("Delta passes to not run, separated by commas. By "
  64. "default, run all delta passes."),
  65. cl::cat(LLVMReduceOptions), cl::CommaSeparated);
  66. #define DELTA_PASSES \
  67. do { \
  68. DELTA_PASS("strip-debug-info", stripDebugInfoDeltaPass) \
  69. DELTA_PASS("functions", reduceFunctionsDeltaPass) \
  70. DELTA_PASS("function-bodies", reduceFunctionBodiesDeltaPass) \
  71. DELTA_PASS("special-globals", reduceSpecialGlobalsDeltaPass) \
  72. DELTA_PASS("aliases", reduceAliasesDeltaPass) \
  73. DELTA_PASS("ifuncs", reduceIFuncsDeltaPass) \
  74. DELTA_PASS("simplify-conditionals-true", reduceConditionalsTrueDeltaPass) \
  75. DELTA_PASS("simplify-conditionals-false", reduceConditionalsFalseDeltaPass)\
  76. DELTA_PASS("invokes", reduceInvokesDeltaPass) \
  77. DELTA_PASS("unreachable-basic-blocks", reduceUnreachableBasicBlocksDeltaPass) \
  78. DELTA_PASS("basic-blocks", reduceBasicBlocksDeltaPass) \
  79. DELTA_PASS("simplify-cfg", reduceUsingSimplifyCFGDeltaPass) \
  80. DELTA_PASS("function-data", reduceFunctionDataDeltaPass) \
  81. DELTA_PASS("global-values", reduceGlobalValuesDeltaPass) \
  82. DELTA_PASS("global-objects", reduceGlobalObjectsDeltaPass) \
  83. DELTA_PASS("global-initializers", reduceGlobalsInitializersDeltaPass) \
  84. DELTA_PASS("global-variables", reduceGlobalsDeltaPass) \
  85. DELTA_PASS("di-metadata", reduceDIMetadataDeltaPass) \
  86. DELTA_PASS("metadata", reduceMetadataDeltaPass) \
  87. DELTA_PASS("named-metadata", reduceNamedMetadataDeltaPass) \
  88. DELTA_PASS("arguments", reduceArgumentsDeltaPass) \
  89. DELTA_PASS("instructions", reduceInstructionsDeltaPass) \
  90. DELTA_PASS("simplify-instructions", simplifyInstructionsDeltaPass) \
  91. DELTA_PASS("ir-passes", runIRPassesDeltaPass) \
  92. DELTA_PASS("operands-zero", reduceOperandsZeroDeltaPass) \
  93. DELTA_PASS("operands-one", reduceOperandsOneDeltaPass) \
  94. DELTA_PASS("operands-nan", reduceOperandsNaNDeltaPass) \
  95. DELTA_PASS("operands-to-args", reduceOperandsToArgsDeltaPass) \
  96. DELTA_PASS("operands-skip", reduceOperandsSkipDeltaPass) \
  97. DELTA_PASS("operand-bundles", reduceOperandBundesDeltaPass) \
  98. DELTA_PASS("attributes", reduceAttributesDeltaPass) \
  99. DELTA_PASS("module-data", reduceModuleDataDeltaPass) \
  100. DELTA_PASS("opcodes", reduceOpcodesDeltaPass) \
  101. DELTA_PASS("volatile", reduceVolatileInstructionsDeltaPass) \
  102. DELTA_PASS("atomic-ordering", reduceAtomicOrderingDeltaPass) \
  103. DELTA_PASS("syncscopes", reduceAtomicSyncScopesDeltaPass) \
  104. DELTA_PASS("instruction-flags", reduceInstructionFlagsDeltaPass) \
  105. } while (false)
  106. #define DELTA_PASSES_MIR \
  107. do { \
  108. DELTA_PASS("instructions", reduceInstructionsMIRDeltaPass) \
  109. DELTA_PASS("ir-instruction-references", \
  110. reduceIRInstructionReferencesDeltaPass) \
  111. DELTA_PASS("ir-block-references", reduceIRBlockReferencesDeltaPass) \
  112. DELTA_PASS("ir-function-references", reduceIRFunctionReferencesDeltaPass) \
  113. DELTA_PASS("instruction-flags", reduceInstructionFlagsMIRDeltaPass) \
  114. DELTA_PASS("register-uses", reduceRegisterUsesMIRDeltaPass) \
  115. DELTA_PASS("register-defs", reduceRegisterDefsMIRDeltaPass) \
  116. DELTA_PASS("register-hints", reduceVirtualRegisterHintsDeltaPass) \
  117. DELTA_PASS("register-masks", reduceRegisterMasksMIRDeltaPass) \
  118. } while (false)
  119. static void runAllDeltaPasses(TestRunner &Tester,
  120. const SmallStringSet &SkipPass) {
  121. #define DELTA_PASS(NAME, FUNC) \
  122. if (!SkipPass.count(NAME)) { \
  123. FUNC(Tester); \
  124. }
  125. if (Tester.getProgram().isMIR()) {
  126. DELTA_PASSES_MIR;
  127. } else {
  128. DELTA_PASSES;
  129. }
  130. #undef DELTA_PASS
  131. }
  132. static void runDeltaPassName(TestRunner &Tester, StringRef PassName) {
  133. #define DELTA_PASS(NAME, FUNC) \
  134. if (PassName == NAME) { \
  135. FUNC(Tester); \
  136. return; \
  137. }
  138. if (Tester.getProgram().isMIR()) {
  139. DELTA_PASSES_MIR;
  140. } else {
  141. DELTA_PASSES;
  142. }
  143. #undef DELTA_PASS
  144. // We should have errored on unrecognized passes before trying to run
  145. // anything.
  146. llvm_unreachable("unknown delta pass");
  147. }
  148. void llvm::printDeltaPasses(raw_ostream &OS) {
  149. OS << "Delta passes (pass to `--delta-passes=` as a comma separated list):\n";
  150. #define DELTA_PASS(NAME, FUNC) OS << " " << NAME << "\n";
  151. OS << " IR:\n";
  152. DELTA_PASSES;
  153. OS << " MIR:\n";
  154. DELTA_PASSES_MIR;
  155. #undef DELTA_PASS
  156. }
  157. // Built a set of available delta passes.
  158. static void collectPassNames(const TestRunner &Tester,
  159. SmallStringSet &NameSet) {
  160. #define DELTA_PASS(NAME, FUNC) NameSet.insert(NAME);
  161. if (Tester.getProgram().isMIR()) {
  162. DELTA_PASSES_MIR;
  163. } else {
  164. DELTA_PASSES;
  165. }
  166. #undef DELTA_PASS
  167. }
  168. /// Verify all requested or skipped passes are valid names, and return them in a
  169. /// set.
  170. static SmallStringSet handlePassList(const TestRunner &Tester,
  171. const cl::list<std::string> &PassList) {
  172. SmallStringSet AllPasses;
  173. collectPassNames(Tester, AllPasses);
  174. SmallStringSet PassSet;
  175. for (StringRef PassName : PassList) {
  176. if (!AllPasses.count(PassName)) {
  177. errs() << "unknown pass \"" << PassName << "\"\n";
  178. exit(1);
  179. }
  180. PassSet.insert(PassName);
  181. }
  182. return PassSet;
  183. }
  184. void llvm::runDeltaPasses(TestRunner &Tester, int MaxPassIterations) {
  185. uint64_t OldComplexity = Tester.getProgram().getComplexityScore();
  186. SmallStringSet RunPassSet, SkipPassSet;
  187. if (!DeltaPasses.empty())
  188. RunPassSet = handlePassList(Tester, DeltaPasses);
  189. if (!SkipDeltaPasses.empty())
  190. SkipPassSet = handlePassList(Tester, SkipDeltaPasses);
  191. for (int Iter = 0; Iter < MaxPassIterations; ++Iter) {
  192. if (DeltaPasses.empty()) {
  193. runAllDeltaPasses(Tester, SkipPassSet);
  194. } else {
  195. for (StringRef PassName : DeltaPasses) {
  196. if (!SkipPassSet.count(PassName))
  197. runDeltaPassName(Tester, PassName);
  198. }
  199. }
  200. uint64_t NewComplexity = Tester.getProgram().getComplexityScore();
  201. if (NewComplexity >= OldComplexity)
  202. break;
  203. OldComplexity = NewComplexity;
  204. }
  205. }