| /* |
| * Copyright 2015 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #ifndef GrTextureMaker_DEFINED |
| #define GrTextureMaker_DEFINED |
| |
| #include "GrTextureParams.h" |
| #include "GrResourceKey.h" |
| #include "GrTexture.h" |
| #include "SkTLazy.h" |
| |
| class GrContext; |
| class GrTextureParams; |
| class GrUniqueKey; |
| class SkBitmap; |
| |
| /** |
| * Different GPUs and API extensions have different requirements with respect to what texture |
| * sampling parameters may be used with textures of various types. This class facilitates making |
| * texture compatible with a given GrTextureParams. There are two immediate subclasses defined |
| * below. One is a base class for sources that are inherently texture-backed (e.g. a texture-backed |
| * SkImage). It supports subsetting the original texture. The other is for use cases where the |
| * source can generate a texture that represents some content (e.g. cpu pixels, SkPicture, ...). |
| */ |
| class GrTextureProducer : public SkNoncopyable { |
| public: |
| struct CopyParams { |
| GrTextureParams::FilterMode fFilter; |
| int fWidth; |
| int fHeight; |
| }; |
| |
| enum FilterConstraint { |
| kYes_FilterConstraint, |
| kNo_FilterConstraint, |
| }; |
| |
| /** |
| * Helper for creating a fragment processor to sample the texture with a given filtering mode. |
| * It attempts to avoid making texture copies or using domains whenever possible. |
| * |
| * @param textureMatrix Matrix used to access the texture. It is applied to |
| * the local coords. The post-transformed coords should |
| * be in texel units (rather than normalized) with |
| * respect to this Producer's bounds (width()/height()). |
| * @param constraintRect A rect that represents the area of the texture to be |
| * sampled. It must be contained in the Producer's bounds |
| * as defined by width()/height(). |
| * @param filterConstriant Indicates whether filtering is limited to |
| * constraintRect. |
| * @param coordsLimitedToConstraintRect Is it known that textureMatrix*localCoords is bound |
| * by the portion of the texture indicated by |
| * constraintRect (without consideration of filter |
| * width, just the raw coords). |
| * @param filterOrNullForBicubic If non-null indicates the filter mode. If null means |
| * use bicubic filtering. |
| **/ |
| virtual sk_sp<GrFragmentProcessor> createFragmentProcessor( |
| const SkMatrix& textureMatrix, |
| const SkRect& constraintRect, |
| FilterConstraint filterConstraint, |
| bool coordsLimitedToConstraintRect, |
| const GrTextureParams::FilterMode* filterOrNullForBicubic, |
| SkColorSpace* dstColorSpace, |
| SkSourceGammaTreatment) = 0; |
| |
| virtual ~GrTextureProducer() {} |
| |
| int width() const { return fWidth; } |
| int height() const { return fHeight; } |
| bool isAlphaOnly() const { return fIsAlphaOnly; } |
| virtual SkAlphaType alphaType() const = 0; |
| virtual SkColorSpace* getColorSpace() = 0; |
| |
| protected: |
| GrTextureProducer(int width, int height, bool isAlphaOnly) |
| : fWidth(width) |
| , fHeight(height) |
| , fIsAlphaOnly(isAlphaOnly) {} |
| |
| /** Helper for creating a key for a copy from an original key. */ |
| static void MakeCopyKeyFromOrigKey(const GrUniqueKey& origKey, |
| const CopyParams& copyParams, |
| GrUniqueKey* copyKey) { |
| SkASSERT(!copyKey->isValid()); |
| if (origKey.isValid()) { |
| static const GrUniqueKey::Domain kDomain = GrUniqueKey::GenerateDomain(); |
| GrUniqueKey::Builder builder(copyKey, origKey, kDomain, 3); |
| builder[0] = copyParams.fFilter; |
| builder[1] = copyParams.fWidth; |
| builder[2] = copyParams.fHeight; |
| } |
| } |
| |
| /** |
| * If we need to make a copy in order to be compatible with GrTextureParams producer is asked to |
| * return a key that identifies its original content + the CopyParms parameter. If the producer |
| * does not want to cache the stretched version (e.g. the producer is volatile), this should |
| * simply return without initializing the copyKey. |
| */ |
| virtual void makeCopyKey(const CopyParams&, GrUniqueKey* copyKey) = 0; |
| |
| /** |
| * If a stretched version of the texture is generated, it may be cached (assuming that |
| * makeCopyKey() returns true). In that case, the maker is notified in case it |
| * wants to note that for when the maker is destroyed. |
| */ |
| virtual void didCacheCopy(const GrUniqueKey& copyKey) = 0; |
| |
| private: |
| const int fWidth; |
| const int fHeight; |
| const bool fIsAlphaOnly; |
| |
| typedef SkNoncopyable INHERITED; |
| }; |
| |
| /** |
| * Base class for sources that start out as textures. Optionally allows for a content area subrect. |
| * The intent is not to use content area for subrect rendering. Rather, the pixels outside the |
| * content area have undefined values and shouldn't be read *regardless* of filtering mode or |
| * the SkCanvas::SrcRectConstraint used for subrect draws. |
| */ |
| class GrTextureAdjuster : public GrTextureProducer { |
| public: |
| /** Makes the subset of the texture safe to use with the given texture parameters. |
| outOffset will be the top-left corner of the subset if a copy is not made. Otherwise, |
| the copy will be tight to the contents and outOffset will be (0, 0). If the copy's size |
| does not match subset's dimensions then the contents are scaled to fit the copy.*/ |
| GrTexture* refTextureSafeForParams(const GrTextureParams&, SkSourceGammaTreatment, |
| SkIPoint* outOffset); |
| |
| sk_sp<GrFragmentProcessor> createFragmentProcessor( |
| const SkMatrix& textureMatrix, |
| const SkRect& constraintRect, |
| FilterConstraint, |
| bool coordsLimitedToConstraintRect, |
| const GrTextureParams::FilterMode* filterOrNullForBicubic, |
| SkColorSpace* dstColorSpace, |
| SkSourceGammaTreatment) override; |
| |
| // We do not ref the texture nor the colorspace, so the caller must keep them in scope while |
| // this Adjuster is alive. |
| GrTextureAdjuster(GrTexture*, SkAlphaType, const SkIRect& area, uint32_t uniqueID, |
| SkColorSpace*); |
| |
| protected: |
| SkAlphaType alphaType() const override { return fAlphaType; } |
| SkColorSpace* getColorSpace() override; |
| void makeCopyKey(const CopyParams& params, GrUniqueKey* copyKey) override; |
| void didCacheCopy(const GrUniqueKey& copyKey) override; |
| |
| GrTexture* originalTexture() const { return fOriginal; } |
| |
| /** Returns the content area or null for the whole original texture */ |
| const SkIRect* contentAreaOrNull() { return fContentArea.getMaybeNull(); } |
| |
| private: |
| SkTLazy<SkIRect> fContentArea; |
| GrTexture* fOriginal; |
| SkAlphaType fAlphaType; |
| SkColorSpace* fColorSpace; |
| uint32_t fUniqueID; |
| |
| GrTexture* refCopy(const CopyParams ©Params); |
| |
| typedef GrTextureProducer INHERITED; |
| }; |
| |
| /** |
| * Base class for sources that start out as something other than a texture (encoded image, |
| * picture, ...). |
| */ |
| class GrTextureMaker : public GrTextureProducer { |
| public: |
| /** Returns a texture that is safe for use with the params. If the size of the returned texture |
| does not match width()/height() then the contents of the original must be scaled to fit |
| the texture. */ |
| GrTexture* refTextureForParams(const GrTextureParams&, SkSourceGammaTreatment); |
| |
| sk_sp<GrFragmentProcessor> createFragmentProcessor( |
| const SkMatrix& textureMatrix, |
| const SkRect& constraintRect, |
| FilterConstraint filterConstraint, |
| bool coordsLimitedToConstraintRect, |
| const GrTextureParams::FilterMode* filterOrNullForBicubic, |
| SkColorSpace* dstColorSpace, |
| SkSourceGammaTreatment) override; |
| |
| protected: |
| GrTextureMaker(GrContext* context, int width, int height, bool isAlphaOnly) |
| : INHERITED(width, height, isAlphaOnly) |
| , fContext(context) {} |
| |
| /** |
| * Return the maker's "original" texture. It is the responsibility of the maker to handle any |
| * caching of the original if desired. |
| */ |
| virtual GrTexture* refOriginalTexture(bool willBeMipped, SkSourceGammaTreatment) = 0; |
| |
| /** |
| * Return a new (uncached) texture that is the stretch of the maker's original. |
| * |
| * The base-class handles general logic for this, and only needs access to the following |
| * method: |
| * - refOriginalTexture() |
| * |
| * Subclass may override this if they can handle creating the texture more directly than |
| * by copying. |
| */ |
| virtual GrTexture* generateTextureForParams(const CopyParams&, bool willBeMipped, |
| SkSourceGammaTreatment); |
| |
| GrContext* context() const { return fContext; } |
| |
| private: |
| GrContext* fContext; |
| |
| typedef GrTextureProducer INHERITED; |
| }; |
| |
| #endif |