blob: 5d978e79512cca9cb4c5839bc1104b693a4d0f58 [file] [log] [blame]
reed85d91782015-09-10 14:33:38 -07001/*
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"
reed85d91782015-09-10 14:33:38 -07009#include "SkImageCacherator.h"
Brian Osmandf7e0752017-04-26 16:20:28 -040010
11#include "SkBitmap.h"
12#include "SkBitmapCache.h"
Brian Osmandf7e0752017-04-26 16:20:28 -040013#include "SkData.h"
14#include "SkImageGenerator.h"
reed85d91782015-09-10 14:33:38 -070015#include "SkImagePriv.h"
Brian Osmandf7e0752017-04-26 16:20:28 -040016#include "SkNextID.h"
reed85d91782015-09-10 14:33:38 -070017#include "SkPixelRef.h"
reed85d91782015-09-10 14:33:38 -070018
Brian Osmandf7e0752017-04-26 16:20:28 -040019#if SK_SUPPORT_GPU
20#include "GrContext.h"
21#include "GrContextPriv.h"
22#include "GrGpuResourcePriv.h"
23#include "GrImageTextureMaker.h"
24#include "GrResourceKey.h"
Robert Phillips1afd4cd2018-01-08 13:40:32 -050025#include "GrProxyProvider.h"
Brian Salomon2bbdcc42017-09-07 12:36:34 -040026#include "GrSamplerState.h"
Brian Osmandf7e0752017-04-26 16:20:28 -040027#include "GrYUVProvider.h"
28#include "SkGr.h"
29#endif
reed85d91782015-09-10 14:33:38 -070030
Brian Osmandf7e0752017-04-26 16:20:28 -040031// Ref-counted tuple(SkImageGenerator, SkMutex) which allows sharing one generator among N images
32class SharedGenerator final : public SkNVRefCnt<SharedGenerator> {
33public:
34 static sk_sp<SharedGenerator> Make(std::unique_ptr<SkImageGenerator> gen) {
35 return gen ? sk_sp<SharedGenerator>(new SharedGenerator(std::move(gen))) : nullptr;
36 }
37
Matt Sarettb2004f72017-05-18 09:26:50 -040038 // This is thread safe. It is a const field set in the constructor.
39 const SkImageInfo& getInfo() { return fGenerator->getInfo(); }
40
Brian Osmandf7e0752017-04-26 16:20:28 -040041private:
42 explicit SharedGenerator(std::unique_ptr<SkImageGenerator> gen)
43 : fGenerator(std::move(gen)) {
44 SkASSERT(fGenerator);
45 }
46
47 friend class ScopedGenerator;
48 friend class SkImage_Lazy;
49
50 std::unique_ptr<SkImageGenerator> fGenerator;
51 SkMutex fMutex;
52};
53
54class SkImage_Lazy : public SkImage_Base, public SkImageCacherator {
55public:
56 struct Validator {
Christopher Cameron77e96662017-07-08 01:47:47 -070057 Validator(sk_sp<SharedGenerator>, const SkIRect* subset, sk_sp<SkColorSpace> colorSpace);
Brian Osmandf7e0752017-04-26 16:20:28 -040058
59 operator bool() const { return fSharedGenerator.get(); }
60
61 sk_sp<SharedGenerator> fSharedGenerator;
62 SkImageInfo fInfo;
63 SkIPoint fOrigin;
Christopher Cameron77e96662017-07-08 01:47:47 -070064 sk_sp<SkColorSpace> fColorSpace;
Brian Osmandf7e0752017-04-26 16:20:28 -040065 uint32_t fUniqueID;
66 };
67
68 SkImage_Lazy(Validator* validator);
69
70 SkImageInfo onImageInfo() const override {
71 return fInfo;
herba7c9d632016-04-19 12:30:22 -070072 }
Greg Daniel56008aa2018-03-14 15:33:42 -040073 SkColorType onColorType() const override {
74 return kUnknown_SkColorType;
75 }
brianosman69c166d2016-08-17 14:01:05 -070076 SkAlphaType onAlphaType() const override {
Brian Osmandf7e0752017-04-26 16:20:28 -040077 return fInfo.alphaType();
brianosman69c166d2016-08-17 14:01:05 -070078 }
herba7c9d632016-04-19 12:30:22 -070079
Mike Reedf2c73642018-05-29 15:41:27 -040080 SkIRect onGetSubset() const override {
81 return SkIRect::MakeXYWH(fOrigin.fX, fOrigin.fY, fInfo.width(), fInfo.height());
82 }
83
Robert Phillipsb726d582017-03-09 16:36:32 -050084 bool onReadPixels(const SkImageInfo&, void*, size_t, int srcX, int srcY,
85 CachingHint) const override;
86#if SK_SUPPORT_GPU
Brian Salomon2bbdcc42017-09-07 12:36:34 -040087 sk_sp<GrTextureProxy> asTextureProxyRef(GrContext*,
88 const GrSamplerState&, SkColorSpace*,
89 sk_sp<SkColorSpace>*,
Robert Phillipsb726d582017-03-09 16:36:32 -050090 SkScalar scaleAdjust[2]) const override;
91#endif
Ben Wagnerbdf54332018-05-15 14:12:14 -040092 sk_sp<SkData> onRefEncoded() const override;
reed7fb4f8b2016-03-11 04:33:52 -080093 sk_sp<SkImage> onMakeSubset(const SkIRect&) const override;
Brian Osman61624f02016-12-09 14:51:59 -050094 bool getROPixels(SkBitmap*, SkColorSpace* dstColorSpace, CachingHint) const override;
reed85d91782015-09-10 14:33:38 -070095 bool onIsLazyGenerated() const override { return true; }
Mike Reed7f1d0202017-05-08 16:13:39 -040096 bool onCanLazyGenerateOnGPU() const override;
Matt Sarett9f3dcb32017-05-04 08:53:32 -040097 sk_sp<SkImage> onMakeColorSpace(sk_sp<SkColorSpace>, SkColorType,
98 SkTransferFunctionBehavior) const override;
reed85d91782015-09-10 14:33:38 -070099
Brian Osman5bbd0762017-05-08 11:07:42 -0400100 bool onIsValid(GrContext*) const override;
101
Brian Osmandf7e0752017-04-26 16:20:28 -0400102 SkImageCacherator* peekCacherator() const override {
103 return const_cast<SkImage_Lazy*>(this);
104 }
105
106 // Only return true if the generate has already been cached.
107 bool lockAsBitmapOnlyIfAlreadyCached(SkBitmap*, CachedFormat) const;
108 // Call the underlying generator directly
109 bool directGeneratePixels(const SkImageInfo& dstInfo, void* dstPixels, size_t dstRB,
110 int srcX, int srcY, SkTransferFunctionBehavior behavior) const;
111
112 // SkImageCacherator interface
113#if SK_SUPPORT_GPU
114 // Returns the texture proxy. If the cacherator is generating the texture and wants to cache it,
115 // it should use the passed in key (if the key is valid).
116 sk_sp<GrTextureProxy> lockTextureProxy(GrContext*,
117 const GrUniqueKey& key,
118 SkImage::CachingHint,
119 bool willBeMipped,
Stan Ilievba81af22017-06-08 15:16:53 -0400120 SkColorSpace* dstColorSpace,
121 GrTextureMaker::AllowedTexGenType genType) override;
Brian Osmandf7e0752017-04-26 16:20:28 -0400122
123 // Returns the color space of the texture that would be returned if you called lockTexture.
124 // Separate code path to allow querying of the color space for textures that cached (even
125 // externally).
126 sk_sp<SkColorSpace> getColorSpace(GrContext*, SkColorSpace* dstColorSpace) override;
127 void makeCacheKeyFromOrigKey(const GrUniqueKey& origKey, CachedFormat,
128 GrUniqueKey* cacheKey) override;
129#endif
130
131 CachedFormat chooseCacheFormat(SkColorSpace* dstColorSpace,
132 const GrCaps* = nullptr) const override;
133 SkImageInfo buildCacheInfo(CachedFormat) const override;
134
reed85d91782015-09-10 14:33:38 -0700135private:
Brian Osmandf7e0752017-04-26 16:20:28 -0400136 class ScopedGenerator;
137
138 /**
139 * On success (true), bitmap will point to the pixels for this generator. If this returns
140 * false, the bitmap will be reset to empty.
141 */
Christopher Cameron77e96662017-07-08 01:47:47 -0700142 bool lockAsBitmap(SkBitmap*, SkImage::CachingHint, CachedFormat, const SkImageInfo&,
143 SkTransferFunctionBehavior) const;
144
145 /**
146 * Populates parameters to pass to the generator for reading pixels or generating a texture.
147 * For image generators, legacy versus true color blending is indicated using a
148 * SkTransferFunctionBehavior, and the target color space is specified on the SkImageInfo.
149 * If generatorImageInfo has no color space set, set its color space to this SkImage's color
150 * space, and return "ignore" behavior, indicating legacy mode. If generatorImageInfo has a
151 * color space set, return "respect" behavior, indicating linear blending mode.
152 */
153 SkTransferFunctionBehavior getGeneratorBehaviorAndInfo(SkImageInfo* generatorImageInfo) const;
Brian Osmandf7e0752017-04-26 16:20:28 -0400154
155 sk_sp<SharedGenerator> fSharedGenerator;
Christopher Cameron77e96662017-07-08 01:47:47 -0700156 // Note that fInfo is not necessarily the info from the generator. It may be cropped by
157 // onMakeSubset and its color space may be changed by onMakeColorSpace.
Brian Osmandf7e0752017-04-26 16:20:28 -0400158 const SkImageInfo fInfo;
159 const SkIPoint fOrigin;
160
161 struct IDRec {
162 SkOnce fOnce;
163 uint32_t fUniqueID;
164 };
165 mutable IDRec fIDRecs[kNumCachedFormats];
166
167 uint32_t getUniqueID(CachedFormat) const;
reed85d91782015-09-10 14:33:38 -0700168
Christopher Camerond4b67872017-07-13 15:18:08 -0700169 // Repeated calls to onMakeColorSpace will result in a proliferation of unique IDs and
170 // SkImage_Lazy instances. Cache the result of the last successful onMakeColorSpace call.
171 mutable SkMutex fOnMakeColorSpaceMutex;
172 mutable sk_sp<SkColorSpace> fOnMakeColorSpaceTarget;
173 mutable sk_sp<SkImage> fOnMakeColorSpaceResult;
174
reed85d91782015-09-10 14:33:38 -0700175 typedef SkImage_Base INHERITED;
176};
177
178///////////////////////////////////////////////////////////////////////////////
179
Christopher Cameron77e96662017-07-08 01:47:47 -0700180SkImage_Lazy::Validator::Validator(sk_sp<SharedGenerator> gen, const SkIRect* subset,
181 sk_sp<SkColorSpace> colorSpace)
Brian Osmandf7e0752017-04-26 16:20:28 -0400182 : fSharedGenerator(std::move(gen)) {
Brian Osmandf7e0752017-04-26 16:20:28 -0400183 if (!fSharedGenerator) {
184 return;
185 }
186
187 // The following generator accessors are safe without acquiring the mutex (const getters).
188 // TODO: refactor to use a ScopedGenerator instead, for clarity.
189 const SkImageInfo& info = fSharedGenerator->fGenerator->getInfo();
190 if (info.isEmpty()) {
191 fSharedGenerator.reset();
192 return;
193 }
194
195 fUniqueID = fSharedGenerator->fGenerator->uniqueID();
196 const SkIRect bounds = SkIRect::MakeWH(info.width(), info.height());
197 if (subset) {
198 if (!bounds.contains(*subset)) {
199 fSharedGenerator.reset();
200 return;
201 }
202 if (*subset != bounds) {
203 // we need a different uniqueID since we really are a subset of the raw generator
204 fUniqueID = SkNextID::ImageID();
205 }
206 } else {
207 subset = &bounds;
208 }
209
210 fInfo = info.makeWH(subset->width(), subset->height());
211 fOrigin = SkIPoint::Make(subset->x(), subset->y());
Christopher Cameron77e96662017-07-08 01:47:47 -0700212 if (colorSpace) {
213 fInfo = fInfo.makeColorSpace(colorSpace);
214 fUniqueID = SkNextID::ImageID();
215 }
Brian Osmandf7e0752017-04-26 16:20:28 -0400216}
217
218///////////////////////////////////////////////////////////////////////////////
219
220// Helper for exclusive access to a shared generator.
221class SkImage_Lazy::ScopedGenerator {
222public:
223 ScopedGenerator(const sk_sp<SharedGenerator>& gen)
224 : fSharedGenerator(gen)
225 , fAutoAquire(gen->fMutex) {}
226
227 SkImageGenerator* operator->() const {
228 fSharedGenerator->fMutex.assertHeld();
229 return fSharedGenerator->fGenerator.get();
230 }
231
232 operator SkImageGenerator*() const {
233 fSharedGenerator->fMutex.assertHeld();
234 return fSharedGenerator->fGenerator.get();
235 }
236
237private:
238 const sk_sp<SharedGenerator>& fSharedGenerator;
239 SkAutoExclusive fAutoAquire;
240};
241
242///////////////////////////////////////////////////////////////////////////////
243
244SkImage_Lazy::SkImage_Lazy(Validator* validator)
245 : INHERITED(validator->fInfo.width(), validator->fInfo.height(), validator->fUniqueID)
246 , fSharedGenerator(std::move(validator->fSharedGenerator))
247 , fInfo(validator->fInfo)
248 , fOrigin(validator->fOrigin) {
249 SkASSERT(fSharedGenerator);
Brian Osmandf7e0752017-04-26 16:20:28 -0400250 // We explicit set the legacy format slot, but leave the others uninitialized (via SkOnce)
251 // and only resolove them to IDs as needed (by calling getUniqueID()).
252 fIDRecs[kLegacy_CachedFormat].fOnce([this, validator] {
253 fIDRecs[kLegacy_CachedFormat].fUniqueID = validator->fUniqueID;
254 });
255}
256
257uint32_t SkImage_Lazy::getUniqueID(CachedFormat format) const {
258 IDRec* rec = &fIDRecs[format];
259 rec->fOnce([rec] {
260 rec->fUniqueID = SkNextID::ImageID();
261 });
262 return rec->fUniqueID;
263}
264
265//////////////////////////////////////////////////////////////////////////////////////////////////
266
267// Abstraction of GrCaps that handles the cases where we don't have a caps pointer (because
268// we're in raster mode), or where GPU support is entirely missing. In theory, we only need the
269// chosen format to be texturable, but that lets us choose F16 on GLES implemenations where we
270// won't be able to read the texture back. We'd like to ensure that SkImake::makeNonTextureImage
271// works, so we require that the formats we choose are renderable (as a proxy for being readable).
272struct CacheCaps {
273 CacheCaps(const GrCaps* caps) : fCaps(caps) {}
274
275#if SK_SUPPORT_GPU
276 bool supportsHalfFloat() const {
Brian Salomonbdecacf2018-02-02 20:32:49 -0500277 return !fCaps || (fCaps->isConfigTexturable(kRGBA_half_GrPixelConfig) &&
278 fCaps->isConfigRenderable(kRGBA_half_GrPixelConfig));
Brian Osmandf7e0752017-04-26 16:20:28 -0400279 }
280
281 bool supportsSRGB() const {
282 return !fCaps ||
283 (fCaps->srgbSupport() && fCaps->isConfigTexturable(kSRGBA_8888_GrPixelConfig));
284 }
285
286 bool supportsSBGR() const {
287 return !fCaps || fCaps->srgbSupport();
288 }
289#else
290 bool supportsHalfFloat() const { return true; }
291 bool supportsSRGB() const { return true; }
292 bool supportsSBGR() const { return true; }
293#endif
294
295 const GrCaps* fCaps;
296};
297
298SkImageCacherator::CachedFormat SkImage_Lazy::chooseCacheFormat(SkColorSpace* dstColorSpace,
299 const GrCaps* grCaps) const {
Brian Osmanbfc33e52018-06-27 14:24:11 -0400300#ifdef SK_SUPPORT_LEGACY_LAZY_IMAGE_DECODE_BEHAVIOR
Brian Osmane2693102018-06-26 19:19:49 +0000301 SkColorSpace* cs = fInfo.colorSpace();
302 if (!cs || !dstColorSpace) {
303 return kLegacy_CachedFormat;
304 }
305
306 CacheCaps caps(grCaps);
307 switch (fInfo.colorType()) {
308 case kUnknown_SkColorType:
309 case kAlpha_8_SkColorType:
310 case kRGB_565_SkColorType:
311 case kARGB_4444_SkColorType:
312 case kRGB_888x_SkColorType:
313 case kRGBA_1010102_SkColorType:
314 case kRGB_101010x_SkColorType:
315 // We don't support color space on these formats, so always decode in legacy mode:
316 // TODO: Ask the codec to decode these to something else (at least sRGB 8888)?
317 return kLegacy_CachedFormat;
318
319 case kGray_8_SkColorType:
320 // TODO: What do we do with grayscale sources that have strange color spaces attached?
321 // The codecs and color space xform don't handle this correctly (yet), so drop it on
322 // the floor. (Also, inflating by a factor of 8 is going to be unfortunate).
323 // As it is, we don't directly support sRGB grayscale, so ask the codec to convert
324 // it for us. This bypasses some really sketchy code GrUploadPixmapToTexture.
325 if (cs->gammaCloseToSRGB() && caps.supportsSRGB()) {
326 return kSRGB8888_CachedFormat;
327 } else {
328 return kLegacy_CachedFormat;
329 }
330
331 case kRGBA_8888_SkColorType:
332 if (cs->gammaCloseToSRGB()) {
333 if (caps.supportsSRGB()) {
334 return kSRGB8888_CachedFormat;
335 } else if (caps.supportsHalfFloat()) {
336 return kLinearF16_CachedFormat;
337 } else {
338 return kLegacy_CachedFormat;
339 }
340 } else {
341 if (caps.supportsHalfFloat()) {
342 return kLinearF16_CachedFormat;
343 } else if (caps.supportsSRGB()) {
344 return kSRGB8888_CachedFormat;
345 } else {
346 return kLegacy_CachedFormat;
347 }
348 }
349
350 case kBGRA_8888_SkColorType:
351 // Odd case. sBGRA isn't a real thing, so we may not have this texturable.
352 if (caps.supportsSBGR()) {
353 if (cs->gammaCloseToSRGB()) {
354 return kSBGR8888_CachedFormat;
355 } else if (caps.supportsHalfFloat()) {
356 return kLinearF16_CachedFormat;
357 } else if (caps.supportsSRGB()) {
358 return kSRGB8888_CachedFormat;
359 } else {
360 // sBGRA support without sRGBA is highly unlikely (impossible?) Nevertheless.
361 return kLegacy_CachedFormat;
362 }
363 } else {
364 if (cs->gammaCloseToSRGB()) {
365 if (caps.supportsSRGB()) {
366 return kSRGB8888_CachedFormat;
367 } else if (caps.supportsHalfFloat()) {
368 return kLinearF16_CachedFormat;
369 } else {
370 return kLegacy_CachedFormat;
371 }
372 } else {
373 if (caps.supportsHalfFloat()) {
374 return kLinearF16_CachedFormat;
375 } else if (caps.supportsSRGB()) {
376 return kSRGB8888_CachedFormat;
377 } else {
378 return kLegacy_CachedFormat;
379 }
380 }
381 }
382
383 case kRGBA_F16_SkColorType:
Brian Osmand47fe092018-06-26 15:27:10 -0400384 case kRGBA_F32_SkColorType:
Brian Osmane2693102018-06-26 19:19:49 +0000385 if (caps.supportsHalfFloat()) {
386 return kLinearF16_CachedFormat;
387 } else if (caps.supportsSRGB()) {
388 return kSRGB8888_CachedFormat;
389 } else {
390 return kLegacy_CachedFormat;
391 }
392 }
393 SkDEBUGFAIL("Unreachable");
Brian Osmanbfc33e52018-06-27 14:24:11 -0400394#endif
Brian Osmandf7e0752017-04-26 16:20:28 -0400395 return kLegacy_CachedFormat;
396}
397
398SkImageInfo SkImage_Lazy::buildCacheInfo(CachedFormat format) const {
Brian Osmanbfc33e52018-06-27 14:24:11 -0400399#ifdef SK_SUPPORT_LEGACY_LAZY_IMAGE_DECODE_BEHAVIOR
Brian Osmane2693102018-06-26 19:19:49 +0000400 switch (format) {
401 case kLegacy_CachedFormat:
402 return fInfo.makeColorSpace(nullptr);
403 case kLinearF16_CachedFormat:
404 return fInfo.makeColorType(kRGBA_F16_SkColorType)
405 .makeColorSpace(fInfo.colorSpace()->makeLinearGamma());
406 case kSRGB8888_CachedFormat:
407 // If the transfer function is nearly (but not exactly) sRGB, we don't want the codec
408 // to bother trans-coding. It would be slow, and do more harm than good visually,
409 // so we make sure to leave the colorspace as-is.
410 if (fInfo.colorSpace()->gammaCloseToSRGB()) {
411 return fInfo.makeColorType(kRGBA_8888_SkColorType);
412 } else {
413 return fInfo.makeColorType(kRGBA_8888_SkColorType)
414 .makeColorSpace(fInfo.colorSpace()->makeSRGBGamma());
415 }
416 case kSBGR8888_CachedFormat:
417 // See note above about not-quite-sRGB transfer functions.
418 if (fInfo.colorSpace()->gammaCloseToSRGB()) {
419 return fInfo.makeColorType(kBGRA_8888_SkColorType);
420 } else {
421 return fInfo.makeColorType(kBGRA_8888_SkColorType)
422 .makeColorSpace(fInfo.colorSpace()->makeSRGBGamma());
423 }
424 default:
425 SkDEBUGFAIL("Invalid cached format");
426 return fInfo;
Brian Osmandf7e0752017-04-26 16:20:28 -0400427 }
Brian Osmanbfc33e52018-06-27 14:24:11 -0400428#else
429 if (kGray_8_SkColorType == fInfo.colorType()) {
430 return fInfo.makeColorSpace(nullptr);
431 } else {
432 return fInfo;
433 }
434#endif
Brian Osmandf7e0752017-04-26 16:20:28 -0400435}
436
437//////////////////////////////////////////////////////////////////////////////////////////////////
438
439static bool check_output_bitmap(const SkBitmap& bitmap, uint32_t expectedID) {
440 SkASSERT(bitmap.getGenerationID() == expectedID);
441 SkASSERT(bitmap.isImmutable());
442 SkASSERT(bitmap.getPixels());
443 return true;
444}
445
446bool SkImage_Lazy::directGeneratePixels(const SkImageInfo& info, void* pixels, size_t rb,
447 int srcX, int srcY,
448 SkTransferFunctionBehavior behavior) const {
449 ScopedGenerator generator(fSharedGenerator);
450 const SkImageInfo& genInfo = generator->getInfo();
451 // Currently generators do not natively handle subsets, so check that first.
452 if (srcX || srcY || genInfo.width() != info.width() || genInfo.height() != info.height()) {
453 return false;
454 }
455
456 SkImageGenerator::Options opts;
Christopher Cameron77e96662017-07-08 01:47:47 -0700457 // TODO: This should respect the behavior argument.
458 opts.fBehavior = SkTransferFunctionBehavior::kIgnore;
Brian Osmandf7e0752017-04-26 16:20:28 -0400459 return generator->getPixels(info, pixels, rb, &opts);
460}
461
462//////////////////////////////////////////////////////////////////////////////////////////////////
463
464bool SkImage_Lazy::lockAsBitmapOnlyIfAlreadyCached(SkBitmap* bitmap, CachedFormat format) const {
465 uint32_t uniqueID = this->getUniqueID(format);
466 return SkBitmapCache::Find(SkBitmapCacheDesc::Make(uniqueID,
467 fInfo.width(), fInfo.height()), bitmap) &&
468 check_output_bitmap(*bitmap, uniqueID);
469}
470
Christopher Cameron77e96662017-07-08 01:47:47 -0700471static bool generate_pixels(SkImageGenerator* gen, const SkPixmap& pmap, int originX, int originY,
472 SkTransferFunctionBehavior behavior) {
Brian Osmandf7e0752017-04-26 16:20:28 -0400473 const int genW = gen->getInfo().width();
474 const int genH = gen->getInfo().height();
475 const SkIRect srcR = SkIRect::MakeWH(genW, genH);
476 const SkIRect dstR = SkIRect::MakeXYWH(originX, originY, pmap.width(), pmap.height());
477 if (!srcR.contains(dstR)) {
478 return false;
479 }
480
481 // If they are requesting a subset, we have to have a temp allocation for full image, and
482 // then copy the subset into their allocation
483 SkBitmap full;
484 SkPixmap fullPM;
485 const SkPixmap* dstPM = &pmap;
486 if (srcR != dstR) {
487 if (!full.tryAllocPixels(pmap.info().makeWH(genW, genH))) {
488 return false;
489 }
490 if (!full.peekPixels(&fullPM)) {
491 return false;
492 }
493 dstPM = &fullPM;
494 }
495
Christopher Cameron77e96662017-07-08 01:47:47 -0700496 SkImageGenerator::Options opts;
497 opts.fBehavior = behavior;
498 if (!gen->getPixels(dstPM->info(), dstPM->writable_addr(), dstPM->rowBytes(), &opts)) {
Brian Osmandf7e0752017-04-26 16:20:28 -0400499 return false;
500 }
501
502 if (srcR != dstR) {
503 if (!full.readPixels(pmap, originX, originY)) {
504 return false;
505 }
506 }
507 return true;
508}
509
Christopher Cameron77e96662017-07-08 01:47:47 -0700510bool SkImage_Lazy::lockAsBitmap(SkBitmap* bitmap, SkImage::CachingHint chint, CachedFormat format,
511 const SkImageInfo& info,
512 SkTransferFunctionBehavior behavior) const {
Brian Osmandf7e0752017-04-26 16:20:28 -0400513 if (this->lockAsBitmapOnlyIfAlreadyCached(bitmap, format)) {
514 return true;
515 }
516
517 uint32_t uniqueID = this->getUniqueID(format);
518
519 SkBitmap tmpBitmap;
520 SkBitmapCache::RecPtr cacheRec;
521 SkPixmap pmap;
522 if (SkImage::kAllow_CachingHint == chint) {
523 auto desc = SkBitmapCacheDesc::Make(uniqueID, info.width(), info.height());
524 cacheRec = SkBitmapCache::Alloc(desc, info, &pmap);
525 if (!cacheRec) {
526 return false;
527 }
528 } else {
529 if (!tmpBitmap.tryAllocPixels(info)) {
530 return false;
531 }
532 if (!tmpBitmap.peekPixels(&pmap)) {
533 return false;
534 }
535 }
536
537 ScopedGenerator generator(fSharedGenerator);
Christopher Cameron77e96662017-07-08 01:47:47 -0700538 if (!generate_pixels(generator, pmap, fOrigin.x(), fOrigin.y(), behavior)) {
Brian Osmandf7e0752017-04-26 16:20:28 -0400539 return false;
540 }
541
542 if (cacheRec) {
543 SkBitmapCache::Add(std::move(cacheRec), bitmap);
544 SkASSERT(bitmap->getPixels()); // we're locked
545 SkASSERT(bitmap->isImmutable());
546 SkASSERT(bitmap->getGenerationID() == uniqueID);
547 this->notifyAddedToCache();
548 } else {
549 *bitmap = tmpBitmap;
550 bitmap->pixelRef()->setImmutableWithID(uniqueID);
551 }
552
553 check_output_bitmap(*bitmap, uniqueID);
554 return true;
555}
556
557//////////////////////////////////////////////////////////////////////////////////////////////////
558
Brian Osmanf1b43822017-04-20 13:43:23 -0400559bool SkImage_Lazy::onReadPixels(const SkImageInfo& dstInfo, void* dstPixels, size_t dstRB,
560 int srcX, int srcY, CachingHint chint) const {
Brian Osman61624f02016-12-09 14:51:59 -0500561 SkColorSpace* dstColorSpace = dstInfo.colorSpace();
reed85d91782015-09-10 14:33:38 -0700562 SkBitmap bm;
reed6868c3f2015-11-24 11:44:47 -0800563 if (kDisallow_CachingHint == chint) {
Brian Osmandf7e0752017-04-26 16:20:28 -0400564 CachedFormat cacheFormat = this->chooseCacheFormat(dstColorSpace);
Christopher Cameron77e96662017-07-08 01:47:47 -0700565 SkImageInfo genPixelsInfo = dstInfo;
566 SkTransferFunctionBehavior behavior = getGeneratorBehaviorAndInfo(&genPixelsInfo);
Brian Osmandf7e0752017-04-26 16:20:28 -0400567 if (this->lockAsBitmapOnlyIfAlreadyCached(&bm, cacheFormat)) {
reed6868c3f2015-11-24 11:44:47 -0800568 return bm.readPixels(dstInfo, dstPixels, dstRB, srcX, srcY);
569 } else {
570 // Try passing the caller's buffer directly down to the generator. If this fails we
571 // may still succeed in the general case, as the generator may prefer some other
572 // config, which we could then convert via SkBitmap::readPixels.
Christopher Cameron77e96662017-07-08 01:47:47 -0700573 if (this->directGeneratePixels(genPixelsInfo, dstPixels, dstRB, srcX, srcY, behavior)) {
reed6868c3f2015-11-24 11:44:47 -0800574 return true;
575 }
576 // else fall through
577 }
578 }
579
Brian Osman61624f02016-12-09 14:51:59 -0500580 if (this->getROPixels(&bm, dstColorSpace, chint)) {
reed85d91782015-09-10 14:33:38 -0700581 return bm.readPixels(dstInfo, dstPixels, dstRB, srcX, srcY);
582 }
583 return false;
584}
585
Ben Wagnerbdf54332018-05-15 14:12:14 -0400586sk_sp<SkData> SkImage_Lazy::onRefEncoded() const {
587 ScopedGenerator generator(fSharedGenerator);
588 return generator->refEncodedData();
589}
reed85d91782015-09-10 14:33:38 -0700590
Brian Osmanf1b43822017-04-20 13:43:23 -0400591bool SkImage_Lazy::getROPixels(SkBitmap* bitmap, SkColorSpace* dstColorSpace,
592 CachingHint chint) const {
Brian Osmandf7e0752017-04-26 16:20:28 -0400593 CachedFormat cacheFormat = this->chooseCacheFormat(dstColorSpace);
Christopher Cameron77e96662017-07-08 01:47:47 -0700594 const SkImageInfo cacheInfo = this->buildCacheInfo(cacheFormat);
595 SkImageInfo genPixelsInfo = cacheInfo;
596 SkTransferFunctionBehavior behavior = getGeneratorBehaviorAndInfo(&genPixelsInfo);
597 return this->lockAsBitmap(bitmap, chint, cacheFormat, genPixelsInfo, behavior);
reed85d91782015-09-10 14:33:38 -0700598}
599
Brian Osman5bbd0762017-05-08 11:07:42 -0400600bool SkImage_Lazy::onIsValid(GrContext* context) const {
601 ScopedGenerator generator(fSharedGenerator);
602 return generator->isValid(context);
603}
604
Mike Reed7f1d0202017-05-08 16:13:39 -0400605bool SkImage_Lazy::onCanLazyGenerateOnGPU() const {
606#if SK_SUPPORT_GPU
607 ScopedGenerator generator(fSharedGenerator);
Stan Ilievba81af22017-06-08 15:16:53 -0400608 return SkImageGenerator::TexGenType::kNone != generator->onCanGenerateTexture();
Mike Reed7f1d0202017-05-08 16:13:39 -0400609#else
610 return false;
611#endif
612}
613
Christopher Cameron77e96662017-07-08 01:47:47 -0700614SkTransferFunctionBehavior SkImage_Lazy::getGeneratorBehaviorAndInfo(SkImageInfo* generatorImageInfo) const {
Brian Osmanbfc33e52018-06-27 14:24:11 -0400615#ifdef SK_SUPPORT_LEGACY_LAZY_IMAGE_DECODE_BEHAVIOR
Brian Osmane2693102018-06-26 19:19:49 +0000616 if (generatorImageInfo->colorSpace()) {
617 return SkTransferFunctionBehavior::kRespect;
618 }
619 // Only specify an output color space if color conversion can be done on the color type.
620 switch (generatorImageInfo->colorType()) {
621 case kRGBA_8888_SkColorType:
622 case kBGRA_8888_SkColorType:
623 case kRGBA_F16_SkColorType:
624 case kRGB_565_SkColorType:
625 *generatorImageInfo = generatorImageInfo->makeColorSpace(fInfo.refColorSpace());
626 break;
627 default:
628 break;
629 }
Brian Osmanbfc33e52018-06-27 14:24:11 -0400630#endif
Christopher Cameron77e96662017-07-08 01:47:47 -0700631 return SkTransferFunctionBehavior::kIgnore;
632}
633
Brian Osmandf7e0752017-04-26 16:20:28 -0400634///////////////////////////////////////////////////////////////////////////////////////////////////
635
Robert Phillipsb726d582017-03-09 16:36:32 -0500636#if SK_SUPPORT_GPU
Brian Osmanf1b43822017-04-20 13:43:23 -0400637sk_sp<GrTextureProxy> SkImage_Lazy::asTextureProxyRef(GrContext* context,
Brian Salomon2bbdcc42017-09-07 12:36:34 -0400638 const GrSamplerState& params,
Brian Osmanf1b43822017-04-20 13:43:23 -0400639 SkColorSpace* dstColorSpace,
640 sk_sp<SkColorSpace>* texColorSpace,
641 SkScalar scaleAdjust[2]) const {
Brian Osmandf7e0752017-04-26 16:20:28 -0400642 if (!context) {
643 return nullptr;
644 }
645
646 GrImageTextureMaker textureMaker(context, this, kAllow_CachingHint);
647 return textureMaker.refTextureProxyForParams(params, dstColorSpace, texColorSpace, scaleAdjust);
Robert Phillipsb726d582017-03-09 16:36:32 -0500648}
649#endif
650
Brian Osmanf1b43822017-04-20 13:43:23 -0400651sk_sp<SkImage> SkImage_Lazy::onMakeSubset(const SkIRect& subset) const {
Brian Osmandf7e0752017-04-26 16:20:28 -0400652 SkASSERT(fInfo.bounds().contains(subset));
653 SkASSERT(fInfo.bounds() != subset);
reed7b6945b2015-09-24 00:50:58 -0700654
Brian Osmandf7e0752017-04-26 16:20:28 -0400655 const SkIRect generatorSubset = subset.makeOffset(fOrigin.x(), fOrigin.y());
Christopher Cameron77e96662017-07-08 01:47:47 -0700656 Validator validator(fSharedGenerator, &generatorSubset, fInfo.refColorSpace());
Brian Osmanf1b43822017-04-20 13:43:23 -0400657 return validator ? sk_sp<SkImage>(new SkImage_Lazy(&validator)) : nullptr;
reed7b6945b2015-09-24 00:50:58 -0700658}
659
Matt Sarett9f3dcb32017-05-04 08:53:32 -0400660sk_sp<SkImage> SkImage_Lazy::onMakeColorSpace(sk_sp<SkColorSpace> target,
661 SkColorType targetColorType,
662 SkTransferFunctionBehavior premulBehavior) const {
Christopher Camerond4b67872017-07-13 15:18:08 -0700663 SkAutoExclusive autoAquire(fOnMakeColorSpaceMutex);
664 if (target && fOnMakeColorSpaceTarget &&
665 SkColorSpace::Equals(target.get(), fOnMakeColorSpaceTarget.get())) {
666 return fOnMakeColorSpaceResult;
667 }
Christopher Cameron77e96662017-07-08 01:47:47 -0700668 const SkIRect generatorSubset =
669 SkIRect::MakeXYWH(fOrigin.x(), fOrigin.y(), fInfo.width(), fInfo.height());
670 Validator validator(fSharedGenerator, &generatorSubset, target);
Christopher Camerond4b67872017-07-13 15:18:08 -0700671 sk_sp<SkImage> result = validator ? sk_sp<SkImage>(new SkImage_Lazy(&validator)) : nullptr;
672 if (result) {
673 fOnMakeColorSpaceTarget = target;
674 fOnMakeColorSpaceResult = result;
675 }
676 return result;
Matt Sarett6de13102017-03-14 14:10:48 -0400677}
678
Mike Reed185130c2017-02-15 15:14:16 -0500679sk_sp<SkImage> SkImage::MakeFromGenerator(std::unique_ptr<SkImageGenerator> generator,
680 const SkIRect* subset) {
Christopher Cameron77e96662017-07-08 01:47:47 -0700681 SkImage_Lazy::Validator validator(SharedGenerator::Make(std::move(generator)), subset, nullptr);
fmalita7929e3a2016-10-27 08:15:44 -0700682
Brian Osmanf1b43822017-04-20 13:43:23 -0400683 return validator ? sk_make_sp<SkImage_Lazy>(&validator) : nullptr;
reed85d91782015-09-10 14:33:38 -0700684}
Brian Osmandf7e0752017-04-26 16:20:28 -0400685
686//////////////////////////////////////////////////////////////////////////////////////////////////
687
688/**
689 * Implementation of SkImageCacherator interface, as needed by GrImageTextureMaker
690 */
691
692#if SK_SUPPORT_GPU
693
694void SkImage_Lazy::makeCacheKeyFromOrigKey(const GrUniqueKey& origKey, CachedFormat format,
695 GrUniqueKey* cacheKey) {
696 SkASSERT(!cacheKey->isValid());
697 if (origKey.isValid()) {
698 static const GrUniqueKey::Domain kDomain = GrUniqueKey::GenerateDomain();
Derek Sollenbergere1c60d62018-04-04 11:53:35 -0400699 GrUniqueKey::Builder builder(cacheKey, origKey, kDomain, 1, "Image");
Brian Osmandf7e0752017-04-26 16:20:28 -0400700 builder[0] = format;
701 }
702}
703
704class Generator_GrYUVProvider : public GrYUVProvider {
705 SkImageGenerator* fGen;
706
707public:
708 Generator_GrYUVProvider(SkImageGenerator* gen) : fGen(gen) {}
709
710 uint32_t onGetID() override { return fGen->uniqueID(); }
711 bool onQueryYUV8(SkYUVSizeInfo* sizeInfo, SkYUVColorSpace* colorSpace) const override {
712 return fGen->queryYUV8(sizeInfo, colorSpace);
713 }
714 bool onGetYUV8Planes(const SkYUVSizeInfo& sizeInfo, void* planes[3]) override {
715 return fGen->getYUV8Planes(sizeInfo, planes);
716 }
717};
718
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500719static void set_key_on_proxy(GrProxyProvider* proxyProvider,
Greg Danielfc5060d2017-10-04 18:36:15 +0000720 GrTextureProxy* proxy, GrTextureProxy* originalProxy,
721 const GrUniqueKey& key) {
Brian Osmandf7e0752017-04-26 16:20:28 -0400722 if (key.isValid()) {
Robert Phillips8a90f502017-07-24 15:09:56 -0400723 SkASSERT(proxy->origin() == kTopLeft_GrSurfaceOrigin);
Greg Danielf6f7b672018-02-15 13:06:26 -0500724 if (originalProxy && originalProxy->getUniqueKey().isValid()) {
725 SkASSERT(originalProxy->getUniqueKey() == key);
Greg Daniele252f082017-10-23 16:05:23 -0400726 SkASSERT(GrMipMapped::kYes == proxy->mipMapped() &&
727 GrMipMapped::kNo == originalProxy->mipMapped());
Greg Danielf6f7b672018-02-15 13:06:26 -0500728 // If we had an originalProxy with a valid key, that means there already is a proxy in
729 // the cache which matches the key, but it does not have mip levels and we require them.
730 // Thus we must remove the unique key from that proxy.
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500731 proxyProvider->removeUniqueKeyFromProxy(key, originalProxy);
Greg Danielfc5060d2017-10-04 18:36:15 +0000732 }
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500733 proxyProvider->assignUniqueKeyToProxy(key, proxy);
Brian Osmandf7e0752017-04-26 16:20:28 -0400734 }
735}
736
737sk_sp<SkColorSpace> SkImage_Lazy::getColorSpace(GrContext* ctx, SkColorSpace* dstColorSpace) {
Brian Osmana8ac9242017-09-07 10:19:08 -0400738 if (!dstColorSpace) {
739 // In legacy mode, we do no modification to the image's color space or encoding.
740 // Subsequent legacy drawing is likely to ignore the color space, but some clients
741 // may want to know what space the image data is in, so return it.
742 return fInfo.refColorSpace();
743 } else {
Brian Salomonc7fe0f72018-05-11 10:14:21 -0400744 CachedFormat format = this->chooseCacheFormat(dstColorSpace, ctx->contextPriv().caps());
Brian Osmana8ac9242017-09-07 10:19:08 -0400745 SkImageInfo cacheInfo = this->buildCacheInfo(format);
746 return cacheInfo.refColorSpace();
747 }
Brian Osmandf7e0752017-04-26 16:20:28 -0400748}
749
750/*
751 * We have 4 ways to try to return a texture (in sorted order)
752 *
753 * 1. Check the cache for a pre-existing one
754 * 2. Ask the generator to natively create one
755 * 3. Ask the generator to return YUV planes, which the GPU can convert
756 * 4. Ask the generator to return RGB(A) data, which the GPU can convert
757 */
758sk_sp<GrTextureProxy> SkImage_Lazy::lockTextureProxy(GrContext* ctx,
759 const GrUniqueKey& origKey,
760 SkImage::CachingHint chint,
761 bool willBeMipped,
Stan Ilievba81af22017-06-08 15:16:53 -0400762 SkColorSpace* dstColorSpace,
763 GrTextureMaker::AllowedTexGenType genType) {
Brian Osmandf7e0752017-04-26 16:20:28 -0400764 // Values representing the various texture lock paths we can take. Used for logging the path
765 // taken to a histogram.
766 enum LockTexturePath {
767 kFailure_LockTexturePath,
768 kPreExisting_LockTexturePath,
769 kNative_LockTexturePath,
770 kCompressed_LockTexturePath, // Deprecated
771 kYUV_LockTexturePath,
772 kRGBA_LockTexturePath,
773 };
774
775 enum { kLockTexturePathCount = kRGBA_LockTexturePath + 1 };
776
777 // Determine which cached format we're going to use (which may involve decoding to a different
778 // info than the generator provides).
Brian Salomonc7fe0f72018-05-11 10:14:21 -0400779 CachedFormat format = this->chooseCacheFormat(dstColorSpace, ctx->contextPriv().caps());
Brian Osmandf7e0752017-04-26 16:20:28 -0400780
781 // Fold the cache format into our texture key
782 GrUniqueKey key;
783 this->makeCacheKeyFromOrigKey(origKey, format, &key);
784
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500785 GrProxyProvider* proxyProvider = ctx->contextPriv().proxyProvider();
Greg Danielfc5060d2017-10-04 18:36:15 +0000786 sk_sp<GrTextureProxy> proxy;
787
Brian Osmandf7e0752017-04-26 16:20:28 -0400788 // 1. Check the cache for a pre-existing one
789 if (key.isValid()) {
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500790 proxy = proxyProvider->findOrCreateProxyByUniqueKey(key, kTopLeft_GrSurfaceOrigin);
Greg Danielfc5060d2017-10-04 18:36:15 +0000791 if (proxy) {
Brian Osmandf7e0752017-04-26 16:20:28 -0400792 SK_HISTOGRAM_ENUMERATION("LockTexturePath", kPreExisting_LockTexturePath,
793 kLockTexturePathCount);
Greg Daniele252f082017-10-23 16:05:23 -0400794 if (!willBeMipped || GrMipMapped::kYes == proxy->mipMapped()) {
Greg Danielfc5060d2017-10-04 18:36:15 +0000795 return proxy;
796 }
Brian Osmandf7e0752017-04-26 16:20:28 -0400797 }
798 }
799
800 // The CachedFormat is both an index for which cache "slot" we'll use to store this particular
801 // decoded variant of the encoded data, and also a recipe for how to transform the original
802 // info to get the one that we're going to decode to.
Christopher Cameron77e96662017-07-08 01:47:47 -0700803 const SkImageInfo cacheInfo = this->buildCacheInfo(format);
804 SkImageInfo genPixelsInfo = cacheInfo;
805 SkTransferFunctionBehavior behavior = getGeneratorBehaviorAndInfo(&genPixelsInfo);
Brian Osmandf7e0752017-04-26 16:20:28 -0400806
807 // 2. Ask the generator to natively create one
Greg Danielfc5060d2017-10-04 18:36:15 +0000808 if (!proxy) {
Brian Osmandf7e0752017-04-26 16:20:28 -0400809 ScopedGenerator generator(fSharedGenerator);
Stan Ilievba81af22017-06-08 15:16:53 -0400810 if (GrTextureMaker::AllowedTexGenType::kCheap == genType &&
811 SkImageGenerator::TexGenType::kCheap != generator->onCanGenerateTexture()) {
812 return nullptr;
813 }
Greg Danielf88c12e2017-10-09 09:57:35 -0400814 if ((proxy = generator->generateTexture(ctx, genPixelsInfo, fOrigin, behavior,
815 willBeMipped))) {
Brian Osmandf7e0752017-04-26 16:20:28 -0400816 SK_HISTOGRAM_ENUMERATION("LockTexturePath", kNative_LockTexturePath,
817 kLockTexturePathCount);
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500818 set_key_on_proxy(proxyProvider, proxy.get(), nullptr, key);
Greg Daniele252f082017-10-23 16:05:23 -0400819 if (!willBeMipped || GrMipMapped::kYes == proxy->mipMapped()) {
Greg Danielfc5060d2017-10-04 18:36:15 +0000820 return proxy;
821 }
Brian Osmandf7e0752017-04-26 16:20:28 -0400822 }
823 }
824
Greg Daniel3e70fa32017-10-05 16:27:06 -0400825 // 3. Ask the generator to return YUV planes, which the GPU can convert. If we will be mipping
826 // the texture we fall through here and have the CPU generate the mip maps for us.
827 if (!proxy && !willBeMipped && !ctx->contextPriv().disableGpuYUVConversion()) {
Brian Osman2b23c4b2018-06-01 12:25:08 -0400828 const GrSurfaceDesc desc = GrImageInfoToSurfaceDesc(cacheInfo);
Brian Osmandf7e0752017-04-26 16:20:28 -0400829 ScopedGenerator generator(fSharedGenerator);
830 Generator_GrYUVProvider provider(generator);
Christopher Cameron77e96662017-07-08 01:47:47 -0700831
832 // The pixels in the texture will be in the generator's color space. If onMakeColorSpace
833 // has been called then this will not match this image's color space. To correct this, apply
834 // a color space conversion from the generator's color space to this image's color space.
Brian Osman56893cd2018-06-08 14:11:37 -0400835 // Note that we can only do this conversion (on the GPU) if both color spaces are XYZ type.
Brian Osman861ea5b2018-06-14 09:14:03 -0400836 SkColorSpace* generatorColorSpace = fSharedGenerator->fGenerator->getInfo().colorSpace();
837 SkColorSpace* thisColorSpace = fInfo.colorSpace();
Christopher Cameron77e96662017-07-08 01:47:47 -0700838
Brian Osman56893cd2018-06-08 14:11:37 -0400839 if ((!generatorColorSpace || generatorColorSpace->toXYZD50()) &&
840 (!thisColorSpace || thisColorSpace->toXYZD50())) {
841 // TODO: Update to create the mipped surface in the YUV generator and draw the base
842 // layer directly into the mipped surface.
843 proxy = provider.refAsTextureProxy(ctx, desc, generatorColorSpace, thisColorSpace);
844 if (proxy) {
845 SK_HISTOGRAM_ENUMERATION("LockTexturePath", kYUV_LockTexturePath,
846 kLockTexturePathCount);
847 set_key_on_proxy(proxyProvider, proxy.get(), nullptr, key);
848 return proxy;
849 }
Brian Osmandf7e0752017-04-26 16:20:28 -0400850 }
851 }
852
853 // 4. Ask the generator to return RGB(A) data, which the GPU can convert
854 SkBitmap bitmap;
Greg Danielfc5060d2017-10-04 18:36:15 +0000855 if (!proxy && this->lockAsBitmap(&bitmap, chint, format, genPixelsInfo, behavior)) {
Brian Osmandf7e0752017-04-26 16:20:28 -0400856 if (willBeMipped) {
Brian Osman2b23c4b2018-06-01 12:25:08 -0400857 proxy = proxyProvider->createMipMapProxyFromBitmap(bitmap);
Brian Osmandf7e0752017-04-26 16:20:28 -0400858 }
859 if (!proxy) {
Brian Osman2b23c4b2018-06-01 12:25:08 -0400860 proxy = GrUploadBitmapToTextureProxy(proxyProvider, bitmap);
Brian Osmandf7e0752017-04-26 16:20:28 -0400861 }
Greg Daniele252f082017-10-23 16:05:23 -0400862 if (proxy && (!willBeMipped || GrMipMapped::kYes == proxy->mipMapped())) {
Brian Osmandf7e0752017-04-26 16:20:28 -0400863 SK_HISTOGRAM_ENUMERATION("LockTexturePath", kRGBA_LockTexturePath,
864 kLockTexturePathCount);
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500865 set_key_on_proxy(proxyProvider, proxy.get(), nullptr, key);
Brian Osmandf7e0752017-04-26 16:20:28 -0400866 return proxy;
867 }
868 }
Greg Danielfc5060d2017-10-04 18:36:15 +0000869
870 if (proxy) {
871 // We need a mipped proxy, but we either found a proxy earlier that wasn't mipped, generated
872 // a native non mipped proxy, or generated a non-mipped yuv proxy. Thus we generate a new
873 // mipped surface and copy the original proxy into the base layer. We will then let the gpu
874 // generate the rest of the mips.
875 SkASSERT(willBeMipped);
Greg Daniele252f082017-10-23 16:05:23 -0400876 SkASSERT(GrMipMapped::kNo == proxy->mipMapped());
Greg Daniele1da1d92017-10-06 15:59:27 -0400877 if (auto mippedProxy = GrCopyBaseMipMapToTextureProxy(ctx, proxy.get())) {
Robert Phillips1afd4cd2018-01-08 13:40:32 -0500878 set_key_on_proxy(proxyProvider, mippedProxy.get(), proxy.get(), key);
Greg Danielfc5060d2017-10-04 18:36:15 +0000879 return mippedProxy;
880 }
Greg Daniel8f5bbda2018-06-08 17:22:23 -0400881 // We failed to make a mipped proxy with the base copied into it. This could have
882 // been from failure to make the proxy or failure to do the copy. Thus we will fall
883 // back to just using the non mipped proxy; See skbug.com/7094.
884 return proxy;
Greg Danielfc5060d2017-10-04 18:36:15 +0000885 }
886
Brian Osmandf7e0752017-04-26 16:20:28 -0400887 SK_HISTOGRAM_ENUMERATION("LockTexturePath", kFailure_LockTexturePath,
888 kLockTexturePathCount);
889 return nullptr;
890}
891
892///////////////////////////////////////////////////////////////////////////////////////////////////
893
894#endif