/* * Copyright 2015 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "SkCodecImageGenerator.h" SkImageGenerator* SkCodecImageGenerator::NewFromEncodedCodec(SkData* data) { SkCodec* codec = SkCodec::NewFromData(data); if (nullptr == codec) { return nullptr; } return new SkCodecImageGenerator(codec, data); } static SkImageInfo make_premul(const SkImageInfo& info) { if (kUnpremul_SkAlphaType == info.alphaType()) { return info.makeAlphaType(kPremul_SkAlphaType); } return info; } SkCodecImageGenerator::SkCodecImageGenerator(SkCodec* codec, SkData* data) : INHERITED(make_premul(codec->getInfo())) , fCodec(codec) , fData(SkRef(data)) , fYWidth(0) , fUWidth(0) , fVWidth(0) {} SkData* SkCodecImageGenerator::onRefEncodedData(SK_REFENCODEDDATA_CTXPARAM) { return SkRef(fData.get()); } bool SkCodecImageGenerator::onGetPixels(const SkImageInfo& info, void* pixels, size_t rowBytes, SkPMColor ctable[], int* ctableCount) { SkCodec::Result result = fCodec->getPixels(info, pixels, rowBytes, nullptr, ctable, ctableCount); switch (result) { case SkCodec::kSuccess: case SkCodec::kIncompleteInput: return true; default: return false; } } bool SkCodecImageGenerator::onGetYUV8Planes(SkISize sizes[3], void* planes[3], size_t rowBytes[3], SkYUVColorSpace* colorSpace) { // TODO (msarett): Change the YUV API in ImageGenerator to match SkCodec. // This function is currently a hack to match the implementation // in SkCodec with the old API. SkCodec::YUVSizeInfo sizeInfo; // If planes is NULL, we just need to return the size. if (nullptr == planes) { bool result = fCodec->queryYUV8(&sizeInfo, colorSpace); if (result) { // Save the true widths fYWidth = sizeInfo.fYSize.width(); fUWidth = sizeInfo.fUSize.width(); fVWidth = sizeInfo.fVSize.width(); // Set the sizes so that the client allocates enough memory sizes[0].fWidth = (int) sizeInfo.fYWidthBytes; sizes[0].fHeight = sizeInfo.fYSize.height(); sizes[1].fWidth = (int) sizeInfo.fUWidthBytes; sizes[1].fHeight = sizeInfo.fUSize.height(); sizes[2].fWidth = (int) sizeInfo.fVWidthBytes; sizes[2].fHeight = sizeInfo.fVSize.height(); } return result; } // Set the sizeInfo with the true widths and heights SkASSERT(fYWidth != 0 && fUWidth != 0 && fVWidth != 0); sizeInfo.fYSize.set(fYWidth, sizes[0].height()); sizeInfo.fUSize.set(fUWidth, sizes[1].height()); sizeInfo.fVSize.set(fVWidth, sizes[2].height()); // Set the sizeInfo with the allocated widths sizeInfo.fYWidthBytes = sizes[0].width(); sizeInfo.fUWidthBytes = sizes[1].width(); sizeInfo.fVWidthBytes = sizes[2].width(); SkCodec::Result result = fCodec->getYUV8Planes(sizeInfo, planes); if ((result == SkCodec::kSuccess || result == SkCodec::kIncompleteInput) && colorSpace) { *colorSpace = kJPEG_SkYUVColorSpace; } switch (result) { case SkCodec::kSuccess: case SkCodec::kIncompleteInput: return true; default: return false; } }