| /* | 
 |  * 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 const GrFragmentProcessor* createFragmentProcessor( | 
 |                                     const SkMatrix& textureMatrix, | 
 |                                     const SkRect& constraintRect, | 
 |                                     FilterConstraint filterConstraint, | 
 |                                     bool coordsLimitedToConstraintRect, | 
 |                                     const GrTextureParams::FilterMode* filterOrNullForBicubic) = 0; | 
 |  | 
 |     virtual ~GrTextureProducer() {} | 
 |  | 
 |     int width() const { return fWidth; } | 
 |     int height() const { return fHeight; } | 
 |     bool isAlphaOnly() const { return fIsAlphaOnly; } | 
 |  | 
 | 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&, SkIPoint* outOffset); | 
 |  | 
 |     const GrFragmentProcessor* createFragmentProcessor( | 
 |                                 const SkMatrix& textureMatrix, | 
 |                                 const SkRect& constraintRect, | 
 |                                 FilterConstraint, | 
 |                                 bool coordsLimitedToConstraintRect, | 
 |                                 const GrTextureParams::FilterMode* filterOrNullForBicubic) override; | 
 |  | 
 | protected: | 
 |     /** The whole texture is content. */ | 
 |     explicit GrTextureAdjuster(GrTexture* original, bool isAlphaOnly) | 
 |         : INHERITED(original->width(), original->height(), isAlphaOnly) | 
 |         , fOriginal(original) {} | 
 |  | 
 |     GrTextureAdjuster(GrTexture* original, const SkIRect& contentArea, bool isAlphaOnly); | 
 |  | 
 |     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; | 
 |  | 
 |     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&); | 
 |  | 
 |     const GrFragmentProcessor* createFragmentProcessor( | 
 |                                 const SkMatrix& textureMatrix, | 
 |                                 const SkRect& constraintRect, | 
 |                                 FilterConstraint filterConstraint, | 
 |                                 bool coordsLimitedToConstraintRect, | 
 |                                 const GrTextureParams::FilterMode* filterOrNullForBicubic) 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() = 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&); | 
 |  | 
 |     GrContext* context() const { return fContext; } | 
 |  | 
 | private: | 
 |     GrContext*  fContext; | 
 |  | 
 |     typedef GrTextureProducer INHERITED; | 
 | }; | 
 |  | 
 | #endif |