blob: dd192e555942fee800c9819eaa82b58b30e4edf7 [file] [log] [blame]
Brian Salomon764e5462018-08-21 12:07:00 -04001/*
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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "include/gpu/GrContext.h"
9#include "src/gpu/GrContextPriv.h"
10#include "src/gpu/GrMemoryPool.h"
11#include "src/gpu/GrOpFlushState.h"
Greg Danielf41b2bd2019-08-22 16:19:24 -040012#include "src/gpu/GrOpsTask.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050013#include "src/gpu/ops/GrOp.h"
14#include "tests/Test.h"
Brian Salomon764e5462018-08-21 12:07:00 -040015
Brian Salomona7682c82018-10-24 10:04:37 -040016// 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).
19namespace {
20struct Range {
21 unsigned fOffset;
22 unsigned fLength;
23};
Brian Salomon764e5462018-08-21 12:07:00 -040024
Brian Salomona7682c82018-10-24 10:04:37 -040025static constexpr int kNumOpPositions = 4;
26static constexpr Range kRanges[] = {{0, 4,}, {1, 2}};
27static constexpr int kNumRanges = (int)SK_ARRAY_COUNT(kRanges);
28static constexpr int kNumRepeats = 2;
29static constexpr int kNumOps = kNumRepeats * kNumOpPositions * kNumRanges;
30
31static constexpr uint64_t fact(int n) {
Brian Salomon764e5462018-08-21 12:07:00 -040032 assert(n > 0);
33 return n > 1 ? n * fact(n - 1) : 1;
34}
35
Brian Salomona7682c82018-10-24 10:04:37 -040036// How wide should our result buffer be to hold values written by the ranges of the ops.
37static 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;
Brian Salomon764e5462018-08-21 12:07:00 -040043}
44
Brian Salomona7682c82018-10-24 10:04:37 -040045// Number of possible allowable binary chainings among the kNumOps ops.
46static constexpr int kNumCombinableValues = fact(kNumOps) / fact(kNumOps - 2);
47using Combinable = std::array<GrOp::CombineResult, kNumCombinableValues>;
48
Brian Salomon588cec72018-11-14 13:56:37 -050049/**
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 */
53int64_t combinable_index(int a, int b) {
Brian Salomona7682c82018-10-24 10:04:37 -040054 SkASSERT(b != a);
55 // Each index gets kNumOps - 1 contiguous bools
Brian Salomon588cec72018-11-14 13:56:37 -050056 int64_t aOffset = a * (kNumOps - 1);
Brian Salomona7682c82018-10-24 10:04:37 -040057 // 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;
Brian Salomon588cec72018-11-14 13:56:37 -050059 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 */
66static 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 }
Brian Salomona7682c82018-10-24 10:04:37 -040087}
88
Brian Salomon764e5462018-08-21 12:07:00 -040089/**
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 */
93class TestOp : public GrOp {
94public:
95 DEFINE_OP_CLASS_ID
96
Brian Salomona7682c82018-10-24 10:04:37 -040097 static std::unique_ptr<TestOp> Make(GrContext* context, int value, const Range& range,
98 int result[], const Combinable* combinable) {
Robert Phillips9da87e02019-02-04 13:26:26 -050099 GrOpMemoryPool* pool = context->priv().opMemoryPool();
Brian Salomona7682c82018-10-24 10:04:37 -0400100 return pool->allocate<TestOp>(value, range, result, combinable);
Brian Salomon764e5462018-08-21 12:07:00 -0400101 }
102
103 const char* name() const override { return "TestOp"; }
104
Brian Salomona7682c82018-10-24 10:04:37 -0400105 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 }
Brian Salomon764e5462018-08-21 12:07:00 -0400114
115private:
116 friend class ::GrOpMemoryPool; // for ctor
117
Brian Salomona7682c82018-10-24 10:04:37 -0400118 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),
Greg Daniel5faf4742019-10-01 15:14:44 -0400122 HasAABloat::kNo, IsHairline::kNo);
Brian Salomon764e5462018-08-21 12:07:00 -0400123 }
124
125 void onPrepare(GrOpFlushState*) override {}
126
Brian Salomon588cec72018-11-14 13:56:37 -0500127 void onExecute(GrOpFlushState*, const SkRect& chainBounds) override {
Brian Salomon764e5462018-08-21 12:07:00 -0400128 for (auto& op : ChainRange<TestOp>(this)) {
129 op.writeResult(fResult);
130 }
131 }
132
Michael Ludwig28b0c5d2019-12-19 14:51:00 -0500133 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;
Brian Salomona7682c82018-10-24 10:04:37 -0400138 auto that = t->cast<TestOp>();
Brian Salomon588cec72018-11-14 13:56:37 -0500139 int v0 = fValueRanges[0].fValue;
140 int v1 = that->fValueRanges[0].fValue;
141 auto result = (*fCombinable)[combinable_index(v0, v1)];
Brian Salomona7682c82018-10-24 10:04:37 -0400142 if (result == GrOp::CombineResult::kMerged) {
143 std::move(that->fValueRanges.begin(), that->fValueRanges.end(),
144 std::back_inserter(fValueRanges));
Brian Salomona7682c82018-10-24 10:04:37 -0400145 }
146 return result;
Brian Salomon764e5462018-08-21 12:07:00 -0400147 }
148
Brian Salomona7682c82018-10-24 10:04:37 -0400149 struct ValueRange {
150 int fValue;
151 Range fRange;
152 };
153 std::vector<ValueRange> fValueRanges;
Brian Salomon764e5462018-08-21 12:07:00 -0400154 int* fResult;
Brian Salomona7682c82018-10-24 10:04:37 -0400155 const Combinable* fCombinable;
Brian Salomon764e5462018-08-21 12:07:00 -0400156
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 */
166DEF_GPUTEST(OpChainTest, reporter, /*ctxInfo*/) {
167 auto context = GrContext::MakeMock(nullptr);
168 SkASSERT(context);
169 GrSurfaceDesc desc;
170 desc.fConfig = kRGBA_8888_GrPixelConfig;
171 desc.fWidth = kNumOps + 1;
172 desc.fHeight = 1;
Brian Salomon764e5462018-08-21 12:07:00 -0400173
Greg Daniel4065d452018-11-16 15:43:41 -0500174 const GrBackendFormat format =
Robert Phillips0a15cc62019-07-30 12:49:10 -0400175 context->priv().caps()->getDefaultBackendFormat(GrColorType::kRGBA_8888,
176 GrRenderable::kYes);
Greg Daniel47c20e82020-01-21 14:29:57 -0500177 GrSwizzle swizzle = context->priv().caps()->getReadSwizzle(format, GrColorType::kRGBA_8888);
Greg Daniel4065d452018-11-16 15:43:41 -0500178
Greg Daniel16f5c652019-10-29 11:26:01 -0400179 static const GrSurfaceOrigin kOrigin = kTopLeft_GrSurfaceOrigin;
Robert Phillips9da87e02019-02-04 13:26:26 -0500180 auto proxy = context->priv().proxyProvider()->createProxy(
Greg Daniel47c20e82020-01-21 14:29:57 -0500181 format, desc, swizzle, GrRenderable::kYes, 1, kOrigin, GrMipMapped::kNo,
Brian Salomone8a766b2019-07-19 14:24:36 -0400182 SkBackingFit::kExact, SkBudgeted::kNo, GrProtected::kNo, GrInternalSurfaceFlags::kNone);
Brian Salomon764e5462018-08-21 12:07:00 -0400183 SkASSERT(proxy);
Robert Phillips9da87e02019-02-04 13:26:26 -0500184 proxy->instantiate(context->priv().resourceProvider());
Greg Daniel16f5c652019-10-29 11:26:01 -0400185
186 GrSwizzle outSwizzle = context->priv().caps()->getOutputSwizzle(format,
187 GrColorType::kRGBA_8888);
188
Brian Salomona7682c82018-10-24 10:04:37 -0400189 int result[result_width()];
190 int validResult[result_width()];
Brian Salomon764e5462018-08-21 12:07:00 -0400191
192 int permutation[kNumOps];
193 for (int i = 0; i < kNumOps; ++i) {
194 permutation[i] = i;
195 }
Brian Salomon588cec72018-11-14 13:56:37 -0500196 // Op order permutations.
Brian Salomona7682c82018-10-24 10:04:37 -0400197 static constexpr int kNumPermutations = 100;
Brian Salomon588cec72018-11-14 13:56:37 -0500198 // For a given number of chainability groups, this is the number of random combinability reuslts
199 // we will test.
200 static constexpr int kNumCombinabilitiesPerGrouping = 20;
Brian Salomona7682c82018-10-24 10:04:37 -0400201 SkRandom random;
202 bool repeat = false;
203 Combinable combinable;
204 for (int p = 0; p < kNumPermutations; ++p) {
205 for (int i = 0; i < kNumOps - 2 && !repeat; ++i) {
206 // The current implementation of nextULessThan() is biased. :(
207 unsigned j = i + random.nextULessThan(kNumOps - i);
208 std::swap(permutation[i], permutation[j]);
209 }
Brian Salomon588cec72018-11-14 13:56:37 -0500210 // g is the number of chainable groups that we partition the ops into.
211 for (int g = 1; g < kNumOps; ++g) {
212 for (int c = 0; c < kNumCombinabilitiesPerGrouping; ++c) {
213 init_combinable(g, &combinable, &random);
214 GrTokenTracker tracker;
Robert Phillips9da87e02019-02-04 13:26:26 -0500215 GrOpFlushState flushState(context->priv().getGpu(),
Brian Salomon2c791fc2019-04-02 11:52:03 -0400216 context->priv().resourceProvider(),
Brian Salomon2c791fc2019-04-02 11:52:03 -0400217 &tracker);
Michael Ludwig28b0c5d2019-12-19 14:51:00 -0500218 GrOpsTask opsTask(context->priv().arenas(),
Greg Daniel16f5c652019-10-29 11:26:01 -0400219 GrSurfaceProxyView(proxy, kOrigin, outSwizzle),
Robert Phillips61fc7992019-10-22 11:58:17 -0400220 context->priv().auditTrail());
Brian Salomon588cec72018-11-14 13:56:37 -0500221 // This assumes the particular values of kRanges.
222 std::fill_n(result, result_width(), -1);
223 std::fill_n(validResult, result_width(), -1);
224 for (int i = 0; i < kNumOps; ++i) {
225 int value = permutation[i];
226 // factor out the repeats and then use the canonical starting position and range
227 // to determine an actual range.
228 int j = value % (kNumRanges * kNumOpPositions);
229 int pos = j % kNumOpPositions;
230 Range range = kRanges[j / kNumOpPositions];
231 range.fOffset += pos;
232 auto op = TestOp::Make(context.get(), value, range, result, &combinable);
233 op->writeResult(validResult);
Greg Danielf41b2bd2019-08-22 16:19:24 -0400234 opsTask.addOp(std::move(op),
235 GrTextureResolveManager(context->priv().drawingManager()),
236 *context->priv().caps());
Brian Salomon588cec72018-11-14 13:56:37 -0500237 }
Greg Danielf41b2bd2019-08-22 16:19:24 -0400238 opsTask.makeClosed(*context->priv().caps());
239 opsTask.prepare(&flushState);
240 opsTask.execute(&flushState);
241 opsTask.endFlush();
Brian Salomona7682c82018-10-24 10:04:37 -0400242#if 0 // Useful to repeat a random configuration that fails the test while debugger attached.
Brian Salomon588cec72018-11-14 13:56:37 -0500243 if (!std::equal(result, result + result_width(), validResult)) {
244 repeat = true;
245 }
Brian Salomona7682c82018-10-24 10:04:37 -0400246#endif
Brian Salomon588cec72018-11-14 13:56:37 -0500247 (void)repeat;
248 REPORTER_ASSERT(reporter, std::equal(result, result + result_width(), validResult));
249 }
Brian Salomon764e5462018-08-21 12:07:00 -0400250 }
Brian Salomona7682c82018-10-24 10:04:37 -0400251 }
Brian Salomon764e5462018-08-21 12:07:00 -0400252}