android_view_Surface.cpp revision 5c3d927e17e98e8fd4a9f3c86f7f4def0bcfa816
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>
428790be6de3644e332ec6a17c855da89ffc13a9bfLeon Scroggins III#include <SkImage.h>
436158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian#include <SkRegion.h>
449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/misc.h>
469e316a1a2a8d734315bbd56a85308f9657a92913Jeff Brown#include <utils/Log.h>
479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4864a55af0ac700baecb0877235eb42caac59a3560Jeff Brown#include <ScopedUtfChars.h>
4964a55af0ac700baecb0877235eb42caac59a3560Jeff Brown
509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectnamespace android {
539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic const char* const OutOfResourcesException =
559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    "android/view/Surface$OutOfResourcesException";
569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5764a55af0ac700baecb0877235eb42caac59a3560Jeff Brownstatic struct {
5864a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    jclass clazz;
593866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    jfieldID mNativeObject;
60fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown    jfieldID mLock;
61cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown    jmethodID ctor;
6264a55af0ac700baecb0877235eb42caac59a3560Jeff Brown} gSurfaceClassInfo;
6364a55af0ac700baecb0877235eb42caac59a3560Jeff Brown
6464a55af0ac700baecb0877235eb42caac59a3560Jeff Brownstatic struct {
6564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    jfieldID left;
6664a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    jfieldID top;
6764a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    jfieldID right;
6864a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    jfieldID bottom;
6964a55af0ac700baecb0877235eb42caac59a3560Jeff Brown} gRectClassInfo;
7064a55af0ac700baecb0877235eb42caac59a3560Jeff Brown
7164a55af0ac700baecb0877235eb42caac59a3560Jeff Brownstatic struct {
7264a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    jfieldID mSurfaceFormat;
735c3d927e17e98e8fd4a9f3c86f7f4def0bcfa816Florin Malita    jmethodID setNativeBitmap;
7464a55af0ac700baecb0877235eb42caac59a3560Jeff Brown} gCanvasClassInfo;
7564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown
769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
78b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopian// this is just a pointer we use to pass to inc/decStrong
79b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopianstatic const void *sRefBaseOwner;
80b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopian
8164a55af0ac700baecb0877235eb42caac59a3560Jeff Brownbool android_view_Surface_isInstanceOf(JNIEnv* env, jobject obj) {
8264a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    return env->IsInstanceOf(obj, gSurfaceClassInfo.clazz);
83050316184b01c0d1a01c46afae7429b89a27c31btedbo}
84050316184b01c0d1a01c46afae7429b89a27c31btedbo
853866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopiansp<ANativeWindow> android_view_Surface_getNativeWindow(JNIEnv* env, jobject surfaceObj) {
863866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    return android_view_Surface_getSurface(env, surfaceObj);
879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
893866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopiansp<Surface> android_view_Surface_getSurface(JNIEnv* env, jobject surfaceObj) {
907c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian    sp<Surface> sur;
917c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian    jobject lock = env->GetObjectField(surfaceObj,
92fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown            gSurfaceClassInfo.mLock);
937c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian    if (env->MonitorEnter(lock) == JNI_OK) {
947c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian        sur = reinterpret_cast<Surface *>(
9536bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat                env->GetLongField(surfaceObj, gSurfaceClassInfo.mNativeObject));
967c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian        env->MonitorExit(lock);
977c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian    }
987c116b54b743cc3e92ac42abdbbe324d63b50a81Mathias Agopian    return sur;
99cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown}
100cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown
10129479ebe1007361222bf6ab4d5e2a27927d4b8e8Mathias Agopianjobject android_view_Surface_createFromIGraphicBufferProducer(JNIEnv* env,
102d47f7d8b5fe3a3861d7cbdc5f912235407823c8eAndy McFadden        const sp<IGraphicBufferProducer>& bufferProducer) {
103d47f7d8b5fe3a3861d7cbdc5f912235407823c8eAndy McFadden    if (bufferProducer == NULL) {
104cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown        return NULL;
105cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown    }
106cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown
107e32632682ca9207bd247ca27012cf670b5c23f54Mathias Agopian    sp<Surface> surface(new Surface(bufferProducer, true));
108cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown    if (surface == NULL) {
109cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown        return NULL;
110cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown    }
111cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown
112af608e81d450b962a443a21fb1a6feadeb355fe1Chong Zhang    jobject surfaceObj = env->NewObject(gSurfaceClassInfo.clazz,
113af608e81d450b962a443a21fb1a6feadeb355fe1Chong Zhang            gSurfaceClassInfo.ctor, (jlong)surface.get());
114cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown    if (surfaceObj == NULL) {
115cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown        if (env->ExceptionCheck()) {
116d47f7d8b5fe3a3861d7cbdc5f912235407823c8eAndy McFadden            ALOGE("Could not create instance of Surface from IGraphicBufferProducer.");
117cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown            LOGE_EX(env);
118cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown            env->ExceptionClear();
119cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown        }
120cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown        return NULL;
121cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown    }
122b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopian    surface->incStrong(&sRefBaseOwner);
123cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown    return surfaceObj;
124cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown}
125cbad976b2a36a0895ca94510d5208a86f66cf596Jeff Brown
1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
128f3d3b4d80618d498fb669c8bfdc492ce9ea25691Mathias Agopianstatic inline bool isSurfaceValid(const sp<Surface>& sur) {
129f3d3b4d80618d498fb669c8bfdc492ce9ea25691Mathias Agopian    return Surface::isValid(sur);
13052800617946c456e78ed010c82d0ec4358368164Mathias Agopian}
13152800617946c456e78ed010c82d0ec4358368164Mathias Agopian
13252800617946c456e78ed010c82d0ec4358368164Mathias Agopian// ----------------------------------------------------------------------------
13352800617946c456e78ed010c82d0ec4358368164Mathias Agopian
13436bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhatstatic jlong nativeCreateFromSurfaceTexture(JNIEnv* env, jclass clazz,
13564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown        jobject surfaceTextureObj) {
13652a9a10b6b8c7b7a9f97777541841b94d4fd9754Mathias Agopian    sp<IGraphicBufferProducer> producer(SurfaceTexture_getProducer(env, surfaceTextureObj));
13752a9a10b6b8c7b7a9f97777541841b94d4fd9754Mathias Agopian    if (producer == NULL) {
1382e76c992bfa955cb637417005941e3e93810a99dDaniel Lam        jniThrowException(env, "java/lang/IllegalArgumentException",
1392e76c992bfa955cb637417005941e3e93810a99dDaniel Lam                "SurfaceTexture has already been released");
1403866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian        return 0;
1412e76c992bfa955cb637417005941e3e93810a99dDaniel Lam    }
14264a55af0ac700baecb0877235eb42caac59a3560Jeff Brown
14352a9a10b6b8c7b7a9f97777541841b94d4fd9754Mathias Agopian    sp<Surface> surface(new Surface(producer, true));
1440de171b0d490a5928d54d2fb67c912d140aac643Ted Bonkenburg    if (surface == NULL) {
1450de171b0d490a5928d54d2fb67c912d140aac643Ted Bonkenburg        jniThrowException(env, OutOfResourcesException, NULL);
1463866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian        return 0;
1470de171b0d490a5928d54d2fb67c912d140aac643Ted Bonkenburg    }
1480de171b0d490a5928d54d2fb67c912d140aac643Ted Bonkenburg
149b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopian    surface->incStrong(&sRefBaseOwner);
15036bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    return jlong(surface.get());
1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
15336bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhatstatic void nativeRelease(JNIEnv* env, jclass clazz, jlong nativeObject) {
1543866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    sp<Surface> sur(reinterpret_cast<Surface *>(nativeObject));
155b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopian    sur->decStrong(&sRefBaseOwner);
1568138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen}
1578138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen
15836bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhatstatic jboolean nativeIsValid(JNIEnv* env, jclass clazz, jlong nativeObject) {
1593866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    sp<Surface> sur(reinterpret_cast<Surface *>(nativeObject));
16052800617946c456e78ed010c82d0ec4358368164Mathias Agopian    return isSurfaceValid(sur) ? JNI_TRUE : JNI_FALSE;
1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
16336bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhatstatic jboolean nativeIsConsumerRunningBehind(JNIEnv* env, jclass clazz, jlong nativeObject) {
1643866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    sp<Surface> sur(reinterpret_cast<Surface *>(nativeObject));
16552800617946c456e78ed010c82d0ec4358368164Mathias Agopian    if (!isSurfaceValid(sur)) {
166c14bacf1fb511472138eeb5dc84a9423fc003214Mathias Agopian        doThrowIAE(env);
16764a55af0ac700baecb0877235eb42caac59a3560Jeff Brown        return JNI_FALSE;
168c14bacf1fb511472138eeb5dc84a9423fc003214Mathias Agopian    }
16964a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    int value = 0;
1703866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    ANativeWindow* anw = static_cast<ANativeWindow*>(sur.get());
171c14bacf1fb511472138eeb5dc84a9423fc003214Mathias Agopian    anw->query(anw, NATIVE_WINDOW_CONSUMER_RUNNING_BEHIND, &value);
17264a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    return value;
173c14bacf1fb511472138eeb5dc84a9423fc003214Mathias Agopian}
174c14bacf1fb511472138eeb5dc84a9423fc003214Mathias Agopian
17564a55af0ac700baecb0877235eb42caac59a3560Jeff Brownstatic inline SkBitmap::Config convertPixelFormat(PixelFormat format) {
176f1e5b0d4e5d0d2a78c234cd0cbd3005a74a79429Mathias Agopian    /* note: if PIXEL_FORMAT_RGBX_8888 means that all alpha bytes are 0xFF, then
1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        we can map to SkBitmap::kARGB_8888_Config, and optionally call
1788790be6de3644e332ec6a17c855da89ffc13a9bfLeon Scroggins III        bitmap.setAlphaType(kOpaque_SkAlphaType) on the resulting SkBitmap
1798790be6de3644e332ec6a17c855da89ffc13a9bfLeon Scroggins III        (as an accelerator)
1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    */
1818138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen    switch (format) {
1828138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen    case PIXEL_FORMAT_RGBX_8888:    return SkBitmap::kARGB_8888_Config;
1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_RGBA_8888:    return SkBitmap::kARGB_8888_Config;
1848138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen    case PIXEL_FORMAT_RGB_565:      return SkBitmap::kRGB_565_Config;
1858138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen    default:                        return SkBitmap::kNo_Config;
1868138cb49e47f9a9905a316a217757710185c66aaMarco Nelissen    }
1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
18936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhatstatic jlong nativeLockCanvas(JNIEnv* env, jclass clazz,
19036bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        jlong nativeObject, jobject canvasObj, jobject dirtyRectObj) {
1913866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    sp<Surface> surface(reinterpret_cast<Surface *>(nativeObject));
1923866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian
19352800617946c456e78ed010c82d0ec4358368164Mathias Agopian    if (!isSurfaceValid(surface)) {
1948451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes        doThrowIAE(env);
195ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden        return 0;
196fea12b87f2988f20ea87314c4a1b47e1a0f4d95eRomain Guy    }
1979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
198667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian    Rect dirtyRect;
199667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian    Rect* dirtyRectPtr = NULL;
200667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian
20164a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    if (dirtyRectObj) {
202667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian        dirtyRect.left   = env->GetIntField(dirtyRectObj, gRectClassInfo.left);
203667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian        dirtyRect.top    = env->GetIntField(dirtyRectObj, gRectClassInfo.top);
204667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian        dirtyRect.right  = env->GetIntField(dirtyRectObj, gRectClassInfo.right);
205667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian        dirtyRect.bottom = env->GetIntField(dirtyRectObj, gRectClassInfo.bottom);
206667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian        dirtyRectPtr = &dirtyRect;
2079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
20952800617946c456e78ed010c82d0ec4358368164Mathias Agopian    ANativeWindow_Buffer outBuffer;
210667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian    status_t err = surface->lock(&outBuffer, dirtyRectPtr);
2119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (err < 0) {
2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const char* const exception = (err == NO_MEMORY) ?
21364a55af0ac700baecb0877235eb42caac59a3560Jeff Brown                OutOfResourcesException :
21464a55af0ac700baecb0877235eb42caac59a3560Jeff Brown                "java/lang/IllegalArgumentException";
2158451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes        jniThrowException(env, exception, NULL);
216ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden        return 0;
2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // Associate a SkCanvas object to this surface
22052800617946c456e78ed010c82d0ec4358368164Mathias Agopian    env->SetIntField(canvasObj, gCanvasClassInfo.mSurfaceFormat, outBuffer.format);
2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    SkBitmap bitmap;
22352800617946c456e78ed010c82d0ec4358368164Mathias Agopian    ssize_t bpr = outBuffer.stride * bytesPerPixel(outBuffer.format);
22452800617946c456e78ed010c82d0ec4358368164Mathias Agopian    bitmap.setConfig(convertPixelFormat(outBuffer.format), outBuffer.width, outBuffer.height, bpr);
22552800617946c456e78ed010c82d0ec4358368164Mathias Agopian    if (outBuffer.format == PIXEL_FORMAT_RGBX_8888) {
2268790be6de3644e332ec6a17c855da89ffc13a9bfLeon Scroggins III        bitmap.setAlphaType(kOpaque_SkAlphaType);
227f1e5b0d4e5d0d2a78c234cd0cbd3005a74a79429Mathias Agopian    }
22852800617946c456e78ed010c82d0ec4358368164Mathias Agopian    if (outBuffer.width > 0 && outBuffer.height > 0) {
22952800617946c456e78ed010c82d0ec4358368164Mathias Agopian        bitmap.setPixels(outBuffer.bits);
2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // be safe with an empty bitmap.
2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        bitmap.setPixels(NULL);
2339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
234fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger
2355c3d927e17e98e8fd4a9f3c86f7f4def0bcfa816Florin Malita    env->CallVoidMethod(canvasObj, gCanvasClassInfo.setNativeBitmap,
2365c3d927e17e98e8fd4a9f3c86f7f4def0bcfa816Florin Malita                        reinterpret_cast<jlong>(&bitmap));
2378451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
238667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian    if (dirtyRectPtr) {
2395c3d927e17e98e8fd4a9f3c86f7f4def0bcfa816Florin Malita        SkCanvas* nativeCanvas = GraphicsJNI::getNativeCanvas(env, canvasObj);
240667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian        nativeCanvas->clipRect( SkRect::Make(reinterpret_cast<const SkIRect&>(dirtyRect)) );
2416158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    }
2426158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian
24364a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    if (dirtyRectObj) {
244667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian        env->SetIntField(dirtyRectObj, gRectClassInfo.left,   dirtyRect.left);
245667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian        env->SetIntField(dirtyRectObj, gRectClassInfo.top,    dirtyRect.top);
246667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian        env->SetIntField(dirtyRectObj, gRectClassInfo.right,  dirtyRect.right);
247667809ef5d2d1fe3796cdc9bdd09503a70d2ba6cMathias Agopian        env->SetIntField(dirtyRectObj, gRectClassInfo.bottom, dirtyRect.bottom);
2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
249ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden
250ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden    // Create another reference to the surface and return it.  This reference
251ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden    // should be passed to nativeUnlockCanvasAndPost in place of mNativeObject,
252ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden    // because the latter could be replaced while the surface is locked.
253ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden    sp<Surface> lockedSurface(surface);
254ed55c8db1c0d47492423fc54f4b0dd5cd585e593Andy McFadden    lockedSurface->incStrong(&sRefBaseOwner);
25536bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    return (jlong) lockedSurface.get();
2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
258fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brownstatic void nativeUnlockCanvasAndPost(JNIEnv* env, jclass clazz,
25936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        jlong nativeObject, jobject canvasObj) {
2603866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    sp<Surface> surface(reinterpret_cast<Surface *>(nativeObject));
26152800617946c456e78ed010c82d0ec4358368164Mathias Agopian    if (!isSurfaceValid(surface)) {
2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return;
26364a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    }
2649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // detach the canvas from the surface
2665c3d927e17e98e8fd4a9f3c86f7f4def0bcfa816Florin Malita    env->CallVoidMethod(canvasObj, gCanvasClassInfo.setNativeBitmap, (jlong)0);
2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // unlock surface
2699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    status_t err = surface->unlockAndPost();
2709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (err < 0) {
2718451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes        doThrowIAE(env);
2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
27564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown// ----------------------------------------------------------------------------
27664a55af0ac700baecb0877235eb42caac59a3560Jeff Brown
27736bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhatstatic jlong nativeCreateFromSurfaceControl(JNIEnv* env, jclass clazz,
27836bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        jlong surfaceControlNativeObj) {
27917f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian    /*
28017f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian     * This is used by the WindowManagerService just after constructing
28117f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian     * a Surface and is necessary for returning the Surface reference to
28217f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian     * the caller. At this point, we should only have a SurfaceControl.
28317f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian     */
284fae5cb2b356a1fef172b43066180a7ab4c32dbacMathias Agopian
2853866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    sp<SurfaceControl> ctrl(reinterpret_cast<SurfaceControl *>(surfaceControlNativeObj));
286fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown    sp<Surface> surface(ctrl->getSurface());
287fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown    if (surface != NULL) {
288fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown        surface->incStrong(&sRefBaseOwner);
2899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
29036bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    return reinterpret_cast<jlong>(surface.get());
29161566cc1932468720a831ad5cbc68ee080d613c9Dianne Hackborn}
29261566cc1932468720a831ad5cbc68ee080d613c9Dianne Hackborn
29336bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhatstatic jlong nativeReadFromParcel(JNIEnv* env, jclass clazz,
29436bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        jlong nativeObject, jobject parcelObj) {
29564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    Parcel* parcel = parcelForJavaObject(env, parcelObj);
2969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (parcel == NULL) {
2978451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes        doThrowNPE(env);
2983866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian        return 0;
2999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
300c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian
3013866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    sp<Surface> self(reinterpret_cast<Surface *>(nativeObject));
302c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian    sp<IBinder> binder(parcel->readStrongBinder());
303c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian
304c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian    // update the Surface only if the underlying IGraphicBufferProducer
305c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian    // has changed.
306c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian    if (self != NULL
307c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian            && (self->getIGraphicBufferProducer()->asBinder() == binder)) {
308c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian        // same IGraphicBufferProducer, return ourselves
30936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return jlong(self.get());
3103866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    }
3114a05f436989a49e06ec762aac9507ad13fcc5e23Mathias Agopian
3124a05f436989a49e06ec762aac9507ad13fcc5e23Mathias Agopian    sp<Surface> sur;
313c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian    sp<IGraphicBufferProducer> gbp(interface_cast<IGraphicBufferProducer>(binder));
3144a05f436989a49e06ec762aac9507ad13fcc5e23Mathias Agopian    if (gbp != NULL) {
315c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian        // we have a new IGraphicBufferProducer, create a new Surface for it
316e32632682ca9207bd247ca27012cf670b5c23f54Mathias Agopian        sur = new Surface(gbp, true);
317c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian        // and keep a reference before passing to java
318b1d90c8f60f71422196c337f1d078b68867f5710Mathias Agopian        sur->incStrong(&sRefBaseOwner);
3193866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    }
3204a05f436989a49e06ec762aac9507ad13fcc5e23Mathias Agopian
321c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian    if (self != NULL) {
322c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian        // and loose the java reference to ourselves
323c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian        self->decStrong(&sRefBaseOwner);
324c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian    }
325c3b9cd635163ae48a9151fc6e1650388c5d716beMathias Agopian
32636bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    return jlong(sur.get());
3279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
32929479ebe1007361222bf6ab4d5e2a27927d4b8e8Mathias Agopianstatic void nativeWriteToParcel(JNIEnv* env, jclass clazz,
33036bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        jlong nativeObject, jobject parcelObj) {
33164a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    Parcel* parcel = parcelForJavaObject(env, parcelObj);
3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (parcel == NULL) {
3338451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes        doThrowNPE(env);
3349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return;
3359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3363866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    sp<Surface> self(reinterpret_cast<Surface *>(nativeObject));
3374a05f436989a49e06ec762aac9507ad13fcc5e23Mathias Agopian    parcel->writeStrongBinder( self != 0 ? self->getIGraphicBufferProducer()->asBinder() : NULL);
3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic JNINativeMethod gSurfaceMethods[] = {
34336bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    {"nativeCreateFromSurfaceTexture", "(Landroid/graphics/SurfaceTexture;)J",
34464a55af0ac700baecb0877235eb42caac59a3560Jeff Brown            (void*)nativeCreateFromSurfaceTexture },
34536bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    {"nativeRelease", "(J)V",
34664a55af0ac700baecb0877235eb42caac59a3560Jeff Brown            (void*)nativeRelease },
34736bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    {"nativeIsValid", "(J)Z",
34864a55af0ac700baecb0877235eb42caac59a3560Jeff Brown            (void*)nativeIsValid },
34936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    {"nativeIsConsumerRunningBehind", "(J)Z",
35064a55af0ac700baecb0877235eb42caac59a3560Jeff Brown            (void*)nativeIsConsumerRunningBehind },
35136bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    {"nativeLockCanvas", "(JLandroid/graphics/Canvas;Landroid/graphics/Rect;)J",
35264a55af0ac700baecb0877235eb42caac59a3560Jeff Brown            (void*)nativeLockCanvas },
35336bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    {"nativeUnlockCanvasAndPost", "(JLandroid/graphics/Canvas;)V",
35464a55af0ac700baecb0877235eb42caac59a3560Jeff Brown            (void*)nativeUnlockCanvasAndPost },
35536bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    {"nativeCreateFromSurfaceControl", "(J)J",
356fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown            (void*)nativeCreateFromSurfaceControl },
35736bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    {"nativeReadFromParcel", "(JLandroid/os/Parcel;)J",
35864a55af0ac700baecb0877235eb42caac59a3560Jeff Brown            (void*)nativeReadFromParcel },
35936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    {"nativeWriteToParcel", "(JLandroid/os/Parcel;)V",
36064a55af0ac700baecb0877235eb42caac59a3560Jeff Brown            (void*)nativeWriteToParcel },
3619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project};
3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectint register_android_view_Surface(JNIEnv* env)
3649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
36564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    int err = AndroidRuntime::registerNativeMethods(env, "android/view/Surface",
3669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            gSurfaceMethods, NELEM(gSurfaceMethods));
36764a55af0ac700baecb0877235eb42caac59a3560Jeff Brown
36864a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    jclass clazz = env->FindClass("android/view/Surface");
36964a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    gSurfaceClassInfo.clazz = jclass(env->NewGlobalRef(clazz));
3703866f0d581ceaa165710feeee9f37fe1b0d7067dMathias Agopian    gSurfaceClassInfo.mNativeObject =
37136bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat            env->GetFieldID(gSurfaceClassInfo.clazz, "mNativeObject", "J");
372fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown    gSurfaceClassInfo.mLock =
373fc0ebd7d379ff63c00ebf78ca252fab5070213daJeff Brown            env->GetFieldID(gSurfaceClassInfo.clazz, "mLock", "Ljava/lang/Object;");
37436bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    gSurfaceClassInfo.ctor = env->GetMethodID(gSurfaceClassInfo.clazz, "<init>", "(J)V");
37564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown
37664a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    clazz = env->FindClass("android/graphics/Canvas");
37764a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    gCanvasClassInfo.mSurfaceFormat = env->GetFieldID(clazz, "mSurfaceFormat", "I");
3785c3d927e17e98e8fd4a9f3c86f7f4def0bcfa816Florin Malita    gCanvasClassInfo.setNativeBitmap = env->GetMethodID(clazz, "setNativeBitmap", "(J)V");
379fc615a0f643408956fc0dc1b997871e2b27cee7eDerek Sollenberger
38064a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    clazz = env->FindClass("android/graphics/Rect");
38164a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    gRectClassInfo.left = env->GetFieldID(clazz, "left", "I");
38264a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    gRectClassInfo.top = env->GetFieldID(clazz, "top", "I");
38364a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    gRectClassInfo.right = env->GetFieldID(clazz, "right", "I");
38464a55af0ac700baecb0877235eb42caac59a3560Jeff Brown    gRectClassInfo.bottom = env->GetFieldID(clazz, "bottom", "I");
38564a55af0ac700baecb0877235eb42caac59a3560Jeff Brown
3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return err;
3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project};
390