blob: e4d31a61bb7423dfeb28710cfc7613a4f7e9a6c9 [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"
11#include "src/core/SkDraw.h"
12#include "src/core/SkMaskFilterBase.h"
13#include "src/gpu/GrBitmapTextureMaker.h"
14#include "src/gpu/GrBlurUtils.h"
15#include "src/gpu/GrCaps.h"
16#include "src/gpu/GrColorSpaceXform.h"
17#include "src/gpu/GrImageTextureMaker.h"
18#include "src/gpu/GrRenderTargetContext.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050019#include "src/gpu/GrStyle.h"
20#include "src/gpu/GrTextureAdjuster.h"
21#include "src/gpu/GrTextureMaker.h"
22#include "src/gpu/SkGr.h"
23#include "src/gpu/effects/GrBicubicEffect.h"
24#include "src/gpu/effects/GrTextureDomain.h"
Brian Salomonb8f098d2020-01-07 11:15:44 -050025#include "src/gpu/effects/GrTextureEffect.h"
Michael Ludwig663afe52019-06-03 16:46:19 -040026#include "src/gpu/geometry/GrShape.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050027#include "src/image/SkImage_Base.h"
bsalomonc55271f2015-11-09 11:55:57 -080028
Michael Ludwig1433cfd2019-02-27 17:12:30 -050029namespace {
30
bsalomonc55271f2015-11-09 11:55:57 -080031static inline bool use_shader(bool textureIsAlphaOnly, const SkPaint& paint) {
32 return textureIsAlphaOnly && paint.getShader();
33}
34
bsalomonb1b01992015-11-18 10:56:08 -080035//////////////////////////////////////////////////////////////////////////////
36// Helper functions for dropping src rect constraint in bilerp mode.
37
38static const SkScalar kColorBleedTolerance = 0.001f;
39
40static bool has_aligned_samples(const SkRect& srcRect, const SkRect& transformedRect) {
41 // detect pixel disalignment
Brian Salomona911f8f2015-11-18 15:19:57 -050042 if (SkScalarAbs(SkScalarRoundToScalar(transformedRect.left()) - transformedRect.left()) < kColorBleedTolerance &&
43 SkScalarAbs(SkScalarRoundToScalar(transformedRect.top()) - transformedRect.top()) < kColorBleedTolerance &&
44 SkScalarAbs(transformedRect.width() - srcRect.width()) < kColorBleedTolerance &&
bsalomonb1b01992015-11-18 10:56:08 -080045 SkScalarAbs(transformedRect.height() - srcRect.height()) < kColorBleedTolerance) {
46 return true;
47 }
48 return false;
49}
50
51static bool may_color_bleed(const SkRect& srcRect,
52 const SkRect& transformedRect,
53 const SkMatrix& m,
Chris Dalton6ce447a2019-06-23 18:07:38 -060054 int numSamples) {
bsalomonb1b01992015-11-18 10:56:08 -080055 // Only gets called if has_aligned_samples returned false.
56 // So we can assume that sampling is axis aligned but not texel aligned.
57 SkASSERT(!has_aligned_samples(srcRect, transformedRect));
58 SkRect innerSrcRect(srcRect), innerTransformedRect, outerTransformedRect(transformedRect);
Chris Dalton6ce447a2019-06-23 18:07:38 -060059 if (numSamples > 1) {
bsalomonb1b01992015-11-18 10:56:08 -080060 innerSrcRect.inset(SK_Scalar1, SK_Scalar1);
61 } else {
62 innerSrcRect.inset(SK_ScalarHalf, SK_ScalarHalf);
63 }
64 m.mapRect(&innerTransformedRect, innerSrcRect);
65
66 // The gap between outerTransformedRect and innerTransformedRect
67 // represents the projection of the source border area, which is
68 // problematic for color bleeding. We must check whether any
69 // destination pixels sample the border area.
70 outerTransformedRect.inset(kColorBleedTolerance, kColorBleedTolerance);
71 innerTransformedRect.outset(kColorBleedTolerance, kColorBleedTolerance);
72 SkIRect outer, inner;
73 outerTransformedRect.round(&outer);
74 innerTransformedRect.round(&inner);
75 // If the inner and outer rects round to the same result, it means the
76 // border does not overlap any pixel centers. Yay!
77 return inner != outer;
78}
79
80static bool can_ignore_bilerp_constraint(const GrTextureProducer& producer,
81 const SkRect& srcRect,
82 const SkMatrix& srcRectToDeviceSpace,
Chris Dalton6ce447a2019-06-23 18:07:38 -060083 int numSamples) {
bsalomonb1b01992015-11-18 10:56:08 -080084 if (srcRectToDeviceSpace.rectStaysRect()) {
85 // sampling is axis-aligned
86 SkRect transformedRect;
87 srcRectToDeviceSpace.mapRect(&transformedRect, srcRect);
88
89 if (has_aligned_samples(srcRect, transformedRect) ||
Chris Dalton6ce447a2019-06-23 18:07:38 -060090 !may_color_bleed(srcRect, transformedRect, srcRectToDeviceSpace, numSamples)) {
bsalomonb1b01992015-11-18 10:56:08 -080091 return true;
92 }
93 }
94 return false;
95}
96
Michael Ludwig1433cfd2019-02-27 17:12:30 -050097enum class ImageDrawMode {
98 // Src and dst have been restricted to the image content. May need to clamp, no need to decal.
99 kOptimized,
100 // Src and dst are their original sizes, requires use of a decal instead of plain clamping.
101 // This is used when a dst clip is provided and extends outside of the optimized dst rect.
102 kDecal,
103 // Src or dst are empty, or do not intersect the image content so don't draw anything.
104 kSkip
105};
106
107/**
108 * Optimize the src rect sampling area within an image (sized 'width' x 'height') such that
109 * 'outSrcRect' will be completely contained in the image's bounds. The corresponding rect
110 * to draw will be output to 'outDstRect'. The mapping between src and dst will be cached in
111 * 'srcToDst'. Outputs are not always updated when kSkip is returned.
112 *
113 * If 'origSrcRect' is null, implicitly use the image bounds. If 'origDstRect' is null, use the
114 * original src rect. 'dstClip' should be null when there is no additional clipping.
115 */
116static ImageDrawMode optimize_sample_area(const SkISize& image, const SkRect* origSrcRect,
117 const SkRect* origDstRect, const SkPoint dstClip[4],
118 SkRect* outSrcRect, SkRect* outDstRect,
119 SkMatrix* srcToDst) {
120 SkRect srcBounds = SkRect::MakeIWH(image.fWidth, image.fHeight);
121
122 SkRect src = origSrcRect ? *origSrcRect : srcBounds;
123 SkRect dst = origDstRect ? *origDstRect : src;
124
125 if (src.isEmpty() || dst.isEmpty()) {
126 return ImageDrawMode::kSkip;
127 }
128
129 if (outDstRect) {
130 srcToDst->setRectToRect(src, dst, SkMatrix::kFill_ScaleToFit);
131 } else {
132 srcToDst->setIdentity();
133 }
134
135 if (origSrcRect && !srcBounds.contains(src)) {
136 if (!src.intersect(srcBounds)) {
137 return ImageDrawMode::kSkip;
138 }
139 srcToDst->mapRect(&dst, src);
140
141 // Both src and dst have gotten smaller. If dstClip is provided, confirm it is still
142 // contained in dst, otherwise cannot optimize the sample area and must use a decal instead
143 if (dstClip) {
144 for (int i = 0; i < 4; ++i) {
145 if (!dst.contains(dstClip[i].fX, dstClip[i].fY)) {
146 // Must resort to using a decal mode restricted to the clipped 'src', and
147 // use the original dst rect (filling in src bounds as needed)
148 *outSrcRect = src;
149 *outDstRect = (origDstRect ? *origDstRect
150 : (origSrcRect ? *origSrcRect : srcBounds));
151 return ImageDrawMode::kDecal;
152 }
153 }
154 }
155 }
156
157 // The original src and dst were fully contained in the image, or there was no dst clip to
158 // worry about, or the clip was still contained in the restricted dst rect.
159 *outSrcRect = src;
160 *outDstRect = dst;
161 return ImageDrawMode::kOptimized;
162}
163
Brian Salomon34169692017-08-28 15:32:01 -0400164/**
Brian Salomonb80ffee2018-05-23 16:39:39 -0400165 * Checks whether the paint is compatible with using GrRenderTargetContext::drawTexture. It is more
166 * efficient than the GrTextureProducer general case.
Brian Salomon34169692017-08-28 15:32:01 -0400167 */
Brian Salomonb80ffee2018-05-23 16:39:39 -0400168static bool can_use_draw_texture(const SkPaint& paint) {
Brian Salomon34169692017-08-28 15:32:01 -0400169 return (!paint.getColorFilter() && !paint.getShader() && !paint.getMaskFilter() &&
Michael Ludwigd54ca8f2019-02-13 13:25:21 -0500170 !paint.getImageFilter() && paint.getFilterQuality() < kMedium_SkFilterQuality);
Brian Salomon34169692017-08-28 15:32:01 -0400171}
172
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500173// Assumes srcRect and dstRect have already been optimized to fit the proxy
174static void draw_texture(GrRenderTargetContext* rtc, const GrClip& clip, const SkMatrix& ctm,
175 const SkPaint& paint, const SkRect& srcRect, const SkRect& dstRect,
176 const SkPoint dstClip[4], GrAA aa, GrQuadAAFlags aaFlags,
Greg Daniel2f3cd4f2020-02-07 11:07:25 -0500177 SkCanvas::SrcRectConstraint constraint, GrSurfaceProxyView view,
Greg Daniela4828a12019-10-11 13:51:02 -0400178 const GrColorInfo& srcColorInfo) {
Brian Salomon4bc0c1f2019-09-30 15:12:27 -0400179 const GrColorInfo& dstInfo(rtc->colorInfo());
Mike Kleine03a1762018-08-22 11:52:16 -0400180 auto textureXform =
Greg Daniela4828a12019-10-11 13:51:02 -0400181 GrColorSpaceXform::Make(srcColorInfo.colorSpace(), srcColorInfo.alphaType(),
Brian Osman3d139a42018-11-19 10:42:10 -0500182 dstInfo.colorSpace(), kPremul_SkAlphaType);
Brian Salomon2bbdcc42017-09-07 12:36:34 -0400183 GrSamplerState::Filter filter;
Brian Salomon34169692017-08-28 15:32:01 -0400184 switch (paint.getFilterQuality()) {
185 case kNone_SkFilterQuality:
Brian Salomon2bbdcc42017-09-07 12:36:34 -0400186 filter = GrSamplerState::Filter::kNearest;
Brian Salomon34169692017-08-28 15:32:01 -0400187 break;
188 case kLow_SkFilterQuality:
Brian Salomon2bbdcc42017-09-07 12:36:34 -0400189 filter = GrSamplerState::Filter::kBilerp;
Brian Salomon34169692017-08-28 15:32:01 -0400190 break;
191 case kMedium_SkFilterQuality:
192 case kHigh_SkFilterQuality:
193 SK_ABORT("Quality level not allowed.");
194 }
Greg Daniel2f3cd4f2020-02-07 11:07:25 -0500195 GrSurfaceProxy* proxy = view.proxy();
Michael Ludwigd54ca8f2019-02-13 13:25:21 -0500196 // Must specify the strict constraint when the proxy is not functionally exact and the src
197 // rect would access pixels outside the proxy's content area without the constraint.
Brian Salomon5c60b752019-12-13 15:03:43 -0500198 if (constraint != SkCanvas::kStrict_SrcRectConstraint && !proxy->isFunctionallyExact()) {
Michael Ludwigd54ca8f2019-02-13 13:25:21 -0500199 // Conservative estimate of how much a coord could be outset from src rect:
200 // 1/2 pixel for AA and 1/2 pixel for bilerp
201 float buffer = 0.5f * (aa == GrAA::kYes) +
202 0.5f * (filter == GrSamplerState::Filter::kBilerp);
Brian Salomon9f2b86c2019-10-22 10:37:46 -0400203 SkRect safeBounds = proxy->getBoundsRect();
Michael Ludwigd54ca8f2019-02-13 13:25:21 -0500204 safeBounds.inset(buffer, buffer);
205 if (!safeBounds.contains(srcRect)) {
206 constraint = SkCanvas::kStrict_SrcRectConstraint;
207 }
208 }
Brian Osman3d139a42018-11-19 10:42:10 -0500209 SkPMColor4f color;
Greg Daniela4828a12019-10-11 13:51:02 -0400210 if (GrColorTypeIsAlphaOnly(srcColorInfo.colorType())) {
Brian Osman8fa7ab42019-03-18 10:22:42 -0400211 color = SkColor4fPrepForDst(paint.getColor4f(), dstInfo).premul();
Brian Osman3ebd3542018-07-30 14:36:53 -0400212 } else {
Brian Osman3d139a42018-11-19 10:42:10 -0500213 float paintAlpha = paint.getColor4f().fA;
214 color = { paintAlpha, paintAlpha, paintAlpha, paintAlpha };
Brian Osman3ebd3542018-07-30 14:36:53 -0400215 }
Brian Salomon34169692017-08-28 15:32:01 -0400216
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500217 if (dstClip) {
218 // Get source coords corresponding to dstClip
219 SkPoint srcQuad[4];
220 GrMapRectPoints(dstRect, srcRect, dstClip, srcQuad, 4);
Michael Ludwig24adb3a2019-02-27 19:42:20 +0000221
Greg Daniel2f3cd4f2020-02-07 11:07:25 -0500222 rtc->drawTextureQuad(clip, std::move(view), srcColorInfo.colorType(),
Brian Salomonfc118442019-11-22 19:09:27 -0500223 srcColorInfo.alphaType(), filter, paint.getBlendMode(), color, srcQuad,
224 dstClip, aa, aaFlags,
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500225 constraint == SkCanvas::kStrict_SrcRectConstraint ? &srcRect : nullptr,
226 ctm, std::move(textureXform));
227 } else {
Greg Daniel40903af2020-01-30 14:55:05 -0500228 rtc->drawTexture(clip, std::move(view), srcColorInfo.alphaType(), filter,
229 paint.getBlendMode(), color, srcRect, dstRect, aa, aaFlags, constraint,
230 ctm, std::move(textureXform));
Michael Ludwig24adb3a2019-02-27 19:42:20 +0000231 }
Michael Ludwig24adb3a2019-02-27 19:42:20 +0000232}
233
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500234// Assumes srcRect and dstRect have already been optimized to fit the proxy.
Brian Salomon777e1462020-02-28 21:10:31 -0500235static void draw_texture_producer(GrContext* context,
236 GrRenderTargetContext* rtc,
237 const GrClip& clip,
238 const SkMatrix& ctm,
239 const SkPaint& paint,
240 GrTextureProducer* producer,
241 const SkRect& src,
242 const SkRect& dst,
243 const SkPoint dstClip[4],
244 const SkMatrix& srcToDst,
245 GrAA aa,
246 GrQuadAAFlags aaFlags,
247 SkCanvas::SrcRectConstraint constraint,
248 GrSamplerState::WrapMode wm) {
249 if (wm == GrSamplerState::WrapMode::kClamp && !producer->isPlanar() &&
250 can_use_draw_texture(paint)) {
Jim Van Verth30e0d7f2018-11-02 13:36:42 -0400251 // We've done enough checks above to allow us to pass ClampNearest() and not check for
252 // scaling adjustments.
Brian Salomonecbb0fb2020-02-28 18:07:32 -0500253 auto view = producer->view(GrMipMapped::kNo);
Greg Danielcc21d0c2020-02-05 16:58:40 -0500254 if (!view) {
Jim Van Verth30e0d7f2018-11-02 13:36:42 -0400255 return;
256 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500257
Brian Salomonecbb0fb2020-02-28 18:07:32 -0500258 draw_texture(
259 rtc, clip, ctm, paint, src, dst, dstClip, aa, aaFlags, constraint, std::move(view),
260 {producer->colorType(), producer->alphaType(), sk_ref_sp(producer->colorSpace())});
Jim Van Verth30e0d7f2018-11-02 13:36:42 -0400261 return;
262 }
263
bsalomonc55271f2015-11-09 11:55:57 -0800264 const SkMaskFilter* mf = paint.getMaskFilter();
Michael Ludwigb7d64b92019-02-11 11:09:15 -0500265
bsalomonc55271f2015-11-09 11:55:57 -0800266 // The shader expects proper local coords, so we can't replace local coords with texture coords
267 // if the shader will be used. If we have a mask filter we will change the underlying geometry
268 // that is rendered.
bsalomonf1ecd212015-12-09 17:06:02 -0800269 bool canUseTextureCoordsAsLocalCoords = !use_shader(producer->isAlphaOnly(), paint) && !mf;
bsalomonc55271f2015-11-09 11:55:57 -0800270
Michael Ludwigb7d64b92019-02-11 11:09:15 -0500271 // Specifying the texture coords as local coordinates is an attempt to enable more GrDrawOp
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500272 // combining by not baking anything about the srcRect, dstRect, or ctm, into the texture
Michael Ludwigb7d64b92019-02-11 11:09:15 -0500273 // FP. In the future this should be an opaque optimization enabled by the combination of
274 // GrDrawOp/GP and FP.
275 if (mf && as_MFB(mf)->hasFragmentProcessor()) {
276 mf = nullptr;
277 }
bsalomonc55271f2015-11-09 11:55:57 -0800278 bool doBicubic;
Brian Salomon2bbdcc42017-09-07 12:36:34 -0400279 GrSamplerState::Filter fm = GrSkFilterQualityToGrFilterMode(
Chris Dalton309c6c02019-08-13 10:32:47 -0600280 producer->width(), producer->height(), paint.getFilterQuality(), ctm, srcToDst,
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500281 context->priv().options().fSharpenMipmappedTextures, &doBicubic);
Brian Salomon2bbdcc42017-09-07 12:36:34 -0400282 const GrSamplerState::Filter* filterMode = doBicubic ? nullptr : &fm;
bsalomonc55271f2015-11-09 11:55:57 -0800283
Brian Osmane8e54582016-11-28 10:06:27 -0500284 GrTextureProducer::FilterConstraint constraintMode;
bsalomonc55271f2015-11-09 11:55:57 -0800285 if (SkCanvas::kFast_SrcRectConstraint == constraint) {
286 constraintMode = GrTextureAdjuster::kNo_FilterConstraint;
287 } else {
288 constraintMode = GrTextureAdjuster::kYes_FilterConstraint;
289 }
halcanary9d524f22016-03-29 09:03:52 -0700290
bsalomonc55271f2015-11-09 11:55:57 -0800291 // If we have to outset for AA then we will generate texture coords outside the src rect. The
292 // same happens for any mask filter that extends the bounds rendered in the dst.
293 // This is conservative as a mask filter does not have to expand the bounds rendered.
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500294 bool coordsAllInsideSrcRect = aaFlags == GrQuadAAFlags::kNone && !mf;
bsalomonc55271f2015-11-09 11:55:57 -0800295
bsalomonb1b01992015-11-18 10:56:08 -0800296 // Check for optimization to drop the src rect constraint when on bilerp.
Brian Salomon2bbdcc42017-09-07 12:36:34 -0400297 if (filterMode && GrSamplerState::Filter::kBilerp == *filterMode &&
Michael Ludwiga6a84002019-04-12 15:03:02 -0400298 GrTextureAdjuster::kYes_FilterConstraint == constraintMode && coordsAllInsideSrcRect &&
Brian Salomon777e1462020-02-28 21:10:31 -0500299 !producer->isPlanar()) {
bsalomonb1b01992015-11-18 10:56:08 -0800300 SkMatrix combinedMatrix;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500301 combinedMatrix.setConcat(ctm, srcToDst);
Chris Dalton6ce447a2019-06-23 18:07:38 -0600302 if (can_ignore_bilerp_constraint(*producer, src, combinedMatrix, rtc->numSamples())) {
bsalomonb1b01992015-11-18 10:56:08 -0800303 constraintMode = GrTextureAdjuster::kNo_FilterConstraint;
304 }
305 }
306
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500307 SkMatrix textureMatrix;
bsalomon3aa5fce2015-11-12 09:59:44 -0800308 if (canUseTextureCoordsAsLocalCoords) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500309 textureMatrix = SkMatrix::I();
bsalomon3aa5fce2015-11-12 09:59:44 -0800310 } else {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500311 if (!srcToDst.invert(&textureMatrix)) {
bsalomon3aa5fce2015-11-12 09:59:44 -0800312 return;
313 }
bsalomon3aa5fce2015-11-12 09:59:44 -0800314 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500315 auto fp = producer->createFragmentProcessor(textureMatrix, src, constraintMode,
Brian Salomon777e1462020-02-28 21:10:31 -0500316 coordsAllInsideSrcRect, wm, wm, filterMode);
Brian Osman05c8f462018-10-22 17:13:36 -0400317 fp = GrColorSpaceXformEffect::Make(std::move(fp), producer->colorSpace(), producer->alphaType(),
Brian Salomon4bc0c1f2019-09-30 15:12:27 -0400318 rtc->colorInfo().colorSpace());
bsalomonc55271f2015-11-09 11:55:57 -0800319 if (!fp) {
320 return;
321 }
joshualitt33a5fce2015-11-18 13:28:51 -0800322
bsalomonc55271f2015-11-09 11:55:57 -0800323 GrPaint grPaint;
Brian Salomon4bc0c1f2019-09-30 15:12:27 -0400324 if (!SkPaintToGrPaintWithTexture(context, rtc->colorInfo(), paint, ctm, std::move(fp),
325 producer->isAlphaOnly(), &grPaint)) {
bsalomonc55271f2015-11-09 11:55:57 -0800326 return;
327 }
328
329 if (!mf) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500330 // Can draw the image directly (any mask filter on the paint was converted to an FP already)
331 if (dstClip) {
332 SkPoint srcClipPoints[4];
333 SkPoint* srcClip = nullptr;
334 if (canUseTextureCoordsAsLocalCoords) {
335 // Calculate texture coordinates that match the dst clip
336 GrMapRectPoints(dst, src, dstClip, srcClipPoints, 4);
337 srcClip = srcClipPoints;
338 }
339 rtc->fillQuadWithEdgeAA(clip, std::move(grPaint), aa, aaFlags, ctm, dstClip, srcClip);
340 } else {
341 // Provide explicit texture coords when possible, otherwise rely on texture matrix
342 rtc->fillRectWithEdgeAA(clip, std::move(grPaint), aa, aaFlags, ctm, dst,
343 canUseTextureCoordsAsLocalCoords ? &src : nullptr);
344 }
345 } else {
346 // Must draw the mask filter as a GrShape. For now, this loses the per-edge AA information
347 // since it always draws with AA, but that is should not be noticeable since the mask filter
348 // is probably a blur.
349 GrShape shape;
350 if (dstClip) {
351 // Represent it as an SkPath formed from the dstClip
352 SkPath path;
353 path.addPoly(dstClip, 4, true);
354 shape = GrShape(path);
355 } else {
356 shape = GrShape(dst);
357 }
358
359 GrBlurUtils::drawShapeWithMaskFilter(
360 context, rtc, clip, shape, std::move(grPaint), ctm, mf);
361 }
362}
363
364} // anonymous namespace
365
366//////////////////////////////////////////////////////////////////////////////
367
368void SkGpuDevice::drawImageQuad(const SkImage* image, const SkRect* srcRect, const SkRect* dstRect,
369 const SkPoint dstClip[4], GrAA aa, GrQuadAAFlags aaFlags,
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500370 const SkMatrix* preViewMatrix, const SkPaint& paint,
371 SkCanvas::SrcRectConstraint constraint) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500372 SkRect src;
373 SkRect dst;
374 SkMatrix srcToDst;
375 ImageDrawMode mode = optimize_sample_area(SkISize::Make(image->width(), image->height()),
376 srcRect, dstRect, dstClip, &src, &dst, &srcToDst);
377 if (mode == ImageDrawMode::kSkip) {
bsalomonc55271f2015-11-09 11:55:57 -0800378 return;
379 }
380
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500381 if (src.contains(image->bounds())) {
382 constraint = SkCanvas::kFast_SrcRectConstraint;
383 }
384 // Depending on the nature of image, it can flow through more or less optimal pipelines
Brian Salomon777e1462020-02-28 21:10:31 -0500385 GrSamplerState::WrapMode wrapMode = mode == ImageDrawMode::kDecal
386 ? GrSamplerState::WrapMode::kClampToBorder
387 : GrSamplerState::WrapMode::kClamp;
Robert Phillips27927a52018-08-20 13:18:12 -0400388
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500389 // Get final CTM matrix
Michael Ludwigc89d1b52019-10-18 11:32:56 -0400390 SkMatrix ctm = this->localToDevice();
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500391 if (preViewMatrix) {
392 ctm.preConcat(*preViewMatrix);
393 }
394
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500395 // YUVA images can be stored in multiple images with different plane resolutions, so this
396 // uses an effect to combine them dynamically on the GPU. This is done before requesting a
397 // pinned texture proxy because YUV images force-flatten to RGBA in that scenario.
398 if (as_IB(image)->isYUVA()) {
399 SK_HISTOGRAM_BOOLEAN("DrawTiled", false);
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500400 LogDrawScaleFactor(ctm, srcToDst, paint.getFilterQuality());
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500401
Brian Salomon777e1462020-02-28 21:10:31 -0500402 GrYUVAImageTextureMaker maker(fContext.get(), image);
403 draw_texture_producer(fContext.get(), fRenderTargetContext.get(), this->clip(), ctm, paint,
404 &maker, src, dst, dstClip, srcToDst, aa, aaFlags, constraint,
405 wrapMode);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500406 return;
407 }
408
409 // Pinned texture proxies can be rendered directly as textures, or with relatively simple
410 // adjustments applied to the image content (scaling, mipmaps, color space, etc.)
411 uint32_t pinnedUniqueID;
Greg Danielcc21d0c2020-02-05 16:58:40 -0500412 if (GrSurfaceProxyView view = as_IB(image)->refPinnedView(this->context(), &pinnedUniqueID)) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500413 SK_HISTOGRAM_BOOLEAN("DrawTiled", false);
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500414 LogDrawScaleFactor(ctm, srcToDst, paint.getFilterQuality());
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500415
Greg Daniel82c6b102020-01-21 10:33:22 -0500416 GrColorInfo colorInfo;
Greg Danielcc21d0c2020-02-05 16:58:40 -0500417 if (fContext->priv().caps()->isFormatSRGB(view.proxy()->backendFormat())) {
Greg Daniel82c6b102020-01-21 10:33:22 -0500418 SkASSERT(image->imageInfo().colorType() == kRGBA_8888_SkColorType);
419 colorInfo = GrColorInfo(GrColorType::kRGBA_8888_SRGB, image->imageInfo().alphaType(),
420 image->imageInfo().refColorSpace());
421 } else {
422 colorInfo = GrColorInfo(image->imageInfo().colorInfo());
423 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500424
Brian Salomon777e1462020-02-28 21:10:31 -0500425 GrTextureAdjuster adjuster(fContext.get(), std::move(view), colorInfo, pinnedUniqueID);
426 draw_texture_producer(fContext.get(), fRenderTargetContext.get(), this->clip(), ctm, paint,
427 &adjuster, src, dst, dstClip, srcToDst, aa, aaFlags, constraint,
428 wrapMode);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500429 return;
430 }
431
432 // Next up, try tiling the image
433 // TODO (michaelludwig): Implement this with per-edge AA flags to handle seaming properly
434 // instead of going through drawBitmapRect (which will be removed from SkDevice in the future)
435 SkBitmap bm;
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500436 if (this->shouldTileImage(image, &src, constraint, paint.getFilterQuality(), ctm, srcToDst)) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500437 // only support tiling as bitmap at the moment, so force raster-version
Jim Van Verth352f4f72019-10-29 15:26:00 -0400438 if (as_IB(image)->getROPixels(&bm)) {
439 this->drawBitmapRect(bm, &src, dst, paint, constraint);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500440 return;
441 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500442 }
443
444 // This is the funnel for all non-tiled bitmap/image draw calls. Log a histogram entry.
445 SK_HISTOGRAM_BOOLEAN("DrawTiled", false);
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500446 LogDrawScaleFactor(ctm, srcToDst, paint.getFilterQuality());
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500447
448 // Lazily generated images must get drawn as a texture producer that handles the final
449 // texture creation.
450 if (image->isLazyGenerated()) {
Brian Salomon777e1462020-02-28 21:10:31 -0500451 GrImageTextureMaker maker(fContext.get(), image, SkImage::kAllow_CachingHint);
452 draw_texture_producer(fContext.get(), fRenderTargetContext.get(), this->clip(), ctm, paint,
453 &maker, src, dst, dstClip, srcToDst, aa, aaFlags, constraint,
454 wrapMode);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500455 return;
456 }
457 if (as_IB(image)->getROPixels(&bm)) {
Greg Daniel6f5441a2020-01-28 17:02:49 -0500458 GrBitmapTextureMaker maker(fContext.get(), bm, GrBitmapTextureMaker::Cached::kYes,
Brian Salomon777e1462020-02-28 21:10:31 -0500459 SkBackingFit::kExact);
460 draw_texture_producer(fContext.get(), fRenderTargetContext.get(), this->clip(), ctm, paint,
461 &maker, src, dst, dstClip, srcToDst, aa, aaFlags, constraint,
462 wrapMode);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500463 }
464
465 // Otherwise don't know how to draw it
466}
467
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400468void SkGpuDevice::drawEdgeAAImageSet(const SkCanvas::ImageSetEntry set[], int count,
469 const SkPoint dstClips[], const SkMatrix preViewMatrices[],
470 const SkPaint& paint, SkCanvas::SrcRectConstraint constraint) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500471 SkASSERT(count > 0);
Michael Ludwig31ba7182019-04-03 10:38:06 -0400472 if (!can_use_draw_texture(paint)) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500473 // Send every entry through drawImageQuad() to handle the more complicated paint
474 int dstClipIndex = 0;
475 for (int i = 0; i < count; ++i) {
476 // Only no clip or quad clip are supported
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400477 SkASSERT(!set[i].fHasClip || dstClips);
478 SkASSERT(set[i].fMatrixIndex < 0 || preViewMatrices);
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500479
Brian Salomondb151e02019-09-17 12:11:16 -0400480 SkTCopyOnFirstWrite<SkPaint> entryPaint(paint);
481 if (set[i].fAlpha != 1.f) {
482 auto paintAlpha = paint.getAlphaf();
483 entryPaint.writable()->setAlphaf(paintAlpha * set[i].fAlpha);
484 }
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500485 // Always send GrAA::kYes to preserve seaming across tiling in MSAA
Brian Salomondb151e02019-09-17 12:11:16 -0400486 this->drawImageQuad(
487 set[i].fImage.get(), &set[i].fSrcRect, &set[i].fDstRect,
488 set[i].fHasClip ? dstClips + dstClipIndex : nullptr, GrAA::kYes,
489 SkToGrQuadAAFlags(set[i].fAAFlags),
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400490 set[i].fMatrixIndex < 0 ? nullptr : preViewMatrices + set[i].fMatrixIndex,
Brian Salomondb151e02019-09-17 12:11:16 -0400491 *entryPaint, constraint);
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400492 dstClipIndex += 4 * set[i].fHasClip;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500493 }
494 return;
495 }
496
497 GrSamplerState::Filter filter = kNone_SkFilterQuality == paint.getFilterQuality() ?
498 GrSamplerState::Filter::kNearest : GrSamplerState::Filter::kBilerp;
499 SkBlendMode mode = paint.getBlendMode();
500
501 SkAutoTArray<GrRenderTargetContext::TextureSetEntry> textures(count);
502 // We accumulate compatible proxies until we find an an incompatible one or reach the end and
Michael Ludwig379e4962019-12-06 13:21:26 -0500503 // issue the accumulated 'n' draws starting at 'base'. 'p' represents the number of proxy
504 // switches that occur within the 'n' entries.
505 int base = 0, n = 0, p = 0;
506 auto draw = [&](int nextBase) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500507 if (n > 0) {
508 auto textureXform = GrColorSpaceXform::Make(
509 set[base].fImage->colorSpace(), set[base].fImage->alphaType(),
Brian Salomon4bc0c1f2019-09-30 15:12:27 -0400510 fRenderTargetContext->colorInfo().colorSpace(), kPremul_SkAlphaType);
Michael Ludwig379e4962019-12-06 13:21:26 -0500511 fRenderTargetContext->drawTextureSet(this->clip(), textures.get() + base, n, p,
Michael Ludwigc89d1b52019-10-18 11:32:56 -0400512 filter, mode, GrAA::kYes, constraint,
513 this->localToDevice(), std::move(textureXform));
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500514 }
Michael Ludwig379e4962019-12-06 13:21:26 -0500515 base = nextBase;
516 n = 0;
517 p = 0;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500518 };
519 int dstClipIndex = 0;
520 for (int i = 0; i < count; ++i) {
Michael Ludwigd9958f82019-03-21 13:08:36 -0400521 SkASSERT(!set[i].fHasClip || dstClips);
522 SkASSERT(set[i].fMatrixIndex < 0 || preViewMatrices);
523
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500524 // Manage the dst clip pointer tracking before any continues are used so we don't lose
525 // our place in the dstClips array.
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400526 const SkPoint* clip = set[i].fHasClip ? dstClips + dstClipIndex : nullptr;
527 dstClipIndex += 4 * set[i].fHasClip;
528
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500529 // The default SkBaseDevice implementation is based on drawImageRect which does not allow
530 // non-sorted src rects. TODO: Decide this is OK or make sure we handle it.
531 if (!set[i].fSrcRect.isSorted()) {
Michael Ludwig379e4962019-12-06 13:21:26 -0500532 draw(i + 1);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500533 continue;
534 }
535
Greg Danielcc21d0c2020-02-05 16:58:40 -0500536 GrSurfaceProxyView view;
Michael Ludwigd9958f82019-03-21 13:08:36 -0400537 const SkImage_Base* image = as_IB(set[i].fImage.get());
Greg Danielcc21d0c2020-02-05 16:58:40 -0500538 // Extract view from image, but skip YUV images so they get processed through
Michael Ludwigd9958f82019-03-21 13:08:36 -0400539 // drawImageQuad and the proper effect to dynamically sample their planes.
540 if (!image->isYUVA()) {
541 uint32_t uniqueID;
Greg Danielcc21d0c2020-02-05 16:58:40 -0500542 view = image->refPinnedView(this->context(), &uniqueID);
543 if (!view) {
Brian Salomonecbb0fb2020-02-28 18:07:32 -0500544 view = image->refView(this->context(), GrMipMapped::kNo);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500545 }
546 }
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500547
Greg Danielcc21d0c2020-02-05 16:58:40 -0500548 if (!view) {
Michael Ludwigd9958f82019-03-21 13:08:36 -0400549 // This image can't go through the texture op, send through general image pipeline
550 // after flushing current batch.
Michael Ludwig379e4962019-12-06 13:21:26 -0500551 draw(i + 1);
Brian Salomondb151e02019-09-17 12:11:16 -0400552 SkTCopyOnFirstWrite<SkPaint> entryPaint(paint);
553 if (set[i].fAlpha != 1.f) {
554 auto paintAlpha = paint.getAlphaf();
555 entryPaint.writable()->setAlphaf(paintAlpha * set[i].fAlpha);
556 }
557 this->drawImageQuad(
558 image, &set[i].fSrcRect, &set[i].fDstRect, clip, GrAA::kYes,
Michael Ludwigd9958f82019-03-21 13:08:36 -0400559 SkToGrQuadAAFlags(set[i].fAAFlags),
560 set[i].fMatrixIndex < 0 ? nullptr : preViewMatrices + set[i].fMatrixIndex,
Brian Salomondb151e02019-09-17 12:11:16 -0400561 *entryPaint, constraint);
Michael Ludwigd9958f82019-03-21 13:08:36 -0400562 continue;
563 }
Michael Ludwig7ae2ab52019-03-05 16:00:20 -0500564
Greg Danielcc21d0c2020-02-05 16:58:40 -0500565 textures[i].fProxyView = std::move(view);
Brian Salomonfc118442019-11-22 19:09:27 -0500566 textures[i].fSrcAlphaType = image->alphaType();
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500567 textures[i].fSrcRect = set[i].fSrcRect;
568 textures[i].fDstRect = set[i].fDstRect;
569 textures[i].fDstClipQuad = clip;
Michael Ludwig390f0cc2019-03-19 09:16:38 -0400570 textures[i].fPreViewMatrix =
571 set[i].fMatrixIndex < 0 ? nullptr : preViewMatrices + set[i].fMatrixIndex;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500572 textures[i].fAlpha = set[i].fAlpha * paint.getAlphaf();
573 textures[i].fAAFlags = SkToGrQuadAAFlags(set[i].fAAFlags);
574
575 if (n > 0 &&
Greg Daniel549325c2019-10-30 16:19:20 -0400576 (!GrTextureProxy::ProxiesAreCompatibleAsDynamicState(
Michael Ludwigfcdd0612019-11-25 08:34:31 -0500577 textures[i].fProxyView.proxy(),
578 textures[base].fProxyView.proxy()) ||
Greg Daniel507736f2020-01-17 15:36:10 -0500579 textures[i].fProxyView.swizzle() != textures[base].fProxyView.swizzle() ||
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500580 set[i].fImage->alphaType() != set[base].fImage->alphaType() ||
581 !SkColorSpace::Equals(set[i].fImage->colorSpace(), set[base].fImage->colorSpace()))) {
Michael Ludwig379e4962019-12-06 13:21:26 -0500582 draw(i);
583 }
584 // Whether or not we submitted a draw in the above if(), this ith entry is in the current
585 // set being accumulated so increment n, and increment p if proxies are different.
586 ++n;
587 if (n == 1 || textures[i - 1].fProxyView.proxy() != textures[i].fProxyView.proxy()) {
588 // First proxy or a different proxy (that is compatible, otherwise we'd have drawn up
589 // to i - 1).
590 ++p;
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500591 }
592 }
Michael Ludwig379e4962019-12-06 13:21:26 -0500593 draw(count);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500594}
595
596// TODO (michaelludwig) - to be removed when drawBitmapRect doesn't need it anymore
597void SkGpuDevice::drawTextureProducer(GrTextureProducer* producer,
598 const SkRect* srcRect,
599 const SkRect* dstRect,
600 SkCanvas::SrcRectConstraint constraint,
601 const SkMatrix& viewMatrix,
Brian Salomon777e1462020-02-28 21:10:31 -0500602 const SkPaint& paint) {
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500603 // The texture refactor split the old logic of drawTextureProducer into the beginning of
604 // drawImageQuad() and into the static draw_texture_producer. Replicate necessary logic that
605 // drawImageQuad() handles.
606 SkRect src;
607 SkRect dst;
608 SkMatrix srcToDst;
Brian Salomon1a217eb2019-10-24 10:50:36 -0400609 ImageDrawMode mode = optimize_sample_area(producer->dimensions(), srcRect, dstRect, nullptr,
610 &src, &dst, &srcToDst);
Michael Ludwig1433cfd2019-02-27 17:12:30 -0500611 if (mode == ImageDrawMode::kSkip) {
612 return;
613 }
614 // There's no dstClip to worry about and the producer is already made so we wouldn't be able
615 // to tell it to use decals if we had to
616 SkASSERT(mode != ImageDrawMode::kDecal);
617
618 draw_texture_producer(fContext.get(), fRenderTargetContext.get(), this->clip(), viewMatrix,
619 paint, producer, src, dst, /* clip */ nullptr, srcToDst,
620 GrAA(paint.isAntiAlias()),
621 paint.isAntiAlias() ? GrQuadAAFlags::kAll : GrQuadAAFlags::kNone,
Brian Salomon777e1462020-02-28 21:10:31 -0500622 constraint, GrSamplerState::WrapMode::kClamp);
bsalomonc55271f2015-11-09 11:55:57 -0800623}