blob: 152090061416cff42e1238603f6ee79406f23c72 [file] [log] [blame]
reed@google.coma513efb2013-09-17 20:14:52 +00001/*
2 * Copyright 2013 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#include "SkDeviceLooper.h"
10#include "SkRasterClip.h"
11
12static void make_bm(SkBitmap* bm, int w, int h) {
13 bm->setConfig(SkBitmap::kA8_Config, w, h);
14 bm->allocPixels();
15}
16
17static bool equal(const SkRasterClip& a, const SkRasterClip& b) {
18 if (a.isBW()) {
19 return b.isBW() && a.bwRgn() == b.bwRgn();
20 } else {
21 return a.isAA() && a.aaRgn() == b.aaRgn();
22 }
23}
24
25static void test_simple(skiatest::Reporter* reporter) {
26 struct {
27 SkISize fDevSize;
28 SkIRect fRCBounds;
29 SkIRect fRect;
30 } const gRec[] = {
31 { { 4000, 10 }, { 0, 0, 4000, 10 }, { 0, 0, 4000, 4000 } },
32 { { 10, 4000 }, { 0, 0, 10, 4000 }, { 0, 0, 4000, 4000 } },
33 // very large devce, small rect
34 { { 32000, 10 }, { 0, 0, 32000, 10 }, { 0, 0, 4000, 4000 } },
35 { { 10, 32000 }, { 0, 0, 10, 32000 }, { 0, 0, 4000, 4000 } },
36 // very large device, small clip
37 { { 32000, 10 }, { 0, 0, 4000, 10 }, { 0, 0, 32000, 32000 } },
38 { { 10, 32000 }, { 0, 0, 10, 4000 }, { 0, 0, 32000, 32000 } },
39 };
40
41 for (size_t i = 0; i < SK_ARRAY_COUNT(gRec); ++i) {
42 SkBitmap bitmap;
43 make_bm(&bitmap, gRec[i].fDevSize.width(), gRec[i].fDevSize.height());
44
45 SkRasterClip rc(gRec[i].fRCBounds);
46
47 for (int aa = 0; aa <= 1; ++aa) {
48 SkDeviceLooper looper(bitmap, rc, gRec[i].fRect, SkToBool(aa));
49
50 bool valid = looper.next();
51 REPORTER_ASSERT(reporter, valid);
52 if (valid) {
53 REPORTER_ASSERT(reporter, looper.getBitmap().width() == bitmap.width());
54 REPORTER_ASSERT(reporter, looper.getBitmap().height() == bitmap.height());
55 REPORTER_ASSERT(reporter, equal(looper.getRC(), rc));
56
57 REPORTER_ASSERT(reporter, !looper.next());
58 }
59 }
60 // test that a rect that doesn't intersect returns no loops
61 {
62 SkIRect r = rc.getBounds();
63 r.offset(r.width(), 0);
64 SkDeviceLooper looper(bitmap, rc, r, false);
65 REPORTER_ASSERT(reporter, !looper.next());
66 }
67 }
68}
69
70// mask-bits are interpreted as the areas where the clip is visible
71// [ 0x01 0x02 ]
72// [ 0x04 0x08 ]
73//
74static void make_rgn(SkRegion* rgn, int w, int h, unsigned mask) {
75 SkASSERT(SkAlign2(w));
76 SkASSERT(SkAlign2(h));
77 w >>= 1;
78 h >>= 1;
79 const SkIRect baseR = SkIRect::MakeWH(w, h);
80
81 int bit = 1;
82 for (int y = 0; y <= 1; ++y) {
83 for (int x = 0; x <= 1; ++x) {
84 if (mask & bit) {
85 SkIRect r = baseR;
86 r.offset(x * w, y * h);
87 rgn->op(r, SkRegion::kUnion_Op);
88 }
89 bit <<= 1;
90 }
91 }
92}
93
94static void test_complex(skiatest::Reporter* reporter) {
95 // choose size values that will result in 4 quadrants, given fAA setting
96 const int BW_SIZE = 17 * 1000;
97 const int AA_SIZE = 7 * 1000;
98
99 struct {
100 SkISize fSize;
101 bool fAA;
102 } const gRec[] = {
103 { { BW_SIZE, BW_SIZE }, false },
104 { { AA_SIZE, AA_SIZE }, true },
105 };
106
107 for (size_t i = 0; i < SK_ARRAY_COUNT(gRec); ++i) {
108 const int w = gRec[i].fSize.width();
109 const int h = gRec[i].fSize.height();
110
111 SkBitmap bitmap;
112 make_bm(&bitmap, w, h);
113
114 const SkIRect rect = SkIRect::MakeWH(w, h);
115
116 // mask-bits are interpreted as the areas where the clip is visible
117 // [ 0x01 0x02 ]
118 // [ 0x04 0x08 ]
119 //
120 for (int mask = 0; mask <= 15; ++mask) {
121 SkRegion rgn;
122 make_rgn(&rgn, w, h, mask);
123
124 SkRasterClip rc;
125 rc.op(rgn, SkRegion::kReplace_Op);
126
127 SkDeviceLooper looper(bitmap, rc, rect, gRec[i].fAA);
128 while (looper.next()) {
129 REPORTER_ASSERT(reporter, !looper.getRC().isEmpty());
130 }
131 }
132 }
133}
134
135static void TestDeviceLooper(skiatest::Reporter* reporter) {
136 test_simple(reporter);
137 test_complex(reporter);
138}
139
140#include "TestClassDef.h"
141DEFINE_TESTCLASS("DeviceLooper", DeviceLooperClass, TestDeviceLooper)