| OLD | NEW |
| (Empty) |
| 1 /* | |
| 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 | |
| 8 #include "Test.h" | |
| 9 // This is a GR test | |
| 10 #if SK_SUPPORT_GPU | |
| 11 #include "GrClipMaskManager.h" | |
| 12 #include "GrContextFactory.h" | |
| 13 #include "SkGpuDevice.h" | |
| 14 | |
| 15 static const int X_SIZE = 12; | |
| 16 static const int Y_SIZE = 12; | |
| 17 | |
| 18 //////////////////////////////////////////////////////////////////////////////// | |
| 19 // note: this is unused | |
| 20 static GrTexture* create_texture(GrContext* context) { | |
| 21 unsigned char textureData[X_SIZE][Y_SIZE][4]; | |
| 22 | |
| 23 memset(textureData, 0, 4* X_SIZE * Y_SIZE); | |
| 24 | |
| 25 GrSurfaceDesc desc; | |
| 26 | |
| 27 // let Skia know we will be using this texture as a render target | |
| 28 desc.fFlags = kRenderTarget_GrSurfaceFlag; | |
| 29 desc.fConfig = kSkia8888_GrPixelConfig; | |
| 30 desc.fWidth = X_SIZE; | |
| 31 desc.fHeight = Y_SIZE; | |
| 32 | |
| 33 // We are initializing the texture with zeros here | |
| 34 GrTexture* texture = context->textureProvider()->createTexture(desc, false,
textureData, 0); | |
| 35 if (!texture) { | |
| 36 return nullptr; | |
| 37 } | |
| 38 | |
| 39 return texture; | |
| 40 } | |
| 41 | |
| 42 // Ensure that the 'getConservativeBounds' calls are returning bounds clamped | |
| 43 // to the render target | |
| 44 static void test_clip_bounds(skiatest::Reporter* reporter, GrContext* context) { | |
| 45 | |
| 46 static const int kXSize = 100; | |
| 47 static const int kYSize = 100; | |
| 48 | |
| 49 GrSurfaceDesc desc; | |
| 50 desc.fFlags = kRenderTarget_GrSurfaceFlag; | |
| 51 desc.fConfig = kAlpha_8_GrPixelConfig; | |
| 52 desc.fWidth = kXSize; | |
| 53 desc.fHeight = kYSize; | |
| 54 | |
| 55 GrTexture* texture = context->textureProvider()->createTexture(desc, false,
nullptr, 0); | |
| 56 if (!texture) { | |
| 57 return; | |
| 58 } | |
| 59 | |
| 60 SkAutoTUnref<GrTexture> au(texture); | |
| 61 | |
| 62 SkIRect intScreen = SkIRect::MakeWH(kXSize, kYSize); | |
| 63 SkRect screen; | |
| 64 | |
| 65 screen = SkRect::MakeWH(SkIntToScalar(kXSize), | |
| 66 SkIntToScalar(kYSize)); | |
| 67 | |
| 68 SkRect clipRect(screen); | |
| 69 clipRect.outset(10, 10); | |
| 70 | |
| 71 // create a clip stack that will (trivially) reduce to a single rect that | |
| 72 // is larger than the screen | |
| 73 SkClipStack stack; | |
| 74 stack.clipDevRect(clipRect, SkRegion::kReplace_Op, false); | |
| 75 | |
| 76 bool isIntersectionOfRects = true; | |
| 77 SkRect devStackBounds; | |
| 78 | |
| 79 stack.getConservativeBounds(0, 0, kXSize, kYSize, | |
| 80 &devStackBounds, | |
| 81 &isIntersectionOfRects); | |
| 82 | |
| 83 // make sure that the SkClipStack is behaving itself | |
| 84 REPORTER_ASSERT(reporter, screen == devStackBounds); | |
| 85 REPORTER_ASSERT(reporter, isIntersectionOfRects); | |
| 86 | |
| 87 // wrap the SkClipStack in a GrClip | |
| 88 GrClip clipData; | |
| 89 clipData.setClipStack(&stack); | |
| 90 | |
| 91 SkIRect devGrClipBound; | |
| 92 clipData.getConservativeBounds(texture, | |
| 93 &devGrClipBound, | |
| 94 &isIntersectionOfRects); | |
| 95 | |
| 96 // make sure that GrClip is behaving itself | |
| 97 REPORTER_ASSERT(reporter, intScreen == devGrClipBound); | |
| 98 REPORTER_ASSERT(reporter, isIntersectionOfRects); | |
| 99 } | |
| 100 | |
| 101 //////////////////////////////////////////////////////////////////////////////// | |
| 102 // verify that the top state of the stack matches the passed in state | |
| 103 static void check_state(skiatest::Reporter* reporter, | |
| 104 const GrClipMaskCache& cache, | |
| 105 const SkClipStack& clip, | |
| 106 GrTexture* mask, | |
| 107 const SkIRect& bound) { | |
| 108 REPORTER_ASSERT(reporter, clip.getTopmostGenID() == cache.getLastClipGenID()
); | |
| 109 | |
| 110 REPORTER_ASSERT(reporter, mask == cache.getLastMask()); | |
| 111 | |
| 112 SkIRect cacheBound; | |
| 113 cache.getLastBound(&cacheBound); | |
| 114 REPORTER_ASSERT(reporter, bound == cacheBound); | |
| 115 } | |
| 116 | |
| 117 static void check_empty_state(skiatest::Reporter* reporter, | |
| 118 const GrClipMaskCache& cache) { | |
| 119 REPORTER_ASSERT(reporter, SkClipStack::kInvalidGenID == cache.getLastClipGen
ID()); | |
| 120 REPORTER_ASSERT(reporter, nullptr == cache.getLastMask()); | |
| 121 | |
| 122 SkIRect emptyBound; | |
| 123 emptyBound.setEmpty(); | |
| 124 | |
| 125 SkIRect cacheBound; | |
| 126 cache.getLastBound(&cacheBound); | |
| 127 REPORTER_ASSERT(reporter, emptyBound == cacheBound); | |
| 128 } | |
| 129 | |
| 130 //////////////////////////////////////////////////////////////////////////////// | |
| 131 // basic test of the cache's base functionality: | |
| 132 // push, pop, set, canReuse & getters | |
| 133 static void test_cache(skiatest::Reporter* reporter, GrContext* context) { | |
| 134 | |
| 135 if (false) { // avoid bit rot, suppress warning | |
| 136 create_texture(context); | |
| 137 } | |
| 138 GrClipMaskCache cache(context->resourceProvider()); | |
| 139 | |
| 140 // check initial state | |
| 141 check_empty_state(reporter, cache); | |
| 142 | |
| 143 // set the current state | |
| 144 SkIRect bound1; | |
| 145 bound1.set(0, 0, 100, 100); | |
| 146 | |
| 147 SkClipStack clip1(bound1); | |
| 148 | |
| 149 GrSurfaceDesc desc; | |
| 150 desc.fFlags = kRenderTarget_GrSurfaceFlag; | |
| 151 desc.fWidth = X_SIZE; | |
| 152 desc.fHeight = Y_SIZE; | |
| 153 desc.fConfig = kSkia8888_GrPixelConfig; | |
| 154 | |
| 155 cache.acquireMask(clip1.getTopmostGenID(), desc, bound1); | |
| 156 | |
| 157 GrTexture* texture1 = cache.getLastMask(); | |
| 158 REPORTER_ASSERT(reporter, texture1); | |
| 159 if (nullptr == texture1) { | |
| 160 return; | |
| 161 } | |
| 162 | |
| 163 // check that the set took | |
| 164 check_state(reporter, cache, clip1, texture1, bound1); | |
| 165 | |
| 166 // push the state | |
| 167 cache.push(); | |
| 168 | |
| 169 // verify that the pushed state is initially empty | |
| 170 check_empty_state(reporter, cache); | |
| 171 | |
| 172 // modify the new state | |
| 173 SkIRect bound2; | |
| 174 bound2.set(-10, -10, 10, 10); | |
| 175 | |
| 176 SkClipStack clip2(bound2); | |
| 177 | |
| 178 cache.acquireMask(clip2.getTopmostGenID(), desc, bound2); | |
| 179 | |
| 180 GrTexture* texture2 = cache.getLastMask(); | |
| 181 REPORTER_ASSERT(reporter, texture2); | |
| 182 if (nullptr == texture2) { | |
| 183 return; | |
| 184 } | |
| 185 | |
| 186 // check that the changes took | |
| 187 check_state(reporter, cache, clip2, texture2, bound2); | |
| 188 | |
| 189 // check to make sure canReuse works | |
| 190 REPORTER_ASSERT(reporter, cache.canReuse(clip2.getTopmostGenID(), bound2)); | |
| 191 REPORTER_ASSERT(reporter, !cache.canReuse(clip1.getTopmostGenID(), bound1)); | |
| 192 | |
| 193 // pop the state | |
| 194 cache.pop(); | |
| 195 | |
| 196 // verify that the old state is restored | |
| 197 check_state(reporter, cache, clip1, texture1, bound1); | |
| 198 | |
| 199 // manually clear the state | |
| 200 cache.reset(); | |
| 201 | |
| 202 // verify it is now empty | |
| 203 check_empty_state(reporter, cache); | |
| 204 | |
| 205 // pop again - so there is no state | |
| 206 cache.pop(); | |
| 207 | |
| 208 #if !defined(SK_DEBUG) | |
| 209 // verify that the getters don't crash | |
| 210 // only do in release since it generates asserts in debug | |
| 211 check_empty_state(reporter, cache); | |
| 212 #endif | |
| 213 } | |
| 214 | |
| 215 DEF_GPUTEST(ClipCache, reporter, factory) { | |
| 216 for (int type = 0; type < GrContextFactory::kLastGLContextType; ++type) { | |
| 217 GrContextFactory::GLContextType glType = static_cast<GrContextFactory::G
LContextType>(type); | |
| 218 if (!GrContextFactory::IsRenderingGLContext(glType)) { | |
| 219 continue; | |
| 220 } | |
| 221 GrContext* context = factory->get(glType); | |
| 222 if (nullptr == context) { | |
| 223 continue; | |
| 224 } | |
| 225 | |
| 226 test_cache(reporter, context); | |
| 227 test_clip_bounds(reporter, context); | |
| 228 } | |
| 229 } | |
| 230 | |
| 231 #endif | |
| OLD | NEW |