blob: 7a41cacc137c081c6d7d6e20ab2b864bc6eda392 [file] [log] [blame]
joshualitta751c972015-11-20 13:37:32 -08001/*
2 * Copyright 2015 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "src/gpu/ops/GrAtlasTextOp.h"
Robert Phillipse4fda6c2018-02-21 12:10:41 -05009
Mike Kleinc0bd9f92019-04-23 12:05:21 -050010#include "include/core/SkPoint3.h"
11#include "include/private/GrRecordingContext.h"
12#include "src/core/SkMathPriv.h"
13#include "src/core/SkMatrixPriv.h"
14#include "src/core/SkStrikeCache.h"
15#include "src/gpu/GrCaps.h"
16#include "src/gpu/GrMemoryPool.h"
17#include "src/gpu/GrOpFlushState.h"
18#include "src/gpu/GrRecordingContextPriv.h"
19#include "src/gpu/GrResourceProvider.h"
20#include "src/gpu/effects/GrBitmapTextGeoProc.h"
21#include "src/gpu/effects/GrDistanceFieldGeoProc.h"
Robert Phillips3968fcb2019-12-05 16:40:31 -050022#include "src/gpu/ops/GrSimpleMeshDrawOpHelper.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050023#include "src/gpu/text/GrAtlasManager.h"
24#include "src/gpu/text/GrStrikeCache.h"
joshualitta751c972015-11-20 13:37:32 -080025
joshualitt60ce86d2015-11-23 13:08:22 -080026///////////////////////////////////////////////////////////////////////////////////////////////////
27
Robert Phillipsb97da532019-02-12 15:24:12 -050028std::unique_ptr<GrAtlasTextOp> GrAtlasTextOp::MakeBitmap(GrRecordingContext* context,
Robert Phillips7c525e62018-06-12 10:11:12 -040029 GrPaint&& paint,
30 GrMaskFormat maskFormat,
31 int glyphCount,
32 bool needsTransform) {
Robert Phillips9da87e02019-02-04 13:26:26 -050033 GrOpMemoryPool* pool = context->priv().opMemoryPool();
Robert Phillipsc994a932018-06-19 13:09:54 -040034
35 std::unique_ptr<GrAtlasTextOp> op = pool->allocate<GrAtlasTextOp>(std::move(paint));
Robert Phillips7c525e62018-06-12 10:11:12 -040036
37 switch (maskFormat) {
38 case kA8_GrMaskFormat:
39 op->fMaskType = kGrayscaleCoverageMask_MaskType;
40 break;
41 case kA565_GrMaskFormat:
42 op->fMaskType = kLCDCoverageMask_MaskType;
43 break;
44 case kARGB_GrMaskFormat:
45 op->fMaskType = kColorBitmapMask_MaskType;
46 break;
47 }
48 op->fNumGlyphs = glyphCount;
49 op->fGeoCount = 1;
50 op->fLuminanceColor = 0;
51 op->fNeedsGlyphTransform = needsTransform;
52 return op;
53 }
54
55std::unique_ptr<GrAtlasTextOp> GrAtlasTextOp::MakeDistanceField(
Robert Phillipsb97da532019-02-12 15:24:12 -050056 GrRecordingContext* context,
Robert Phillips7c525e62018-06-12 10:11:12 -040057 GrPaint&& paint,
58 int glyphCount,
59 const GrDistanceFieldAdjustTable* distanceAdjustTable,
60 bool useGammaCorrectDistanceTable,
61 SkColor luminanceColor,
62 const SkSurfaceProps& props,
63 bool isAntiAliased,
64 bool useLCD) {
Robert Phillips9da87e02019-02-04 13:26:26 -050065 GrOpMemoryPool* pool = context->priv().opMemoryPool();
Robert Phillipsc994a932018-06-19 13:09:54 -040066
67 std::unique_ptr<GrAtlasTextOp> op = pool->allocate<GrAtlasTextOp>(std::move(paint));
Robert Phillips7c525e62018-06-12 10:11:12 -040068
69 bool isBGR = SkPixelGeometryIsBGR(props.pixelGeometry());
70 bool isLCD = useLCD && SkPixelGeometryIsH(props.pixelGeometry());
71 op->fMaskType = !isAntiAliased ? kAliasedDistanceField_MaskType
72 : isLCD ? (isBGR ? kLCDBGRDistanceField_MaskType
73 : kLCDDistanceField_MaskType)
74 : kGrayscaleDistanceField_MaskType;
75 op->fDistanceAdjustTable.reset(SkRef(distanceAdjustTable));
76 op->fUseGammaCorrectDistanceTable = useGammaCorrectDistanceTable;
77 op->fLuminanceColor = luminanceColor;
78 op->fNumGlyphs = glyphCount;
79 op->fGeoCount = 1;
80 return op;
81 }
82
joshualitta751c972015-11-20 13:37:32 -080083static const int kDistanceAdjustLumShift = 5;
84
Brian Salomon5c6ac642017-12-19 11:09:32 -050085void GrAtlasTextOp::init() {
86 const Geometry& geo = fGeoData[0];
Brian Salomon5c6ac642017-12-19 11:09:32 -050087 if (this->usesDistanceFields()) {
88 bool isLCD = this->isLCD();
89
Herb Derby1c5be7b2019-12-13 12:03:06 -050090 const SkMatrix& drawMatrix = geo.fDrawMatrix;
Brian Salomon5c6ac642017-12-19 11:09:32 -050091
Herb Derby1c5be7b2019-12-13 12:03:06 -050092 fDFGPFlags = drawMatrix.isSimilarity() ? kSimilarity_DistanceFieldEffectFlag : 0;
93 fDFGPFlags |= drawMatrix.isScaleTranslate() ? kScaleOnly_DistanceFieldEffectFlag : 0;
94 fDFGPFlags |= drawMatrix.hasPerspective() ? kPerspective_DistanceFieldEffectFlag : 0;
Brian Salomon5c6ac642017-12-19 11:09:32 -050095 fDFGPFlags |= fUseGammaCorrectDistanceTable ? kGammaCorrect_DistanceFieldEffectFlag : 0;
96 fDFGPFlags |= (kAliasedDistanceField_MaskType == fMaskType)
97 ? kAliased_DistanceFieldEffectFlag
98 : 0;
99
100 if (isLCD) {
101 fDFGPFlags |= kUseLCD_DistanceFieldEffectFlag;
102 fDFGPFlags |=
103 (kLCDBGRDistanceField_MaskType == fMaskType) ? kBGR_DistanceFieldEffectFlag : 0;
104 }
Jim Van Verthb515ae72018-05-23 16:44:55 -0400105
106 fNeedsGlyphTransform = true;
Brian Salomon5c6ac642017-12-19 11:09:32 -0500107 }
Jim Van Verth70276912018-06-01 13:46:46 -0400108
109 SkRect bounds;
Herb Derby5bf5b042019-12-12 16:37:03 -0500110 geo.fBlob->computeSubRunBounds(
111 &bounds, *geo.fSubRunPtr, geo.fDrawMatrix, geo.fDrawOrigin, fNeedsGlyphTransform);
Jim Van Verth70276912018-06-01 13:46:46 -0400112 // We don't have tight bounds on the glyph paths in device space. For the purposes of bounds
113 // we treat this as a set of non-AA rects rendered with a texture.
Greg Daniel5faf4742019-10-01 15:14:44 -0400114 this->setBounds(bounds, HasAABloat::kNo, IsHairline::kNo);
Brian Salomon5c6ac642017-12-19 11:09:32 -0500115}
116
Chris Dalton1706cbf2019-05-21 19:35:29 -0600117void GrAtlasTextOp::visitProxies(const VisitProxyFunc& func) const {
Robert Phillipse4fda6c2018-02-21 12:10:41 -0500118 fProcessors.visitProxies(func);
Robert Phillipse4fda6c2018-02-21 12:10:41 -0500119}
120
Brian Osman9a390ac2018-11-12 09:47:48 -0500121#ifdef SK_DEBUG
Brian Salomon344ec422016-12-15 10:58:41 -0500122SkString GrAtlasTextOp::dumpInfo() const {
joshualitta751c972015-11-20 13:37:32 -0800123 SkString str;
124
125 for (int i = 0; i < fGeoCount; ++i) {
Herb Derby1b8dcd12019-11-15 15:21:15 -0500126 str.appendf("%d: Color: 0x%08x Trans: %.2f,%.2f\n",
joshualitta751c972015-11-20 13:37:32 -0800127 i,
Brian Osmancf860852018-10-31 14:04:39 -0400128 fGeoData[i].fColor.toBytes_RGBA(),
Herb Derby5bf5b042019-12-12 16:37:03 -0500129 fGeoData[i].fDrawOrigin.x(),
130 fGeoData[i].fDrawOrigin.y());
joshualitta751c972015-11-20 13:37:32 -0800131 }
132
Brian Salomon44acb5b2017-07-18 19:59:24 -0400133 str += fProcessors.dumpProcessors();
134 str += INHERITED::dumpInfo();
joshualitta751c972015-11-20 13:37:32 -0800135 return str;
136}
Brian Osman9a390ac2018-11-12 09:47:48 -0500137#endif
joshualitta751c972015-11-20 13:37:32 -0800138
Brian Salomon44acb5b2017-07-18 19:59:24 -0400139GrDrawOp::FixedFunctionFlags GrAtlasTextOp::fixedFunctionFlags() const {
140 return FixedFunctionFlags::kNone;
141}
142
Chris Dalton6ce447a2019-06-23 18:07:38 -0600143GrProcessorSet::Analysis GrAtlasTextOp::finalize(
144 const GrCaps& caps, const GrAppliedClip* clip, bool hasMixedSampledCoverage,
145 GrClampType clampType) {
Brian Salomon44acb5b2017-07-18 19:59:24 -0400146 GrProcessorAnalysisCoverage coverage;
147 GrProcessorAnalysisColor color;
joshualitta751c972015-11-20 13:37:32 -0800148 if (kColorBitmapMask_MaskType == fMaskType) {
Brian Salomon44acb5b2017-07-18 19:59:24 -0400149 color.setToUnknown();
joshualitta751c972015-11-20 13:37:32 -0800150 } else {
Brian Osman09068252018-01-03 09:57:29 -0500151 color.setToConstant(this->color());
joshualitta751c972015-11-20 13:37:32 -0800152 }
joshualitta751c972015-11-20 13:37:32 -0800153 switch (fMaskType) {
joshualitta751c972015-11-20 13:37:32 -0800154 case kGrayscaleCoverageMask_MaskType:
Jim Van Verth90e89b32017-07-06 16:36:55 -0400155 case kAliasedDistanceField_MaskType:
156 case kGrayscaleDistanceField_MaskType:
Brian Salomon44acb5b2017-07-18 19:59:24 -0400157 coverage = GrProcessorAnalysisCoverage::kSingleChannel;
joshualitta751c972015-11-20 13:37:32 -0800158 break;
159 case kLCDCoverageMask_MaskType:
160 case kLCDDistanceField_MaskType:
Jim Van Verth90e89b32017-07-06 16:36:55 -0400161 case kLCDBGRDistanceField_MaskType:
Brian Salomon44acb5b2017-07-18 19:59:24 -0400162 coverage = GrProcessorAnalysisCoverage::kLCD;
joshualitta751c972015-11-20 13:37:32 -0800163 break;
164 case kColorBitmapMask_MaskType:
Brian Salomon44acb5b2017-07-18 19:59:24 -0400165 coverage = GrProcessorAnalysisCoverage::kNone;
Brian Salomonc0b642c2017-03-27 13:09:36 -0400166 break;
joshualitta751c972015-11-20 13:37:32 -0800167 }
Chris Daltonb8fff0d2019-03-05 10:11:58 -0700168 auto analysis = fProcessors.finalize(
Chris Dalton6ce447a2019-06-23 18:07:38 -0600169 color, coverage, clip, &GrUserStencilSettings::kUnused, hasMixedSampledCoverage, caps,
170 clampType, &fGeoData[0].fColor);
Brian Salomon44acb5b2017-07-18 19:59:24 -0400171 fUsesLocalCoords = analysis.usesLocalCoords();
Chris Dalton4b62aed2019-01-15 11:53:00 -0700172 return analysis;
joshualitta751c972015-11-20 13:37:32 -0800173}
174
Brian Salomon18923f92017-11-06 16:26:02 -0500175static void clip_quads(const SkIRect& clipRect, char* currVertex, const char* blobVertices,
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400176 size_t vertexStride, int glyphCount) {
177 for (int i = 0; i < glyphCount; ++i) {
Brian Salomon18923f92017-11-06 16:26:02 -0500178 const SkPoint* blobPositionLT = reinterpret_cast<const SkPoint*>(blobVertices);
179 const SkPoint* blobPositionRB =
180 reinterpret_cast<const SkPoint*>(blobVertices + 3 * vertexStride);
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400181
Jim Van Verth328a33f2017-10-20 12:14:22 -0400182 // positions for bitmap glyphs are pixel boundary aligned
Brian Osman7ca90d22017-11-10 15:31:27 -0500183 SkIRect positionRect = SkIRect::MakeLTRB(SkScalarRoundToInt(blobPositionLT->fX),
184 SkScalarRoundToInt(blobPositionLT->fY),
185 SkScalarRoundToInt(blobPositionRB->fX),
186 SkScalarRoundToInt(blobPositionRB->fY));
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400187 if (clipRect.contains(positionRect)) {
188 memcpy(currVertex, blobVertices, 4 * vertexStride);
189 currVertex += 4 * vertexStride;
190 } else {
191 // Pull out some more data that we'll need.
192 // In the LCD case the color will be garbage, but we'll overwrite it with the texcoords
193 // and it avoids a lot of conditionals.
Brian Salomon18923f92017-11-06 16:26:02 -0500194 auto color = *reinterpret_cast<const SkColor*>(blobVertices + sizeof(SkPoint));
Jim Van Verthdea2f342020-01-15 10:48:25 -0500195 size_t coordOffset = vertexStride - 2*sizeof(int16_t);
196 auto* blobCoordsLT = reinterpret_cast<const int16_t*>(blobVertices + coordOffset);
197 auto* blobCoordsRB = reinterpret_cast<const int16_t*>(blobVertices + 3 * vertexStride +
Brian Salomon18923f92017-11-06 16:26:02 -0500198 coordOffset);
Jim Van Verth328a33f2017-10-20 12:14:22 -0400199 // Pull out the texel coordinates and texture index bits
Jim Van Verthdea2f342020-01-15 10:48:25 -0500200 int16_t coordsRectL = blobCoordsLT[0];
201 int16_t coordsRectT = blobCoordsLT[1];
202 int16_t coordsRectR = blobCoordsRB[0];
203 int16_t coordsRectB = blobCoordsRB[1];
204 int index0, index1;
205
206 std::tie(coordsRectL, coordsRectT, index0) =
207 GrDrawOpAtlas::UnpackIndexFromTexCoords(coordsRectL, coordsRectT);
208 std::tie(coordsRectR, coordsRectB, index1) =
209 GrDrawOpAtlas::UnpackIndexFromTexCoords(coordsRectR, coordsRectB);
210 SkASSERT(index0 == index1);
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400211
Jim Van Verth328a33f2017-10-20 12:14:22 -0400212 int positionRectWidth = positionRect.width();
213 int positionRectHeight = positionRect.height();
214 SkASSERT(positionRectWidth == (coordsRectR - coordsRectL));
215 SkASSERT(positionRectHeight == (coordsRectB - coordsRectT));
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400216
217 // Clip position and texCoords to the clipRect
Jim Van Verth328a33f2017-10-20 12:14:22 -0400218 unsigned int delta;
219 delta = SkTMin(SkTMax(clipRect.fLeft - positionRect.fLeft, 0), positionRectWidth);
220 coordsRectL += delta;
221 positionRect.fLeft += delta;
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400222
Jim Van Verth328a33f2017-10-20 12:14:22 -0400223 delta = SkTMin(SkTMax(clipRect.fTop - positionRect.fTop, 0), positionRectHeight);
224 coordsRectT += delta;
225 positionRect.fTop += delta;
226
227 delta = SkTMin(SkTMax(positionRect.fRight - clipRect.fRight, 0), positionRectWidth);
228 coordsRectR -= delta;
229 positionRect.fRight -= delta;
230
231 delta = SkTMin(SkTMax(positionRect.fBottom - clipRect.fBottom, 0), positionRectHeight);
232 coordsRectB -= delta;
233 positionRect.fBottom -= delta;
234
235 // Repack texel coordinates and index
Jim Van Verthdea2f342020-01-15 10:48:25 -0500236 std::tie(coordsRectL, coordsRectT) =
237 GrDrawOpAtlas::PackIndexInTexCoords(coordsRectL, coordsRectT, index0);
238 std::tie(coordsRectR, coordsRectB) =
239 GrDrawOpAtlas::PackIndexInTexCoords(coordsRectR, coordsRectB, index1);
Jim Van Verth328a33f2017-10-20 12:14:22 -0400240
241 // Set new positions and coords
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400242 SkPoint* currPosition = reinterpret_cast<SkPoint*>(currVertex);
243 currPosition->fX = positionRect.fLeft;
244 currPosition->fY = positionRect.fTop;
245 *(reinterpret_cast<SkColor*>(currVertex + sizeof(SkPoint))) = color;
Jim Van Verth328a33f2017-10-20 12:14:22 -0400246 uint16_t* currCoords = reinterpret_cast<uint16_t*>(currVertex + coordOffset);
247 currCoords[0] = coordsRectL;
248 currCoords[1] = coordsRectT;
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400249 currVertex += vertexStride;
250
251 currPosition = reinterpret_cast<SkPoint*>(currVertex);
252 currPosition->fX = positionRect.fLeft;
253 currPosition->fY = positionRect.fBottom;
254 *(reinterpret_cast<SkColor*>(currVertex + sizeof(SkPoint))) = color;
Jim Van Verth328a33f2017-10-20 12:14:22 -0400255 currCoords = reinterpret_cast<uint16_t*>(currVertex + coordOffset);
256 currCoords[0] = coordsRectL;
257 currCoords[1] = coordsRectB;
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400258 currVertex += vertexStride;
259
260 currPosition = reinterpret_cast<SkPoint*>(currVertex);
261 currPosition->fX = positionRect.fRight;
262 currPosition->fY = positionRect.fTop;
263 *(reinterpret_cast<SkColor*>(currVertex + sizeof(SkPoint))) = color;
Jim Van Verth328a33f2017-10-20 12:14:22 -0400264 currCoords = reinterpret_cast<uint16_t*>(currVertex + coordOffset);
265 currCoords[0] = coordsRectR;
266 currCoords[1] = coordsRectT;
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400267 currVertex += vertexStride;
268
269 currPosition = reinterpret_cast<SkPoint*>(currVertex);
270 currPosition->fX = positionRect.fRight;
271 currPosition->fY = positionRect.fBottom;
272 *(reinterpret_cast<SkColor*>(currVertex + sizeof(SkPoint))) = color;
Jim Van Verth328a33f2017-10-20 12:14:22 -0400273 currCoords = reinterpret_cast<uint16_t*>(currVertex + coordOffset);
274 currCoords[0] = coordsRectR;
275 currCoords[1] = coordsRectB;
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400276 currVertex += vertexStride;
277 }
278
279 blobVertices += 4 * vertexStride;
280 }
281}
282
Brian Salomon91326c32017-08-09 16:02:19 -0400283void GrAtlasTextOp::onPrepareDraws(Target* target) {
Robert Phillips4bc70112018-03-01 10:24:02 -0500284 auto resourceProvider = target->resourceProvider();
285
joshualitta751c972015-11-20 13:37:32 -0800286 // if we have RGB, then we won't have any SkShaders so no need to use a localmatrix.
287 // TODO actually only invert if we don't have RGBA
288 SkMatrix localMatrix;
Herb Derby1c5be7b2019-12-13 12:03:06 -0500289 if (this->usesLocalCoords() && !fGeoData[0].fDrawMatrix.invert(&localMatrix)) {
joshualitta751c972015-11-20 13:37:32 -0800290 return;
291 }
292
Robert Phillips5a66efb2018-03-07 15:13:18 -0500293 GrAtlasManager* atlasManager = target->atlasManager();
Herb Derby081e6f32019-01-16 13:46:02 -0500294 GrStrikeCache* glyphCache = target->glyphCache();
Robert Phillipsc4039ea2018-03-01 11:36:45 -0500295
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400296 GrMaskFormat maskFormat = this->maskFormat();
297
Greg Daniel9715b6c2019-12-10 15:03:10 -0500298 unsigned int numActiveViews;
299 const GrSurfaceProxyView* views = atlasManager->getViews(maskFormat, &numActiveViews);
300 if (!views) {
joshualitta751c972015-11-20 13:37:32 -0800301 SkDebugf("Could not allocate backing texture for atlas\n");
302 return;
303 }
Greg Daniel9715b6c2019-12-10 15:03:10 -0500304 SkASSERT(views[0].proxy());
joshualitta751c972015-11-20 13:37:32 -0800305
Brian Salomon7eae3e02018-08-07 14:02:38 +0000306 static constexpr int kMaxTextures = GrBitmapTextGeoProc::kMaxTextures;
Brian Salomon4dea72a2019-12-18 10:43:10 -0500307 static_assert(GrDistanceFieldA8TextGeoProc::kMaxTextures == kMaxTextures);
308 static_assert(GrDistanceFieldLCDTextGeoProc::kMaxTextures == kMaxTextures);
Brian Salomon7eae3e02018-08-07 14:02:38 +0000309
Chris Dalton07cdcfc92019-02-26 11:13:22 -0700310 auto fixedDynamicState = target->makeFixedDynamicState(kMaxTextures);
Greg Daniel9715b6c2019-12-10 15:03:10 -0500311 for (unsigned i = 0; i < numActiveViews; ++i) {
312 fixedDynamicState->fPrimitiveProcessorTextures[i] = views[i].proxy();
Greg Danielb20d7e52019-09-03 13:54:39 -0400313 // This op does not know its atlas proxies when it is added to a GrOpsTasks, so the proxies
314 // don't get added during the visitProxies call. Thus we add them here.
Greg Daniel9715b6c2019-12-10 15:03:10 -0500315 target->sampledProxyArray()->push_back(views[i].proxy());
Brian Salomon7eae3e02018-08-07 14:02:38 +0000316 }
Brian Salomon49348902018-06-26 09:12:38 -0400317
318 FlushInfo flushInfo;
Chris Dalton07cdcfc92019-02-26 11:13:22 -0700319 flushInfo.fFixedDynamicState = fixedDynamicState;
Brian Salomon49348902018-06-26 09:12:38 -0400320
Herb Derby1c5be7b2019-12-13 12:03:06 -0500321 bool vmPerspective = fGeoData[0].fDrawMatrix.hasPerspective();
joshualittd9d30f72015-12-08 10:47:55 -0800322 if (this->usesDistanceFields()) {
Robert Phillips7cd0bfe2019-11-20 16:08:10 -0500323 flushInfo.fGeometryProcessor = this->setupDfProcessor(target->allocator(),
324 *target->caps().shaderCaps(),
Greg Daniel9715b6c2019-12-10 15:03:10 -0500325 views, numActiveViews);
joshualitta751c972015-11-20 13:37:32 -0800326 } else {
Brian Salomonccb61422020-01-09 10:46:36 -0500327 auto filter = fNeedsGlyphTransform ? GrSamplerState::Filter::kBilerp
328 : GrSamplerState::Filter::kNearest;
329 flushInfo.fGeometryProcessor = GrBitmapTextGeoProc::Make(
330 target->allocator(), *target->caps().shaderCaps(), this->color(), false, views,
331 numActiveViews, filter, maskFormat, localMatrix, vmPerspective);
joshualitta751c972015-11-20 13:37:32 -0800332 }
333
Brian Osmanf04fb3c2018-11-12 15:34:00 -0500334 size_t vertexStride = flushInfo.fGeometryProcessor->vertexStride();
joshualitta751c972015-11-20 13:37:32 -0800335
Brian Salomon43cbd722020-01-03 22:09:12 -0500336 // Ensure we don't request an insanely large contiguous vertex allocation.
337 static const size_t kMaxVertexBytes = GrBufferAllocPool::kDefaultBufferSize;
338 int maxGlyphsInBuffer = kMaxVertexBytes / (vertexStride * kVerticesPerGlyph);
339 int totalGlyphCount = this->numGlyphs();
340 int bufferGlyphCount = std::min(totalGlyphCount, maxGlyphsInBuffer);
341 void* vertices = target->makeVertexSpace(vertexStride, bufferGlyphCount * kVerticesPerGlyph,
Brian Salomon12d22642019-01-29 14:38:50 -0500342 &flushInfo.fVertexBuffer, &flushInfo.fVertexOffset);
Robert Phillipsee08d522019-10-28 16:34:44 -0400343 flushInfo.fIndexBuffer = resourceProvider->refNonAAQuadIndexBuffer();
joshualitta751c972015-11-20 13:37:32 -0800344 if (!vertices || !flushInfo.fVertexBuffer) {
345 SkDebugf("Could not allocate vertices\n");
346 return;
347 }
348
Brian Salomon18923f92017-11-06 16:26:02 -0500349 char* currVertex = reinterpret_cast<char*>(vertices);
joshualitta751c972015-11-20 13:37:32 -0800350
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400351 // each of these is a SubRun
joshualitta751c972015-11-20 13:37:32 -0800352 for (int i = 0; i < fGeoCount; i++) {
joshualitt144c3c82015-11-30 12:30:13 -0800353 const Geometry& args = fGeoData[i];
Brian Osman1be2b7c2018-10-29 16:07:15 -0400354 // TODO4F: Preserve float colors
Herb Derby86240592018-05-24 16:12:31 -0400355 GrTextBlob::VertexRegenerator regenerator(
Herb Derby5bf5b042019-12-12 16:37:03 -0500356 resourceProvider, args.fSubRunPtr, args.fDrawMatrix, args.fDrawOrigin,
Brian Osmancf860852018-10-31 14:04:39 -0400357 args.fColor.toBytes_RGBA(), target->deferredUploadTarget(), glyphCache,
Herb Derbycb443a52019-11-11 18:01:36 -0500358 atlasManager);
Brian Salomon43cbd722020-01-03 22:09:12 -0500359 // This loop issues draws until regenerator says we're done with this geo. Regenerator
360 // breaks things up if inline uploads are necessary.
Herb Derby5f6f8512020-01-10 12:50:35 -0500361 GrTextBlob::VertexRegenerator::Result result;
362 while (!result.fFinished) {
Brian Salomon43cbd722020-01-03 22:09:12 -0500363 // Copy regenerated vertices from the blob to our vertex buffer. If we overflow our
364 // vertex buffer we'll issue a draw and then get more vertex buffer space.
365 do {
366 if (!bufferGlyphCount) {
367 this->flush(target, &flushInfo);
368 bufferGlyphCount = std::min(totalGlyphCount, maxGlyphsInBuffer);
369 vertices = target->makeVertexSpace(
370 vertexStride, bufferGlyphCount * kVerticesPerGlyph,
371 &flushInfo.fVertexBuffer, &flushInfo.fVertexOffset);
372 currVertex = reinterpret_cast<char*>(vertices);
373 }
374 if (!regenerator.regenerate(&result, bufferGlyphCount)) {
375 return;
376 }
377 int glyphCount = std::min(result.fGlyphsRegenerated, bufferGlyphCount);
378 int vertexCount = glyphCount * kVerticesPerGlyph;
379 size_t vertexBytes = vertexCount * vertexStride;
380 if (args.fClipRect.isEmpty()) {
381 memcpy(currVertex, result.fFirstVertex, vertexBytes);
382 } else {
383 SkASSERT(!vmPerspective);
384 clip_quads(args.fClipRect, currVertex, result.fFirstVertex, vertexStride,
385 glyphCount);
386 }
387 if (fNeedsGlyphTransform && !args.fDrawMatrix.isIdentity()) {
388 // We always do the distance field view matrix transformation after copying
389 // rather than during blob vertex generation time in the blob as handling
390 // successive arbitrary transformations would be complicated and accumulate
391 // error.
392 if (args.fDrawMatrix.hasPerspective()) {
393 auto* pos = reinterpret_cast<SkPoint3*>(currVertex);
394 SkMatrixPriv::MapHomogeneousPointsWithStride(args.fDrawMatrix, pos,
395 vertexStride, pos,
396 vertexStride, vertexCount);
397 } else {
398 auto* pos = reinterpret_cast<SkPoint*>(currVertex);
399 SkMatrixPriv::MapPointsWithStride(args.fDrawMatrix, pos, vertexStride,
400 vertexCount);
401 }
402 }
403 flushInfo.fGlyphsToFlush += glyphCount;
404 currVertex += vertexBytes;
405 result.fFirstVertex += vertexBytes;
406 result.fGlyphsRegenerated -= glyphCount;
407 bufferGlyphCount -= glyphCount;
408 totalGlyphCount -= glyphCount;
409 } while (result.fGlyphsRegenerated);
Herb Derby5f6f8512020-01-10 12:50:35 -0500410 if (!result.fFinished) {
411 this->flush(target, &flushInfo);
Robert Phillipsd2e9f762018-03-07 11:54:37 -0500412 }
Herb Derby5f6f8512020-01-10 12:50:35 -0500413 } // for all vertices
414 } // for all geometries
Brian Salomon43cbd722020-01-03 22:09:12 -0500415 SkASSERT(!bufferGlyphCount);
416 SkASSERT(!totalGlyphCount);
joshualitta751c972015-11-20 13:37:32 -0800417 this->flush(target, &flushInfo);
418}
419
Chris Dalton07cdcfc92019-02-26 11:13:22 -0700420void GrAtlasTextOp::onExecute(GrOpFlushState* flushState, const SkRect& chainBounds) {
Robert Phillips3968fcb2019-12-05 16:40:31 -0500421 auto pipeline = GrSimpleMeshDrawOpHelper::CreatePipeline(flushState,
422 std::move(fProcessors),
423 GrPipeline::InputFlags::kNone);
424
425 flushState->executeDrawsAndUploadsForMeshDrawOp(this, chainBounds, pipeline);
Chris Dalton07cdcfc92019-02-26 11:13:22 -0700426}
427
Brian Salomone5b399e2017-07-19 13:50:54 -0400428void GrAtlasTextOp::flush(GrMeshDrawOp::Target* target, FlushInfo* flushInfo) const {
Robert Phillipsd2e9f762018-03-07 11:54:37 -0500429 if (!flushInfo->fGlyphsToFlush) {
430 return;
431 }
432
Robert Phillips5a66efb2018-03-07 15:13:18 -0500433 auto atlasManager = target->atlasManager();
Robert Phillipsc4039ea2018-03-01 11:36:45 -0500434
Robert Phillips7cd0bfe2019-11-20 16:08:10 -0500435 GrGeometryProcessor* gp = flushInfo->fGeometryProcessor;
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400436 GrMaskFormat maskFormat = this->maskFormat();
Robert Phillipsf3690dd2018-02-20 15:18:59 -0500437
Greg Daniel9715b6c2019-12-10 15:03:10 -0500438 unsigned int numActiveViews;
439 const GrSurfaceProxyView* views = atlasManager->getViews(maskFormat, &numActiveViews);
440 SkASSERT(views);
Jim Van Verth9f2516f2019-11-22 14:58:37 -0500441 // Something has gone terribly wrong, bail
Greg Daniel9715b6c2019-12-10 15:03:10 -0500442 if (!views || 0 == numActiveViews) {
Jim Van Verth9f2516f2019-11-22 14:58:37 -0500443 return;
444 }
Greg Daniel9715b6c2019-12-10 15:03:10 -0500445 if (gp->numTextureSamplers() != (int) numActiveViews) {
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400446 // During preparation the number of atlas pages has increased.
447 // Update the proxies used in the GP to match.
Greg Daniel9715b6c2019-12-10 15:03:10 -0500448 for (unsigned i = gp->numTextureSamplers(); i < numActiveViews; ++i) {
449 flushInfo->fFixedDynamicState->fPrimitiveProcessorTextures[i] = views[i].proxy();
Greg Danielb20d7e52019-09-03 13:54:39 -0400450 // This op does not know its atlas proxies when it is added to a GrOpsTasks, so the
451 // proxies don't get added during the visitProxies call. Thus we add them here.
Greg Daniel9715b6c2019-12-10 15:03:10 -0500452 target->sampledProxyArray()->push_back(views[i].proxy());
Brian Salomon43cbd722020-01-03 22:09:12 -0500453 // These will get unreffed when the previously recorded draws destruct.
454 for (int d = 0; d < flushInfo->fNumDraws; ++d) {
455 flushInfo->fFixedDynamicState->fPrimitiveProcessorTextures[i]->ref();
456 }
Brian Salomon7eae3e02018-08-07 14:02:38 +0000457 }
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400458 if (this->usesDistanceFields()) {
459 if (this->isLCD()) {
Greg Daniel9715b6c2019-12-10 15:03:10 -0500460 reinterpret_cast<GrDistanceFieldLCDTextGeoProc*>(gp)->addNewViews(
Brian Salomonccb61422020-01-09 10:46:36 -0500461 views, numActiveViews, GrSamplerState::Filter::kBilerp);
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400462 } else {
Greg Daniel9715b6c2019-12-10 15:03:10 -0500463 reinterpret_cast<GrDistanceFieldA8TextGeoProc*>(gp)->addNewViews(
Brian Salomonccb61422020-01-09 10:46:36 -0500464 views, numActiveViews, GrSamplerState::Filter::kBilerp);
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400465 }
466 } else {
Brian Salomonccb61422020-01-09 10:46:36 -0500467 auto filter = fNeedsGlyphTransform ? GrSamplerState::Filter::kBilerp
468 : GrSamplerState::Filter::kNearest;
469 reinterpret_cast<GrBitmapTextGeoProc*>(gp)->addNewViews(views, numActiveViews, filter);
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400470 }
471 }
Brian Salomondbf70722019-02-07 11:31:24 -0500472 int maxGlyphsPerDraw = static_cast<int>(flushInfo->fIndexBuffer->size() / sizeof(uint16_t) / 6);
Brian Salomon7eae3e02018-08-07 14:02:38 +0000473 GrMesh* mesh = target->allocMesh(GrPrimitiveType::kTriangles);
Brian Salomon12d22642019-01-29 14:38:50 -0500474 mesh->setIndexedPatterned(flushInfo->fIndexBuffer, kIndicesPerGlyph, kVerticesPerGlyph,
Brian Salomon7eae3e02018-08-07 14:02:38 +0000475 flushInfo->fGlyphsToFlush, maxGlyphsPerDraw);
Brian Salomon12d22642019-01-29 14:38:50 -0500476 mesh->setVertexData(flushInfo->fVertexBuffer, flushInfo->fVertexOffset);
Robert Phillipscea290f2019-11-06 11:21:03 -0500477 target->recordDraw(flushInfo->fGeometryProcessor, mesh, 1, flushInfo->fFixedDynamicState,
478 nullptr, GrPrimitiveType::kTriangles);
joshualitta751c972015-11-20 13:37:32 -0800479 flushInfo->fVertexOffset += kVerticesPerGlyph * flushInfo->fGlyphsToFlush;
480 flushInfo->fGlyphsToFlush = 0;
Brian Salomon43cbd722020-01-03 22:09:12 -0500481 ++flushInfo->fNumDraws;
joshualitta751c972015-11-20 13:37:32 -0800482}
483
Michael Ludwig28b0c5d2019-12-19 14:51:00 -0500484GrOp::CombineResult GrAtlasTextOp::onCombineIfPossible(GrOp* t, GrRecordingContext::Arenas*,
485 const GrCaps& caps) {
Brian Salomon344ec422016-12-15 10:58:41 -0500486 GrAtlasTextOp* that = t->cast<GrAtlasTextOp>();
Brian Salomon44acb5b2017-07-18 19:59:24 -0400487 if (fProcessors != that->fProcessors) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000488 return CombineResult::kCannotCombine;
Brian Salomon44acb5b2017-07-18 19:59:24 -0400489 }
490
joshualitta751c972015-11-20 13:37:32 -0800491 if (fMaskType != that->fMaskType) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000492 return CombineResult::kCannotCombine;
joshualitta751c972015-11-20 13:37:32 -0800493 }
494
Herb Derby1c5be7b2019-12-13 12:03:06 -0500495 const SkMatrix& thisFirstMatrix = fGeoData[0].fDrawMatrix;
496 const SkMatrix& thatFirstMatrix = that->fGeoData[0].fDrawMatrix;
Brian Salomon5c6ac642017-12-19 11:09:32 -0500497
Mike Reed2c383152019-12-18 16:47:47 -0500498 if (this->usesLocalCoords() && !SkMatrixPriv::CheapEqual(thisFirstMatrix, thatFirstMatrix)) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000499 return CombineResult::kCannotCombine;
Brian Salomon5c6ac642017-12-19 11:09:32 -0500500 }
501
Jim Van Verthb515ae72018-05-23 16:44:55 -0400502 if (fNeedsGlyphTransform != that->fNeedsGlyphTransform) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000503 return CombineResult::kCannotCombine;
Jim Van Verthb515ae72018-05-23 16:44:55 -0400504 }
505
506 if (fNeedsGlyphTransform &&
507 (thisFirstMatrix.hasPerspective() != thatFirstMatrix.hasPerspective())) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000508 return CombineResult::kCannotCombine;
Jim Van Verthb515ae72018-05-23 16:44:55 -0400509 }
510
Brian Salomon5c6ac642017-12-19 11:09:32 -0500511 if (this->usesDistanceFields()) {
512 if (fDFGPFlags != that->fDFGPFlags) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000513 return CombineResult::kCannotCombine;
joshualitta751c972015-11-20 13:37:32 -0800514 }
515
Jim Van Verthbc2cdd12017-06-08 11:14:35 -0400516 if (fLuminanceColor != that->fLuminanceColor) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000517 return CombineResult::kCannotCombine;
joshualitta751c972015-11-20 13:37:32 -0800518 }
Brian Salomon5c6ac642017-12-19 11:09:32 -0500519 } else {
520 if (kColorBitmapMask_MaskType == fMaskType && this->color() != that->color()) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000521 return CombineResult::kCannotCombine;
Brian Salomon5c6ac642017-12-19 11:09:32 -0500522 }
joshualitta751c972015-11-20 13:37:32 -0800523 }
524
Brian Salomon344ec422016-12-15 10:58:41 -0500525 fNumGlyphs += that->numGlyphs();
joshualitta751c972015-11-20 13:37:32 -0800526
Jim Van Verth56c37142017-10-31 14:44:25 -0400527 // Reallocate space for geo data if necessary and then import that geo's data.
joshualitta751c972015-11-20 13:37:32 -0800528 int newGeoCount = that->fGeoCount + fGeoCount;
joshualitta751c972015-11-20 13:37:32 -0800529
Jim Van Verthc8a65e32017-10-25 14:25:27 -0400530 // We reallocate at a rate of 1.5x to try to get better total memory usage
531 if (newGeoCount > fGeoDataAllocSize) {
Jim Van Verth56c37142017-10-31 14:44:25 -0400532 int newAllocSize = fGeoDataAllocSize + fGeoDataAllocSize / 2;
Jim Van Verthc8a65e32017-10-25 14:25:27 -0400533 while (newAllocSize < newGeoCount) {
534 newAllocSize += newAllocSize / 2;
535 }
joshualitta751c972015-11-20 13:37:32 -0800536 fGeoData.realloc(newAllocSize);
Jim Van Verthc8a65e32017-10-25 14:25:27 -0400537 fGeoDataAllocSize = newAllocSize;
joshualitta751c972015-11-20 13:37:32 -0800538 }
539
Brian Salomon344ec422016-12-15 10:58:41 -0500540 // We steal the ref on the blobs from the other AtlasTextOp and set its count to 0 so that
joshualitta751c972015-11-20 13:37:32 -0800541 // it doesn't try to unref them.
Brian Salomon344ec422016-12-15 10:58:41 -0500542 memcpy(&fGeoData[fGeoCount], that->fGeoData.get(), that->fGeoCount * sizeof(Geometry));
joshualitta751c972015-11-20 13:37:32 -0800543#ifdef SK_DEBUG
544 for (int i = 0; i < that->fGeoCount; ++i) {
Herb Derbyc514e7d2019-12-11 17:00:31 -0500545 that->fGeoData.get()[i].fBlob = (GrTextBlob*)0x1;
joshualitta751c972015-11-20 13:37:32 -0800546 }
547#endif
548 that->fGeoCount = 0;
549 fGeoCount = newGeoCount;
550
Brian Salomon7eae3e02018-08-07 14:02:38 +0000551 return CombineResult::kMerged;
joshualitta751c972015-11-20 13:37:32 -0800552}
553
joshualitta751c972015-11-20 13:37:32 -0800554// TODO trying to figure out why lcd is so whack
Herb Derby26cbe512018-05-24 14:39:01 -0400555// (see comments in GrTextContext::ComputeCanonicalColor)
Robert Phillips7cd0bfe2019-11-20 16:08:10 -0500556GrGeometryProcessor* GrAtlasTextOp::setupDfProcessor(SkArenaAlloc* arena,
557 const GrShaderCaps& caps,
Greg Daniel9715b6c2019-12-10 15:03:10 -0500558 const GrSurfaceProxyView* views,
559 unsigned int numActiveViews) const {
joshualitta751c972015-11-20 13:37:32 -0800560 bool isLCD = this->isLCD();
Brian Salomon5c6ac642017-12-19 11:09:32 -0500561
562 SkMatrix localMatrix = SkMatrix::I();
563 if (this->usesLocalCoords()) {
564 // If this fails we'll just use I().
Herb Derby1c5be7b2019-12-13 12:03:06 -0500565 bool result = fGeoData[0].fDrawMatrix.invert(&localMatrix);
Brian Salomon5c6ac642017-12-19 11:09:32 -0500566 (void)result;
567 }
joshualitta751c972015-11-20 13:37:32 -0800568
569 // see if we need to create a new effect
570 if (isLCD) {
brianosman0586f5c2016-04-12 12:48:21 -0700571 float redCorrection = fDistanceAdjustTable->getAdjustment(
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400572 SkColorGetR(fLuminanceColor) >> kDistanceAdjustLumShift,
Brian Salomon344ec422016-12-15 10:58:41 -0500573 fUseGammaCorrectDistanceTable);
brianosman0586f5c2016-04-12 12:48:21 -0700574 float greenCorrection = fDistanceAdjustTable->getAdjustment(
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400575 SkColorGetG(fLuminanceColor) >> kDistanceAdjustLumShift,
Brian Salomon344ec422016-12-15 10:58:41 -0500576 fUseGammaCorrectDistanceTable);
brianosman0586f5c2016-04-12 12:48:21 -0700577 float blueCorrection = fDistanceAdjustTable->getAdjustment(
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400578 SkColorGetB(fLuminanceColor) >> kDistanceAdjustLumShift,
Brian Salomon344ec422016-12-15 10:58:41 -0500579 fUseGammaCorrectDistanceTable);
joshualitta751c972015-11-20 13:37:32 -0800580 GrDistanceFieldLCDTextGeoProc::DistanceAdjust widthAdjust =
Brian Salomon344ec422016-12-15 10:58:41 -0500581 GrDistanceFieldLCDTextGeoProc::DistanceAdjust::Make(
582 redCorrection, greenCorrection, blueCorrection);
Greg Daniel9715b6c2019-12-10 15:03:10 -0500583 return GrDistanceFieldLCDTextGeoProc::Make(arena, caps, views, numActiveViews,
Brian Salomonccb61422020-01-09 10:46:36 -0500584 GrSamplerState::Filter::kBilerp, widthAdjust,
Brian Osman09068252018-01-03 09:57:29 -0500585 fDFGPFlags, localMatrix);
joshualitta751c972015-11-20 13:37:32 -0800586 } else {
joshualitta751c972015-11-20 13:37:32 -0800587#ifdef SK_GAMMA_APPLY_TO_A8
Jim Van Verth90e89b32017-07-06 16:36:55 -0400588 float correction = 0;
589 if (kAliasedDistanceField_MaskType != fMaskType) {
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400590 U8CPU lum = SkColorSpaceLuminance::computeLuminance(SK_GAMMA_EXPONENT,
591 fLuminanceColor);
Jim Van Verth90e89b32017-07-06 16:36:55 -0400592 correction = fDistanceAdjustTable->getAdjustment(lum >> kDistanceAdjustLumShift,
593 fUseGammaCorrectDistanceTable);
594 }
Greg Daniel9715b6c2019-12-10 15:03:10 -0500595 return GrDistanceFieldA8TextGeoProc::Make(arena, caps, views, numActiveViews,
Brian Salomonccb61422020-01-09 10:46:36 -0500596 GrSamplerState::Filter::kBilerp, correction,
597 fDFGPFlags, localMatrix);
joshualitta751c972015-11-20 13:37:32 -0800598#else
Greg Daniel9715b6c2019-12-10 15:03:10 -0500599 return GrDistanceFieldA8TextGeoProc::Make(arena, caps, views, numActiveViews,
Brian Salomonccb61422020-01-09 10:46:36 -0500600 GrSamplerState::Filter::kBilerp,
Brian Salomon5c6ac642017-12-19 11:09:32 -0500601 fDFGPFlags, localMatrix);
joshualitta751c972015-11-20 13:37:32 -0800602#endif
603 }
joshualitta751c972015-11-20 13:37:32 -0800604}
joshualittddd22d82016-02-16 06:47:52 -0800605