blob: 667e230251122e354974485c43142a1d45c508bf [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
Brian Salomon09e019e2016-12-15 10:20:35 -05008#include "GrAtlasTextOp.h"
Robert Phillipse4fda6c2018-02-21 12:10:41 -05009
Brian Osman4a3f5c82018-09-18 16:16:38 -040010#include "GrCaps.h"
Robert Phillips296b1cc2017-03-15 10:42:12 -040011#include "GrContext.h"
Robert Phillipsc994a932018-06-19 13:09:54 -040012#include "GrContextPriv.h"
Robert Phillips7c525e62018-06-12 10:11:12 -040013#include "GrMemoryPool.h"
Brian Salomon742e31d2016-12-07 17:06:19 -050014#include "GrOpFlushState.h"
joshualitta751c972015-11-20 13:37:32 -080015#include "GrResourceProvider.h"
halcanary4dbbd042016-06-07 17:21:10 -070016#include "SkMathPriv.h"
Brian Salomon5c6ac642017-12-19 11:09:32 -050017#include "SkMatrixPriv.h"
18#include "SkPoint3.h"
Herb Derbydce19a72018-04-18 16:02:17 -040019#include "SkStrikeCache.h"
joshualitta751c972015-11-20 13:37:32 -080020#include "effects/GrBitmapTextGeoProc.h"
21#include "effects/GrDistanceFieldGeoProc.h"
Robert Phillipsc4039ea2018-03-01 11:36:45 -050022#include "text/GrAtlasManager.h"
Herb Derby081e6f32019-01-16 13:46:02 -050023#include "text/GrStrikeCache.h"
joshualitta751c972015-11-20 13:37:32 -080024
joshualitt60ce86d2015-11-23 13:08:22 -080025///////////////////////////////////////////////////////////////////////////////////////////////////
26
Robert Phillips7c525e62018-06-12 10:11:12 -040027std::unique_ptr<GrAtlasTextOp> GrAtlasTextOp::MakeBitmap(GrContext* context,
28 GrPaint&& paint,
29 GrMaskFormat maskFormat,
30 int glyphCount,
31 bool needsTransform) {
Robert Phillipsc994a932018-06-19 13:09:54 -040032 GrOpMemoryPool* pool = context->contextPriv().opMemoryPool();
33
34 std::unique_ptr<GrAtlasTextOp> op = pool->allocate<GrAtlasTextOp>(std::move(paint));
Robert Phillips7c525e62018-06-12 10:11:12 -040035
36 switch (maskFormat) {
37 case kA8_GrMaskFormat:
38 op->fMaskType = kGrayscaleCoverageMask_MaskType;
39 break;
40 case kA565_GrMaskFormat:
41 op->fMaskType = kLCDCoverageMask_MaskType;
42 break;
43 case kARGB_GrMaskFormat:
44 op->fMaskType = kColorBitmapMask_MaskType;
45 break;
46 }
47 op->fNumGlyphs = glyphCount;
48 op->fGeoCount = 1;
49 op->fLuminanceColor = 0;
50 op->fNeedsGlyphTransform = needsTransform;
51 return op;
52 }
53
54std::unique_ptr<GrAtlasTextOp> GrAtlasTextOp::MakeDistanceField(
55 GrContext* context,
56 GrPaint&& paint,
57 int glyphCount,
58 const GrDistanceFieldAdjustTable* distanceAdjustTable,
59 bool useGammaCorrectDistanceTable,
60 SkColor luminanceColor,
61 const SkSurfaceProps& props,
62 bool isAntiAliased,
63 bool useLCD) {
Robert Phillipsc994a932018-06-19 13:09:54 -040064 GrOpMemoryPool* pool = context->contextPriv().opMemoryPool();
65
66 std::unique_ptr<GrAtlasTextOp> op = pool->allocate<GrAtlasTextOp>(std::move(paint));
Robert Phillips7c525e62018-06-12 10:11:12 -040067
68 bool isBGR = SkPixelGeometryIsBGR(props.pixelGeometry());
69 bool isLCD = useLCD && SkPixelGeometryIsH(props.pixelGeometry());
70 op->fMaskType = !isAntiAliased ? kAliasedDistanceField_MaskType
71 : isLCD ? (isBGR ? kLCDBGRDistanceField_MaskType
72 : kLCDDistanceField_MaskType)
73 : kGrayscaleDistanceField_MaskType;
74 op->fDistanceAdjustTable.reset(SkRef(distanceAdjustTable));
75 op->fUseGammaCorrectDistanceTable = useGammaCorrectDistanceTable;
76 op->fLuminanceColor = luminanceColor;
77 op->fNumGlyphs = glyphCount;
78 op->fGeoCount = 1;
79 return op;
80 }
81
joshualitta751c972015-11-20 13:37:32 -080082static const int kDistanceAdjustLumShift = 5;
83
Brian Salomon5c6ac642017-12-19 11:09:32 -050084void GrAtlasTextOp::init() {
85 const Geometry& geo = fGeoData[0];
Brian Salomon5c6ac642017-12-19 11:09:32 -050086 if (this->usesDistanceFields()) {
87 bool isLCD = this->isLCD();
88
89 const SkMatrix& viewMatrix = geo.fViewMatrix;
90
91 fDFGPFlags = viewMatrix.isSimilarity() ? kSimilarity_DistanceFieldEffectFlag : 0;
92 fDFGPFlags |= viewMatrix.isScaleTranslate() ? kScaleOnly_DistanceFieldEffectFlag : 0;
93 fDFGPFlags |= viewMatrix.hasPerspective() ? kPerspective_DistanceFieldEffectFlag : 0;
94 fDFGPFlags |= fUseGammaCorrectDistanceTable ? kGammaCorrect_DistanceFieldEffectFlag : 0;
95 fDFGPFlags |= (kAliasedDistanceField_MaskType == fMaskType)
96 ? kAliased_DistanceFieldEffectFlag
97 : 0;
98
99 if (isLCD) {
100 fDFGPFlags |= kUseLCD_DistanceFieldEffectFlag;
101 fDFGPFlags |=
102 (kLCDBGRDistanceField_MaskType == fMaskType) ? kBGR_DistanceFieldEffectFlag : 0;
103 }
Jim Van Verthb515ae72018-05-23 16:44:55 -0400104
105 fNeedsGlyphTransform = true;
Brian Salomon5c6ac642017-12-19 11:09:32 -0500106 }
Jim Van Verth70276912018-06-01 13:46:46 -0400107
108 SkRect bounds;
109 geo.fBlob->computeSubRunBounds(&bounds, geo.fRun, geo.fSubRun, geo.fViewMatrix, geo.fX, geo.fY,
110 fNeedsGlyphTransform);
111 // We don't have tight bounds on the glyph paths in device space. For the purposes of bounds
112 // we treat this as a set of non-AA rects rendered with a texture.
113 this->setBounds(bounds, HasAABloat::kNo, IsZeroArea::kNo);
Brian Salomon5c6ac642017-12-19 11:09:32 -0500114}
115
Brian Salomon7d94bb52018-10-12 14:37:19 -0400116void GrAtlasTextOp::visitProxies(const VisitProxyFunc& func, VisitorType) const {
Robert Phillipse4fda6c2018-02-21 12:10:41 -0500117 fProcessors.visitProxies(func);
Robert Phillipse4fda6c2018-02-21 12:10:41 -0500118}
119
Brian Osman9a390ac2018-11-12 09:47:48 -0500120#ifdef SK_DEBUG
Brian Salomon344ec422016-12-15 10:58:41 -0500121SkString GrAtlasTextOp::dumpInfo() const {
joshualitta751c972015-11-20 13:37:32 -0800122 SkString str;
123
124 for (int i = 0; i < fGeoCount; ++i) {
125 str.appendf("%d: Color: 0x%08x Trans: %.2f,%.2f Runs: %d\n",
126 i,
Brian Osmancf860852018-10-31 14:04:39 -0400127 fGeoData[i].fColor.toBytes_RGBA(),
joshualitt8e0ef292016-02-19 14:13:03 -0800128 fGeoData[i].fX,
129 fGeoData[i].fY,
Herb Derby2bb343c2018-11-08 15:03:48 -0500130 fGeoData[i].fBlob->runCountLimit());
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 Dalton4b62aed2019-01-15 11:53:00 -0700143GrProcessorSet::Analysis GrAtlasTextOp::finalize(const GrCaps& caps, const GrAppliedClip* clip) {
Brian Salomon44acb5b2017-07-18 19:59:24 -0400144 GrProcessorAnalysisCoverage coverage;
145 GrProcessorAnalysisColor color;
joshualitta751c972015-11-20 13:37:32 -0800146 if (kColorBitmapMask_MaskType == fMaskType) {
Brian Salomon44acb5b2017-07-18 19:59:24 -0400147 color.setToUnknown();
joshualitta751c972015-11-20 13:37:32 -0800148 } else {
Brian Osman09068252018-01-03 09:57:29 -0500149 color.setToConstant(this->color());
joshualitta751c972015-11-20 13:37:32 -0800150 }
joshualitta751c972015-11-20 13:37:32 -0800151 switch (fMaskType) {
joshualitta751c972015-11-20 13:37:32 -0800152 case kGrayscaleCoverageMask_MaskType:
Jim Van Verth90e89b32017-07-06 16:36:55 -0400153 case kAliasedDistanceField_MaskType:
154 case kGrayscaleDistanceField_MaskType:
Brian Salomon44acb5b2017-07-18 19:59:24 -0400155 coverage = GrProcessorAnalysisCoverage::kSingleChannel;
joshualitta751c972015-11-20 13:37:32 -0800156 break;
157 case kLCDCoverageMask_MaskType:
158 case kLCDDistanceField_MaskType:
Jim Van Verth90e89b32017-07-06 16:36:55 -0400159 case kLCDBGRDistanceField_MaskType:
Brian Salomon44acb5b2017-07-18 19:59:24 -0400160 coverage = GrProcessorAnalysisCoverage::kLCD;
joshualitta751c972015-11-20 13:37:32 -0800161 break;
162 case kColorBitmapMask_MaskType:
Brian Salomon44acb5b2017-07-18 19:59:24 -0400163 coverage = GrProcessorAnalysisCoverage::kNone;
Brian Salomonc0b642c2017-03-27 13:09:36 -0400164 break;
joshualitta751c972015-11-20 13:37:32 -0800165 }
Brian Osman532b3f92018-07-11 10:02:07 -0400166 auto analysis = fProcessors.finalize(color, coverage, clip, false, caps, &fGeoData[0].fColor);
Brian Salomon44acb5b2017-07-18 19:59:24 -0400167 fUsesLocalCoords = analysis.usesLocalCoords();
Chris Dalton4b62aed2019-01-15 11:53:00 -0700168 return analysis;
joshualitta751c972015-11-20 13:37:32 -0800169}
170
Brian Salomon18923f92017-11-06 16:26:02 -0500171static void clip_quads(const SkIRect& clipRect, char* currVertex, const char* blobVertices,
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400172 size_t vertexStride, int glyphCount) {
173 for (int i = 0; i < glyphCount; ++i) {
Brian Salomon18923f92017-11-06 16:26:02 -0500174 const SkPoint* blobPositionLT = reinterpret_cast<const SkPoint*>(blobVertices);
175 const SkPoint* blobPositionRB =
176 reinterpret_cast<const SkPoint*>(blobVertices + 3 * vertexStride);
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400177
Jim Van Verth328a33f2017-10-20 12:14:22 -0400178 // positions for bitmap glyphs are pixel boundary aligned
Brian Osman7ca90d22017-11-10 15:31:27 -0500179 SkIRect positionRect = SkIRect::MakeLTRB(SkScalarRoundToInt(blobPositionLT->fX),
180 SkScalarRoundToInt(blobPositionLT->fY),
181 SkScalarRoundToInt(blobPositionRB->fX),
182 SkScalarRoundToInt(blobPositionRB->fY));
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400183 if (clipRect.contains(positionRect)) {
184 memcpy(currVertex, blobVertices, 4 * vertexStride);
185 currVertex += 4 * vertexStride;
186 } else {
187 // Pull out some more data that we'll need.
188 // In the LCD case the color will be garbage, but we'll overwrite it with the texcoords
189 // and it avoids a lot of conditionals.
Brian Salomon18923f92017-11-06 16:26:02 -0500190 auto color = *reinterpret_cast<const SkColor*>(blobVertices + sizeof(SkPoint));
Jim Van Verth328a33f2017-10-20 12:14:22 -0400191 size_t coordOffset = vertexStride - 2*sizeof(uint16_t);
Brian Salomon18923f92017-11-06 16:26:02 -0500192 auto* blobCoordsLT = reinterpret_cast<const uint16_t*>(blobVertices + coordOffset);
193 auto* blobCoordsRB = reinterpret_cast<const uint16_t*>(blobVertices + 3 * vertexStride +
194 coordOffset);
Jim Van Verth328a33f2017-10-20 12:14:22 -0400195 // Pull out the texel coordinates and texture index bits
196 uint16_t coordsRectL = blobCoordsLT[0] >> 1;
197 uint16_t coordsRectT = blobCoordsLT[1] >> 1;
198 uint16_t coordsRectR = blobCoordsRB[0] >> 1;
199 uint16_t coordsRectB = blobCoordsRB[1] >> 1;
200 uint16_t pageIndexX = blobCoordsLT[0] & 0x1;
201 uint16_t pageIndexY = blobCoordsLT[1] & 0x1;
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400202
Jim Van Verth328a33f2017-10-20 12:14:22 -0400203 int positionRectWidth = positionRect.width();
204 int positionRectHeight = positionRect.height();
205 SkASSERT(positionRectWidth == (coordsRectR - coordsRectL));
206 SkASSERT(positionRectHeight == (coordsRectB - coordsRectT));
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400207
208 // Clip position and texCoords to the clipRect
Jim Van Verth328a33f2017-10-20 12:14:22 -0400209 unsigned int delta;
210 delta = SkTMin(SkTMax(clipRect.fLeft - positionRect.fLeft, 0), positionRectWidth);
211 coordsRectL += delta;
212 positionRect.fLeft += delta;
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400213
Jim Van Verth328a33f2017-10-20 12:14:22 -0400214 delta = SkTMin(SkTMax(clipRect.fTop - positionRect.fTop, 0), positionRectHeight);
215 coordsRectT += delta;
216 positionRect.fTop += delta;
217
218 delta = SkTMin(SkTMax(positionRect.fRight - clipRect.fRight, 0), positionRectWidth);
219 coordsRectR -= delta;
220 positionRect.fRight -= delta;
221
222 delta = SkTMin(SkTMax(positionRect.fBottom - clipRect.fBottom, 0), positionRectHeight);
223 coordsRectB -= delta;
224 positionRect.fBottom -= delta;
225
226 // Repack texel coordinates and index
227 coordsRectL = coordsRectL << 1 | pageIndexX;
228 coordsRectT = coordsRectT << 1 | pageIndexY;
229 coordsRectR = coordsRectR << 1 | pageIndexX;
230 coordsRectB = coordsRectB << 1 | pageIndexY;
231
232 // Set new positions and coords
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400233 SkPoint* currPosition = reinterpret_cast<SkPoint*>(currVertex);
234 currPosition->fX = positionRect.fLeft;
235 currPosition->fY = positionRect.fTop;
236 *(reinterpret_cast<SkColor*>(currVertex + sizeof(SkPoint))) = color;
Jim Van Verth328a33f2017-10-20 12:14:22 -0400237 uint16_t* currCoords = reinterpret_cast<uint16_t*>(currVertex + coordOffset);
238 currCoords[0] = coordsRectL;
239 currCoords[1] = coordsRectT;
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400240 currVertex += vertexStride;
241
242 currPosition = reinterpret_cast<SkPoint*>(currVertex);
243 currPosition->fX = positionRect.fLeft;
244 currPosition->fY = positionRect.fBottom;
245 *(reinterpret_cast<SkColor*>(currVertex + sizeof(SkPoint))) = color;
Jim Van Verth328a33f2017-10-20 12:14:22 -0400246 currCoords = reinterpret_cast<uint16_t*>(currVertex + coordOffset);
247 currCoords[0] = coordsRectL;
248 currCoords[1] = coordsRectB;
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400249 currVertex += vertexStride;
250
251 currPosition = reinterpret_cast<SkPoint*>(currVertex);
252 currPosition->fX = positionRect.fRight;
253 currPosition->fY = positionRect.fTop;
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] = coordsRectR;
257 currCoords[1] = coordsRectT;
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.fBottom;
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] = coordsRectB;
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400267 currVertex += vertexStride;
268 }
269
270 blobVertices += 4 * vertexStride;
271 }
272}
273
Brian Salomon91326c32017-08-09 16:02:19 -0400274void GrAtlasTextOp::onPrepareDraws(Target* target) {
Robert Phillips4bc70112018-03-01 10:24:02 -0500275 auto resourceProvider = target->resourceProvider();
276
joshualitta751c972015-11-20 13:37:32 -0800277 // if we have RGB, then we won't have any SkShaders so no need to use a localmatrix.
278 // TODO actually only invert if we don't have RGBA
279 SkMatrix localMatrix;
Brian Salomon5c6ac642017-12-19 11:09:32 -0500280 if (this->usesLocalCoords() && !fGeoData[0].fViewMatrix.invert(&localMatrix)) {
joshualitta751c972015-11-20 13:37:32 -0800281 return;
282 }
283
Robert Phillips5a66efb2018-03-07 15:13:18 -0500284 GrAtlasManager* atlasManager = target->atlasManager();
Herb Derby081e6f32019-01-16 13:46:02 -0500285 GrStrikeCache* glyphCache = target->glyphCache();
Robert Phillipsc4039ea2018-03-01 11:36:45 -0500286
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400287 GrMaskFormat maskFormat = this->maskFormat();
288
Jim Van Verthcbeae032018-05-16 14:54:41 -0400289 unsigned int numActiveProxies;
290 const sk_sp<GrTextureProxy>* proxies = atlasManager->getProxies(maskFormat, &numActiveProxies);
291 if (!proxies) {
joshualitta751c972015-11-20 13:37:32 -0800292 SkDebugf("Could not allocate backing texture for atlas\n");
293 return;
294 }
Jim Van Verthcbeae032018-05-16 14:54:41 -0400295 SkASSERT(proxies[0]);
joshualitta751c972015-11-20 13:37:32 -0800296
Brian Salomon7eae3e02018-08-07 14:02:38 +0000297 static constexpr int kMaxTextures = GrBitmapTextGeoProc::kMaxTextures;
298 GR_STATIC_ASSERT(GrDistanceFieldA8TextGeoProc::kMaxTextures == kMaxTextures);
299 GR_STATIC_ASSERT(GrDistanceFieldLCDTextGeoProc::kMaxTextures == kMaxTextures);
300
Brian Osman9aa30c62018-07-02 15:21:46 -0400301 static const uint32_t kPipelineFlags = 0;
302 auto pipe = target->makePipeline(kPipelineFlags, std::move(fProcessors),
Brian Salomon7eae3e02018-08-07 14:02:38 +0000303 target->detachAppliedClip(), kMaxTextures);
304 for (unsigned i = 0; i < numActiveProxies; ++i) {
305 pipe.fFixedDynamicState->fPrimitiveProcessorTextures[i] = proxies[i].get();
306 }
Brian Salomon49348902018-06-26 09:12:38 -0400307
308 FlushInfo flushInfo;
309 flushInfo.fPipeline = pipe.fPipeline;
310 flushInfo.fFixedDynamicState = pipe.fFixedDynamicState;
311
Jim Van Verthb515ae72018-05-23 16:44:55 -0400312 bool vmPerspective = fGeoData[0].fViewMatrix.hasPerspective();
joshualittd9d30f72015-12-08 10:47:55 -0800313 if (this->usesDistanceFields()) {
Brian Osman4a3f5c82018-09-18 16:16:38 -0400314 flushInfo.fGeometryProcessor = this->setupDfProcessor(*target->caps().shaderCaps(),
315 proxies, numActiveProxies);
joshualitta751c972015-11-20 13:37:32 -0800316 } else {
Jim Van Verthb515ae72018-05-23 16:44:55 -0400317 GrSamplerState samplerState = fNeedsGlyphTransform ? GrSamplerState::ClampBilerp()
318 : GrSamplerState::ClampNearest();
Brian Salomon2bbdcc42017-09-07 12:36:34 -0400319 flushInfo.fGeometryProcessor = GrBitmapTextGeoProc::Make(
Brian Osmanc906d252018-12-04 11:17:46 -0500320 *target->caps().shaderCaps(), this->color(), false, proxies, numActiveProxies,
321 samplerState, maskFormat, localMatrix, vmPerspective);
joshualitta751c972015-11-20 13:37:32 -0800322 }
323
joshualitta751c972015-11-20 13:37:32 -0800324 flushInfo.fGlyphsToFlush = 0;
Brian Osmanf04fb3c2018-11-12 15:34:00 -0500325 size_t vertexStride = flushInfo.fGeometryProcessor->vertexStride();
joshualitta751c972015-11-20 13:37:32 -0800326
joshualitta751c972015-11-20 13:37:32 -0800327 int glyphCount = this->numGlyphs();
joshualitta751c972015-11-20 13:37:32 -0800328
Brian Salomon12d22642019-01-29 14:38:50 -0500329 void* vertices = target->makeVertexSpace(vertexStride, glyphCount * kVerticesPerGlyph,
330 &flushInfo.fVertexBuffer, &flushInfo.fVertexOffset);
Brian Salomond28a79d2017-10-16 13:01:07 -0400331 flushInfo.fIndexBuffer = target->resourceProvider()->refQuadIndexBuffer();
joshualitta751c972015-11-20 13:37:32 -0800332 if (!vertices || !flushInfo.fVertexBuffer) {
333 SkDebugf("Could not allocate vertices\n");
334 return;
335 }
336
Brian Salomon18923f92017-11-06 16:26:02 -0500337 char* currVertex = reinterpret_cast<char*>(vertices);
joshualitta751c972015-11-20 13:37:32 -0800338
Herb Derby7956b592018-03-22 16:10:30 -0400339 SkExclusiveStrikePtr autoGlyphCache;
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400340 // each of these is a SubRun
joshualitta751c972015-11-20 13:37:32 -0800341 for (int i = 0; i < fGeoCount; i++) {
joshualitt144c3c82015-11-30 12:30:13 -0800342 const Geometry& args = fGeoData[i];
joshualitta751c972015-11-20 13:37:32 -0800343 Blob* blob = args.fBlob;
Brian Osman1be2b7c2018-10-29 16:07:15 -0400344 // TODO4F: Preserve float colors
Herb Derby86240592018-05-24 16:12:31 -0400345 GrTextBlob::VertexRegenerator regenerator(
Robert Phillips4bc70112018-03-01 10:24:02 -0500346 resourceProvider, blob, args.fRun, args.fSubRun, args.fViewMatrix, args.fX, args.fY,
Brian Osmancf860852018-10-31 14:04:39 -0400347 args.fColor.toBytes_RGBA(), target->deferredUploadTarget(), glyphCache,
348 atlasManager, &autoGlyphCache);
Robert Phillipsd2e9f762018-03-07 11:54:37 -0500349 bool done = false;
350 while (!done) {
Herb Derby86240592018-05-24 16:12:31 -0400351 GrTextBlob::VertexRegenerator::Result result;
Robert Phillipsd2e9f762018-03-07 11:54:37 -0500352 if (!regenerator.regenerate(&result)) {
353 break;
354 }
355 done = result.fFinished;
356
Brian Salomon18923f92017-11-06 16:26:02 -0500357 // Copy regenerated vertices from the blob to our vertex buffer.
358 size_t vertexBytes = result.fGlyphsRegenerated * kVerticesPerGlyph * vertexStride;
359 if (args.fClipRect.isEmpty()) {
360 memcpy(currVertex, result.fFirstVertex, vertexBytes);
361 } else {
Jim Van Verthb515ae72018-05-23 16:44:55 -0400362 SkASSERT(!vmPerspective);
Brian Salomon18923f92017-11-06 16:26:02 -0500363 clip_quads(args.fClipRect, currVertex, result.fFirstVertex, vertexStride,
364 result.fGlyphsRegenerated);
365 }
Jim Van Verthb515ae72018-05-23 16:44:55 -0400366 if (fNeedsGlyphTransform && !args.fViewMatrix.isIdentity()) {
Brian Salomon5c6ac642017-12-19 11:09:32 -0500367 // We always do the distance field view matrix transformation after copying rather
368 // than during blob vertex generation time in the blob as handling successive
369 // arbitrary transformations would be complicated and accumulate error.
370 if (args.fViewMatrix.hasPerspective()) {
371 auto* pos = reinterpret_cast<SkPoint3*>(currVertex);
Brian Salomon7c2192b2018-01-08 09:47:57 -0500372 SkMatrixPriv::MapHomogeneousPointsWithStride(
373 args.fViewMatrix, pos, vertexStride, pos, vertexStride,
374 result.fGlyphsRegenerated * kVerticesPerGlyph);
Brian Salomon5c6ac642017-12-19 11:09:32 -0500375 } else {
376 auto* pos = reinterpret_cast<SkPoint*>(currVertex);
Brian Salomonfa3783f2018-01-05 13:49:07 -0500377 SkMatrixPriv::MapPointsWithStride(
378 args.fViewMatrix, pos, vertexStride,
379 result.fGlyphsRegenerated * kVerticesPerGlyph);
Brian Salomon5c6ac642017-12-19 11:09:32 -0500380 }
381 }
Brian Salomon18923f92017-11-06 16:26:02 -0500382 flushInfo.fGlyphsToFlush += result.fGlyphsRegenerated;
383 if (!result.fFinished) {
384 this->flush(target, &flushInfo);
385 }
386 currVertex += vertexBytes;
Robert Phillipsd2e9f762018-03-07 11:54:37 -0500387 }
joshualitta751c972015-11-20 13:37:32 -0800388 }
joshualitta751c972015-11-20 13:37:32 -0800389 this->flush(target, &flushInfo);
390}
391
Brian Salomone5b399e2017-07-19 13:50:54 -0400392void GrAtlasTextOp::flush(GrMeshDrawOp::Target* target, FlushInfo* flushInfo) const {
Robert Phillipsd2e9f762018-03-07 11:54:37 -0500393 if (!flushInfo->fGlyphsToFlush) {
394 return;
395 }
396
Robert Phillips5a66efb2018-03-07 15:13:18 -0500397 auto atlasManager = target->atlasManager();
Robert Phillipsc4039ea2018-03-01 11:36:45 -0500398
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400399 GrGeometryProcessor* gp = flushInfo->fGeometryProcessor.get();
400 GrMaskFormat maskFormat = this->maskFormat();
Robert Phillipsf3690dd2018-02-20 15:18:59 -0500401
Jim Van Verthcbeae032018-05-16 14:54:41 -0400402 unsigned int numActiveProxies;
403 const sk_sp<GrTextureProxy>* proxies = atlasManager->getProxies(maskFormat, &numActiveProxies);
404 SkASSERT(proxies);
405 if (gp->numTextureSamplers() != (int) numActiveProxies) {
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400406 // During preparation the number of atlas pages has increased.
407 // Update the proxies used in the GP to match.
Brian Salomon7eae3e02018-08-07 14:02:38 +0000408 for (unsigned i = gp->numTextureSamplers(); i < numActiveProxies; ++i) {
409 flushInfo->fFixedDynamicState->fPrimitiveProcessorTextures[i] = proxies[i].get();
410 }
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400411 if (this->usesDistanceFields()) {
412 if (this->isLCD()) {
413 reinterpret_cast<GrDistanceFieldLCDTextGeoProc*>(gp)->addNewProxies(
Jim Van Verthcbeae032018-05-16 14:54:41 -0400414 proxies, numActiveProxies, GrSamplerState::ClampBilerp());
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400415 } else {
416 reinterpret_cast<GrDistanceFieldA8TextGeoProc*>(gp)->addNewProxies(
Jim Van Verthcbeae032018-05-16 14:54:41 -0400417 proxies, numActiveProxies, GrSamplerState::ClampBilerp());
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400418 }
419 } else {
Jim Van Verthb515ae72018-05-23 16:44:55 -0400420 GrSamplerState samplerState = fNeedsGlyphTransform ? GrSamplerState::ClampBilerp()
421 : GrSamplerState::ClampNearest();
Jim Van Verthcbeae032018-05-16 14:54:41 -0400422 reinterpret_cast<GrBitmapTextGeoProc*>(gp)->addNewProxies(proxies, numActiveProxies,
Jim Van Verthcf838c72018-03-05 14:40:36 -0500423 samplerState);
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400424 }
425 }
cdalton397536c2016-03-25 12:15:03 -0700426 int maxGlyphsPerDraw =
Brian Salomon344ec422016-12-15 10:58:41 -0500427 static_cast<int>(flushInfo->fIndexBuffer->gpuMemorySize() / sizeof(uint16_t) / 6);
Brian Salomon7eae3e02018-08-07 14:02:38 +0000428 GrMesh* mesh = target->allocMesh(GrPrimitiveType::kTriangles);
Brian Salomon12d22642019-01-29 14:38:50 -0500429 mesh->setIndexedPatterned(flushInfo->fIndexBuffer, kIndicesPerGlyph, kVerticesPerGlyph,
Brian Salomon7eae3e02018-08-07 14:02:38 +0000430 flushInfo->fGlyphsToFlush, maxGlyphsPerDraw);
Brian Salomon12d22642019-01-29 14:38:50 -0500431 mesh->setVertexData(flushInfo->fVertexBuffer, flushInfo->fVertexOffset);
Brian Salomon7eae3e02018-08-07 14:02:38 +0000432 target->draw(flushInfo->fGeometryProcessor, flushInfo->fPipeline, flushInfo->fFixedDynamicState,
433 mesh);
joshualitta751c972015-11-20 13:37:32 -0800434 flushInfo->fVertexOffset += kVerticesPerGlyph * flushInfo->fGlyphsToFlush;
435 flushInfo->fGlyphsToFlush = 0;
436}
437
Brian Salomon7eae3e02018-08-07 14:02:38 +0000438GrOp::CombineResult GrAtlasTextOp::onCombineIfPossible(GrOp* t, const GrCaps& caps) {
Brian Salomon344ec422016-12-15 10:58:41 -0500439 GrAtlasTextOp* that = t->cast<GrAtlasTextOp>();
Brian Salomon44acb5b2017-07-18 19:59:24 -0400440 if (fProcessors != that->fProcessors) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000441 return CombineResult::kCannotCombine;
Brian Salomon44acb5b2017-07-18 19:59:24 -0400442 }
443
joshualitta751c972015-11-20 13:37:32 -0800444 if (fMaskType != that->fMaskType) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000445 return CombineResult::kCannotCombine;
joshualitta751c972015-11-20 13:37:32 -0800446 }
447
Brian Salomon5c6ac642017-12-19 11:09:32 -0500448 const SkMatrix& thisFirstMatrix = fGeoData[0].fViewMatrix;
449 const SkMatrix& thatFirstMatrix = that->fGeoData[0].fViewMatrix;
450
451 if (this->usesLocalCoords() && !thisFirstMatrix.cheapEqualTo(thatFirstMatrix)) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000452 return CombineResult::kCannotCombine;
Brian Salomon5c6ac642017-12-19 11:09:32 -0500453 }
454
Jim Van Verthb515ae72018-05-23 16:44:55 -0400455 if (fNeedsGlyphTransform != that->fNeedsGlyphTransform) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000456 return CombineResult::kCannotCombine;
Jim Van Verthb515ae72018-05-23 16:44:55 -0400457 }
458
459 if (fNeedsGlyphTransform &&
460 (thisFirstMatrix.hasPerspective() != thatFirstMatrix.hasPerspective())) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000461 return CombineResult::kCannotCombine;
Jim Van Verthb515ae72018-05-23 16:44:55 -0400462 }
463
Brian Salomon5c6ac642017-12-19 11:09:32 -0500464 if (this->usesDistanceFields()) {
465 if (fDFGPFlags != that->fDFGPFlags) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000466 return CombineResult::kCannotCombine;
joshualitta751c972015-11-20 13:37:32 -0800467 }
468
Jim Van Verthbc2cdd12017-06-08 11:14:35 -0400469 if (fLuminanceColor != that->fLuminanceColor) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000470 return CombineResult::kCannotCombine;
joshualitta751c972015-11-20 13:37:32 -0800471 }
Brian Salomon5c6ac642017-12-19 11:09:32 -0500472 } else {
473 if (kColorBitmapMask_MaskType == fMaskType && this->color() != that->color()) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000474 return CombineResult::kCannotCombine;
Brian Salomon5c6ac642017-12-19 11:09:32 -0500475 }
joshualitta751c972015-11-20 13:37:32 -0800476 }
477
Jim Van Verthc8a65e32017-10-25 14:25:27 -0400478 // Keep the batch vertex buffer size below 32K so we don't have to create a special one
479 // We use the largest possible vertex size for this
480 static const int kVertexSize = sizeof(SkPoint) + sizeof(SkColor) + 2 * sizeof(uint16_t);
Jim Van Verth56c37142017-10-31 14:44:25 -0400481 static const int kMaxGlyphs = 32768 / (kVerticesPerGlyph * kVertexSize);
Jim Van Verthc8a65e32017-10-25 14:25:27 -0400482 if (this->fNumGlyphs + that->fNumGlyphs > kMaxGlyphs) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000483 return CombineResult::kCannotCombine;
Jim Van Verthc8a65e32017-10-25 14:25:27 -0400484 }
485
Brian Salomon344ec422016-12-15 10:58:41 -0500486 fNumGlyphs += that->numGlyphs();
joshualitta751c972015-11-20 13:37:32 -0800487
Jim Van Verth56c37142017-10-31 14:44:25 -0400488 // Reallocate space for geo data if necessary and then import that geo's data.
joshualitta751c972015-11-20 13:37:32 -0800489 int newGeoCount = that->fGeoCount + fGeoCount;
joshualitta751c972015-11-20 13:37:32 -0800490
Jim Van Verthc8a65e32017-10-25 14:25:27 -0400491 // We reallocate at a rate of 1.5x to try to get better total memory usage
492 if (newGeoCount > fGeoDataAllocSize) {
Jim Van Verth56c37142017-10-31 14:44:25 -0400493 int newAllocSize = fGeoDataAllocSize + fGeoDataAllocSize / 2;
Jim Van Verthc8a65e32017-10-25 14:25:27 -0400494 while (newAllocSize < newGeoCount) {
495 newAllocSize += newAllocSize / 2;
496 }
joshualitta751c972015-11-20 13:37:32 -0800497 fGeoData.realloc(newAllocSize);
Jim Van Verthc8a65e32017-10-25 14:25:27 -0400498 fGeoDataAllocSize = newAllocSize;
joshualitta751c972015-11-20 13:37:32 -0800499 }
500
Brian Salomon344ec422016-12-15 10:58:41 -0500501 // 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 -0800502 // it doesn't try to unref them.
Brian Salomon344ec422016-12-15 10:58:41 -0500503 memcpy(&fGeoData[fGeoCount], that->fGeoData.get(), that->fGeoCount * sizeof(Geometry));
joshualitta751c972015-11-20 13:37:32 -0800504#ifdef SK_DEBUG
505 for (int i = 0; i < that->fGeoCount; ++i) {
506 that->fGeoData.get()[i].fBlob = (Blob*)0x1;
507 }
508#endif
509 that->fGeoCount = 0;
510 fGeoCount = newGeoCount;
511
Brian Salomon7eae3e02018-08-07 14:02:38 +0000512 return CombineResult::kMerged;
joshualitta751c972015-11-20 13:37:32 -0800513}
514
joshualitta751c972015-11-20 13:37:32 -0800515// TODO trying to figure out why lcd is so whack
Herb Derby26cbe512018-05-24 14:39:01 -0400516// (see comments in GrTextContext::ComputeCanonicalColor)
Brian Osman4a3f5c82018-09-18 16:16:38 -0400517sk_sp<GrGeometryProcessor> GrAtlasTextOp::setupDfProcessor(const GrShaderCaps& caps,
518 const sk_sp<GrTextureProxy>* proxies,
Jim Van Verthcbeae032018-05-16 14:54:41 -0400519 unsigned int numActiveProxies) const {
joshualitta751c972015-11-20 13:37:32 -0800520 bool isLCD = this->isLCD();
Brian Salomon5c6ac642017-12-19 11:09:32 -0500521
522 SkMatrix localMatrix = SkMatrix::I();
523 if (this->usesLocalCoords()) {
524 // If this fails we'll just use I().
525 bool result = fGeoData[0].fViewMatrix.invert(&localMatrix);
526 (void)result;
527 }
joshualitta751c972015-11-20 13:37:32 -0800528
529 // see if we need to create a new effect
530 if (isLCD) {
brianosman0586f5c2016-04-12 12:48:21 -0700531 float redCorrection = fDistanceAdjustTable->getAdjustment(
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400532 SkColorGetR(fLuminanceColor) >> kDistanceAdjustLumShift,
Brian Salomon344ec422016-12-15 10:58:41 -0500533 fUseGammaCorrectDistanceTable);
brianosman0586f5c2016-04-12 12:48:21 -0700534 float greenCorrection = fDistanceAdjustTable->getAdjustment(
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400535 SkColorGetG(fLuminanceColor) >> kDistanceAdjustLumShift,
Brian Salomon344ec422016-12-15 10:58:41 -0500536 fUseGammaCorrectDistanceTable);
brianosman0586f5c2016-04-12 12:48:21 -0700537 float blueCorrection = fDistanceAdjustTable->getAdjustment(
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400538 SkColorGetB(fLuminanceColor) >> kDistanceAdjustLumShift,
Brian Salomon344ec422016-12-15 10:58:41 -0500539 fUseGammaCorrectDistanceTable);
joshualitta751c972015-11-20 13:37:32 -0800540 GrDistanceFieldLCDTextGeoProc::DistanceAdjust widthAdjust =
Brian Salomon344ec422016-12-15 10:58:41 -0500541 GrDistanceFieldLCDTextGeoProc::DistanceAdjust::Make(
542 redCorrection, greenCorrection, blueCorrection);
Brian Osman4a3f5c82018-09-18 16:16:38 -0400543 return GrDistanceFieldLCDTextGeoProc::Make(caps, proxies, numActiveProxies,
Robert Phillips4bc70112018-03-01 10:24:02 -0500544 GrSamplerState::ClampBilerp(), widthAdjust,
Brian Osman09068252018-01-03 09:57:29 -0500545 fDFGPFlags, localMatrix);
joshualitta751c972015-11-20 13:37:32 -0800546 } else {
joshualitta751c972015-11-20 13:37:32 -0800547#ifdef SK_GAMMA_APPLY_TO_A8
Jim Van Verth90e89b32017-07-06 16:36:55 -0400548 float correction = 0;
549 if (kAliasedDistanceField_MaskType != fMaskType) {
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400550 U8CPU lum = SkColorSpaceLuminance::computeLuminance(SK_GAMMA_EXPONENT,
551 fLuminanceColor);
Jim Van Verth90e89b32017-07-06 16:36:55 -0400552 correction = fDistanceAdjustTable->getAdjustment(lum >> kDistanceAdjustLumShift,
553 fUseGammaCorrectDistanceTable);
554 }
Brian Osman4a3f5c82018-09-18 16:16:38 -0400555 return GrDistanceFieldA8TextGeoProc::Make(caps, proxies, numActiveProxies,
Robert Phillips4bc70112018-03-01 10:24:02 -0500556 GrSamplerState::ClampBilerp(),
Brian Salomon5c6ac642017-12-19 11:09:32 -0500557 correction, fDFGPFlags, localMatrix);
joshualitta751c972015-11-20 13:37:32 -0800558#else
Brian Osman4a3f5c82018-09-18 16:16:38 -0400559 return GrDistanceFieldA8TextGeoProc::Make(caps, proxies, numActiveProxies,
Robert Phillips4bc70112018-03-01 10:24:02 -0500560 GrSamplerState::ClampBilerp(),
Brian Salomon5c6ac642017-12-19 11:09:32 -0500561 fDFGPFlags, localMatrix);
joshualitta751c972015-11-20 13:37:32 -0800562#endif
563 }
joshualitta751c972015-11-20 13:37:32 -0800564}
joshualittddd22d82016-02-16 06:47:52 -0800565