• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //===- ListReducer.h - Trim down list while retaining property --*- C++ -*-===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
10 // This class is to be used as a base class for operations that want to zero in
11 // on a subset of the input which still causes the bug we are tracking.
12 //
13 //===----------------------------------------------------------------------===//
14 
15 #ifndef LLVM_TOOLS_BUGPOINT_LISTREDUCER_H
16 #define LLVM_TOOLS_BUGPOINT_LISTREDUCER_H
17 
18 #include "llvm/Support/Error.h"
19 #include "llvm/Support/raw_ostream.h"
20 #include <algorithm>
21 #include <cstdlib>
22 #include <random>
23 #include <vector>
24 
25 namespace llvm {
26 
27 extern bool BugpointIsInterrupted;
28 
29 template <typename ElTy> struct ListReducer {
30   enum TestResult {
31     NoFailure,  // No failure of the predicate was detected
32     KeepSuffix, // The suffix alone satisfies the predicate
33     KeepPrefix  // The prefix alone satisfies the predicate
34   };
35 
~ListReducerListReducer36   virtual ~ListReducer() {}
37 
38   /// This virtual function should be overriden by subclasses to implement the
39   /// test desired.  The testcase is only required to test to see if the Kept
40   /// list still satisfies the property, but if it is going to check the prefix
41   /// anyway, it can.
42   virtual Expected<TestResult> doTest(std::vector<ElTy> &Prefix,
43                                       std::vector<ElTy> &Kept) = 0;
44 
45   /// This function attempts to reduce the length of the specified list while
46   /// still maintaining the "test" property.  This is the core of the "work"
47   /// that bugpoint does.
reduceListListReducer48   Expected<bool> reduceList(std::vector<ElTy> &TheList) {
49     std::vector<ElTy> empty;
50     std::mt19937 randomness(0x6e5ea738);  // Seed the random number generator
51     Expected<TestResult> Result = doTest(TheList, empty);
52     if (Error E = Result.takeError())
53       return std::move(E);
54     switch (*Result) {
55     case KeepPrefix:
56       if (TheList.size() == 1) // we are done, it's the base case and it fails
57         return true;
58       else
59         break; // there's definitely an error, but we need to narrow it down
60 
61     case KeepSuffix:
62       // cannot be reached!
63       llvm_unreachable("bugpoint ListReducer internal error: "
64                        "selected empty set.");
65 
66     case NoFailure:
67       return false; // there is no failure with the full set of passes/funcs!
68     }
69 
70     // Maximal number of allowed splitting iterations,
71     // before the elements are randomly shuffled.
72     const unsigned MaxIterationsWithoutProgress = 3;
73 
74     // Maximal number of allowed single-element trim iterations. We add a
75     // threshold here as single-element reductions may otherwise take a
76     // very long time to complete.
77     const unsigned MaxTrimIterationsWithoutBackJump = 3;
78     bool ShufflingEnabled = true;
79 
80   Backjump:
81     unsigned MidTop = TheList.size();
82     unsigned MaxIterations = MaxIterationsWithoutProgress;
83     unsigned NumOfIterationsWithoutProgress = 0;
84     while (MidTop > 1) { // Binary split reduction loop
85       // Halt if the user presses ctrl-c.
86       if (BugpointIsInterrupted) {
87         errs() << "\n\n*** Reduction Interrupted, cleaning up...\n\n";
88         return true;
89       }
90 
91       // If the loop doesn't make satisfying progress, try shuffling.
92       // The purpose of shuffling is to avoid the heavy tails of the
93       // distribution (improving the speed of convergence).
94       if (ShufflingEnabled && NumOfIterationsWithoutProgress > MaxIterations) {
95         std::vector<ElTy> ShuffledList(TheList);
96         std::shuffle(ShuffledList.begin(), ShuffledList.end(), randomness);
97         errs() << "\n\n*** Testing shuffled set...\n\n";
98         // Check that random shuffle doesn't lose the bug
99         Expected<TestResult> Result = doTest(ShuffledList, empty);
100         // TODO: Previously, this error was ignored and we treated it as if
101         // shuffling hid the bug. This should really either be consumeError if
102         // that behaviour was sensible, or we should propagate the error.
103         assert(!Result.takeError() && "Shuffling caused internal error?");
104 
105         if (*Result == KeepPrefix) {
106           // If the bug is still here, use the shuffled list.
107           TheList.swap(ShuffledList);
108           MidTop = TheList.size();
109           // Must increase the shuffling treshold to avoid the small
110           // probability of infinite looping without making progress.
111           MaxIterations += 2;
112           errs() << "\n\n*** Shuffling does not hide the bug...\n\n";
113         } else {
114           ShufflingEnabled = false; // Disable shuffling further on
115           errs() << "\n\n*** Shuffling hides the bug...\n\n";
116         }
117         NumOfIterationsWithoutProgress = 0;
118       }
119 
120       unsigned Mid = MidTop / 2;
121       std::vector<ElTy> Prefix(TheList.begin(), TheList.begin() + Mid);
122       std::vector<ElTy> Suffix(TheList.begin() + Mid, TheList.end());
123 
124       Expected<TestResult> Result = doTest(Prefix, Suffix);
125       if (Error E = Result.takeError())
126         return std::move(E);
127       switch (*Result) {
128       case KeepSuffix:
129         // The property still holds.  We can just drop the prefix elements, and
130         // shorten the list to the "kept" elements.
131         TheList.swap(Suffix);
132         MidTop = TheList.size();
133         // Reset progress treshold and progress counter
134         MaxIterations = MaxIterationsWithoutProgress;
135         NumOfIterationsWithoutProgress = 0;
136         break;
137       case KeepPrefix:
138         // The predicate still holds, shorten the list to the prefix elements.
139         TheList.swap(Prefix);
140         MidTop = TheList.size();
141         // Reset progress treshold and progress counter
142         MaxIterations = MaxIterationsWithoutProgress;
143         NumOfIterationsWithoutProgress = 0;
144         break;
145       case NoFailure:
146         // Otherwise the property doesn't hold.  Some of the elements we removed
147         // must be necessary to maintain the property.
148         MidTop = Mid;
149         NumOfIterationsWithoutProgress++;
150         break;
151       }
152     }
153 
154     // Probability of backjumping from the trimming loop back to the binary
155     // split reduction loop.
156     const int BackjumpProbability = 10;
157 
158     // Okay, we trimmed as much off the top and the bottom of the list as we
159     // could.  If there is more than two elements in the list, try deleting
160     // interior elements and testing that.
161     //
162     if (TheList.size() > 2) {
163       bool Changed = true;
164       std::vector<ElTy> EmptyList;
165       unsigned TrimIterations = 0;
166       while (Changed) { // Trimming loop.
167         Changed = false;
168 
169         // If the binary split reduction loop made an unfortunate sequence of
170         // splits, the trimming loop might be left off with a huge number of
171         // remaining elements (large search space). Backjumping out of that
172         // search space and attempting a different split can significantly
173         // improve the convergence speed.
174         if (std::rand() % 100 < BackjumpProbability)
175           goto Backjump;
176 
177         for (unsigned i = 1; i < TheList.size() - 1; ++i) {
178           // Check interior elts
179           if (BugpointIsInterrupted) {
180             errs() << "\n\n*** Reduction Interrupted, cleaning up...\n\n";
181             return true;
182           }
183 
184           std::vector<ElTy> TestList(TheList);
185           TestList.erase(TestList.begin() + i);
186 
187           Expected<TestResult> Result = doTest(EmptyList, TestList);
188           if (Error E = Result.takeError())
189             return std::move(E);
190           if (*Result == KeepSuffix) {
191             // We can trim down the list!
192             TheList.swap(TestList);
193             --i; // Don't skip an element of the list
194             Changed = true;
195           }
196         }
197         if (TrimIterations >= MaxTrimIterationsWithoutBackJump)
198           break;
199         TrimIterations++;
200       }
201     }
202 
203     return true; // there are some failure and we've narrowed them down
204   }
205 };
206 
207 } // End llvm namespace
208 
209 #endif
210