OLD | NEW |
1 | 1 |
2 /* | 2 /* |
3 * Copyright 2012 Google Inc. | 3 * Copyright 2012 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 "GrAAConvexPathRenderer.h" | 9 #include "GrAAConvexPathRenderer.h" |
10 | 10 |
| 11 #include "GrAAConvexTessellator.h" |
11 #include "GrBatch.h" | 12 #include "GrBatch.h" |
12 #include "GrBatchTarget.h" | 13 #include "GrBatchTarget.h" |
13 #include "GrContext.h" | 14 #include "GrContext.h" |
| 15 #include "GrDefaultGeoProcFactory.h" |
14 #include "GrDrawTargetCaps.h" | 16 #include "GrDrawTargetCaps.h" |
15 #include "GrGeometryProcessor.h" | 17 #include "GrGeometryProcessor.h" |
16 #include "GrInvariantOutput.h" | 18 #include "GrInvariantOutput.h" |
17 #include "GrPathUtils.h" | 19 #include "GrPathUtils.h" |
18 #include "GrProcessor.h" | 20 #include "GrProcessor.h" |
19 #include "GrPipelineBuilder.h" | 21 #include "GrPipelineBuilder.h" |
20 #include "GrStrokeInfo.h" | 22 #include "GrStrokeInfo.h" |
21 #include "SkGeometry.h" | 23 #include "SkGeometry.h" |
22 #include "SkString.h" | 24 #include "SkString.h" |
23 #include "SkTraceEvent.h" | 25 #include "SkTraceEvent.h" |
(...skipping 672 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
696 bool GrAAConvexPathRenderer::canDrawPath(const GrDrawTarget* target, | 698 bool GrAAConvexPathRenderer::canDrawPath(const GrDrawTarget* target, |
697 const GrPipelineBuilder*, | 699 const GrPipelineBuilder*, |
698 const SkMatrix& viewMatrix, | 700 const SkMatrix& viewMatrix, |
699 const SkPath& path, | 701 const SkPath& path, |
700 const GrStrokeInfo& stroke, | 702 const GrStrokeInfo& stroke, |
701 bool antiAlias) const { | 703 bool antiAlias) const { |
702 return (target->caps()->shaderCaps()->shaderDerivativeSupport() && antiAlias
&& | 704 return (target->caps()->shaderCaps()->shaderDerivativeSupport() && antiAlias
&& |
703 stroke.isFillStyle() && !path.isInverseFillType() && path.isConvex()
); | 705 stroke.isFillStyle() && !path.isInverseFillType() && path.isConvex()
); |
704 } | 706 } |
705 | 707 |
| 708 // extract the result vertices and indices from the GrAAConvexTessellator |
| 709 static void extract_verts(const GrAAConvexTessellator& tess, |
| 710 void* vertices, |
| 711 size_t vertexStride, |
| 712 GrColor color, |
| 713 uint16_t* idxs, |
| 714 bool tweakAlphaForCoverage) { |
| 715 intptr_t verts = reinterpret_cast<intptr_t>(vertices); |
| 716 |
| 717 for (int i = 0; i < tess.numPts(); ++i) { |
| 718 *((SkPoint*)((intptr_t)verts + i * vertexStride)) = tess.point(i); |
| 719 } |
| 720 |
| 721 // Make 'verts' point to the colors |
| 722 verts += sizeof(SkPoint); |
| 723 for (int i = 0; i < tess.numPts(); ++i) { |
| 724 SkASSERT(tess.depth(i) >= -0.5f && tess.depth(i) <= 0.5f); |
| 725 if (tweakAlphaForCoverage) { |
| 726 SkASSERT(SkScalarRoundToInt(255.0f * (tess.depth(i) + 0.5f)) <= 255)
; |
| 727 unsigned scale = SkScalarRoundToInt(255.0f * (tess.depth(i) + 0.5f))
; |
| 728 GrColor scaledColor = (0xff == scale) ? color : SkAlphaMulQ(color, s
cale); |
| 729 *reinterpret_cast<GrColor*>(verts + i * vertexStride) = scaledColor; |
| 730 } else { |
| 731 *reinterpret_cast<GrColor*>(verts + i * vertexStride) = color; |
| 732 *reinterpret_cast<float*>(verts + i * vertexStride + sizeof(GrColor)
) = |
| 733 tess.depth(i
) + 0.5f; |
| 734 } |
| 735 } |
| 736 |
| 737 for (int i = 0; i < tess.numIndices(); ++i) { |
| 738 idxs[i] = tess.index(i); |
| 739 } |
| 740 } |
| 741 |
| 742 static const GrGeometryProcessor* create_fill_gp(bool tweakAlphaForCoverage, |
| 743 const SkMatrix& localMatrix) { |
| 744 uint32_t flags = GrDefaultGeoProcFactory::kColor_GPType; |
| 745 if (!tweakAlphaForCoverage) { |
| 746 flags |= GrDefaultGeoProcFactory::kCoverage_GPType; |
| 747 } |
| 748 |
| 749 return GrDefaultGeoProcFactory::Create(flags, GrColor_WHITE, SkMatrix::I(),
localMatrix, |
| 750 false, 0xff); |
| 751 } |
| 752 |
706 class AAConvexPathBatch : public GrBatch { | 753 class AAConvexPathBatch : public GrBatch { |
707 public: | 754 public: |
708 struct Geometry { | 755 struct Geometry { |
709 GrColor fColor; | 756 GrColor fColor; |
710 SkMatrix fViewMatrix; | 757 SkMatrix fViewMatrix; |
711 SkPath fPath; | 758 SkPath fPath; |
712 }; | 759 }; |
713 | 760 |
714 static GrBatch* Create(const Geometry& geometry) { | 761 static GrBatch* Create(const Geometry& geometry) { |
715 return SkNEW_ARGS(AAConvexPathBatch, (geometry)); | 762 return SkNEW_ARGS(AAConvexPathBatch, (geometry)); |
(...skipping 15 matching lines...) Expand all Loading... |
731 fGeoData[0].fColor = GrColor_ILLEGAL; | 778 fGeoData[0].fColor = GrColor_ILLEGAL; |
732 } else if (GrColor_ILLEGAL != init.fOverrideColor) { | 779 } else if (GrColor_ILLEGAL != init.fOverrideColor) { |
733 fGeoData[0].fColor = init.fOverrideColor; | 780 fGeoData[0].fColor = init.fOverrideColor; |
734 } | 781 } |
735 | 782 |
736 // setup batch properties | 783 // setup batch properties |
737 fBatch.fColorIgnored = init.fColorIgnored; | 784 fBatch.fColorIgnored = init.fColorIgnored; |
738 fBatch.fColor = fGeoData[0].fColor; | 785 fBatch.fColor = fGeoData[0].fColor; |
739 fBatch.fUsesLocalCoords = init.fUsesLocalCoords; | 786 fBatch.fUsesLocalCoords = init.fUsesLocalCoords; |
740 fBatch.fCoverageIgnored = init.fCoverageIgnored; | 787 fBatch.fCoverageIgnored = init.fCoverageIgnored; |
| 788 fBatch.fLinesOnly = SkPath::kLine_SegmentMask == fGeoData[0].fPath.getSe
gmentMasks(); |
| 789 fBatch.fCanTweakAlphaForCoverage = init.fCanTweakAlphaForCoverage; |
| 790 } |
| 791 |
| 792 void generateGeometryLinesOnly(GrBatchTarget* batchTarget, const GrPipeline*
pipeline) { |
| 793 bool canTweakAlphaForCoverage = this->canTweakAlphaForCoverage(); |
| 794 |
| 795 SkMatrix invert; |
| 796 if (this->usesLocalCoords() && !this->viewMatrix().invert(&invert)) { |
| 797 SkDebugf("Could not invert viewmatrix\n"); |
| 798 return; |
| 799 } |
| 800 |
| 801 // Setup GrGeometryProcessor |
| 802 SkAutoTUnref<const GrGeometryProcessor> gp( |
| 803 create_fill_gp(canTweakAlphaForC
overage, invert)); |
| 804 |
| 805 batchTarget->initDraw(gp, pipeline); |
| 806 |
| 807 // TODO remove this when batch is everywhere |
| 808 GrPipelineInfo init; |
| 809 init.fColorIgnored = fBatch.fColorIgnored; |
| 810 init.fOverrideColor = GrColor_ILLEGAL; |
| 811 init.fCoverageIgnored = fBatch.fCoverageIgnored; |
| 812 init.fUsesLocalCoords = this->usesLocalCoords(); |
| 813 gp->initBatchTracker(batchTarget->currentBatchTracker(), init); |
| 814 |
| 815 size_t vertexStride = gp->getVertexStride(); |
| 816 |
| 817 SkASSERT(canTweakAlphaForCoverage ? |
| 818 vertexStride == sizeof(GrDefaultGeoProcFactory::PositionColorAt
tr) : |
| 819 vertexStride == sizeof(GrDefaultGeoProcFactory::PositionColorCo
verageAttr)); |
| 820 |
| 821 GrAAConvexTessellator tess; |
| 822 |
| 823 int instanceCount = fGeoData.count(); |
| 824 |
| 825 for (int i = 0; i < instanceCount; i++) { |
| 826 tess.rewind(); |
| 827 |
| 828 Geometry& args = fGeoData[i]; |
| 829 |
| 830 if (!tess.tessellate(args.fViewMatrix, args.fPath)) { |
| 831 continue; |
| 832 } |
| 833 |
| 834 const GrVertexBuffer* vertexBuffer; |
| 835 int firstVertex; |
| 836 |
| 837 void* verts = batchTarget->makeVertSpace(vertexStride, tess.numPts()
, |
| 838 &vertexBuffer, &firstVertex
); |
| 839 if (!verts) { |
| 840 SkDebugf("Could not allocate vertices\n"); |
| 841 return; |
| 842 } |
| 843 |
| 844 const GrIndexBuffer* indexBuffer; |
| 845 int firstIndex; |
| 846 |
| 847 uint16_t* idxs = batchTarget->makeIndexSpace(tess.numIndices(), |
| 848 &indexBuffer, &firstInde
x); |
| 849 if (!idxs) { |
| 850 SkDebugf("Could not allocate indices\n"); |
| 851 return; |
| 852 } |
| 853 |
| 854 extract_verts(tess, verts, vertexStride, args.fColor, idxs, canTweak
AlphaForCoverage); |
| 855 |
| 856 GrVertices info; |
| 857 info.initIndexed(kTriangles_GrPrimitiveType, |
| 858 vertexBuffer, indexBuffer, |
| 859 firstVertex, firstIndex, |
| 860 tess.numPts(), tess.numIndices()); |
| 861 batchTarget->draw(info); |
| 862 } |
741 } | 863 } |
742 | 864 |
743 void generateGeometry(GrBatchTarget* batchTarget, const GrPipeline* pipeline
) override { | 865 void generateGeometry(GrBatchTarget* batchTarget, const GrPipeline* pipeline
) override { |
| 866 #ifndef SK_IGNORE_LINEONLY_AA_CONVEX_PATH_OPTS |
| 867 if (this->linesOnly()) { |
| 868 this->generateGeometryLinesOnly(batchTarget, pipeline); |
| 869 return; |
| 870 } |
| 871 #endif |
| 872 |
744 int instanceCount = fGeoData.count(); | 873 int instanceCount = fGeoData.count(); |
745 | 874 |
746 SkMatrix invert; | 875 SkMatrix invert; |
747 if (this->usesLocalCoords() && !this->viewMatrix().invert(&invert)) { | 876 if (this->usesLocalCoords() && !this->viewMatrix().invert(&invert)) { |
748 SkDebugf("Could not invert viewmatrix\n"); | 877 SkDebugf("Could not invert viewmatrix\n"); |
749 return; | 878 return; |
750 } | 879 } |
751 | 880 |
752 // Setup GrGeometryProcessor | 881 // Setup GrGeometryProcessor |
753 SkAutoTUnref<GrGeometryProcessor> quadProcessor(QuadEdgeEffect::Create(t
his->color(), | 882 SkAutoTUnref<GrGeometryProcessor> quadProcessor(QuadEdgeEffect::Create(t
his->color(), |
(...skipping 91 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
845 | 974 |
846 if (this->color() != that->color()) { | 975 if (this->color() != that->color()) { |
847 return false; | 976 return false; |
848 } | 977 } |
849 | 978 |
850 SkASSERT(this->usesLocalCoords() == that->usesLocalCoords()); | 979 SkASSERT(this->usesLocalCoords() == that->usesLocalCoords()); |
851 if (this->usesLocalCoords() && !this->viewMatrix().cheapEqualTo(that->vi
ewMatrix())) { | 980 if (this->usesLocalCoords() && !this->viewMatrix().cheapEqualTo(that->vi
ewMatrix())) { |
852 return false; | 981 return false; |
853 } | 982 } |
854 | 983 |
| 984 if (this->linesOnly() != that->linesOnly()) { |
| 985 return false; |
| 986 } |
| 987 |
| 988 // In the event of two batches, one who can tweak, one who cannot, we ju
st fall back to |
| 989 // not tweaking |
| 990 if (this->canTweakAlphaForCoverage() != that->canTweakAlphaForCoverage()
) { |
| 991 fBatch.fCanTweakAlphaForCoverage = false; |
| 992 } |
| 993 |
855 fGeoData.push_back_n(that->geoData()->count(), that->geoData()->begin())
; | 994 fGeoData.push_back_n(that->geoData()->count(), that->geoData()->begin())
; |
856 this->joinBounds(that->bounds()); | 995 this->joinBounds(that->bounds()); |
857 return true; | 996 return true; |
858 } | 997 } |
859 | 998 |
860 GrColor color() const { return fBatch.fColor; } | 999 GrColor color() const { return fBatch.fColor; } |
| 1000 bool linesOnly() const { return fBatch.fLinesOnly; } |
861 bool usesLocalCoords() const { return fBatch.fUsesLocalCoords; } | 1001 bool usesLocalCoords() const { return fBatch.fUsesLocalCoords; } |
| 1002 bool canTweakAlphaForCoverage() const { return fBatch.fCanTweakAlphaForCover
age; } |
862 const SkMatrix& viewMatrix() const { return fGeoData[0].fViewMatrix; } | 1003 const SkMatrix& viewMatrix() const { return fGeoData[0].fViewMatrix; } |
863 | 1004 |
864 struct BatchTracker { | 1005 struct BatchTracker { |
865 GrColor fColor; | 1006 GrColor fColor; |
866 bool fUsesLocalCoords; | 1007 bool fUsesLocalCoords; |
867 bool fColorIgnored; | 1008 bool fColorIgnored; |
868 bool fCoverageIgnored; | 1009 bool fCoverageIgnored; |
| 1010 bool fLinesOnly; |
| 1011 bool fCanTweakAlphaForCoverage; |
869 }; | 1012 }; |
870 | 1013 |
871 BatchTracker fBatch; | 1014 BatchTracker fBatch; |
872 SkSTArray<1, Geometry, true> fGeoData; | 1015 SkSTArray<1, Geometry, true> fGeoData; |
873 }; | 1016 }; |
874 | 1017 |
875 bool GrAAConvexPathRenderer::onDrawPath(GrDrawTarget* target, | 1018 bool GrAAConvexPathRenderer::onDrawPath(GrDrawTarget* target, |
876 GrPipelineBuilder* pipelineBuilder, | 1019 GrPipelineBuilder* pipelineBuilder, |
877 GrColor color, | 1020 GrColor color, |
878 const SkMatrix& vm, | 1021 const SkMatrix& vm, |
879 const SkPath& path, | 1022 const SkPath& path, |
880 const GrStrokeInfo&, | 1023 const GrStrokeInfo&, |
881 bool antiAlias) { | 1024 bool antiAlias) { |
882 if (path.isEmpty()) { | 1025 if (path.isEmpty()) { |
883 return true; | 1026 return true; |
884 } | 1027 } |
885 | 1028 |
886 AAConvexPathBatch::Geometry geometry; | 1029 AAConvexPathBatch::Geometry geometry; |
887 geometry.fColor = color; | 1030 geometry.fColor = color; |
888 geometry.fViewMatrix = vm; | 1031 geometry.fViewMatrix = vm; |
889 geometry.fPath = path; | 1032 geometry.fPath = path; |
890 | 1033 |
891 SkAutoTUnref<GrBatch> batch(AAConvexPathBatch::Create(geometry)); | 1034 SkAutoTUnref<GrBatch> batch(AAConvexPathBatch::Create(geometry)); |
892 target->drawBatch(pipelineBuilder, batch); | 1035 target->drawBatch(pipelineBuilder, batch); |
893 | 1036 |
894 return true; | 1037 return true; |
895 | 1038 |
896 } | 1039 } |
OLD | NEW |