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