blob: c9c0dca808a63d63ef8d8d13b00dfb15449a25be [file] [log] [blame]
bsalomonc55271f2015-11-09 11:55:57 -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/SkGpuDevice.h"
Michael Ludwig1433cfd2019-02-27 17:12:30 -05009
Mike Kleinc0bd9f92019-04-23 12:05:21 -050010#include "include/core/SkYUVAIndex.h"
Robert Phillips16bf7d32020-07-07 10:20:27 -040011#include "include/gpu/GrDirectContext.h"
12#include "include/gpu/GrRecordingContext.h"
Mike Klein8aa0edf2020-10-16 11:04:18 -050013#include "include/private/SkTPin.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050014#include "src/core/SkDraw.h"
15#include "src/core/SkMaskFilterBase.h"
Michael Ludwig16d5b0a2020-08-31 13:07:16 -040016#include "src/core/SkSpecialImage.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050017#include "src/gpu/GrBitmapTextureMaker.h"
18#include "src/gpu/GrBlurUtils.h"
19#include "src/gpu/GrCaps.h"
20#include "src/gpu/GrColorSpaceXform.h"
21#include "src/gpu/GrImageTextureMaker.h"
Robert Phillips16bf7d32020-07-07 10:20:27 -040022#include "src/gpu/GrRecordingContextPriv.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050023#include "src/gpu/GrRenderTargetContext.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050024#include "src/gpu/GrStyle.h"
25#include "src/gpu/GrTextureAdjuster.h"
26#include "src/gpu/GrTextureMaker.h"
27#include "src/gpu/SkGr.h"
28#include "src/gpu/effects/GrBicubicEffect.h"
John Stilesf743d4e2020-07-23 11:35:08 -040029#include "src/gpu/effects/GrBlendFragmentProcessor.h"
Brian Salomonb8f098d2020-01-07 11:15:44 -050030#include "src/gpu/effects/GrTextureEffect.h"
Michael Ludwig2686d692020-04-17 20:21:37 +000031#include "src/gpu/geometry/GrStyledShape.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050032#include "src/image/SkImage_Base.h"
bsalomonc55271f2015-11-09 11:55:57 -080033
Michael Ludwig1433cfd2019-02-27 17:12:30 -050034namespace {
35
bsalomonc55271f2015-11-09 11:55:57 -080036static inline bool use_shader(bool textureIsAlphaOnly, const SkPaint& paint) {
37 return textureIsAlphaOnly && paint.getShader();
38}
39
bsalomonb1b01992015-11-18 10:56:08 -080040//////////////////////////////////////////////////////////////////////////////
Brian Salomona3b02f52020-07-15 16:02:01 -040041// Helper functions for dropping src rect subset with GrSamplerState::Filter::kLinear.
bsalomonb1b01992015-11-18 10:56:08 -080042
43static const SkScalar kColorBleedTolerance = 0.001f;
44
45static bool has_aligned_samples(const SkRect& srcRect, const SkRect& transformedRect) {
46 // detect pixel disalignment
Brian Salomona911f8f2015-11-18 15:19:57 -050047 if (SkScalarAbs(SkScalarRoundToScalar(transformedRect.left()) - transformedRect.left()) < kColorBleedTolerance &&
48 SkScalarAbs(SkScalarRoundToScalar(transformedRect.top()) - transformedRect.top()) < kColorBleedTolerance &&
49 SkScalarAbs(transformedRect.width() - srcRect.width()) < kColorBleedTolerance &&
bsalomonb1b01992015-11-18 10:56:08 -080050 SkScalarAbs(transformedRect.height() - srcRect.height()) < kColorBleedTolerance) {
51 return true;
52 }
53 return false;
54}
55
56static bool may_color_bleed(const SkRect& srcRect,
57 const SkRect& transformedRect,
58 const SkMatrix& m,
Chris Dalton6ce447a2019-06-23 18:07:38 -060059 int numSamples) {
bsalomonb1b01992015-11-18 10:56:08 -080060 // Only gets called if has_aligned_samples returned false.
61 // So we can assume that sampling is axis aligned but not texel aligned.
62 SkASSERT(!has_aligned_samples(srcRect, transformedRect));
63 SkRect innerSrcRect(srcRect), innerTransformedRect, outerTransformedRect(transformedRect);
Chris Dalton6ce447a2019-06-23 18:07:38 -060064 if (numSamples > 1) {
bsalomonb1b01992015-11-18 10:56:08 -080065 innerSrcRect.inset(SK_Scalar1, SK_Scalar1);
66 } else {
67 innerSrcRect.inset(SK_ScalarHalf, SK_ScalarHalf);
68 }
69 m.mapRect(&innerTransformedRect, innerSrcRect);
70
71 // The gap between outerTransformedRect and innerTransformedRect
72 // represents the projection of the source border area, which is
73 // problematic for color bleeding. We must check whether any
74 // destination pixels sample the border area.
75 outerTransformedRect.inset(kColorBleedTolerance, kColorBleedTolerance);
76 innerTransformedRect.outset(kColorBleedTolerance, kColorBleedTolerance);
77 SkIRect outer, inner;
78 outerTransformedRect.round(&outer);
79 innerTransformedRect.round(&inner);
80 // If the inner and outer rects round to the same result, it means the
81 // border does not overlap any pixel centers. Yay!
82 return inner != outer;
83}
84
Brian Salomona3b02f52020-07-15 16:02:01 -040085static bool can_ignore_linear_filtering_subset(const GrTextureProducer& producer,
86 const SkRect& srcSubset,
87 const SkMatrix& srcRectToDeviceSpace,
88 int numSamples) {
bsalomonb1b01992015-11-18 10:56:08 -080089 if (srcRectToDeviceSpace.rectStaysRect()) {
90 // sampling is axis-aligned
91 SkRect transformedRect;
Brian Salomon8f32f132020-07-14 12:30:12 -040092 srcRectToDeviceSpace.mapRect(&transformedRect, srcSubset);
bsalomonb1b01992015-11-18 10:56:08 -080093
Brian Salomon8f32f132020-07-14 12:30:12 -040094 if (has_aligned_samples(srcSubset, transformedRect) ||
95 !may_color_bleed(srcSubset, transformedRect, srcRectToDeviceSpace, numSamples)) {
bsalomonb1b01992015-11-18 10:56:08 -080096 return true;
97 }
98 }
99 return false;
100}
101
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400102//////////////////////////////////////////////////////////////////////////////
103// Helper functions for tiling a large SkBitmap
104
105static const int kBmpSmallTileSize = 1 << 10;
106
107static inline int get_tile_count(const SkIRect& srcRect, int tileSize) {
108 int tilesX = (srcRect.fRight / tileSize) - (srcRect.fLeft / tileSize) + 1;
109 int tilesY = (srcRect.fBottom / tileSize) - (srcRect.fTop / tileSize) + 1;
110 return tilesX * tilesY;
111}
112
113static int determine_tile_size(const SkIRect& src, int maxTileSize) {
114 if (maxTileSize <= kBmpSmallTileSize) {
115 return maxTileSize;
116 }
117
118 size_t maxTileTotalTileSize = get_tile_count(src, maxTileSize);
119 size_t smallTotalTileSize = get_tile_count(src, kBmpSmallTileSize);
120
121 maxTileTotalTileSize *= maxTileSize * maxTileSize;
122 smallTotalTileSize *= kBmpSmallTileSize * kBmpSmallTileSize;
123
124 if (maxTileTotalTileSize > 2 * smallTotalTileSize) {
125 return kBmpSmallTileSize;
126 } else {
127 return maxTileSize;
128 }
129}
130
131// Given a bitmap, an optional src rect, and a context with a clip and matrix determine what
132// pixels from the bitmap are necessary.
Michael Ludwigc002d562020-05-13 14:17:57 -0400133static SkIRect determine_clipped_src_rect(int width, int height,
Michael Ludwig7c12e282020-05-29 09:54:07 -0400134 const GrClip* clip,
Michael Ludwigc002d562020-05-13 14:17:57 -0400135 const SkMatrix& viewMatrix,
136 const SkMatrix& srcToDstRect,
137 const SkISize& imageDimensions,
138 const SkRect* srcRectPtr) {
Michael Ludwige06a8972020-06-11 10:29:00 -0400139 SkIRect clippedSrcIRect = clip ? clip->getConservativeBounds()
Michael Ludwig7c12e282020-05-29 09:54:07 -0400140 : SkIRect::MakeWH(width, height);
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400141 SkMatrix inv = SkMatrix::Concat(viewMatrix, srcToDstRect);
142 if (!inv.invert(&inv)) {
Michael Ludwigc002d562020-05-13 14:17:57 -0400143 return SkIRect::MakeEmpty();
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400144 }
Michael Ludwigc002d562020-05-13 14:17:57 -0400145 SkRect clippedSrcRect = SkRect::Make(clippedSrcIRect);
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400146 inv.mapRect(&clippedSrcRect);
147 if (srcRectPtr) {
148 if (!clippedSrcRect.intersect(*srcRectPtr)) {
Michael Ludwigc002d562020-05-13 14:17:57 -0400149 return SkIRect::MakeEmpty();
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400150 }
151 }
Michael Ludwigc002d562020-05-13 14:17:57 -0400152 clippedSrcRect.roundOut(&clippedSrcIRect);
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400153 SkIRect bmpBounds = SkIRect::MakeSize(imageDimensions);
Michael Ludwigc002d562020-05-13 14:17:57 -0400154 if (!clippedSrcIRect.intersect(bmpBounds)) {
155 return SkIRect::MakeEmpty();
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400156 }
Michael Ludwigc002d562020-05-13 14:17:57 -0400157
158 return clippedSrcIRect;
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400159}
160
Michael Ludwig47237242020-03-10 16:16:17 -0400161// tileSize and clippedSubset are valid if true is returned
Robert Phillips16bf7d32020-07-07 10:20:27 -0400162static bool should_tile_image_id(GrRecordingContext* context,
Michael Ludwig47237242020-03-10 16:16:17 -0400163 SkISize rtSize,
Michael Ludwig7c12e282020-05-29 09:54:07 -0400164 const GrClip* clip,
Michael Ludwig47237242020-03-10 16:16:17 -0400165 uint32_t imageID,
166 const SkISize& imageSize,
167 const SkMatrix& ctm,
168 const SkMatrix& srcToDst,
169 const SkRect* src,
170 int maxTileSize,
171 int* tileSize,
172 SkIRect* clippedSubset) {
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400173 // if it's larger than the max tile size, then we have no choice but tiling.
Michael Ludwig47237242020-03-10 16:16:17 -0400174 if (imageSize.width() > maxTileSize || imageSize.height() > maxTileSize) {
Michael Ludwigc002d562020-05-13 14:17:57 -0400175 *clippedSubset = determine_clipped_src_rect(rtSize.width(), rtSize.height(), clip, ctm,
176 srcToDst, imageSize, src);
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400177 *tileSize = determine_tile_size(*clippedSubset, maxTileSize);
178 return true;
179 }
180
181 // If the image would only produce 4 tiles of the smaller size, don't bother tiling it.
Michael Ludwig47237242020-03-10 16:16:17 -0400182 const size_t area = imageSize.width() * imageSize.height();
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400183 if (area < 4 * kBmpSmallTileSize * kBmpSmallTileSize) {
184 return false;
185 }
186
Michael Ludwig46d91382020-05-07 09:51:12 -0400187 // At this point we know we could do the draw by uploading the entire bitmap as a texture.
188 // However, if the texture would be large compared to the cache size and we don't require most
189 // of it for this draw then tile to reduce the amount of upload and cache spill.
190 // NOTE: if the context is not a direct context, it doesn't have access to the resource cache,
191 // and theoretically, the resource cache's limits could be being changed on another thread, so
192 // even having access to just the limit wouldn't be a reliable test during recording here.
193 // Instead, we will just upload the entire image to be on the safe side and not tile.
Robert Phillips16bf7d32020-07-07 10:20:27 -0400194 auto direct = context->asDirectContext();
195 if (!direct) {
Michael Ludwig46d91382020-05-07 09:51:12 -0400196 return false;
197 }
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400198
199 // assumption here is that sw bitmap size is a good proxy for its size as
200 // a texture
201 size_t bmpSize = area * sizeof(SkPMColor); // assume 32bit pixels
Robert Phillips16bf7d32020-07-07 10:20:27 -0400202 size_t cacheSize = direct->getResourceCacheLimit();
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400203 if (bmpSize < cacheSize / 2) {
204 return false;
205 }
206
207 // Figure out how much of the src we will need based on the src rect and clipping. Reject if
208 // tiling memory savings would be < 50%.
Michael Ludwigc002d562020-05-13 14:17:57 -0400209 *clippedSubset = determine_clipped_src_rect(rtSize.width(), rtSize.height(), clip, ctm,
210 srcToDst, imageSize, src);
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400211 *tileSize = kBmpSmallTileSize; // already know whole bitmap fits in one max sized tile.
212 size_t usedTileBytes = get_tile_count(*clippedSubset, kBmpSmallTileSize) *
213 kBmpSmallTileSize * kBmpSmallTileSize *
214 sizeof(SkPMColor); // assume 32bit pixels;
215
216 return usedTileBytes * 2 < bmpSize;
217}
218
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400219// This method outsets 'iRect' by 'outset' all around and then clamps its extents to
220// 'clamp'. 'offset' is adjusted to remain positioned over the top-left corner
221// of 'iRect' for all possible outsets/clamps.
Michael Ludwig47237242020-03-10 16:16:17 -0400222static inline void clamped_outset_with_offset(SkIRect* iRect, int outset, SkPoint* offset,
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400223 const SkIRect& clamp) {
224 iRect->outset(outset, outset);
225
226 int leftClampDelta = clamp.fLeft - iRect->fLeft;
227 if (leftClampDelta > 0) {
228 offset->fX -= outset - leftClampDelta;
229 iRect->fLeft = clamp.fLeft;
230 } else {
231 offset->fX -= outset;
232 }
233
234 int topClampDelta = clamp.fTop - iRect->fTop;
235 if (topClampDelta > 0) {
236 offset->fY -= outset - topClampDelta;
237 iRect->fTop = clamp.fTop;
238 } else {
239 offset->fY -= outset;
240 }
241
242 if (iRect->fRight > clamp.fRight) {
243 iRect->fRight = clamp.fRight;
244 }
245 if (iRect->fBottom > clamp.fBottom) {
246 iRect->fBottom = clamp.fBottom;
247 }
248}
249
250//////////////////////////////////////////////////////////////////////////////
251// Helper functions for drawing an image with GrRenderTargetContext
252
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500253enum class ImageDrawMode {
254 // Src and dst have been restricted to the image content. May need to clamp, no need to decal.
255 kOptimized,
256 // Src and dst are their original sizes, requires use of a decal instead of plain clamping.
257 // This is used when a dst clip is provided and extends outside of the optimized dst rect.
258 kDecal,
259 // Src or dst are empty, or do not intersect the image content so don't draw anything.
260 kSkip
261};
262
263/**
264 * Optimize the src rect sampling area within an image (sized 'width' x 'height') such that
265 * 'outSrcRect' will be completely contained in the image's bounds. The corresponding rect
266 * to draw will be output to 'outDstRect'. The mapping between src and dst will be cached in
267 * 'srcToDst'. Outputs are not always updated when kSkip is returned.
268 *
269 * If 'origSrcRect' is null, implicitly use the image bounds. If 'origDstRect' is null, use the
270 * original src rect. 'dstClip' should be null when there is no additional clipping.
271 */
272static ImageDrawMode optimize_sample_area(const SkISize& image, const SkRect* origSrcRect,
273 const SkRect* origDstRect, const SkPoint dstClip[4],
274 SkRect* outSrcRect, SkRect* outDstRect,
275 SkMatrix* srcToDst) {
276 SkRect srcBounds = SkRect::MakeIWH(image.fWidth, image.fHeight);
277
278 SkRect src = origSrcRect ? *origSrcRect : srcBounds;
279 SkRect dst = origDstRect ? *origDstRect : src;
280
281 if (src.isEmpty() || dst.isEmpty()) {
282 return ImageDrawMode::kSkip;
283 }
284
285 if (outDstRect) {
286 srcToDst->setRectToRect(src, dst, SkMatrix::kFill_ScaleToFit);
287 } else {
288 srcToDst->setIdentity();
289 }
290
291 if (origSrcRect && !srcBounds.contains(src)) {
292 if (!src.intersect(srcBounds)) {
293 return ImageDrawMode::kSkip;
294 }
295 srcToDst->mapRect(&dst, src);
296
297 // Both src and dst have gotten smaller. If dstClip is provided, confirm it is still
298 // contained in dst, otherwise cannot optimize the sample area and must use a decal instead
299 if (dstClip) {
300 for (int i = 0; i < 4; ++i) {
301 if (!dst.contains(dstClip[i].fX, dstClip[i].fY)) {
302 // Must resort to using a decal mode restricted to the clipped 'src', and
303 // use the original dst rect (filling in src bounds as needed)
304 *outSrcRect = src;
305 *outDstRect = (origDstRect ? *origDstRect
306 : (origSrcRect ? *origSrcRect : srcBounds));
307 return ImageDrawMode::kDecal;
308 }
309 }
310 }
311 }
312
313 // The original src and dst were fully contained in the image, or there was no dst clip to
314 // worry about, or the clip was still contained in the restricted dst rect.
315 *outSrcRect = src;
316 *outDstRect = dst;
317 return ImageDrawMode::kOptimized;
318}
319
Brian Salomon34169692017-08-28 15:32:01 -0400320/**
Brian Salomonb80ffee2018-05-23 16:39:39 -0400321 * Checks whether the paint is compatible with using GrRenderTargetContext::drawTexture. It is more
322 * efficient than the GrTextureProducer general case.
Brian Salomon34169692017-08-28 15:32:01 -0400323 */
Brian Salomonb80ffee2018-05-23 16:39:39 -0400324static bool can_use_draw_texture(const SkPaint& paint) {
Brian Salomon34169692017-08-28 15:32:01 -0400325 return (!paint.getColorFilter() && !paint.getShader() && !paint.getMaskFilter() &&
Michael Ludwigd54ca8f2019-02-13 13:25:21 -0500326 !paint.getImageFilter() && paint.getFilterQuality() < kMedium_SkFilterQuality);
Brian Salomon34169692017-08-28 15:32:01 -0400327}
328
Michael Ludwig1c66ad92020-07-10 08:59:44 -0400329static SkPMColor4f texture_color(SkColor4f paintColor, float entryAlpha, GrColorType srcColorType,
330 const GrColorInfo& dstColorInfo) {
331 paintColor.fA *= entryAlpha;
332 if (GrColorTypeIsAlphaOnly(srcColorType)) {
333 return SkColor4fPrepForDst(paintColor, dstColorInfo).premul();
334 } else {
335 float paintAlpha = SkTPin(paintColor.fA, 0.f, 1.f);
336 return { paintAlpha, paintAlpha, paintAlpha, paintAlpha };
337 }
338}
339
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500340// Assumes srcRect and dstRect have already been optimized to fit the proxy
Brian Salomone69b9ef2020-07-22 11:18:06 -0400341static void draw_texture(GrRenderTargetContext* rtc,
342 const GrClip* clip,
343 const SkMatrix& ctm,
344 const SkPaint& paint,
345 const SkRect& srcRect,
346 const SkRect& dstRect,
347 const SkPoint dstClip[4],
348 GrAA aa,
349 GrQuadAAFlags aaFlags,
350 SkCanvas::SrcRectConstraint constraint,
351 GrSurfaceProxyView view,
Greg Daniela4828a12019-10-11 13:51:02 -0400352 const GrColorInfo& srcColorInfo) {
Brian Salomon4bc0c1f2019-09-30 15:12:27 -0400353 const GrColorInfo& dstInfo(rtc->colorInfo());
Mike Kleine03a1762018-08-22 11:52:16 -0400354 auto textureXform =
Greg Daniela4828a12019-10-11 13:51:02 -0400355 GrColorSpaceXform::Make(srcColorInfo.colorSpace(), srcColorInfo.alphaType(),
Brian Osman3d139a42018-11-19 10:42:10 -0500356 dstInfo.colorSpace(), kPremul_SkAlphaType);
Brian Salomon2bbdcc42017-09-07 12:36:34 -0400357 GrSamplerState::Filter filter;
Brian Salomon34169692017-08-28 15:32:01 -0400358 switch (paint.getFilterQuality()) {
359 case kNone_SkFilterQuality:
Brian Salomon2bbdcc42017-09-07 12:36:34 -0400360 filter = GrSamplerState::Filter::kNearest;
Brian Salomon34169692017-08-28 15:32:01 -0400361 break;
362 case kLow_SkFilterQuality:
Brian Salomona3b02f52020-07-15 16:02:01 -0400363 filter = GrSamplerState::Filter::kLinear;
Brian Salomon34169692017-08-28 15:32:01 -0400364 break;
365 case kMedium_SkFilterQuality:
366 case kHigh_SkFilterQuality:
367 SK_ABORT("Quality level not allowed.");
368 }
Greg Daniel2f3cd4f2020-02-07 11:07:25 -0500369 GrSurfaceProxy* proxy = view.proxy();
Michael Ludwigd54ca8f2019-02-13 13:25:21 -0500370 // Must specify the strict constraint when the proxy is not functionally exact and the src
371 // rect would access pixels outside the proxy's content area without the constraint.
Brian Salomon5c60b752019-12-13 15:03:43 -0500372 if (constraint != SkCanvas::kStrict_SrcRectConstraint && !proxy->isFunctionallyExact()) {
Michael Ludwigd54ca8f2019-02-13 13:25:21 -0500373 // Conservative estimate of how much a coord could be outset from src rect:
Brian Salomona3b02f52020-07-15 16:02:01 -0400374 // 1/2 pixel for AA and 1/2 pixel for linear filtering
Michael Ludwigd54ca8f2019-02-13 13:25:21 -0500375 float buffer = 0.5f * (aa == GrAA::kYes) +
Brian Salomona3b02f52020-07-15 16:02:01 -0400376 0.5f * (filter == GrSamplerState::Filter::kLinear);
Brian Salomon9f2b86c2019-10-22 10:37:46 -0400377 SkRect safeBounds = proxy->getBoundsRect();
Michael Ludwigd54ca8f2019-02-13 13:25:21 -0500378 safeBounds.inset(buffer, buffer);
379 if (!safeBounds.contains(srcRect)) {
380 constraint = SkCanvas::kStrict_SrcRectConstraint;
381 }
382 }
Brian Salomon34169692017-08-28 15:32:01 -0400383
Michael Ludwig1c66ad92020-07-10 08:59:44 -0400384 SkPMColor4f color = texture_color(paint.getColor4f(), 1.f, srcColorInfo.colorType(), dstInfo);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500385 if (dstClip) {
386 // Get source coords corresponding to dstClip
387 SkPoint srcQuad[4];
388 GrMapRectPoints(dstRect, srcRect, dstClip, srcQuad, 4);
Michael Ludwig24adb3a2019-02-27 19:42:20 +0000389
Brian Salomone69b9ef2020-07-22 11:18:06 -0400390 rtc->drawTextureQuad(clip,
391 std::move(view),
392 srcColorInfo.colorType(),
393 srcColorInfo.alphaType(),
394 filter,
395 GrSamplerState::MipmapMode::kNone,
396 paint.getBlendMode(),
397 color,
398 srcQuad,
399 dstClip,
400 aa,
401 aaFlags,
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500402 constraint == SkCanvas::kStrict_SrcRectConstraint ? &srcRect : nullptr,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400403 ctm,
404 std::move(textureXform));
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500405 } else {
Brian Salomone69b9ef2020-07-22 11:18:06 -0400406 rtc->drawTexture(clip,
407 std::move(view),
408 srcColorInfo.alphaType(),
409 filter,
410 GrSamplerState::MipmapMode::kNone,
411 paint.getBlendMode(),
412 color,
413 srcRect,
414 dstRect,
415 aa,
416 aaFlags,
417 constraint,
418 ctm,
419 std::move(textureXform));
Michael Ludwig24adb3a2019-02-27 19:42:20 +0000420 }
Michael Ludwig24adb3a2019-02-27 19:42:20 +0000421}
422
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500423// Assumes srcRect and dstRect have already been optimized to fit the proxy.
Robert Phillips16bf7d32020-07-07 10:20:27 -0400424static void draw_texture_producer(GrRecordingContext* context,
Brian Salomon777e1462020-02-28 21:10:31 -0500425 GrRenderTargetContext* rtc,
Michael Ludwig7c12e282020-05-29 09:54:07 -0400426 const GrClip* clip,
Brian Osman449b1152020-04-15 16:43:00 -0400427 const SkMatrixProvider& matrixProvider,
Brian Salomon777e1462020-02-28 21:10:31 -0500428 const SkPaint& paint,
429 GrTextureProducer* producer,
430 const SkRect& src,
431 const SkRect& dst,
432 const SkPoint dstClip[4],
433 const SkMatrix& srcToDst,
434 GrAA aa,
435 GrQuadAAFlags aaFlags,
436 SkCanvas::SrcRectConstraint constraint,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400437 GrSamplerState sampler,
Michael Ludwig47237242020-03-10 16:16:17 -0400438 bool doBicubic) {
Brian Osman449b1152020-04-15 16:43:00 -0400439 const SkMatrix& ctm(matrixProvider.localToDevice());
Brian Salomone69b9ef2020-07-22 11:18:06 -0400440 if (sampler.wrapModeX() == GrSamplerState::WrapMode::kClamp &&
441 sampler.wrapModeY() == GrSamplerState::WrapMode::kClamp && !producer->isPlanar() &&
Brian Salomon777e1462020-02-28 21:10:31 -0500442 can_use_draw_texture(paint)) {
Jim Van Verth30e0d7f2018-11-02 13:36:42 -0400443 // We've done enough checks above to allow us to pass ClampNearest() and not check for
444 // scaling adjustments.
Brian Salomon7e67dca2020-07-21 09:27:25 -0400445 auto view = producer->view(GrMipmapped::kNo);
Greg Danielcc21d0c2020-02-05 16:58:40 -0500446 if (!view) {
Jim Van Verth30e0d7f2018-11-02 13:36:42 -0400447 return;
448 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500449
Brian Salomonecbb0fb2020-02-28 18:07:32 -0500450 draw_texture(
451 rtc, clip, ctm, paint, src, dst, dstClip, aa, aaFlags, constraint, std::move(view),
452 {producer->colorType(), producer->alphaType(), sk_ref_sp(producer->colorSpace())});
Jim Van Verth30e0d7f2018-11-02 13:36:42 -0400453 return;
454 }
455
bsalomonc55271f2015-11-09 11:55:57 -0800456 const SkMaskFilter* mf = paint.getMaskFilter();
Michael Ludwigb7d64b92019-02-11 11:09:15 -0500457
bsalomonc55271f2015-11-09 11:55:57 -0800458 // The shader expects proper local coords, so we can't replace local coords with texture coords
459 // if the shader will be used. If we have a mask filter we will change the underlying geometry
460 // that is rendered.
bsalomonf1ecd212015-12-09 17:06:02 -0800461 bool canUseTextureCoordsAsLocalCoords = !use_shader(producer->isAlphaOnly(), paint) && !mf;
bsalomonc55271f2015-11-09 11:55:57 -0800462
Michael Ludwigb7d64b92019-02-11 11:09:15 -0500463 // Specifying the texture coords as local coordinates is an attempt to enable more GrDrawOp
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500464 // combining by not baking anything about the srcRect, dstRect, or ctm, into the texture
Michael Ludwigb7d64b92019-02-11 11:09:15 -0500465 // FP. In the future this should be an opaque optimization enabled by the combination of
466 // GrDrawOp/GP and FP.
467 if (mf && as_MFB(mf)->hasFragmentProcessor()) {
468 mf = nullptr;
469 }
bsalomonc55271f2015-11-09 11:55:57 -0800470
Brian Salomon0ea33072020-07-14 10:43:42 -0400471 bool restrictToSubset = SkCanvas::kStrict_SrcRectConstraint == constraint;
halcanary9d524f22016-03-29 09:03:52 -0700472
bsalomonc55271f2015-11-09 11:55:57 -0800473 // If we have to outset for AA then we will generate texture coords outside the src rect. The
474 // same happens for any mask filter that extends the bounds rendered in the dst.
475 // This is conservative as a mask filter does not have to expand the bounds rendered.
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500476 bool coordsAllInsideSrcRect = aaFlags == GrQuadAAFlags::kNone && !mf;
bsalomonc55271f2015-11-09 11:55:57 -0800477
Brian Salomona3b02f52020-07-15 16:02:01 -0400478 // Check for optimization to drop the src rect constraint when using linear filtering.
Brian Salomone69b9ef2020-07-22 11:18:06 -0400479 if (!doBicubic && sampler.filter() == GrSamplerState::Filter::kLinear && restrictToSubset &&
480 sampler.mipmapped() == GrMipmapped::kNo && coordsAllInsideSrcRect &&
481 !producer->isPlanar()) {
bsalomonb1b01992015-11-18 10:56:08 -0800482 SkMatrix combinedMatrix;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500483 combinedMatrix.setConcat(ctm, srcToDst);
Brian Salomona3b02f52020-07-15 16:02:01 -0400484 if (can_ignore_linear_filtering_subset(*producer, src, combinedMatrix, rtc->numSamples())) {
Brian Salomon0ea33072020-07-14 10:43:42 -0400485 restrictToSubset = false;
bsalomonb1b01992015-11-18 10:56:08 -0800486 }
487 }
488
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500489 SkMatrix textureMatrix;
bsalomon3aa5fce2015-11-12 09:59:44 -0800490 if (canUseTextureCoordsAsLocalCoords) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500491 textureMatrix = SkMatrix::I();
bsalomon3aa5fce2015-11-12 09:59:44 -0800492 } else {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500493 if (!srcToDst.invert(&textureMatrix)) {
bsalomon3aa5fce2015-11-12 09:59:44 -0800494 return;
495 }
bsalomon3aa5fce2015-11-12 09:59:44 -0800496 }
Brian Salomon0ea33072020-07-14 10:43:42 -0400497 const SkRect* subset = restrictToSubset ? &src : nullptr;
498 const SkRect* domain = coordsAllInsideSrcRect ? &src : nullptr;
499 std::unique_ptr<GrFragmentProcessor> fp;
500 if (doBicubic) {
Brian Salomone69b9ef2020-07-22 11:18:06 -0400501 fp = producer->createBicubicFragmentProcessor(textureMatrix, subset, domain,
Mike Reed3867c702020-09-01 13:28:10 -0400502 sampler.wrapModeX(), sampler.wrapModeY(),
503 GrBicubicEffect::gMitchell);
Brian Salomon0ea33072020-07-14 10:43:42 -0400504 } else {
Brian Salomone69b9ef2020-07-22 11:18:06 -0400505 fp = producer->createFragmentProcessor(textureMatrix, subset, domain, sampler);
Brian Salomon0ea33072020-07-14 10:43:42 -0400506 }
bsalomonc55271f2015-11-09 11:55:57 -0800507 if (!fp) {
508 return;
509 }
Brian Osman958a3bb2020-07-30 14:13:23 -0400510 fp = GrColorSpaceXformEffect::Make(std::move(fp), producer->colorSpace(), producer->alphaType(),
511 rtc->colorInfo().colorSpace(), kPremul_SkAlphaType);
512 fp = GrBlendFragmentProcessor::Make(std::move(fp), nullptr, SkBlendMode::kModulate);
joshualitt33a5fce2015-11-18 13:28:51 -0800513
bsalomonc55271f2015-11-09 11:55:57 -0800514 GrPaint grPaint;
Brian Osman449b1152020-04-15 16:43:00 -0400515 if (!SkPaintToGrPaintWithTexture(context, rtc->colorInfo(), paint, matrixProvider,
516 std::move(fp), producer->isAlphaOnly(), &grPaint)) {
bsalomonc55271f2015-11-09 11:55:57 -0800517 return;
518 }
519
520 if (!mf) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500521 // Can draw the image directly (any mask filter on the paint was converted to an FP already)
522 if (dstClip) {
523 SkPoint srcClipPoints[4];
524 SkPoint* srcClip = nullptr;
525 if (canUseTextureCoordsAsLocalCoords) {
526 // Calculate texture coordinates that match the dst clip
527 GrMapRectPoints(dst, src, dstClip, srcClipPoints, 4);
528 srcClip = srcClipPoints;
529 }
530 rtc->fillQuadWithEdgeAA(clip, std::move(grPaint), aa, aaFlags, ctm, dstClip, srcClip);
531 } else {
532 // Provide explicit texture coords when possible, otherwise rely on texture matrix
533 rtc->fillRectWithEdgeAA(clip, std::move(grPaint), aa, aaFlags, ctm, dst,
534 canUseTextureCoordsAsLocalCoords ? &src : nullptr);
535 }
536 } else {
Michael Ludwig2686d692020-04-17 20:21:37 +0000537 // Must draw the mask filter as a GrStyledShape. For now, this loses the per-edge AA
538 // information since it always draws with AA, but that should not be noticeable since the
539 // mask filter is probably a blur.
540 GrStyledShape shape;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500541 if (dstClip) {
542 // Represent it as an SkPath formed from the dstClip
543 SkPath path;
544 path.addPoly(dstClip, 4, true);
Michael Ludwig2686d692020-04-17 20:21:37 +0000545 shape = GrStyledShape(path);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500546 } else {
Michael Ludwig2686d692020-04-17 20:21:37 +0000547 shape = GrStyledShape(dst);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500548 }
549
550 GrBlurUtils::drawShapeWithMaskFilter(
551 context, rtc, clip, shape, std::move(grPaint), ctm, mf);
552 }
553}
554
Robert Phillips16bf7d32020-07-07 10:20:27 -0400555void draw_tiled_bitmap(GrRecordingContext* context,
Michael Ludwig47237242020-03-10 16:16:17 -0400556 GrRenderTargetContext* rtc,
Michael Ludwig7c12e282020-05-29 09:54:07 -0400557 const GrClip* clip,
Michael Ludwig47237242020-03-10 16:16:17 -0400558 const SkBitmap& bitmap,
559 int tileSize,
Brian Osman449b1152020-04-15 16:43:00 -0400560 const SkMatrixProvider& matrixProvider,
Michael Ludwig47237242020-03-10 16:16:17 -0400561 const SkMatrix& srcToDst,
562 const SkRect& srcRect,
563 const SkIRect& clippedSrcIRect,
564 const SkPaint& paint,
565 GrAA aa,
566 SkCanvas::SrcRectConstraint constraint,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400567 GrSamplerState sampler,
Michael Ludwig47237242020-03-10 16:16:17 -0400568 bool doBicubic) {
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400569 SkRect clippedSrcRect = SkRect::Make(clippedSrcIRect);
570
571 int nx = bitmap.width() / tileSize;
572 int ny = bitmap.height() / tileSize;
Michael Ludwig47237242020-03-10 16:16:17 -0400573
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400574 for (int x = 0; x <= nx; x++) {
575 for (int y = 0; y <= ny; y++) {
576 SkRect tileR;
577 tileR.setLTRB(SkIntToScalar(x * tileSize), SkIntToScalar(y * tileSize),
578 SkIntToScalar((x + 1) * tileSize), SkIntToScalar((y + 1) * tileSize));
579
580 if (!SkRect::Intersects(tileR, clippedSrcRect)) {
581 continue;
582 }
583
584 if (!tileR.intersect(srcRect)) {
585 continue;
586 }
587
588 SkIRect iTileR;
589 tileR.roundOut(&iTileR);
590 SkVector offset = SkPoint::Make(SkIntToScalar(iTileR.fLeft),
591 SkIntToScalar(iTileR.fTop));
592 SkRect rectToDraw = tileR;
Michael Ludwig47237242020-03-10 16:16:17 -0400593 srcToDst.mapRect(&rectToDraw);
Brian Salomone69b9ef2020-07-22 11:18:06 -0400594 if (sampler.filter() != GrSamplerState::Filter::kNearest || doBicubic) {
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400595 SkIRect iClampRect;
596
597 if (SkCanvas::kFast_SrcRectConstraint == constraint) {
598 // In bleed mode we want to always expand the tile on all edges
599 // but stay within the bitmap bounds
600 iClampRect = SkIRect::MakeWH(bitmap.width(), bitmap.height());
601 } else {
602 // In texture-domain/clamp mode we only want to expand the
603 // tile on edges interior to "srcRect" (i.e., we want to
604 // not bleed across the original clamped edges)
605 srcRect.roundOut(&iClampRect);
606 }
Michael Ludwig47237242020-03-10 16:16:17 -0400607 int outset = doBicubic ? GrBicubicEffect::kFilterTexelPad : 1;
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400608 clamped_outset_with_offset(&iTileR, outset, &offset, iClampRect);
609 }
610
611 SkBitmap tmpB;
612 if (bitmap.extractSubset(&tmpB, iTileR)) {
Michael Ludwig47237242020-03-10 16:16:17 -0400613 // We should have already handled bitmaps larger than the max texture size.
614 SkASSERT(tmpB.width() <= context->priv().caps()->maxTextureSize() &&
615 tmpB.height() <= context->priv().caps()->maxTextureSize());
616 // We should be respecting the max tile size by the time we get here.
617 SkASSERT(tmpB.width() <= context->priv().caps()->maxTileSize() &&
618 tmpB.height() <= context->priv().caps()->maxTileSize());
619
Brian Salomonbc074a62020-03-18 10:06:13 -0400620 GrBitmapTextureMaker tileProducer(context, tmpB, GrImageTexGenPolicy::kDraw);
Michael Ludwig47237242020-03-10 16:16:17 -0400621
622 GrQuadAAFlags aaFlags = GrQuadAAFlags::kNone;
623 if (aa == GrAA::kYes) {
624 // If the entire bitmap was anti-aliased, turn on AA for the outside tile edges.
625 if (tileR.fLeft <= srcRect.fLeft) {
626 aaFlags |= GrQuadAAFlags::kLeft;
627 }
628 if (tileR.fRight >= srcRect.fRight) {
629 aaFlags |= GrQuadAAFlags::kRight;
630 }
631 if (tileR.fTop <= srcRect.fTop) {
632 aaFlags |= GrQuadAAFlags::kTop;
633 }
634 if (tileR.fBottom >= srcRect.fBottom) {
635 aaFlags |= GrQuadAAFlags::kBottom;
636 }
637 }
638
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400639 // now offset it to make it "local" to our tmp bitmap
640 tileR.offset(-offset.fX, -offset.fY);
Michael Ludwig47237242020-03-10 16:16:17 -0400641 SkMatrix offsetSrcToDst = srcToDst;
642 offsetSrcToDst.preTranslate(offset.fX, offset.fY);
Brian Osman449b1152020-04-15 16:43:00 -0400643 draw_texture_producer(context, rtc, clip, matrixProvider, paint, &tileProducer,
644 tileR, rectToDraw, nullptr, offsetSrcToDst, aa, aaFlags,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400645 constraint, sampler, doBicubic);
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400646 }
647 }
648 }
649}
650
Michael Ludwig47237242020-03-10 16:16:17 -0400651} // anonymous namespace
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400652
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500653//////////////////////////////////////////////////////////////////////////////
654
Michael Ludwig278263d2020-10-12 19:49:24 +0000655void SkGpuDevice::drawSpecial(SkSpecialImage* special, const SkMatrix& localToDevice,
656 const SkPaint& paint) {
Michael Ludwig16d5b0a2020-08-31 13:07:16 -0400657 SkASSERT(!paint.getMaskFilter() && !paint.getImageFilter());
658 SkASSERT(special->isTextureBacked());
659
660 SkRect src = SkRect::Make(special->subset());
Michael Ludwig278263d2020-10-12 19:49:24 +0000661 SkRect dst = SkRect::MakeWH(special->width(), special->height());
Michael Ludwig16d5b0a2020-08-31 13:07:16 -0400662 SkMatrix srcToDst = SkMatrix::MakeRectToRect(src, dst, SkMatrix::kFill_ScaleToFit);
663
Michael Ludwig278263d2020-10-12 19:49:24 +0000664 GrSamplerState sampler(GrSamplerState::WrapMode::kClamp,
665 paint.getFilterQuality() >= kLow_SkFilterQuality ?
666 GrSamplerState::Filter::kLinear : GrSamplerState::Filter::kNearest);
667 GrAA aa = paint.isAntiAlias() ? GrAA::kYes : GrAA::kNo;
668 GrQuadAAFlags aaFlags = paint.isAntiAlias() ? GrQuadAAFlags::kAll : GrQuadAAFlags::kNone;
Michael Ludwig16d5b0a2020-08-31 13:07:16 -0400669
670 GrColorInfo colorInfo(SkColorTypeToGrColorType(special->colorType()),
671 special->alphaType(), sk_ref_sp(special->getColorSpace()));
672
673 GrSurfaceProxyView view = special->view(this->recordingContext());
674 GrTextureAdjuster texture(fContext.get(), std::move(view), colorInfo, special->uniqueID());
675 // In most cases this ought to hit draw_texture since there won't be a color filter,
676 // alpha-only texture+shader, or a high filter quality.
Michael Ludwig278263d2020-10-12 19:49:24 +0000677 SkOverrideDeviceMatrixProvider matrixProvider(this->asMatrixProvider(), localToDevice);
Michael Ludwig16d5b0a2020-08-31 13:07:16 -0400678 draw_texture_producer(fContext.get(), fRenderTargetContext.get(), this->clip(),
Michael Ludwig278263d2020-10-12 19:49:24 +0000679 matrixProvider, paint, &texture, src, dst, nullptr, srcToDst, aa,
680 aaFlags, SkCanvas::kStrict_SrcRectConstraint, sampler, false);
Michael Ludwig16d5b0a2020-08-31 13:07:16 -0400681}
682
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500683void SkGpuDevice::drawImageQuad(const SkImage* image, const SkRect* srcRect, const SkRect* dstRect,
684 const SkPoint dstClip[4], GrAA aa, GrQuadAAFlags aaFlags,
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500685 const SkMatrix* preViewMatrix, const SkPaint& paint,
686 SkCanvas::SrcRectConstraint constraint) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500687 SkRect src;
688 SkRect dst;
689 SkMatrix srcToDst;
690 ImageDrawMode mode = optimize_sample_area(SkISize::Make(image->width(), image->height()),
691 srcRect, dstRect, dstClip, &src, &dst, &srcToDst);
692 if (mode == ImageDrawMode::kSkip) {
bsalomonc55271f2015-11-09 11:55:57 -0800693 return;
694 }
695
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500696 if (src.contains(image->bounds())) {
697 constraint = SkCanvas::kFast_SrcRectConstraint;
698 }
699 // Depending on the nature of image, it can flow through more or less optimal pipelines
Brian Salomon777e1462020-02-28 21:10:31 -0500700 GrSamplerState::WrapMode wrapMode = mode == ImageDrawMode::kDecal
701 ? GrSamplerState::WrapMode::kClampToBorder
702 : GrSamplerState::WrapMode::kClamp;
Robert Phillips27927a52018-08-20 13:18:12 -0400703
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500704 // Get final CTM matrix
Brian Osman449b1152020-04-15 16:43:00 -0400705 SkPreConcatMatrixProvider matrixProvider(this->asMatrixProvider(),
706 preViewMatrix ? *preViewMatrix : SkMatrix::I());
707 const SkMatrix& ctm(matrixProvider.localToDevice());
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500708
Brian Salomone69b9ef2020-07-22 11:18:06 -0400709 bool sharpenMM = fContext->priv().options().fSharpenMipmappedTextures;
710 auto [fm, mm, bicubic] = GrInterpretFilterQuality(image->dimensions(), paint.getFilterQuality(),
John Stilesf2f9b0d2020-08-25 13:33:45 -0400711 ctm, srcToDst, sharpenMM,
712 /*allowFilterQualityReduction=*/true);
Michael Ludwig47237242020-03-10 16:16:17 -0400713
714 auto clip = this->clip();
715
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500716 // YUVA images can be stored in multiple images with different plane resolutions, so this
717 // uses an effect to combine them dynamically on the GPU. This is done before requesting a
718 // pinned texture proxy because YUV images force-flatten to RGBA in that scenario.
719 if (as_IB(image)->isYUVA()) {
720 SK_HISTOGRAM_BOOLEAN("DrawTiled", false);
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500721 LogDrawScaleFactor(ctm, srcToDst, paint.getFilterQuality());
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500722
Brian Salomon777e1462020-02-28 21:10:31 -0500723 GrYUVAImageTextureMaker maker(fContext.get(), image);
Brian Osman449b1152020-04-15 16:43:00 -0400724 draw_texture_producer(fContext.get(), fRenderTargetContext.get(), clip, matrixProvider,
725 paint, &maker, src, dst, dstClip, srcToDst, aa, aaFlags, constraint,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400726 {wrapMode, fm, mm}, bicubic);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500727 return;
728 }
729
730 // Pinned texture proxies can be rendered directly as textures, or with relatively simple
731 // adjustments applied to the image content (scaling, mipmaps, color space, etc.)
732 uint32_t pinnedUniqueID;
Robert Phillipsbedaef32020-06-29 10:37:57 -0400733 if (GrSurfaceProxyView view = as_IB(image)->refPinnedView(this->recordingContext(),
734 &pinnedUniqueID)) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500735 SK_HISTOGRAM_BOOLEAN("DrawTiled", false);
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500736 LogDrawScaleFactor(ctm, srcToDst, paint.getFilterQuality());
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500737
Greg Daniel82c6b102020-01-21 10:33:22 -0500738 GrColorInfo colorInfo;
Greg Danielcc21d0c2020-02-05 16:58:40 -0500739 if (fContext->priv().caps()->isFormatSRGB(view.proxy()->backendFormat())) {
Greg Daniel82c6b102020-01-21 10:33:22 -0500740 SkASSERT(image->imageInfo().colorType() == kRGBA_8888_SkColorType);
741 colorInfo = GrColorInfo(GrColorType::kRGBA_8888_SRGB, image->imageInfo().alphaType(),
742 image->imageInfo().refColorSpace());
743 } else {
744 colorInfo = GrColorInfo(image->imageInfo().colorInfo());
745 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500746
Brian Salomon777e1462020-02-28 21:10:31 -0500747 GrTextureAdjuster adjuster(fContext.get(), std::move(view), colorInfo, pinnedUniqueID);
Brian Osman449b1152020-04-15 16:43:00 -0400748 draw_texture_producer(fContext.get(), fRenderTargetContext.get(), clip, matrixProvider,
749 paint, &adjuster, src, dst, dstClip, srcToDst, aa, aaFlags,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400750 constraint, {wrapMode, fm, mm}, bicubic);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500751 return;
752 }
753
Michael Ludwig47237242020-03-10 16:16:17 -0400754 // Next up, determine if the image must be tiled
755 {
756 // If image is explicitly already texture backed then we shouldn't get here.
757 SkASSERT(!image->isTextureBacked());
758
759 int tileFilterPad;
Brian Salomone69b9ef2020-07-22 11:18:06 -0400760 if (bicubic) {
Michael Ludwig47237242020-03-10 16:16:17 -0400761 tileFilterPad = GrBicubicEffect::kFilterTexelPad;
762 } else if (GrSamplerState::Filter::kNearest == fm) {
763 tileFilterPad = 0;
764 } else {
765 tileFilterPad = 1;
766 }
767 int maxTileSize = fContext->priv().caps()->maxTileSize() - 2 * tileFilterPad;
768 int tileSize;
769 SkIRect clippedSubset;
770 if (should_tile_image_id(fContext.get(), SkISize::Make(fRenderTargetContext->width(),
771 fRenderTargetContext->height()),
772 clip, image->unique(), image->dimensions(), ctm, srcToDst, &src,
773 maxTileSize, &tileSize, &clippedSubset)) {
774 // Extract pixels on the CPU, since we have to split into separate textures before
775 // sending to the GPU.
776 SkBitmap bm;
Adlai Hollerbcfc5542020-08-27 12:44:07 -0400777 if (as_IB(image)->getROPixels(nullptr, &bm)) {
Michael Ludwig47237242020-03-10 16:16:17 -0400778 // This is the funnel for all paths that draw tiled bitmaps/images. Log histogram
779 SK_HISTOGRAM_BOOLEAN("DrawTiled", true);
780 LogDrawScaleFactor(ctm, srcToDst, paint.getFilterQuality());
781 draw_tiled_bitmap(fContext.get(), fRenderTargetContext.get(), clip, bm, tileSize,
Brian Osman449b1152020-04-15 16:43:00 -0400782 matrixProvider, srcToDst, src, clippedSubset, paint, aa,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400783 constraint, {wrapMode, fm, mm}, bicubic);
Michael Ludwig47237242020-03-10 16:16:17 -0400784 return;
785 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500786 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500787 }
788
789 // This is the funnel for all non-tiled bitmap/image draw calls. Log a histogram entry.
790 SK_HISTOGRAM_BOOLEAN("DrawTiled", false);
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500791 LogDrawScaleFactor(ctm, srcToDst, paint.getFilterQuality());
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500792
793 // Lazily generated images must get drawn as a texture producer that handles the final
794 // texture creation.
795 if (image->isLazyGenerated()) {
Brian Salomonbc074a62020-03-18 10:06:13 -0400796 GrImageTextureMaker maker(fContext.get(), image, GrImageTexGenPolicy::kDraw);
Brian Osman449b1152020-04-15 16:43:00 -0400797 draw_texture_producer(fContext.get(), fRenderTargetContext.get(), clip, matrixProvider,
798 paint, &maker, src, dst, dstClip, srcToDst, aa, aaFlags, constraint,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400799 {wrapMode, fm, mm}, bicubic);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500800 return;
801 }
Michael Ludwig47237242020-03-10 16:16:17 -0400802
803 SkBitmap bm;
Adlai Hollerbcfc5542020-08-27 12:44:07 -0400804 if (as_IB(image)->getROPixels(nullptr, &bm)) {
Brian Salomonbc074a62020-03-18 10:06:13 -0400805 GrBitmapTextureMaker maker(fContext.get(), bm, GrImageTexGenPolicy::kDraw);
Brian Osman449b1152020-04-15 16:43:00 -0400806 draw_texture_producer(fContext.get(), fRenderTargetContext.get(), clip, matrixProvider,
807 paint, &maker, src, dst, dstClip, srcToDst, aa, aaFlags, constraint,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400808 {wrapMode, fm, mm}, bicubic);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500809 }
810
811 // Otherwise don't know how to draw it
812}
813
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400814void SkGpuDevice::drawEdgeAAImageSet(const SkCanvas::ImageSetEntry set[], int count,
815 const SkPoint dstClips[], const SkMatrix preViewMatrices[],
816 const SkPaint& paint, SkCanvas::SrcRectConstraint constraint) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500817 SkASSERT(count > 0);
Michael Ludwig31ba7182019-04-03 10:38:06 -0400818 if (!can_use_draw_texture(paint)) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500819 // Send every entry through drawImageQuad() to handle the more complicated paint
820 int dstClipIndex = 0;
821 for (int i = 0; i < count; ++i) {
822 // Only no clip or quad clip are supported
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400823 SkASSERT(!set[i].fHasClip || dstClips);
824 SkASSERT(set[i].fMatrixIndex < 0 || preViewMatrices);
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500825
Brian Salomondb151e02019-09-17 12:11:16 -0400826 SkTCopyOnFirstWrite<SkPaint> entryPaint(paint);
827 if (set[i].fAlpha != 1.f) {
828 auto paintAlpha = paint.getAlphaf();
829 entryPaint.writable()->setAlphaf(paintAlpha * set[i].fAlpha);
830 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500831 // Always send GrAA::kYes to preserve seaming across tiling in MSAA
Brian Salomondb151e02019-09-17 12:11:16 -0400832 this->drawImageQuad(
833 set[i].fImage.get(), &set[i].fSrcRect, &set[i].fDstRect,
834 set[i].fHasClip ? dstClips + dstClipIndex : nullptr, GrAA::kYes,
835 SkToGrQuadAAFlags(set[i].fAAFlags),
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400836 set[i].fMatrixIndex < 0 ? nullptr : preViewMatrices + set[i].fMatrixIndex,
Brian Salomondb151e02019-09-17 12:11:16 -0400837 *entryPaint, constraint);
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400838 dstClipIndex += 4 * set[i].fHasClip;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500839 }
840 return;
841 }
842
Brian Salomona3b02f52020-07-15 16:02:01 -0400843 GrSamplerState::Filter filter = kNone_SkFilterQuality == paint.getFilterQuality()
844 ? GrSamplerState::Filter::kNearest
845 : GrSamplerState::Filter::kLinear;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500846 SkBlendMode mode = paint.getBlendMode();
847
848 SkAutoTArray<GrRenderTargetContext::TextureSetEntry> textures(count);
849 // We accumulate compatible proxies until we find an an incompatible one or reach the end and
Michael Ludwig379e4962019-12-06 13:21:26 -0500850 // issue the accumulated 'n' draws starting at 'base'. 'p' represents the number of proxy
851 // switches that occur within the 'n' entries.
852 int base = 0, n = 0, p = 0;
853 auto draw = [&](int nextBase) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500854 if (n > 0) {
855 auto textureXform = GrColorSpaceXform::Make(
856 set[base].fImage->colorSpace(), set[base].fImage->alphaType(),
Brian Salomon4bc0c1f2019-09-30 15:12:27 -0400857 fRenderTargetContext->colorInfo().colorSpace(), kPremul_SkAlphaType);
Brian Salomone69b9ef2020-07-22 11:18:06 -0400858 fRenderTargetContext->drawTextureSet(this->clip(),
859 textures.get() + base,
860 n,
861 p,
862 filter,
863 GrSamplerState::MipmapMode::kNone,
864 mode,
865 GrAA::kYes,
866 constraint,
867 this->localToDevice(),
868 std::move(textureXform));
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500869 }
Michael Ludwig379e4962019-12-06 13:21:26 -0500870 base = nextBase;
871 n = 0;
872 p = 0;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500873 };
874 int dstClipIndex = 0;
875 for (int i = 0; i < count; ++i) {
Michael Ludwigd9958f82019-03-21 13:08:36 -0400876 SkASSERT(!set[i].fHasClip || dstClips);
877 SkASSERT(set[i].fMatrixIndex < 0 || preViewMatrices);
878
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500879 // Manage the dst clip pointer tracking before any continues are used so we don't lose
880 // our place in the dstClips array.
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400881 const SkPoint* clip = set[i].fHasClip ? dstClips + dstClipIndex : nullptr;
882 dstClipIndex += 4 * set[i].fHasClip;
883
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500884 // The default SkBaseDevice implementation is based on drawImageRect which does not allow
885 // non-sorted src rects. TODO: Decide this is OK or make sure we handle it.
886 if (!set[i].fSrcRect.isSorted()) {
Michael Ludwig379e4962019-12-06 13:21:26 -0500887 draw(i + 1);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500888 continue;
889 }
890
Greg Danielcc21d0c2020-02-05 16:58:40 -0500891 GrSurfaceProxyView view;
Michael Ludwigd9958f82019-03-21 13:08:36 -0400892 const SkImage_Base* image = as_IB(set[i].fImage.get());
Greg Danielcc21d0c2020-02-05 16:58:40 -0500893 // Extract view from image, but skip YUV images so they get processed through
Michael Ludwigd9958f82019-03-21 13:08:36 -0400894 // drawImageQuad and the proper effect to dynamically sample their planes.
895 if (!image->isYUVA()) {
896 uint32_t uniqueID;
Robert Phillipsbedaef32020-06-29 10:37:57 -0400897 view = image->refPinnedView(this->recordingContext(), &uniqueID);
Greg Danielcc21d0c2020-02-05 16:58:40 -0500898 if (!view) {
Brian Salomon7e67dca2020-07-21 09:27:25 -0400899 view = image->refView(this->recordingContext(), GrMipmapped::kNo);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500900 }
901 }
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500902
Greg Danielcc21d0c2020-02-05 16:58:40 -0500903 if (!view) {
Michael Ludwigd9958f82019-03-21 13:08:36 -0400904 // This image can't go through the texture op, send through general image pipeline
905 // after flushing current batch.
Michael Ludwig379e4962019-12-06 13:21:26 -0500906 draw(i + 1);
Brian Salomondb151e02019-09-17 12:11:16 -0400907 SkTCopyOnFirstWrite<SkPaint> entryPaint(paint);
908 if (set[i].fAlpha != 1.f) {
909 auto paintAlpha = paint.getAlphaf();
910 entryPaint.writable()->setAlphaf(paintAlpha * set[i].fAlpha);
911 }
912 this->drawImageQuad(
913 image, &set[i].fSrcRect, &set[i].fDstRect, clip, GrAA::kYes,
Michael Ludwigd9958f82019-03-21 13:08:36 -0400914 SkToGrQuadAAFlags(set[i].fAAFlags),
915 set[i].fMatrixIndex < 0 ? nullptr : preViewMatrices + set[i].fMatrixIndex,
Brian Salomondb151e02019-09-17 12:11:16 -0400916 *entryPaint, constraint);
Michael Ludwigd9958f82019-03-21 13:08:36 -0400917 continue;
918 }
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500919
Greg Danielcc21d0c2020-02-05 16:58:40 -0500920 textures[i].fProxyView = std::move(view);
Brian Salomonfc118442019-11-22 19:09:27 -0500921 textures[i].fSrcAlphaType = image->alphaType();
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500922 textures[i].fSrcRect = set[i].fSrcRect;
923 textures[i].fDstRect = set[i].fDstRect;
924 textures[i].fDstClipQuad = clip;
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400925 textures[i].fPreViewMatrix =
926 set[i].fMatrixIndex < 0 ? nullptr : preViewMatrices + set[i].fMatrixIndex;
Michael Ludwig1c66ad92020-07-10 08:59:44 -0400927 textures[i].fColor = texture_color(paint.getColor4f(), set[i].fAlpha,
928 SkColorTypeToGrColorType(image->colorType()),
929 fRenderTargetContext->colorInfo());
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500930 textures[i].fAAFlags = SkToGrQuadAAFlags(set[i].fAAFlags);
931
932 if (n > 0 &&
Greg Daniel549325c2019-10-30 16:19:20 -0400933 (!GrTextureProxy::ProxiesAreCompatibleAsDynamicState(
Michael Ludwigfcdd0612019-11-25 08:34:31 -0500934 textures[i].fProxyView.proxy(),
935 textures[base].fProxyView.proxy()) ||
Greg Daniel507736f2020-01-17 15:36:10 -0500936 textures[i].fProxyView.swizzle() != textures[base].fProxyView.swizzle() ||
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500937 set[i].fImage->alphaType() != set[base].fImage->alphaType() ||
938 !SkColorSpace::Equals(set[i].fImage->colorSpace(), set[base].fImage->colorSpace()))) {
Michael Ludwig379e4962019-12-06 13:21:26 -0500939 draw(i);
940 }
941 // Whether or not we submitted a draw in the above if(), this ith entry is in the current
942 // set being accumulated so increment n, and increment p if proxies are different.
943 ++n;
944 if (n == 1 || textures[i - 1].fProxyView.proxy() != textures[i].fProxyView.proxy()) {
945 // First proxy or a different proxy (that is compatible, otherwise we'd have drawn up
946 // to i - 1).
947 ++p;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500948 }
949 }
Michael Ludwig379e4962019-12-06 13:21:26 -0500950 draw(count);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500951}