blob: cd3f7748035261062386a41bca7dfc47580d3d34 [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.comaeb21602011-08-30 18:13:44 +000013#include "GrGpu.h"
14#include "GrIndexBuffer.h"
bsalomon@google.comdbeeac32011-09-12 14:59:34 +000015#include "GrPathUtils.h"
bsalomon@google.comaeb21602011-08-30 18:13:44 +000016#include "SkGeometry.h"
17#include "SkTemplates.h"
18
19namespace {
20// quadratics are rendered as 5-sided polys in order to bound the
21// AA stroke around the center-curve. See comments in push_quad_index_buffer and
22// bloat_quad.
23static const int kVertsPerQuad = 5;
24static const int kIdxsPerQuad = 9;
25
26static const int kVertsPerLineSeg = 4;
27static const int kIdxsPerLineSeg = 6;
28
29static const int kNumQuadsInIdxBuffer = 256;
30static const size_t kQuadIdxSBufize = kIdxsPerQuad *
31 sizeof(uint16_t) *
32 kNumQuadsInIdxBuffer;
33
34bool push_quad_index_data(GrIndexBuffer* qIdxBuffer) {
35 uint16_t* data = (uint16_t*) qIdxBuffer->lock();
36 bool tempData = NULL == data;
37 if (tempData) {
tomhudson@google.comc377baf2012-07-09 20:17:56 +000038 data = SkNEW_ARRAY(uint16_t, kNumQuadsInIdxBuffer * kIdxsPerQuad);
bsalomon@google.comaeb21602011-08-30 18:13:44 +000039 }
40 for (int i = 0; i < kNumQuadsInIdxBuffer; ++i) {
41
42 // Each quadratic is rendered as a five sided polygon. This poly bounds
43 // the quadratic's bounding triangle but has been expanded so that the
44 // 1-pixel wide area around the curve is inside the poly.
45 // If a,b,c are the original control points then the poly a0,b0,c0,c1,a1
46 // that is rendered would look like this:
47 // b0
48 // b
49 //
50 // a0 c0
51 // a c
52 // a1 c1
bsalomon@google.com0e5104c2012-04-10 16:20:41 +000053 // Each is drawn as three triangles specified by these 9 indices:
bsalomon@google.comaeb21602011-08-30 18:13:44 +000054 int baseIdx = i * kIdxsPerQuad;
55 uint16_t baseVert = (uint16_t)(i * kVertsPerQuad);
56 data[0 + baseIdx] = baseVert + 0; // a0
57 data[1 + baseIdx] = baseVert + 1; // a1
58 data[2 + baseIdx] = baseVert + 2; // b0
59 data[3 + baseIdx] = baseVert + 2; // b0
60 data[4 + baseIdx] = baseVert + 4; // c1
61 data[5 + baseIdx] = baseVert + 3; // c0
62 data[6 + baseIdx] = baseVert + 1; // a1
63 data[7 + baseIdx] = baseVert + 4; // c1
64 data[8 + baseIdx] = baseVert + 2; // b0
65 }
66 if (tempData) {
67 bool ret = qIdxBuffer->updateData(data, kQuadIdxSBufize);
68 delete[] data;
69 return ret;
70 } else {
71 qIdxBuffer->unlock();
72 return true;
73 }
74}
75}
76
77GrPathRenderer* GrAAHairLinePathRenderer::Create(GrContext* context) {
bsalomon@google.coma8a6a322011-09-23 14:19:58 +000078 const GrIndexBuffer* lIdxBuffer = context->getQuadIndexBuffer();
79 if (NULL == lIdxBuffer) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +000080 return NULL;
81 }
bsalomon@google.coma8a6a322011-09-23 14:19:58 +000082 GrGpu* gpu = context->getGpu();
83 GrIndexBuffer* qIdxBuf = gpu->createIndexBuffer(kQuadIdxSBufize, false);
84 SkAutoTUnref<GrIndexBuffer> qIdxBuffer(qIdxBuf);
85 if (NULL == qIdxBuf ||
86 !push_quad_index_data(qIdxBuf)) {
87 return NULL;
88 }
tomhudson@google.comc377baf2012-07-09 20:17:56 +000089 return SkNEW_ARGS(GrAAHairLinePathRenderer,
90 (context, lIdxBuffer, qIdxBuf));
bsalomon@google.comaeb21602011-08-30 18:13:44 +000091}
92
93GrAAHairLinePathRenderer::GrAAHairLinePathRenderer(
94 const GrContext* context,
95 const GrIndexBuffer* linesIndexBuffer,
96 const GrIndexBuffer* quadsIndexBuffer) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +000097 fLinesIndexBuffer = linesIndexBuffer;
98 linesIndexBuffer->ref();
99 fQuadsIndexBuffer = quadsIndexBuffer;
100 quadsIndexBuffer->ref();
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000101}
102
103GrAAHairLinePathRenderer::~GrAAHairLinePathRenderer() {
104 fLinesIndexBuffer->unref();
105 fQuadsIndexBuffer->unref();
106}
107
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000108namespace {
109
bsalomon@google.com49313f62011-09-14 13:54:05 +0000110typedef SkTArray<SkPoint, true> PtArray;
bsalomon@google.com92669012011-09-27 19:10:05 +0000111#define PREALLOC_PTARRAY(N) SkSTArray<(N),SkPoint, true>
bsalomon@google.com49313f62011-09-14 13:54:05 +0000112typedef SkTArray<int, true> IntArray;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000113
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000114// Takes 178th time of logf on Z600 / VC2010
115int get_float_exp(float x) {
116 GR_STATIC_ASSERT(sizeof(int) == sizeof(float));
117#if GR_DEBUG
118 static bool tested;
119 if (!tested) {
120 tested = true;
121 GrAssert(get_float_exp(0.25f) == -2);
122 GrAssert(get_float_exp(0.3f) == -2);
123 GrAssert(get_float_exp(0.5f) == -1);
124 GrAssert(get_float_exp(1.f) == 0);
125 GrAssert(get_float_exp(2.f) == 1);
126 GrAssert(get_float_exp(2.5f) == 1);
127 GrAssert(get_float_exp(8.f) == 3);
128 GrAssert(get_float_exp(100.f) == 6);
129 GrAssert(get_float_exp(1000.f) == 9);
130 GrAssert(get_float_exp(1024.f) == 10);
131 GrAssert(get_float_exp(3000000.f) == 21);
132 }
133#endif
bsalomon@google.com2ec72802011-09-21 21:46:03 +0000134 const int* iptr = (const int*)&x;
135 return (((*iptr) & 0x7f800000) >> 23) - 127;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000136}
137
138// we subdivide the quads to avoid huge overfill
139// if it returns -1 then should be drawn as lines
140int num_quad_subdivs(const SkPoint p[3]) {
141 static const SkScalar gDegenerateToLineTol = SK_Scalar1;
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000142 static const SkScalar gDegenerateToLineTolSqd =
bsalomon@google.com46a2a1e2011-09-06 22:10:52 +0000143 SkScalarMul(gDegenerateToLineTol, gDegenerateToLineTol);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000144
bsalomon@google.com46a2a1e2011-09-06 22:10:52 +0000145 if (p[0].distanceToSqd(p[1]) < gDegenerateToLineTolSqd ||
146 p[1].distanceToSqd(p[2]) < gDegenerateToLineTolSqd) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000147 return -1;
148 }
bsalomon@google.com46a2a1e2011-09-06 22:10:52 +0000149
bsalomon@google.com81712882012-11-01 17:12:34 +0000150 SkScalar dsqd = p[1].distanceToLineBetweenSqd(p[0], p[2]);
bsalomon@google.com46a2a1e2011-09-06 22:10:52 +0000151 if (dsqd < gDegenerateToLineTolSqd) {
152 return -1;
153 }
154
155 if (p[2].distanceToLineBetweenSqd(p[1], p[0]) < gDegenerateToLineTolSqd) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000156 return -1;
157 }
158
159 static const int kMaxSub = 4;
160 // tolerance of triangle height in pixels
161 // tuned on windows Quadro FX 380 / Z600
162 // trade off of fill vs cpu time on verts
163 // maybe different when do this using gpu (geo or tess shaders)
164 static const SkScalar gSubdivTol = 175 * SK_Scalar1;
165
robertphillips@google.com7460b372012-04-25 16:54:51 +0000166 if (dsqd <= SkScalarMul(gSubdivTol, gSubdivTol)) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000167 return 0;
168 } else {
169 // subdividing the quad reduces d by 4. so we want x = log4(d/tol)
170 // = log4(d*d/tol*tol)/2
171 // = log2(d*d/tol*tol)
172
173#ifdef SK_SCALAR_IS_FLOAT
174 // +1 since we're ignoring the mantissa contribution.
175 int log = get_float_exp(dsqd/(gSubdivTol*gSubdivTol)) + 1;
176 log = GrMin(GrMax(0, log), kMaxSub);
177 return log;
178#else
robertphillips@google.com7460b372012-04-25 16:54:51 +0000179 SkScalar log = SkScalarLog(
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000180 SkScalarDiv(dsqd,
robertphillips@google.com7460b372012-04-25 16:54:51 +0000181 SkScalarMul(gSubdivTol, gSubdivTol)));
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000182 static const SkScalar conv = SkScalarInvert(SkScalarLog(2));
183 log = SkScalarMul(log, conv);
184 return GrMin(GrMax(0, SkScalarCeilToInt(log)),kMaxSub);
185#endif
186 }
187}
188
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000189/**
190 * Generates the lines and quads to be rendered. Lines are always recorded in
191 * device space. We will do a device space bloat to account for the 1pixel
192 * thickness.
193 * Quads are recorded in device space unless m contains
194 * perspective, then in they are in src space. We do this because we will
195 * subdivide large quads to reduce over-fill. This subdivision has to be
196 * performed before applying the perspective matrix.
197 */
198int generate_lines_and_quads(const SkPath& path,
199 const SkMatrix& m,
robertphillips@google.com7b112892012-07-31 15:18:21 +0000200 const GrIRect& devClipBounds,
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000201 PtArray* lines,
202 PtArray* quads,
203 IntArray* quadSubdivCnts) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000204 SkPath::Iter iter(path, false);
205
206 int totalQuadCount = 0;
207 GrRect bounds;
208 GrIRect ibounds;
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000209
210 bool persp = m.hasPerspective();
211
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000212 for (;;) {
213 GrPoint pts[4];
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000214 GrPoint devPts[4];
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000215 GrPathCmd cmd = (GrPathCmd)iter.next(pts);
216 switch (cmd) {
217 case kMove_PathCmd:
218 break;
219 case kLine_PathCmd:
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000220 m.mapPoints(devPts, pts, 2);
221 bounds.setBounds(devPts, 2);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000222 bounds.outset(SK_Scalar1, SK_Scalar1);
223 bounds.roundOut(&ibounds);
robertphillips@google.com7b112892012-07-31 15:18:21 +0000224 if (SkIRect::Intersects(devClipBounds, ibounds)) {
bsalomon@google.coma996fec2011-09-13 18:49:13 +0000225 SkPoint* pts = lines->push_back_n(2);
226 pts[0] = devPts[0];
227 pts[1] = devPts[1];
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000228 }
229 break;
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000230 case kQuadratic_PathCmd:
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000231 m.mapPoints(devPts, pts, 3);
232 bounds.setBounds(devPts, 3);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000233 bounds.outset(SK_Scalar1, SK_Scalar1);
234 bounds.roundOut(&ibounds);
robertphillips@google.com7b112892012-07-31 15:18:21 +0000235 if (SkIRect::Intersects(devClipBounds, ibounds)) {
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000236 int subdiv = num_quad_subdivs(devPts);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000237 GrAssert(subdiv >= -1);
238 if (-1 == subdiv) {
bsalomon@google.coma996fec2011-09-13 18:49:13 +0000239 SkPoint* pts = lines->push_back_n(4);
240 pts[0] = devPts[0];
241 pts[1] = devPts[1];
242 pts[2] = devPts[1];
243 pts[3] = devPts[2];
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000244 } else {
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000245 // when in perspective keep quads in src space
246 SkPoint* qPts = persp ? pts : devPts;
bsalomon@google.coma996fec2011-09-13 18:49:13 +0000247 SkPoint* pts = quads->push_back_n(3);
248 pts[0] = qPts[0];
249 pts[1] = qPts[1];
250 pts[2] = qPts[2];
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000251 quadSubdivCnts->push_back() = subdiv;
252 totalQuadCount += 1 << subdiv;
253 }
254 }
bsalomon@google.coma51ab842012-07-10 19:53:34 +0000255 break;
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000256 case kCubic_PathCmd:
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000257 m.mapPoints(devPts, pts, 4);
258 bounds.setBounds(devPts, 4);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000259 bounds.outset(SK_Scalar1, SK_Scalar1);
260 bounds.roundOut(&ibounds);
robertphillips@google.com7b112892012-07-31 15:18:21 +0000261 if (SkIRect::Intersects(devClipBounds, ibounds)) {
bsalomon@google.com92669012011-09-27 19:10:05 +0000262 PREALLOC_PTARRAY(32) q;
bsalomon@google.coma51ab842012-07-10 19:53:34 +0000263 // we don't need a direction if we aren't constraining the subdivision
264 static const SkPath::Direction kDummyDir = SkPath::kCCW_Direction;
bsalomon@google.com69cc6ad2012-01-17 14:25:10 +0000265 // We convert cubics to quadratics (for now).
266 // In perspective have to do conversion in src space.
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000267 if (persp) {
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000268 SkScalar tolScale =
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000269 GrPathUtils::scaleToleranceToSrc(SK_Scalar1, m,
270 path.getBounds());
bsalomon@google.coma51ab842012-07-10 19:53:34 +0000271 GrPathUtils::convertCubicToQuads(pts, tolScale, false, kDummyDir, &q);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000272 } else {
bsalomon@google.coma51ab842012-07-10 19:53:34 +0000273 GrPathUtils::convertCubicToQuads(devPts, SK_Scalar1, false, kDummyDir, &q);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000274 }
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000275 for (int i = 0; i < q.count(); i += 3) {
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000276 SkPoint* qInDevSpace;
277 // bounds has to be calculated in device space, but q is
278 // in src space when there is perspective.
279 if (persp) {
280 m.mapPoints(devPts, &q[i], 3);
281 bounds.setBounds(devPts, 3);
282 qInDevSpace = devPts;
283 } else {
284 bounds.setBounds(&q[i], 3);
285 qInDevSpace = &q[i];
286 }
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000287 bounds.outset(SK_Scalar1, SK_Scalar1);
288 bounds.roundOut(&ibounds);
robertphillips@google.com7b112892012-07-31 15:18:21 +0000289 if (SkIRect::Intersects(devClipBounds, ibounds)) {
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000290 int subdiv = num_quad_subdivs(qInDevSpace);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000291 GrAssert(subdiv >= -1);
292 if (-1 == subdiv) {
bsalomon@google.coma996fec2011-09-13 18:49:13 +0000293 SkPoint* pts = lines->push_back_n(4);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000294 // lines should always be in device coords
bsalomon@google.coma996fec2011-09-13 18:49:13 +0000295 pts[0] = qInDevSpace[0];
296 pts[1] = qInDevSpace[1];
297 pts[2] = qInDevSpace[1];
298 pts[3] = qInDevSpace[2];
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000299 } else {
bsalomon@google.coma996fec2011-09-13 18:49:13 +0000300 SkPoint* pts = quads->push_back_n(3);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000301 // q is already in src space when there is no
302 // perspective and dev coords otherwise.
bsalomon@google.coma996fec2011-09-13 18:49:13 +0000303 pts[0] = q[0 + i];
304 pts[1] = q[1 + i];
305 pts[2] = q[2 + i];
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000306 quadSubdivCnts->push_back() = subdiv;
307 totalQuadCount += 1 << subdiv;
308 }
309 }
310 }
311 }
bsalomon@google.coma51ab842012-07-10 19:53:34 +0000312 break;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000313 case kClose_PathCmd:
314 break;
315 case kEnd_PathCmd:
316 return totalQuadCount;
317 }
318 }
319}
320
321struct Vertex {
322 GrPoint fPos;
323 union {
324 struct {
bsalomon@google.com81712882012-11-01 17:12:34 +0000325 SkScalar fA;
326 SkScalar fB;
327 SkScalar fC;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000328 } fLine;
329 GrVec fQuadCoord;
330 struct {
bsalomon@google.com81712882012-11-01 17:12:34 +0000331 SkScalar fBogus[4];
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000332 };
333 };
334};
335GR_STATIC_ASSERT(sizeof(Vertex) == 3 * sizeof(GrPoint));
336
337void intersect_lines(const SkPoint& ptA, const SkVector& normA,
338 const SkPoint& ptB, const SkVector& normB,
339 SkPoint* result) {
340
341 SkScalar lineAW = -normA.dot(ptA);
342 SkScalar lineBW = -normB.dot(ptB);
343
344 SkScalar wInv = SkScalarMul(normA.fX, normB.fY) -
345 SkScalarMul(normA.fY, normB.fX);
346 wInv = SkScalarInvert(wInv);
347
348 result->fX = SkScalarMul(normA.fY, lineBW) - SkScalarMul(lineAW, normB.fY);
349 result->fX = SkScalarMul(result->fX, wInv);
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000350
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000351 result->fY = SkScalarMul(lineAW, normB.fX) - SkScalarMul(normA.fX, lineBW);
352 result->fY = SkScalarMul(result->fY, wInv);
353}
354
bsalomon@google.comb9086a02012-11-01 18:02:54 +0000355void bloat_quad(const SkPoint qpts[3], const SkMatrix* toDevice,
356 const SkMatrix* toSrc, Vertex verts[kVertsPerQuad]) {
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000357 GrAssert(!toDevice == !toSrc);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000358 // original quad is specified by tri a,b,c
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000359 SkPoint a = qpts[0];
360 SkPoint b = qpts[1];
361 SkPoint c = qpts[2];
362
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000363 // this should be in the src space, not dev coords, when we have perspective
bsalomon@google.com19713172012-03-15 13:51:08 +0000364 GrPathUtils::QuadUVMatrix DevToUV(qpts);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000365
366 if (toDevice) {
367 toDevice->mapPoints(&a, 1);
368 toDevice->mapPoints(&b, 1);
369 toDevice->mapPoints(&c, 1);
370 }
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000371 // make a new poly where we replace a and c by a 1-pixel wide edges orthog
372 // to edges ab and bc:
373 //
374 // before | after
375 // | b0
376 // b |
377 // |
378 // | a0 c0
379 // a c | a1 c1
380 //
381 // edges a0->b0 and b0->c0 are parallel to original edges a->b and b->c,
382 // respectively.
383 Vertex& a0 = verts[0];
384 Vertex& a1 = verts[1];
385 Vertex& b0 = verts[2];
386 Vertex& c0 = verts[3];
387 Vertex& c1 = verts[4];
388
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000389 SkVector ab = b;
390 ab -= a;
391 SkVector ac = c;
392 ac -= a;
393 SkVector cb = b;
394 cb -= c;
395
396 // We should have already handled degenerates
397 GrAssert(ab.length() > 0 && cb.length() > 0);
398
399 ab.normalize();
400 SkVector abN;
401 abN.setOrthog(ab, SkVector::kLeft_Side);
402 if (abN.dot(ac) > 0) {
403 abN.negate();
404 }
405
406 cb.normalize();
407 SkVector cbN;
408 cbN.setOrthog(cb, SkVector::kLeft_Side);
409 if (cbN.dot(ac) < 0) {
410 cbN.negate();
411 }
412
413 a0.fPos = a;
414 a0.fPos += abN;
415 a1.fPos = a;
416 a1.fPos -= abN;
417
418 c0.fPos = c;
419 c0.fPos += cbN;
420 c1.fPos = c;
421 c1.fPos -= cbN;
422
423 intersect_lines(a0.fPos, abN, c0.fPos, cbN, &b0.fPos);
424
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000425 if (toSrc) {
426 toSrc->mapPointsWithStride(&verts[0].fPos, sizeof(Vertex), kVertsPerQuad);
427 }
bsalomon@google.com19713172012-03-15 13:51:08 +0000428 DevToUV.apply<kVertsPerQuad, sizeof(Vertex), sizeof(GrPoint)>(verts);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000429}
430
431void add_quads(const SkPoint p[3],
432 int subdiv,
bsalomon@google.comb9086a02012-11-01 18:02:54 +0000433 const SkMatrix* toDevice,
434 const SkMatrix* toSrc,
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000435 Vertex** vert) {
436 GrAssert(subdiv >= 0);
437 if (subdiv) {
438 SkPoint newP[5];
439 SkChopQuadAtHalf(p, newP);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000440 add_quads(newP + 0, subdiv-1, toDevice, toSrc, vert);
441 add_quads(newP + 2, subdiv-1, toDevice, toSrc, vert);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000442 } else {
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000443 bloat_quad(p, toDevice, toSrc, *vert);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000444 *vert += kVertsPerQuad;
445 }
446}
447
448void add_line(const SkPoint p[2],
449 int rtHeight,
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000450 const SkMatrix* toSrc,
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000451 Vertex** vert) {
452 const SkPoint& a = p[0];
453 const SkPoint& b = p[1];
454
455 SkVector orthVec = b;
456 orthVec -= a;
457
458 if (orthVec.setLength(SK_Scalar1)) {
459 orthVec.setOrthog(orthVec);
460
bsalomon@google.com706f6682012-10-23 14:53:55 +0000461 SkScalar lineC = -(a.dot(orthVec));
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000462 for (int i = 0; i < kVertsPerLineSeg; ++i) {
463 (*vert)[i].fPos = (i < 2) ? a : b;
464 if (0 == i || 3 == i) {
465 (*vert)[i].fPos -= orthVec;
466 } else {
467 (*vert)[i].fPos += orthVec;
468 }
bsalomon@google.com706f6682012-10-23 14:53:55 +0000469 (*vert)[i].fLine.fA = orthVec.fX;
470 (*vert)[i].fLine.fB = orthVec.fY;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000471 (*vert)[i].fLine.fC = lineC;
472 }
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000473 if (NULL != toSrc) {
474 toSrc->mapPointsWithStride(&(*vert)->fPos,
475 sizeof(Vertex),
476 kVertsPerLineSeg);
477 }
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000478 } else {
479 // just make it degenerate and likely offscreen
480 (*vert)[0].fPos.set(SK_ScalarMax, SK_ScalarMax);
481 (*vert)[1].fPos.set(SK_ScalarMax, SK_ScalarMax);
482 (*vert)[2].fPos.set(SK_ScalarMax, SK_ScalarMax);
483 (*vert)[3].fPos.set(SK_ScalarMax, SK_ScalarMax);
484 }
485
486 *vert += kVertsPerLineSeg;
487}
488
489}
490
bsalomon@google.comb3729422012-03-07 19:13:28 +0000491bool GrAAHairLinePathRenderer::createGeom(
492 const SkPath& path,
bsalomon@google.comb3729422012-03-07 19:13:28 +0000493 GrDrawTarget* target,
bsalomon@google.comb3729422012-03-07 19:13:28 +0000494 int* lineCnt,
495 int* quadCnt,
496 GrDrawTarget::AutoReleaseGeometry* arg) {
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000497 const GrDrawState& drawState = target->getDrawState();
bsalomon@google.com8f9cbd62011-12-09 15:55:34 +0000498 int rtHeight = drawState.getRenderTarget()->height();
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000499
robertphillips@google.com7b112892012-07-31 15:18:21 +0000500 GrIRect devClipBounds;
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000501 target->getClip()->getConservativeBounds(drawState.getRenderTarget(),
robertphillips@google.com7b112892012-07-31 15:18:21 +0000502 &devClipBounds);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000503
504 GrVertexLayout layout = GrDrawTarget::kEdge_VertexLayoutBit;
bsalomon@google.comb9086a02012-11-01 18:02:54 +0000505 SkMatrix viewM = drawState.getViewMatrix();
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000506
bsalomon@google.com92669012011-09-27 19:10:05 +0000507 PREALLOC_PTARRAY(128) lines;
508 PREALLOC_PTARRAY(128) quads;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000509 IntArray qSubdivs;
bsalomon@google.com0f11e1a2012-10-08 14:48:36 +0000510 *quadCnt = generate_lines_and_quads(path, viewM, devClipBounds,
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000511 &lines, &quads, &qSubdivs);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000512
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000513 *lineCnt = lines.count() / 2;
514 int vertCnt = kVertsPerLineSeg * *lineCnt + kVertsPerQuad * *quadCnt;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000515
516 GrAssert(sizeof(Vertex) == GrDrawTarget::VertexSize(layout));
517
bsalomon@google.comb3729422012-03-07 19:13:28 +0000518 if (!arg->set(target, layout, vertCnt, 0)) {
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000519 return false;
520 }
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000521
bsalomon@google.comb3729422012-03-07 19:13:28 +0000522 Vertex* verts = reinterpret_cast<Vertex*>(arg->vertices());
523
bsalomon@google.comb9086a02012-11-01 18:02:54 +0000524 const SkMatrix* toDevice = NULL;
525 const SkMatrix* toSrc = NULL;
526 SkMatrix ivm;
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000527
528 if (viewM.hasPerspective()) {
529 if (viewM.invert(&ivm)) {
530 toDevice = &viewM;
531 toSrc = &ivm;
532 }
533 }
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000534
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000535 for (int i = 0; i < *lineCnt; ++i) {
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000536 add_line(&lines[2*i], rtHeight, toSrc, &verts);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000537 }
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000538
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000539 int unsubdivQuadCnt = quads.count() / 3;
540 for (int i = 0; i < unsubdivQuadCnt; ++i) {
541 GrAssert(qSubdivs[i] >= 0);
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000542 add_quads(&quads[3*i], qSubdivs[i], toDevice, toSrc, &verts);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000543 }
544
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000545 return true;
546}
547
robertphillips@google.com8a4fc402012-05-24 12:42:24 +0000548bool GrAAHairLinePathRenderer::canDrawPath(const SkPath& path,
549 GrPathFill fill,
550 const GrDrawTarget* target,
551 bool antiAlias) const {
bsalomon@google.com47059542012-06-06 20:51:20 +0000552 if (fill != kHairLine_GrPathFill || !antiAlias) {
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000553 return false;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000554 }
555
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000556 static const uint32_t gReqDerivMask = SkPath::kCubic_SegmentMask |
557 SkPath::kQuad_SegmentMask;
bsalomon@google.comf6601872012-08-28 21:11:35 +0000558 if (!target->getCaps().shaderDerivativeSupport() &&
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000559 (gReqDerivMask & path.getSegmentMasks())) {
560 return false;
561 }
562 return true;
563}
564
565bool GrAAHairLinePathRenderer::onDrawPath(const SkPath& path,
566 GrPathFill fill,
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000567 GrDrawTarget* target,
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000568 bool antiAlias) {
569
570 int lineCnt;
571 int quadCnt;
bsalomon@google.comb3729422012-03-07 19:13:28 +0000572 GrDrawTarget::AutoReleaseGeometry arg;
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000573 if (!this->createGeom(path,
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000574 target,
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000575 &lineCnt,
bsalomon@google.comb3729422012-03-07 19:13:28 +0000576 &quadCnt,
577 &arg)) {
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000578 return false;
579 }
580
bsalomon@google.coma8347462012-10-08 18:59:39 +0000581 GrDrawState::AutoDeviceCoordDraw adcd;
bsalomon@google.com873ea0c2012-03-30 15:55:32 +0000582 GrDrawState* drawState = target->drawState();
bsalomon@google.coma8347462012-10-08 18:59:39 +0000583 // createGeom transforms the geometry to device space when the matrix does not have
584 // perspective.
bsalomon@google.com8f9cbd62011-12-09 15:55:34 +0000585 if (!drawState->getViewMatrix().hasPerspective()) {
bsalomon@google.coma8347462012-10-08 18:59:39 +0000586 adcd.set(drawState);
587 if (!adcd.succeeded()) {
bsalomon@google.come3d32162012-07-20 13:37:06 +0000588 return false;
bsalomon@google.comdbeeac32011-09-12 14:59:34 +0000589 }
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000590 }
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000591
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000592 // TODO: See whether rendering lines as degenerate quads improves perf
593 // when we have a mix
bsalomon@google.coma8347462012-10-08 18:59:39 +0000594
595 GrDrawState::VertexEdgeType oldEdgeType = drawState->getVertexEdgeType();
596
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000597 target->setIndexSourceToBuffer(fLinesIndexBuffer);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000598 int lines = 0;
599 int nBufLines = fLinesIndexBuffer->maxQuads();
bsalomon@google.coma8347462012-10-08 18:59:39 +0000600 drawState->setVertexEdgeType(GrDrawState::kHairLine_EdgeType);
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000601 while (lines < lineCnt) {
602 int n = GrMin(lineCnt - lines, nBufLines);
bsalomon@google.com47059542012-06-06 20:51:20 +0000603 target->drawIndexed(kTriangles_GrPrimitiveType,
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000604 kVertsPerLineSeg*lines, // startV
605 0, // startI
606 kVertsPerLineSeg*n, // vCount
607 kIdxsPerLineSeg*n); // iCount
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000608 lines += n;
609 }
610
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000611 target->setIndexSourceToBuffer(fQuadsIndexBuffer);
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000612 int quads = 0;
bsalomon@google.coma8347462012-10-08 18:59:39 +0000613 drawState->setVertexEdgeType(GrDrawState::kHairQuad_EdgeType);
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000614 while (quads < quadCnt) {
615 int n = GrMin(quadCnt - quads, kNumQuadsInIdxBuffer);
bsalomon@google.com47059542012-06-06 20:51:20 +0000616 target->drawIndexed(kTriangles_GrPrimitiveType,
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000617 4 * lineCnt + kVertsPerQuad*quads, // startV
618 0, // startI
619 kVertsPerQuad*n, // vCount
620 kIdxsPerQuad*n); // iCount
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000621 quads += n;
622 }
bsalomon@google.coma8347462012-10-08 18:59:39 +0000623 drawState->setVertexEdgeType(oldEdgeType);
bsalomon@google.comc2099d22012-03-02 21:26:50 +0000624 return true;
bsalomon@google.comaeb21602011-08-30 18:13:44 +0000625}
626