2013-11-21 15:32:08 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2013 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef SkImageGenerator_DEFINED
|
|
|
|
#define SkImageGenerator_DEFINED
|
|
|
|
|
2015-09-17 20:35:19 +00:00
|
|
|
#include "SkBitmap.h"
|
2014-05-29 15:57:20 +00:00
|
|
|
#include "SkColor.h"
|
2014-09-13 00:45:58 +00:00
|
|
|
#include "SkImageInfo.h"
|
2013-11-21 15:32:08 +00:00
|
|
|
|
2015-08-13 20:32:39 +00:00
|
|
|
class GrContext;
|
2015-10-12 17:39:46 +00:00
|
|
|
class GrTexture;
|
|
|
|
class GrTextureParams;
|
2013-12-10 21:11:12 +00:00
|
|
|
class SkBitmap;
|
2013-11-21 15:32:08 +00:00
|
|
|
class SkData;
|
2013-12-10 21:11:12 +00:00
|
|
|
class SkImageGenerator;
|
2015-08-04 20:53:14 +00:00
|
|
|
class SkMatrix;
|
|
|
|
class SkPaint;
|
|
|
|
class SkPicture;
|
2013-12-10 21:11:12 +00:00
|
|
|
|
2016-01-05 17:16:19 +00:00
|
|
|
#ifdef SK_SUPPORT_LEGACY_REFENCODEDDATA_NOCTX
|
|
|
|
#define SK_REFENCODEDDATA_CTXPARAM
|
|
|
|
#else
|
|
|
|
#define SK_REFENCODEDDATA_CTXPARAM GrContext* ctx
|
|
|
|
#endif
|
|
|
|
|
2013-12-10 21:11:12 +00:00
|
|
|
/**
|
|
|
|
* Takes ownership of SkImageGenerator. If this method fails for
|
|
|
|
* whatever reason, it will return false and immediatetely delete
|
|
|
|
* the generator. If it succeeds, it will modify destination
|
|
|
|
* bitmap.
|
|
|
|
*
|
Add Options to SkDecodingImageGenerator, simplify API.
Motivation: We want to remove redundant classes from Skia. To
that end we want to remove SkImageRef and its subclasses and
replace their uses with SkDiscardablePixelRef +
SkDecodingImageGenerator. Since Android uses SkImageRef, we need
to make sure that SkDecodingImageGenerator allows all of the
settings that Android exposes in BitmapFactory.Options.
To that end, we have created an Options struct for the
SkDecodingImageGenerator which lets the client of the generator set
sample size, dithering, and bitmap config.
We have made the SkDecodingImageGenerator constructor private
and replaced the SkDecodingImageGenerator::Install functions
with a SkDecodingImageGenerator::Create functions (one for
SkData and one for SkStream) which now take a
SkDecodingImageGenerator::Options struct.
Also added a ImageDecoderOptions test which loops through a list
of sets of options and tries them on a set of 5 small encoded
images.
Also updated several users of SkDecodingImageGenerator::Install to
follow new call signature - gm/factory.cpp, LazyDecodeBitmap.cpp,
and PictureTest.cpp, CachedDecodingPixelRefTest.cpp.
We also added a new ImprovedBitmapFactory Test which simulates the
exact function that Android will need to modify to use this,
installPixelRef() in BitmapFactory.
R=reed@google.com, scroggo@google.com
Committed: https://code.google.com/p/skia/source/detail?r=12744
Review URL: https://codereview.chromium.org/93703004
git-svn-id: http://skia.googlecode.com/svn/trunk@12855 2bbb7eff-a529-9590-31e7-b0007b416f81
2014-01-02 13:15:13 +00:00
|
|
|
* If generator is NULL, will safely return false.
|
|
|
|
*
|
2013-12-10 21:11:12 +00:00
|
|
|
* If this fails or when the SkDiscardablePixelRef that is
|
2015-09-22 13:10:35 +00:00
|
|
|
* installed into destination is destroyed, it will
|
|
|
|
* delete the generator. Therefore, generator should be
|
|
|
|
* allocated with new.
|
2013-12-10 21:11:12 +00:00
|
|
|
*
|
|
|
|
* @param destination Upon success, this bitmap will be
|
|
|
|
* configured and have a pixelref installed.
|
|
|
|
*
|
|
|
|
* @return true iff successful.
|
|
|
|
*/
|
2015-09-25 13:56:57 +00:00
|
|
|
SK_API bool SkDEPRECATED_InstallDiscardablePixelRef(SkImageGenerator*, SkBitmap* destination);
|
2013-12-10 21:11:12 +00:00
|
|
|
|
2015-01-08 02:04:45 +00:00
|
|
|
/**
|
2015-01-08 02:20:16 +00:00
|
|
|
* On success, installs a discardable pixelref into destination, based on encoded data.
|
2015-01-08 02:04:45 +00:00
|
|
|
* Regardless of success or failure, the caller must still balance their ownership of encoded.
|
|
|
|
*/
|
2015-09-25 13:56:57 +00:00
|
|
|
SK_API bool SkDEPRECATED_InstallDiscardablePixelRef(SkData* encoded, SkBitmap* destination);
|
2013-11-21 15:32:08 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* An interface that allows a purgeable PixelRef (such as a
|
|
|
|
* SkDiscardablePixelRef) to decode and re-decode an image as needed.
|
|
|
|
*/
|
2015-02-13 19:13:34 +00:00
|
|
|
class SK_API SkImageGenerator : public SkNoncopyable {
|
2013-11-21 15:32:08 +00:00
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* The PixelRef which takes ownership of this SkImageGenerator
|
|
|
|
* will call the image generator's destructor.
|
|
|
|
*/
|
|
|
|
virtual ~SkImageGenerator() { }
|
|
|
|
|
2015-08-13 20:32:39 +00:00
|
|
|
uint32_t uniqueID() const { return fUniqueID; }
|
|
|
|
|
2013-11-21 15:32:08 +00:00
|
|
|
/**
|
|
|
|
* Return a ref to the encoded (i.e. compressed) representation,
|
2016-01-05 17:16:19 +00:00
|
|
|
* of this data. If the GrContext is non-null, then the caller is only interested in
|
|
|
|
* gpu-specific formats, so the impl may return null even if they have encoded data,
|
|
|
|
* assuming they know it is not suitable for the gpu.
|
2013-11-21 15:32:08 +00:00
|
|
|
*
|
|
|
|
* If non-NULL is returned, the caller is responsible for calling
|
|
|
|
* unref() on the data when it is finished.
|
|
|
|
*/
|
2016-01-05 17:16:19 +00:00
|
|
|
SkData* refEncodedData(GrContext* ctx = nullptr) {
|
|
|
|
#ifdef SK_SUPPORT_LEGACY_REFENCODEDDATA_NOCTX
|
|
|
|
return this->onRefEncodedData();
|
|
|
|
#else
|
|
|
|
return this->onRefEncodedData(ctx);
|
|
|
|
#endif
|
|
|
|
}
|
2013-11-21 15:32:08 +00:00
|
|
|
|
|
|
|
/**
|
2015-03-19 15:31:14 +00:00
|
|
|
* Return the ImageInfo associated with this generator.
|
2013-11-21 15:32:08 +00:00
|
|
|
*/
|
2015-03-19 15:31:14 +00:00
|
|
|
const SkImageInfo& getInfo() const { return fInfo; }
|
2013-11-21 15:32:08 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Decode into the given pixels, a block of memory of size at
|
|
|
|
* least (info.fHeight - 1) * rowBytes + (info.fWidth *
|
|
|
|
* bytesPerPixel)
|
|
|
|
*
|
2013-12-02 22:32:54 +00:00
|
|
|
* Repeated calls to this function should give the same results,
|
|
|
|
* allowing the PixelRef to be immutable.
|
|
|
|
*
|
2013-11-21 15:32:08 +00:00
|
|
|
* @param info A description of the format (config, size)
|
|
|
|
* expected by the caller. This can simply be identical
|
|
|
|
* to the info returned by getInfo().
|
|
|
|
*
|
|
|
|
* This contract also allows the caller to specify
|
|
|
|
* different output-configs, which the implementation can
|
|
|
|
* decide to support or not.
|
|
|
|
*
|
2015-02-13 19:13:34 +00:00
|
|
|
* A size that does not match getInfo() implies a request
|
|
|
|
* to scale. If the generator cannot perform this scale,
|
|
|
|
* it will return kInvalidScale.
|
|
|
|
*
|
2014-05-29 15:57:20 +00:00
|
|
|
* If info is kIndex8_SkColorType, then the caller must provide storage for up to 256
|
|
|
|
* SkPMColor values in ctable. On success the generator must copy N colors into that storage,
|
|
|
|
* (where N is the logical number of table entries) and set ctableCount to N.
|
|
|
|
*
|
|
|
|
* If info is not kIndex8_SkColorType, then the last two parameters may be NULL. If ctableCount
|
|
|
|
* is not null, it will be set to 0.
|
|
|
|
*
|
2015-07-09 16:08:00 +00:00
|
|
|
* @return true on success.
|
2013-11-21 15:32:08 +00:00
|
|
|
*/
|
2015-07-09 16:08:00 +00:00
|
|
|
bool getPixels(const SkImageInfo& info, void* pixels, size_t rowBytes,
|
|
|
|
SkPMColor ctable[], int* ctableCount);
|
2014-05-29 15:57:20 +00:00
|
|
|
|
|
|
|
/**
|
2015-03-17 12:02:17 +00:00
|
|
|
* Simplified version of getPixels() that asserts that info is NOT kIndex8_SkColorType and
|
|
|
|
* uses the default Options.
|
2014-05-29 15:57:20 +00:00
|
|
|
*/
|
2015-07-09 16:08:00 +00:00
|
|
|
bool getPixels(const SkImageInfo& info, void* pixels, size_t rowBytes);
|
2014-05-29 15:57:20 +00:00
|
|
|
|
2014-07-21 18:37:39 +00:00
|
|
|
/**
|
|
|
|
* If planes or rowBytes is NULL or if any entry in planes is NULL or if any entry in rowBytes
|
|
|
|
* is 0, this imagegenerator should output the sizes and return true if it can efficiently
|
|
|
|
* return YUV planar data. If it cannot, it should return false. Note that either planes and
|
|
|
|
* rowBytes are both fully defined and non NULL/non 0 or they are both NULL or have NULL or 0
|
|
|
|
* entries only. Having only partial planes/rowBytes information is not supported.
|
|
|
|
*
|
|
|
|
* If all planes and rowBytes entries are non NULL or non 0, then it should copy the
|
|
|
|
* associated YUV data into those planes of memory supplied by the caller. It should validate
|
|
|
|
* that the sizes match what it expected. If the sizes do not match, it should return false.
|
|
|
|
*/
|
2014-09-13 00:45:58 +00:00
|
|
|
bool getYUV8Planes(SkISize sizes[3], void* planes[3], size_t rowBytes[3],
|
|
|
|
SkYUVColorSpace* colorSpace);
|
2014-07-21 18:37:39 +00:00
|
|
|
|
2015-08-13 20:32:39 +00:00
|
|
|
/**
|
|
|
|
* If the generator can natively/efficiently return its pixels as a GPU image (backed by a
|
|
|
|
* texture) this will return that image. If not, this will return NULL.
|
|
|
|
*
|
|
|
|
* Regarding the GrContext parameter:
|
|
|
|
*
|
|
|
|
* The caller may pass NULL for the context. In that case the generator may assume that its
|
|
|
|
* internal context is current. If it has no internal context, then it should just return
|
|
|
|
* null.
|
|
|
|
*
|
|
|
|
* If the caller passes a non-null context, then the generator should only succeed if:
|
|
|
|
* - it has no intrinsic context, and will use the caller's
|
|
|
|
* - its internal context is the same
|
|
|
|
* - it can somehow convert its texture into one that is valid for the provided context.
|
|
|
|
*
|
2015-10-12 17:39:46 +00:00
|
|
|
* Regarding the GrTextureParams parameter:
|
2015-08-13 20:32:39 +00:00
|
|
|
*
|
|
|
|
* If the context (the provided one or the generator's intrinsic one) determines that to
|
2015-08-13 21:06:46 +00:00
|
|
|
* support the specified usage, it must return a different sized texture it may,
|
|
|
|
* so the caller must inspect the texture's width/height and compare them to the generator's
|
2015-10-12 17:39:46 +00:00
|
|
|
* getInfo() width/height. For readback usage use GrTextureParams::ClampNoFilter()
|
2015-08-13 20:32:39 +00:00
|
|
|
*/
|
2015-10-15 19:14:55 +00:00
|
|
|
GrTexture* generateTexture(GrContext*, const SkIRect* subset = nullptr);
|
2015-08-13 20:32:39 +00:00
|
|
|
|
2015-12-02 22:19:47 +00:00
|
|
|
struct SupportedSizes {
|
|
|
|
SkISize fSizes[2];
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Some generators can efficiently scale their contents. If this is supported, the generator
|
|
|
|
* may only support certain scaled dimensions. Call this with the desired scale factor,
|
|
|
|
* and it will return true if scaling is supported, and in supportedSizes[] it will return
|
|
|
|
* the nearest supported dimensions.
|
|
|
|
*
|
|
|
|
* If no native scaling is supported, or scale is invalid (e.g. scale <= 0 || scale > 1)
|
|
|
|
* this will return false, and the supportedsizes will be undefined.
|
|
|
|
*/
|
|
|
|
bool computeScaledDimensions(SkScalar scale, SupportedSizes*);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Scale the generator's pixels to fit into scaledSize.
|
|
|
|
* This routine also support retrieving only a subset of the pixels. That subset is specified
|
|
|
|
* by the following rectangle (in the scaled space):
|
|
|
|
*
|
|
|
|
* subset = SkIRect::MakeXYWH(subsetOrigin.x(), subsetOrigin.y(),
|
|
|
|
* subsetPixels.width(), subsetPixels.height())
|
|
|
|
*
|
|
|
|
* If subset is not contained inside the scaledSize, this returns false.
|
|
|
|
*
|
|
|
|
* whole = SkIRect::MakeWH(scaledSize.width(), scaledSize.height())
|
|
|
|
* if (!whole.contains(subset)) {
|
|
|
|
* return false;
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* If the requested colortype/alphatype in pixels is not supported,
|
|
|
|
* or the requested scaledSize is not supported, or the generator encounters an error,
|
|
|
|
* this returns false.
|
|
|
|
*/
|
|
|
|
bool generateScaledPixels(const SkISize& scaledSize, const SkIPoint& subsetOrigin,
|
|
|
|
const SkPixmap& subsetPixels);
|
|
|
|
|
|
|
|
bool generateScaledPixels(const SkPixmap& scaledPixels) {
|
|
|
|
return this->generateScaledPixels(SkISize::Make(scaledPixels.width(),
|
|
|
|
scaledPixels.height()),
|
|
|
|
SkIPoint::Make(0, 0), scaledPixels);
|
|
|
|
}
|
|
|
|
|
2015-01-06 19:30:45 +00:00
|
|
|
/**
|
|
|
|
* If the default image decoder system can interpret the specified (encoded) data, then
|
|
|
|
* this returns a new ImageGenerator for it. Otherwise this returns NULL. Either way
|
|
|
|
* the caller is still responsible for managing their ownership of the data.
|
|
|
|
*/
|
2015-07-09 18:47:36 +00:00
|
|
|
static SkImageGenerator* NewFromEncoded(SkData*);
|
2015-01-06 19:30:45 +00:00
|
|
|
|
2015-08-04 20:53:14 +00:00
|
|
|
/** Return a new image generator backed by the specified picture. If the size is empty or
|
|
|
|
* the picture is NULL, this returns NULL.
|
|
|
|
* The optional matrix and paint arguments are passed to drawPicture() at rasterization
|
|
|
|
* time.
|
|
|
|
*/
|
|
|
|
static SkImageGenerator* NewFromPicture(const SkISize&, const SkPicture*, const SkMatrix*,
|
|
|
|
const SkPaint*);
|
|
|
|
|
2015-09-13 18:03:32 +00:00
|
|
|
bool tryGenerateBitmap(SkBitmap* bm) {
|
2015-09-17 20:35:19 +00:00
|
|
|
return this->tryGenerateBitmap(bm, nullptr, nullptr);
|
2015-09-13 18:03:32 +00:00
|
|
|
}
|
2015-09-17 20:35:19 +00:00
|
|
|
bool tryGenerateBitmap(SkBitmap* bm, const SkImageInfo& info, SkBitmap::Allocator* allocator) {
|
|
|
|
return this->tryGenerateBitmap(bm, &info, allocator);
|
2015-09-13 18:03:32 +00:00
|
|
|
}
|
|
|
|
void generateBitmap(SkBitmap* bm) {
|
2015-09-17 20:35:19 +00:00
|
|
|
if (!this->tryGenerateBitmap(bm, nullptr, nullptr)) {
|
2015-09-13 18:03:32 +00:00
|
|
|
sk_throw();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void generateBitmap(SkBitmap* bm, const SkImageInfo& info) {
|
2015-09-17 20:35:19 +00:00
|
|
|
if (!this->tryGenerateBitmap(bm, &info, nullptr)) {
|
2015-09-13 18:03:32 +00:00
|
|
|
sk_throw();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-29 15:57:20 +00:00
|
|
|
protected:
|
2015-08-13 20:32:39 +00:00
|
|
|
SkImageGenerator(const SkImageInfo& info);
|
2015-03-19 15:31:14 +00:00
|
|
|
|
2016-01-05 17:16:19 +00:00
|
|
|
virtual SkData* onRefEncodedData(SK_REFENCODEDDATA_CTXPARAM);
|
2015-03-19 15:31:14 +00:00
|
|
|
|
2015-07-09 16:08:00 +00:00
|
|
|
virtual bool onGetPixels(const SkImageInfo& info, void* pixels, size_t rowBytes,
|
|
|
|
SkPMColor ctable[], int* ctableCount);
|
2014-07-21 18:37:39 +00:00
|
|
|
virtual bool onGetYUV8Planes(SkISize sizes[3], void* planes[3], size_t rowBytes[3]);
|
2014-09-13 00:45:58 +00:00
|
|
|
virtual bool onGetYUV8Planes(SkISize sizes[3], void* planes[3], size_t rowBytes[3],
|
|
|
|
SkYUVColorSpace* colorSpace);
|
2015-03-19 15:31:14 +00:00
|
|
|
|
2015-10-15 19:14:55 +00:00
|
|
|
virtual GrTexture* onGenerateTexture(GrContext*, const SkIRect*) {
|
2015-08-18 18:16:09 +00:00
|
|
|
return nullptr;
|
|
|
|
}
|
2015-08-13 20:32:39 +00:00
|
|
|
|
2015-12-02 22:19:47 +00:00
|
|
|
virtual bool onComputeScaledDimensions(SkScalar, SupportedSizes*) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
virtual bool onGenerateScaledPixels(const SkISize&, const SkIPoint&, const SkPixmap&) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-09-17 20:35:19 +00:00
|
|
|
bool tryGenerateBitmap(SkBitmap* bm, const SkImageInfo* optionalInfo, SkBitmap::Allocator*);
|
2015-09-13 18:03:32 +00:00
|
|
|
|
2015-03-19 15:31:14 +00:00
|
|
|
private:
|
|
|
|
const SkImageInfo fInfo;
|
2015-08-13 20:32:39 +00:00
|
|
|
const uint32_t fUniqueID;
|
2015-07-09 18:47:36 +00:00
|
|
|
|
2015-07-09 21:08:49 +00:00
|
|
|
// This is our default impl, which may be different on different platforms.
|
|
|
|
// It is called from NewFromEncoded() after it has checked for any runtime factory.
|
|
|
|
// The SkData will never be NULL, as that will have been checked by NewFromEncoded.
|
2015-07-09 18:47:36 +00:00
|
|
|
static SkImageGenerator* NewFromEncodedImpl(SkData*);
|
2013-11-21 15:32:08 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif // SkImageGenerator_DEFINED
|