| OLD | NEW |
| 1 | 1 |
| 2 /* | 2 /* |
| 3 * Copyright 2014 Google Inc. | 3 * Copyright 2014 Google Inc. |
| 4 * | 4 * |
| 5 * Use of this source code is governed by a BSD-style license that can be | 5 * Use of this source code is governed by a BSD-style license that can be |
| 6 * found in the LICENSE file. | 6 * found in the LICENSE file. |
| 7 */ | 7 */ |
| 8 | 8 |
| 9 #include "GrAADistanceFieldPathRenderer.h" | 9 #include "GrAADistanceFieldPathRenderer.h" |
| 10 | 10 |
| (...skipping 125 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 136 fStroke.setStrokeStyle(-1.0f); | 136 fStroke.setStrokeStyle(-1.0f); |
| 137 } | 137 } |
| 138 } | 138 } |
| 139 } | 139 } |
| 140 SkPath fPath; | 140 SkPath fPath; |
| 141 // The unique ID of the path involved in this draw. This may be differen
t than the ID | 141 // The unique ID of the path involved in this draw. This may be differen
t than the ID |
| 142 // in fPath since that path may have resulted from a SkStrokeRec::applyT
oPath call. | 142 // in fPath since that path may have resulted from a SkStrokeRec::applyT
oPath call. |
| 143 uint32_t fGenID; | 143 uint32_t fGenID; |
| 144 SkStrokeRec fStroke; | 144 SkStrokeRec fStroke; |
| 145 bool fAntiAlias; | 145 bool fAntiAlias; |
| 146 PathData* fPathData; | |
| 147 }; | 146 }; |
| 148 | 147 |
| 149 static GrDrawBatch* Create(const Geometry& geometry, GrColor color, const Sk
Matrix& viewMatrix, | 148 static GrDrawBatch* Create(const Geometry& geometry, GrColor color, const Sk
Matrix& viewMatrix, |
| 150 GrBatchAtlas* atlas, PathCache* pathCache, PathDa
taList* pathList) { | 149 GrBatchAtlas* atlas, PathCache* pathCache, PathDa
taList* pathList) { |
| 151 return new AADistanceFieldPathBatch(geometry, color, viewMatrix, atlas,
pathCache, | 150 return new AADistanceFieldPathBatch(geometry, color, viewMatrix, atlas,
pathCache, |
| 152 pathList); | 151 pathList); |
| 153 } | 152 } |
| 154 | 153 |
| 155 const char* name() const override { return "AADistanceFieldPathBatch"; } | 154 const char* name() const override { return "AADistanceFieldPathBatch"; } |
| 156 | 155 |
| (...skipping 19 matching lines...) Expand all Loading... |
| 176 fBatch.fCoverageIgnored = !overrides.readsCoverage(); | 175 fBatch.fCoverageIgnored = !overrides.readsCoverage(); |
| 177 } | 176 } |
| 178 | 177 |
| 179 struct FlushInfo { | 178 struct FlushInfo { |
| 180 SkAutoTUnref<const GrVertexBuffer> fVertexBuffer; | 179 SkAutoTUnref<const GrVertexBuffer> fVertexBuffer; |
| 181 SkAutoTUnref<const GrIndexBuffer> fIndexBuffer; | 180 SkAutoTUnref<const GrIndexBuffer> fIndexBuffer; |
| 182 int fVertexOffset; | 181 int fVertexOffset; |
| 183 int fInstancesToFlush; | 182 int fInstancesToFlush; |
| 184 }; | 183 }; |
| 185 | 184 |
| 186 void onPrepareDraws(Target* target) override { | 185 void onPrepareDraws(Target* target) const override { |
| 187 int instanceCount = fGeoData.count(); | 186 int instanceCount = fGeoData.count(); |
| 188 | 187 |
| 189 SkMatrix invert; | 188 SkMatrix invert; |
| 190 if (this->usesLocalCoords() && !this->viewMatrix().invert(&invert)) { | 189 if (this->usesLocalCoords() && !this->viewMatrix().invert(&invert)) { |
| 191 SkDebugf("Could not invert viewmatrix\n"); | 190 SkDebugf("Could not invert viewmatrix\n"); |
| 192 return; | 191 return; |
| 193 } | 192 } |
| 194 | 193 |
| 195 uint32_t flags = 0; | 194 uint32_t flags = 0; |
| 196 flags |= this->viewMatrix().isSimilarity() ? kSimilarity_DistanceFieldEf
fectFlag : 0; | 195 flags |= this->viewMatrix().isSimilarity() ? kSimilarity_DistanceFieldEf
fectFlag : 0; |
| (...skipping 25 matching lines...) Expand all Loading... |
| 222 &flushInfo.fVertexOffset); | 221 &flushInfo.fVertexOffset); |
| 223 flushInfo.fVertexBuffer.reset(SkRef(vertexBuffer)); | 222 flushInfo.fVertexBuffer.reset(SkRef(vertexBuffer)); |
| 224 flushInfo.fIndexBuffer.reset(target->resourceProvider()->refQuadIndexBuf
fer()); | 223 flushInfo.fIndexBuffer.reset(target->resourceProvider()->refQuadIndexBuf
fer()); |
| 225 if (!vertices || !flushInfo.fIndexBuffer) { | 224 if (!vertices || !flushInfo.fIndexBuffer) { |
| 226 SkDebugf("Could not allocate vertices\n"); | 225 SkDebugf("Could not allocate vertices\n"); |
| 227 return; | 226 return; |
| 228 } | 227 } |
| 229 | 228 |
| 230 flushInfo.fInstancesToFlush = 0; | 229 flushInfo.fInstancesToFlush = 0; |
| 231 for (int i = 0; i < instanceCount; i++) { | 230 for (int i = 0; i < instanceCount; i++) { |
| 232 Geometry& args = fGeoData[i]; | 231 const Geometry& args = fGeoData[i]; |
| 233 | 232 |
| 234 // get mip level | 233 // get mip level |
| 235 SkScalar maxScale = this->viewMatrix().getMaxScale(); | 234 SkScalar maxScale = this->viewMatrix().getMaxScale(); |
| 236 const SkRect& bounds = args.fPath.getBounds(); | 235 const SkRect& bounds = args.fPath.getBounds(); |
| 237 SkScalar maxDim = SkMaxScalar(bounds.width(), bounds.height()); | 236 SkScalar maxDim = SkMaxScalar(bounds.width(), bounds.height()); |
| 238 SkScalar size = maxScale * maxDim; | 237 SkScalar size = maxScale * maxDim; |
| 239 uint32_t desiredDimension; | 238 uint32_t desiredDimension; |
| 240 if (size <= kSmallMIP) { | 239 if (size <= kSmallMIP) { |
| 241 desiredDimension = kSmallMIP; | 240 desiredDimension = kSmallMIP; |
| 242 } else if (size <= kMediumMIP) { | 241 } else if (size <= kMediumMIP) { |
| 243 desiredDimension = kMediumMIP; | 242 desiredDimension = kMediumMIP; |
| 244 } else { | 243 } else { |
| 245 desiredDimension = kLargeMIP; | 244 desiredDimension = kLargeMIP; |
| 246 } | 245 } |
| 247 | 246 |
| 248 // check to see if path is cached | 247 // check to see if path is cached |
| 249 PathData::Key key(args.fGenID, desiredDimension, args.fStroke); | 248 PathData::Key key(args.fGenID, desiredDimension, args.fStroke); |
| 250 args.fPathData = fPathCache->find(key); | 249 PathData* pathData = fPathCache->find(key); |
| 251 if (nullptr == args.fPathData || !atlas->hasID(args.fPathData->fID))
{ | 250 if (nullptr == pathData || !atlas->hasID(pathData->fID)) { |
| 252 // Remove the stale cache entry | 251 // Remove the stale cache entry |
| 253 if (args.fPathData) { | 252 if (pathData) { |
| 254 fPathCache->remove(args.fPathData->fKey); | 253 fPathCache->remove(pathData->fKey); |
| 255 fPathList->remove(args.fPathData); | 254 fPathList->remove(pathData); |
| 256 delete args.fPathData; | 255 delete pathData; |
| 257 } | 256 } |
| 258 SkScalar scale = desiredDimension/maxDim; | 257 SkScalar scale = desiredDimension/maxDim; |
| 259 args.fPathData = new PathData; | 258 pathData = new PathData; |
| 260 if (!this->addPathToAtlas(target, | 259 if (!this->addPathToAtlas(target, |
| 261 dfProcessor, | 260 dfProcessor, |
| 262 this->pipeline(), | 261 this->pipeline(), |
| 263 &flushInfo, | 262 &flushInfo, |
| 264 atlas, | 263 atlas, |
| 265 args.fPathData, | 264 pathData, |
| 266 args.fPath, | 265 args.fPath, |
| 267 args.fGenID, | 266 args.fGenID, |
| 268 args.fStroke, | 267 args.fStroke, |
| 269 args.fAntiAlias, | 268 args.fAntiAlias, |
| 270 desiredDimension, | 269 desiredDimension, |
| 271 scale)) { | 270 scale)) { |
| 272 SkDebugf("Can't rasterize path\n"); | 271 SkDebugf("Can't rasterize path\n"); |
| 273 return; | 272 return; |
| 274 } | 273 } |
| 275 } | 274 } |
| 276 | 275 |
| 277 atlas->setLastUseToken(args.fPathData->fID, target->currentToken()); | 276 atlas->setLastUseToken(pathData->fID, target->currentToken()); |
| 278 | 277 |
| 279 // Now set vertices | 278 // Now set vertices |
| 280 intptr_t offset = reinterpret_cast<intptr_t>(vertices); | 279 intptr_t offset = reinterpret_cast<intptr_t>(vertices); |
| 281 offset += i * kVerticesPerQuad * vertexStride; | 280 offset += i * kVerticesPerQuad * vertexStride; |
| 282 SkPoint* positions = reinterpret_cast<SkPoint*>(offset); | 281 SkPoint* positions = reinterpret_cast<SkPoint*>(offset); |
| 283 this->writePathVertices(target, | 282 this->writePathVertices(target, |
| 284 atlas, | 283 atlas, |
| 285 this->pipeline(), | 284 this->pipeline(), |
| 286 dfProcessor, | 285 dfProcessor, |
| 287 positions, | 286 positions, |
| 288 vertexStride, | 287 vertexStride, |
| 289 this->viewMatrix(), | 288 this->viewMatrix(), |
| 290 args.fPath, | 289 args.fPath, |
| 291 args.fPathData); | 290 pathData); |
| 292 flushInfo.fInstancesToFlush++; | 291 flushInfo.fInstancesToFlush++; |
| 293 } | 292 } |
| 294 | 293 |
| 295 this->flush(target, &flushInfo); | 294 this->flush(target, &flushInfo); |
| 296 } | 295 } |
| 297 | 296 |
| 298 SkSTArray<1, Geometry, true>* geoData() { return &fGeoData; } | 297 SkSTArray<1, Geometry, true>* geoData() { return &fGeoData; } |
| 299 | 298 |
| 300 AADistanceFieldPathBatch(const Geometry& geometry, GrColor color, const SkMa
trix& viewMatrix, | 299 AADistanceFieldPathBatch(const Geometry& geometry, GrColor color, const SkMa
trix& viewMatrix, |
| 301 GrBatchAtlas* atlas, | 300 GrBatchAtlas* atlas, |
| 302 PathCache* pathCache, PathDataList* pathList) | 301 PathCache* pathCache, PathDataList* pathList) |
| 303 : INHERITED(ClassID()) { | 302 : INHERITED(ClassID()) { |
| 304 fBatch.fColor = color; | 303 fBatch.fColor = color; |
| 305 fBatch.fViewMatrix = viewMatrix; | 304 fBatch.fViewMatrix = viewMatrix; |
| 306 fGeoData.push_back(geometry); | 305 fGeoData.push_back(geometry); |
| 307 fGeoData.back().fPathData = nullptr; | |
| 308 | 306 |
| 309 fAtlas = atlas; | 307 fAtlas = atlas; |
| 310 fPathCache = pathCache; | 308 fPathCache = pathCache; |
| 311 fPathList = pathList; | 309 fPathList = pathList; |
| 312 | 310 |
| 313 // Compute bounds | 311 // Compute bounds |
| 314 fBounds = geometry.fPath.getBounds(); | 312 fBounds = geometry.fPath.getBounds(); |
| 315 viewMatrix.mapRect(&fBounds); | 313 viewMatrix.mapRect(&fBounds); |
| 316 } | 314 } |
| 317 | 315 |
| 318 bool addPathToAtlas(GrVertexBatch::Target* target, | 316 bool addPathToAtlas(GrVertexBatch::Target* target, |
| 319 const GrGeometryProcessor* dfProcessor, | 317 const GrGeometryProcessor* dfProcessor, |
| 320 const GrPipeline* pipeline, | 318 const GrPipeline* pipeline, |
| 321 FlushInfo* flushInfo, | 319 FlushInfo* flushInfo, |
| 322 GrBatchAtlas* atlas, | 320 GrBatchAtlas* atlas, |
| 323 PathData* pathData, | 321 PathData* pathData, |
| 324 const SkPath& path, | 322 const SkPath& path, |
| 325 uint32_t genID, | 323 uint32_t genID, |
| 326 const SkStrokeRec& stroke, | 324 const SkStrokeRec& stroke, |
| 327 bool antiAlias, | 325 bool antiAlias, |
| 328 uint32_t dimension, | 326 uint32_t dimension, |
| 329 SkScalar scale) { | 327 SkScalar scale) const { |
| 330 const SkRect& bounds = path.getBounds(); | 328 const SkRect& bounds = path.getBounds(); |
| 331 | 329 |
| 332 // generate bounding rect for bitmap draw | 330 // generate bounding rect for bitmap draw |
| 333 SkRect scaledBounds = bounds; | 331 SkRect scaledBounds = bounds; |
| 334 // scale to mip level size | 332 // scale to mip level size |
| 335 scaledBounds.fLeft *= scale; | 333 scaledBounds.fLeft *= scale; |
| 336 scaledBounds.fTop *= scale; | 334 scaledBounds.fTop *= scale; |
| 337 scaledBounds.fRight *= scale; | 335 scaledBounds.fRight *= scale; |
| 338 scaledBounds.fBottom *= scale; | 336 scaledBounds.fBottom *= scale; |
| 339 // move the origin to an integer boundary (gives better results) | 337 // move the origin to an integer boundary (gives better results) |
| (...skipping 96 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 436 } | 434 } |
| 437 | 435 |
| 438 void writePathVertices(GrDrawBatch::Target* target, | 436 void writePathVertices(GrDrawBatch::Target* target, |
| 439 GrBatchAtlas* atlas, | 437 GrBatchAtlas* atlas, |
| 440 const GrPipeline* pipeline, | 438 const GrPipeline* pipeline, |
| 441 const GrGeometryProcessor* gp, | 439 const GrGeometryProcessor* gp, |
| 442 SkPoint* positions, | 440 SkPoint* positions, |
| 443 size_t vertexStride, | 441 size_t vertexStride, |
| 444 const SkMatrix& viewMatrix, | 442 const SkMatrix& viewMatrix, |
| 445 const SkPath& path, | 443 const SkPath& path, |
| 446 const PathData* pathData) { | 444 const PathData* pathData) const { |
| 447 GrTexture* texture = atlas->getTexture(); | 445 GrTexture* texture = atlas->getTexture(); |
| 448 | 446 |
| 449 SkScalar dx = pathData->fBounds.fLeft; | 447 SkScalar dx = pathData->fBounds.fLeft; |
| 450 SkScalar dy = pathData->fBounds.fTop; | 448 SkScalar dy = pathData->fBounds.fTop; |
| 451 SkScalar width = pathData->fBounds.width(); | 449 SkScalar width = pathData->fBounds.width(); |
| 452 SkScalar height = pathData->fBounds.height(); | 450 SkScalar height = pathData->fBounds.height(); |
| 453 | 451 |
| 454 SkScalar invScale = 1.0f / pathData->fScale; | 452 SkScalar invScale = 1.0f / pathData->fScale; |
| 455 dx *= invScale; | 453 dx *= invScale; |
| 456 dy *= invScale; | 454 dy *= invScale; |
| (...skipping 12 matching lines...) Expand all Loading... |
| 469 | 467 |
| 470 // vertex texture coords | 468 // vertex texture coords |
| 471 SkPoint* textureCoords = positions + 1; | 469 SkPoint* textureCoords = positions + 1; |
| 472 textureCoords->setRectFan(SkFixedToFloat(texture->texturePriv().normaliz
eFixedX(tx)), | 470 textureCoords->setRectFan(SkFixedToFloat(texture->texturePriv().normaliz
eFixedX(tx)), |
| 473 SkFixedToFloat(texture->texturePriv().normaliz
eFixedY(ty)), | 471 SkFixedToFloat(texture->texturePriv().normaliz
eFixedY(ty)), |
| 474 SkFixedToFloat(texture->texturePriv().normaliz
eFixedX(tx + tw)), | 472 SkFixedToFloat(texture->texturePriv().normaliz
eFixedX(tx + tw)), |
| 475 SkFixedToFloat(texture->texturePriv().normaliz
eFixedY(ty + th)), | 473 SkFixedToFloat(texture->texturePriv().normaliz
eFixedY(ty + th)), |
| 476 vertexStride); | 474 vertexStride); |
| 477 } | 475 } |
| 478 | 476 |
| 479 void flush(GrVertexBatch::Target* target, FlushInfo* flushInfo) { | 477 void flush(GrVertexBatch::Target* target, FlushInfo* flushInfo) const { |
| 480 GrVertices vertices; | 478 GrVertices vertices; |
| 481 int maxInstancesPerDraw = flushInfo->fIndexBuffer->maxQuads(); | 479 int maxInstancesPerDraw = flushInfo->fIndexBuffer->maxQuads(); |
| 482 vertices.initInstanced(kTriangles_GrPrimitiveType, flushInfo->fVertexBuf
fer, | 480 vertices.initInstanced(kTriangles_GrPrimitiveType, flushInfo->fVertexBuf
fer, |
| 483 flushInfo->fIndexBuffer, flushInfo->fVertexOffset, kVerticesPerQuad, | 481 flushInfo->fIndexBuffer, flushInfo->fVertexOffset, kVerticesPerQuad, |
| 484 kIndicesPerQuad, flushInfo->fInstancesToFlush, maxInstancesPerDraw); | 482 kIndicesPerQuad, flushInfo->fInstancesToFlush, maxInstancesPerDraw); |
| 485 target->draw(vertices); | 483 target->draw(vertices); |
| 486 flushInfo->fVertexOffset += kVerticesPerQuad * flushInfo->fInstancesToFl
ush; | 484 flushInfo->fVertexOffset += kVerticesPerQuad * flushInfo->fInstancesToFl
ush; |
| 487 flushInfo->fInstancesToFlush = 0; | 485 flushInfo->fInstancesToFlush = 0; |
| 488 } | 486 } |
| 489 | 487 |
| (...skipping 145 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 635 geometry.fAntiAlias = random->nextBool(); | 633 geometry.fAntiAlias = random->nextBool(); |
| 636 geometry.fGenID = random->nextU(); | 634 geometry.fGenID = random->nextU(); |
| 637 | 635 |
| 638 return AADistanceFieldPathBatch::Create(geometry, color, viewMatrix, | 636 return AADistanceFieldPathBatch::Create(geometry, color, viewMatrix, |
| 639 gTestStruct.fAtlas, | 637 gTestStruct.fAtlas, |
| 640 &gTestStruct.fPathCache, | 638 &gTestStruct.fPathCache, |
| 641 &gTestStruct.fPathList); | 639 &gTestStruct.fPathList); |
| 642 } | 640 } |
| 643 | 641 |
| 644 #endif | 642 #endif |
| OLD | NEW |