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