blob: ec175c7ea4393952fed50be405032d9cc9f84db0 [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"
Robert Phillipsb7bfbc22020-07-01 12:55:01 -040011#include "include/gpu/GrRecordingContext.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "src/core/SkMathPriv.h"
13#include "src/core/SkMatrixPriv.h"
Herb Derby7a1d9422020-07-06 14:18:01 -040014#include "src/core/SkMatrixProvider.h"
Herb Derbyfd894ff2020-07-15 13:23:33 -040015#include "src/core/SkSpan.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050016#include "src/core/SkStrikeCache.h"
17#include "src/gpu/GrCaps.h"
18#include "src/gpu/GrMemoryPool.h"
19#include "src/gpu/GrOpFlushState.h"
20#include "src/gpu/GrRecordingContextPriv.h"
Herb Derby4598fa12020-06-10 14:54:22 -040021#include "src/gpu/GrRenderTargetContext.h"
Herb Derby7a1d9422020-07-06 14:18:01 -040022#include "src/gpu/GrRenderTargetContextPriv.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050023#include "src/gpu/GrResourceProvider.h"
Herb Derby7a1d9422020-07-06 14:18:01 -040024#include "src/gpu/SkGr.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050025#include "src/gpu/effects/GrBitmapTextGeoProc.h"
26#include "src/gpu/effects/GrDistanceFieldGeoProc.h"
Robert Phillips3968fcb2019-12-05 16:40:31 -050027#include "src/gpu/ops/GrSimpleMeshDrawOpHelper.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050028#include "src/gpu/text/GrAtlasManager.h"
Robert Phillips841c9a52020-03-27 12:41:31 -040029#include "src/gpu/text/GrDistanceFieldAdjustTable.h"
joshualitta751c972015-11-20 13:37:32 -080030
Herb Derbya08bde62020-06-12 15:46:06 -040031#if GR_TEST_UTILS
32#include "src/gpu/GrDrawOpTest.h"
33#endif
34
joshualitt60ce86d2015-11-23 13:08:22 -080035///////////////////////////////////////////////////////////////////////////////////////////////////
36
Herb Derby3c873af2020-04-29 15:56:07 -040037GrAtlasTextOp::GrAtlasTextOp(MaskType maskType,
Herb Derby268e48b2020-07-16 12:56:58 -040038 bool needsTransform,
39 int glyphCount,
40 SkRect deviceRect,
Herb Derby1d17e492020-07-21 11:45:04 -040041 const Geometry& geo,
42 GrPaint&& paint)
Herb Derby268e48b2020-07-16 12:56:58 -040043 : INHERITED{ClassID()}
44 , fMaskType{maskType}
45 , fNeedsGlyphTransform{needsTransform}
46 , fLuminanceColor{0}
47 , fUseGammaCorrectDistanceTable{false}
48 , fDFGPFlags{0}
49 , fGeoDataAllocSize{kMinGeometryAllocated}
50 , fProcessors{std::move(paint)}
51 , fNumGlyphs{glyphCount} {
Herb Derby1d17e492020-07-21 11:45:04 -040052 new (&fGeoData[0]) Geometry{geo};
Herb Derby268e48b2020-07-16 12:56:58 -040053 fGeoCount = 1;
54
55 // We don't have tight bounds on the glyph paths in device space. For the purposes of bounds
56 // we treat this as a set of non-AA rects rendered with a texture.
57 this->setBounds(deviceRect, HasAABloat::kNo, IsHairline::kNo);
58}
59
60GrAtlasTextOp::GrAtlasTextOp(MaskType maskType,
61 bool needsTransform,
62 int glyphCount,
63 SkRect deviceRect,
Herb Derby3c873af2020-04-29 15:56:07 -040064 SkColor luminanceColor,
65 bool useGammaCorrectDistanceTable,
Herb Derby268e48b2020-07-16 12:56:58 -040066 uint32_t DFGPFlags,
Herb Derby1d17e492020-07-21 11:45:04 -040067 const Geometry& geo,
68 GrPaint&& paint)
Herb Derby268e48b2020-07-16 12:56:58 -040069 : INHERITED{ClassID()}
Herb Derby3c873af2020-04-29 15:56:07 -040070 , fMaskType{maskType}
Herb Derby268e48b2020-07-16 12:56:58 -040071 , fNeedsGlyphTransform{needsTransform}
Herb Derby3c873af2020-04-29 15:56:07 -040072 , fLuminanceColor{luminanceColor}
73 , fUseGammaCorrectDistanceTable{useGammaCorrectDistanceTable}
74 , fDFGPFlags{DFGPFlags}
75 , fGeoDataAllocSize{kMinGeometryAllocated}
76 , fProcessors{std::move(paint)}
Herb Derby268e48b2020-07-16 12:56:58 -040077 , fNumGlyphs{glyphCount} {
Herb Derby1d17e492020-07-21 11:45:04 -040078 new (&fGeoData[0]) Geometry{geo};
Herb Derby3c873af2020-04-29 15:56:07 -040079 fGeoCount = 1;
80
Herb Derby3c873af2020-04-29 15:56:07 -040081 // We don't have tight bounds on the glyph paths in device space. For the purposes of bounds
82 // we treat this as a set of non-AA rects rendered with a texture.
Herb Derby268e48b2020-07-16 12:56:58 -040083 this->setBounds(deviceRect, HasAABloat::kNo, IsHairline::kNo);
Herb Derby3c873af2020-04-29 15:56:07 -040084}
85
Herb Derby64391c42020-05-16 14:32:15 -040086void GrAtlasTextOp::Geometry::fillVertexData(void *dst, int offset, int count) const {
Herb Derby43ad7912020-07-20 16:14:19 -040087 fSubRun.fillVertexData(dst, offset, count, fColor.toBytes_RGBA(),
88 fDrawMatrix, fDrawOrigin, fClipRect);
Herb Derby64391c42020-05-16 14:32:15 -040089}
90
Chris Dalton1706cbf2019-05-21 19:35:29 -060091void GrAtlasTextOp::visitProxies(const VisitProxyFunc& func) const {
Robert Phillipse4fda6c2018-02-21 12:10:41 -050092 fProcessors.visitProxies(func);
Robert Phillipse4fda6c2018-02-21 12:10:41 -050093}
94
John Stiles8d9bf642020-08-12 15:07:45 -040095#if GR_TEST_UTILS
John Stiles8dd1e222020-08-12 19:06:24 -040096SkString GrAtlasTextOp::onDumpInfo() const {
joshualitta751c972015-11-20 13:37:32 -080097 SkString str;
98
99 for (int i = 0; i < fGeoCount; ++i) {
Herb Derby1b8dcd12019-11-15 15:21:15 -0500100 str.appendf("%d: Color: 0x%08x Trans: %.2f,%.2f\n",
joshualitta751c972015-11-20 13:37:32 -0800101 i,
Brian Osmancf860852018-10-31 14:04:39 -0400102 fGeoData[i].fColor.toBytes_RGBA(),
Herb Derby5bf5b042019-12-12 16:37:03 -0500103 fGeoData[i].fDrawOrigin.x(),
104 fGeoData[i].fDrawOrigin.y());
joshualitta751c972015-11-20 13:37:32 -0800105 }
106
Brian Salomon44acb5b2017-07-18 19:59:24 -0400107 str += fProcessors.dumpProcessors();
joshualitta751c972015-11-20 13:37:32 -0800108 return str;
109}
Brian Osman9a390ac2018-11-12 09:47:48 -0500110#endif
joshualitta751c972015-11-20 13:37:32 -0800111
Brian Salomon44acb5b2017-07-18 19:59:24 -0400112GrDrawOp::FixedFunctionFlags GrAtlasTextOp::fixedFunctionFlags() const {
113 return FixedFunctionFlags::kNone;
114}
115
Chris Dalton6ce447a2019-06-23 18:07:38 -0600116GrProcessorSet::Analysis GrAtlasTextOp::finalize(
117 const GrCaps& caps, const GrAppliedClip* clip, bool hasMixedSampledCoverage,
118 GrClampType clampType) {
Brian Salomon44acb5b2017-07-18 19:59:24 -0400119 GrProcessorAnalysisCoverage coverage;
120 GrProcessorAnalysisColor color;
joshualitta751c972015-11-20 13:37:32 -0800121 if (kColorBitmapMask_MaskType == fMaskType) {
Brian Salomon44acb5b2017-07-18 19:59:24 -0400122 color.setToUnknown();
joshualitta751c972015-11-20 13:37:32 -0800123 } else {
Brian Osman09068252018-01-03 09:57:29 -0500124 color.setToConstant(this->color());
joshualitta751c972015-11-20 13:37:32 -0800125 }
joshualitta751c972015-11-20 13:37:32 -0800126 switch (fMaskType) {
joshualitta751c972015-11-20 13:37:32 -0800127 case kGrayscaleCoverageMask_MaskType:
Jim Van Verth90e89b32017-07-06 16:36:55 -0400128 case kAliasedDistanceField_MaskType:
129 case kGrayscaleDistanceField_MaskType:
Brian Salomon44acb5b2017-07-18 19:59:24 -0400130 coverage = GrProcessorAnalysisCoverage::kSingleChannel;
joshualitta751c972015-11-20 13:37:32 -0800131 break;
132 case kLCDCoverageMask_MaskType:
133 case kLCDDistanceField_MaskType:
Jim Van Verth90e89b32017-07-06 16:36:55 -0400134 case kLCDBGRDistanceField_MaskType:
Brian Salomon44acb5b2017-07-18 19:59:24 -0400135 coverage = GrProcessorAnalysisCoverage::kLCD;
joshualitta751c972015-11-20 13:37:32 -0800136 break;
137 case kColorBitmapMask_MaskType:
Brian Salomon44acb5b2017-07-18 19:59:24 -0400138 coverage = GrProcessorAnalysisCoverage::kNone;
Brian Salomonc0b642c2017-03-27 13:09:36 -0400139 break;
joshualitta751c972015-11-20 13:37:32 -0800140 }
Chris Daltonb8fff0d2019-03-05 10:11:58 -0700141 auto analysis = fProcessors.finalize(
Chris Dalton6ce447a2019-06-23 18:07:38 -0600142 color, coverage, clip, &GrUserStencilSettings::kUnused, hasMixedSampledCoverage, caps,
143 clampType, &fGeoData[0].fColor);
Brian Salomon44acb5b2017-07-18 19:59:24 -0400144 fUsesLocalCoords = analysis.usesLocalCoords();
Chris Dalton4b62aed2019-01-15 11:53:00 -0700145 return analysis;
joshualitta751c972015-11-20 13:37:32 -0800146}
147
Brian Salomon91326c32017-08-09 16:02:19 -0400148void GrAtlasTextOp::onPrepareDraws(Target* target) {
Robert Phillips4bc70112018-03-01 10:24:02 -0500149 auto resourceProvider = target->resourceProvider();
150
joshualitta751c972015-11-20 13:37:32 -0800151 // if we have RGB, then we won't have any SkShaders so no need to use a localmatrix.
152 // TODO actually only invert if we don't have RGBA
153 SkMatrix localMatrix;
Herb Derby1c5be7b2019-12-13 12:03:06 -0500154 if (this->usesLocalCoords() && !fGeoData[0].fDrawMatrix.invert(&localMatrix)) {
joshualitta751c972015-11-20 13:37:32 -0800155 return;
156 }
157
Robert Phillips5a66efb2018-03-07 15:13:18 -0500158 GrAtlasManager* atlasManager = target->atlasManager();
Mike Klein99e002f2020-01-16 16:45:03 +0000159
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400160 GrMaskFormat maskFormat = this->maskFormat();
161
Greg Daniel9715b6c2019-12-10 15:03:10 -0500162 unsigned int numActiveViews;
163 const GrSurfaceProxyView* views = atlasManager->getViews(maskFormat, &numActiveViews);
164 if (!views) {
joshualitta751c972015-11-20 13:37:32 -0800165 SkDebugf("Could not allocate backing texture for atlas\n");
166 return;
167 }
Greg Daniel9715b6c2019-12-10 15:03:10 -0500168 SkASSERT(views[0].proxy());
joshualitta751c972015-11-20 13:37:32 -0800169
Brian Salomon7eae3e02018-08-07 14:02:38 +0000170 static constexpr int kMaxTextures = GrBitmapTextGeoProc::kMaxTextures;
Brian Salomon4dea72a2019-12-18 10:43:10 -0500171 static_assert(GrDistanceFieldA8TextGeoProc::kMaxTextures == kMaxTextures);
172 static_assert(GrDistanceFieldLCDTextGeoProc::kMaxTextures == kMaxTextures);
Brian Salomon7eae3e02018-08-07 14:02:38 +0000173
Chris Dalton304e14d2020-03-17 14:29:06 -0600174 auto primProcProxies = target->allocPrimProcProxyPtrs(kMaxTextures);
Greg Daniel9715b6c2019-12-10 15:03:10 -0500175 for (unsigned i = 0; i < numActiveViews; ++i) {
Chris Dalton304e14d2020-03-17 14:29:06 -0600176 primProcProxies[i] = views[i].proxy();
Greg Danielb20d7e52019-09-03 13:54:39 -0400177 // This op does not know its atlas proxies when it is added to a GrOpsTasks, so the proxies
178 // don't get added during the visitProxies call. Thus we add them here.
Greg Daniel9715b6c2019-12-10 15:03:10 -0500179 target->sampledProxyArray()->push_back(views[i].proxy());
Brian Salomon7eae3e02018-08-07 14:02:38 +0000180 }
Brian Salomon49348902018-06-26 09:12:38 -0400181
182 FlushInfo flushInfo;
Chris Dalton304e14d2020-03-17 14:29:06 -0600183 flushInfo.fPrimProcProxies = primProcProxies;
Herb Derbyfd894ff2020-07-15 13:23:33 -0400184 flushInfo.fIndexBuffer = resourceProvider->refNonAAQuadIndexBuffer();
Brian Salomon49348902018-06-26 09:12:38 -0400185
Herb Derby1c5be7b2019-12-13 12:03:06 -0500186 bool vmPerspective = fGeoData[0].fDrawMatrix.hasPerspective();
joshualittd9d30f72015-12-08 10:47:55 -0800187 if (this->usesDistanceFields()) {
Robert Phillips7cd0bfe2019-11-20 16:08:10 -0500188 flushInfo.fGeometryProcessor = this->setupDfProcessor(target->allocator(),
189 *target->caps().shaderCaps(),
Greg Daniel9715b6c2019-12-10 15:03:10 -0500190 views, numActiveViews);
joshualitta751c972015-11-20 13:37:32 -0800191 } else {
Brian Salomona3b02f52020-07-15 16:02:01 -0400192 auto filter = fNeedsGlyphTransform ? GrSamplerState::Filter::kLinear
Brian Salomonccb61422020-01-09 10:46:36 -0500193 : GrSamplerState::Filter::kNearest;
194 flushInfo.fGeometryProcessor = GrBitmapTextGeoProc::Make(
195 target->allocator(), *target->caps().shaderCaps(), this->color(), false, views,
196 numActiveViews, filter, maskFormat, localMatrix, vmPerspective);
joshualitta751c972015-11-20 13:37:32 -0800197 }
198
Herb Derbyfd894ff2020-07-15 13:23:33 -0400199 const int vertexStride = (int)flushInfo.fGeometryProcessor->vertexStride();
joshualitta751c972015-11-20 13:37:32 -0800200
Brian Salomon43cbd722020-01-03 22:09:12 -0500201 // Ensure we don't request an insanely large contiguous vertex allocation.
Herb Derby23f29762020-01-10 16:26:14 -0500202 static const int kMaxVertexBytes = GrBufferAllocPool::kDefaultBufferSize;
203 const int quadSize = vertexStride * kVerticesPerGlyph;
204 const int maxQuadsPerBuffer = kMaxVertexBytes / quadSize;
205
Herb Derbyfd894ff2020-07-15 13:23:33 -0400206 int allGlyphsCursor = 0;
207 const int allGlyphsEnd = this->numGlyphs();
208 int quadCursor;
209 int quadEnd;
210 char* vertices;
Herb Derby23f29762020-01-10 16:26:14 -0500211
Herb Derbyfd894ff2020-07-15 13:23:33 -0400212 auto resetVertexBuffer = [&] {
213 quadCursor = 0;
214 quadEnd = std::min(maxQuadsPerBuffer, allGlyphsEnd - allGlyphsCursor);
joshualitta751c972015-11-20 13:37:32 -0800215
Herb Derbyfd894ff2020-07-15 13:23:33 -0400216 vertices = (char*)target->makeVertexSpace(
217 vertexStride,
218 kVerticesPerGlyph * quadEnd,
219 &flushInfo.fVertexBuffer,
220 &flushInfo.fVertexOffset);
221
222 if (!vertices || !flushInfo.fVertexBuffer) {
223 SkDebugf("Could not allocate vertices\n");
224 return false;
225 }
226 return true;
227 };
228
229 resetVertexBuffer();
230
Herb Derbya80ce1a2020-05-26 12:58:59 -0400231 for (const Geometry& geo : SkSpan(fGeoData.get(), fGeoCount)) {
Herb Derby43ad7912020-07-20 16:14:19 -0400232 const GrAtlasSubRun& subRun = geo.fSubRun;
233 SkASSERT((int)subRun.vertexStride() == vertexStride);
Herb Derby62b12fe2020-01-14 17:57:24 -0500234
Herb Derby43ad7912020-07-20 16:14:19 -0400235 const int subRunEnd = subRun.glyphCount();
Herb Derbyfd894ff2020-07-15 13:23:33 -0400236 for (int subRunCursor = 0; subRunCursor < subRunEnd;) {
237 // Regenerate the atlas for the remainder of the glyphs in the run, or the remainder
238 // of the glyphs to fill the vertex buffer.
239 int regenEnd = subRunCursor + std::min(subRunEnd - subRunCursor, quadEnd - quadCursor);
Herb Derby43ad7912020-07-20 16:14:19 -0400240 auto[ok, glyphsRegenerated] = subRun.regenerateAtlas(subRunCursor, regenEnd, target);
Herb Derby23f29762020-01-10 16:26:14 -0500241 // There was a problem allocating the glyph in the atlas. Bail.
Robert Phillips1576e4e2020-04-01 12:49:45 -0400242 if (!ok) {
243 return;
244 }
Herb Derby23f29762020-01-10 16:26:14 -0500245
Herb Derbyfd894ff2020-07-15 13:23:33 -0400246 geo.fillVertexData(vertices + quadCursor * quadSize, subRunCursor, glyphsRegenerated);
Herb Derby23f29762020-01-10 16:26:14 -0500247
Herb Derbyfd894ff2020-07-15 13:23:33 -0400248 subRunCursor += glyphsRegenerated;
249 quadCursor += glyphsRegenerated;
250 allGlyphsCursor += glyphsRegenerated;
Herb Derby23f29762020-01-10 16:26:14 -0500251 flushInfo.fGlyphsToFlush += glyphsRegenerated;
252
Herb Derbyfd894ff2020-07-15 13:23:33 -0400253 if (quadCursor == quadEnd || subRunCursor < subRunEnd) {
254 // Flush if not all the glyphs are drawn because either the quad buffer is full or
255 // the atlas is out of space.
Herb Derby4513cdd2020-01-31 13:28:49 -0500256 this->createDrawForGeneratedGlyphs(target, &flushInfo);
Herb Derbyfd894ff2020-07-15 13:23:33 -0400257 if (quadCursor == quadEnd && allGlyphsCursor < allGlyphsEnd) {
258 // If the vertex buffer is full and there are still glyphs to draw then
259 // get a new buffer.
260 if(!resetVertexBuffer()) {
Herb Derby23f29762020-01-10 16:26:14 -0500261 return;
262 }
263 }
264 }
265 }
Herb Derbyfd894ff2020-07-15 13:23:33 -0400266 }
joshualitta751c972015-11-20 13:37:32 -0800267}
268
Chris Dalton07cdcfc92019-02-26 11:13:22 -0700269void GrAtlasTextOp::onExecute(GrOpFlushState* flushState, const SkRect& chainBounds) {
Robert Phillips3968fcb2019-12-05 16:40:31 -0500270 auto pipeline = GrSimpleMeshDrawOpHelper::CreatePipeline(flushState,
271 std::move(fProcessors),
272 GrPipeline::InputFlags::kNone);
273
Chris Dalton1b6a43c2020-09-25 12:21:18 -0600274 flushState->executeDrawsAndUploadsForMeshDrawOp(this, chainBounds, pipeline,
275 &GrUserStencilSettings::kUnused);
Chris Dalton07cdcfc92019-02-26 11:13:22 -0700276}
277
Herb Derby4513cdd2020-01-31 13:28:49 -0500278void GrAtlasTextOp::createDrawForGeneratedGlyphs(
279 GrMeshDrawOp::Target* target, FlushInfo* flushInfo) const {
Robert Phillipsd2e9f762018-03-07 11:54:37 -0500280 if (!flushInfo->fGlyphsToFlush) {
281 return;
282 }
283
Robert Phillips5a66efb2018-03-07 15:13:18 -0500284 auto atlasManager = target->atlasManager();
Robert Phillipsc4039ea2018-03-01 11:36:45 -0500285
Robert Phillips7cd0bfe2019-11-20 16:08:10 -0500286 GrGeometryProcessor* gp = flushInfo->fGeometryProcessor;
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400287 GrMaskFormat maskFormat = this->maskFormat();
Robert Phillipsf3690dd2018-02-20 15:18:59 -0500288
Greg Daniel9715b6c2019-12-10 15:03:10 -0500289 unsigned int numActiveViews;
290 const GrSurfaceProxyView* views = atlasManager->getViews(maskFormat, &numActiveViews);
291 SkASSERT(views);
Jim Van Verth9f2516f2019-11-22 14:58:37 -0500292 // Something has gone terribly wrong, bail
Greg Daniel9715b6c2019-12-10 15:03:10 -0500293 if (!views || 0 == numActiveViews) {
Jim Van Verth9f2516f2019-11-22 14:58:37 -0500294 return;
295 }
Greg Daniel9715b6c2019-12-10 15:03:10 -0500296 if (gp->numTextureSamplers() != (int) numActiveViews) {
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400297 // During preparation the number of atlas pages has increased.
298 // Update the proxies used in the GP to match.
Greg Daniel9715b6c2019-12-10 15:03:10 -0500299 for (unsigned i = gp->numTextureSamplers(); i < numActiveViews; ++i) {
Chris Dalton304e14d2020-03-17 14:29:06 -0600300 flushInfo->fPrimProcProxies[i] = views[i].proxy();
Greg Danielb20d7e52019-09-03 13:54:39 -0400301 // This op does not know its atlas proxies when it is added to a GrOpsTasks, so the
302 // proxies don't get added during the visitProxies call. Thus we add them here.
Greg Daniel9715b6c2019-12-10 15:03:10 -0500303 target->sampledProxyArray()->push_back(views[i].proxy());
Brian Salomon43cbd722020-01-03 22:09:12 -0500304 // These will get unreffed when the previously recorded draws destruct.
305 for (int d = 0; d < flushInfo->fNumDraws; ++d) {
Chris Dalton304e14d2020-03-17 14:29:06 -0600306 flushInfo->fPrimProcProxies[i]->ref();
Brian Salomon43cbd722020-01-03 22:09:12 -0500307 }
Brian Salomon7eae3e02018-08-07 14:02:38 +0000308 }
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400309 if (this->usesDistanceFields()) {
310 if (this->isLCD()) {
Greg Daniel9715b6c2019-12-10 15:03:10 -0500311 reinterpret_cast<GrDistanceFieldLCDTextGeoProc*>(gp)->addNewViews(
Brian Salomona3b02f52020-07-15 16:02:01 -0400312 views, numActiveViews, GrSamplerState::Filter::kLinear);
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400313 } else {
Greg Daniel9715b6c2019-12-10 15:03:10 -0500314 reinterpret_cast<GrDistanceFieldA8TextGeoProc*>(gp)->addNewViews(
Brian Salomona3b02f52020-07-15 16:02:01 -0400315 views, numActiveViews, GrSamplerState::Filter::kLinear);
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400316 }
317 } else {
Brian Salomona3b02f52020-07-15 16:02:01 -0400318 auto filter = fNeedsGlyphTransform ? GrSamplerState::Filter::kLinear
Brian Salomonccb61422020-01-09 10:46:36 -0500319 : GrSamplerState::Filter::kNearest;
320 reinterpret_cast<GrBitmapTextGeoProc*>(gp)->addNewViews(views, numActiveViews, filter);
Jim Van Vertheafa64b2017-09-18 10:05:00 -0400321 }
322 }
Brian Salomondbf70722019-02-07 11:31:24 -0500323 int maxGlyphsPerDraw = static_cast<int>(flushInfo->fIndexBuffer->size() / sizeof(uint16_t) / 6);
Chris Daltoneb694b72020-03-16 09:25:50 -0600324 GrSimpleMesh* mesh = target->allocMesh();
Chris Dalton37c7bdd2020-03-13 09:21:12 -0600325 mesh->setIndexedPatterned(flushInfo->fIndexBuffer, kIndicesPerGlyph, flushInfo->fGlyphsToFlush,
326 maxGlyphsPerDraw, flushInfo->fVertexBuffer, kVerticesPerGlyph,
327 flushInfo->fVertexOffset);
Chris Dalton304e14d2020-03-17 14:29:06 -0600328 target->recordDraw(flushInfo->fGeometryProcessor, mesh, 1, flushInfo->fPrimProcProxies,
Chris Dalton3bf2f3a2020-03-17 11:48:23 -0600329 GrPrimitiveType::kTriangles);
joshualitta751c972015-11-20 13:37:32 -0800330 flushInfo->fVertexOffset += kVerticesPerGlyph * flushInfo->fGlyphsToFlush;
331 flushInfo->fGlyphsToFlush = 0;
Brian Salomon43cbd722020-01-03 22:09:12 -0500332 ++flushInfo->fNumDraws;
joshualitta751c972015-11-20 13:37:32 -0800333}
334
Herb Derbye25c3002020-10-27 15:57:27 -0400335GrOp::CombineResult GrAtlasTextOp::onCombineIfPossible(GrOp* t, SkArenaAlloc*, const GrCaps& caps) {
Brian Salomon344ec422016-12-15 10:58:41 -0500336 GrAtlasTextOp* that = t->cast<GrAtlasTextOp>();
Brian Salomon44acb5b2017-07-18 19:59:24 -0400337 if (fProcessors != that->fProcessors) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000338 return CombineResult::kCannotCombine;
Brian Salomon44acb5b2017-07-18 19:59:24 -0400339 }
340
joshualitta751c972015-11-20 13:37:32 -0800341 if (fMaskType != that->fMaskType) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000342 return CombineResult::kCannotCombine;
joshualitta751c972015-11-20 13:37:32 -0800343 }
344
Herb Derby1c5be7b2019-12-13 12:03:06 -0500345 const SkMatrix& thisFirstMatrix = fGeoData[0].fDrawMatrix;
346 const SkMatrix& thatFirstMatrix = that->fGeoData[0].fDrawMatrix;
Brian Salomon5c6ac642017-12-19 11:09:32 -0500347
Mike Reed2c383152019-12-18 16:47:47 -0500348 if (this->usesLocalCoords() && !SkMatrixPriv::CheapEqual(thisFirstMatrix, thatFirstMatrix)) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000349 return CombineResult::kCannotCombine;
Brian Salomon5c6ac642017-12-19 11:09:32 -0500350 }
351
Jim Van Verthb515ae72018-05-23 16:44:55 -0400352 if (fNeedsGlyphTransform != that->fNeedsGlyphTransform) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000353 return CombineResult::kCannotCombine;
Jim Van Verthb515ae72018-05-23 16:44:55 -0400354 }
355
356 if (fNeedsGlyphTransform &&
357 (thisFirstMatrix.hasPerspective() != thatFirstMatrix.hasPerspective())) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000358 return CombineResult::kCannotCombine;
Jim Van Verthb515ae72018-05-23 16:44:55 -0400359 }
360
Brian Salomon5c6ac642017-12-19 11:09:32 -0500361 if (this->usesDistanceFields()) {
362 if (fDFGPFlags != that->fDFGPFlags) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000363 return CombineResult::kCannotCombine;
joshualitta751c972015-11-20 13:37:32 -0800364 }
365
Jim Van Verthbc2cdd12017-06-08 11:14:35 -0400366 if (fLuminanceColor != that->fLuminanceColor) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000367 return CombineResult::kCannotCombine;
joshualitta751c972015-11-20 13:37:32 -0800368 }
Brian Salomon5c6ac642017-12-19 11:09:32 -0500369 } else {
370 if (kColorBitmapMask_MaskType == fMaskType && this->color() != that->color()) {
Brian Salomon7eae3e02018-08-07 14:02:38 +0000371 return CombineResult::kCannotCombine;
Brian Salomon5c6ac642017-12-19 11:09:32 -0500372 }
joshualitta751c972015-11-20 13:37:32 -0800373 }
374
Brian Salomon344ec422016-12-15 10:58:41 -0500375 fNumGlyphs += that->numGlyphs();
joshualitta751c972015-11-20 13:37:32 -0800376
Jim Van Verth56c37142017-10-31 14:44:25 -0400377 // Reallocate space for geo data if necessary and then import that geo's data.
joshualitta751c972015-11-20 13:37:32 -0800378 int newGeoCount = that->fGeoCount + fGeoCount;
joshualitta751c972015-11-20 13:37:32 -0800379
Jim Van Verthc8a65e32017-10-25 14:25:27 -0400380 // We reallocate at a rate of 1.5x to try to get better total memory usage
381 if (newGeoCount > fGeoDataAllocSize) {
Jim Van Verth56c37142017-10-31 14:44:25 -0400382 int newAllocSize = fGeoDataAllocSize + fGeoDataAllocSize / 2;
Jim Van Verthc8a65e32017-10-25 14:25:27 -0400383 while (newAllocSize < newGeoCount) {
384 newAllocSize += newAllocSize / 2;
385 }
joshualitta751c972015-11-20 13:37:32 -0800386 fGeoData.realloc(newAllocSize);
Jim Van Verthc8a65e32017-10-25 14:25:27 -0400387 fGeoDataAllocSize = newAllocSize;
joshualitta751c972015-11-20 13:37:32 -0800388 }
389
Brian Salomon344ec422016-12-15 10:58:41 -0500390 // 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 -0800391 // it doesn't try to unref them.
Herb Derby1d17e492020-07-21 11:45:04 -0400392 for (int i = 0; i < that->fGeoCount; i++) {
393 new (&fGeoData[fGeoCount + i]) Geometry{that->fGeoData[i]};
joshualitta751c972015-11-20 13:37:32 -0800394 }
Herb Derby1d17e492020-07-21 11:45:04 -0400395
joshualitta751c972015-11-20 13:37:32 -0800396 that->fGeoCount = 0;
397 fGeoCount = newGeoCount;
398
Brian Salomon7eae3e02018-08-07 14:02:38 +0000399 return CombineResult::kMerged;
joshualitta751c972015-11-20 13:37:32 -0800400}
401
Herb Derby3c873af2020-04-29 15:56:07 -0400402static const int kDistanceAdjustLumShift = 5;
403
joshualitta751c972015-11-20 13:37:32 -0800404// TODO trying to figure out why lcd is so whack
Robert Phillips7cd0bfe2019-11-20 16:08:10 -0500405GrGeometryProcessor* GrAtlasTextOp::setupDfProcessor(SkArenaAlloc* arena,
406 const GrShaderCaps& caps,
Greg Daniel9715b6c2019-12-10 15:03:10 -0500407 const GrSurfaceProxyView* views,
408 unsigned int numActiveViews) const {
joshualitta751c972015-11-20 13:37:32 -0800409 bool isLCD = this->isLCD();
Brian Salomon5c6ac642017-12-19 11:09:32 -0500410
411 SkMatrix localMatrix = SkMatrix::I();
412 if (this->usesLocalCoords()) {
413 // If this fails we'll just use I().
Herb Derby1c5be7b2019-12-13 12:03:06 -0500414 bool result = fGeoData[0].fDrawMatrix.invert(&localMatrix);
Brian Salomon5c6ac642017-12-19 11:09:32 -0500415 (void)result;
416 }
joshualitta751c972015-11-20 13:37:32 -0800417
Robert Phillips841c9a52020-03-27 12:41:31 -0400418 auto dfAdjustTable = GrDistanceFieldAdjustTable::Get();
419
joshualitta751c972015-11-20 13:37:32 -0800420 // see if we need to create a new effect
421 if (isLCD) {
Robert Phillips841c9a52020-03-27 12:41:31 -0400422 float redCorrection = dfAdjustTable->getAdjustment(
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400423 SkColorGetR(fLuminanceColor) >> kDistanceAdjustLumShift,
Brian Salomon344ec422016-12-15 10:58:41 -0500424 fUseGammaCorrectDistanceTable);
Robert Phillips841c9a52020-03-27 12:41:31 -0400425 float greenCorrection = dfAdjustTable->getAdjustment(
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400426 SkColorGetG(fLuminanceColor) >> kDistanceAdjustLumShift,
Brian Salomon344ec422016-12-15 10:58:41 -0500427 fUseGammaCorrectDistanceTable);
Robert Phillips841c9a52020-03-27 12:41:31 -0400428 float blueCorrection = dfAdjustTable->getAdjustment(
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400429 SkColorGetB(fLuminanceColor) >> kDistanceAdjustLumShift,
Brian Salomon344ec422016-12-15 10:58:41 -0500430 fUseGammaCorrectDistanceTable);
joshualitta751c972015-11-20 13:37:32 -0800431 GrDistanceFieldLCDTextGeoProc::DistanceAdjust widthAdjust =
Brian Salomon344ec422016-12-15 10:58:41 -0500432 GrDistanceFieldLCDTextGeoProc::DistanceAdjust::Make(
433 redCorrection, greenCorrection, blueCorrection);
Greg Daniel9715b6c2019-12-10 15:03:10 -0500434 return GrDistanceFieldLCDTextGeoProc::Make(arena, caps, views, numActiveViews,
Brian Salomona3b02f52020-07-15 16:02:01 -0400435 GrSamplerState::Filter::kLinear, widthAdjust,
Brian Osman09068252018-01-03 09:57:29 -0500436 fDFGPFlags, localMatrix);
joshualitta751c972015-11-20 13:37:32 -0800437 } else {
joshualitta751c972015-11-20 13:37:32 -0800438#ifdef SK_GAMMA_APPLY_TO_A8
Jim Van Verth90e89b32017-07-06 16:36:55 -0400439 float correction = 0;
440 if (kAliasedDistanceField_MaskType != fMaskType) {
Jim Van Verth58c3cce2017-10-19 15:50:24 -0400441 U8CPU lum = SkColorSpaceLuminance::computeLuminance(SK_GAMMA_EXPONENT,
442 fLuminanceColor);
Robert Phillips841c9a52020-03-27 12:41:31 -0400443 correction = dfAdjustTable->getAdjustment(lum >> kDistanceAdjustLumShift,
444 fUseGammaCorrectDistanceTable);
Jim Van Verth90e89b32017-07-06 16:36:55 -0400445 }
Greg Daniel9715b6c2019-12-10 15:03:10 -0500446 return GrDistanceFieldA8TextGeoProc::Make(arena, caps, views, numActiveViews,
Brian Salomona3b02f52020-07-15 16:02:01 -0400447 GrSamplerState::Filter::kLinear, correction,
Brian Salomonccb61422020-01-09 10:46:36 -0500448 fDFGPFlags, localMatrix);
joshualitta751c972015-11-20 13:37:32 -0800449#else
Greg Daniel9715b6c2019-12-10 15:03:10 -0500450 return GrDistanceFieldA8TextGeoProc::Make(arena, caps, views, numActiveViews,
Brian Salomona3b02f52020-07-15 16:02:01 -0400451 GrSamplerState::Filter::kLinear, fDFGPFlags,
452 localMatrix);
joshualitta751c972015-11-20 13:37:32 -0800453#endif
454 }
joshualitta751c972015-11-20 13:37:32 -0800455}
joshualittddd22d82016-02-16 06:47:52 -0800456
Herb Derby4598fa12020-06-10 14:54:22 -0400457#if GR_TEST_UTILS
Herb Derbyc76d4092020-10-07 16:46:15 -0400458
459GrOp::Owner GrAtlasTextOp::CreateOpTestingOnly(GrRenderTargetContext* rtc,
460 const SkPaint& skPaint,
461 const SkFont& font,
462 const SkMatrixProvider& mtxProvider,
463 const char* text,
464 int x,
465 int y) {
Herb Derby4598fa12020-06-10 14:54:22 -0400466 size_t textLen = (int)strlen(text);
467
468 const SkMatrix& drawMatrix(mtxProvider.localToDevice());
469
470 auto drawOrigin = SkPoint::Make(x, y);
471 SkGlyphRunBuilder builder;
472 builder.drawTextUTF8(skPaint, font, text, textLen, drawOrigin);
473
474 auto glyphRunList = builder.useGlyphRunList();
Ben Wagner525e8762020-07-09 16:19:35 -0400475 if (glyphRunList.empty()) {
476 return nullptr;
477 }
Herb Derby4598fa12020-06-10 14:54:22 -0400478
Robert Phillipsa9306eb2020-07-21 13:01:25 -0400479
480 auto rContext = rtc->priv().recordingContext();
481 GrSDFTOptions SDFOptions = rContext->priv().SDFTOptions();
Herb Derby4598fa12020-06-10 14:54:22 -0400482
Herb Derby4598fa12020-06-10 14:54:22 -0400483 sk_sp<GrTextBlob> blob = GrTextBlob::Make(glyphRunList, drawMatrix);
Herb Derby411e7aa2020-07-09 16:02:08 -0400484 SkGlyphRunListPainter* painter = rtc->priv().testingOnly_glyphRunPainter();
Herb Derby4598fa12020-06-10 14:54:22 -0400485 painter->processGlyphRunList(
Herb Derbyd5764642020-07-08 09:57:11 -0400486 glyphRunList, drawMatrix, rtc->surfaceProps(),
Robert Phillipsa9306eb2020-07-21 13:01:25 -0400487 rContext->priv().caps()->shaderCaps()->supportsDistanceFieldText(),
Herb Derby4598fa12020-06-10 14:54:22 -0400488 SDFOptions, blob.get());
Herb Derby342273c2020-07-13 10:22:32 -0400489 if (!blob->subRunList().head()) {
Ben Wagner525e8762020-07-09 16:19:35 -0400490 return nullptr;
491 }
Herb Derby4598fa12020-06-10 14:54:22 -0400492
Herb Derby252a3c02020-07-14 12:15:34 -0400493 GrAtlasSubRun* subRun = static_cast<GrAtlasSubRun*>(blob->subRunList().head());
Herb Derbyc76d4092020-10-07 16:46:15 -0400494 GrOp::Owner op;
Herb Derby252a3c02020-07-14 12:15:34 -0400495 std::tie(std::ignore, op) = subRun->makeAtlasTextOp(nullptr, mtxProvider, glyphRunList, rtc);
Herb Derby411e7aa2020-07-09 16:02:08 -0400496 return op;
Herb Derby4598fa12020-06-10 14:54:22 -0400497}
498
499GR_DRAW_OP_TEST_DEFINE(GrAtlasTextOp) {
500 // Setup dummy SkPaint / GrPaint / GrRenderTargetContext
501 auto rtc = GrRenderTargetContext::Make(
502 context, GrColorType::kRGBA_8888, nullptr, SkBackingFit::kApprox, {1024, 1024});
503
504 SkSimpleMatrixProvider matrixProvider(GrTest::TestMatrixInvertible(random));
505
506 SkPaint skPaint;
507 skPaint.setColor(random->nextU());
508
509 SkFont font;
510 if (random->nextBool()) {
511 font.setEdging(SkFont::Edging::kSubpixelAntiAlias);
512 } else {
513 font.setEdging(random->nextBool() ? SkFont::Edging::kAntiAlias : SkFont::Edging::kAlias);
514 }
515 font.setSubpixel(random->nextBool());
516
517 const char* text = "The quick brown fox jumps over the lazy dog.";
518
519 // create some random x/y offsets, including negative offsets
520 static const int kMaxTrans = 1024;
521 int xPos = (random->nextU() % 2) * 2 - 1;
522 int yPos = (random->nextU() % 2) * 2 - 1;
523 int xInt = (random->nextU() % kMaxTrans) * xPos;
524 int yInt = (random->nextU() % kMaxTrans) * yPos;
525
526 return GrAtlasTextOp::CreateOpTestingOnly(
527 rtc.get(), skPaint, font, matrixProvider, text, xInt, yInt);
528}
529
530#endif
531
532