blob: 4c3de12a84bb999c82634d362cbc25fb8f847a24 [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
Robert Phillips16bf7d32020-07-07 10:20:27 -040010#include "include/gpu/GrDirectContext.h"
11#include "include/gpu/GrRecordingContext.h"
Mike Klein8aa0edf2020-10-16 11:04:18 -050012#include "include/private/SkTPin.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050013#include "src/core/SkDraw.h"
14#include "src/core/SkMaskFilterBase.h"
Michael Ludwig16d5b0a2020-08-31 13:07:16 -040015#include "src/core/SkSpecialImage.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050016#include "src/gpu/GrBitmapTextureMaker.h"
17#include "src/gpu/GrBlurUtils.h"
18#include "src/gpu/GrCaps.h"
19#include "src/gpu/GrColorSpaceXform.h"
20#include "src/gpu/GrImageTextureMaker.h"
Robert Phillips16bf7d32020-07-07 10:20:27 -040021#include "src/gpu/GrRecordingContextPriv.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050022#include "src/gpu/GrStyle.h"
Brian Salomoneebe7352020-12-09 16:37:04 -050023#include "src/gpu/GrSurfaceDrawContext.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050024#include "src/gpu/GrTextureAdjuster.h"
25#include "src/gpu/GrTextureMaker.h"
26#include "src/gpu/SkGr.h"
27#include "src/gpu/effects/GrBicubicEffect.h"
John Stilesf743d4e2020-07-23 11:35:08 -040028#include "src/gpu/effects/GrBlendFragmentProcessor.h"
Brian Salomonb8f098d2020-01-07 11:15:44 -050029#include "src/gpu/effects/GrTextureEffect.h"
Michael Ludwig2686d692020-04-17 20:21:37 +000030#include "src/gpu/geometry/GrStyledShape.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050031#include "src/image/SkImage_Base.h"
bsalomonc55271f2015-11-09 11:55:57 -080032
Michael Ludwig1433cfd2019-02-27 17:12:30 -050033namespace {
34
bsalomonc55271f2015-11-09 11:55:57 -080035static inline bool use_shader(bool textureIsAlphaOnly, const SkPaint& paint) {
36 return textureIsAlphaOnly && paint.getShader();
37}
38
bsalomonb1b01992015-11-18 10:56:08 -080039//////////////////////////////////////////////////////////////////////////////
Brian Salomona3b02f52020-07-15 16:02:01 -040040// Helper functions for dropping src rect subset with GrSamplerState::Filter::kLinear.
bsalomonb1b01992015-11-18 10:56:08 -080041
42static const SkScalar kColorBleedTolerance = 0.001f;
43
44static bool has_aligned_samples(const SkRect& srcRect, const SkRect& transformedRect) {
45 // detect pixel disalignment
Brian Salomona911f8f2015-11-18 15:19:57 -050046 if (SkScalarAbs(SkScalarRoundToScalar(transformedRect.left()) - transformedRect.left()) < kColorBleedTolerance &&
47 SkScalarAbs(SkScalarRoundToScalar(transformedRect.top()) - transformedRect.top()) < kColorBleedTolerance &&
48 SkScalarAbs(transformedRect.width() - srcRect.width()) < kColorBleedTolerance &&
bsalomonb1b01992015-11-18 10:56:08 -080049 SkScalarAbs(transformedRect.height() - srcRect.height()) < kColorBleedTolerance) {
50 return true;
51 }
52 return false;
53}
54
55static bool may_color_bleed(const SkRect& srcRect,
56 const SkRect& transformedRect,
57 const SkMatrix& m,
Chris Dalton6ce447a2019-06-23 18:07:38 -060058 int numSamples) {
bsalomonb1b01992015-11-18 10:56:08 -080059 // Only gets called if has_aligned_samples returned false.
60 // So we can assume that sampling is axis aligned but not texel aligned.
61 SkASSERT(!has_aligned_samples(srcRect, transformedRect));
62 SkRect innerSrcRect(srcRect), innerTransformedRect, outerTransformedRect(transformedRect);
Chris Dalton6ce447a2019-06-23 18:07:38 -060063 if (numSamples > 1) {
bsalomonb1b01992015-11-18 10:56:08 -080064 innerSrcRect.inset(SK_Scalar1, SK_Scalar1);
65 } else {
66 innerSrcRect.inset(SK_ScalarHalf, SK_ScalarHalf);
67 }
68 m.mapRect(&innerTransformedRect, innerSrcRect);
69
70 // The gap between outerTransformedRect and innerTransformedRect
71 // represents the projection of the source border area, which is
72 // problematic for color bleeding. We must check whether any
73 // destination pixels sample the border area.
74 outerTransformedRect.inset(kColorBleedTolerance, kColorBleedTolerance);
75 innerTransformedRect.outset(kColorBleedTolerance, kColorBleedTolerance);
76 SkIRect outer, inner;
77 outerTransformedRect.round(&outer);
78 innerTransformedRect.round(&inner);
79 // If the inner and outer rects round to the same result, it means the
80 // border does not overlap any pixel centers. Yay!
81 return inner != outer;
82}
83
Brian Salomona3b02f52020-07-15 16:02:01 -040084static bool can_ignore_linear_filtering_subset(const GrTextureProducer& producer,
85 const SkRect& srcSubset,
86 const SkMatrix& srcRectToDeviceSpace,
87 int numSamples) {
bsalomonb1b01992015-11-18 10:56:08 -080088 if (srcRectToDeviceSpace.rectStaysRect()) {
89 // sampling is axis-aligned
90 SkRect transformedRect;
Brian Salomon8f32f132020-07-14 12:30:12 -040091 srcRectToDeviceSpace.mapRect(&transformedRect, srcSubset);
bsalomonb1b01992015-11-18 10:56:08 -080092
Brian Salomon8f32f132020-07-14 12:30:12 -040093 if (has_aligned_samples(srcSubset, transformedRect) ||
94 !may_color_bleed(srcSubset, transformedRect, srcRectToDeviceSpace, numSamples)) {
bsalomonb1b01992015-11-18 10:56:08 -080095 return true;
96 }
97 }
98 return false;
99}
100
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400101//////////////////////////////////////////////////////////////////////////////
102// Helper functions for tiling a large SkBitmap
103
104static const int kBmpSmallTileSize = 1 << 10;
105
106static inline int get_tile_count(const SkIRect& srcRect, int tileSize) {
107 int tilesX = (srcRect.fRight / tileSize) - (srcRect.fLeft / tileSize) + 1;
108 int tilesY = (srcRect.fBottom / tileSize) - (srcRect.fTop / tileSize) + 1;
109 return tilesX * tilesY;
110}
111
112static int determine_tile_size(const SkIRect& src, int maxTileSize) {
113 if (maxTileSize <= kBmpSmallTileSize) {
114 return maxTileSize;
115 }
116
117 size_t maxTileTotalTileSize = get_tile_count(src, maxTileSize);
118 size_t smallTotalTileSize = get_tile_count(src, kBmpSmallTileSize);
119
120 maxTileTotalTileSize *= maxTileSize * maxTileSize;
121 smallTotalTileSize *= kBmpSmallTileSize * kBmpSmallTileSize;
122
123 if (maxTileTotalTileSize > 2 * smallTotalTileSize) {
124 return kBmpSmallTileSize;
125 } else {
126 return maxTileSize;
127 }
128}
129
130// Given a bitmap, an optional src rect, and a context with a clip and matrix determine what
131// pixels from the bitmap are necessary.
Michael Ludwigc002d562020-05-13 14:17:57 -0400132static SkIRect determine_clipped_src_rect(int width, int height,
Michael Ludwig7c12e282020-05-29 09:54:07 -0400133 const GrClip* clip,
Michael Ludwigc002d562020-05-13 14:17:57 -0400134 const SkMatrix& viewMatrix,
135 const SkMatrix& srcToDstRect,
136 const SkISize& imageDimensions,
137 const SkRect* srcRectPtr) {
Michael Ludwige06a8972020-06-11 10:29:00 -0400138 SkIRect clippedSrcIRect = clip ? clip->getConservativeBounds()
Michael Ludwig7c12e282020-05-29 09:54:07 -0400139 : SkIRect::MakeWH(width, height);
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400140 SkMatrix inv = SkMatrix::Concat(viewMatrix, srcToDstRect);
141 if (!inv.invert(&inv)) {
Michael Ludwigc002d562020-05-13 14:17:57 -0400142 return SkIRect::MakeEmpty();
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400143 }
Michael Ludwigc002d562020-05-13 14:17:57 -0400144 SkRect clippedSrcRect = SkRect::Make(clippedSrcIRect);
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400145 inv.mapRect(&clippedSrcRect);
146 if (srcRectPtr) {
147 if (!clippedSrcRect.intersect(*srcRectPtr)) {
Michael Ludwigc002d562020-05-13 14:17:57 -0400148 return SkIRect::MakeEmpty();
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400149 }
150 }
Michael Ludwigc002d562020-05-13 14:17:57 -0400151 clippedSrcRect.roundOut(&clippedSrcIRect);
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400152 SkIRect bmpBounds = SkIRect::MakeSize(imageDimensions);
Michael Ludwigc002d562020-05-13 14:17:57 -0400153 if (!clippedSrcIRect.intersect(bmpBounds)) {
154 return SkIRect::MakeEmpty();
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400155 }
Michael Ludwigc002d562020-05-13 14:17:57 -0400156
157 return clippedSrcIRect;
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400158}
159
Michael Ludwig47237242020-03-10 16:16:17 -0400160// tileSize and clippedSubset are valid if true is returned
Robert Phillips16bf7d32020-07-07 10:20:27 -0400161static bool should_tile_image_id(GrRecordingContext* context,
Michael Ludwig47237242020-03-10 16:16:17 -0400162 SkISize rtSize,
Michael Ludwig7c12e282020-05-29 09:54:07 -0400163 const GrClip* clip,
Michael Ludwig47237242020-03-10 16:16:17 -0400164 uint32_t imageID,
165 const SkISize& imageSize,
166 const SkMatrix& ctm,
167 const SkMatrix& srcToDst,
168 const SkRect* src,
169 int maxTileSize,
170 int* tileSize,
171 SkIRect* clippedSubset) {
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400172 // if it's larger than the max tile size, then we have no choice but tiling.
Michael Ludwig47237242020-03-10 16:16:17 -0400173 if (imageSize.width() > maxTileSize || imageSize.height() > maxTileSize) {
Michael Ludwigc002d562020-05-13 14:17:57 -0400174 *clippedSubset = determine_clipped_src_rect(rtSize.width(), rtSize.height(), clip, ctm,
175 srcToDst, imageSize, src);
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400176 *tileSize = determine_tile_size(*clippedSubset, maxTileSize);
177 return true;
178 }
179
180 // If the image would only produce 4 tiles of the smaller size, don't bother tiling it.
Michael Ludwig47237242020-03-10 16:16:17 -0400181 const size_t area = imageSize.width() * imageSize.height();
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400182 if (area < 4 * kBmpSmallTileSize * kBmpSmallTileSize) {
183 return false;
184 }
185
Michael Ludwig46d91382020-05-07 09:51:12 -0400186 // At this point we know we could do the draw by uploading the entire bitmap as a texture.
187 // However, if the texture would be large compared to the cache size and we don't require most
188 // of it for this draw then tile to reduce the amount of upload and cache spill.
189 // NOTE: if the context is not a direct context, it doesn't have access to the resource cache,
190 // and theoretically, the resource cache's limits could be being changed on another thread, so
191 // even having access to just the limit wouldn't be a reliable test during recording here.
192 // Instead, we will just upload the entire image to be on the safe side and not tile.
Robert Phillips16bf7d32020-07-07 10:20:27 -0400193 auto direct = context->asDirectContext();
194 if (!direct) {
Michael Ludwig46d91382020-05-07 09:51:12 -0400195 return false;
196 }
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400197
198 // assumption here is that sw bitmap size is a good proxy for its size as
199 // a texture
200 size_t bmpSize = area * sizeof(SkPMColor); // assume 32bit pixels
Robert Phillips16bf7d32020-07-07 10:20:27 -0400201 size_t cacheSize = direct->getResourceCacheLimit();
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400202 if (bmpSize < cacheSize / 2) {
203 return false;
204 }
205
206 // Figure out how much of the src we will need based on the src rect and clipping. Reject if
207 // tiling memory savings would be < 50%.
Michael Ludwigc002d562020-05-13 14:17:57 -0400208 *clippedSubset = determine_clipped_src_rect(rtSize.width(), rtSize.height(), clip, ctm,
209 srcToDst, imageSize, src);
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400210 *tileSize = kBmpSmallTileSize; // already know whole bitmap fits in one max sized tile.
211 size_t usedTileBytes = get_tile_count(*clippedSubset, kBmpSmallTileSize) *
212 kBmpSmallTileSize * kBmpSmallTileSize *
213 sizeof(SkPMColor); // assume 32bit pixels;
214
215 return usedTileBytes * 2 < bmpSize;
216}
217
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400218// This method outsets 'iRect' by 'outset' all around and then clamps its extents to
219// 'clamp'. 'offset' is adjusted to remain positioned over the top-left corner
220// of 'iRect' for all possible outsets/clamps.
Michael Ludwig47237242020-03-10 16:16:17 -0400221static inline void clamped_outset_with_offset(SkIRect* iRect, int outset, SkPoint* offset,
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400222 const SkIRect& clamp) {
223 iRect->outset(outset, outset);
224
225 int leftClampDelta = clamp.fLeft - iRect->fLeft;
226 if (leftClampDelta > 0) {
227 offset->fX -= outset - leftClampDelta;
228 iRect->fLeft = clamp.fLeft;
229 } else {
230 offset->fX -= outset;
231 }
232
233 int topClampDelta = clamp.fTop - iRect->fTop;
234 if (topClampDelta > 0) {
235 offset->fY -= outset - topClampDelta;
236 iRect->fTop = clamp.fTop;
237 } else {
238 offset->fY -= outset;
239 }
240
241 if (iRect->fRight > clamp.fRight) {
242 iRect->fRight = clamp.fRight;
243 }
244 if (iRect->fBottom > clamp.fBottom) {
245 iRect->fBottom = clamp.fBottom;
246 }
247}
248
249//////////////////////////////////////////////////////////////////////////////
Brian Salomoneebe7352020-12-09 16:37:04 -0500250// Helper functions for drawing an image with GrSurfaceDrawContext
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400251
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500252enum class ImageDrawMode {
253 // Src and dst have been restricted to the image content. May need to clamp, no need to decal.
254 kOptimized,
255 // Src and dst are their original sizes, requires use of a decal instead of plain clamping.
256 // This is used when a dst clip is provided and extends outside of the optimized dst rect.
257 kDecal,
258 // Src or dst are empty, or do not intersect the image content so don't draw anything.
259 kSkip
260};
261
262/**
263 * Optimize the src rect sampling area within an image (sized 'width' x 'height') such that
264 * 'outSrcRect' will be completely contained in the image's bounds. The corresponding rect
265 * to draw will be output to 'outDstRect'. The mapping between src and dst will be cached in
266 * 'srcToDst'. Outputs are not always updated when kSkip is returned.
267 *
268 * If 'origSrcRect' is null, implicitly use the image bounds. If 'origDstRect' is null, use the
269 * original src rect. 'dstClip' should be null when there is no additional clipping.
270 */
271static ImageDrawMode optimize_sample_area(const SkISize& image, const SkRect* origSrcRect,
272 const SkRect* origDstRect, const SkPoint dstClip[4],
273 SkRect* outSrcRect, SkRect* outDstRect,
274 SkMatrix* srcToDst) {
275 SkRect srcBounds = SkRect::MakeIWH(image.fWidth, image.fHeight);
276
277 SkRect src = origSrcRect ? *origSrcRect : srcBounds;
278 SkRect dst = origDstRect ? *origDstRect : src;
279
280 if (src.isEmpty() || dst.isEmpty()) {
281 return ImageDrawMode::kSkip;
282 }
283
284 if (outDstRect) {
Mike Reed2ac6ce82021-01-15 12:26:22 -0500285 *srcToDst = SkMatrix::RectToRect(src, dst);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500286 } else {
287 srcToDst->setIdentity();
288 }
289
290 if (origSrcRect && !srcBounds.contains(src)) {
291 if (!src.intersect(srcBounds)) {
292 return ImageDrawMode::kSkip;
293 }
294 srcToDst->mapRect(&dst, src);
295
296 // Both src and dst have gotten smaller. If dstClip is provided, confirm it is still
297 // contained in dst, otherwise cannot optimize the sample area and must use a decal instead
298 if (dstClip) {
299 for (int i = 0; i < 4; ++i) {
300 if (!dst.contains(dstClip[i].fX, dstClip[i].fY)) {
301 // Must resort to using a decal mode restricted to the clipped 'src', and
302 // use the original dst rect (filling in src bounds as needed)
303 *outSrcRect = src;
304 *outDstRect = (origDstRect ? *origDstRect
305 : (origSrcRect ? *origSrcRect : srcBounds));
306 return ImageDrawMode::kDecal;
307 }
308 }
309 }
310 }
311
312 // The original src and dst were fully contained in the image, or there was no dst clip to
313 // worry about, or the clip was still contained in the restricted dst rect.
314 *outSrcRect = src;
315 *outDstRect = dst;
316 return ImageDrawMode::kOptimized;
317}
318
Brian Salomon34169692017-08-28 15:32:01 -0400319/**
Brian Salomoneebe7352020-12-09 16:37:04 -0500320 * Checks whether the paint is compatible with using GrSurfaceDrawContext::drawTexture. It is more
Brian Salomonb80ffee2018-05-23 16:39:39 -0400321 * efficient than the GrTextureProducer general case.
Brian Salomon34169692017-08-28 15:32:01 -0400322 */
Mike Reed4f23dec2020-12-30 14:22:42 +0000323static bool can_use_draw_texture(const SkPaint& paint, bool useCubicResampler, SkMipmapMode mm) {
Brian Salomon34169692017-08-28 15:32:01 -0400324 return (!paint.getColorFilter() && !paint.getShader() && !paint.getMaskFilter() &&
Mike Reed4f23dec2020-12-30 14:22:42 +0000325 !paint.getImageFilter() && !useCubicResampler && mm == SkMipmapMode::kNone);
Brian Salomon34169692017-08-28 15:32:01 -0400326}
327
Michael Ludwig1c66ad92020-07-10 08:59:44 -0400328static SkPMColor4f texture_color(SkColor4f paintColor, float entryAlpha, GrColorType srcColorType,
329 const GrColorInfo& dstColorInfo) {
330 paintColor.fA *= entryAlpha;
331 if (GrColorTypeIsAlphaOnly(srcColorType)) {
332 return SkColor4fPrepForDst(paintColor, dstColorInfo).premul();
333 } else {
334 float paintAlpha = SkTPin(paintColor.fA, 0.f, 1.f);
335 return { paintAlpha, paintAlpha, paintAlpha, paintAlpha };
336 }
337}
338
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500339// Assumes srcRect and dstRect have already been optimized to fit the proxy
Brian Salomoneebe7352020-12-09 16:37:04 -0500340static void draw_texture(GrSurfaceDrawContext* rtc,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400341 const GrClip* clip,
342 const SkMatrix& ctm,
343 const SkPaint& paint,
Mike Reed869a7172020-12-28 14:45:28 -0500344 GrSamplerState::Filter filter,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400345 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 Salomonb43d6992021-01-05 14:37:40 -0500353 if (GrColorTypeIsAlphaOnly(srcColorInfo.colorType())) {
354 view.concatSwizzle(GrSwizzle("aaaa"));
355 }
Brian Salomonbacbb922021-01-21 19:48:00 -0500356 const GrColorInfo& dstInfo = rtc->colorInfo();
357 auto textureXform = GrColorSpaceXform::Make(srcColorInfo, rtc->colorInfo());
Greg Daniel2f3cd4f2020-02-07 11:07:25 -0500358 GrSurfaceProxy* proxy = view.proxy();
Michael Ludwigd54ca8f2019-02-13 13:25:21 -0500359 // Must specify the strict constraint when the proxy is not functionally exact and the src
360 // rect would access pixels outside the proxy's content area without the constraint.
Brian Salomon5c60b752019-12-13 15:03:43 -0500361 if (constraint != SkCanvas::kStrict_SrcRectConstraint && !proxy->isFunctionallyExact()) {
Michael Ludwigd54ca8f2019-02-13 13:25:21 -0500362 // Conservative estimate of how much a coord could be outset from src rect:
Brian Salomona3b02f52020-07-15 16:02:01 -0400363 // 1/2 pixel for AA and 1/2 pixel for linear filtering
Michael Ludwigd54ca8f2019-02-13 13:25:21 -0500364 float buffer = 0.5f * (aa == GrAA::kYes) +
Brian Salomona3b02f52020-07-15 16:02:01 -0400365 0.5f * (filter == GrSamplerState::Filter::kLinear);
Brian Salomon9f2b86c2019-10-22 10:37:46 -0400366 SkRect safeBounds = proxy->getBoundsRect();
Michael Ludwigd54ca8f2019-02-13 13:25:21 -0500367 safeBounds.inset(buffer, buffer);
368 if (!safeBounds.contains(srcRect)) {
369 constraint = SkCanvas::kStrict_SrcRectConstraint;
370 }
371 }
Brian Salomon34169692017-08-28 15:32:01 -0400372
Michael Ludwig1c66ad92020-07-10 08:59:44 -0400373 SkPMColor4f color = texture_color(paint.getColor4f(), 1.f, srcColorInfo.colorType(), dstInfo);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500374 if (dstClip) {
375 // Get source coords corresponding to dstClip
376 SkPoint srcQuad[4];
377 GrMapRectPoints(dstRect, srcRect, dstClip, srcQuad, 4);
Michael Ludwig24adb3a2019-02-27 19:42:20 +0000378
Brian Salomone69b9ef2020-07-22 11:18:06 -0400379 rtc->drawTextureQuad(clip,
380 std::move(view),
381 srcColorInfo.colorType(),
382 srcColorInfo.alphaType(),
383 filter,
384 GrSamplerState::MipmapMode::kNone,
385 paint.getBlendMode(),
386 color,
387 srcQuad,
388 dstClip,
389 aa,
390 aaFlags,
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500391 constraint == SkCanvas::kStrict_SrcRectConstraint ? &srcRect : nullptr,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400392 ctm,
393 std::move(textureXform));
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500394 } else {
Brian Salomone69b9ef2020-07-22 11:18:06 -0400395 rtc->drawTexture(clip,
396 std::move(view),
397 srcColorInfo.alphaType(),
398 filter,
399 GrSamplerState::MipmapMode::kNone,
400 paint.getBlendMode(),
401 color,
402 srcRect,
403 dstRect,
404 aa,
405 aaFlags,
406 constraint,
407 ctm,
408 std::move(textureXform));
Michael Ludwig24adb3a2019-02-27 19:42:20 +0000409 }
Michael Ludwig24adb3a2019-02-27 19:42:20 +0000410}
411
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500412// Assumes srcRect and dstRect have already been optimized to fit the proxy.
Robert Phillips16bf7d32020-07-07 10:20:27 -0400413static void draw_texture_producer(GrRecordingContext* context,
Brian Salomoneebe7352020-12-09 16:37:04 -0500414 GrSurfaceDrawContext* rtc,
Michael Ludwig7c12e282020-05-29 09:54:07 -0400415 const GrClip* clip,
Brian Osman449b1152020-04-15 16:43:00 -0400416 const SkMatrixProvider& matrixProvider,
Brian Salomon777e1462020-02-28 21:10:31 -0500417 const SkPaint& paint,
418 GrTextureProducer* producer,
419 const SkRect& src,
420 const SkRect& dst,
421 const SkPoint dstClip[4],
422 const SkMatrix& srcToDst,
423 GrAA aa,
424 GrQuadAAFlags aaFlags,
425 SkCanvas::SrcRectConstraint constraint,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400426 GrSamplerState sampler,
Mike Reed52130b02020-12-28 15:33:13 -0500427 SkCubicResampler cubic) {
Brian Osman449b1152020-04-15 16:43:00 -0400428 const SkMatrix& ctm(matrixProvider.localToDevice());
Brian Salomone69b9ef2020-07-22 11:18:06 -0400429 if (sampler.wrapModeX() == GrSamplerState::WrapMode::kClamp &&
430 sampler.wrapModeY() == GrSamplerState::WrapMode::kClamp && !producer->isPlanar() &&
Mike Reed4f23dec2020-12-30 14:22:42 +0000431 can_use_draw_texture(paint, GrValidCubicResampler(cubic), sampler.mipmapMode())) {
Jim Van Verth30e0d7f2018-11-02 13:36:42 -0400432 // We've done enough checks above to allow us to pass ClampNearest() and not check for
433 // scaling adjustments.
Brian Salomon7e67dca2020-07-21 09:27:25 -0400434 auto view = producer->view(GrMipmapped::kNo);
Greg Danielcc21d0c2020-02-05 16:58:40 -0500435 if (!view) {
Jim Van Verth30e0d7f2018-11-02 13:36:42 -0400436 return;
437 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500438
Mike Reed869a7172020-12-28 14:45:28 -0500439 draw_texture(rtc,
440 clip,
441 ctm,
442 paint,
443 sampler.filter(),
444 src,
445 dst,
446 dstClip,
447 aa,
448 aaFlags,
449 constraint,
450 std::move(view),
Brian Salomon490f1922021-02-01 13:14:37 -0500451 producer->colorInfo());
Jim Van Verth30e0d7f2018-11-02 13:36:42 -0400452 return;
453 }
454
bsalomonc55271f2015-11-09 11:55:57 -0800455 const SkMaskFilter* mf = paint.getMaskFilter();
Michael Ludwigb7d64b92019-02-11 11:09:15 -0500456
bsalomonc55271f2015-11-09 11:55:57 -0800457 // The shader expects proper local coords, so we can't replace local coords with texture coords
458 // if the shader will be used. If we have a mask filter we will change the underlying geometry
459 // that is rendered.
bsalomonf1ecd212015-12-09 17:06:02 -0800460 bool canUseTextureCoordsAsLocalCoords = !use_shader(producer->isAlphaOnly(), paint) && !mf;
bsalomonc55271f2015-11-09 11:55:57 -0800461
Michael Ludwigb7d64b92019-02-11 11:09:15 -0500462 // Specifying the texture coords as local coordinates is an attempt to enable more GrDrawOp
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500463 // combining by not baking anything about the srcRect, dstRect, or ctm, into the texture
Michael Ludwigb7d64b92019-02-11 11:09:15 -0500464 // FP. In the future this should be an opaque optimization enabled by the combination of
465 // GrDrawOp/GP and FP.
466 if (mf && as_MFB(mf)->hasFragmentProcessor()) {
467 mf = nullptr;
468 }
bsalomonc55271f2015-11-09 11:55:57 -0800469
Brian Salomon0ea33072020-07-14 10:43:42 -0400470 bool restrictToSubset = SkCanvas::kStrict_SrcRectConstraint == constraint;
halcanary9d524f22016-03-29 09:03:52 -0700471
bsalomonc55271f2015-11-09 11:55:57 -0800472 // If we have to outset for AA then we will generate texture coords outside the src rect. The
473 // same happens for any mask filter that extends the bounds rendered in the dst.
474 // This is conservative as a mask filter does not have to expand the bounds rendered.
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500475 bool coordsAllInsideSrcRect = aaFlags == GrQuadAAFlags::kNone && !mf;
bsalomonc55271f2015-11-09 11:55:57 -0800476
Brian Salomona3b02f52020-07-15 16:02:01 -0400477 // Check for optimization to drop the src rect constraint when using linear filtering.
Mike Reed52130b02020-12-28 15:33:13 -0500478 if (!GrValidCubicResampler(cubic) &&
479 sampler.filter() == GrSamplerState::Filter::kLinear && restrictToSubset &&
Brian Salomone69b9ef2020-07-22 11:18:06 -0400480 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;
Mike Reed52130b02020-12-28 15:33:13 -0500500 if (GrValidCubicResampler(cubic)) {
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(),
Mike Reed52130b02020-12-28 15:33:13 -0500503 cubic);
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);
Brian Salomonb43d6992021-01-05 14:37:40 -0500512 if (producer->isAlphaOnly()) {
513 fp = GrBlendFragmentProcessor::Make(std::move(fp), nullptr, SkBlendMode::kDstIn);
514 } else {
515 fp = GrBlendFragmentProcessor::Make(std::move(fp), nullptr, SkBlendMode::kSrcIn);
516 }
joshualitt33a5fce2015-11-18 13:28:51 -0800517
bsalomonc55271f2015-11-09 11:55:57 -0800518 GrPaint grPaint;
Brian Osman449b1152020-04-15 16:43:00 -0400519 if (!SkPaintToGrPaintWithTexture(context, rtc->colorInfo(), paint, matrixProvider,
520 std::move(fp), producer->isAlphaOnly(), &grPaint)) {
bsalomonc55271f2015-11-09 11:55:57 -0800521 return;
522 }
523
524 if (!mf) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500525 // Can draw the image directly (any mask filter on the paint was converted to an FP already)
526 if (dstClip) {
527 SkPoint srcClipPoints[4];
528 SkPoint* srcClip = nullptr;
529 if (canUseTextureCoordsAsLocalCoords) {
530 // Calculate texture coordinates that match the dst clip
531 GrMapRectPoints(dst, src, dstClip, srcClipPoints, 4);
532 srcClip = srcClipPoints;
533 }
534 rtc->fillQuadWithEdgeAA(clip, std::move(grPaint), aa, aaFlags, ctm, dstClip, srcClip);
535 } else {
536 // Provide explicit texture coords when possible, otherwise rely on texture matrix
537 rtc->fillRectWithEdgeAA(clip, std::move(grPaint), aa, aaFlags, ctm, dst,
538 canUseTextureCoordsAsLocalCoords ? &src : nullptr);
539 }
540 } else {
Michael Ludwig2686d692020-04-17 20:21:37 +0000541 // Must draw the mask filter as a GrStyledShape. For now, this loses the per-edge AA
542 // information since it always draws with AA, but that should not be noticeable since the
543 // mask filter is probably a blur.
544 GrStyledShape shape;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500545 if (dstClip) {
546 // Represent it as an SkPath formed from the dstClip
547 SkPath path;
548 path.addPoly(dstClip, 4, true);
Michael Ludwig2686d692020-04-17 20:21:37 +0000549 shape = GrStyledShape(path);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500550 } else {
Michael Ludwig2686d692020-04-17 20:21:37 +0000551 shape = GrStyledShape(dst);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500552 }
553
554 GrBlurUtils::drawShapeWithMaskFilter(
555 context, rtc, clip, shape, std::move(grPaint), ctm, mf);
556 }
557}
558
Robert Phillips16bf7d32020-07-07 10:20:27 -0400559void draw_tiled_bitmap(GrRecordingContext* context,
Brian Salomoneebe7352020-12-09 16:37:04 -0500560 GrSurfaceDrawContext* rtc,
Michael Ludwig7c12e282020-05-29 09:54:07 -0400561 const GrClip* clip,
Michael Ludwig47237242020-03-10 16:16:17 -0400562 const SkBitmap& bitmap,
563 int tileSize,
Brian Osman449b1152020-04-15 16:43:00 -0400564 const SkMatrixProvider& matrixProvider,
Michael Ludwig47237242020-03-10 16:16:17 -0400565 const SkMatrix& srcToDst,
566 const SkRect& srcRect,
567 const SkIRect& clippedSrcIRect,
568 const SkPaint& paint,
569 GrAA aa,
570 SkCanvas::SrcRectConstraint constraint,
Brian Salomone69b9ef2020-07-22 11:18:06 -0400571 GrSamplerState sampler,
Mike Reed52130b02020-12-28 15:33:13 -0500572 SkCubicResampler cubic) {
573 const bool doBicubic = GrValidCubicResampler(cubic);
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400574 SkRect clippedSrcRect = SkRect::Make(clippedSrcIRect);
575
576 int nx = bitmap.width() / tileSize;
577 int ny = bitmap.height() / tileSize;
Michael Ludwig47237242020-03-10 16:16:17 -0400578
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400579 for (int x = 0; x <= nx; x++) {
580 for (int y = 0; y <= ny; y++) {
581 SkRect tileR;
582 tileR.setLTRB(SkIntToScalar(x * tileSize), SkIntToScalar(y * tileSize),
583 SkIntToScalar((x + 1) * tileSize), SkIntToScalar((y + 1) * tileSize));
584
585 if (!SkRect::Intersects(tileR, clippedSrcRect)) {
586 continue;
587 }
588
589 if (!tileR.intersect(srcRect)) {
590 continue;
591 }
592
593 SkIRect iTileR;
594 tileR.roundOut(&iTileR);
595 SkVector offset = SkPoint::Make(SkIntToScalar(iTileR.fLeft),
596 SkIntToScalar(iTileR.fTop));
597 SkRect rectToDraw = tileR;
Michael Ludwig47237242020-03-10 16:16:17 -0400598 srcToDst.mapRect(&rectToDraw);
Brian Salomone69b9ef2020-07-22 11:18:06 -0400599 if (sampler.filter() != GrSamplerState::Filter::kNearest || doBicubic) {
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400600 SkIRect iClampRect;
601
602 if (SkCanvas::kFast_SrcRectConstraint == constraint) {
603 // In bleed mode we want to always expand the tile on all edges
604 // but stay within the bitmap bounds
605 iClampRect = SkIRect::MakeWH(bitmap.width(), bitmap.height());
606 } else {
607 // In texture-domain/clamp mode we only want to expand the
608 // tile on edges interior to "srcRect" (i.e., we want to
609 // not bleed across the original clamped edges)
610 srcRect.roundOut(&iClampRect);
611 }
Michael Ludwig47237242020-03-10 16:16:17 -0400612 int outset = doBicubic ? GrBicubicEffect::kFilterTexelPad : 1;
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400613 clamped_outset_with_offset(&iTileR, outset, &offset, iClampRect);
614 }
615
616 SkBitmap tmpB;
617 if (bitmap.extractSubset(&tmpB, iTileR)) {
Michael Ludwig47237242020-03-10 16:16:17 -0400618 // We should have already handled bitmaps larger than the max texture size.
619 SkASSERT(tmpB.width() <= context->priv().caps()->maxTextureSize() &&
620 tmpB.height() <= context->priv().caps()->maxTextureSize());
Michael Ludwig47237242020-03-10 16:16:17 -0400621
Brian Salomonbc074a62020-03-18 10:06:13 -0400622 GrBitmapTextureMaker tileProducer(context, tmpB, GrImageTexGenPolicy::kDraw);
Michael Ludwig47237242020-03-10 16:16:17 -0400623
624 GrQuadAAFlags aaFlags = GrQuadAAFlags::kNone;
625 if (aa == GrAA::kYes) {
626 // If the entire bitmap was anti-aliased, turn on AA for the outside tile edges.
627 if (tileR.fLeft <= srcRect.fLeft) {
628 aaFlags |= GrQuadAAFlags::kLeft;
629 }
630 if (tileR.fRight >= srcRect.fRight) {
631 aaFlags |= GrQuadAAFlags::kRight;
632 }
633 if (tileR.fTop <= srcRect.fTop) {
634 aaFlags |= GrQuadAAFlags::kTop;
635 }
636 if (tileR.fBottom >= srcRect.fBottom) {
637 aaFlags |= GrQuadAAFlags::kBottom;
638 }
639 }
640
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400641 // now offset it to make it "local" to our tmp bitmap
642 tileR.offset(-offset.fX, -offset.fY);
Michael Ludwig47237242020-03-10 16:16:17 -0400643 SkMatrix offsetSrcToDst = srcToDst;
644 offsetSrcToDst.preTranslate(offset.fX, offset.fY);
Brian Osman449b1152020-04-15 16:43:00 -0400645 draw_texture_producer(context, rtc, clip, matrixProvider, paint, &tileProducer,
646 tileR, rectToDraw, nullptr, offsetSrcToDst, aa, aaFlags,
Mike Reed52130b02020-12-28 15:33:13 -0500647 constraint, sampler, cubic);
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400648 }
649 }
650 }
651}
652
Michael Ludwig47237242020-03-10 16:16:17 -0400653} // anonymous namespace
Michael Ludwigdd86fb32020-03-10 09:55:35 -0400654
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500655//////////////////////////////////////////////////////////////////////////////
656
Mike Reedd8ff3132021-01-02 12:57:04 -0500657static SkFilterMode downgrade_to_filter(const SkSamplingOptions& sampling) {
658 SkFilterMode filter = sampling.filter;
659 if (sampling.useCubic || sampling.mipmap != SkMipmapMode::kNone) {
660 // if we were "fancier" than just bilerp, only do bilerp
661 filter = SkFilterMode::kLinear;
662 }
663 return filter;
664}
665
Michael Ludwig278263d2020-10-12 19:49:24 +0000666void SkGpuDevice::drawSpecial(SkSpecialImage* special, const SkMatrix& localToDevice,
Mike Reedd8ff3132021-01-02 12:57:04 -0500667 const SkSamplingOptions& sampling, const SkPaint& paint) {
Michael Ludwig16d5b0a2020-08-31 13:07:16 -0400668 SkASSERT(!paint.getMaskFilter() && !paint.getImageFilter());
669 SkASSERT(special->isTextureBacked());
670
671 SkRect src = SkRect::Make(special->subset());
Michael Ludwig278263d2020-10-12 19:49:24 +0000672 SkRect dst = SkRect::MakeWH(special->width(), special->height());
Mike Reed2ac6ce82021-01-15 12:26:22 -0500673 SkMatrix srcToDst = SkMatrix::RectToRect(src, dst);
Michael Ludwig16d5b0a2020-08-31 13:07:16 -0400674
Mike Reedd8ff3132021-01-02 12:57:04 -0500675 GrSamplerState sampler(GrSamplerState::WrapMode::kClamp, downgrade_to_filter(sampling));
Michael Ludwig278263d2020-10-12 19:49:24 +0000676 GrAA aa = paint.isAntiAlias() ? GrAA::kYes : GrAA::kNo;
677 GrQuadAAFlags aaFlags = paint.isAntiAlias() ? GrQuadAAFlags::kAll : GrQuadAAFlags::kNone;
Michael Ludwig16d5b0a2020-08-31 13:07:16 -0400678
679 GrColorInfo colorInfo(SkColorTypeToGrColorType(special->colorType()),
680 special->alphaType(), sk_ref_sp(special->getColorSpace()));
681
682 GrSurfaceProxyView view = special->view(this->recordingContext());
683 GrTextureAdjuster texture(fContext.get(), std::move(view), colorInfo, special->uniqueID());
684 // In most cases this ought to hit draw_texture since there won't be a color filter,
685 // alpha-only texture+shader, or a high filter quality.
Michael Ludwig278263d2020-10-12 19:49:24 +0000686 SkOverrideDeviceMatrixProvider matrixProvider(this->asMatrixProvider(), localToDevice);
Brian Salomon8f7d9532020-12-23 09:16:59 -0500687 draw_texture_producer(fContext.get(), fSurfaceDrawContext.get(), this->clip(), matrixProvider,
688 paint, &texture, src, dst, nullptr, srcToDst, aa, aaFlags,
Mike Reed52130b02020-12-28 15:33:13 -0500689 SkCanvas::kStrict_SrcRectConstraint, sampler, kInvalidCubicResampler);
Michael Ludwig16d5b0a2020-08-31 13:07:16 -0400690}
691
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500692void SkGpuDevice::drawImageQuad(const SkImage* image, const SkRect* srcRect, const SkRect* dstRect,
693 const SkPoint dstClip[4], GrAA aa, GrQuadAAFlags aaFlags,
Mike Reed4f23dec2020-12-30 14:22:42 +0000694 const SkMatrix* preViewMatrix, const SkSamplingOptions& sampling,
695 const SkPaint& paint, SkCanvas::SrcRectConstraint constraint) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500696 SkRect src;
697 SkRect dst;
698 SkMatrix srcToDst;
699 ImageDrawMode mode = optimize_sample_area(SkISize::Make(image->width(), image->height()),
700 srcRect, dstRect, dstClip, &src, &dst, &srcToDst);
701 if (mode == ImageDrawMode::kSkip) {
bsalomonc55271f2015-11-09 11:55:57 -0800702 return;
703 }
704
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500705 if (src.contains(image->bounds())) {
706 constraint = SkCanvas::kFast_SrcRectConstraint;
707 }
708 // Depending on the nature of image, it can flow through more or less optimal pipelines
Brian Salomon777e1462020-02-28 21:10:31 -0500709 GrSamplerState::WrapMode wrapMode = mode == ImageDrawMode::kDecal
710 ? GrSamplerState::WrapMode::kClampToBorder
711 : GrSamplerState::WrapMode::kClamp;
Robert Phillips27927a52018-08-20 13:18:12 -0400712
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500713 // Get final CTM matrix
Brian Osman449b1152020-04-15 16:43:00 -0400714 SkPreConcatMatrixProvider matrixProvider(this->asMatrixProvider(),
715 preViewMatrix ? *preViewMatrix : SkMatrix::I());
716 const SkMatrix& ctm(matrixProvider.localToDevice());
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500717
Brian Salomone69b9ef2020-07-22 11:18:06 -0400718 bool sharpenMM = fContext->priv().options().fSharpenMipmappedTextures;
Mike Reed52130b02020-12-28 15:33:13 -0500719 auto [fm, mm, cubic] = GrInterpretSamplingOptions(image->dimensions(), sampling,
John Stilesf2f9b0d2020-08-25 13:33:45 -0400720 ctm, srcToDst, sharpenMM,
721 /*allowFilterQualityReduction=*/true);
Michael Ludwig47237242020-03-10 16:16:17 -0400722
723 auto clip = this->clip();
724
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500725 // YUVA images can be stored in multiple images with different plane resolutions, so this
726 // uses an effect to combine them dynamically on the GPU. This is done before requesting a
727 // pinned texture proxy because YUV images force-flatten to RGBA in that scenario.
728 if (as_IB(image)->isYUVA()) {
Brian Salomon777e1462020-02-28 21:10:31 -0500729 GrYUVAImageTextureMaker maker(fContext.get(), image);
Brian Salomon8f7d9532020-12-23 09:16:59 -0500730 draw_texture_producer(fContext.get(), fSurfaceDrawContext.get(), clip, matrixProvider,
Brian Osman449b1152020-04-15 16:43:00 -0400731 paint, &maker, src, dst, dstClip, srcToDst, aa, aaFlags, constraint,
Mike Reed52130b02020-12-28 15:33:13 -0500732 {wrapMode, fm, mm}, cubic);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500733 return;
734 }
735
Brian Salomon490f1922021-02-01 13:14:37 -0500736 if (!image->isTextureBacked() && !as_IB(image)->isPinnedOnContext(fContext.get())) {
Brian Salomon86446572021-01-29 03:14:12 +0000737 int tileFilterPad;
738 if (GrValidCubicResampler(cubic)) {
739 tileFilterPad = GrBicubicEffect::kFilterTexelPad;
Brian Salomon490f1922021-02-01 13:14:37 -0500740 } else if (fm == GrSamplerState::Filter::kNearest) {
Brian Salomon86446572021-01-29 03:14:12 +0000741 tileFilterPad = 0;
742 } else {
743 tileFilterPad = 1;
744 }
Brian Salomon010bb462021-01-29 14:42:38 -0500745 int maxTileSize = fContext->priv().caps()->maxTextureSize() - 2*tileFilterPad;
Brian Salomon86446572021-01-29 03:14:12 +0000746 int tileSize;
747 SkIRect clippedSubset;
748 if (should_tile_image_id(fContext.get(), fSurfaceDrawContext->dimensions(), clip,
749 image->unique(), image->dimensions(), ctm, srcToDst, &src,
750 maxTileSize, &tileSize, &clippedSubset)) {
751 // Extract pixels on the CPU, since we have to split into separate textures before
Brian Salomon490f1922021-02-01 13:14:37 -0500752 // sending to the GPU if tiling.
753 if (SkBitmap bm; as_IB(image)->getROPixels(nullptr, &bm)) {
Brian Salomon86446572021-01-29 03:14:12 +0000754 // This is the funnel for all paths that draw tiled bitmaps/images.
755 draw_tiled_bitmap(fContext.get(), fSurfaceDrawContext.get(), clip, bm, tileSize,
756 matrixProvider, srcToDst, src, clippedSubset, paint, aa,
757 constraint, {wrapMode, fm, mm}, cubic);
758 return;
759 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500760 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500761 }
Brian Salomon86446572021-01-29 03:14:12 +0000762
Brian Salomond0924f32021-02-03 10:15:31 -0500763 if (auto [view, ct] = as_IB(image)->asView(this->recordingContext(),
764 GrMipmapped(mm != SkMipmapMode::kNone)); view) {
Brian Salomon490f1922021-02-01 13:14:37 -0500765 // This adjuster shouldn't do anything since we already asked for mip maps if necessary.
766 // TODO: Pull YUVA out of draw_texture_producer and make it work directly from a view.
Brian Salomond0924f32021-02-03 10:15:31 -0500767 GrColorInfo colorInfo(ct, image->alphaType(), image->refColorSpace());
Brian Salomon490f1922021-02-01 13:14:37 -0500768 GrTextureAdjuster adjuster(fContext.get(),
769 std::move(view),
Brian Salomond0924f32021-02-03 10:15:31 -0500770 std::move(colorInfo),
Brian Salomon490f1922021-02-01 13:14:37 -0500771 image->uniqueID());
Brian Salomon86446572021-01-29 03:14:12 +0000772 draw_texture_producer(fContext.get(), fSurfaceDrawContext.get(), clip, matrixProvider,
Brian Salomon490f1922021-02-01 13:14:37 -0500773 paint, &adjuster, src, dst, dstClip, srcToDst, aa, aaFlags,
774 constraint, {wrapMode, fm, mm}, cubic);
Brian Salomon86446572021-01-29 03:14:12 +0000775 return;
776 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500777}
778
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400779void SkGpuDevice::drawEdgeAAImageSet(const SkCanvas::ImageSetEntry set[], int count,
780 const SkPoint dstClips[], const SkMatrix preViewMatrices[],
Mike Reed4f509342021-01-05 12:03:47 -0500781 const SkSamplingOptions& sampling, const SkPaint& paint,
782 SkCanvas::SrcRectConstraint constraint) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500783 SkASSERT(count > 0);
Mike Reed4f23dec2020-12-30 14:22:42 +0000784 if (!can_use_draw_texture(paint, sampling.useCubic, sampling.mipmap)) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500785 // Send every entry through drawImageQuad() to handle the more complicated paint
786 int dstClipIndex = 0;
787 for (int i = 0; i < count; ++i) {
788 // Only no clip or quad clip are supported
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400789 SkASSERT(!set[i].fHasClip || dstClips);
790 SkASSERT(set[i].fMatrixIndex < 0 || preViewMatrices);
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500791
Brian Salomondb151e02019-09-17 12:11:16 -0400792 SkTCopyOnFirstWrite<SkPaint> entryPaint(paint);
793 if (set[i].fAlpha != 1.f) {
794 auto paintAlpha = paint.getAlphaf();
795 entryPaint.writable()->setAlphaf(paintAlpha * set[i].fAlpha);
796 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500797 // Always send GrAA::kYes to preserve seaming across tiling in MSAA
Brian Salomondb151e02019-09-17 12:11:16 -0400798 this->drawImageQuad(
799 set[i].fImage.get(), &set[i].fSrcRect, &set[i].fDstRect,
800 set[i].fHasClip ? dstClips + dstClipIndex : nullptr, GrAA::kYes,
801 SkToGrQuadAAFlags(set[i].fAAFlags),
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400802 set[i].fMatrixIndex < 0 ? nullptr : preViewMatrices + set[i].fMatrixIndex,
Mike Reed4f23dec2020-12-30 14:22:42 +0000803 sampling, *entryPaint, constraint);
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400804 dstClipIndex += 4 * set[i].fHasClip;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500805 }
806 return;
807 }
808
Mike Reed4f23dec2020-12-30 14:22:42 +0000809 GrSamplerState::Filter filter = sampling.filter == SkFilterMode::kNearest
Brian Salomona3b02f52020-07-15 16:02:01 -0400810 ? GrSamplerState::Filter::kNearest
811 : GrSamplerState::Filter::kLinear;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500812 SkBlendMode mode = paint.getBlendMode();
813
Brian Salomoneebe7352020-12-09 16:37:04 -0500814 SkAutoTArray<GrSurfaceDrawContext::TextureSetEntry> textures(count);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500815 // We accumulate compatible proxies until we find an an incompatible one or reach the end and
Michael Ludwig379e4962019-12-06 13:21:26 -0500816 // issue the accumulated 'n' draws starting at 'base'. 'p' represents the number of proxy
817 // switches that occur within the 'n' entries.
818 int base = 0, n = 0, p = 0;
819 auto draw = [&](int nextBase) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500820 if (n > 0) {
Brian Salomonbacbb922021-01-21 19:48:00 -0500821 auto textureXform = GrColorSpaceXform::Make(set[base].fImage->imageInfo().colorInfo(),
822 fSurfaceDrawContext->colorInfo());
Brian Salomon8f7d9532020-12-23 09:16:59 -0500823 fSurfaceDrawContext->drawTextureSet(this->clip(),
824 textures.get() + base,
825 n,
826 p,
827 filter,
828 GrSamplerState::MipmapMode::kNone,
829 mode,
830 GrAA::kYes,
831 constraint,
832 this->localToDevice(),
833 std::move(textureXform));
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500834 }
Michael Ludwig379e4962019-12-06 13:21:26 -0500835 base = nextBase;
836 n = 0;
837 p = 0;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500838 };
839 int dstClipIndex = 0;
840 for (int i = 0; i < count; ++i) {
Michael Ludwigd9958f82019-03-21 13:08:36 -0400841 SkASSERT(!set[i].fHasClip || dstClips);
842 SkASSERT(set[i].fMatrixIndex < 0 || preViewMatrices);
843
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500844 // Manage the dst clip pointer tracking before any continues are used so we don't lose
845 // our place in the dstClips array.
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400846 const SkPoint* clip = set[i].fHasClip ? dstClips + dstClipIndex : nullptr;
847 dstClipIndex += 4 * set[i].fHasClip;
848
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500849 // The default SkBaseDevice implementation is based on drawImageRect which does not allow
850 // non-sorted src rects. TODO: Decide this is OK or make sure we handle it.
851 if (!set[i].fSrcRect.isSorted()) {
Michael Ludwig379e4962019-12-06 13:21:26 -0500852 draw(i + 1);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500853 continue;
854 }
855
Greg Danielcc21d0c2020-02-05 16:58:40 -0500856 GrSurfaceProxyView view;
Michael Ludwigd9958f82019-03-21 13:08:36 -0400857 const SkImage_Base* image = as_IB(set[i].fImage.get());
Greg Danielcc21d0c2020-02-05 16:58:40 -0500858 // Extract view from image, but skip YUV images so they get processed through
Michael Ludwigd9958f82019-03-21 13:08:36 -0400859 // drawImageQuad and the proper effect to dynamically sample their planes.
860 if (!image->isYUVA()) {
Brian Salomond0924f32021-02-03 10:15:31 -0500861 std::tie(view, std::ignore) = image->asView(this->recordingContext(), GrMipmapped::kNo);
Brian Salomonb43d6992021-01-05 14:37:40 -0500862 if (image->isAlphaOnly()) {
863 GrSwizzle swizzle = GrSwizzle::Concat(view.swizzle(), GrSwizzle("aaaa"));
864 view = {view.detachProxy(), view.origin(), swizzle};
865 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500866 }
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500867
Greg Danielcc21d0c2020-02-05 16:58:40 -0500868 if (!view) {
Michael Ludwigd9958f82019-03-21 13:08:36 -0400869 // This image can't go through the texture op, send through general image pipeline
870 // after flushing current batch.
Michael Ludwig379e4962019-12-06 13:21:26 -0500871 draw(i + 1);
Brian Salomondb151e02019-09-17 12:11:16 -0400872 SkTCopyOnFirstWrite<SkPaint> entryPaint(paint);
873 if (set[i].fAlpha != 1.f) {
874 auto paintAlpha = paint.getAlphaf();
875 entryPaint.writable()->setAlphaf(paintAlpha * set[i].fAlpha);
876 }
877 this->drawImageQuad(
878 image, &set[i].fSrcRect, &set[i].fDstRect, clip, GrAA::kYes,
Michael Ludwigd9958f82019-03-21 13:08:36 -0400879 SkToGrQuadAAFlags(set[i].fAAFlags),
880 set[i].fMatrixIndex < 0 ? nullptr : preViewMatrices + set[i].fMatrixIndex,
Mike Reed4f23dec2020-12-30 14:22:42 +0000881 sampling, *entryPaint, constraint);
Michael Ludwigd9958f82019-03-21 13:08:36 -0400882 continue;
883 }
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500884
Greg Danielcc21d0c2020-02-05 16:58:40 -0500885 textures[i].fProxyView = std::move(view);
Brian Salomonfc118442019-11-22 19:09:27 -0500886 textures[i].fSrcAlphaType = image->alphaType();
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500887 textures[i].fSrcRect = set[i].fSrcRect;
888 textures[i].fDstRect = set[i].fDstRect;
889 textures[i].fDstClipQuad = clip;
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400890 textures[i].fPreViewMatrix =
891 set[i].fMatrixIndex < 0 ? nullptr : preViewMatrices + set[i].fMatrixIndex;
Michael Ludwig1c66ad92020-07-10 08:59:44 -0400892 textures[i].fColor = texture_color(paint.getColor4f(), set[i].fAlpha,
893 SkColorTypeToGrColorType(image->colorType()),
Brian Salomon8f7d9532020-12-23 09:16:59 -0500894 fSurfaceDrawContext->colorInfo());
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500895 textures[i].fAAFlags = SkToGrQuadAAFlags(set[i].fAAFlags);
896
897 if (n > 0 &&
Greg Daniel549325c2019-10-30 16:19:20 -0400898 (!GrTextureProxy::ProxiesAreCompatibleAsDynamicState(
Michael Ludwigfcdd0612019-11-25 08:34:31 -0500899 textures[i].fProxyView.proxy(),
900 textures[base].fProxyView.proxy()) ||
Greg Daniel507736f2020-01-17 15:36:10 -0500901 textures[i].fProxyView.swizzle() != textures[base].fProxyView.swizzle() ||
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500902 set[i].fImage->alphaType() != set[base].fImage->alphaType() ||
903 !SkColorSpace::Equals(set[i].fImage->colorSpace(), set[base].fImage->colorSpace()))) {
Michael Ludwig379e4962019-12-06 13:21:26 -0500904 draw(i);
905 }
906 // Whether or not we submitted a draw in the above if(), this ith entry is in the current
907 // set being accumulated so increment n, and increment p if proxies are different.
908 ++n;
909 if (n == 1 || textures[i - 1].fProxyView.proxy() != textures[i].fProxyView.proxy()) {
910 // First proxy or a different proxy (that is compatible, otherwise we'd have drawn up
911 // to i - 1).
912 ++p;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500913 }
914 }
Michael Ludwig379e4962019-12-06 13:21:26 -0500915 draw(count);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500916}