android_view_Surface.cpp revision 87eac99a21772ae56018cb81db6966557b459554
19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* 29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2007 The Android Open Source Project 39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License"); 59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License. 69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at 79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * http://www.apache.org/licenses/LICENSE-2.0 99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software 119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS, 129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and 149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License. 159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 17fae5cb2b356a1fef172b43066180a7ab4c32dbacMathias Agopian#define LOG_TAG "Surface" 18fae5cb2b356a1fef172b43066180a7ab4c32dbacMathias Agopian 199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <stdio.h> 209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 213866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian#include "jni.h" 223866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian#include "JNIHelp.h" 2364a55af0ac700baecb0877235eb42caac59a3560Jeff Brown#include "android_os_Parcel.h" 24a111187a5b4dc5d287e29ced7d65e8d20edca140Dianne Hackborn#include "android/graphics/GraphicsJNI.h" 259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 263866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian#include <android_runtime/AndroidRuntime.h> 273866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian#include <android_runtime/android_view_Surface.h> 283866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian#include <android_runtime/android_graphics_SurfaceTexture.h> 2987eac99a21772ae56018cb81db6966557b459554Ruben Brunk#include <android_runtime/Log.h> 308335f1ccccedb6655d96d9d5b697a7f0938235ddMathias Agopian 314a05f436989a49e06ec762aac9507ad13fcc5e23Mathias Agopian#include <binder/Parcel.h> 324a05f436989a49e06ec762aac9507ad13fcc5e23Mathias Agopian 338335f1ccccedb6655d96d9d5b697a7f0938235ddMathias Agopian#include <gui/Surface.h> 3452800617946c456e78ed010c82d0ec4358368164Mathias Agopian#include <gui/SurfaceControl.h> 35d47f7d8b5fe3a3861d7cbdc5f912235407823c8eAndy McFadden#include <gui/GLConsumer.h> 368335f1ccccedb6655d96d9d5b697a7f0938235ddMathias Agopian 379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <ui/Rect.h> 388335f1ccccedb6655d96d9d5b697a7f0938235ddMathias Agopian#include <ui/Region.h> 399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <SkCanvas.h> 419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <SkBitmap.h> 426158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian#include <SkRegion.h> 439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/misc.h> 459e316a1a2a8d734315bbd56a85308f9657a92913Jeff Brown#include <utils/Log.h> 469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4764a55af0ac700baecb0877235eb42caac59a3560Jeff Brown#include <ScopedUtfChars.h> 4864a55af0ac700baecb0877235eb42caac59a3560Jeff Brown 499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------- 509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectnamespace android { 529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic const char* const OutOfResourcesException = 549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project "android/view/Surface$OutOfResourcesException"; 559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5664a55af0ac700baecb0877235eb42caac59a3560Jeff Brownstatic struct { 5764a55af0ac700baecb0877235eb42caac59a3560Jeff Brown jclass clazz; 583866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian jfieldID mNativeObject; 59fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown jfieldID mLock; 60cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown jmethodID ctor; 6164a55af0ac700baecb0877235eb42caac59a3560Jeff Brown} gSurfaceClassInfo; 6264a55af0ac700baecb0877235eb42caac59a3560Jeff Brown 6364a55af0ac700baecb0877235eb42caac59a3560Jeff Brownstatic struct { 6464a55af0ac700baecb0877235eb42caac59a3560Jeff Brown jfieldID left; 6564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown jfieldID top; 6664a55af0ac700baecb0877235eb42caac59a3560Jeff Brown jfieldID right; 6764a55af0ac700baecb0877235eb42caac59a3560Jeff Brown jfieldID bottom; 6864a55af0ac700baecb0877235eb42caac59a3560Jeff Brown} gRectClassInfo; 6964a55af0ac700baecb0877235eb42caac59a3560Jeff Brown 7064a55af0ac700baecb0877235eb42caac59a3560Jeff Brownstatic struct { 71fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger jfieldID mFinalizer; 7264a55af0ac700baecb0877235eb42caac59a3560Jeff Brown jfieldID mNativeCanvas; 7364a55af0ac700baecb0877235eb42caac59a3560Jeff Brown jfieldID mSurfaceFormat; 7464a55af0ac700baecb0877235eb42caac59a3560Jeff Brown} gCanvasClassInfo; 7564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown 7664a55af0ac700baecb0877235eb42caac59a3560Jeff Brownstatic struct { 77fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger jfieldID mNativeCanvas; 78fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger} gCanvasFinalizerClassInfo; 79fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger 809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------- 819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 82b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopian// this is just a pointer we use to pass to inc/decStrong 83b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopianstatic const void *sRefBaseOwner; 84b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopian 8564a55af0ac700baecb0877235eb42caac59a3560Jeff Brownbool android_view_Surface_isInstanceOf(JNIEnv* env, jobject obj) { 8664a55af0ac700baecb0877235eb42caac59a3560Jeff Brown return env->IsInstanceOf(obj, gSurfaceClassInfo.clazz); 87050316184b01c0d1a01c46afae7429b89a27c31btedbo} 88050316184b01c0d1a01c46afae7429b89a27c31btedbo 893866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopiansp<ANativeWindow> android_view_Surface_getNativeWindow(JNIEnv* env, jobject surfaceObj) { 903866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian return android_view_Surface_getSurface(env, surfaceObj); 919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 933866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopiansp<Surface> android_view_Surface_getSurface(JNIEnv* env, jobject surfaceObj) { 947c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian sp<Surface> sur; 957c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian jobject lock = env->GetObjectField(surfaceObj, 96fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown gSurfaceClassInfo.mLock); 977c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian if (env->MonitorEnter(lock) == JNI_OK) { 987c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian sur = reinterpret_cast<Surface *>( 997c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian env->GetIntField(surfaceObj, gSurfaceClassInfo.mNativeObject)); 1007c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian env->MonitorExit(lock); 1017c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian } 1027c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian return sur; 103cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown} 104cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown 10529479ebe1007361222bf6ab4d5e2a27927d4b8e8Mathias Agopianjobject android_view_Surface_createFromIGraphicBufferProducer(JNIEnv* env, 106d47f7d8b5fe3a3861d7cbdc5f912235407823c8eAndy McFadden const sp<IGraphicBufferProducer>& bufferProducer) { 107d47f7d8b5fe3a3861d7cbdc5f912235407823c8eAndy McFadden if (bufferProducer == NULL) { 108cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown return NULL; 109cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown } 110cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown 111e32632682ca9207bd247ca27012cf670b5c23f54Mathias Agopian sp<Surface> surface(new Surface(bufferProducer, true)); 112cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown if (surface == NULL) { 113cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown return NULL; 114cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown } 115cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown 1163866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian jobject surfaceObj = env->NewObject(gSurfaceClassInfo.clazz, gSurfaceClassInfo.ctor, surface.get()); 117cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown if (surfaceObj == NULL) { 118cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown if (env->ExceptionCheck()) { 119d47f7d8b5fe3a3861d7cbdc5f912235407823c8eAndy McFadden ALOGE("Could not create instance of Surface from IGraphicBufferProducer."); 120cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown LOGE_EX(env); 121cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown env->ExceptionClear(); 122cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown } 123cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown return NULL; 124cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown } 125b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopian surface->incStrong(&sRefBaseOwner); 126cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown return surfaceObj; 127cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown} 128cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown 1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------- 1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 131f3d3b4d80618d498fb669c8bfdc492ce9ea25691Mathias Agopianstatic inline bool isSurfaceValid(const sp<Surface>& sur) { 132f3d3b4d80618d498fb669c8bfdc492ce9ea25691Mathias Agopian return Surface::isValid(sur); 13352800617946c456e78ed010c82d0ec4358368164Mathias Agopian} 13452800617946c456e78ed010c82d0ec4358368164Mathias Agopian 13552800617946c456e78ed010c82d0ec4358368164Mathias Agopian// ---------------------------------------------------------------------------- 13652800617946c456e78ed010c82d0ec4358368164Mathias Agopian 13752800617946c456e78ed010c82d0ec4358368164Mathias Agopianstatic jint nativeCreateFromSurfaceTexture(JNIEnv* env, jclass clazz, 13864a55af0ac700baecb0877235eb42caac59a3560Jeff Brown jobject surfaceTextureObj) { 13952a9a10b6b8c7b7a9f97777541841b94d4fd9754Mathias Agopian sp<IGraphicBufferProducer> producer(SurfaceTexture_getProducer(env, surfaceTextureObj)); 14052a9a10b6b8c7b7a9f97777541841b94d4fd9754Mathias Agopian if (producer == NULL) { 1412e76c992bfa955cb637417005941e3e93810a99dDaniel Lam jniThrowException(env, "java/lang/IllegalArgumentException", 1422e76c992bfa955cb637417005941e3e93810a99dDaniel Lam "SurfaceTexture has already been released"); 1433866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian return 0; 1442e76c992bfa955cb637417005941e3e93810a99dDaniel Lam } 14564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown 14652a9a10b6b8c7b7a9f97777541841b94d4fd9754Mathias Agopian sp<Surface> surface(new Surface(producer, true)); 1470de171b0d490a5928d54d2fb67c912d140aac643Ted Bonkenburg if (surface == NULL) { 1480de171b0d490a5928d54d2fb67c912d140aac643Ted Bonkenburg jniThrowException(env, OutOfResourcesException, NULL); 1493866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian return 0; 1500de171b0d490a5928d54d2fb67c912d140aac643Ted Bonkenburg } 1510de171b0d490a5928d54d2fb67c912d140aac643Ted Bonkenburg 152b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopian surface->incStrong(&sRefBaseOwner); 1533866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian return int(surface.get()); 1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 15629479ebe1007361222bf6ab4d5e2a27927d4b8e8Mathias Agopianstatic void nativeRelease(JNIEnv* env, jclass clazz, jint nativeObject) { 1573866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian sp<Surface> sur(reinterpret_cast<Surface *>(nativeObject)); 158b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopian sur->decStrong(&sRefBaseOwner); 1598138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen} 1608138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen 16129479ebe1007361222bf6ab4d5e2a27927d4b8e8Mathias Agopianstatic jboolean nativeIsValid(JNIEnv* env, jclass clazz, jint nativeObject) { 1623866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian sp<Surface> sur(reinterpret_cast<Surface *>(nativeObject)); 16352800617946c456e78ed010c82d0ec4358368164Mathias Agopian return isSurfaceValid(sur) ? JNI_TRUE : JNI_FALSE; 1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16629479ebe1007361222bf6ab4d5e2a27927d4b8e8Mathias Agopianstatic jboolean nativeIsConsumerRunningBehind(JNIEnv* env, jclass clazz, jint nativeObject) { 1673866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian sp<Surface> sur(reinterpret_cast<Surface *>(nativeObject)); 16852800617946c456e78ed010c82d0ec4358368164Mathias Agopian if (!isSurfaceValid(sur)) { 169c14bacf1fb511472138eeb5dc84a9423fc003214Mathias Agopian doThrowIAE(env); 17064a55af0ac700baecb0877235eb42caac59a3560Jeff Brown return JNI_FALSE; 171c14bacf1fb511472138eeb5dc84a9423fc003214Mathias Agopian } 17264a55af0ac700baecb0877235eb42caac59a3560Jeff Brown int value = 0; 1733866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian ANativeWindow* anw = static_cast<ANativeWindow*>(sur.get()); 174c14bacf1fb511472138eeb5dc84a9423fc003214Mathias Agopian anw->query(anw, NATIVE_WINDOW_CONSUMER_RUNNING_BEHIND, &value); 17564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown return value; 176c14bacf1fb511472138eeb5dc84a9423fc003214Mathias Agopian} 177c14bacf1fb511472138eeb5dc84a9423fc003214Mathias Agopian 17864a55af0ac700baecb0877235eb42caac59a3560Jeff Brownstatic inline SkBitmap::Config convertPixelFormat(PixelFormat format) { 179f1e5b0d4e5d0d2a78c234cd0cbd3005a74a79429Mathias Agopian /* note: if PIXEL_FORMAT_RGBX_8888 means that all alpha bytes are 0xFF, then 1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project we can map to SkBitmap::kARGB_8888_Config, and optionally call 1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project bitmap.setIsOpaque(true) on the resulting SkBitmap (as an accelerator) 1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1838138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen switch (format) { 1848138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen case PIXEL_FORMAT_RGBX_8888: return SkBitmap::kARGB_8888_Config; 1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project case PIXEL_FORMAT_RGBA_8888: return SkBitmap::kARGB_8888_Config; 1868138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen case PIXEL_FORMAT_RGB_565: return SkBitmap::kRGB_565_Config; 1878138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen default: return SkBitmap::kNo_Config; 1888138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen } 1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 191fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenbergerstatic inline void swapCanvasPtr(JNIEnv* env, jobject canvasObj, SkCanvas* newCanvas) { 192fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger jobject canvasFinalizerObj = env->GetObjectField(canvasObj, gCanvasClassInfo.mFinalizer); 193fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger SkCanvas* previousCanvas = reinterpret_cast<SkCanvas*>( 194fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger env->GetIntField(canvasObj, gCanvasClassInfo.mNativeCanvas)); 195fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger env->SetIntField(canvasObj, gCanvasClassInfo.mNativeCanvas, (int)newCanvas); 196fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger env->SetIntField(canvasFinalizerObj, gCanvasFinalizerClassInfo.mNativeCanvas, (int)newCanvas); 197fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger SkSafeUnref(previousCanvas); 198fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger} 199fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger 200ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFaddenstatic jint nativeLockCanvas(JNIEnv* env, jclass clazz, 201fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown jint nativeObject, jobject canvasObj, jobject dirtyRectObj) { 2023866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian sp<Surface> surface(reinterpret_cast<Surface *>(nativeObject)); 2033866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian 20452800617946c456e78ed010c82d0ec4358368164Mathias Agopian if (!isSurfaceValid(surface)) { 2058451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes doThrowIAE(env); 206ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden return 0; 207fea12b87f2988f20ea87314c4a1b47e1a0f4d95eRomain Guy } 2089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 209667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian Rect dirtyRect; 210667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian Rect* dirtyRectPtr = NULL; 211667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian 21264a55af0ac700baecb0877235eb42caac59a3560Jeff Brown if (dirtyRectObj) { 213667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian dirtyRect.left = env->GetIntField(dirtyRectObj, gRectClassInfo.left); 214667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian dirtyRect.top = env->GetIntField(dirtyRectObj, gRectClassInfo.top); 215667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian dirtyRect.right = env->GetIntField(dirtyRectObj, gRectClassInfo.right); 216667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian dirtyRect.bottom = env->GetIntField(dirtyRectObj, gRectClassInfo.bottom); 217667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian dirtyRectPtr = &dirtyRect; 2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 22052800617946c456e78ed010c82d0ec4358368164Mathias Agopian ANativeWindow_Buffer outBuffer; 221667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian status_t err = surface->lock(&outBuffer, dirtyRectPtr); 2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (err < 0) { 2239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const char* const exception = (err == NO_MEMORY) ? 22464a55af0ac700baecb0877235eb42caac59a3560Jeff Brown OutOfResourcesException : 22564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown "java/lang/IllegalArgumentException"; 2268451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes jniThrowException(env, exception, NULL); 227ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden return 0; 2289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Associate a SkCanvas object to this surface 23152800617946c456e78ed010c82d0ec4358368164Mathias Agopian env->SetIntField(canvasObj, gCanvasClassInfo.mSurfaceFormat, outBuffer.format); 2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project SkBitmap bitmap; 23452800617946c456e78ed010c82d0ec4358368164Mathias Agopian ssize_t bpr = outBuffer.stride * bytesPerPixel(outBuffer.format); 23552800617946c456e78ed010c82d0ec4358368164Mathias Agopian bitmap.setConfig(convertPixelFormat(outBuffer.format), outBuffer.width, outBuffer.height, bpr); 23652800617946c456e78ed010c82d0ec4358368164Mathias Agopian if (outBuffer.format == PIXEL_FORMAT_RGBX_8888) { 237f1e5b0d4e5d0d2a78c234cd0cbd3005a74a79429Mathias Agopian bitmap.setIsOpaque(true); 238f1e5b0d4e5d0d2a78c234cd0cbd3005a74a79429Mathias Agopian } 23952800617946c456e78ed010c82d0ec4358368164Mathias Agopian if (outBuffer.width > 0 && outBuffer.height > 0) { 24052800617946c456e78ed010c82d0ec4358368164Mathias Agopian bitmap.setPixels(outBuffer.bits); 2419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // be safe with an empty bitmap. 2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project bitmap.setPixels(NULL); 2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 245fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger 246fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger SkCanvas* nativeCanvas = SkNEW_ARGS(SkCanvas, (bitmap)); 247fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger swapCanvasPtr(env, canvasObj, nativeCanvas); 2488451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes 249667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian if (dirtyRectPtr) { 250667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian nativeCanvas->clipRect( SkRect::Make(reinterpret_cast<const SkIRect&>(dirtyRect)) ); 2516158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian } 2526158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian 25364a55af0ac700baecb0877235eb42caac59a3560Jeff Brown if (dirtyRectObj) { 254667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian env->SetIntField(dirtyRectObj, gRectClassInfo.left, dirtyRect.left); 255667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian env->SetIntField(dirtyRectObj, gRectClassInfo.top, dirtyRect.top); 256667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian env->SetIntField(dirtyRectObj, gRectClassInfo.right, dirtyRect.right); 257667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian env->SetIntField(dirtyRectObj, gRectClassInfo.bottom, dirtyRect.bottom); 2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 259ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden 260ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden // Create another reference to the surface and return it. This reference 261ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden // should be passed to nativeUnlockCanvasAndPost in place of mNativeObject, 262ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden // because the latter could be replaced while the surface is locked. 263ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden sp<Surface> lockedSurface(surface); 264ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden lockedSurface->incStrong(&sRefBaseOwner); 265ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden return (int) lockedSurface.get(); 2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 268fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brownstatic void nativeUnlockCanvasAndPost(JNIEnv* env, jclass clazz, 269fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown jint nativeObject, jobject canvasObj) { 2703866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian sp<Surface> surface(reinterpret_cast<Surface *>(nativeObject)); 27152800617946c456e78ed010c82d0ec4358368164Mathias Agopian if (!isSurfaceValid(surface)) { 2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 27364a55af0ac700baecb0877235eb42caac59a3560Jeff Brown } 2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // detach the canvas from the surface 276fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger SkCanvas* nativeCanvas = SkNEW(SkCanvas); 277fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger swapCanvasPtr(env, canvasObj, nativeCanvas); 2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // unlock surface 2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project status_t err = surface->unlockAndPost(); 2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (err < 0) { 2828451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes doThrowIAE(env); 2839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 2859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 28664a55af0ac700baecb0877235eb42caac59a3560Jeff Brown// ---------------------------------------------------------------------------- 28764a55af0ac700baecb0877235eb42caac59a3560Jeff Brown 288fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brownstatic jint nativeCreateFromSurfaceControl(JNIEnv* env, jclass clazz, 289fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown jint surfaceControlNativeObj) { 29017f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian /* 29117f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian * This is used by the WindowManagerService just after constructing 29217f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian * a Surface and is necessary for returning the Surface reference to 29317f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian * the caller. At this point, we should only have a SurfaceControl. 29417f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian */ 295fae5cb2b356a1fef172b43066180a7ab4c32dbacMathias Agopian 2963866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian sp<SurfaceControl> ctrl(reinterpret_cast<SurfaceControl *>(surfaceControlNativeObj)); 297fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown sp<Surface> surface(ctrl->getSurface()); 298fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown if (surface != NULL) { 299fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown surface->incStrong(&sRefBaseOwner); 3009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 301fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown return reinterpret_cast<jint>(surface.get()); 30261566cc1932468720a831ad5cbc68ee080d613c9Dianne Hackborn} 30361566cc1932468720a831ad5cbc68ee080d613c9Dianne Hackborn 30429479ebe1007361222bf6ab4d5e2a27927d4b8e8Mathias Agopianstatic jint nativeReadFromParcel(JNIEnv* env, jclass clazz, 3053866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian jint nativeObject, jobject parcelObj) { 30664a55af0ac700baecb0877235eb42caac59a3560Jeff Brown Parcel* parcel = parcelForJavaObject(env, parcelObj); 3079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (parcel == NULL) { 3088451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes doThrowNPE(env); 3093866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian return 0; 3109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 311c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian 3123866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian sp<Surface> self(reinterpret_cast<Surface *>(nativeObject)); 313c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian sp<IBinder> binder(parcel->readStrongBinder()); 314c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian 315c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian // update the Surface only if the underlying IGraphicBufferProducer 316c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian // has changed. 317c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian if (self != NULL 318c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian && (self->getIGraphicBufferProducer()->asBinder() == binder)) { 319c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian // same IGraphicBufferProducer, return ourselves 320c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian return int(self.get()); 3213866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian } 3224a05f436989a49e06ec762aac9507ad13fcc5e23Mathias Agopian 3234a05f436989a49e06ec762aac9507ad13fcc5e23Mathias Agopian sp<Surface> sur; 324c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian sp<IGraphicBufferProducer> gbp(interface_cast<IGraphicBufferProducer>(binder)); 3254a05f436989a49e06ec762aac9507ad13fcc5e23Mathias Agopian if (gbp != NULL) { 326c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian // we have a new IGraphicBufferProducer, create a new Surface for it 327e32632682ca9207bd247ca27012cf670b5c23f54Mathias Agopian sur = new Surface(gbp, true); 328c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian // and keep a reference before passing to java 329b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopian sur->incStrong(&sRefBaseOwner); 3303866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian } 3314a05f436989a49e06ec762aac9507ad13fcc5e23Mathias Agopian 332c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian if (self != NULL) { 333c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian // and loose the java reference to ourselves 334c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian self->decStrong(&sRefBaseOwner); 335c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian } 336c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian 3373866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian return int(sur.get()); 3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 3399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 34029479ebe1007361222bf6ab4d5e2a27927d4b8e8Mathias Agopianstatic void nativeWriteToParcel(JNIEnv* env, jclass clazz, 3413866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian jint nativeObject, jobject parcelObj) { 34264a55af0ac700baecb0877235eb42caac59a3560Jeff Brown Parcel* parcel = parcelForJavaObject(env, parcelObj); 3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (parcel == NULL) { 3448451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes doThrowNPE(env); 3459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3473866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian sp<Surface> self(reinterpret_cast<Surface *>(nativeObject)); 3484a05f436989a49e06ec762aac9507ad13fcc5e23Mathias Agopian parcel->writeStrongBinder( self != 0 ? self->getIGraphicBufferProducer()->asBinder() : NULL); 3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------- 3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic JNINativeMethod gSurfaceMethods[] = { 3543866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian {"nativeCreateFromSurfaceTexture", "(Landroid/graphics/SurfaceTexture;)I", 35564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown (void*)nativeCreateFromSurfaceTexture }, 3563866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian {"nativeRelease", "(I)V", 35764a55af0ac700baecb0877235eb42caac59a3560Jeff Brown (void*)nativeRelease }, 3583866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian {"nativeIsValid", "(I)Z", 35964a55af0ac700baecb0877235eb42caac59a3560Jeff Brown (void*)nativeIsValid }, 3603866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian {"nativeIsConsumerRunningBehind", "(I)Z", 36164a55af0ac700baecb0877235eb42caac59a3560Jeff Brown (void*)nativeIsConsumerRunningBehind }, 362ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden {"nativeLockCanvas", "(ILandroid/graphics/Canvas;Landroid/graphics/Rect;)I", 36364a55af0ac700baecb0877235eb42caac59a3560Jeff Brown (void*)nativeLockCanvas }, 3643866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian {"nativeUnlockCanvasAndPost", "(ILandroid/graphics/Canvas;)V", 36564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown (void*)nativeUnlockCanvasAndPost }, 366fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown {"nativeCreateFromSurfaceControl", "(I)I", 367fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown (void*)nativeCreateFromSurfaceControl }, 3683866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian {"nativeReadFromParcel", "(ILandroid/os/Parcel;)I", 36964a55af0ac700baecb0877235eb42caac59a3560Jeff Brown (void*)nativeReadFromParcel }, 3703866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian {"nativeWriteToParcel", "(ILandroid/os/Parcel;)V", 37164a55af0ac700baecb0877235eb42caac59a3560Jeff Brown (void*)nativeWriteToParcel }, 3729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}; 3739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectint register_android_view_Surface(JNIEnv* env) 3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 37664a55af0ac700baecb0877235eb42caac59a3560Jeff Brown int err = AndroidRuntime::registerNativeMethods(env, "android/view/Surface", 3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project gSurfaceMethods, NELEM(gSurfaceMethods)); 37864a55af0ac700baecb0877235eb42caac59a3560Jeff Brown 37964a55af0ac700baecb0877235eb42caac59a3560Jeff Brown jclass clazz = env->FindClass("android/view/Surface"); 38064a55af0ac700baecb0877235eb42caac59a3560Jeff Brown gSurfaceClassInfo.clazz = jclass(env->NewGlobalRef(clazz)); 3813866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian gSurfaceClassInfo.mNativeObject = 3823866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian env->GetFieldID(gSurfaceClassInfo.clazz, "mNativeObject", "I"); 383fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown gSurfaceClassInfo.mLock = 384fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown env->GetFieldID(gSurfaceClassInfo.clazz, "mLock", "Ljava/lang/Object;"); 385ffddc9b8045235a493ec506965ae4892601eb23dMathias Agopian gSurfaceClassInfo.ctor = env->GetMethodID(gSurfaceClassInfo.clazz, "<init>", "(I)V"); 38664a55af0ac700baecb0877235eb42caac59a3560Jeff Brown 38764a55af0ac700baecb0877235eb42caac59a3560Jeff Brown clazz = env->FindClass("android/graphics/Canvas"); 388fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger gCanvasClassInfo.mFinalizer = env->GetFieldID(clazz, "mFinalizer", "Landroid/graphics/Canvas$CanvasFinalizer;"); 38964a55af0ac700baecb0877235eb42caac59a3560Jeff Brown gCanvasClassInfo.mNativeCanvas = env->GetFieldID(clazz, "mNativeCanvas", "I"); 39064a55af0ac700baecb0877235eb42caac59a3560Jeff Brown gCanvasClassInfo.mSurfaceFormat = env->GetFieldID(clazz, "mSurfaceFormat", "I"); 39164a55af0ac700baecb0877235eb42caac59a3560Jeff Brown 392fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger clazz = env->FindClass("android/graphics/Canvas$CanvasFinalizer"); 393fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger gCanvasFinalizerClassInfo.mNativeCanvas = env->GetFieldID(clazz, "mNativeCanvas", "I"); 394fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger 39564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown clazz = env->FindClass("android/graphics/Rect"); 39664a55af0ac700baecb0877235eb42caac59a3560Jeff Brown gRectClassInfo.left = env->GetFieldID(clazz, "left", "I"); 39764a55af0ac700baecb0877235eb42caac59a3560Jeff Brown gRectClassInfo.top = env->GetFieldID(clazz, "top", "I"); 39864a55af0ac700baecb0877235eb42caac59a3560Jeff Brown gRectClassInfo.right = env->GetFieldID(clazz, "right", "I"); 39964a55af0ac700baecb0877235eb42caac59a3560Jeff Brown gRectClassInfo.bottom = env->GetFieldID(clazz, "bottom", "I"); 40064a55af0ac700baecb0877235eb42caac59a3560Jeff Brown 4019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return err; 4029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 4039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}; 405