/* * Copyright (C) 2017 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "Bitmap.h" #include "BitmapFactory.h" #include "ByteBufferStreamAdaptor.h" #include "CreateJavaOutputStreamAdaptor.h" #include "GraphicsJNI.h" #include "ImageDecoder.h" #include "Utils.h" #include "core_jni_helpers.h" #include #include #include #include #include #include #include #include #include using namespace android; static jclass gImageDecoder_class; static jclass gSize_class; static jclass gDecodeException_class; static jclass gCanvas_class; static jmethodID gImageDecoder_constructorMethodID; static jmethodID gImageDecoder_postProcessMethodID; static jmethodID gSize_constructorMethodID; static jmethodID gDecodeException_constructorMethodID; static jmethodID gCallback_onPartialImageMethodID; static jmethodID gCanvas_constructorMethodID; static jmethodID gCanvas_releaseMethodID; // Clear and return any pending exception for handling other than throwing directly. static jthrowable get_and_clear_exception(JNIEnv* env) { jthrowable jexception = env->ExceptionOccurred(); if (jexception) { env->ExceptionClear(); } return jexception; } // Throw a new ImageDecoder.DecodeException. Returns null for convenience. static jobject throw_exception(JNIEnv* env, ImageDecoder::Error error, const char* msg, jthrowable cause, jobject source) { jstring jstr = nullptr; if (msg) { jstr = env->NewStringUTF(msg); if (!jstr) { // Out of memory. return nullptr; } } jthrowable exception = (jthrowable) env->NewObject(gDecodeException_class, gDecodeException_constructorMethodID, error, jstr, cause, source); // Only throw if not out of memory. if (exception) { env->Throw(exception); } return nullptr; } static jobject native_create(JNIEnv* env, std::unique_ptr stream, jobject source) { if (!stream.get()) { return throw_exception(env, ImageDecoder::kSourceMalformedData, "Failed to create a stream", nullptr, source); } std::unique_ptr decoder(new ImageDecoder); SkCodec::Result result; auto codec = SkCodec::MakeFromStream(std::move(stream), &result, decoder->mPeeker.get()); if (jthrowable jexception = get_and_clear_exception(env)) { return throw_exception(env, ImageDecoder::kSourceException, "", jexception, source); } if (!codec) { switch (result) { case SkCodec::kIncompleteInput: return throw_exception(env, ImageDecoder::kSourceIncomplete, "", nullptr, source); default: SkString msg; msg.printf("Failed to create image decoder with message '%s'", SkCodec::ResultToString(result)); return throw_exception(env, ImageDecoder::kSourceMalformedData, msg.c_str(), nullptr, source); } } const bool animated = codec->getFrameCount() > 1; if (jthrowable jexception = get_and_clear_exception(env)) { return throw_exception(env, ImageDecoder::kSourceException, "", jexception, source); } decoder->mCodec = SkAndroidCodec::MakeFromCodec(std::move(codec), SkAndroidCodec::ExifOrientationBehavior::kRespect); if (!decoder->mCodec.get()) { return throw_exception(env, ImageDecoder::kSourceMalformedData, "", nullptr, source); } const auto& info = decoder->mCodec->getInfo(); const int width = info.width(); const int height = info.height(); const bool isNinePatch = decoder->mPeeker->mPatch != nullptr; return env->NewObject(gImageDecoder_class, gImageDecoder_constructorMethodID, reinterpret_cast(decoder.release()), width, height, animated, isNinePatch); } static jobject ImageDecoder_nCreateFd(JNIEnv* env, jobject /*clazz*/, jobject fileDescriptor, jobject source) { int descriptor = jniGetFDFromFileDescriptor(env, fileDescriptor); struct stat fdStat; if (fstat(descriptor, &fdStat) == -1) { return throw_exception(env, ImageDecoder::kSourceMalformedData, "broken file descriptor; fstat returned -1", nullptr, source); } int dupDescriptor = fcntl(descriptor, F_DUPFD_CLOEXEC, 0); FILE* file = fdopen(dupDescriptor, "r"); if (file == NULL) { close(dupDescriptor); return throw_exception(env, ImageDecoder::kSourceMalformedData, "Could not open file", nullptr, source); } std::unique_ptr fileStream(new SkFILEStream(file)); return native_create(env, std::move(fileStream), source); } static jobject ImageDecoder_nCreateInputStream(JNIEnv* env, jobject /*clazz*/, jobject is, jbyteArray storage, jobject source) { std::unique_ptr stream(CreateJavaInputStreamAdaptor(env, is, storage, false)); if (!stream.get()) { return throw_exception(env, ImageDecoder::kSourceMalformedData, "Failed to create a stream", nullptr, source); } std::unique_ptr bufferedStream( SkFrontBufferedStream::Make(std::move(stream), SkCodec::MinBufferedBytesNeeded())); return native_create(env, std::move(bufferedStream), source); } static jobject ImageDecoder_nCreateAsset(JNIEnv* env, jobject /*clazz*/, jlong assetPtr, jobject source) { Asset* asset = reinterpret_cast(assetPtr); std::unique_ptr stream(new AssetStreamAdaptor(asset)); return native_create(env, std::move(stream), source); } static jobject ImageDecoder_nCreateByteBuffer(JNIEnv* env, jobject /*clazz*/, jobject jbyteBuffer, jint initialPosition, jint limit, jobject source) { std::unique_ptr stream = CreateByteBufferStreamAdaptor(env, jbyteBuffer, initialPosition, limit); if (!stream) { return throw_exception(env, ImageDecoder::kSourceMalformedData, "Failed to read ByteBuffer", nullptr, source); } return native_create(env, std::move(stream), source); } static jobject ImageDecoder_nCreateByteArray(JNIEnv* env, jobject /*clazz*/, jbyteArray byteArray, jint offset, jint length, jobject source) { std::unique_ptr stream(CreateByteArrayStreamAdaptor(env, byteArray, offset, length)); return native_create(env, std::move(stream), source); } jint postProcessAndRelease(JNIEnv* env, jobject jimageDecoder, std::unique_ptr canvas) { jobject jcanvas = env->NewObject(gCanvas_class, gCanvas_constructorMethodID, reinterpret_cast(canvas.get())); if (!jcanvas) { doThrowOOME(env, "Failed to create Java Canvas for PostProcess!"); return ImageDecoder::kUnknown; } // jcanvas now owns canvas. canvas.release(); return env->CallIntMethod(jimageDecoder, gImageDecoder_postProcessMethodID, jcanvas); } static jobject ImageDecoder_nDecodeBitmap(JNIEnv* env, jobject /*clazz*/, jlong nativePtr, jobject jdecoder, jboolean jpostProcess, jint desiredWidth, jint desiredHeight, jobject jsubset, jboolean requireMutable, jint allocator, jboolean requireUnpremul, jboolean preferRamOverQuality, jboolean asAlphaMask, jlong colorSpaceHandle, jboolean extended) { auto* decoder = reinterpret_cast(nativePtr); SkAndroidCodec* codec = decoder->mCodec.get(); const SkISize desiredSize = SkISize::Make(desiredWidth, desiredHeight); SkISize decodeSize = desiredSize; const int sampleSize = codec->computeSampleSize(&decodeSize); const bool scale = desiredSize != decodeSize; SkImageInfo decodeInfo = codec->getInfo().makeWH(decodeSize.width(), decodeSize.height()); if (scale && requireUnpremul && kOpaque_SkAlphaType != decodeInfo.alphaType()) { doThrowISE(env, "Cannot scale unpremultiplied pixels!"); return nullptr; } switch (decodeInfo.alphaType()) { case kUnpremul_SkAlphaType: if (!requireUnpremul) { decodeInfo = decodeInfo.makeAlphaType(kPremul_SkAlphaType); } break; case kPremul_SkAlphaType: if (requireUnpremul) { decodeInfo = decodeInfo.makeAlphaType(kUnpremul_SkAlphaType); } break; case kOpaque_SkAlphaType: break; case kUnknown_SkAlphaType: doThrowIOE(env, "Unknown alpha type"); return nullptr; } SkColorType colorType = kN32_SkColorType; if (asAlphaMask && decodeInfo.colorType() == kGray_8_SkColorType) { // We have to trick Skia to decode this to a single channel. colorType = kGray_8_SkColorType; } else if (preferRamOverQuality) { // FIXME: The post-process might add alpha, which would make a 565 // result incorrect. If we call the postProcess before now and record // to a picture, we can know whether alpha was added, and if not, we // can still use 565. if (decodeInfo.alphaType() == kOpaque_SkAlphaType && !jpostProcess) { // If the final result will be hardware, decoding to 565 and then // uploading to the gpu as 8888 will not save memory. This still // may save us from using F16, but do not go down to 565. if (allocator != ImageDecoder::kHardware_Allocator && (allocator != ImageDecoder::kDefault_Allocator || requireMutable)) { colorType = kRGB_565_SkColorType; } } // Otherwise, stick with N32 } else if (extended) { colorType = kRGBA_F16_SkColorType; } else { colorType = codec->computeOutputColorType(colorType); } const bool isHardware = !requireMutable && (allocator == ImageDecoder::kDefault_Allocator || allocator == ImageDecoder::kHardware_Allocator) && colorType != kGray_8_SkColorType; if (colorType == kRGBA_F16_SkColorType && isHardware && !uirenderer::HardwareBitmapUploader::hasFP16Support()) { colorType = kN32_SkColorType; } sk_sp colorSpace = GraphicsJNI::getNativeColorSpace(colorSpaceHandle); colorSpace = codec->computeOutputColorSpace(colorType, colorSpace); decodeInfo = decodeInfo.makeColorType(colorType).makeColorSpace(colorSpace); SkBitmap bm; auto bitmapInfo = decodeInfo; if (asAlphaMask && colorType == kGray_8_SkColorType) { bitmapInfo = bitmapInfo.makeColorType(kAlpha_8_SkColorType); } if (!bm.setInfo(bitmapInfo)) { doThrowIOE(env, "Failed to setInfo properly"); return nullptr; } sk_sp nativeBitmap; // If we are going to scale or subset, we will create a new bitmap later on, // so use the heap for the temporary. // FIXME: Use scanline decoding on only a couple lines to save memory. b/70709380. if (allocator == ImageDecoder::kSharedMemory_Allocator && !scale && !jsubset) { nativeBitmap = Bitmap::allocateAshmemBitmap(&bm); } else { nativeBitmap = Bitmap::allocateHeapBitmap(&bm); } if (!nativeBitmap) { SkString msg; msg.printf("OOM allocating Bitmap with dimensions %i x %i", decodeInfo.width(), decodeInfo.height()); doThrowOOME(env, msg.c_str()); return nullptr; } SkAndroidCodec::AndroidOptions options; options.fSampleSize = sampleSize; auto result = codec->getAndroidPixels(decodeInfo, bm.getPixels(), bm.rowBytes(), &options); jthrowable jexception = get_and_clear_exception(env); int onPartialImageError = jexception ? ImageDecoder::kSourceException : 0; // No error. switch (result) { case SkCodec::kSuccess: // Ignore the exception, since the decode was successful anyway. jexception = nullptr; onPartialImageError = 0; break; case SkCodec::kIncompleteInput: if (!jexception) { onPartialImageError = ImageDecoder::kSourceIncomplete; } break; case SkCodec::kErrorInInput: if (!jexception) { onPartialImageError = ImageDecoder::kSourceMalformedData; } break; default: SkString msg; msg.printf("getPixels failed with error %s", SkCodec::ResultToString(result)); doThrowIOE(env, msg.c_str()); return nullptr; } if (onPartialImageError) { env->CallVoidMethod(jdecoder, gCallback_onPartialImageMethodID, onPartialImageError, jexception); if (env->ExceptionCheck()) { return nullptr; } } jbyteArray ninePatchChunk = nullptr; jobject ninePatchInsets = nullptr; // Ignore ninepatch when post-processing. if (!jpostProcess) { // FIXME: Share more code with BitmapFactory.cpp. if (decoder->mPeeker->mPatch != nullptr) { size_t ninePatchArraySize = decoder->mPeeker->mPatch->serializedSize(); ninePatchChunk = env->NewByteArray(ninePatchArraySize); if (ninePatchChunk == nullptr) { doThrowOOME(env, "Failed to allocate nine patch chunk."); return nullptr; } env->SetByteArrayRegion(ninePatchChunk, 0, decoder->mPeeker->mPatchSize, reinterpret_cast(decoder->mPeeker->mPatch)); } if (decoder->mPeeker->mHasInsets) { ninePatchInsets = decoder->mPeeker->createNinePatchInsets(env, 1.0f); if (ninePatchInsets == nullptr) { doThrowOOME(env, "Failed to allocate nine patch insets."); return nullptr; } } } if (scale || jsubset) { int translateX = 0; int translateY = 0; if (jsubset) { SkIRect subset; GraphicsJNI::jrect_to_irect(env, jsubset, &subset); translateX = -subset.fLeft; translateY = -subset.fTop; desiredWidth = subset.width(); desiredHeight = subset.height(); } SkImageInfo scaledInfo = bitmapInfo.makeWH(desiredWidth, desiredHeight); SkBitmap scaledBm; if (!scaledBm.setInfo(scaledInfo)) { doThrowIOE(env, "Failed scaled setInfo"); return nullptr; } sk_sp scaledPixelRef; if (allocator == ImageDecoder::kSharedMemory_Allocator) { scaledPixelRef = Bitmap::allocateAshmemBitmap(&scaledBm); } else { scaledPixelRef = Bitmap::allocateHeapBitmap(&scaledBm); } if (!scaledPixelRef) { SkString msg; msg.printf("OOM allocating scaled Bitmap with dimensions %i x %i", desiredWidth, desiredHeight); doThrowOOME(env, msg.c_str()); return nullptr; } SkPaint paint; paint.setBlendMode(SkBlendMode::kSrc); paint.setFilterQuality(kLow_SkFilterQuality); // bilinear filtering SkCanvas canvas(scaledBm, SkCanvas::ColorBehavior::kLegacy); canvas.translate(translateX, translateY); if (scale) { float scaleX = (float) desiredWidth / decodeInfo.width(); float scaleY = (float) desiredHeight / decodeInfo.height(); canvas.scale(scaleX, scaleY); } canvas.drawBitmap(bm, 0.0f, 0.0f, &paint); bm.swap(scaledBm); nativeBitmap = std::move(scaledPixelRef); } if (jpostProcess) { std::unique_ptr canvas(Canvas::create_canvas(bm)); jint pixelFormat = postProcessAndRelease(env, jdecoder, std::move(canvas)); if (env->ExceptionCheck()) { return nullptr; } SkAlphaType newAlphaType = bm.alphaType(); switch (pixelFormat) { case ImageDecoder::kUnknown: break; case ImageDecoder::kTranslucent: newAlphaType = kPremul_SkAlphaType; break; case ImageDecoder::kOpaque: newAlphaType = kOpaque_SkAlphaType; break; default: SkString msg; msg.printf("invalid return from postProcess: %i", pixelFormat); doThrowIAE(env, msg.c_str()); return nullptr; } if (newAlphaType != bm.alphaType()) { if (!bm.setAlphaType(newAlphaType)) { SkString msg; msg.printf("incompatible return from postProcess: %i", pixelFormat); doThrowIAE(env, msg.c_str()); return nullptr; } nativeBitmap->setAlphaType(newAlphaType); } } int bitmapCreateFlags = 0x0; if (!requireUnpremul) { // Even if the image is opaque, setting this flag means that // if alpha is added (e.g. by PostProcess), it will be marked as // premultiplied. bitmapCreateFlags |= bitmap::kBitmapCreateFlag_Premultiplied; } if (requireMutable) { bitmapCreateFlags |= bitmap::kBitmapCreateFlag_Mutable; } else { if (isHardware) { sk_sp hwBitmap = Bitmap::allocateHardwareBitmap(bm); if (hwBitmap) { hwBitmap->setImmutable(); return bitmap::createBitmap(env, hwBitmap.release(), bitmapCreateFlags, ninePatchChunk, ninePatchInsets); } if (allocator == ImageDecoder::kHardware_Allocator) { doThrowOOME(env, "failed to allocate hardware Bitmap!"); return nullptr; } // If we failed to create a hardware bitmap, go ahead and create a // software one. } nativeBitmap->setImmutable(); } return bitmap::createBitmap(env, nativeBitmap.release(), bitmapCreateFlags, ninePatchChunk, ninePatchInsets); } static jobject ImageDecoder_nGetSampledSize(JNIEnv* env, jobject /*clazz*/, jlong nativePtr, jint sampleSize) { auto* decoder = reinterpret_cast(nativePtr); SkISize size = decoder->mCodec->getSampledDimensions(sampleSize); return env->NewObject(gSize_class, gSize_constructorMethodID, size.width(), size.height()); } static void ImageDecoder_nGetPadding(JNIEnv* env, jobject /*clazz*/, jlong nativePtr, jobject outPadding) { auto* decoder = reinterpret_cast(nativePtr); decoder->mPeeker->getPadding(env, outPadding); } static void ImageDecoder_nClose(JNIEnv* /*env*/, jobject /*clazz*/, jlong nativePtr) { delete reinterpret_cast(nativePtr); } static jstring ImageDecoder_nGetMimeType(JNIEnv* env, jobject /*clazz*/, jlong nativePtr) { auto* decoder = reinterpret_cast(nativePtr); return encodedFormatToString(env, decoder->mCodec->getEncodedFormat()); } static jobject ImageDecoder_nGetColorSpace(JNIEnv* env, jobject /*clazz*/, jlong nativePtr) { auto* codec = reinterpret_cast(nativePtr)->mCodec.get(); auto colorType = codec->computeOutputColorType(kN32_SkColorType); sk_sp colorSpace = codec->computeOutputColorSpace(colorType); return GraphicsJNI::getColorSpace(env, colorSpace.get(), colorType); } static const JNINativeMethod gImageDecoderMethods[] = { { "nCreate", "(JLandroid/graphics/ImageDecoder$Source;)Landroid/graphics/ImageDecoder;", (void*) ImageDecoder_nCreateAsset }, { "nCreate", "(Ljava/nio/ByteBuffer;IILandroid/graphics/ImageDecoder$Source;)Landroid/graphics/ImageDecoder;", (void*) ImageDecoder_nCreateByteBuffer }, { "nCreate", "([BIILandroid/graphics/ImageDecoder$Source;)Landroid/graphics/ImageDecoder;", (void*) ImageDecoder_nCreateByteArray }, { "nCreate", "(Ljava/io/InputStream;[BLandroid/graphics/ImageDecoder$Source;)Landroid/graphics/ImageDecoder;", (void*) ImageDecoder_nCreateInputStream }, { "nCreate", "(Ljava/io/FileDescriptor;Landroid/graphics/ImageDecoder$Source;)Landroid/graphics/ImageDecoder;", (void*) ImageDecoder_nCreateFd }, { "nDecodeBitmap", "(JLandroid/graphics/ImageDecoder;ZIILandroid/graphics/Rect;ZIZZZJZ)Landroid/graphics/Bitmap;", (void*) ImageDecoder_nDecodeBitmap }, { "nGetSampledSize","(JI)Landroid/util/Size;", (void*) ImageDecoder_nGetSampledSize }, { "nGetPadding", "(JLandroid/graphics/Rect;)V", (void*) ImageDecoder_nGetPadding }, { "nClose", "(J)V", (void*) ImageDecoder_nClose}, { "nGetMimeType", "(J)Ljava/lang/String;", (void*) ImageDecoder_nGetMimeType }, { "nGetColorSpace", "(J)Landroid/graphics/ColorSpace;", (void*) ImageDecoder_nGetColorSpace }, }; int register_android_graphics_ImageDecoder(JNIEnv* env) { gImageDecoder_class = MakeGlobalRefOrDie(env, FindClassOrDie(env, "android/graphics/ImageDecoder")); gImageDecoder_constructorMethodID = GetMethodIDOrDie(env, gImageDecoder_class, "", "(JIIZZ)V"); gImageDecoder_postProcessMethodID = GetMethodIDOrDie(env, gImageDecoder_class, "postProcessAndRelease", "(Landroid/graphics/Canvas;)I"); gSize_class = MakeGlobalRefOrDie(env, FindClassOrDie(env, "android/util/Size")); gSize_constructorMethodID = GetMethodIDOrDie(env, gSize_class, "", "(II)V"); gDecodeException_class = MakeGlobalRefOrDie(env, FindClassOrDie(env, "android/graphics/ImageDecoder$DecodeException")); gDecodeException_constructorMethodID = GetMethodIDOrDie(env, gDecodeException_class, "", "(ILjava/lang/String;Ljava/lang/Throwable;Landroid/graphics/ImageDecoder$Source;)V"); gCallback_onPartialImageMethodID = GetMethodIDOrDie(env, gImageDecoder_class, "onPartialImage", "(ILjava/lang/Throwable;)V"); gCanvas_class = MakeGlobalRefOrDie(env, FindClassOrDie(env, "android/graphics/Canvas")); gCanvas_constructorMethodID = GetMethodIDOrDie(env, gCanvas_class, "", "(J)V"); gCanvas_releaseMethodID = GetMethodIDOrDie(env, gCanvas_class, "release", "()V"); return android::RegisterMethodsOrDie(env, "android/graphics/ImageDecoder", gImageDecoderMethods, NELEM(gImageDecoderMethods)); }