• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2018 Google Inc.
3  *
4  * Use of this source code is governed by a BSD-style license that can be
5  * found in the LICENSE file.
6  */
7 
8 #include "include/gpu/GrContext.h"
9 #include "src/gpu/GrContextPriv.h"
10 #include "src/gpu/GrMemoryPool.h"
11 #include "src/gpu/GrOpFlushState.h"
12 #include "src/gpu/GrOpsTask.h"
13 #include "src/gpu/ops/GrOp.h"
14 #include "tests/Test.h"
15 
16 // We create Ops that write a value into a range of a buffer. We create ranges from
17 // kNumOpPositions starting positions x kRanges canonical ranges. We repeat each range kNumRepeats
18 // times (with a different value written by each of the repeats).
19 namespace {
20 struct Range {
21     unsigned fOffset;
22     unsigned fLength;
23 };
24 
25 static constexpr int kNumOpPositions = 4;
26 static constexpr Range kRanges[] = {{0, 4,}, {1, 2}};
27 static constexpr int kNumRanges = (int)SK_ARRAY_COUNT(kRanges);
28 static constexpr int kNumRepeats = 2;
29 static constexpr int kNumOps = kNumRepeats * kNumOpPositions * kNumRanges;
30 
fact(int n)31 static constexpr uint64_t fact(int n) {
32     assert(n > 0);
33     return n > 1 ? n * fact(n - 1) : 1;
34 }
35 
36 // How wide should our result buffer be to hold values written by the ranges of the ops.
result_width()37 static constexpr unsigned result_width() {
38     unsigned maxLength = 0;
39     for (size_t i = 0; i < kNumRanges; ++i) {
40         maxLength = maxLength > kRanges[i].fLength ? maxLength : kRanges[i].fLength;
41     }
42     return kNumOpPositions + maxLength - 1;
43 }
44 
45 // Number of possible allowable binary chainings among the kNumOps ops.
46 static constexpr int kNumCombinableValues = fact(kNumOps) / fact(kNumOps - 2);
47 using Combinable = std::array<GrOp::CombineResult, kNumCombinableValues>;
48 
49 /**
50  * The index in Combinable for the result for combining op 'b' into op 'a', i.e. the result of
51  * op[a]->combineIfPossible(op[b]).
52  */
combinable_index(int a,int b)53 int64_t combinable_index(int a, int b) {
54     SkASSERT(b != a);
55     // Each index gets kNumOps - 1 contiguous bools
56     int64_t aOffset = a * (kNumOps - 1);
57     // Within a's range we have one value each other op, but not one for a itself.
58     int64_t bIdxInA = b < a ? b : b - 1;
59     return aOffset + bIdxInA;
60 }
61 
62 /**
63  * Creates a legal set of combinability results for the ops. The likelihood that any two ops
64  * in a group can merge is randomly chosen.
65  */
init_combinable(int numGroups,Combinable * combinable,SkRandom * random)66 static void init_combinable(int numGroups, Combinable* combinable, SkRandom* random) {
67     SkScalar mergeProbability = random->nextUScalar1();
68     std::fill_n(combinable->begin(), kNumCombinableValues, GrOp::CombineResult::kCannotCombine);
69     SkTDArray<int> groups[kNumOps];
70     for (int i = 0; i < kNumOps; ++i) {
71         auto& group = groups[random->nextULessThan(numGroups)];
72         for (int g = 0; g < group.count(); ++g) {
73             int j = group[g];
74             if (random->nextUScalar1() < mergeProbability) {
75                 (*combinable)[combinable_index(i, j)] = GrOp::CombineResult::kMerged;
76             } else {
77                 (*combinable)[combinable_index(i, j)] = GrOp::CombineResult::kMayChain;
78             }
79             if (random->nextUScalar1() < mergeProbability) {
80                 (*combinable)[combinable_index(j, i)] = GrOp::CombineResult::kMerged;
81             } else {
82                 (*combinable)[combinable_index(j, i)] = GrOp::CombineResult::kMayChain;
83             }
84         }
85         group.push_back(i);
86     }
87 }
88 
89 /**
90  * A simple test op. It has an integer position, p. When it executes it writes p into an array
91  * of ints at index p and p+1. It takes a bitfield that indicates allowed pair-wise chainings.
92  */
93 class TestOp : public GrOp {
94 public:
95     DEFINE_OP_CLASS_ID
96 
Make(GrContext * context,int value,const Range & range,int result[],const Combinable * combinable)97     static std::unique_ptr<TestOp> Make(GrContext* context, int value, const Range& range,
98                                         int result[], const Combinable* combinable) {
99         GrOpMemoryPool* pool = context->priv().opMemoryPool();
100         return pool->allocate<TestOp>(value, range, result, combinable);
101     }
102 
name() const103     const char* name() const override { return "TestOp"; }
104 
writeResult(int result[]) const105     void writeResult(int result[]) const {
106         for (const auto& op : ChainRange<TestOp>(this)) {
107             for (const auto& vr : op.fValueRanges) {
108                 for (unsigned i = 0; i < vr.fRange.fLength; ++i) {
109                     result[vr.fRange.fOffset + i] = vr.fValue;
110                 }
111             }
112         }
113     }
114 
115 private:
116     friend class ::GrOpMemoryPool;  // for ctor
117 
TestOp(int value,const Range & range,int result[],const Combinable * combinable)118     TestOp(int value, const Range& range, int result[], const Combinable* combinable)
119             : INHERITED(ClassID()), fResult(result), fCombinable(combinable) {
120         fValueRanges.push_back({value, range});
121         this->setBounds(SkRect::MakeXYWH(range.fOffset, 0, range.fOffset + range.fLength, 1),
122                         HasAABloat::kNo, IsHairline::kNo);
123     }
124 
onPrepare(GrOpFlushState *)125     void onPrepare(GrOpFlushState*) override {}
126 
onExecute(GrOpFlushState *,const SkRect & chainBounds)127     void onExecute(GrOpFlushState*, const SkRect& chainBounds) override {
128         for (auto& op : ChainRange<TestOp>(this)) {
129             op.writeResult(fResult);
130         }
131     }
132 
onCombineIfPossible(GrOp * t,GrRecordingContext::Arenas * arenas,const GrCaps &)133     CombineResult onCombineIfPossible(GrOp* t, GrRecordingContext::Arenas* arenas,
134                                       const GrCaps&) override {
135         // This op doesn't use the arenas, but make sure the GrOpsTask is sending it
136         SkASSERT(arenas);
137         (void) arenas;
138         auto that = t->cast<TestOp>();
139         int v0 = fValueRanges[0].fValue;
140         int v1 = that->fValueRanges[0].fValue;
141         auto result = (*fCombinable)[combinable_index(v0, v1)];
142         if (result == GrOp::CombineResult::kMerged) {
143             std::move(that->fValueRanges.begin(), that->fValueRanges.end(),
144                       std::back_inserter(fValueRanges));
145         }
146         return result;
147     }
148 
149     struct ValueRange {
150         int fValue;
151         Range fRange;
152     };
153     std::vector<ValueRange> fValueRanges;
154     int* fResult;
155     const Combinable* fCombinable;
156 
157     typedef GrOp INHERITED;
158 };
159 }  // namespace
160 
161 /**
162  * Tests adding kNumOps to an op list with all possible allowed chaining configurations. Tests
163  * adding the ops in all possible orders and verifies that the chained executions don't violate
164  * painter's order.
165  */
166 DEF_GPUTEST(OpChainTest, reporter, /*ctxInfo*/) {
167     auto context = GrContext::MakeMock(nullptr);
168     SkASSERT(context);
169     static constexpr SkISize kDims = {kNumOps + 1, 1};
170 
171     const GrBackendFormat format =
172         context->priv().caps()->getDefaultBackendFormat(GrColorType::kRGBA_8888,
173                                                         GrRenderable::kYes);
174     GrSwizzle swizzle = context->priv().caps()->getReadSwizzle(format, GrColorType::kRGBA_8888);
175 
176     static const GrSurfaceOrigin kOrigin = kTopLeft_GrSurfaceOrigin;
177     auto proxy = context->priv().proxyProvider()->createProxy(
178             format, kDims, swizzle, GrRenderable::kYes, 1, GrMipMapped::kNo, SkBackingFit::kExact,
179             SkBudgeted::kNo, GrProtected::kNo, GrInternalSurfaceFlags::kNone);
180     SkASSERT(proxy);
181     proxy->instantiate(context->priv().resourceProvider());
182 
183     GrSwizzle outSwizzle = context->priv().caps()->getOutputSwizzle(format,
184                                                                     GrColorType::kRGBA_8888);
185 
186     int result[result_width()];
187     int validResult[result_width()];
188 
189     int permutation[kNumOps];
190     for (int i = 0; i < kNumOps; ++i) {
191         permutation[i] = i;
192     }
193     // Op order permutations.
194     static constexpr int kNumPermutations = 100;
195     // For a given number of chainability groups, this is the number of random combinability reuslts
196     // we will test.
197     static constexpr int kNumCombinabilitiesPerGrouping = 20;
198     SkRandom random;
199     bool repeat = false;
200     Combinable combinable;
201     for (int p = 0; p < kNumPermutations; ++p) {
202         for (int i = 0; i < kNumOps - 2 && !repeat; ++i) {
203             // The current implementation of nextULessThan() is biased. :(
204             unsigned j = i + random.nextULessThan(kNumOps - i);
205             std::swap(permutation[i], permutation[j]);
206         }
207         // g is the number of chainable groups that we partition the ops into.
208         for (int g = 1; g < kNumOps; ++g) {
209             for (int c = 0; c < kNumCombinabilitiesPerGrouping; ++c) {
210                 init_combinable(g, &combinable, &random);
211                 GrTokenTracker tracker;
212                 GrOpFlushState flushState(context->priv().getGpu(),
213                                           context->priv().resourceProvider(),
214                                           &tracker);
215                 GrOpsTask opsTask(context->priv().arenas(),
216                                   GrSurfaceProxyView(proxy, kOrigin, outSwizzle),
217                                   context->priv().auditTrail());
218                 // This assumes the particular values of kRanges.
219                 std::fill_n(result, result_width(), -1);
220                 std::fill_n(validResult, result_width(), -1);
221                 for (int i = 0; i < kNumOps; ++i) {
222                     int value = permutation[i];
223                     // factor out the repeats and then use the canonical starting position and range
224                     // to determine an actual range.
225                     int j = value % (kNumRanges * kNumOpPositions);
226                     int pos = j % kNumOpPositions;
227                     Range range = kRanges[j / kNumOpPositions];
228                     range.fOffset += pos;
229                     auto op = TestOp::Make(context.get(), value, range, result, &combinable);
230                     op->writeResult(validResult);
231                     opsTask.addOp(std::move(op),
232                                   GrTextureResolveManager(context->priv().drawingManager()),
233                                   *context->priv().caps());
234                 }
235                 opsTask.makeClosed(*context->priv().caps());
236                 opsTask.prepare(&flushState);
237                 opsTask.execute(&flushState);
238                 opsTask.endFlush();
239 #if 0  // Useful to repeat a random configuration that fails the test while debugger attached.
240                 if (!std::equal(result, result + result_width(), validResult)) {
241                     repeat = true;
242                 }
243 #endif
244                 (void)repeat;
245                 REPORTER_ASSERT(reporter, std::equal(result, result + result_width(), validResult));
246             }
247         }
248     }
249 }
250