• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
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 
8 #include "SkImage_Base.h"
9 #include "SkImageCacherator.h"
10 
11 #include "SkBitmap.h"
12 #include "SkBitmapCache.h"
13 #include "SkColorSpace_Base.h"
14 #include "SkData.h"
15 #include "SkImageGenerator.h"
16 #include "SkImagePriv.h"
17 #include "SkNextID.h"
18 #include "SkPixelRef.h"
19 
20 #if SK_SUPPORT_GPU
21 #include "GrContext.h"
22 #include "GrContextPriv.h"
23 #include "GrGpuResourcePriv.h"
24 #include "GrImageTextureMaker.h"
25 #include "GrResourceKey.h"
26 #include "GrResourceProvider.h"
27 #include "GrSamplerParams.h"
28 #include "GrYUVProvider.h"
29 #include "SkGr.h"
30 #endif
31 
32 // Ref-counted tuple(SkImageGenerator, SkMutex) which allows sharing one generator among N images
33 class SharedGenerator final : public SkNVRefCnt<SharedGenerator> {
34 public:
Make(std::unique_ptr<SkImageGenerator> gen)35     static sk_sp<SharedGenerator> Make(std::unique_ptr<SkImageGenerator> gen) {
36         return gen ? sk_sp<SharedGenerator>(new SharedGenerator(std::move(gen))) : nullptr;
37     }
38 
39     // This is thread safe.  It is a const field set in the constructor.
getInfo()40     const SkImageInfo& getInfo() { return fGenerator->getInfo(); }
41 
42 private:
SharedGenerator(std::unique_ptr<SkImageGenerator> gen)43     explicit SharedGenerator(std::unique_ptr<SkImageGenerator> gen)
44             : fGenerator(std::move(gen)) {
45         SkASSERT(fGenerator);
46     }
47 
48     friend class ScopedGenerator;
49     friend class SkImage_Lazy;
50 
51     std::unique_ptr<SkImageGenerator> fGenerator;
52     SkMutex                           fMutex;
53 };
54 
55 class SkImage_Lazy : public SkImage_Base, public SkImageCacherator {
56 public:
57     struct Validator {
58         Validator(sk_sp<SharedGenerator>, const SkIRect* subset, sk_sp<SkColorSpace> colorSpace);
59 
operator boolSkImage_Lazy::Validator60         operator bool() const { return fSharedGenerator.get(); }
61 
62         sk_sp<SharedGenerator> fSharedGenerator;
63         SkImageInfo            fInfo;
64         SkIPoint               fOrigin;
65         sk_sp<SkColorSpace>    fColorSpace;
66         uint32_t               fUniqueID;
67     };
68 
69     SkImage_Lazy(Validator* validator);
70 
onImageInfo() const71     SkImageInfo onImageInfo() const override {
72         return fInfo;
73     }
onAlphaType() const74     SkAlphaType onAlphaType() const override {
75         return fInfo.alphaType();
76     }
77 
78     bool onReadPixels(const SkImageInfo&, void*, size_t, int srcX, int srcY,
79                       CachingHint) const override;
80 #if SK_SUPPORT_GPU
81     sk_sp<GrTextureProxy> asTextureProxyRef(GrContext*, const GrSamplerParams&,
82                                             SkColorSpace*, sk_sp<SkColorSpace>*,
83                                             SkScalar scaleAdjust[2]) const override;
84 #endif
85     SkData* onRefEncoded() const override;
86     sk_sp<SkImage> onMakeSubset(const SkIRect&) const override;
87     bool getROPixels(SkBitmap*, SkColorSpace* dstColorSpace, CachingHint) const override;
onIsLazyGenerated() const88     bool onIsLazyGenerated() const override { return true; }
89     bool onCanLazyGenerateOnGPU() const override;
90     sk_sp<SkImage> onMakeColorSpace(sk_sp<SkColorSpace>, SkColorType,
91                                     SkTransferFunctionBehavior) const override;
92 
93     bool onIsValid(GrContext*) const override;
94 
peekCacherator() const95     SkImageCacherator* peekCacherator() const override {
96         return const_cast<SkImage_Lazy*>(this);
97     }
98 
99     // Only return true if the generate has already been cached.
100     bool lockAsBitmapOnlyIfAlreadyCached(SkBitmap*, CachedFormat) const;
101     // Call the underlying generator directly
102     bool directGeneratePixels(const SkImageInfo& dstInfo, void* dstPixels, size_t dstRB,
103                               int srcX, int srcY, SkTransferFunctionBehavior behavior) const;
104 
105     // SkImageCacherator interface
106 #if SK_SUPPORT_GPU
107     // Returns the texture proxy. If the cacherator is generating the texture and wants to cache it,
108     // it should use the passed in key (if the key is valid).
109     sk_sp<GrTextureProxy> lockTextureProxy(GrContext*,
110                                            const GrUniqueKey& key,
111                                            SkImage::CachingHint,
112                                            bool willBeMipped,
113                                            SkColorSpace* dstColorSpace,
114                                            GrTextureMaker::AllowedTexGenType genType) override;
115 
116     // Returns the color space of the texture that would be returned if you called lockTexture.
117     // Separate code path to allow querying of the color space for textures that cached (even
118     // externally).
119     sk_sp<SkColorSpace> getColorSpace(GrContext*, SkColorSpace* dstColorSpace) override;
120     void makeCacheKeyFromOrigKey(const GrUniqueKey& origKey, CachedFormat,
121                                  GrUniqueKey* cacheKey) override;
122 #endif
123 
124     CachedFormat chooseCacheFormat(SkColorSpace* dstColorSpace,
125                                    const GrCaps* = nullptr) const override;
126     SkImageInfo buildCacheInfo(CachedFormat) const override;
127 
128 private:
129     class ScopedGenerator;
130 
131     /**
132      *  On success (true), bitmap will point to the pixels for this generator. If this returns
133      *  false, the bitmap will be reset to empty.
134      */
135     bool lockAsBitmap(SkBitmap*, SkImage::CachingHint, CachedFormat, const SkImageInfo&,
136                       SkTransferFunctionBehavior) const;
137 
138     /**
139      * Populates parameters to pass to the generator for reading pixels or generating a texture.
140      * For image generators, legacy versus true color blending is indicated using a
141      * SkTransferFunctionBehavior, and the target color space is specified on the SkImageInfo.
142      * If generatorImageInfo has no color space set, set its color space to this SkImage's color
143      * space, and return "ignore" behavior, indicating legacy mode. If generatorImageInfo has a
144      * color space set, return "respect" behavior, indicating linear blending mode.
145      */
146     SkTransferFunctionBehavior getGeneratorBehaviorAndInfo(SkImageInfo* generatorImageInfo) const;
147 
148     sk_sp<SharedGenerator> fSharedGenerator;
149     // Note that fInfo is not necessarily the info from the generator. It may be cropped by
150     // onMakeSubset and its color space may be changed by onMakeColorSpace.
151     const SkImageInfo      fInfo;
152     const SkIPoint         fOrigin;
153 
154     struct IDRec {
155         SkOnce      fOnce;
156         uint32_t    fUniqueID;
157     };
158     mutable IDRec fIDRecs[kNumCachedFormats];
159 
160     uint32_t getUniqueID(CachedFormat) const;
161 
162     // Repeated calls to onMakeColorSpace will result in a proliferation of unique IDs and
163     // SkImage_Lazy instances. Cache the result of the last successful onMakeColorSpace call.
164     mutable SkMutex             fOnMakeColorSpaceMutex;
165     mutable sk_sp<SkColorSpace> fOnMakeColorSpaceTarget;
166     mutable sk_sp<SkImage>      fOnMakeColorSpaceResult;
167 
168     typedef SkImage_Base INHERITED;
169 };
170 
171 ///////////////////////////////////////////////////////////////////////////////
172 
Validator(sk_sp<SharedGenerator> gen,const SkIRect * subset,sk_sp<SkColorSpace> colorSpace)173 SkImage_Lazy::Validator::Validator(sk_sp<SharedGenerator> gen, const SkIRect* subset,
174                                    sk_sp<SkColorSpace> colorSpace)
175         : fSharedGenerator(std::move(gen)) {
176     if (!fSharedGenerator) {
177         return;
178     }
179 
180     // The following generator accessors are safe without acquiring the mutex (const getters).
181     // TODO: refactor to use a ScopedGenerator instead, for clarity.
182     const SkImageInfo& info = fSharedGenerator->fGenerator->getInfo();
183     if (info.isEmpty()) {
184         fSharedGenerator.reset();
185         return;
186     }
187 
188     fUniqueID = fSharedGenerator->fGenerator->uniqueID();
189     const SkIRect bounds = SkIRect::MakeWH(info.width(), info.height());
190     if (subset) {
191         if (!bounds.contains(*subset)) {
192             fSharedGenerator.reset();
193             return;
194         }
195         if (*subset != bounds) {
196             // we need a different uniqueID since we really are a subset of the raw generator
197             fUniqueID = SkNextID::ImageID();
198         }
199     } else {
200         subset = &bounds;
201     }
202 
203     fInfo   = info.makeWH(subset->width(), subset->height());
204     fOrigin = SkIPoint::Make(subset->x(), subset->y());
205     if (colorSpace) {
206         fInfo = fInfo.makeColorSpace(colorSpace);
207         fUniqueID = SkNextID::ImageID();
208     }
209 }
210 
211 ///////////////////////////////////////////////////////////////////////////////
212 
213 // Helper for exclusive access to a shared generator.
214 class SkImage_Lazy::ScopedGenerator {
215 public:
ScopedGenerator(const sk_sp<SharedGenerator> & gen)216     ScopedGenerator(const sk_sp<SharedGenerator>& gen)
217       : fSharedGenerator(gen)
218       , fAutoAquire(gen->fMutex) {}
219 
operator ->() const220     SkImageGenerator* operator->() const {
221         fSharedGenerator->fMutex.assertHeld();
222         return fSharedGenerator->fGenerator.get();
223     }
224 
operator SkImageGenerator*() const225     operator SkImageGenerator*() const {
226         fSharedGenerator->fMutex.assertHeld();
227         return fSharedGenerator->fGenerator.get();
228     }
229 
230 private:
231     const sk_sp<SharedGenerator>& fSharedGenerator;
232     SkAutoExclusive               fAutoAquire;
233 };
234 
235 ///////////////////////////////////////////////////////////////////////////////
236 
SkImage_Lazy(Validator * validator)237 SkImage_Lazy::SkImage_Lazy(Validator* validator)
238         : INHERITED(validator->fInfo.width(), validator->fInfo.height(), validator->fUniqueID)
239         , fSharedGenerator(std::move(validator->fSharedGenerator))
240         , fInfo(validator->fInfo)
241         , fOrigin(validator->fOrigin) {
242     SkASSERT(fSharedGenerator);
243     // We explicit set the legacy format slot, but leave the others uninitialized (via SkOnce)
244     // and only resolove them to IDs as needed (by calling getUniqueID()).
245     fIDRecs[kLegacy_CachedFormat].fOnce([this, validator] {
246         fIDRecs[kLegacy_CachedFormat].fUniqueID = validator->fUniqueID;
247     });
248 }
249 
getUniqueID(CachedFormat format) const250 uint32_t SkImage_Lazy::getUniqueID(CachedFormat format) const {
251     IDRec* rec = &fIDRecs[format];
252     rec->fOnce([rec] {
253         rec->fUniqueID = SkNextID::ImageID();
254     });
255     return rec->fUniqueID;
256 }
257 
258 //////////////////////////////////////////////////////////////////////////////////////////////////
259 
260 // Abstraction of GrCaps that handles the cases where we don't have a caps pointer (because
261 // we're in raster mode), or where GPU support is entirely missing. In theory, we only need the
262 // chosen format to be texturable, but that lets us choose F16 on GLES implemenations where we
263 // won't be able to read the texture back. We'd like to ensure that SkImake::makeNonTextureImage
264 // works, so we require that the formats we choose are renderable (as a proxy for being readable).
265 struct CacheCaps {
CacheCapsCacheCaps266     CacheCaps(const GrCaps* caps) : fCaps(caps) {}
267 
268 #if SK_SUPPORT_GPU
supportsHalfFloatCacheCaps269     bool supportsHalfFloat() const {
270         return !fCaps ||
271             (fCaps->isConfigTexturable(kRGBA_half_GrPixelConfig) &&
272              fCaps->isConfigRenderable(kRGBA_half_GrPixelConfig, false));
273     }
274 
supportsSRGBCacheCaps275     bool supportsSRGB() const {
276         return !fCaps ||
277             (fCaps->srgbSupport() && fCaps->isConfigTexturable(kSRGBA_8888_GrPixelConfig));
278     }
279 
supportsSBGRCacheCaps280     bool supportsSBGR() const {
281         return !fCaps || fCaps->srgbSupport();
282     }
283 #else
supportsHalfFloatCacheCaps284     bool supportsHalfFloat() const { return true; }
supportsSRGBCacheCaps285     bool supportsSRGB() const { return true; }
supportsSBGRCacheCaps286     bool supportsSBGR() const { return true; }
287 #endif
288 
289     const GrCaps* fCaps;
290 };
291 
chooseCacheFormat(SkColorSpace * dstColorSpace,const GrCaps * grCaps) const292 SkImageCacherator::CachedFormat SkImage_Lazy::chooseCacheFormat(SkColorSpace* dstColorSpace,
293                                                                 const GrCaps* grCaps) const {
294     SkColorSpace* cs = fInfo.colorSpace();
295     if (!cs || !dstColorSpace) {
296         return kLegacy_CachedFormat;
297     }
298 
299     CacheCaps caps(grCaps);
300     switch (fInfo.colorType()) {
301         case kUnknown_SkColorType:
302         case kAlpha_8_SkColorType:
303         case kRGB_565_SkColorType:
304         case kARGB_4444_SkColorType:
305             // We don't support color space on these formats, so always decode in legacy mode:
306             // TODO: Ask the codec to decode these to something else (at least sRGB 8888)?
307             return kLegacy_CachedFormat;
308 
309         case kGray_8_SkColorType:
310             // TODO: What do we do with grayscale sources that have strange color spaces attached?
311             // The codecs and color space xform don't handle this correctly (yet), so drop it on
312             // the floor. (Also, inflating by a factor of 8 is going to be unfortunate).
313             // As it is, we don't directly support sRGB grayscale, so ask the codec to convert
314             // it for us. This bypasses some really sketchy code GrUploadPixmapToTexture.
315             if (cs->gammaCloseToSRGB() && caps.supportsSRGB()) {
316                 return kSRGB8888_CachedFormat;
317             } else {
318                 return kLegacy_CachedFormat;
319             }
320 
321         case kRGBA_8888_SkColorType:
322             if (cs->gammaCloseToSRGB()) {
323                 if (caps.supportsSRGB()) {
324                     return kSRGB8888_CachedFormat;
325                 } else if (caps.supportsHalfFloat()) {
326                     return kLinearF16_CachedFormat;
327                 } else {
328                     return kLegacy_CachedFormat;
329                 }
330             } else {
331                 if (caps.supportsHalfFloat()) {
332                     return kLinearF16_CachedFormat;
333                 } else if (caps.supportsSRGB()) {
334                     return kSRGB8888_CachedFormat;
335                 } else {
336                     return kLegacy_CachedFormat;
337                 }
338             }
339 
340         case kBGRA_8888_SkColorType:
341             // Odd case. sBGRA isn't a real thing, so we may not have this texturable.
342             if (caps.supportsSBGR()) {
343                 if (cs->gammaCloseToSRGB()) {
344                     return kSBGR8888_CachedFormat;
345                 } else if (caps.supportsHalfFloat()) {
346                     return kLinearF16_CachedFormat;
347                 } else if (caps.supportsSRGB()) {
348                     return kSRGB8888_CachedFormat;
349                 } else {
350                     // sBGRA support without sRGBA is highly unlikely (impossible?) Nevertheless.
351                     return kLegacy_CachedFormat;
352                 }
353             } else {
354                 if (cs->gammaCloseToSRGB()) {
355                     if (caps.supportsSRGB()) {
356                         return kSRGB8888_CachedFormat;
357                     } else if (caps.supportsHalfFloat()) {
358                         return kLinearF16_CachedFormat;
359                     } else {
360                         return kLegacy_CachedFormat;
361                     }
362                 } else {
363                     if (caps.supportsHalfFloat()) {
364                         return kLinearF16_CachedFormat;
365                     } else if (caps.supportsSRGB()) {
366                         return kSRGB8888_CachedFormat;
367                     } else {
368                         return kLegacy_CachedFormat;
369                     }
370                 }
371             }
372 
373         case kRGBA_F16_SkColorType:
374             if (caps.supportsHalfFloat()) {
375                 return kLinearF16_CachedFormat;
376             } else if (caps.supportsSRGB()) {
377                 return kSRGB8888_CachedFormat;
378             } else {
379                 return kLegacy_CachedFormat;
380             }
381     }
382     SkDEBUGFAIL("Unreachable");
383     return kLegacy_CachedFormat;
384 }
385 
buildCacheInfo(CachedFormat format) const386 SkImageInfo SkImage_Lazy::buildCacheInfo(CachedFormat format) const {
387     switch (format) {
388         case kLegacy_CachedFormat:
389             return fInfo.makeColorSpace(nullptr);
390         case kLinearF16_CachedFormat:
391             return fInfo.makeColorType(kRGBA_F16_SkColorType)
392                         .makeColorSpace(as_CSB(fInfo.colorSpace())->makeLinearGamma());
393         case kSRGB8888_CachedFormat:
394             // If the transfer function is nearly (but not exactly) sRGB, we don't want the codec
395             // to bother trans-coding. It would be slow, and do more harm than good visually,
396             // so we make sure to leave the colorspace as-is.
397             if (fInfo.colorSpace()->gammaCloseToSRGB()) {
398                 return fInfo.makeColorType(kRGBA_8888_SkColorType);
399             } else {
400                 return fInfo.makeColorType(kRGBA_8888_SkColorType)
401                             .makeColorSpace(as_CSB(fInfo.colorSpace())->makeSRGBGamma());
402             }
403         case kSBGR8888_CachedFormat:
404             // See note above about not-quite-sRGB transfer functions.
405             if (fInfo.colorSpace()->gammaCloseToSRGB()) {
406                 return fInfo.makeColorType(kBGRA_8888_SkColorType);
407             } else {
408                 return fInfo.makeColorType(kBGRA_8888_SkColorType)
409                             .makeColorSpace(as_CSB(fInfo.colorSpace())->makeSRGBGamma());
410             }
411         default:
412             SkDEBUGFAIL("Invalid cached format");
413             return fInfo;
414     }
415 }
416 
417 //////////////////////////////////////////////////////////////////////////////////////////////////
418 
check_output_bitmap(const SkBitmap & bitmap,uint32_t expectedID)419 static bool check_output_bitmap(const SkBitmap& bitmap, uint32_t expectedID) {
420     SkASSERT(bitmap.getGenerationID() == expectedID);
421     SkASSERT(bitmap.isImmutable());
422     SkASSERT(bitmap.getPixels());
423     return true;
424 }
425 
directGeneratePixels(const SkImageInfo & info,void * pixels,size_t rb,int srcX,int srcY,SkTransferFunctionBehavior behavior) const426 bool SkImage_Lazy::directGeneratePixels(const SkImageInfo& info, void* pixels, size_t rb,
427                                         int srcX, int srcY,
428                                         SkTransferFunctionBehavior behavior) const {
429     ScopedGenerator generator(fSharedGenerator);
430     const SkImageInfo& genInfo = generator->getInfo();
431     // Currently generators do not natively handle subsets, so check that first.
432     if (srcX || srcY || genInfo.width() != info.width() || genInfo.height() != info.height()) {
433         return false;
434     }
435 
436     SkImageGenerator::Options opts;
437     // TODO: This should respect the behavior argument.
438     opts.fBehavior = SkTransferFunctionBehavior::kIgnore;
439     return generator->getPixels(info, pixels, rb, &opts);
440 }
441 
442 //////////////////////////////////////////////////////////////////////////////////////////////////
443 
lockAsBitmapOnlyIfAlreadyCached(SkBitmap * bitmap,CachedFormat format) const444 bool SkImage_Lazy::lockAsBitmapOnlyIfAlreadyCached(SkBitmap* bitmap, CachedFormat format) const {
445     uint32_t uniqueID = this->getUniqueID(format);
446     return SkBitmapCache::Find(SkBitmapCacheDesc::Make(uniqueID,
447                                                        fInfo.width(), fInfo.height()), bitmap) &&
448            check_output_bitmap(*bitmap, uniqueID);
449 }
450 
generate_pixels(SkImageGenerator * gen,const SkPixmap & pmap,int originX,int originY,SkTransferFunctionBehavior behavior)451 static bool generate_pixels(SkImageGenerator* gen, const SkPixmap& pmap, int originX, int originY,
452                             SkTransferFunctionBehavior behavior) {
453     const int genW = gen->getInfo().width();
454     const int genH = gen->getInfo().height();
455     const SkIRect srcR = SkIRect::MakeWH(genW, genH);
456     const SkIRect dstR = SkIRect::MakeXYWH(originX, originY, pmap.width(), pmap.height());
457     if (!srcR.contains(dstR)) {
458         return false;
459     }
460 
461     // If they are requesting a subset, we have to have a temp allocation for full image, and
462     // then copy the subset into their allocation
463     SkBitmap full;
464     SkPixmap fullPM;
465     const SkPixmap* dstPM = &pmap;
466     if (srcR != dstR) {
467         if (!full.tryAllocPixels(pmap.info().makeWH(genW, genH))) {
468             return false;
469         }
470         if (!full.peekPixels(&fullPM)) {
471             return false;
472         }
473         dstPM = &fullPM;
474     }
475 
476     SkImageGenerator::Options opts;
477     opts.fBehavior = behavior;
478     if (!gen->getPixels(dstPM->info(), dstPM->writable_addr(), dstPM->rowBytes(), &opts)) {
479         return false;
480     }
481 
482     if (srcR != dstR) {
483         if (!full.readPixels(pmap, originX, originY)) {
484             return false;
485         }
486     }
487     return true;
488 }
489 
lockAsBitmap(SkBitmap * bitmap,SkImage::CachingHint chint,CachedFormat format,const SkImageInfo & info,SkTransferFunctionBehavior behavior) const490 bool SkImage_Lazy::lockAsBitmap(SkBitmap* bitmap, SkImage::CachingHint chint, CachedFormat format,
491                                 const SkImageInfo& info,
492                                 SkTransferFunctionBehavior behavior) const {
493     if (this->lockAsBitmapOnlyIfAlreadyCached(bitmap, format)) {
494         return true;
495     }
496 
497     uint32_t uniqueID = this->getUniqueID(format);
498 
499     SkBitmap tmpBitmap;
500     SkBitmapCache::RecPtr cacheRec;
501     SkPixmap pmap;
502     if (SkImage::kAllow_CachingHint == chint) {
503         auto desc = SkBitmapCacheDesc::Make(uniqueID, info.width(), info.height());
504         cacheRec = SkBitmapCache::Alloc(desc, info, &pmap);
505         if (!cacheRec) {
506             return false;
507         }
508     } else {
509         if (!tmpBitmap.tryAllocPixels(info)) {
510             return false;
511         }
512         if (!tmpBitmap.peekPixels(&pmap)) {
513             return false;
514         }
515     }
516 
517     ScopedGenerator generator(fSharedGenerator);
518     if (!generate_pixels(generator, pmap, fOrigin.x(), fOrigin.y(), behavior)) {
519         return false;
520     }
521 
522     if (cacheRec) {
523         SkBitmapCache::Add(std::move(cacheRec), bitmap);
524         SkASSERT(bitmap->getPixels());  // we're locked
525         SkASSERT(bitmap->isImmutable());
526         SkASSERT(bitmap->getGenerationID() == uniqueID);
527         this->notifyAddedToCache();
528     } else {
529         *bitmap = tmpBitmap;
530         bitmap->pixelRef()->setImmutableWithID(uniqueID);
531     }
532 
533     check_output_bitmap(*bitmap, uniqueID);
534     return true;
535 }
536 
537 //////////////////////////////////////////////////////////////////////////////////////////////////
538 
onReadPixels(const SkImageInfo & dstInfo,void * dstPixels,size_t dstRB,int srcX,int srcY,CachingHint chint) const539 bool SkImage_Lazy::onReadPixels(const SkImageInfo& dstInfo, void* dstPixels, size_t dstRB,
540                                 int srcX, int srcY, CachingHint chint) const {
541     SkColorSpace* dstColorSpace = dstInfo.colorSpace();
542     SkBitmap bm;
543     if (kDisallow_CachingHint == chint) {
544         CachedFormat cacheFormat = this->chooseCacheFormat(dstColorSpace);
545         SkImageInfo genPixelsInfo = dstInfo;
546         SkTransferFunctionBehavior behavior = getGeneratorBehaviorAndInfo(&genPixelsInfo);
547         if (this->lockAsBitmapOnlyIfAlreadyCached(&bm, cacheFormat)) {
548             return bm.readPixels(dstInfo, dstPixels, dstRB, srcX, srcY);
549         } else {
550             // Try passing the caller's buffer directly down to the generator. If this fails we
551             // may still succeed in the general case, as the generator may prefer some other
552             // config, which we could then convert via SkBitmap::readPixels.
553             if (this->directGeneratePixels(genPixelsInfo, dstPixels, dstRB, srcX, srcY, behavior)) {
554                 return true;
555             }
556             // else fall through
557         }
558     }
559 
560     if (this->getROPixels(&bm, dstColorSpace, chint)) {
561         return bm.readPixels(dstInfo, dstPixels, dstRB, srcX, srcY);
562     }
563     return false;
564 }
565 
onRefEncoded() const566 SkData* SkImage_Lazy::onRefEncoded() const {
567     ScopedGenerator generator(fSharedGenerator);
568     return generator->refEncodedData();
569 }
570 
getROPixels(SkBitmap * bitmap,SkColorSpace * dstColorSpace,CachingHint chint) const571 bool SkImage_Lazy::getROPixels(SkBitmap* bitmap, SkColorSpace* dstColorSpace,
572                                CachingHint chint) const {
573     CachedFormat cacheFormat = this->chooseCacheFormat(dstColorSpace);
574     const SkImageInfo cacheInfo = this->buildCacheInfo(cacheFormat);
575     SkImageInfo genPixelsInfo = cacheInfo;
576     SkTransferFunctionBehavior behavior = getGeneratorBehaviorAndInfo(&genPixelsInfo);
577     return this->lockAsBitmap(bitmap, chint, cacheFormat, genPixelsInfo, behavior);
578 }
579 
onIsValid(GrContext * context) const580 bool SkImage_Lazy::onIsValid(GrContext* context) const {
581     ScopedGenerator generator(fSharedGenerator);
582     return generator->isValid(context);
583 }
584 
onCanLazyGenerateOnGPU() const585 bool SkImage_Lazy::onCanLazyGenerateOnGPU() const {
586 #if SK_SUPPORT_GPU
587     ScopedGenerator generator(fSharedGenerator);
588     return SkImageGenerator::TexGenType::kNone != generator->onCanGenerateTexture();
589 #else
590     return false;
591 #endif
592 }
593 
getGeneratorBehaviorAndInfo(SkImageInfo * generatorImageInfo) const594 SkTransferFunctionBehavior SkImage_Lazy::getGeneratorBehaviorAndInfo(SkImageInfo* generatorImageInfo) const {
595     if (generatorImageInfo->colorSpace()) {
596         return SkTransferFunctionBehavior::kRespect;
597     }
598     // Only specify an output color space if color conversion can be done on the color type.
599     switch (generatorImageInfo->colorType()) {
600         case kRGBA_8888_SkColorType:
601         case kBGRA_8888_SkColorType:
602         case kRGBA_F16_SkColorType:
603         case kRGB_565_SkColorType:
604             *generatorImageInfo = generatorImageInfo->makeColorSpace(fInfo.refColorSpace());
605             break;
606         default:
607             break;
608     }
609     return SkTransferFunctionBehavior::kIgnore;
610 }
611 
612 ///////////////////////////////////////////////////////////////////////////////////////////////////
613 
614 #if SK_SUPPORT_GPU
asTextureProxyRef(GrContext * context,const GrSamplerParams & params,SkColorSpace * dstColorSpace,sk_sp<SkColorSpace> * texColorSpace,SkScalar scaleAdjust[2]) const615 sk_sp<GrTextureProxy> SkImage_Lazy::asTextureProxyRef(GrContext* context,
616                                                       const GrSamplerParams& params,
617                                                       SkColorSpace* dstColorSpace,
618                                                       sk_sp<SkColorSpace>* texColorSpace,
619                                                       SkScalar scaleAdjust[2]) const {
620     if (!context) {
621         return nullptr;
622     }
623 
624     GrImageTextureMaker textureMaker(context, this, kAllow_CachingHint);
625     return textureMaker.refTextureProxyForParams(params, dstColorSpace, texColorSpace, scaleAdjust);
626 }
627 #endif
628 
onMakeSubset(const SkIRect & subset) const629 sk_sp<SkImage> SkImage_Lazy::onMakeSubset(const SkIRect& subset) const {
630     SkASSERT(fInfo.bounds().contains(subset));
631     SkASSERT(fInfo.bounds() != subset);
632 
633     const SkIRect generatorSubset = subset.makeOffset(fOrigin.x(), fOrigin.y());
634     Validator validator(fSharedGenerator, &generatorSubset, fInfo.refColorSpace());
635     return validator ? sk_sp<SkImage>(new SkImage_Lazy(&validator)) : nullptr;
636 }
637 
onMakeColorSpace(sk_sp<SkColorSpace> target,SkColorType targetColorType,SkTransferFunctionBehavior premulBehavior) const638 sk_sp<SkImage> SkImage_Lazy::onMakeColorSpace(sk_sp<SkColorSpace> target,
639                                               SkColorType targetColorType,
640                                               SkTransferFunctionBehavior premulBehavior) const {
641     SkAutoExclusive autoAquire(fOnMakeColorSpaceMutex);
642     if (target && fOnMakeColorSpaceTarget &&
643         SkColorSpace::Equals(target.get(), fOnMakeColorSpaceTarget.get())) {
644         return fOnMakeColorSpaceResult;
645     }
646     const SkIRect generatorSubset =
647             SkIRect::MakeXYWH(fOrigin.x(), fOrigin.y(), fInfo.width(), fInfo.height());
648     Validator validator(fSharedGenerator, &generatorSubset, target);
649     sk_sp<SkImage> result = validator ? sk_sp<SkImage>(new SkImage_Lazy(&validator)) : nullptr;
650     if (result) {
651         fOnMakeColorSpaceTarget = target;
652         fOnMakeColorSpaceResult = result;
653     }
654     return result;
655 }
656 
MakeFromGenerator(std::unique_ptr<SkImageGenerator> generator,const SkIRect * subset)657 sk_sp<SkImage> SkImage::MakeFromGenerator(std::unique_ptr<SkImageGenerator> generator,
658                                           const SkIRect* subset) {
659     SkImage_Lazy::Validator validator(SharedGenerator::Make(std::move(generator)), subset, nullptr);
660 
661     return validator ? sk_make_sp<SkImage_Lazy>(&validator) : nullptr;
662 }
663 
664 //////////////////////////////////////////////////////////////////////////////////////////////////
665 
666 /**
667  *  Implementation of SkImageCacherator interface, as needed by GrImageTextureMaker
668  */
669 
670 #if SK_SUPPORT_GPU
671 
makeCacheKeyFromOrigKey(const GrUniqueKey & origKey,CachedFormat format,GrUniqueKey * cacheKey)672 void SkImage_Lazy::makeCacheKeyFromOrigKey(const GrUniqueKey& origKey, CachedFormat format,
673                                            GrUniqueKey* cacheKey) {
674     SkASSERT(!cacheKey->isValid());
675     if (origKey.isValid()) {
676         static const GrUniqueKey::Domain kDomain = GrUniqueKey::GenerateDomain();
677         GrUniqueKey::Builder builder(cacheKey, origKey, kDomain, 1);
678         builder[0] = format;
679     }
680 }
681 
682 class Generator_GrYUVProvider : public GrYUVProvider {
683     SkImageGenerator* fGen;
684 
685 public:
Generator_GrYUVProvider(SkImageGenerator * gen)686     Generator_GrYUVProvider(SkImageGenerator* gen) : fGen(gen) {}
687 
onGetID()688     uint32_t onGetID() override { return fGen->uniqueID(); }
onQueryYUV8(SkYUVSizeInfo * sizeInfo,SkYUVColorSpace * colorSpace) const689     bool onQueryYUV8(SkYUVSizeInfo* sizeInfo, SkYUVColorSpace* colorSpace) const override {
690         return fGen->queryYUV8(sizeInfo, colorSpace);
691     }
onGetYUV8Planes(const SkYUVSizeInfo & sizeInfo,void * planes[3])692     bool onGetYUV8Planes(const SkYUVSizeInfo& sizeInfo, void* planes[3]) override {
693         return fGen->getYUV8Planes(sizeInfo, planes);
694     }
695 };
696 
set_key_on_proxy(GrResourceProvider * resourceProvider,GrTextureProxy * proxy,const GrUniqueKey & key)697 static void set_key_on_proxy(GrResourceProvider* resourceProvider,
698                              GrTextureProxy* proxy, const GrUniqueKey& key) {
699     if (key.isValid()) {
700         resourceProvider->assignUniqueKeyToProxy(key, proxy);
701     }
702 }
703 
getColorSpace(GrContext * ctx,SkColorSpace * dstColorSpace)704 sk_sp<SkColorSpace> SkImage_Lazy::getColorSpace(GrContext* ctx, SkColorSpace* dstColorSpace) {
705     // TODO: This isn't always correct. Picture generator currently produces textures in N32,
706     // and will (soon) emit them in an arbitrary (destination) space. We will need to stash that
707     // information in/on the key so we can return the correct space in case #1 of lockTexture.
708     CachedFormat format = this->chooseCacheFormat(dstColorSpace, ctx->caps());
709     SkImageInfo cacheInfo = this->buildCacheInfo(format);
710     return sk_ref_sp(cacheInfo.colorSpace());
711 }
712 
713 /*
714  *  We have 4 ways to try to return a texture (in sorted order)
715  *
716  *  1. Check the cache for a pre-existing one
717  *  2. Ask the generator to natively create one
718  *  3. Ask the generator to return YUV planes, which the GPU can convert
719  *  4. Ask the generator to return RGB(A) data, which the GPU can convert
720  */
lockTextureProxy(GrContext * ctx,const GrUniqueKey & origKey,SkImage::CachingHint chint,bool willBeMipped,SkColorSpace * dstColorSpace,GrTextureMaker::AllowedTexGenType genType)721 sk_sp<GrTextureProxy> SkImage_Lazy::lockTextureProxy(GrContext* ctx,
722                                                      const GrUniqueKey& origKey,
723                                                      SkImage::CachingHint chint,
724                                                      bool willBeMipped,
725                                                      SkColorSpace* dstColorSpace,
726                                                      GrTextureMaker::AllowedTexGenType genType) {
727     // Values representing the various texture lock paths we can take. Used for logging the path
728     // taken to a histogram.
729     enum LockTexturePath {
730         kFailure_LockTexturePath,
731         kPreExisting_LockTexturePath,
732         kNative_LockTexturePath,
733         kCompressed_LockTexturePath, // Deprecated
734         kYUV_LockTexturePath,
735         kRGBA_LockTexturePath,
736     };
737 
738     enum { kLockTexturePathCount = kRGBA_LockTexturePath + 1 };
739 
740     // Determine which cached format we're going to use (which may involve decoding to a different
741     // info than the generator provides).
742     CachedFormat format = this->chooseCacheFormat(dstColorSpace, ctx->caps());
743 
744     // Fold the cache format into our texture key
745     GrUniqueKey key;
746     this->makeCacheKeyFromOrigKey(origKey, format, &key);
747 
748     // 1. Check the cache for a pre-existing one
749     if (key.isValid()) {
750         if (sk_sp<GrTextureProxy> proxy = ctx->resourceProvider()->findProxyByUniqueKey(key)) {
751             SK_HISTOGRAM_ENUMERATION("LockTexturePath", kPreExisting_LockTexturePath,
752                                      kLockTexturePathCount);
753             return proxy;
754         }
755     }
756 
757     // The CachedFormat is both an index for which cache "slot" we'll use to store this particular
758     // decoded variant of the encoded data, and also a recipe for how to transform the original
759     // info to get the one that we're going to decode to.
760     const SkImageInfo cacheInfo = this->buildCacheInfo(format);
761     SkImageInfo genPixelsInfo = cacheInfo;
762     SkTransferFunctionBehavior behavior = getGeneratorBehaviorAndInfo(&genPixelsInfo);
763 
764     // 2. Ask the generator to natively create one
765     {
766         ScopedGenerator generator(fSharedGenerator);
767         if (GrTextureMaker::AllowedTexGenType::kCheap == genType &&
768                 SkImageGenerator::TexGenType::kCheap != generator->onCanGenerateTexture()) {
769             return nullptr;
770         }
771         if (sk_sp<GrTextureProxy> proxy =
772                     generator->generateTexture(ctx, genPixelsInfo, fOrigin, behavior)) {
773             SK_HISTOGRAM_ENUMERATION("LockTexturePath", kNative_LockTexturePath,
774                                      kLockTexturePathCount);
775             set_key_on_proxy(ctx->resourceProvider(), proxy.get(), key);
776             return proxy;
777         }
778     }
779 
780     // 3. Ask the generator to return YUV planes, which the GPU can convert
781     if (!ctx->contextPriv().disableGpuYUVConversion()) {
782         const GrSurfaceDesc desc = GrImageInfoToSurfaceDesc(cacheInfo, *ctx->caps());
783         ScopedGenerator generator(fSharedGenerator);
784         Generator_GrYUVProvider provider(generator);
785 
786         // The pixels in the texture will be in the generator's color space. If onMakeColorSpace
787         // has been called then this will not match this image's color space. To correct this, apply
788         // a color space conversion from the generator's color space to this image's color space.
789         const SkColorSpace* generatorColorSpace =
790                 fSharedGenerator->fGenerator->getInfo().colorSpace();
791         const SkColorSpace* thisColorSpace = fInfo.colorSpace();
792 
793         sk_sp<GrTextureProxy> proxy =
794                 provider.refAsTextureProxy(ctx, desc, true, generatorColorSpace, thisColorSpace);
795         if (proxy) {
796             SK_HISTOGRAM_ENUMERATION("LockTexturePath", kYUV_LockTexturePath,
797                                      kLockTexturePathCount);
798             set_key_on_proxy(ctx->resourceProvider(), proxy.get(), key);
799             return proxy;
800         }
801     }
802 
803     // 4. Ask the generator to return RGB(A) data, which the GPU can convert
804     SkBitmap bitmap;
805     if (this->lockAsBitmap(&bitmap, chint, format, genPixelsInfo, behavior)) {
806         sk_sp<GrTextureProxy> proxy;
807         if (willBeMipped) {
808             proxy = GrGenerateMipMapsAndUploadToTextureProxy(ctx, bitmap, dstColorSpace);
809         }
810         if (!proxy) {
811             proxy = GrUploadBitmapToTextureProxy(ctx->resourceProvider(), bitmap, dstColorSpace);
812         }
813         if (proxy) {
814             SK_HISTOGRAM_ENUMERATION("LockTexturePath", kRGBA_LockTexturePath,
815                                      kLockTexturePathCount);
816             set_key_on_proxy(ctx->resourceProvider(), proxy.get(), key);
817             return proxy;
818         }
819     }
820     SK_HISTOGRAM_ENUMERATION("LockTexturePath", kFailure_LockTexturePath,
821                              kLockTexturePathCount);
822     return nullptr;
823 }
824 
825 ///////////////////////////////////////////////////////////////////////////////////////////////////
826 
827 #endif
828