BitmapRegionDecoder.cpp revision 5827cb5059ed0eec4c73adf1acbd7ee47b2c5c8f
16b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen/* 26b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * Copyright (C) 2010 The Android Open Source Project 36b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * 46b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * Licensed under the Apache License, Version 2.0 (the "License"); 56b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * you may not use this file except in compliance with the License. 66b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * You may obtain a copy of the License at 76b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * 86b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * http://www.apache.org/licenses/LICENSE-2.0 96b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * 106b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * Unless required by applicable law or agreed to in writing, software 116b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * distributed under the License is distributed on an "AS IS" BASIS, 126b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 136b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * See the License for the specific language governing permissions and 146b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * limitations under the License. 156b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen */ 166b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 176b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#define LOG_TAG "BitmapRegionDecoder" 186b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 196b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "SkBitmap.h" 205827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger#include "SkData.h" 216b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "SkImageEncoder.h" 226b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "GraphicsJNI.h" 236b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "SkUtils.h" 246b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "SkTemplates.h" 256b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "SkPixelRef.h" 266b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "SkStream.h" 276b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "BitmapFactory.h" 286b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "AutoDecodeCancel.h" 296b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "CreateJavaOutputStreamAdaptor.h" 306b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "Utils.h" 31a3804cf77f0edd93f6247a055cdafb856b117eecElliott Hughes#include "JNIHelp.h" 32f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin#include "SkTScopedPtr.h" 336b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 346b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include <android_runtime/AndroidRuntime.h> 356b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "android_util_Binder.h" 366b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "android_nio_utils.h" 376b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include "CreateJavaOutputStreamAdaptor.h" 386b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 396b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include <binder/Parcel.h> 406b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include <jni.h> 41b13b9bdad2baf6ad1ec2e56b6b7598fa20f55fc4Mathias Agopian#include <androidfw/Asset.h> 426b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include <sys/stat.h> 436b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 446b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#if 0 456b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen #define TRACE_BITMAP(code) code 466b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#else 476b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen #define TRACE_BITMAP(code) 486b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#endif 496b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 506b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chenusing namespace android; 516b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 525827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenbergerclass SkBitmapRegionDecoder { 535827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenbergerpublic: 545827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkBitmapRegionDecoder(SkImageDecoder* decoder, int width, int height) { 555827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger fDecoder = decoder; 565827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger fWidth = width; 575827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger fHeight = height; 585827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger } 595827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger ~SkBitmapRegionDecoder() { 605827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkDELETE(fDecoder); 615827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger } 625827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger 635827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger bool decodeRegion(SkBitmap* bitmap, const SkIRect& rect, 645827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkBitmap::Config pref, int sampleSize) { 655827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger fDecoder->setSampleSize(sampleSize); 665827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger return fDecoder->decodeRegion(bitmap, rect, pref); 675827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger } 685827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger 695827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkImageDecoder* getDecoder() const { return fDecoder; } 705827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger int getWidth() const { return fWidth; } 715827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger int getHeight() const { return fHeight; } 725827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger 735827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenbergerprivate: 745827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkImageDecoder* fDecoder; 755827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger int fWidth; 765827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger int fHeight; 775827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger}; 785827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger 796b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chenstatic SkMemoryStream* buildSkMemoryStream(SkStream *stream) { 806b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen size_t bufferSize = 4096; 816b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen size_t streamLen = 0; 826b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen size_t len; 836b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen char* data = (char*)sk_malloc_throw(bufferSize); 846b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 856b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen while ((len = stream->read(data + streamLen, 866b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen bufferSize - streamLen)) != 0) { 876b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen streamLen += len; 886b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen if (streamLen == bufferSize) { 896b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen bufferSize *= 2; 906b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen data = (char*)sk_realloc_throw(data, bufferSize); 916b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen } 926b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen } 936b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen data = (char*)sk_realloc_throw(data, streamLen); 9458c1579ce2634de31d24429c1b870d4256ee4f21Wei-Ta Chen 956b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen SkMemoryStream* streamMem = new SkMemoryStream(); 966b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen streamMem->setMemoryOwned(data, streamLen); 976b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen return streamMem; 986b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen} 996b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 1005827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenbergerstatic jobject createBitmapRegionDecoder(JNIEnv* env, SkStream* stream) { 1016b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen SkImageDecoder* decoder = SkImageDecoder::Factory(stream); 1026b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen int width, height; 1036b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen if (NULL == decoder) { 1046b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen doThrowIOE(env, "Image format not supported"); 1056b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen return nullObjectReturn("SkImageDecoder::Factory returned null"); 1066b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen } 1076b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 108e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy JavaPixelAllocator *javaAllocator = new JavaPixelAllocator(env); 1096b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen decoder->setAllocator(javaAllocator); 1106b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen javaAllocator->unref(); 1116b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 1126b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen if (!decoder->buildTileIndex(stream, &width, &height)) { 1136b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen char msg[100]; 1146b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen snprintf(msg, sizeof(msg), "Image failed to decode using %s decoder", 1156b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen decoder->getFormatName()); 1166b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen doThrowIOE(env, msg); 1175827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkDELETE(decoder); 1186b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen return nullObjectReturn("decoder->buildTileIndex returned false"); 1196b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen } 1206b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 1215827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkBitmapRegionDecoder *bm = new SkBitmapRegionDecoder(decoder, width, height); 1226b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen return GraphicsJNI::createBitmapRegionDecoder(env, bm); 1236b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen} 1246b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 1256b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chenstatic jobject nativeNewInstanceFromByteArray(JNIEnv* env, jobject, jbyteArray byteArray, 1266b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen int offset, int length, jboolean isShareable) { 1276b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen /* If isShareable we could decide to just wrap the java array and 1286b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen share it, but that means adding a globalref to the java array object 1296b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen For now we just always copy the array's data if isShareable. 1306b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen */ 1316b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen AutoJavaByteArray ar(env, byteArray); 1326b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen SkStream* stream = new SkMemoryStream(ar.ptr() + offset, length, true); 1335827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger 1345827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger jobject brd = createBitmapRegionDecoder(env, stream); 1355827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkSafeUnref(stream); // the decoder now holds a reference 1365827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger return brd; 1376b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen} 1386b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 1396b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chenstatic jobject nativeNewInstanceFromFileDescriptor(JNIEnv* env, jobject clazz, 1406b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen jobject fileDescriptor, jboolean isShareable) { 1416b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen NPE_CHECK_RETURN_ZERO(env, fileDescriptor); 1426b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 143a3804cf77f0edd93f6247a055cdafb856b117eecElliott Hughes jint descriptor = jniGetFDFromFileDescriptor(env, fileDescriptor); 1445827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger 1456b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen struct stat fdStat; 1466b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen if (fstat(descriptor, &fdStat) == -1) { 1476b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen doThrowIOE(env, "broken file descriptor"); 1486b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen return nullObjectReturn("fstat return -1"); 1496b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen } 1506b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 1515827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkAutoTUnref<SkData> data(SkData::NewFromFD(descriptor)); 1525827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkMemoryStream* stream = new SkMemoryStream(data); 1536b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 1545827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger jobject brd = createBitmapRegionDecoder(env, stream); 1555827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkSafeUnref(stream); // the decoder now holds a reference 1565827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger return brd; 1576b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen} 1586b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 1596b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chenstatic jobject nativeNewInstanceFromStream(JNIEnv* env, jobject clazz, 1606b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen jobject is, // InputStream 1616b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen jbyteArray storage, // byte[] 1626b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen jboolean isShareable) { 1635827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger jobject brd = NULL; 1646b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen SkStream* stream = CreateJavaInputStreamAdaptor(env, is, storage, 1024); 1656b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 1666b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen if (stream) { 1676b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen // for now we don't allow shareable with java inputstreams 1685827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkMemoryStream* mStream = buildSkMemoryStream(stream); 1695827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger brd = createBitmapRegionDecoder(env, mStream); 1705827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkSafeUnref(mStream); // the decoder now holds a reference 1716b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen stream->unref(); 1726b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen } 1735827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger return brd; 1746b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen} 1756b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 1766b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chenstatic jobject nativeNewInstanceFromAsset(JNIEnv* env, jobject clazz, 1776b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen jint native_asset, // Asset 1786b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen jboolean isShareable) { 1796b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen SkStream* stream, *assStream; 1806b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen Asset* asset = reinterpret_cast<Asset*>(native_asset); 1816b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen assStream = new AssetStreamAdaptor(asset); 1826b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen stream = buildSkMemoryStream(assStream); 1836b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen assStream->unref(); 1845827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger 1855827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger jobject brd = createBitmapRegionDecoder(env, stream); 1865827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger SkSafeUnref(stream); // the decoder now holds a reference 1875827cb5059ed0eec4c73adf1acbd7ee47b2c5c8fDerek Sollenberger return brd; 1886b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen} 1896b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 1906b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen/* 1916b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * nine patch not supported 1926b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * 1936b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * purgeable not supported 1946b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen * reportSizeToVM not supported 1956b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen */ 1966b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chenstatic jobject nativeDecodeRegion(JNIEnv* env, jobject, SkBitmapRegionDecoder *brd, 197f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin int start_x, int start_y, int width, int height, jobject options) { 198f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin jobject tileBitmap = NULL; 1996b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen SkImageDecoder *decoder = brd->getDecoder(); 2006b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen int sampleSize = 1; 2016b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen SkBitmap::Config prefConfig = SkBitmap::kNo_Config; 2026b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen bool doDither = true; 203953f9094a2ec14594fa8501d5f3e2d9e300b1b62Wei-Ta Chen bool preferQualityOverSpeed = false; 2046b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2056b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen if (NULL != options) { 2066b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen sampleSize = env->GetIntField(options, gOptions_sampleSizeFieldID); 2076b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen // initialize these, in case we fail later on 2086b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen env->SetIntField(options, gOptions_widthFieldID, -1); 2096b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen env->SetIntField(options, gOptions_heightFieldID, -1); 2106b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen env->SetObjectField(options, gOptions_mimeFieldID, 0); 2116b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2126b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen jobject jconfig = env->GetObjectField(options, gOptions_configFieldID); 2136b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen prefConfig = GraphicsJNI::getNativeBitmapConfig(env, jconfig); 2146b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen doDither = env->GetBooleanField(options, gOptions_ditherFieldID); 215953f9094a2ec14594fa8501d5f3e2d9e300b1b62Wei-Ta Chen preferQualityOverSpeed = env->GetBooleanField(options, 216953f9094a2ec14594fa8501d5f3e2d9e300b1b62Wei-Ta Chen gOptions_preferQualityOverSpeedFieldID); 217f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin // Get the bitmap for re-use if it exists. 218f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin tileBitmap = env->GetObjectField(options, gOptions_bitmapFieldID); 2196b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen } 2206b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2216b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen decoder->setDitherImage(doDither); 222953f9094a2ec14594fa8501d5f3e2d9e300b1b62Wei-Ta Chen decoder->setPreferQualityOverSpeed(preferQualityOverSpeed); 2236b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen AutoDecoderCancel adc(options, decoder); 2246b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2256b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen // To fix the race condition in case "requestCancelDecode" 2266b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen // happens earlier than AutoDecoderCancel object is added 2276b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen // to the gAutoDecoderCancelMutex linked list. 2286b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen if (NULL != options && env->GetBooleanField(options, gOptions_mCancelID)) { 2296b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen return nullObjectReturn("gOptions_mCancelID");; 2306b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen } 2316b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2326b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen SkIRect region; 2336b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen region.fLeft = start_x; 2346b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen region.fTop = start_y; 2356b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen region.fRight = start_x + width; 2366b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen region.fBottom = start_y + height; 237f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin SkBitmap* bitmap = NULL; 238f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin SkTScopedPtr<SkBitmap> adb; 239f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin 240f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin if (tileBitmap != NULL) { 241f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin // Re-use bitmap. 242f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin bitmap = GraphicsJNI::getNativeBitmap(env, tileBitmap); 243f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin } 244f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin if (bitmap == NULL) { 245f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin bitmap = new SkBitmap; 246f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin adb.reset(bitmap); 247f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin } 2486b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2496b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen if (!brd->decodeRegion(bitmap, region, prefConfig, sampleSize)) { 2506b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen return nullObjectReturn("decoder->decodeRegion returned false"); 2516b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen } 2526b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2536b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen // update options (if any) 2546b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen if (NULL != options) { 2556b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen env->SetIntField(options, gOptions_widthFieldID, bitmap->width()); 2566b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen env->SetIntField(options, gOptions_heightFieldID, bitmap->height()); 2576b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen // TODO: set the mimeType field with the data from the codec. 2586b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen // but how to reuse a set of strings, rather than allocating new one 2596b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen // each time? 2606b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen env->SetObjectField(options, gOptions_mimeFieldID, 2616b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen getMimeTypeString(env, decoder->getFormat())); 2626b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen } 2636b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 264f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin if (tileBitmap != NULL) { 265f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin return tileBitmap; 266f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin } 2676b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 268f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin // detach bitmap from its autodeleter, since we want to own it now 269f970c2e6de52ef0da91c3c8f3b48a44303d0eb73Owen Lin adb.release(); 270e4ac2d6b5723c95e648c489b187ddde449452c13Patrick Dubroy 271afde46ed008f150e45e1b0d7e1dc588fc047b74fPatrick Dubroy JavaPixelAllocator* allocator = (JavaPixelAllocator*) decoder->getAllocator(); 272afde46ed008f150e45e1b0d7e1dc588fc047b74fPatrick Dubroy jbyteArray buff = allocator->getStorageObjAndReset(); 273ec4a50428d5f26a22df3edaf7e5b08f41d5cb54bAmith Yamasani return GraphicsJNI::createBitmap(env, bitmap, buff, false, NULL, NULL, -1); 2746b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen} 2756b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2766b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chenstatic int nativeGetHeight(JNIEnv* env, jobject, SkBitmapRegionDecoder *brd) { 2776b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen return brd->getHeight(); 2786b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen} 2796b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2806b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chenstatic int nativeGetWidth(JNIEnv* env, jobject, SkBitmapRegionDecoder *brd) { 2816b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen return brd->getWidth(); 2826b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen} 2836b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2846b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chenstatic void nativeClean(JNIEnv* env, jobject, SkBitmapRegionDecoder *brd) { 2856b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen delete brd; 2866b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen} 2876b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2886b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen/////////////////////////////////////////////////////////////////////////////// 2896b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2906b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#include <android_runtime/AndroidRuntime.h> 2916b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2926b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chenstatic JNINativeMethod gBitmapRegionDecoderMethods[] = { 2936b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen { "nativeDecodeRegion", 2946b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen "(IIIIILandroid/graphics/BitmapFactory$Options;)Landroid/graphics/Bitmap;", 2956b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen (void*)nativeDecodeRegion}, 2966b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2976b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen { "nativeGetHeight", "(I)I", (void*)nativeGetHeight}, 2986b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 2996b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen { "nativeGetWidth", "(I)I", (void*)nativeGetWidth}, 3006b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 3016b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen { "nativeClean", "(I)V", (void*)nativeClean}, 3026b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 3036b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen { "nativeNewInstance", 3046b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen "([BIIZ)Landroid/graphics/BitmapRegionDecoder;", 3056b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen (void*)nativeNewInstanceFromByteArray 3066b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen }, 3076b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 3086b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen { "nativeNewInstance", 3096b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen "(Ljava/io/InputStream;[BZ)Landroid/graphics/BitmapRegionDecoder;", 3106b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen (void*)nativeNewInstanceFromStream 3116b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen }, 3126b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 3136b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen { "nativeNewInstance", 3146b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen "(Ljava/io/FileDescriptor;Z)Landroid/graphics/BitmapRegionDecoder;", 3156b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen (void*)nativeNewInstanceFromFileDescriptor 3166b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen }, 3176b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 3186b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen { "nativeNewInstance", 3196b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen "(IZ)Landroid/graphics/BitmapRegionDecoder;", 3206b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen (void*)nativeNewInstanceFromAsset 3216b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen }, 3226b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen}; 3236b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 3246b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen#define kClassPathName "android/graphics/BitmapRegionDecoder" 3256b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen 3266b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chenint register_android_graphics_BitmapRegionDecoder(JNIEnv* env) 3276b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen{ 3286b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen return android::AndroidRuntime::registerNativeMethods(env, kClassPathName, 3296b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen gBitmapRegionDecoderMethods, SK_ARRAY_COUNT(gBitmapRegionDecoderMethods)); 3306b849e2123be98eb2a1a25b8abf0b13a279ce952Wei-Ta Chen} 331