blob: 51d5a79003f2eb85f85076cd2f2559cabde17af8 [file] [log] [blame]
bsalomon@google.comf75b84e2011-09-29 14:58:28 +00001
2/*
3 * Copyright 2011 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
8
bsalomon@google.comaeb21602011-08-30 18:13:44 +00009#include "GrAAHairLinePathRenderer.h"
10
11#include "GrContext.h"
tomhudson@google.com93813632011-10-27 20:21:16 +000012#include "GrDrawState.h"
bsalomon@google.comc26d94f2013-03-25 18:19:00 +000013#include "GrDrawTargetCaps.h"
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +000014#include "GrEffect.h"
bsalomon@google.comaeb21602011-08-30 18:13:44 +000015#include "GrGpu.h"
16#include "GrIndexBuffer.h"
bsalomon@google.comdbeeac32011-09-12 14:59:34 +000017#include "GrPathUtils.h"
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +000018#include "GrTBackendEffectFactory.h"
bsalomon@google.comaeb21602011-08-30 18:13:44 +000019#include "SkGeometry.h"
sugoi@google.com12b4e272012-12-06 20:13:11 +000020#include "SkStroke.h"
bsalomon@google.comaeb21602011-08-30 18:13:44 +000021#include "SkTemplates.h"
22
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +000023#include "gl/GrGLEffect.h"
24#include "gl/GrGLSL.h"
bsalomon@google.com4647f902013-03-26 14:45:27 +000025
bsalomon@google.comaeb21602011-08-30 18:13:44 +000026namespace {
27// quadratics are rendered as 5-sided polys in order to bound the
28// AA stroke around the center-curve. See comments in push_quad_index_buffer and
29// bloat_quad.
30static const int kVertsPerQuad = 5;
31static const int kIdxsPerQuad = 9;
32
33static const int kVertsPerLineSeg = 4;
34static const int kIdxsPerLineSeg = 6;
35
36static const int kNumQuadsInIdxBuffer = 256;
37static const size_t kQuadIdxSBufize = kIdxsPerQuad *
38 sizeof(uint16_t) *
39 kNumQuadsInIdxBuffer;
40
41bool push_quad_index_data(GrIndexBuffer* qIdxBuffer) {
42 uint16_t* data = (uint16_t*) qIdxBuffer->lock();
43 bool tempData = NULL == data;
44 if (tempData) {
tomhudson@google.comc377baf2012-07-09 20:17:56 +000045 data = SkNEW_ARRAY(uint16_t, kNumQuadsInIdxBuffer * kIdxsPerQuad);
bsalomon@google.comaeb21602011-08-30 18:13:44 +000046 }
47 for (int i = 0; i < kNumQuadsInIdxBuffer; ++i) {
48
49 // Each quadratic is rendered as a five sided polygon. This poly bounds
50 // the quadratic's bounding triangle but has been expanded so that the
51 // 1-pixel wide area around the curve is inside the poly.
52 // If a,b,c are the original control points then the poly a0,b0,c0,c1,a1
53 // that is rendered would look like this:
54 // b0
55 // b
56 //
57 // a0 c0
58 // a c
59 // a1 c1
bsalomon@google.com0e5104c2012-04-10 16:20:41 +000060 // Each is drawn as three triangles specified by these 9 indices:
bsalomon@google.comaeb21602011-08-30 18:13:44 +000061 int baseIdx = i * kIdxsPerQuad;
62 uint16_t baseVert = (uint16_t)(i * kVertsPerQuad);
63 data[0 + baseIdx] = baseVert + 0; // a0
64 data[1 + baseIdx] = baseVert + 1; // a1
65 data[2 + baseIdx] = baseVert + 2; // b0
66 data[3 + baseIdx] = baseVert + 2; // b0
67 data[4 + baseIdx] = baseVert + 4; // c1
68 data[5 + baseIdx] = baseVert + 3; // c0
69 data[6 + baseIdx] = baseVert + 1; // a1
70 data[7 + baseIdx] = baseVert + 4; // c1
71 data[8 + baseIdx] = baseVert + 2; // b0
72 }
73 if (tempData) {
74 bool ret = qIdxBuffer->updateData(data, kQuadIdxSBufize);
75 delete[] data;
76 return ret;
77 } else {
78 qIdxBuffer->unlock();
79 return true;
80 }
81}
82}
83
84GrPathRenderer* GrAAHairLinePathRenderer::Create(GrContext* context) {
bsalomon@google.coma8a6a322011-09-23 14:19:58 +000085 const GrIndexBuffer* lIdxBuffer = context->getQuadIndexBuffer();
86 if (NULL == lIdxBuffer) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +000087 return NULL;
88 }
bsalomon@google.coma8a6a322011-09-23 14:19:58 +000089 GrGpu* gpu = context->getGpu();
90 GrIndexBuffer* qIdxBuf = gpu->createIndexBuffer(kQuadIdxSBufize, false);
91 SkAutoTUnref<GrIndexBuffer> qIdxBuffer(qIdxBuf);
92 if (NULL == qIdxBuf ||
93 !push_quad_index_data(qIdxBuf)) {
94 return NULL;
95 }
tomhudson@google.comc377baf2012-07-09 20:17:56 +000096 return SkNEW_ARGS(GrAAHairLinePathRenderer,
97 (context, lIdxBuffer, qIdxBuf));
bsalomon@google.comaeb21602011-08-30 18:13:44 +000098}
99
100GrAAHairLinePathRenderer::GrAAHairLinePathRenderer(
101 const GrContext* context,
102 const GrIndexBuffer* linesIndexBuffer,
103 const GrIndexBuffer* quadsIndexBuffer) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000104 fLinesIndexBuffer = linesIndexBuffer;
105 linesIndexBuffer->ref();
106 fQuadsIndexBuffer = quadsIndexBuffer;
107 quadsIndexBuffer->ref();
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000108}
109
110GrAAHairLinePathRenderer::~GrAAHairLinePathRenderer() {
111 fLinesIndexBuffer->unref();
112 fQuadsIndexBuffer->unref();
113}
114
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000115namespace {
116
bsalomon@google.com49313f62011-09-14 13:54:05 +0000117typedef SkTArray<SkPoint, true> PtArray;
bsalomon@google.com92669012011-09-27 19:10:05 +0000118#define PREALLOC_PTARRAY(N) SkSTArray<(N),SkPoint, true>
bsalomon@google.com49313f62011-09-14 13:54:05 +0000119typedef SkTArray<int, true> IntArray;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000120
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000121// Takes 178th time of logf on Z600 / VC2010
122int get_float_exp(float x) {
123 GR_STATIC_ASSERT(sizeof(int) == sizeof(float));
124#if GR_DEBUG
125 static bool tested;
126 if (!tested) {
127 tested = true;
128 GrAssert(get_float_exp(0.25f) == -2);
129 GrAssert(get_float_exp(0.3f) == -2);
130 GrAssert(get_float_exp(0.5f) == -1);
131 GrAssert(get_float_exp(1.f) == 0);
132 GrAssert(get_float_exp(2.f) == 1);
133 GrAssert(get_float_exp(2.5f) == 1);
134 GrAssert(get_float_exp(8.f) == 3);
135 GrAssert(get_float_exp(100.f) == 6);
136 GrAssert(get_float_exp(1000.f) == 9);
137 GrAssert(get_float_exp(1024.f) == 10);
138 GrAssert(get_float_exp(3000000.f) == 21);
139 }
140#endif
bsalomon@google.com2ec72802011-09-21 21:46:03 +0000141 const int* iptr = (const int*)&x;
142 return (((*iptr) & 0x7f800000) >> 23) - 127;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000143}
144
145// we subdivide the quads to avoid huge overfill
146// if it returns -1 then should be drawn as lines
147int num_quad_subdivs(const SkPoint p[3]) {
148 static const SkScalar gDegenerateToLineTol = SK_Scalar1;
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000149 static const SkScalar gDegenerateToLineTolSqd =
bsalomon@google.com46a2a1e2011-09-06 22:10:52 +0000150 SkScalarMul(gDegenerateToLineTol, gDegenerateToLineTol);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000151
bsalomon@google.com46a2a1e2011-09-06 22:10:52 +0000152 if (p[0].distanceToSqd(p[1]) < gDegenerateToLineTolSqd ||
153 p[1].distanceToSqd(p[2]) < gDegenerateToLineTolSqd) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000154 return -1;
155 }
bsalomon@google.com46a2a1e2011-09-06 22:10:52 +0000156
bsalomon@google.com81712882012-11-01 17:12:34 +0000157 SkScalar dsqd = p[1].distanceToLineBetweenSqd(p[0], p[2]);
bsalomon@google.com46a2a1e2011-09-06 22:10:52 +0000158 if (dsqd < gDegenerateToLineTolSqd) {
159 return -1;
160 }
161
162 if (p[2].distanceToLineBetweenSqd(p[1], p[0]) < gDegenerateToLineTolSqd) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000163 return -1;
164 }
165
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000166 // tolerance of triangle height in pixels
167 // tuned on windows Quadro FX 380 / Z600
168 // trade off of fill vs cpu time on verts
169 // maybe different when do this using gpu (geo or tess shaders)
170 static const SkScalar gSubdivTol = 175 * SK_Scalar1;
171
robertphillips@google.com7460b372012-04-25 16:54:51 +0000172 if (dsqd <= SkScalarMul(gSubdivTol, gSubdivTol)) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000173 return 0;
174 } else {
robertphillips@google.com87379e12013-03-29 12:11:10 +0000175 static const int kMaxSub = 4;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000176 // subdividing the quad reduces d by 4. so we want x = log4(d/tol)
177 // = log4(d*d/tol*tol)/2
178 // = log2(d*d/tol*tol)
179
180#ifdef SK_SCALAR_IS_FLOAT
181 // +1 since we're ignoring the mantissa contribution.
182 int log = get_float_exp(dsqd/(gSubdivTol*gSubdivTol)) + 1;
183 log = GrMin(GrMax(0, log), kMaxSub);
184 return log;
185#else
robertphillips@google.com7460b372012-04-25 16:54:51 +0000186 SkScalar log = SkScalarLog(
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000187 SkScalarDiv(dsqd,
robertphillips@google.com7460b372012-04-25 16:54:51 +0000188 SkScalarMul(gSubdivTol, gSubdivTol)));
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000189 static const SkScalar conv = SkScalarInvert(SkScalarLog(2));
190 log = SkScalarMul(log, conv);
191 return GrMin(GrMax(0, SkScalarCeilToInt(log)),kMaxSub);
192#endif
193 }
194}
195
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000196/**
197 * Generates the lines and quads to be rendered. Lines are always recorded in
198 * device space. We will do a device space bloat to account for the 1pixel
199 * thickness.
200 * Quads are recorded in device space unless m contains
201 * perspective, then in they are in src space. We do this because we will
202 * subdivide large quads to reduce over-fill. This subdivision has to be
203 * performed before applying the perspective matrix.
204 */
205int generate_lines_and_quads(const SkPath& path,
206 const SkMatrix& m,
robertphillips@google.com7b112892012-07-31 15:18:21 +0000207 const GrIRect& devClipBounds,
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000208 PtArray* lines,
209 PtArray* quads,
210 IntArray* quadSubdivCnts) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000211 SkPath::Iter iter(path, false);
212
213 int totalQuadCount = 0;
214 GrRect bounds;
215 GrIRect ibounds;
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000216
217 bool persp = m.hasPerspective();
218
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000219 for (;;) {
commit-bot@chromium.org912e68e2013-05-24 18:51:55 +0000220 GrPoint pathPts[4];
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000221 GrPoint devPts[4];
commit-bot@chromium.org912e68e2013-05-24 18:51:55 +0000222 SkPath::Verb verb = iter.next(pathPts);
bsalomon@google.com94b284d2013-05-10 17:14:06 +0000223 switch (verb) {
224 case SkPath::kMove_Verb:
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000225 break;
bsalomon@google.com94b284d2013-05-10 17:14:06 +0000226 case SkPath::kLine_Verb:
commit-bot@chromium.org912e68e2013-05-24 18:51:55 +0000227 m.mapPoints(devPts, pathPts, 2);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000228 bounds.setBounds(devPts, 2);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000229 bounds.outset(SK_Scalar1, SK_Scalar1);
230 bounds.roundOut(&ibounds);
robertphillips@google.com7b112892012-07-31 15:18:21 +0000231 if (SkIRect::Intersects(devClipBounds, ibounds)) {
bsalomon@google.coma996fec2011-09-13 18:49:13 +0000232 SkPoint* pts = lines->push_back_n(2);
233 pts[0] = devPts[0];
234 pts[1] = devPts[1];
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000235 }
236 break;
commit-bot@chromium.org912e68e2013-05-24 18:51:55 +0000237 case SkPath::kQuad_Verb: {
238 SkPoint choppedPts[5];
239 // Chopping the quad helps when the quad is either degenerate or nearly degenerate.
240 // When it is degenerate it allows the approximation with lines to work since the
241 // chop point (if there is one) will be at the parabola's vertex. In the nearly
242 // degenerate the QuadUVMatrix computed for the points is almost singular which
243 // can cause rendering artifacts.
244 int n = SkChopQuadAtMaxCurvature(pathPts, choppedPts);
245 for (int i = 0; i < n; ++i) {
246 SkPoint* quadPts = choppedPts + i * 2;
247 m.mapPoints(devPts, quadPts, 3);
248 bounds.setBounds(devPts, 3);
249 bounds.outset(SK_Scalar1, SK_Scalar1);
250 bounds.roundOut(&ibounds);
251
252 if (SkIRect::Intersects(devClipBounds, ibounds)) {
253 int subdiv = num_quad_subdivs(devPts);
254 GrAssert(subdiv >= -1);
255 if (-1 == subdiv) {
256 SkPoint* pts = lines->push_back_n(4);
257 pts[0] = devPts[0];
258 pts[1] = devPts[1];
259 pts[2] = devPts[1];
260 pts[3] = devPts[2];
261 } else {
262 // when in perspective keep quads in src space
263 SkPoint* qPts = persp ? quadPts : devPts;
264 SkPoint* pts = quads->push_back_n(3);
265 pts[0] = qPts[0];
266 pts[1] = qPts[1];
267 pts[2] = qPts[2];
268 quadSubdivCnts->push_back() = subdiv;
269 totalQuadCount += 1 << subdiv;
270 }
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000271 }
272 }
bsalomon@google.coma51ab842012-07-10 19:53:34 +0000273 break;
commit-bot@chromium.org912e68e2013-05-24 18:51:55 +0000274 }
bsalomon@google.com94b284d2013-05-10 17:14:06 +0000275 case SkPath::kCubic_Verb:
commit-bot@chromium.org912e68e2013-05-24 18:51:55 +0000276 m.mapPoints(devPts, pathPts, 4);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000277 bounds.setBounds(devPts, 4);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000278 bounds.outset(SK_Scalar1, SK_Scalar1);
279 bounds.roundOut(&ibounds);
robertphillips@google.com7b112892012-07-31 15:18:21 +0000280 if (SkIRect::Intersects(devClipBounds, ibounds)) {
bsalomon@google.com92669012011-09-27 19:10:05 +0000281 PREALLOC_PTARRAY(32) q;
bsalomon@google.coma51ab842012-07-10 19:53:34 +0000282 // we don't need a direction if we aren't constraining the subdivision
283 static const SkPath::Direction kDummyDir = SkPath::kCCW_Direction;
bsalomon@google.com69cc6ad2012-01-17 14:25:10 +0000284 // We convert cubics to quadratics (for now).
285 // In perspective have to do conversion in src space.
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000286 if (persp) {
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000287 SkScalar tolScale =
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000288 GrPathUtils::scaleToleranceToSrc(SK_Scalar1, m,
289 path.getBounds());
commit-bot@chromium.org912e68e2013-05-24 18:51:55 +0000290 GrPathUtils::convertCubicToQuads(pathPts, tolScale, false, kDummyDir, &q);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000291 } else {
bsalomon@google.coma51ab842012-07-10 19:53:34 +0000292 GrPathUtils::convertCubicToQuads(devPts, SK_Scalar1, false, kDummyDir, &q);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000293 }
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000294 for (int i = 0; i < q.count(); i += 3) {
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000295 SkPoint* qInDevSpace;
296 // bounds has to be calculated in device space, but q is
297 // in src space when there is perspective.
298 if (persp) {
299 m.mapPoints(devPts, &q[i], 3);
300 bounds.setBounds(devPts, 3);
301 qInDevSpace = devPts;
302 } else {
303 bounds.setBounds(&q[i], 3);
304 qInDevSpace = &q[i];
305 }
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000306 bounds.outset(SK_Scalar1, SK_Scalar1);
307 bounds.roundOut(&ibounds);
robertphillips@google.com7b112892012-07-31 15:18:21 +0000308 if (SkIRect::Intersects(devClipBounds, ibounds)) {
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000309 int subdiv = num_quad_subdivs(qInDevSpace);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000310 GrAssert(subdiv >= -1);
311 if (-1 == subdiv) {
bsalomon@google.coma996fec2011-09-13 18:49:13 +0000312 SkPoint* pts = lines->push_back_n(4);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000313 // lines should always be in device coords
bsalomon@google.coma996fec2011-09-13 18:49:13 +0000314 pts[0] = qInDevSpace[0];
315 pts[1] = qInDevSpace[1];
316 pts[2] = qInDevSpace[1];
317 pts[3] = qInDevSpace[2];
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000318 } else {
bsalomon@google.coma996fec2011-09-13 18:49:13 +0000319 SkPoint* pts = quads->push_back_n(3);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000320 // q is already in src space when there is no
321 // perspective and dev coords otherwise.
bsalomon@google.coma996fec2011-09-13 18:49:13 +0000322 pts[0] = q[0 + i];
323 pts[1] = q[1 + i];
324 pts[2] = q[2 + i];
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000325 quadSubdivCnts->push_back() = subdiv;
326 totalQuadCount += 1 << subdiv;
327 }
328 }
329 }
330 }
bsalomon@google.coma51ab842012-07-10 19:53:34 +0000331 break;
bsalomon@google.com94b284d2013-05-10 17:14:06 +0000332 case SkPath::kClose_Verb:
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000333 break;
bsalomon@google.com94b284d2013-05-10 17:14:06 +0000334 case SkPath::kDone_Verb:
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000335 return totalQuadCount;
336 }
337 }
338}
339
340struct Vertex {
341 GrPoint fPos;
342 union {
343 struct {
bsalomon@google.com81712882012-11-01 17:12:34 +0000344 SkScalar fA;
345 SkScalar fB;
346 SkScalar fC;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000347 } fLine;
348 GrVec fQuadCoord;
349 struct {
bsalomon@google.com81712882012-11-01 17:12:34 +0000350 SkScalar fBogus[4];
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000351 };
352 };
353};
354GR_STATIC_ASSERT(sizeof(Vertex) == 3 * sizeof(GrPoint));
355
356void intersect_lines(const SkPoint& ptA, const SkVector& normA,
357 const SkPoint& ptB, const SkVector& normB,
358 SkPoint* result) {
359
360 SkScalar lineAW = -normA.dot(ptA);
361 SkScalar lineBW = -normB.dot(ptB);
362
363 SkScalar wInv = SkScalarMul(normA.fX, normB.fY) -
364 SkScalarMul(normA.fY, normB.fX);
365 wInv = SkScalarInvert(wInv);
366
367 result->fX = SkScalarMul(normA.fY, lineBW) - SkScalarMul(lineAW, normB.fY);
368 result->fX = SkScalarMul(result->fX, wInv);
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000369
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000370 result->fY = SkScalarMul(lineAW, normB.fX) - SkScalarMul(normA.fX, lineBW);
371 result->fY = SkScalarMul(result->fY, wInv);
372}
373
bsalomon@google.comb9086a02012-11-01 18:02:54 +0000374void bloat_quad(const SkPoint qpts[3], const SkMatrix* toDevice,
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000375 const SkMatrix* toSrc, Vertex verts[kVertsPerQuad],
376 SkRect* devBounds) {
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000377 GrAssert(!toDevice == !toSrc);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000378 // original quad is specified by tri a,b,c
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000379 SkPoint a = qpts[0];
380 SkPoint b = qpts[1];
381 SkPoint c = qpts[2];
382
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000383 // this should be in the src space, not dev coords, when we have perspective
bsalomon@google.com19713172012-03-15 13:51:08 +0000384 GrPathUtils::QuadUVMatrix DevToUV(qpts);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000385
386 if (toDevice) {
387 toDevice->mapPoints(&a, 1);
388 toDevice->mapPoints(&b, 1);
389 toDevice->mapPoints(&c, 1);
390 }
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000391 // make a new poly where we replace a and c by a 1-pixel wide edges orthog
392 // to edges ab and bc:
393 //
394 // before | after
395 // | b0
396 // b |
397 // |
398 // | a0 c0
399 // a c | a1 c1
400 //
401 // edges a0->b0 and b0->c0 are parallel to original edges a->b and b->c,
402 // respectively.
403 Vertex& a0 = verts[0];
404 Vertex& a1 = verts[1];
405 Vertex& b0 = verts[2];
406 Vertex& c0 = verts[3];
407 Vertex& c1 = verts[4];
408
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000409 SkVector ab = b;
410 ab -= a;
411 SkVector ac = c;
412 ac -= a;
413 SkVector cb = b;
414 cb -= c;
415
416 // We should have already handled degenerates
417 GrAssert(ab.length() > 0 && cb.length() > 0);
418
419 ab.normalize();
420 SkVector abN;
421 abN.setOrthog(ab, SkVector::kLeft_Side);
422 if (abN.dot(ac) > 0) {
423 abN.negate();
424 }
425
426 cb.normalize();
427 SkVector cbN;
428 cbN.setOrthog(cb, SkVector::kLeft_Side);
429 if (cbN.dot(ac) < 0) {
430 cbN.negate();
431 }
432
433 a0.fPos = a;
434 a0.fPos += abN;
435 a1.fPos = a;
436 a1.fPos -= abN;
437
438 c0.fPos = c;
439 c0.fPos += cbN;
440 c1.fPos = c;
441 c1.fPos -= cbN;
442
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000443 // This point may not be within 1 pixel of a control point. We update the bounding box to
444 // include it.
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000445 intersect_lines(a0.fPos, abN, c0.fPos, cbN, &b0.fPos);
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000446 devBounds->growToInclude(b0.fPos.fX, b0.fPos.fY);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000447
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000448 if (toSrc) {
449 toSrc->mapPointsWithStride(&verts[0].fPos, sizeof(Vertex), kVertsPerQuad);
450 }
bsalomon@google.com19713172012-03-15 13:51:08 +0000451 DevToUV.apply<kVertsPerQuad, sizeof(Vertex), sizeof(GrPoint)>(verts);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000452}
453
454void add_quads(const SkPoint p[3],
455 int subdiv,
bsalomon@google.comb9086a02012-11-01 18:02:54 +0000456 const SkMatrix* toDevice,
457 const SkMatrix* toSrc,
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000458 Vertex** vert,
459 SkRect* devBounds) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000460 GrAssert(subdiv >= 0);
461 if (subdiv) {
462 SkPoint newP[5];
463 SkChopQuadAtHalf(p, newP);
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000464 add_quads(newP + 0, subdiv-1, toDevice, toSrc, vert, devBounds);
465 add_quads(newP + 2, subdiv-1, toDevice, toSrc, vert, devBounds);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000466 } else {
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000467 bloat_quad(p, toDevice, toSrc, *vert, devBounds);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000468 *vert += kVertsPerQuad;
469 }
470}
471
472void add_line(const SkPoint p[2],
473 int rtHeight,
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000474 const SkMatrix* toSrc,
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000475 Vertex** vert) {
476 const SkPoint& a = p[0];
477 const SkPoint& b = p[1];
478
479 SkVector orthVec = b;
480 orthVec -= a;
481
482 if (orthVec.setLength(SK_Scalar1)) {
483 orthVec.setOrthog(orthVec);
484
bsalomon@google.com706f6682012-10-23 14:53:55 +0000485 SkScalar lineC = -(a.dot(orthVec));
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000486 for (int i = 0; i < kVertsPerLineSeg; ++i) {
487 (*vert)[i].fPos = (i < 2) ? a : b;
488 if (0 == i || 3 == i) {
489 (*vert)[i].fPos -= orthVec;
490 } else {
491 (*vert)[i].fPos += orthVec;
492 }
bsalomon@google.com706f6682012-10-23 14:53:55 +0000493 (*vert)[i].fLine.fA = orthVec.fX;
494 (*vert)[i].fLine.fB = orthVec.fY;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000495 (*vert)[i].fLine.fC = lineC;
496 }
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000497 if (NULL != toSrc) {
498 toSrc->mapPointsWithStride(&(*vert)->fPos,
499 sizeof(Vertex),
500 kVertsPerLineSeg);
501 }
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000502 } else {
503 // just make it degenerate and likely offscreen
504 (*vert)[0].fPos.set(SK_ScalarMax, SK_ScalarMax);
505 (*vert)[1].fPos.set(SK_ScalarMax, SK_ScalarMax);
506 (*vert)[2].fPos.set(SK_ScalarMax, SK_ScalarMax);
507 (*vert)[3].fPos.set(SK_ScalarMax, SK_ScalarMax);
508 }
509
510 *vert += kVertsPerLineSeg;
511}
512
513}
514
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +0000515///////////////////////////////////////////////////////////////////////////////
516
517/**
518 * The output of this effect is a hairline edge for quadratics.
519 * Quadratic specified by 0=u^2-v canonical coords. u and v are the first
520 * two components of the vertex attribute. Uses unsigned distance.
skia.committer@gmail.com041e2db2013-04-03 07:01:14 +0000521 * Coverage is min(0, 1-distance). 3rd & 4th component unused.
522 * Requires shader derivative instruction support.
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +0000523 */
524class HairQuadEdgeEffect : public GrEffect {
525public:
526
527 static GrEffectRef* Create() {
bsalomon@google.comd42aca32013-04-23 15:37:27 +0000528 GR_CREATE_STATIC_EFFECT(gHairQuadEdgeEffect, HairQuadEdgeEffect, ());
529 gHairQuadEdgeEffect->ref();
530 return gHairQuadEdgeEffect;
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +0000531 }
532
533 virtual ~HairQuadEdgeEffect() {}
534
535 static const char* Name() { return "HairQuadEdge"; }
536
skia.committer@gmail.com041e2db2013-04-03 07:01:14 +0000537 virtual void getConstantColorComponents(GrColor* color,
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +0000538 uint32_t* validFlags) const SK_OVERRIDE {
539 *validFlags = 0;
540 }
541
542 virtual const GrBackendEffectFactory& getFactory() const SK_OVERRIDE {
543 return GrTBackendEffectFactory<HairQuadEdgeEffect>::getInstance();
544 }
545
546 class GLEffect : public GrGLEffect {
547 public:
548 GLEffect(const GrBackendEffectFactory& factory, const GrDrawEffect&)
549 : INHERITED (factory) {}
550
551 virtual void emitCode(GrGLShaderBuilder* builder,
552 const GrDrawEffect& drawEffect,
553 EffectKey key,
554 const char* outputColor,
555 const char* inputColor,
556 const TextureSamplerArray& samplers) SK_OVERRIDE {
557 const char *vsName, *fsName;
558 const SkString* attrName =
559 builder->getEffectAttributeName(drawEffect.getVertexAttribIndices()[0]);
560 builder->fsCodeAppendf("\t\tfloat edgeAlpha;\n");
561
562 SkAssertResult(builder->enableFeature(
563 GrGLShaderBuilder::kStandardDerivatives_GLSLFeature));
564 builder->addVarying(kVec4f_GrSLType, "HairQuadEdge", &vsName, &fsName);
565
566 builder->fsCodeAppendf("\t\tvec2 duvdx = dFdx(%s.xy);\n", fsName);
567 builder->fsCodeAppendf("\t\tvec2 duvdy = dFdy(%s.xy);\n", fsName);
568 builder->fsCodeAppendf("\t\tvec2 gF = vec2(2.0*%s.x*duvdx.x - duvdx.y,\n"
569 "\t\t 2.0*%s.x*duvdy.x - duvdy.y);\n",
570 fsName, fsName);
571 builder->fsCodeAppendf("\t\tedgeAlpha = (%s.x*%s.x - %s.y);\n", fsName, fsName,
572 fsName);
573 builder->fsCodeAppend("\t\tedgeAlpha = sqrt(edgeAlpha*edgeAlpha / dot(gF, gF));\n");
574 builder->fsCodeAppend("\t\tedgeAlpha = max(1.0 - edgeAlpha, 0.0);\n");
575
576 SkString modulate;
bsalomon@google.com018f1792013-04-18 19:36:09 +0000577 GrGLSLModulatef<4>(&modulate, inputColor, "edgeAlpha");
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +0000578 builder->fsCodeAppendf("\t%s = %s;\n", outputColor, modulate.c_str());
579
580 builder->vsCodeAppendf("\t%s = %s;\n", vsName, attrName->c_str());
581 }
582
583 static inline EffectKey GenKey(const GrDrawEffect& drawEffect, const GrGLCaps&) {
584 return 0x0;
585 }
586
587 virtual void setData(const GrGLUniformManager&, const GrDrawEffect&) SK_OVERRIDE {}
588
589 private:
590 typedef GrGLEffect INHERITED;
591 };
592
593private:
skia.committer@gmail.com041e2db2013-04-03 07:01:14 +0000594 HairQuadEdgeEffect() {
595 this->addVertexAttrib(kVec4f_GrSLType);
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +0000596 }
597
598 virtual bool onIsEqual(const GrEffect& other) const SK_OVERRIDE {
599 return true;
600 }
601
602 GR_DECLARE_EFFECT_TEST;
603
604 typedef GrEffect INHERITED;
605};
606
607GR_DEFINE_EFFECT_TEST(HairQuadEdgeEffect);
608
609GrEffectRef* HairQuadEdgeEffect::TestCreate(SkMWCRandom* random,
610 GrContext*,
611 const GrDrawTargetCaps& caps,
612 GrTexture*[]) {
613 // Doesn't work without derivative instructions.
614 return caps.shaderDerivativeSupport() ? HairQuadEdgeEffect::Create() : NULL;}
615
616///////////////////////////////////////////////////////////////////////////////
617
618/**
skia.committer@gmail.com041e2db2013-04-03 07:01:14 +0000619 * The output of this effect is a 1-pixel wide line.
620 * Input is 2D implicit device coord line eq (a*x + b*y +c = 0). 4th component unused.
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +0000621 */
622class HairLineEdgeEffect : public GrEffect {
623public:
624
625 static GrEffectRef* Create() {
bsalomon@google.comd42aca32013-04-23 15:37:27 +0000626 GR_CREATE_STATIC_EFFECT(gHairLineEdge, HairLineEdgeEffect, ());
627 gHairLineEdge->ref();
628 return gHairLineEdge;
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +0000629 }
630
631 virtual ~HairLineEdgeEffect() {}
632
633 static const char* Name() { return "HairLineEdge"; }
634
skia.committer@gmail.com041e2db2013-04-03 07:01:14 +0000635 virtual void getConstantColorComponents(GrColor* color,
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +0000636 uint32_t* validFlags) const SK_OVERRIDE {
637 *validFlags = 0;
638 }
639
640 virtual const GrBackendEffectFactory& getFactory() const SK_OVERRIDE {
641 return GrTBackendEffectFactory<HairLineEdgeEffect>::getInstance();
642 }
643
644 class GLEffect : public GrGLEffect {
645 public:
646 GLEffect(const GrBackendEffectFactory& factory, const GrDrawEffect&)
647 : INHERITED (factory) {}
648
649 virtual void emitCode(GrGLShaderBuilder* builder,
650 const GrDrawEffect& drawEffect,
651 EffectKey key,
652 const char* outputColor,
653 const char* inputColor,
654 const TextureSamplerArray& samplers) SK_OVERRIDE {
655 const char *vsName, *fsName;
656 const SkString* attrName =
657 builder->getEffectAttributeName(drawEffect.getVertexAttribIndices()[0]);
658 builder->fsCodeAppendf("\t\tfloat edgeAlpha;\n");
659
660 builder->addVarying(kVec4f_GrSLType, "HairLineEdge", &vsName, &fsName);
661
662 builder->fsCodeAppendf("\t\tedgeAlpha = abs(dot(vec3(%s.xy,1), %s.xyz));\n",
663 builder->fragmentPosition(), fsName);
664 builder->fsCodeAppendf("\t\tedgeAlpha = max(1.0 - edgeAlpha, 0.0);\n");
665
666 SkString modulate;
bsalomon@google.com018f1792013-04-18 19:36:09 +0000667 GrGLSLModulatef<4>(&modulate, inputColor, "edgeAlpha");
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +0000668 builder->fsCodeAppendf("\t%s = %s;\n", outputColor, modulate.c_str());
669
670 builder->vsCodeAppendf("\t%s = %s;\n", vsName, attrName->c_str());
671 }
672
673 static inline EffectKey GenKey(const GrDrawEffect& drawEffect, const GrGLCaps&) {
674 return 0x0;
675 }
676
677 virtual void setData(const GrGLUniformManager&, const GrDrawEffect&) SK_OVERRIDE {}
678
679 private:
680 typedef GrGLEffect INHERITED;
681 };
682
683private:
skia.committer@gmail.com041e2db2013-04-03 07:01:14 +0000684 HairLineEdgeEffect() {
685 this->addVertexAttrib(kVec4f_GrSLType);
commit-bot@chromium.org8d47ddc2013-05-09 14:55:46 +0000686 this->setWillReadFragmentPosition();
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +0000687 }
688
689 virtual bool onIsEqual(const GrEffect& other) const SK_OVERRIDE {
690 return true;
691 }
692
693 GR_DECLARE_EFFECT_TEST;
694
695 typedef GrEffect INHERITED;
696};
697
698GR_DEFINE_EFFECT_TEST(HairLineEdgeEffect);
699
700GrEffectRef* HairLineEdgeEffect::TestCreate(SkMWCRandom* random,
701 GrContext*,
702 const GrDrawTargetCaps& caps,
703 GrTexture*[]) {
704 return HairLineEdgeEffect::Create();
705}
706
707///////////////////////////////////////////////////////////////////////////////
708
robertphillips@google.com42903302013-04-20 12:26:07 +0000709namespace {
710
711// position + edge
712extern const GrVertexAttrib gHairlineAttribs[] = {
713 {kVec2f_GrVertexAttribType, 0, kPosition_GrVertexAttribBinding},
714 {kVec4f_GrVertexAttribType, sizeof(GrPoint), kEffect_GrVertexAttribBinding}
715};
716
717};
718
bsalomon@google.comb3729422012-03-07 19:13:28 +0000719bool GrAAHairLinePathRenderer::createGeom(
720 const SkPath& path,
bsalomon@google.comb3729422012-03-07 19:13:28 +0000721 GrDrawTarget* target,
bsalomon@google.comb3729422012-03-07 19:13:28 +0000722 int* lineCnt,
723 int* quadCnt,
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000724 GrDrawTarget::AutoReleaseGeometry* arg,
725 SkRect* devBounds) {
jvanverth@google.com9b855c72013-03-01 18:21:22 +0000726 GrDrawState* drawState = target->drawState();
727 int rtHeight = drawState->getRenderTarget()->height();
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000728
robertphillips@google.com7b112892012-07-31 15:18:21 +0000729 GrIRect devClipBounds;
commit-bot@chromium.org912e68e2013-05-24 18:51:55 +0000730 target->getClip()->getConservativeBounds(drawState->getRenderTarget(), &devClipBounds);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000731
jvanverth@google.com9b855c72013-03-01 18:21:22 +0000732 SkMatrix viewM = drawState->getViewMatrix();
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000733
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000734 // All the vertices that we compute are within 1 of path control points with the exception of
735 // one of the bounding vertices for each quad. The add_quads() function will update the bounds
736 // for each quad added.
737 *devBounds = path.getBounds();
738 viewM.mapRect(devBounds);
739 devBounds->outset(SK_Scalar1, SK_Scalar1);
740
bsalomon@google.com92669012011-09-27 19:10:05 +0000741 PREALLOC_PTARRAY(128) lines;
742 PREALLOC_PTARRAY(128) quads;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000743 IntArray qSubdivs;
bsalomon@google.com0f11e1a2012-10-08 14:48:36 +0000744 *quadCnt = generate_lines_and_quads(path, viewM, devClipBounds,
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000745 &lines, &quads, &qSubdivs);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000746
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000747 *lineCnt = lines.count() / 2;
748 int vertCnt = kVertsPerLineSeg * *lineCnt + kVertsPerQuad * *quadCnt;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000749
robertphillips@google.com42903302013-04-20 12:26:07 +0000750 target->drawState()->setVertexAttribs<gHairlineAttribs>(SK_ARRAY_COUNT(gHairlineAttribs));
jvanverth@google.comb75b0a02013-02-05 20:33:30 +0000751 GrAssert(sizeof(Vertex) == target->getDrawState().getVertexSize());
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000752
jvanverth@google.comb75b0a02013-02-05 20:33:30 +0000753 if (!arg->set(target, vertCnt, 0)) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000754 return false;
755 }
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000756
bsalomon@google.comb3729422012-03-07 19:13:28 +0000757 Vertex* verts = reinterpret_cast<Vertex*>(arg->vertices());
758
bsalomon@google.comb9086a02012-11-01 18:02:54 +0000759 const SkMatrix* toDevice = NULL;
760 const SkMatrix* toSrc = NULL;
761 SkMatrix ivm;
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000762
763 if (viewM.hasPerspective()) {
764 if (viewM.invert(&ivm)) {
765 toDevice = &viewM;
766 toSrc = &ivm;
767 }
768 }
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000769
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000770 for (int i = 0; i < *lineCnt; ++i) {
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000771 add_line(&lines[2*i], rtHeight, toSrc, &verts);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000772 }
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000773
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000774 int unsubdivQuadCnt = quads.count() / 3;
775 for (int i = 0; i < unsubdivQuadCnt; ++i) {
776 GrAssert(qSubdivs[i] >= 0);
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000777 add_quads(&quads[3*i], qSubdivs[i], toDevice, toSrc, &verts, devBounds);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000778 }
779
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000780 return true;
781}
782
robertphillips@google.com8a4fc402012-05-24 12:42:24 +0000783bool GrAAHairLinePathRenderer::canDrawPath(const SkPath& path,
sugoi@google.com5f74cf82012-12-17 21:16:45 +0000784 const SkStrokeRec& stroke,
robertphillips@google.com8a4fc402012-05-24 12:42:24 +0000785 const GrDrawTarget* target,
786 bool antiAlias) const {
sugoi@google.com5f74cf82012-12-17 21:16:45 +0000787 if (!stroke.isHairlineStyle() || !antiAlias) {
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000788 return false;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000789 }
790
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000791 static const uint32_t gReqDerivMask = SkPath::kCubic_SegmentMask |
792 SkPath::kQuad_SegmentMask;
bsalomon@google.combcce8922013-03-25 15:38:39 +0000793 if (!target->caps()->shaderDerivativeSupport() &&
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000794 (gReqDerivMask & path.getSegmentMasks())) {
795 return false;
796 }
797 return true;
798}
799
800bool GrAAHairLinePathRenderer::onDrawPath(const SkPath& path,
sugoi@google.com5f74cf82012-12-17 21:16:45 +0000801 const SkStrokeRec&,
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000802 GrDrawTarget* target,
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000803 bool antiAlias) {
804
805 int lineCnt;
806 int quadCnt;
bsalomon@google.comb3729422012-03-07 19:13:28 +0000807 GrDrawTarget::AutoReleaseGeometry arg;
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000808 SkRect devBounds;
809
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000810 if (!this->createGeom(path,
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000811 target,
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000812 &lineCnt,
bsalomon@google.comb3729422012-03-07 19:13:28 +0000813 &quadCnt,
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000814 &arg,
815 &devBounds)) {
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000816 return false;
817 }
818
bsalomon@google.com4647f902013-03-26 14:45:27 +0000819 GrDrawTarget::AutoStateRestore asr(target, GrDrawTarget::kPreserve_ASRInit);
bsalomon@google.com873ea0c2012-03-30 15:55:32 +0000820 GrDrawState* drawState = target->drawState();
bsalomon@google.com4647f902013-03-26 14:45:27 +0000821
822 GrDrawState::AutoDeviceCoordDraw adcd;
bsalomon@google.coma8347462012-10-08 18:59:39 +0000823 // createGeom transforms the geometry to device space when the matrix does not have
824 // perspective.
bsalomon@google.com8f9cbd62011-12-09 15:55:34 +0000825 if (!drawState->getViewMatrix().hasPerspective()) {
bsalomon@google.coma8347462012-10-08 18:59:39 +0000826 adcd.set(drawState);
827 if (!adcd.succeeded()) {
bsalomon@google.come3d32162012-07-20 13:37:06 +0000828 return false;
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000829 }
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000830 }
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000831
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000832 // TODO: See whether rendering lines as degenerate quads improves perf
833 // when we have a mix
bsalomon@google.coma8347462012-10-08 18:59:39 +0000834
bsalomon@google.com4647f902013-03-26 14:45:27 +0000835 enum {
836 // the edge effects share this stage with glyph rendering
837 // (kGlyphMaskStage in GrTextContext) && SW path rendering
838 // (kPathMaskStage in GrSWMaskHelper)
839 kEdgeEffectStage = GrPaint::kTotalStages,
840 };
841 static const int kEdgeAttrIndex = 1;
bsalomon@google.coma8347462012-10-08 18:59:39 +0000842
commit-bot@chromium.org90c240a2013-04-02 17:57:21 +0000843 GrEffectRef* hairLineEffect = HairLineEdgeEffect::Create();
844 GrEffectRef* hairQuadEffect = HairQuadEdgeEffect::Create();
skia.committer@gmail.com37cbc7f2013-03-27 07:01:04 +0000845
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000846 // Check devBounds
847#if GR_DEBUG
848 SkRect tolDevBounds = devBounds;
849 tolDevBounds.outset(SK_Scalar1 / 10000, SK_Scalar1 / 10000);
850 SkRect actualBounds;
851 Vertex* verts = reinterpret_cast<Vertex*>(arg.vertices());
852 int vCount = kVertsPerLineSeg * lineCnt + kVertsPerQuad * quadCnt;
853 bool first = true;
854 for (int i = 0; i < vCount; ++i) {
855 SkPoint pos = verts[i].fPos;
856 // This is a hack to workaround the fact that we move some degenerate segments offscreen.
857 if (SK_ScalarMax == pos.fX) {
858 continue;
859 }
860 drawState->getViewMatrix().mapPoints(&pos, 1);
861 if (first) {
862 actualBounds.set(pos.fX, pos.fY, pos.fX, pos.fY);
863 first = false;
864 } else {
865 actualBounds.growToInclude(pos.fX, pos.fY);
866 }
867 }
868 if (!first) {
869 GrAssert(tolDevBounds.contains(actualBounds));
870 }
871#endif
872
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000873 target->setIndexSourceToBuffer(fLinesIndexBuffer);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000874 int lines = 0;
875 int nBufLines = fLinesIndexBuffer->maxQuads();
bsalomon@google.com4647f902013-03-26 14:45:27 +0000876 drawState->setEffect(kEdgeEffectStage, hairLineEffect, kEdgeAttrIndex)->unref();
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000877 while (lines < lineCnt) {
878 int n = GrMin(lineCnt - lines, nBufLines);
bsalomon@google.com47059542012-06-06 20:51:20 +0000879 target->drawIndexed(kTriangles_GrPrimitiveType,
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000880 kVertsPerLineSeg*lines, // startV
881 0, // startI
882 kVertsPerLineSeg*n, // vCount
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000883 kIdxsPerLineSeg*n,
884 &devBounds); // iCount
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000885 lines += n;
886 }
887
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000888 target->setIndexSourceToBuffer(fQuadsIndexBuffer);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000889 int quads = 0;
bsalomon@google.com4647f902013-03-26 14:45:27 +0000890 drawState->setEffect(kEdgeEffectStage, hairQuadEffect, kEdgeAttrIndex)->unref();
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000891 while (quads < quadCnt) {
892 int n = GrMin(quadCnt - quads, kNumQuadsInIdxBuffer);
bsalomon@google.com47059542012-06-06 20:51:20 +0000893 target->drawIndexed(kTriangles_GrPrimitiveType,
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000894 4 * lineCnt + kVertsPerQuad*quads, // startV
895 0, // startI
896 kVertsPerQuad*n, // vCount
bsalomon@google.com1dd9baa2013-05-20 16:49:06 +0000897 kIdxsPerQuad*n, // iCount
898 &devBounds);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000899 quads += n;
900 }
bsalomon@google.com0406b9e2013-04-02 21:00:15 +0000901 target->resetIndexSource();
bsalomon@google.com4647f902013-03-26 14:45:27 +0000902
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000903 return true;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000904}