blob: 36f37abf40ac61086532a7f268ce281df5dc4353 [file] [log] [blame]
robertphillips@google.comf6747b02012-06-12 00:32: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
8#include "GrAARectRenderer.h"
9#include "GrRefCnt.h"
10#include "GrGpu.h"
11
robertphillips@google.com4d73ac22012-06-13 18:54:08 +000012SK_DEFINE_INST_COUNT(GrAARectRenderer)
robertphillips@google.comf6747b02012-06-12 00:32:28 +000013
14namespace {
15
16static GrVertexLayout aa_rect_layout(const GrDrawTarget* target,
17 bool useCoverage) {
18 GrVertexLayout layout = 0;
robertphillips@google.comf6747b02012-06-12 00:32:28 +000019 if (useCoverage) {
20 layout |= GrDrawTarget::kCoverage_VertexLayoutBit;
21 } else {
22 layout |= GrDrawTarget::kColor_VertexLayoutBit;
23 }
24 return layout;
25}
26
27static void setInsetFan(GrPoint* pts, size_t stride,
28 const GrRect& r, GrScalar dx, GrScalar dy) {
29 pts->setRectFan(r.fLeft + dx, r.fTop + dy,
30 r.fRight - dx, r.fBottom - dy, stride);
31}
32
33};
34
35void GrAARectRenderer::reset() {
36 GrSafeSetNull(fAAFillRectIndexBuffer);
37 GrSafeSetNull(fAAStrokeRectIndexBuffer);
38}
39
40const uint16_t GrAARectRenderer::gFillAARectIdx[] = {
41 0, 1, 5, 5, 4, 0,
42 1, 2, 6, 6, 5, 1,
43 2, 3, 7, 7, 6, 2,
44 3, 0, 4, 4, 7, 3,
45 4, 5, 6, 6, 7, 4,
46};
47
48int GrAARectRenderer::aaFillRectIndexCount() {
49 return GR_ARRAY_COUNT(gFillAARectIdx);
50}
51
52GrIndexBuffer* GrAARectRenderer::aaFillRectIndexBuffer(GrGpu* gpu) {
53 if (NULL == fAAFillRectIndexBuffer) {
54 fAAFillRectIndexBuffer = gpu->createIndexBuffer(sizeof(gFillAARectIdx),
55 false);
56 if (NULL != fAAFillRectIndexBuffer) {
57#if GR_DEBUG
58 bool updated =
59#endif
60 fAAFillRectIndexBuffer->updateData(gFillAARectIdx,
61 sizeof(gFillAARectIdx));
62 GR_DEBUGASSERT(updated);
63 }
64 }
65 return fAAFillRectIndexBuffer;
66}
67
68const uint16_t GrAARectRenderer::gStrokeAARectIdx[] = {
69 0 + 0, 1 + 0, 5 + 0, 5 + 0, 4 + 0, 0 + 0,
70 1 + 0, 2 + 0, 6 + 0, 6 + 0, 5 + 0, 1 + 0,
71 2 + 0, 3 + 0, 7 + 0, 7 + 0, 6 + 0, 2 + 0,
72 3 + 0, 0 + 0, 4 + 0, 4 + 0, 7 + 0, 3 + 0,
73
74 0 + 4, 1 + 4, 5 + 4, 5 + 4, 4 + 4, 0 + 4,
75 1 + 4, 2 + 4, 6 + 4, 6 + 4, 5 + 4, 1 + 4,
76 2 + 4, 3 + 4, 7 + 4, 7 + 4, 6 + 4, 2 + 4,
77 3 + 4, 0 + 4, 4 + 4, 4 + 4, 7 + 4, 3 + 4,
78
79 0 + 8, 1 + 8, 5 + 8, 5 + 8, 4 + 8, 0 + 8,
80 1 + 8, 2 + 8, 6 + 8, 6 + 8, 5 + 8, 1 + 8,
81 2 + 8, 3 + 8, 7 + 8, 7 + 8, 6 + 8, 2 + 8,
82 3 + 8, 0 + 8, 4 + 8, 4 + 8, 7 + 8, 3 + 8,
83};
84
85int GrAARectRenderer::aaStrokeRectIndexCount() {
86 return GR_ARRAY_COUNT(gStrokeAARectIdx);
87}
88
89GrIndexBuffer* GrAARectRenderer::aaStrokeRectIndexBuffer(GrGpu* gpu) {
90 if (NULL == fAAStrokeRectIndexBuffer) {
rmistry@google.comd6176b02012-08-23 18:14:13 +000091 fAAStrokeRectIndexBuffer =
robertphillips@google.comf6747b02012-06-12 00:32:28 +000092 gpu->createIndexBuffer(sizeof(gStrokeAARectIdx), false);
93 if (NULL != fAAStrokeRectIndexBuffer) {
94#if GR_DEBUG
95 bool updated =
96#endif
97 fAAStrokeRectIndexBuffer->updateData(gStrokeAARectIdx,
98 sizeof(gStrokeAARectIdx));
99 GR_DEBUGASSERT(updated);
100 }
101 }
102 return fAAStrokeRectIndexBuffer;
103}
104
105void GrAARectRenderer::fillAARect(GrGpu* gpu,
robertphillips@google.comf69a11b2012-06-15 13:58:07 +0000106 GrDrawTarget* target,
107 const GrRect& devRect,
108 bool useVertexCoverage) {
robertphillips@google.comf6747b02012-06-12 00:32:28 +0000109 GrVertexLayout layout = aa_rect_layout(target, useVertexCoverage);
110
111 size_t vsize = GrDrawTarget::VertexSize(layout);
112
113 GrDrawTarget::AutoReleaseGeometry geo(target, layout, 8, 0);
114 if (!geo.succeeded()) {
115 GrPrintf("Failed to get space for vertices!\n");
116 return;
117 }
118 GrIndexBuffer* indexBuffer = this->aaFillRectIndexBuffer(gpu);
119 if (NULL == indexBuffer) {
120 GrPrintf("Failed to create index buffer!\n");
121 return;
122 }
123
124 intptr_t verts = reinterpret_cast<intptr_t>(geo.vertices());
125
126 GrPoint* fan0Pos = reinterpret_cast<GrPoint*>(verts);
127 GrPoint* fan1Pos = reinterpret_cast<GrPoint*>(verts + 4 * vsize);
128
129 setInsetFan(fan0Pos, vsize, devRect, -GR_ScalarHalf, -GR_ScalarHalf);
130 setInsetFan(fan1Pos, vsize, devRect, GR_ScalarHalf, GR_ScalarHalf);
131
132 verts += sizeof(GrPoint);
133 for (int i = 0; i < 4; ++i) {
134 *reinterpret_cast<GrColor*>(verts + i * vsize) = 0;
135 }
136
137 GrColor innerColor;
138 if (useVertexCoverage) {
139 innerColor = 0xffffffff;
140 } else {
141 innerColor = target->getDrawState().getColor();
142 }
143
144 verts += 4 * vsize;
145 for (int i = 0; i < 4; ++i) {
146 *reinterpret_cast<GrColor*>(verts + i * vsize) = innerColor;
147 }
148
149 target->setIndexSourceToBuffer(indexBuffer);
150
151 target->drawIndexed(kTriangles_GrPrimitiveType, 0,
152 0, 8, this->aaFillRectIndexCount());
153}
154
155void GrAARectRenderer::strokeAARect(GrGpu* gpu,
bsalomon@google.come7249bd2012-08-16 15:28:54 +0000156 GrDrawTarget* target,
157 const GrRect& devRect,
158 const GrVec& devStrokeSize,
159 bool useVertexCoverage) {
robertphillips@google.comf6747b02012-06-12 00:32:28 +0000160 const GrScalar& dx = devStrokeSize.fX;
161 const GrScalar& dy = devStrokeSize.fY;
162 const GrScalar rx = GrMul(dx, GR_ScalarHalf);
163 const GrScalar ry = GrMul(dy, GR_ScalarHalf);
164
165 GrScalar spare;
166 {
167 GrScalar w = devRect.width() - dx;
168 GrScalar h = devRect.height() - dy;
169 spare = GrMin(w, h);
170 }
171
172 if (spare <= 0) {
173 GrRect r(devRect);
174 r.inset(-rx, -ry);
175 this->fillAARect(gpu, target, r, useVertexCoverage);
176 return;
177 }
178 GrVertexLayout layout = aa_rect_layout(target, useVertexCoverage);
179 size_t vsize = GrDrawTarget::VertexSize(layout);
180
181 GrDrawTarget::AutoReleaseGeometry geo(target, layout, 16, 0);
182 if (!geo.succeeded()) {
183 GrPrintf("Failed to get space for vertices!\n");
184 return;
185 }
186 GrIndexBuffer* indexBuffer = this->aaStrokeRectIndexBuffer(gpu);
187 if (NULL == indexBuffer) {
188 GrPrintf("Failed to create index buffer!\n");
189 return;
190 }
191
192 intptr_t verts = reinterpret_cast<intptr_t>(geo.vertices());
193
bsalomon@google.come7249bd2012-08-16 15:28:54 +0000194 // We create vertices for four nested rectangles. There are two ramps from 0 to full
195 // coverage, one on the exterior of the stroke and the other on the interior.
196 // The following pointers refer to the four rects, from outermost to innermost.
robertphillips@google.comf6747b02012-06-12 00:32:28 +0000197 GrPoint* fan0Pos = reinterpret_cast<GrPoint*>(verts);
198 GrPoint* fan1Pos = reinterpret_cast<GrPoint*>(verts + 4 * vsize);
199 GrPoint* fan2Pos = reinterpret_cast<GrPoint*>(verts + 8 * vsize);
200 GrPoint* fan3Pos = reinterpret_cast<GrPoint*>(verts + 12 * vsize);
201
202 setInsetFan(fan0Pos, vsize, devRect,
203 -rx - GR_ScalarHalf, -ry - GR_ScalarHalf);
204 setInsetFan(fan1Pos, vsize, devRect,
205 -rx + GR_ScalarHalf, -ry + GR_ScalarHalf);
206 setInsetFan(fan2Pos, vsize, devRect,
207 rx - GR_ScalarHalf, ry - GR_ScalarHalf);
208 setInsetFan(fan3Pos, vsize, devRect,
209 rx + GR_ScalarHalf, ry + GR_ScalarHalf);
210
bsalomon@google.come7249bd2012-08-16 15:28:54 +0000211 // The outermost rect has 0 coverage
robertphillips@google.comf6747b02012-06-12 00:32:28 +0000212 verts += sizeof(GrPoint);
213 for (int i = 0; i < 4; ++i) {
214 *reinterpret_cast<GrColor*>(verts + i * vsize) = 0;
215 }
216
bsalomon@google.come7249bd2012-08-16 15:28:54 +0000217 // The inner two rects have full coverage
robertphillips@google.comf6747b02012-06-12 00:32:28 +0000218 GrColor innerColor;
219 if (useVertexCoverage) {
220 innerColor = 0xffffffff;
221 } else {
222 innerColor = target->getDrawState().getColor();
223 }
224 verts += 4 * vsize;
225 for (int i = 0; i < 8; ++i) {
226 *reinterpret_cast<GrColor*>(verts + i * vsize) = innerColor;
227 }
228
bsalomon@google.come7249bd2012-08-16 15:28:54 +0000229 // The innermost rect has full coverage
robertphillips@google.comf6747b02012-06-12 00:32:28 +0000230 verts += 8 * vsize;
bsalomon@google.come7249bd2012-08-16 15:28:54 +0000231 for (int i = 0; i < 4; ++i) {
robertphillips@google.comf6747b02012-06-12 00:32:28 +0000232 *reinterpret_cast<GrColor*>(verts + i * vsize) = 0;
233 }
234
235 target->setIndexSourceToBuffer(indexBuffer);
236 target->drawIndexed(kTriangles_GrPrimitiveType,
237 0, 0, 16, aaStrokeRectIndexCount());
238}