blob: 932fddb2a510f554b0186b3ce7d03b9952f24c4f [file] [log] [blame]
robertphillips@google.combeeb97c2012-05-09 21:15:28 +00001/*
2 * Copyright 2012 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
bsalomon@google.comcf8fb1f2012-08-02 14:03:32 +00008
robertphillips@google.combeeb97c2012-05-09 21:15:28 +00009#include "Test.h"
bsalomon@google.coma68937c2012-08-03 15:00:52 +000010// This is a GR test
11#if SK_SUPPORT_GPU
bsalomon@google.com67b915d2013-02-04 16:13:32 +000012#include "GrContextFactory.h"
robertphillips@google.combeeb97c2012-05-09 21:15:28 +000013#include "SkGpuDevice.h"
14#include "../../src/gpu/GrClipMaskManager.h"
15
16static const int X_SIZE = 12;
17static const int Y_SIZE = 12;
18
19////////////////////////////////////////////////////////////////////////////////
caryclark@google.com42639cd2012-06-06 12:03:39 +000020// note: this is unused
robertphillips@google.combeeb97c2012-05-09 21:15:28 +000021static GrTexture* createTexture(GrContext* context) {
robertphillips@google.comd82f3fa2012-05-10 21:26:48 +000022 unsigned char textureData[X_SIZE][Y_SIZE][4];
robertphillips@google.combeeb97c2012-05-09 21:15:28 +000023
robertphillips@google.comd82f3fa2012-05-10 21:26:48 +000024 memset(textureData, 0, 4* X_SIZE * Y_SIZE);
robertphillips@google.combeeb97c2012-05-09 21:15:28 +000025
26 GrTextureDesc desc;
27
28 // let Skia know we will be using this texture as a render target
29 desc.fFlags = kRenderTarget_GrTextureFlagBit;
bsalomon@google.comfec0bc32013-02-07 14:43:04 +000030 desc.fConfig = kSkia8888_GrPixelConfig;
robertphillips@google.combeeb97c2012-05-09 21:15:28 +000031 desc.fWidth = X_SIZE;
32 desc.fHeight = Y_SIZE;
robertphillips@google.combeeb97c2012-05-09 21:15:28 +000033
34 // We are initializing the texture with zeros here
35 GrTexture* texture = context->createUncachedTexture(desc, textureData, 0);
36 if (!texture) {
37 return NULL;
38 }
39
40 return texture;
41}
42
robertphillips@google.comc23a63b2012-07-31 11:47:29 +000043// Ensure that the 'getConservativeBounds' calls are returning bounds clamped
44// to the render target
45static void test_clip_bounds(skiatest::Reporter* reporter, GrContext* context) {
46
47 static const int kXSize = 100;
48 static const int kYSize = 100;
49
50 GrTextureDesc desc;
51 desc.fFlags = kRenderTarget_GrTextureFlagBit;
52 desc.fConfig = kAlpha_8_GrPixelConfig;
53 desc.fWidth = kXSize;
54 desc.fHeight = kYSize;
55
56 GrTexture* texture = context->createUncachedTexture(desc, NULL, 0);
57 if (!texture) {
58 return;
59 }
60
61 GrAutoUnref au(texture);
62
63 SkIRect intScreen = SkIRect::MakeWH(kXSize, kYSize);
robertphillips@google.comb755a2a2013-03-04 14:59:55 +000064 SkRect screen;
skia.committer@gmail.com075b0892013-03-05 07:09:08 +000065
robertphillips@google.comb755a2a2013-03-04 14:59:55 +000066 screen = SkRect::MakeWH(SkIntToScalar(kXSize),
67 SkIntToScalar(kYSize));
68
robertphillips@google.comc23a63b2012-07-31 11:47:29 +000069 SkRect clipRect(screen);
70 clipRect.outset(10, 10);
71
72 // create a clip stack that will (trivially) reduce to a single rect that
73 // is larger than the screen
74 SkClipStack stack;
75 stack.clipDevRect(clipRect, SkRegion::kReplace_Op, false);
76
77 bool isIntersectionOfRects = true;
robertphillips@google.com7b112892012-07-31 15:18:21 +000078 SkRect devStackBounds;
robertphillips@google.comc23a63b2012-07-31 11:47:29 +000079
rmistry@google.comd6176b02012-08-23 18:14:13 +000080 stack.getConservativeBounds(0, 0, kXSize, kYSize,
81 &devStackBounds,
robertphillips@google.comc23a63b2012-07-31 11:47:29 +000082 &isIntersectionOfRects);
83
84 // make sure that the SkClipStack is behaving itself
robertphillips@google.com7b112892012-07-31 15:18:21 +000085 REPORTER_ASSERT(reporter, screen == devStackBounds);
robertphillips@google.comc23a63b2012-07-31 11:47:29 +000086 REPORTER_ASSERT(reporter, isIntersectionOfRects);
87
robertphillips@google.com641f8b12012-07-31 19:15:58 +000088 // wrap the SkClipStack in a GrClipData
robertphillips@google.comc23a63b2012-07-31 11:47:29 +000089 GrClipData clipData;
robertphillips@google.com641f8b12012-07-31 19:15:58 +000090 clipData.fClipStack = &stack;
robertphillips@google.comc23a63b2012-07-31 11:47:29 +000091
robertphillips@google.com7b112892012-07-31 15:18:21 +000092 SkIRect devGrClipDataBound;
robertphillips@google.comc23a63b2012-07-31 11:47:29 +000093 clipData.getConservativeBounds(texture,
robertphillips@google.com7b112892012-07-31 15:18:21 +000094 &devGrClipDataBound,
robertphillips@google.comc23a63b2012-07-31 11:47:29 +000095 &isIntersectionOfRects);
96
97 // make sure that GrClipData is behaving itself
robertphillips@google.com7b112892012-07-31 15:18:21 +000098 REPORTER_ASSERT(reporter, intScreen == devGrClipDataBound);
robertphillips@google.comc23a63b2012-07-31 11:47:29 +000099 REPORTER_ASSERT(reporter, isIntersectionOfRects);
100}
101
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000102////////////////////////////////////////////////////////////////////////////////
103// verify that the top state of the stack matches the passed in state
104static void check_state(skiatest::Reporter* reporter,
105 const GrClipMaskCache& cache,
robertphillips@google.com641f8b12012-07-31 19:15:58 +0000106 const SkClipStack& clip,
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000107 GrTexture* mask,
robertphillips@google.com6623fcd2012-05-15 16:47:23 +0000108 const GrIRect& bound) {
skia.committer@gmail.comd21444a2012-12-07 02:01:25 +0000109 SkClipStack cacheClip;
bsalomon@google.com4c2443e2012-12-06 20:58:57 +0000110 REPORTER_ASSERT(reporter, clip.getTopmostGenID() == cache.getLastClipGenID());
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000111
112 REPORTER_ASSERT(reporter, mask == cache.getLastMask());
113
robertphillips@google.com6623fcd2012-05-15 16:47:23 +0000114 GrIRect cacheBound;
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000115 cache.getLastBound(&cacheBound);
116 REPORTER_ASSERT(reporter, bound == cacheBound);
117}
118
119////////////////////////////////////////////////////////////////////////////////
120// basic test of the cache's base functionality:
121// push, pop, set, canReuse & getters
122static void test_cache(skiatest::Reporter* reporter, GrContext* context) {
123
caryclark@google.com42639cd2012-06-06 12:03:39 +0000124 if (false) { // avoid bit rot, suppress warning
rmistry@google.comd6176b02012-08-23 18:14:13 +0000125 createTexture(context);
caryclark@google.com42639cd2012-06-06 12:03:39 +0000126 }
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000127 GrClipMaskCache cache;
128
robertphillips@google.comf105b102012-05-14 12:18:26 +0000129 cache.setContext(context);
130
robertphillips@google.com641f8b12012-07-31 19:15:58 +0000131 SkClipStack emptyClip;
132 emptyClip.reset();
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000133
robertphillips@google.com6623fcd2012-05-15 16:47:23 +0000134 GrIRect emptyBound;
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000135 emptyBound.setEmpty();
136
137 // check initial state
robertphillips@google.com8fff3562012-05-11 12:53:50 +0000138 check_state(reporter, cache, emptyClip, NULL, emptyBound);
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000139
140 // set the current state
robertphillips@google.com6623fcd2012-05-15 16:47:23 +0000141 GrIRect bound1;
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000142 bound1.set(0, 0, 100, 100);
143
robertphillips@google.com641f8b12012-07-31 19:15:58 +0000144 SkClipStack clip1(bound1);
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000145
robertphillips@google.com75b3c962012-06-07 12:08:45 +0000146 GrTextureDesc desc;
147 desc.fFlags = kRenderTarget_GrTextureFlagBit;
148 desc.fWidth = X_SIZE;
149 desc.fHeight = Y_SIZE;
bsalomon@google.comfec0bc32013-02-07 14:43:04 +0000150 desc.fConfig = kSkia8888_GrPixelConfig;
robertphillips@google.comd82f3fa2012-05-10 21:26:48 +0000151
bsalomon@google.com4c2443e2012-12-06 20:58:57 +0000152 cache.acquireMask(clip1.getTopmostGenID(), desc, bound1);
robertphillips@google.comf105b102012-05-14 12:18:26 +0000153
154 GrTexture* texture1 = cache.getLastMask();
155 REPORTER_ASSERT(reporter, texture1);
156 if (NULL == texture1) {
robertphillips@google.comd82f3fa2012-05-10 21:26:48 +0000157 return;
158 }
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000159
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000160 // check that the set took
robertphillips@google.comf105b102012-05-14 12:18:26 +0000161 check_state(reporter, cache, clip1, texture1, bound1);
bsalomon@google.com95ed55a2013-01-24 14:46:47 +0000162 REPORTER_ASSERT(reporter, texture1->getRefCnt());
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000163
164 // push the state
165 cache.push();
166
167 // verify that the pushed state is initially empty
robertphillips@google.com8fff3562012-05-11 12:53:50 +0000168 check_state(reporter, cache, emptyClip, NULL, emptyBound);
bsalomon@google.com95ed55a2013-01-24 14:46:47 +0000169 REPORTER_ASSERT(reporter, texture1->getRefCnt());
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000170
171 // modify the new state
robertphillips@google.com6623fcd2012-05-15 16:47:23 +0000172 GrIRect bound2;
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000173 bound2.set(-10, -10, 10, 10);
174
robertphillips@google.com641f8b12012-07-31 19:15:58 +0000175 SkClipStack clip2(bound2);
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000176
bsalomon@google.com4c2443e2012-12-06 20:58:57 +0000177 cache.acquireMask(clip2.getTopmostGenID(), desc, bound2);
robertphillips@google.comf105b102012-05-14 12:18:26 +0000178
179 GrTexture* texture2 = cache.getLastMask();
180 REPORTER_ASSERT(reporter, texture2);
181 if (NULL == texture2) {
182 return;
183 }
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000184
185 // check that the changes took
robertphillips@google.comf105b102012-05-14 12:18:26 +0000186 check_state(reporter, cache, clip2, texture2, bound2);
bsalomon@google.com95ed55a2013-01-24 14:46:47 +0000187 REPORTER_ASSERT(reporter, texture1->getRefCnt());
188 REPORTER_ASSERT(reporter, texture2->getRefCnt());
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000189
190 // check to make sure canReuse works
bsalomon@google.com4c2443e2012-12-06 20:58:57 +0000191 REPORTER_ASSERT(reporter, cache.canReuse(clip2.getTopmostGenID(), bound2));
192 REPORTER_ASSERT(reporter, !cache.canReuse(clip1.getTopmostGenID(), bound1));
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000193
194 // pop the state
195 cache.pop();
196
197 // verify that the old state is restored
robertphillips@google.comf105b102012-05-14 12:18:26 +0000198 check_state(reporter, cache, clip1, texture1, bound1);
bsalomon@google.com95ed55a2013-01-24 14:46:47 +0000199 REPORTER_ASSERT(reporter, texture1->getRefCnt());
200 REPORTER_ASSERT(reporter, texture2->getRefCnt());
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000201
202 // manually clear the state
203 cache.reset();
204
205 // verify it is now empty
robertphillips@google.com8fff3562012-05-11 12:53:50 +0000206 check_state(reporter, cache, emptyClip, NULL, emptyBound);
bsalomon@google.com95ed55a2013-01-24 14:46:47 +0000207 REPORTER_ASSERT(reporter, texture1->getRefCnt());
208 REPORTER_ASSERT(reporter, texture2->getRefCnt());
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000209
210 // pop again - so there is no state
211 cache.pop();
212
213#if !defined(SK_DEBUG)
214 // verify that the getters don't crash
215 // only do in release since it generates asserts in debug
robertphillips@google.comf21c7042012-05-11 13:01:16 +0000216 check_state(reporter, cache, emptyClip, NULL, emptyBound);
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000217#endif
bsalomon@google.com95ed55a2013-01-24 14:46:47 +0000218 REPORTER_ASSERT(reporter, texture1->getRefCnt());
219 REPORTER_ASSERT(reporter, texture2->getRefCnt());
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000220}
221
222////////////////////////////////////////////////////////////////////////////////
bsalomon@google.com67b915d2013-02-04 16:13:32 +0000223static void TestClipCache(skiatest::Reporter* reporter, GrContextFactory* factory) {
224 for (int type = 0; type < GrContextFactory::kLastGLContextType; ++type) {
225 GrContextFactory::GLContextType glType = static_cast<GrContextFactory::GLContextType>(type);
226 if (!GrContextFactory::IsRenderingGLContext(glType)) {
227 continue;
228 }
229 GrContext* context = factory->get(glType);
230 if (NULL == context) {
231 continue;
232 }
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000233
bsalomon@google.com67b915d2013-02-04 16:13:32 +0000234 test_cache(reporter, context);
235 test_clip_bounds(reporter, context);
236 }
robertphillips@google.combeeb97c2012-05-09 21:15:28 +0000237}
238
239////////////////////////////////////////////////////////////////////////////////
240#include "TestClassDef.h"
241DEFINE_GPUTESTCLASS("ClipCache", ClipCacheTestClass, TestClipCache)
bsalomon@google.comcf8fb1f2012-08-02 14:03:32 +0000242
243#endif