blob: 6867e405d9babbcfa9c054b22fe7723dd4e01773 [file] [log] [blame]
Brian Osmane8e54582016-11-28 10:06:27 -05001/*
2 * Copyright 2016 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 "include/private/GrRecordingContext.h"
9#include "src/gpu/GrColorSpaceXform.h"
10#include "src/gpu/GrGpu.h"
11#include "src/gpu/GrProxyProvider.h"
12#include "src/gpu/GrRecordingContextPriv.h"
13#include "src/gpu/GrTextureAdjuster.h"
14#include "src/gpu/SkGr.h"
Brian Osmane8e54582016-11-28 10:06:27 -050015
Brian Salomone7499c72019-06-24 12:12:36 -040016GrTextureAdjuster::GrTextureAdjuster(GrRecordingContext* context,
Greg Danielcc104db2020-02-03 14:17:08 -050017 GrSurfaceProxyView original,
Greg Daniela4828a12019-10-11 13:51:02 -040018 const GrColorInfo& colorInfo,
Brian Salomon777e1462020-02-28 21:10:31 -050019 uint32_t uniqueID)
20 : INHERITED(context, {colorInfo, original.proxy()->dimensions()})
Greg Daniela4828a12019-10-11 13:51:02 -040021 , fOriginal(std::move(original))
22 , fUniqueID(uniqueID) {}
23
Brian Salomonc8d092a2020-02-24 10:14:21 -050024GrSurfaceProxyView GrTextureAdjuster::makeMippedCopy() {
Robert Phillips9338c602019-02-19 12:52:29 -050025 GrProxyProvider* proxyProvider = this->context()->priv().proxyProvider();
Robert Phillips1afd4cd2018-01-08 13:40:32 -050026
Brian Salomonb62cee32020-02-28 13:34:16 -050027 GrUniqueKey baseKey, mipMappedKey;
28 GrMakeKeyFromImageID(&baseKey, fUniqueID, SkIRect::MakeSize(this->dimensions()));
29 if (baseKey.isValid()) {
30 static const GrUniqueKey::Domain kMipMappedDomain = GrUniqueKey::GenerateDomain();
31 GrUniqueKey::Builder builder(&mipMappedKey, baseKey, kMipMappedDomain, 0);
32 }
Greg Daniel09c94002018-06-08 22:11:51 +000033 sk_sp<GrTextureProxy> cachedCopy;
Brian Salomonb62cee32020-02-28 13:34:16 -050034 if (mipMappedKey.isValid()) {
35 cachedCopy = proxyProvider->findOrCreateProxyByUniqueKey(mipMappedKey, this->colorType());
Brian Salomonc8d092a2020-02-24 10:14:21 -050036 if (cachedCopy) {
Brian Salomonb62cee32020-02-28 13:34:16 -050037 return {std::move(cachedCopy), fOriginal.origin(), fOriginal.swizzle()};
Robert Phillips0c984a02017-03-16 07:51:56 -040038 }
39 }
40
Brian Salomonc8d092a2020-02-24 10:14:21 -050041 GrSurfaceProxyView copyView = GrCopyBaseMipMapToTextureProxy(
Brian Salomonb62cee32020-02-28 13:34:16 -050042 this->context(), fOriginal.proxy(), fOriginal.origin(), this->colorType());
Brian Salomonc8d092a2020-02-24 10:14:21 -050043 if (!copyView) {
44 return {};
Greg Danielcce65002020-01-23 11:16:49 -050045 }
Brian Salomonb62cee32020-02-28 13:34:16 -050046 if (mipMappedKey.isValid()) {
47 SkASSERT(copyView.origin() == fOriginal.origin());
48 // TODO: If we move listeners up from SkImage_Lazy to SkImage_Base then add one here.
49 proxyProvider->assignUniqueKeyToProxy(mipMappedKey, copyView.asTextureProxy());
Brian Osmane8e54582016-11-28 10:06:27 -050050 }
Greg Danielcc104db2020-02-03 14:17:08 -050051 return copyView;
Brian Osmane8e54582016-11-28 10:06:27 -050052}
53
Brian Salomonecbb0fb2020-02-28 18:07:32 -050054GrSurfaceProxyView GrTextureAdjuster::onView(GrMipMapped mipMapped) {
Robert Phillips9338c602019-02-19 12:52:29 -050055 if (this->context()->priv().abandoned()) {
Brian Osmane8e54582016-11-28 10:06:27 -050056 // The texture was abandoned.
Greg Danielcc104db2020-02-03 14:17:08 -050057 return {};
Brian Osmane8e54582016-11-28 10:06:27 -050058 }
Brian Salomon4df00922017-09-07 16:34:11 +000059
Robert Phillips9338c602019-02-19 12:52:29 -050060 SkASSERT(this->width() <= this->context()->priv().caps()->maxTextureSize() &&
61 this->height() <= this->context()->priv().caps()->maxTextureSize());
Brian Osman875f7852018-04-12 13:29:08 -040062
Brian Salomonb62cee32020-02-28 13:34:16 -050063 GrTextureProxy* texProxy = fOriginal.asTextureProxy();
Greg Danielcc104db2020-02-03 14:17:08 -050064 SkASSERT(texProxy);
Brian Salomonecbb0fb2020-02-28 18:07:32 -050065 if (mipMapped == GrMipMapped::kNo || texProxy->mipMapped() == GrMipMapped::kYes) {
Brian Salomonb62cee32020-02-28 13:34:16 -050066 return fOriginal;
Brian Osmane8e54582016-11-28 10:06:27 -050067 }
68
Brian Salomonc8d092a2020-02-24 10:14:21 -050069 GrSurfaceProxyView copy = this->makeMippedCopy();
70 if (!copy) {
Greg Daniel8f5bbda2018-06-08 17:22:23 -040071 // If we were unable to make a copy and we only needed a copy for mips, then we will return
72 // the source texture here and require that the GPU backend is able to fall back to using
73 // bilerp if mips are required.
Brian Salomonb62cee32020-02-28 13:34:16 -050074 return fOriginal;
Greg Daniel8f5bbda2018-06-08 17:22:23 -040075 }
Brian Salomonc8d092a2020-02-24 10:14:21 -050076 SkASSERT(copy.asTextureProxy());
77 return copy;
Brian Osmane8e54582016-11-28 10:06:27 -050078}
79
Brian Salomonaff329b2017-08-11 09:40:37 -040080std::unique_ptr<GrFragmentProcessor> GrTextureAdjuster::createFragmentProcessor(
Brian Salomonc8d092a2020-02-24 10:14:21 -050081 const SkMatrix& textureMatrix,
Greg Danielc77085d2017-11-01 16:38:48 -040082 const SkRect& constraintRect,
Brian Salomonaff329b2017-08-11 09:40:37 -040083 FilterConstraint filterConstraint,
84 bool coordsLimitedToConstraintRect,
Brian Salomon777e1462020-02-28 21:10:31 -050085 GrSamplerState::WrapMode wrapX,
86 GrSamplerState::WrapMode wrapY,
Brian Osman05c8f462018-10-22 17:13:36 -040087 const GrSamplerState::Filter* filterOrNullForBicubic) {
Brian Salomonecbb0fb2020-02-28 18:07:32 -050088 GrSurfaceProxyView view;
89 if (filterOrNullForBicubic) {
90 view = this->view(*filterOrNullForBicubic);
91 } else {
92 view = this->view(GrMipMapped::kNo);
93 }
Brian Salomonc8d092a2020-02-24 10:14:21 -050094 if (!view) {
Brian Osmane8e54582016-11-28 10:06:27 -050095 return nullptr;
96 }
Greg Danielcc104db2020-02-03 14:17:08 -050097 SkASSERT(view.asTextureProxy());
Brian Osmane8e54582016-11-28 10:06:27 -050098
Michael Ludwigddeed372019-02-20 16:50:10 -050099 SkRect domain;
Brian Osmane8e54582016-11-28 10:06:27 -0500100 DomainMode domainMode =
Greg Danielc77085d2017-11-01 16:38:48 -0400101 DetermineDomainMode(constraintRect, filterConstraint, coordsLimitedToConstraintRect,
Greg Danielcc104db2020-02-03 14:17:08 -0500102 view.proxy(), filterOrNullForBicubic, &domain);
Brian Osmane8e54582016-11-28 10:06:27 -0500103 if (kTightCopy_DomainMode == domainMode) {
104 // TODO: Copy the texture and adjust the texture matrix (both parts need to consider
105 // non-int constraint rect)
106 // For now: treat as bilerp and ignore what goes on above level 0.
107
108 // We only expect MIP maps to require a tight copy.
109 SkASSERT(filterOrNullForBicubic &&
Brian Salomon2bbdcc42017-09-07 12:36:34 -0400110 GrSamplerState::Filter::kMipMap == *filterOrNullForBicubic);
111 static const GrSamplerState::Filter kBilerp = GrSamplerState::Filter::kBilerp;
Brian Osmane8e54582016-11-28 10:06:27 -0500112 domainMode =
Greg Danielc77085d2017-11-01 16:38:48 -0400113 DetermineDomainMode(constraintRect, filterConstraint, coordsLimitedToConstraintRect,
Greg Danielcc104db2020-02-03 14:17:08 -0500114 view.proxy(), &kBilerp, &domain);
Brian Osmane8e54582016-11-28 10:06:27 -0500115 SkASSERT(kTightCopy_DomainMode != domainMode);
116 }
117 SkASSERT(kNoDomain_DomainMode == domainMode ||
118 (domain.fLeft <= domain.fRight && domain.fTop <= domain.fBottom));
Brian Salomon777e1462020-02-28 21:10:31 -0500119 return this->createFragmentProcessorForSubsetAndFilter(std::move(view), textureMatrix,
120 domainMode, domain, wrapX, wrapY,
121 filterOrNullForBicubic);
Brian Osmane8e54582016-11-28 10:06:27 -0500122}