• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2012 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 "include/core/SkBitmap.h"
9 #include "include/core/SkCanvas.h"
10 #include "include/core/SkData.h"
11 #include "include/core/SkImageEncoder.h"
12 #include "include/core/SkImageFilter.h"
13 #include "include/core/SkImageGenerator.h"
14 #include "include/core/SkPicture.h"
15 #include "include/core/SkString.h"
16 #include "include/core/SkSurface.h"
17 #include "src/core/SkBitmapCache.h"
18 #include "src/core/SkCachedData.h"
19 #include "src/core/SkColorSpacePriv.h"
20 #include "src/core/SkImageFilterCache.h"
21 #include "src/core/SkImageFilter_Base.h"
22 #include "src/core/SkImagePriv.h"
23 #include "src/core/SkNextID.h"
24 #include "src/core/SkSpecialImage.h"
25 #include "src/image/SkImage_Base.h"
26 #include "src/image/SkReadPixelsRec.h"
27 #include "src/shaders/SkImageShader.h"
28 
29 #if SK_SUPPORT_GPU
30 #include "include/gpu/GrContext.h"
31 #include "include/gpu/GrTexture.h"
32 #include "src/image/SkImage_Gpu.h"
33 #endif
34 #include "include/gpu/GrBackendSurface.h"
35 
SkImage(const SkImageInfo & info,uint32_t uniqueID)36 SkImage::SkImage(const SkImageInfo& info, uint32_t uniqueID)
37         : fInfo(info)
38         , fUniqueID(kNeedNewImageUniqueID == uniqueID ? SkNextID::ImageID() : uniqueID) {
39     SkASSERT(info.width() > 0);
40     SkASSERT(info.height() > 0);
41 }
42 
peekPixels(SkPixmap * pm) const43 bool SkImage::peekPixels(SkPixmap* pm) const {
44     SkPixmap tmp;
45     if (!pm) {
46         pm = &tmp;
47     }
48     return as_IB(this)->onPeekPixels(pm);
49 }
50 
readPixels(const SkImageInfo & dstInfo,void * dstPixels,size_t dstRowBytes,int srcX,int srcY,CachingHint chint) const51 bool SkImage::readPixels(const SkImageInfo& dstInfo, void* dstPixels, size_t dstRowBytes, int srcX,
52                          int srcY, CachingHint chint) const {
53     return as_IB(this)->onReadPixels(dstInfo, dstPixels, dstRowBytes, srcX, srcY, chint);
54 }
55 
scalePixels(const SkPixmap & dst,SkFilterQuality quality,CachingHint chint) const56 bool SkImage::scalePixels(const SkPixmap& dst, SkFilterQuality quality, CachingHint chint) const {
57     if (this->width() == dst.width() && this->height() == dst.height()) {
58         return this->readPixels(dst, 0, 0, chint);
59     }
60 
61     // Idea: If/when SkImageGenerator supports a native-scaling API (where the generator itself
62     //       can scale more efficiently) we should take advantage of it here.
63     //
64     SkBitmap bm;
65     if (as_IB(this)->getROPixels(&bm, chint)) {
66         SkPixmap pmap;
67         // Note: By calling the pixmap scaler, we never cache the final result, so the chint
68         //       is (currently) only being applied to the getROPixels. If we get a request to
69         //       also attempt to cache the final (scaled) result, we would add that logic here.
70         //
71         return bm.peekPixels(&pmap) && pmap.scalePixels(dst, quality);
72     }
73     return false;
74 }
75 
76 ///////////////////////////////////////////////////////////////////////////////////////////////////
77 
colorType() const78 SkColorType SkImage::colorType() const { return fInfo.colorType(); }
79 
alphaType() const80 SkAlphaType SkImage::alphaType() const { return fInfo.alphaType(); }
81 
colorSpace() const82 SkColorSpace* SkImage::colorSpace() const { return fInfo.colorSpace(); }
83 
refColorSpace() const84 sk_sp<SkColorSpace> SkImage::refColorSpace() const { return fInfo.refColorSpace(); }
85 
makeShader(SkTileMode tmx,SkTileMode tmy,const SkMatrix * localMatrix) const86 sk_sp<SkShader> SkImage::makeShader(SkTileMode tmx, SkTileMode tmy,
87                                     const SkMatrix* localMatrix) const {
88     return SkImageShader::Make(sk_ref_sp(const_cast<SkImage*>(this)), tmx, tmy, localMatrix);
89 }
90 
encodeToData(SkEncodedImageFormat type,int quality) const91 sk_sp<SkData> SkImage::encodeToData(SkEncodedImageFormat type, int quality) const {
92     SkBitmap bm;
93     if (as_IB(this)->getROPixels(&bm)) {
94         return SkEncodeBitmap(bm, type, quality);
95     }
96     return nullptr;
97 }
98 
encodeToData() const99 sk_sp<SkData> SkImage::encodeToData() const {
100     if (auto encoded = this->refEncodedData()) {
101         return encoded;
102     }
103 
104     return this->encodeToData(SkEncodedImageFormat::kPNG, 100);
105 }
106 
refEncodedData() const107 sk_sp<SkData> SkImage::refEncodedData() const {
108     return sk_sp<SkData>(as_IB(this)->onRefEncoded());
109 }
110 
MakeFromEncoded(sk_sp<SkData> encoded,const SkIRect * subset)111 sk_sp<SkImage> SkImage::MakeFromEncoded(sk_sp<SkData> encoded, const SkIRect* subset) {
112     if (nullptr == encoded || 0 == encoded->size()) {
113         return nullptr;
114     }
115     return SkImage::MakeFromGenerator(SkImageGenerator::MakeFromEncoded(std::move(encoded)),
116                                       subset);
117 }
118 
119 ///////////////////////////////////////////////////////////////////////////////////////////////////
120 
makeSubset(const SkIRect & subset) const121 sk_sp<SkImage> SkImage::makeSubset(const SkIRect& subset) const {
122     if (subset.isEmpty()) {
123         return nullptr;
124     }
125 
126     const SkIRect bounds = SkIRect::MakeWH(this->width(), this->height());
127     if (!bounds.contains(subset)) {
128         return nullptr;
129     }
130 
131     // optimization : return self if the subset == our bounds
132     if (bounds == subset) {
133         return sk_ref_sp(const_cast<SkImage*>(this));
134     }
135 
136     // CONTEXT TODO: propagate the context parameter to the top-level API
137 #if SK_SUPPORT_GPU
138     return as_IB(this)->onMakeSubset(as_IB(this)->context(), subset);
139 #else
140     return as_IB(this)->onMakeSubset(nullptr, subset);
141 #endif
142 }
143 
144 #if SK_SUPPORT_GPU
145 
getTexture() const146 GrTexture* SkImage::getTexture() const {
147     return as_IB(this)->onGetTexture();
148 }
149 
isTextureBacked() const150 bool SkImage::isTextureBacked() const { return as_IB(this)->onIsTextureBacked(); }
151 
getBackendTexture(bool flushPendingGrContextIO,GrSurfaceOrigin * origin) const152 GrBackendTexture SkImage::getBackendTexture(bool flushPendingGrContextIO,
153                                             GrSurfaceOrigin* origin) const {
154     return as_IB(this)->onGetBackendTexture(flushPendingGrContextIO, origin);
155 }
156 
isValid(GrContext * context) const157 bool SkImage::isValid(GrContext* context) const {
158     if (context && context->abandoned()) {
159         return false;
160     }
161     return as_IB(this)->onIsValid(context);
162 }
163 
flush(GrContext * context,const GrFlushInfo & flushInfo)164 GrSemaphoresSubmitted SkImage::flush(GrContext* context, const GrFlushInfo& flushInfo) {
165     return as_IB(this)->onFlush(context, flushInfo);
166 }
167 
flush(GrContext * context)168 void SkImage::flush(GrContext* context) { as_IB(this)->onFlush(context, {}); }
169 
170 #else
171 
getTexture() const172 GrTexture* SkImage::getTexture() const { return nullptr; }
173 
isTextureBacked() const174 bool SkImage::isTextureBacked() const { return false; }
175 
getBackendTexture(bool flushPendingGrContextIO,GrSurfaceOrigin * origin) const176 GrBackendTexture SkImage::getBackendTexture(bool flushPendingGrContextIO,
177                                             GrSurfaceOrigin* origin) const {
178     return GrBackendTexture(); // invalid
179 }
180 
isValid(GrContext * context) const181 bool SkImage::isValid(GrContext* context) const {
182     if (context) {
183         return false;
184     }
185     return as_IB(this)->onIsValid(context);
186 }
187 
flush(GrContext *,const GrFlushInfo &)188 GrSemaphoresSubmitted SkImage::flush(GrContext*, const GrFlushInfo&) {
189     return GrSemaphoresSubmitted::kNo;
190 }
191 
flush(GrContext *)192 void SkImage::flush(GrContext*) {}
193 
194 #endif
195 
196 ///////////////////////////////////////////////////////////////////////////////
197 
SkImage_Base(const SkImageInfo & info,uint32_t uniqueID)198 SkImage_Base::SkImage_Base(const SkImageInfo& info, uint32_t uniqueID)
199         : INHERITED(info, uniqueID), fAddedToRasterCache(false) {}
200 
~SkImage_Base()201 SkImage_Base::~SkImage_Base() {
202     if (fAddedToRasterCache.load()) {
203         SkNotifyBitmapGenIDIsStale(this->uniqueID());
204     }
205 }
206 
onGetBackendTexture(bool flushPendingGrContextIO,GrSurfaceOrigin * origin) const207 GrBackendTexture SkImage_Base::onGetBackendTexture(bool flushPendingGrContextIO,
208                                                    GrSurfaceOrigin* origin) const {
209     return GrBackendTexture(); // invalid
210 }
211 
readPixels(const SkPixmap & pmap,int srcX,int srcY,CachingHint chint) const212 bool SkImage::readPixels(const SkPixmap& pmap, int srcX, int srcY, CachingHint chint) const {
213     return this->readPixels(pmap.info(), pmap.writable_addr(), pmap.rowBytes(), srcX, srcY, chint);
214 }
215 
216 ///////////////////////////////////////////////////////////////////////////////////////////////////
217 
MakeFromBitmap(const SkBitmap & bm)218 sk_sp<SkImage> SkImage::MakeFromBitmap(const SkBitmap& bm) {
219     if (!bm.pixelRef()) {
220         return nullptr;
221     }
222 
223     return SkMakeImageFromRasterBitmap(bm, kIfMutable_SkCopyPixelsMode);
224 }
225 
asLegacyBitmap(SkBitmap * bitmap,LegacyBitmapMode) const226 bool SkImage::asLegacyBitmap(SkBitmap* bitmap, LegacyBitmapMode ) const {
227     return as_IB(this)->onAsLegacyBitmap(bitmap);
228 }
229 
getPlanes(SkYUVASizeInfo *,SkYUVAIndex[4],SkYUVColorSpace *,const void * [4])230 sk_sp<SkCachedData> SkImage_Base::getPlanes(SkYUVASizeInfo*, SkYUVAIndex[4],
231                                             SkYUVColorSpace*, const void*[4]) {
232     return nullptr;
233 }
234 
onAsLegacyBitmap(SkBitmap * bitmap) const235 bool SkImage_Base::onAsLegacyBitmap(SkBitmap* bitmap) const {
236     // As the base-class, all we can do is make a copy (regardless of mode).
237     // Subclasses that want to be more optimal should override.
238     SkImageInfo info = fInfo.makeColorType(kN32_SkColorType).makeColorSpace(nullptr);
239     if (!bitmap->tryAllocPixels(info)) {
240         return false;
241     }
242     if (!this->readPixels(bitmap->info(), bitmap->getPixels(), bitmap->rowBytes(), 0, 0)) {
243         bitmap->reset();
244         return false;
245     }
246 
247     bitmap->setImmutable();
248     return true;
249 }
250 
MakeFromPicture(sk_sp<SkPicture> picture,const SkISize & dimensions,const SkMatrix * matrix,const SkPaint * paint,BitDepth bitDepth,sk_sp<SkColorSpace> colorSpace)251 sk_sp<SkImage> SkImage::MakeFromPicture(sk_sp<SkPicture> picture, const SkISize& dimensions,
252                                         const SkMatrix* matrix, const SkPaint* paint,
253                                         BitDepth bitDepth, sk_sp<SkColorSpace> colorSpace) {
254     return MakeFromGenerator(SkImageGenerator::MakeFromPicture(dimensions, std::move(picture),
255                                                                matrix, paint, bitDepth,
256                                                                std::move(colorSpace)));
257 }
258 
makeWithFilter(const SkImageFilter * filter,const SkIRect & subset,const SkIRect & clipBounds,SkIRect * outSubset,SkIPoint * offset) const259 sk_sp<SkImage> SkImage::makeWithFilter(const SkImageFilter* filter, const SkIRect& subset,
260                                        const SkIRect& clipBounds, SkIRect* outSubset,
261                                        SkIPoint* offset) const {
262     GrContext* context = as_IB(this)->context();
263 
264     return this->makeWithFilter(context, filter, subset, clipBounds, outSubset, offset);
265 }
266 
makeWithFilter(GrContext * grContext,const SkImageFilter * filter,const SkIRect & subset,const SkIRect & clipBounds,SkIRect * outSubset,SkIPoint * offset) const267 sk_sp<SkImage> SkImage::makeWithFilter(GrContext* grContext,
268                                        const SkImageFilter* filter, const SkIRect& subset,
269                                        const SkIRect& clipBounds, SkIRect* outSubset,
270                                        SkIPoint* offset) const {
271     if (!filter || !outSubset || !offset || !this->bounds().contains(subset)) {
272         return nullptr;
273     }
274     sk_sp<SkSpecialImage> srcSpecialImage =
275 #if SK_SUPPORT_GPU
276         SkSpecialImage::MakeFromImage(grContext, subset, sk_ref_sp(const_cast<SkImage*>(this)));
277 #else
278         SkSpecialImage::MakeFromImage(nullptr, subset, sk_ref_sp(const_cast<SkImage*>(this)));
279 #endif
280     if (!srcSpecialImage) {
281         return nullptr;
282     }
283 
284     sk_sp<SkImageFilterCache> cache(
285         SkImageFilterCache::Create(SkImageFilterCache::kDefaultTransientSize));
286 
287     // The filters operate in the local space of the src image, where (0,0) corresponds to the
288     // subset's top left corner. But the clip bounds and any crop rects on the filters are in the
289     // original coordinate system, so configure the CTM to correct crop rects and explicitly adjust
290     // the clip bounds (since it is assumed to already be in image space).
291     SkImageFilter_Base::Context context(SkMatrix::MakeTrans(-subset.x(), -subset.y()),
292                                         clipBounds.makeOffset(-subset.x(), -subset.y()),
293                                         cache.get(), fInfo.colorType(), fInfo.colorSpace(),
294                                         srcSpecialImage.get());
295 
296     sk_sp<SkSpecialImage> result = as_IFB(filter)->filterImage(context, offset);
297     if (!result) {
298         return nullptr;
299     }
300 
301     // The output image and offset are relative to the subset rectangle, so the offset needs to
302     // be shifted to put it in the correct spot with respect to the original coordinate system
303     offset->fX += subset.x();
304     offset->fY += subset.y();
305 
306     // Final clip against the exact clipBounds (the clip provided in the context gets adjusted
307     // to account for pixel-moving filters so doesn't always exactly match when finished). The
308     // clipBounds are translated into the clippedDstRect coordinate space, including the
309     // result->subset() ensures that the result's image pixel origin does not affect results.
310     SkIRect dstRect = result->subset();
311     SkIRect clippedDstRect = dstRect;
312     if (!clippedDstRect.intersect(clipBounds.makeOffset(result->subset().x() - offset->x(),
313                                                         result->subset().y() - offset->y()))) {
314         return nullptr;
315     }
316 
317     // Adjust the geometric offset if the top-left corner moved as well
318     offset->fX += (clippedDstRect.x() - dstRect.x());
319     offset->fY += (clippedDstRect.y() - dstRect.y());
320     *outSubset = clippedDstRect;
321     return result->asImage();
322 }
323 
isLazyGenerated() const324 bool SkImage::isLazyGenerated() const {
325     return as_IB(this)->onIsLazyGenerated();
326 }
327 
isAlphaOnly() const328 bool SkImage::isAlphaOnly() const { return SkColorTypeIsAlphaOnly(fInfo.colorType()); }
329 
makeColorSpace(sk_sp<SkColorSpace> target) const330 sk_sp<SkImage> SkImage::makeColorSpace(sk_sp<SkColorSpace> target) const {
331     if (!target) {
332         return nullptr;
333     }
334 
335     // No need to create a new image if:
336     // (1) The color spaces are equal.
337     // (2) The color type is kAlpha8.
338     SkColorSpace* colorSpace = this->colorSpace();
339     if (!colorSpace) {
340         colorSpace = sk_srgb_singleton();
341     }
342     if (SkColorSpace::Equals(colorSpace, target.get()) || this->isAlphaOnly()) {
343         return sk_ref_sp(const_cast<SkImage*>(this));
344     }
345 
346     // CONTEXT TODO: propagate the context parameter to the top-level API
347 #if SK_SUPPORT_GPU
348     return as_IB(this)->onMakeColorTypeAndColorSpace(as_IB(this)->context(),
349 #else
350     return as_IB(this)->onMakeColorTypeAndColorSpace(nullptr,
351 #endif
352                                                      this->colorType(), std::move(target));
353 }
354 
makeColorTypeAndColorSpace(SkColorType targetColorType,sk_sp<SkColorSpace> targetColorSpace) const355 sk_sp<SkImage> SkImage::makeColorTypeAndColorSpace(SkColorType targetColorType,
356                                                    sk_sp<SkColorSpace> targetColorSpace) const {
357     if (kUnknown_SkColorType == targetColorType || !targetColorSpace) {
358         return nullptr;
359     }
360 
361     SkColorType colorType = this->colorType();
362     SkColorSpace* colorSpace = this->colorSpace();
363     if (!colorSpace) {
364         colorSpace = sk_srgb_singleton();
365     }
366     if (colorType == targetColorType &&
367         (SkColorSpace::Equals(colorSpace, targetColorSpace.get()) || this->isAlphaOnly())) {
368         return sk_ref_sp(const_cast<SkImage*>(this));
369     }
370 
371     // CONTEXT TODO: propagate the context parameter to the top-level API
372 #if SK_SUPPORT_GPU
373     return as_IB(this)->onMakeColorTypeAndColorSpace(as_IB(this)->context(),
374 #else
375     return as_IB(this)->onMakeColorTypeAndColorSpace(nullptr,
376 #endif
377                                                      targetColorType, std::move(targetColorSpace));
378 }
379 
reinterpretColorSpace(sk_sp<SkColorSpace> target) const380 sk_sp<SkImage> SkImage::reinterpretColorSpace(sk_sp<SkColorSpace> target) const {
381     if (!target) {
382         return nullptr;
383     }
384 
385     // No need to create a new image if:
386     // (1) The color spaces are equal.
387     // (2) The color type is kAlpha8.
388     SkColorSpace* colorSpace = this->colorSpace();
389     if (!colorSpace) {
390         colorSpace = sk_srgb_singleton();
391     }
392     if (SkColorSpace::Equals(colorSpace, target.get()) || this->isAlphaOnly()) {
393         return sk_ref_sp(const_cast<SkImage*>(this));
394     }
395 
396     return as_IB(this)->onReinterpretColorSpace(std::move(target));
397 }
398 
makeNonTextureImage() const399 sk_sp<SkImage> SkImage::makeNonTextureImage() const {
400     if (!this->isTextureBacked()) {
401         return sk_ref_sp(const_cast<SkImage*>(this));
402     }
403     return this->makeRasterImage();
404 }
405 
makeRasterImage() const406 sk_sp<SkImage> SkImage::makeRasterImage() const {
407     SkPixmap pm;
408     if (this->peekPixels(&pm)) {
409         return sk_ref_sp(const_cast<SkImage*>(this));
410     }
411 
412     const size_t rowBytes = fInfo.minRowBytes();
413     size_t size = fInfo.computeByteSize(rowBytes);
414     if (SkImageInfo::ByteSizeOverflowed(size)) {
415         return nullptr;
416     }
417 
418     sk_sp<SkData> data = SkData::MakeUninitialized(size);
419     pm = {fInfo.makeColorSpace(nullptr), data->writable_data(), fInfo.minRowBytes()};
420     if (!this->readPixels(pm, 0, 0)) {
421         return nullptr;
422     }
423 
424     return SkImage::MakeRasterData(fInfo, std::move(data), rowBytes);
425 }
426 
427 //////////////////////////////////////////////////////////////////////////////////////
428 
429 #if !SK_SUPPORT_GPU
430 
DecodeToTexture(GrContext *,const void *,size_t,const SkIRect *)431 sk_sp<SkImage> SkImage::DecodeToTexture(GrContext*, const void*, size_t, const SkIRect*) {
432     return nullptr;
433 }
434 
MakeFromTexture(GrContext * ctx,const GrBackendTexture & tex,GrSurfaceOrigin origin,SkColorType ct,SkAlphaType at,sk_sp<SkColorSpace> cs,TextureReleaseProc releaseP,ReleaseContext releaseC)435 sk_sp<SkImage> SkImage::MakeFromTexture(GrContext* ctx,
436                                         const GrBackendTexture& tex, GrSurfaceOrigin origin,
437                                         SkColorType ct, SkAlphaType at, sk_sp<SkColorSpace> cs,
438                                         TextureReleaseProc releaseP, ReleaseContext releaseC) {
439     return nullptr;
440 }
441 
MakeBackendTextureFromSkImage(GrContext *,sk_sp<SkImage>,GrBackendTexture *,BackendTextureReleaseProc *)442 bool SkImage::MakeBackendTextureFromSkImage(GrContext*,
443                                             sk_sp<SkImage>,
444                                             GrBackendTexture*,
445                                             BackendTextureReleaseProc*) {
446     return false;
447 }
448 
MakeFromAdoptedTexture(GrContext * ctx,const GrBackendTexture & tex,GrSurfaceOrigin origin,SkColorType ct,SkAlphaType at,sk_sp<SkColorSpace> cs)449 sk_sp<SkImage> SkImage::MakeFromAdoptedTexture(GrContext* ctx,
450                                                const GrBackendTexture& tex, GrSurfaceOrigin origin,
451                                                SkColorType ct, SkAlphaType at,
452                                                sk_sp<SkColorSpace> cs) {
453     return nullptr;
454 }
455 
MakeFromYUVATexturesCopy(GrContext * context,SkYUVColorSpace yuvColorSpace,const GrBackendTexture yuvaTextures[],const SkYUVAIndex yuvaIndices[4],SkISize imageSize,GrSurfaceOrigin imageOrigin,sk_sp<SkColorSpace> imageColorSpace)456 sk_sp<SkImage> SkImage::MakeFromYUVATexturesCopy(GrContext* context,
457                                                  SkYUVColorSpace yuvColorSpace,
458                                                  const GrBackendTexture yuvaTextures[],
459                                                  const SkYUVAIndex yuvaIndices[4],
460                                                  SkISize imageSize,
461                                                  GrSurfaceOrigin imageOrigin,
462                                                  sk_sp<SkColorSpace> imageColorSpace) {
463     return nullptr;
464 }
465 
MakeFromYUVATexturesCopyWithExternalBackend(GrContext * context,SkYUVColorSpace yuvColorSpace,const GrBackendTexture yuvaTextures[],const SkYUVAIndex yuvaIndices[4],SkISize imageSize,GrSurfaceOrigin imageOrigin,const GrBackendTexture & backendTexture,sk_sp<SkColorSpace> imageColorSpace)466 sk_sp<SkImage> SkImage::MakeFromYUVATexturesCopyWithExternalBackend(
467         GrContext* context,
468         SkYUVColorSpace yuvColorSpace,
469         const GrBackendTexture yuvaTextures[],
470         const SkYUVAIndex yuvaIndices[4],
471         SkISize imageSize,
472         GrSurfaceOrigin imageOrigin,
473         const GrBackendTexture& backendTexture,
474         sk_sp<SkColorSpace> imageColorSpace) {
475     return nullptr;
476 }
477 
MakeFromYUVTexturesCopy(GrContext * ctx,SkYUVColorSpace space,const GrBackendTexture[3],GrSurfaceOrigin origin,sk_sp<SkColorSpace> imageColorSpace)478 sk_sp<SkImage> SkImage::MakeFromYUVTexturesCopy(GrContext* ctx, SkYUVColorSpace space,
479                                                 const GrBackendTexture[3],
480                                                 GrSurfaceOrigin origin,
481                                                 sk_sp<SkColorSpace> imageColorSpace) {
482     return nullptr;
483 }
484 
MakeFromYUVTexturesCopyWithExternalBackend(GrContext * context,SkYUVColorSpace yuvColorSpace,const GrBackendTexture yuvTextures[3],GrSurfaceOrigin surfaceOrigin,const GrBackendTexture & backendTexture,sk_sp<SkColorSpace> colorSpace)485 sk_sp<SkImage> SkImage::MakeFromYUVTexturesCopyWithExternalBackend(
486         GrContext* context, SkYUVColorSpace yuvColorSpace, const GrBackendTexture yuvTextures[3],
487         GrSurfaceOrigin surfaceOrigin, const GrBackendTexture& backendTexture,
488         sk_sp<SkColorSpace> colorSpace) {
489     return nullptr;
490 }
491 
MakeFromNV12TexturesCopy(GrContext * ctx,SkYUVColorSpace space,const GrBackendTexture[2],GrSurfaceOrigin origin,sk_sp<SkColorSpace> imageColorSpace)492 sk_sp<SkImage> SkImage::MakeFromNV12TexturesCopy(GrContext* ctx, SkYUVColorSpace space,
493                                                 const GrBackendTexture[2],
494                                                 GrSurfaceOrigin origin,
495                                                 sk_sp<SkColorSpace> imageColorSpace) {
496     return nullptr;
497 }
498 
makeTextureImage(GrContext *,GrMipMapped mipMapped) const499 sk_sp<SkImage> SkImage::makeTextureImage(GrContext*, GrMipMapped mipMapped) const {
500     return nullptr;
501 }
502 
MakeFromNV12TexturesCopyWithExternalBackend(GrContext * context,SkYUVColorSpace yuvColorSpace,const GrBackendTexture nv12Textures[2],GrSurfaceOrigin surfaceOrigin,const GrBackendTexture & backendTexture,sk_sp<SkColorSpace> colorSpace)503 sk_sp<SkImage> MakeFromNV12TexturesCopyWithExternalBackend(GrContext* context,
504                                                            SkYUVColorSpace yuvColorSpace,
505                                                            const GrBackendTexture nv12Textures[2],
506                                                            GrSurfaceOrigin surfaceOrigin,
507                                                            const GrBackendTexture& backendTexture,
508                                                            sk_sp<SkColorSpace> colorSpace) {
509     return nullptr;
510 }
511 
512 #endif
513 
514 ///////////////////////////////////////////////////////////////////////////////////////////////////
515 
SkImage_pinAsTexture(const SkImage * image,GrContext * ctx)516 bool SkImage_pinAsTexture(const SkImage* image, GrContext* ctx) {
517     SkASSERT(image);
518     SkASSERT(ctx);
519     return as_IB(image)->onPinAsTexture(ctx);
520 }
521 
SkImage_unpinAsTexture(const SkImage * image,GrContext * ctx)522 void SkImage_unpinAsTexture(const SkImage* image, GrContext* ctx) {
523     SkASSERT(image);
524     SkASSERT(ctx);
525     as_IB(image)->onUnpinAsTexture(ctx);
526 }
527 
SkImage_getSubset(const SkImage * image)528 SkIRect SkImage_getSubset(const SkImage* image) {
529     SkASSERT(image);
530     return as_IB(image)->onGetSubset();
531 }
532