Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1)

Side by Side Diff: src/gpu/GrAADistanceFieldPathRenderer.cpp

Issue 589103004: Add GrAASmallPathRenderer. (Closed) Base URL: https://skia.googlesource.com/skia.git@master
Patch Set: Rename SmallPathRenderer to DistanceFieldPathRenderer. Created 6 years, 2 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
1
2 /*
3 * Copyright 2014 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
robertphillips 2014/09/26 14:00:30 Is it kosher to include a gl header in gpu?
bsalomon 2014/09/26 17:32:51 We do this for processors all over the place, othe
jvanverth1 2014/10/03 17:28:21 Removed anyway, it's not needed.
9 #include "gl/builders/GrGLProgramBuilder.h"
10 #include "GrAADistanceFieldPathRenderer.h"
11
12 #include "GrAtlas.h"
13 #include "GrContext.h"
14 #include "GrDrawState.h"
15 #include "GrSWMaskHelper.h"
16 #include "effects/GrDistanceFieldTextureEffect.h"
17
18 #include "SkDistanceFieldGen.h"
19 #include "SkRTConf.h"
20
21 #define GR_ATLAS_TEXTURE_WIDTH 1024
bsalomon 2014/09/26 17:32:51 Let's drop the GR_ prefix on these macros. They do
jvanverth1 2014/10/03 17:28:21 Done.
22 #define GR_ATLAS_TEXTURE_HEIGHT 1024
23
24 #define GR_PLOT_WIDTH 256
25 #define GR_PLOT_HEIGHT 256
26
27 #define GR_NUM_PLOTS_X (GR_ATLAS_TEXTURE_WIDTH / GR_PLOT_WIDTH)
28 #define GR_NUM_PLOTS_Y (GR_ATLAS_TEXTURE_HEIGHT / GR_PLOT_HEIGHT)
29
30 SK_CONF_DECLARE(bool, c_DumpPathCache, "gpu.dumpPathCache", false,
31 "Dump the contents of the path cache before every purge.");
32
33 ////////////////////////////////////////////////////////////////////////////////
34 GrAADistanceFieldPathRenderer::~GrAADistanceFieldPathRenderer() {
35 PathDataList::Iter iter;
36 iter.init(fPathList, PathDataList::Iter::kHead_IterStart);
37 PathData* pathData;
38 while ((pathData = iter.get())) {
39 iter.next();
40 fPathList.remove(pathData);
robertphillips 2014/09/26 14:00:29 SkDELETE
jvanverth1 2014/10/03 17:28:23 Done.
41 delete pathData;
42 }
43
robertphillips 2014/09/26 14:00:30 SkDELETE
jvanverth1 2014/10/03 17:28:23 Done.
44 delete fAtlas;
45 }
46
47 ////////////////////////////////////////////////////////////////////////////////
48 bool GrAADistanceFieldPathRenderer::canDrawPath(const SkPath& path,
robertphillips 2014/09/26 14:00:29 make these line up?
jvanverth1 2014/10/03 17:28:22 Done.
49 const SkStrokeRec& stroke,
50 const GrDrawTarget* target,
51 bool antiAlias) const {
52 // TODO: Support inverse fill
bsalomon 2014/09/26 17:32:51 You might want to check out the GrGLPath stuff...
jvanverth1 2014/10/03 17:28:22 Acknowledged.
53 // TODO: Support strokes
54 if (!target->caps()->shaderDerivativeSupport() || !antiAlias || path.isInver seFillType()
55 || SkStrokeRec::kFill_Style != stroke.getStyle()) {
56 return false;
57 }
58
robertphillips 2014/09/26 14:00:30 3x ?
jvanverth1 2014/10/03 17:28:21 Done.
59 // currently don't support scaling more than 4x
egdaniel 2014/09/26 13:30:25 But check is on for max scale of 3?
jvanverth1 2014/10/03 17:28:21 Done.
60 const GrDrawState& drawState = target->getDrawState();
61 const SkMatrix& vm = drawState.getViewMatrix();
62 if (vm.getMaxScale() > 3.0f) {
63 return false;
64 }
65
66 // only support paths smaller than 64 x 64
67 const SkRect& bounds = path.getBounds();
68 return bounds.width() < 64.f && bounds.height() < 64.f;
69 }
70
71
72 //*** not sure if this is needed
bsalomon 2014/09/26 17:32:51 I think it's not, but we should have a GM that cli
jvanverth1 2014/10/03 17:28:22 I think it is needed -- GrSoftwarePathRenderer has
73 GrPathRenderer::StencilSupport GrAADistanceFieldPathRenderer::onGetStencilSuppor t(
74 const Sk Path&,
75 const Sk StrokeRec&,
76 const Gr DrawTarget*) const {
77 return GrPathRenderer::kNoSupport_StencilSupport;
78 }
79
80 ////////////////////////////////////////////////////////////////////////////////
81
82 // position + texture coord
83 extern const GrVertexAttrib gSDFPathVertexAttribs[] = {
84 { kVec2f_GrVertexAttribType, 0, kPosition_GrVertexAttribBinding },
85 { kVec2f_GrVertexAttribType, sizeof(SkPoint), kGeometryProcessor_GrVertexAtt ribBinding }
86 };
87 static const size_t kSDFPathVASize = 2 * sizeof(SkPoint);
88
89 bool GrAADistanceFieldPathRenderer::onDrawPath(const SkPath& path,
robertphillips 2014/09/26 14:00:29 line these up?
jvanverth1 2014/10/03 17:28:22 Done.
90 const SkStrokeRec& stroke,
91 GrDrawTarget* target,
92 bool antiAlias) {
93 if (path.isEmpty()) {
bsalomon 2014/09/26 17:32:51 maybe a comment that we already bailed on inverse
jvanverth1 2014/10/03 17:28:21 Done.
94 return true;
95 }
96
97 if (NULL == fContext) {
bsalomon 2014/09/26 17:32:51 Do we need this?
jvanverth1 2014/10/03 17:28:22 Done.
98 return false;
99 }
100
101 // check to see if path is cached
102 // TODO: handle stroked vs. filled version of same path
103 PathData* pathData = fPathCache.find(path.getGenerationID());
104 if (NULL == pathData) {
robertphillips 2014/09/26 14:00:29 this-> ?
jvanverth1 2014/10/03 17:28:21 Done.
105 pathData = addPathToAtlas(path, stroke, antiAlias);
106 if (NULL == pathData) {
107 return false;
108 }
109 }
110
111 // use signed distance field to render
robertphillips 2014/09/26 14:00:29 this-> ?
jvanverth1 2014/10/03 17:28:22 Done.
112 return internalDrawPath(path, pathData, target);
113 }
114
robertphillips 2014/09/26 14:00:30 // This is the factor by which we scale the path p
jvanverth1 2014/10/03 17:28:21 Done.
115 const SkScalar kScaleFactor = 2.0f;
116 const SkScalar kAntiAliasPad = 1.0f;
117
118 GrAADistanceFieldPathRenderer::PathData* GrAADistanceFieldPathRenderer::addPathT oAtlas(
119 const Sk Path& path,
120 const Sk StrokeRec& stroke,
121 bool ant iAlias) {
122 // generate distance field and add to atlas
123 if (NULL == fAtlas) {
124 SkISize textureSize = SkISize::Make(GR_ATLAS_TEXTURE_WIDTH,
125 GR_ATLAS_TEXTURE_HEIGHT);
126 fAtlas = SkNEW_ARGS(GrAtlas, (fContext->getGpu(), kAlpha_8_GrPixelConfig ,
127 kNone_GrTextureFlags, textureSize,
128 GR_NUM_PLOTS_X, GR_NUM_PLOTS_Y, false));
129 if (NULL == fAtlas) {
130 return NULL;
131 }
132 }
133
134 const SkRect& bounds = path.getBounds();
135
136 // generate bounding rect for bitmap draw
137 SkRect scaledBounds = bounds;
138 // scale up by 2x
139 scaledBounds.fLeft *= kScaleFactor;
140 scaledBounds.fTop *= kScaleFactor;
141 scaledBounds.fRight *= kScaleFactor;
142 scaledBounds.fBottom *= kScaleFactor;
143 // move the origin to an integer boundary (gives better results)
144 SkScalar dx = SkScalarFraction(scaledBounds.fLeft);
145 SkScalar dy = SkScalarFraction(scaledBounds.fTop);
146 scaledBounds.offset(-dx, -dy);
147 // get integer boundary
148 SkIRect devPathBounds;
149 scaledBounds.roundOut(&devPathBounds);
150 // pad by 1 to allow room for antialiasing
robertphillips 2014/09/26 14:00:30 Not CeilToInt ?
jvanverth1 2014/10/03 17:28:21 Done.
151 devPathBounds.outset(SkScalarFloorToInt(kAntiAliasPad), SkScalarFloorToInt(k AntiAliasPad));
152 // move origin to upper left corner
153 devPathBounds.offsetTo(0,0);
154
155 // draw path to bitmap
156 SkMatrix drawMatrix;
157 drawMatrix.setTranslate(-bounds.left(), -bounds.top());
158 drawMatrix.postScale(kScaleFactor, kScaleFactor);
159 drawMatrix.postTranslate(kAntiAliasPad, kAntiAliasPad);
160 GrSWMaskHelper helper(fContext);
161
162 if (!helper.init(devPathBounds, &drawMatrix)) {
163 return NULL;
164 }
165 helper.draw(path, stroke, SkRegion::kReplace_Op, antiAlias, 0xFF);
166
167 // generate signed distance field
168 devPathBounds.outset(SK_DistanceFieldPad, SK_DistanceFieldPad);
169 int width = devPathBounds.width();
170 int height = devPathBounds.height();
171 SkAutoSMalloc<1024> dfStorage(width*height*sizeof(unsigned char));
172 helper.toSDF(dfStorage.get());
173
174 // add to atlas
175 SkIPoint16 atlasLocation;
176 GrPlot* plot = fAtlas->addToAtlas(&fPlotUsage, width, height, dfStorage.get( ),
177 &atlasLocation);
178
179 // if atlas full
180 if (NULL == plot) {
robertphillips 2014/09/26 14:00:29 this-> ?
jvanverth1 2014/10/03 17:28:21 Done.
181 if (freeUnusedPlot()) {
182 plot = fAtlas->addToAtlas(&fPlotUsage, width, height, dfStorage.get( ),
183 &atlasLocation);
184 if (plot) {
185 goto HAS_ATLAS;
186 }
187 }
188
189 if (c_DumpPathCache) {
190 #ifdef SK_DEVELOPER
191 GrTexture* texture = fAtlas->getTexture();
192 texture->savePixels("pathcache.png");
193 #endif
194 }
195
196 // before we purge the cache, we must flush any accumulated draws
197 fContext->flush();
198
robertphillips 2014/09/26 14:00:29 this-> ?
jvanverth1 2014/10/03 17:28:21 Done.
199 if (freeUnusedPlot()) {
200 plot = fAtlas->addToAtlas(&fPlotUsage, width, height, dfStorage.get( ),
201 &atlasLocation);
202 if (plot) {
203 goto HAS_ATLAS;
204 }
205 }
206
207 return NULL;
208 }
209
210 HAS_ATLAS:
robertphillips 2014/09/26 14:00:29 Guard this ?!
jvanverth1 2014/10/03 17:28:22 Whoops, removed.
211 GrTexture* texture = fAtlas->getTexture();
212 texture->savePixels("pathcache.png");
213
214 // add to cache
robertphillips 2014/09/26 14:00:29 SkNEW ?
jvanverth1 2014/10/03 17:28:22 Done.
215 PathData* pathData = new PathData();
216 pathData->fGenID = path.getGenerationID();
217 pathData->fPlot = plot;
218 // change the scaled rect to match the size of the inset distance field
219 scaledBounds.fRight = scaledBounds.fLeft +
220 SkIntToScalar(devPathBounds.width() - 2*SK_DistanceFieldInset);
221 scaledBounds.fBottom = scaledBounds.fTop +
222 SkIntToScalar(devPathBounds.height() - 2*SK_DistanceFieldInset);
223 // shift the origin to the correct place relative to the distance field
224 // need to also restore the fractional translation
225 scaledBounds.offset(-SkIntToScalar(SK_DistanceFieldInset) - kAntiAliasPad + dx,
226 -SkIntToScalar(SK_DistanceFieldInset) - kAntiAliasPad + dy);
227 pathData->fBounds = scaledBounds;
228 // origin we render from is inset from distance field edge
229 atlasLocation.fX += SK_DistanceFieldInset;
230 atlasLocation.fY += SK_DistanceFieldInset;
231 pathData->fAtlasLocation = atlasLocation;
232
233 fPathCache.add(pathData);
234 fPathList.addToTail(pathData);
235
236 return pathData;
237 }
238
239 bool GrAADistanceFieldPathRenderer::freeUnusedPlot() {
240 // find an unused plot
robertphillips 2014/09/26 14:00:30 I don't think this fAtlas alias buys us much.
jvanverth1 2014/10/03 17:28:21 Done.
241 GrAtlas* atlas = fAtlas;
242 GrPlot* plot = atlas->getUnusedPlot();
243 if (NULL == plot) {
244 return false;
245 }
246 plot->resetRects();
247
248 // remove any paths that use this plot
249 PathDataList::Iter iter;
250 iter.init(fPathList, PathDataList::Iter::kHead_IterStart);
251 PathData* pathData;
252 while ((pathData = iter.get())) {
253 iter.next();
254 if (plot == pathData->fPlot) {
255 fPathCache.remove(pathData->fGenID);
256 fPathList.remove(pathData);
robertphillips 2014/09/26 14:00:29 SkDELETE ?
jvanverth1 2014/10/03 17:28:23 Done.
257 delete pathData;
258 }
259 }
260
261 // tell the atlas to free the plot
262 GrAtlas::RemovePlot(&fPlotUsage, plot);
263
264 return true;
265 }
266
267 bool GrAADistanceFieldPathRenderer::internalDrawPath(const SkPath& path,
robertphillips 2014/09/26 14:00:29 line these up ?
jvanverth1 2014/10/03 17:28:21 Done.
268 const PathData* pathData,
269 GrDrawTarget* target) {
270
271 GrTexture* texture = fAtlas->getTexture();
272 GrDrawState* drawState = target->drawState();
273
274 SkASSERT(pathData->fPlot);
275 GrDrawTarget::DrawToken drawToken = target->getCurrentDrawToken();
276 pathData->fPlot->setDrawToken(drawToken);
277
278 // make me some vertices
279 drawState->setVertexAttribs<gSDFPathVertexAttribs>(SK_ARRAY_COUNT(gSDFPathVe rtexAttribs),
280 kSDFPathVASize);
281 void* vertices = NULL;
282 void* indices = NULL;
robertphillips 2014/09/26 14:00:28 Will this fit on one line ?
jvanverth1 2014/10/03 17:28:21 Done.
283 bool success = target->reserveVertexAndIndexSpace(4,
284 6,
285 &vertices,
286 &indices);
287 GrAlwaysAssert(success);
288
289 SkScalar dx = pathData->fBounds.fLeft;
290 SkScalar dy = pathData->fBounds.fTop;
291 SkScalar width = pathData->fBounds.width();
292 SkScalar height = pathData->fBounds.height();
293
robertphillips 2014/09/26 14:00:29 invScale ?
jvanverth1 2014/10/03 17:28:22 Done.
294 SkScalar scale = 1.0f/kScaleFactor;
295 dx *= scale;
296 dy *= scale;
297 width *= scale;
298 height *= scale;
299
300 SkFixed tx = SkIntToFixed(pathData->fAtlasLocation.fX);
301 SkFixed ty = SkIntToFixed(pathData->fAtlasLocation.fY);
302 SkFixed tw = SkScalarToFixed(pathData->fBounds.width());
303 SkFixed th = SkScalarToFixed(pathData->fBounds.height());
304
robertphillips 2014/09/26 14:00:29 SkRect::MakeWH(dx, dy, width, height); ?
jvanverth1 2014/10/03 17:28:22 Done.
305 SkRect r;
306 r.fLeft = dx;
307 r.fTop = dy;
308 r.fRight = dx + width;
309 r.fBottom = dy + height;
310
311 size_t vertSize = 2 * sizeof(SkPoint);
312
313 SkPoint* positions = reinterpret_cast<SkPoint*>(vertices);
314 positions->setRectFan(r.left(), r.top(), r.right(), r.bottom(), vertSize);
315
316 // The texture coords are last in both the with and without color vertex lay outs.
317 SkPoint* textureCoords = reinterpret_cast<SkPoint*>(
robertphillips 2014/09/26 14:00:29 overlength.
jvanverth1 2014/10/03 17:28:22 Done.
318 reinterpret_cast<intptr_ t>(positions)+vertSize - sizeof(SkPoint));
egdaniel 2014/09/26 13:30:25 fix line wrap
jvanverth1 2014/10/03 17:28:22 Done.
319 textureCoords->setRectFan(SkFixedToFloat(texture->normalizeFixedX(tx)),
320 SkFixedToFloat(texture->normalizeFixedY(ty)),
321 SkFixedToFloat(texture->normalizeFixedX(tx + tw)),
322 SkFixedToFloat(texture->normalizeFixedY(ty + th)),
323 vertSize);
324
325 uint16_t* indexPtr = reinterpret_cast<uint16_t*>(indices);
326 *indexPtr++ = 0;
327 *indexPtr++ = 1;
328 *indexPtr++ = 2;
329 *indexPtr++ = 0;
330 *indexPtr++ = 2;
331 *indexPtr++ = 3;
332
333 // set up any flags
334 uint32_t flags = 0;
335 const SkMatrix& vm = drawState->getViewMatrix();
336 flags |= vm.isSimilarity() ? kSimilarity_DistanceFieldEffectFlag : 0;
337
338 GrTextureParams params(SkShader::kRepeat_TileMode, GrTextureParams::kBilerp_ FilterMode);
339 drawState->setGeometryProcessor(GrDistanceFieldNoGammaTextureEffect::Create( texture,
340 params,
341 flags))->unref();
342
robertphillips 2014/09/26 14:00:29 one line ?
jvanverth1 2014/10/03 17:28:22 Done.
343 target->drawIndexedInstances(kTriangles_GrPrimitiveType,
344 1,
345 4, 6, &r);
346 target->resetVertexSource();
347 target->resetIndexSource();
348
349 return true;
350 }
351
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698