1/*
2 * Copyright (C) 2013 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#define LOG_TAG "GraphicBuffer"
18
19#include "jni.h"
20#include "JNIHelp.h"
21
22#include "android_os_Parcel.h"
23#include "android_view_GraphicBuffer.h"
24#include "android/graphics/GraphicsJNI.h"
25
26#include <android_runtime/AndroidRuntime.h>
27
28#include <binder/Parcel.h>
29
30#include <ui/GraphicBuffer.h>
31#include <ui/PixelFormat.h>
32
33#include <gui/IGraphicBufferAlloc.h>
34#include <gui/ISurfaceComposer.h>
35
36#include <SkCanvas.h>
37#include <SkBitmap.h>
38
39#include <private/gui/ComposerService.h>
40
41namespace android {
42
43// ----------------------------------------------------------------------------
44// Defines
45// ----------------------------------------------------------------------------
46
47// Debug
48#define DEBUG_GRAPHIC_BUFFER 0
49
50// Debug
51#if DEBUG_GRAPHIC_BUFFER
52    #define GB_LOGD(...) ALOGD(__VA_ARGS__)
53    #define GB_LOGW(...) ALOGW(__VA_ARGS__)
54#else
55    #define GB_LOGD(...)
56    #define GB_LOGW(...)
57#endif
58
59#define LOCK_CANVAS_USAGE GraphicBuffer::USAGE_SW_READ_OFTEN | GraphicBuffer::USAGE_SW_WRITE_OFTEN
60
61// ----------------------------------------------------------------------------
62// JNI Helpers
63// ----------------------------------------------------------------------------
64
65static struct {
66    jfieldID mNativeObject;
67} gGraphicBufferClassInfo;
68
69static struct {
70    jmethodID set;
71    jfieldID left;
72    jfieldID top;
73    jfieldID right;
74    jfieldID bottom;
75} gRectClassInfo;
76
77static struct {
78    jfieldID mSurfaceFormat;
79    jmethodID setNativeBitmap;
80} gCanvasClassInfo;
81
82#define GET_INT(object, field) \
83    env->GetIntField(object, field)
84
85#define SET_INT(object, field, value) \
86    env->SetIntField(object, field, value)
87
88#define GET_LONG(object, field) \
89    env->GetLongField(object, field)
90
91#define SET_LONG(object, field, value) \
92    env->SetLongField(object, field, value)
93
94#define INVOKEV(object, method, ...) \
95    env->CallVoidMethod(object, method, __VA_ARGS__)
96
97// ----------------------------------------------------------------------------
98// Types
99// ----------------------------------------------------------------------------
100
101class GraphicBufferWrapper {
102public:
103    GraphicBufferWrapper(const sp<GraphicBuffer>& buffer): buffer(buffer) {
104    }
105
106    sp<GraphicBuffer> buffer;
107};
108
109// ----------------------------------------------------------------------------
110// GraphicBuffer lifecycle
111// ----------------------------------------------------------------------------
112
113static jlong android_view_GraphiceBuffer_create(JNIEnv* env, jobject clazz,
114        jint width, jint height, jint format, jint usage) {
115
116    sp<ISurfaceComposer> composer(ComposerService::getComposerService());
117    sp<IGraphicBufferAlloc> alloc(composer->createGraphicBufferAlloc());
118    if (alloc == NULL) {
119        GB_LOGW("createGraphicBufferAlloc() failed in GraphicBuffer.create()");
120        return NULL;
121    }
122
123    status_t error;
124    sp<GraphicBuffer> buffer(alloc->createGraphicBuffer(width, height, format, usage, &error));
125    if (buffer == NULL) {
126        GB_LOGW("createGraphicBuffer() failed in GraphicBuffer.create()");
127        return NULL;
128    }
129
130    GraphicBufferWrapper* wrapper = new GraphicBufferWrapper(buffer);
131    return reinterpret_cast<jlong>(wrapper);
132}
133
134static void android_view_GraphiceBuffer_destroy(JNIEnv* env, jobject clazz,
135        jlong wrapperHandle) {
136    GraphicBufferWrapper* wrapper =
137                reinterpret_cast<GraphicBufferWrapper*>(wrapperHandle);
138    delete wrapper;
139}
140
141// ----------------------------------------------------------------------------
142// Canvas management
143// ----------------------------------------------------------------------------
144
145static inline SkColorType convertPixelFormat(int32_t format) {
146    switch (format) {
147        case PIXEL_FORMAT_RGBA_8888:
148            return kN32_SkColorType;
149        case PIXEL_FORMAT_RGBX_8888:
150            return kN32_SkColorType;
151        case PIXEL_FORMAT_RGB_565:
152            return kRGB_565_SkColorType;
153        default:
154            return kUnknown_SkColorType;
155    }
156}
157
158static jboolean android_view_GraphicBuffer_lockCanvas(JNIEnv* env, jobject,
159        jlong wrapperHandle, jobject canvas, jobject dirtyRect) {
160
161    GraphicBufferWrapper* wrapper =
162                reinterpret_cast<GraphicBufferWrapper*>(wrapperHandle);
163    if (!wrapper) {
164        return JNI_FALSE;
165    }
166
167    sp<GraphicBuffer> buffer(wrapper->buffer);
168
169    Rect rect;
170    if (dirtyRect) {
171        rect.left = GET_INT(dirtyRect, gRectClassInfo.left);
172        rect.top = GET_INT(dirtyRect, gRectClassInfo.top);
173        rect.right = GET_INT(dirtyRect, gRectClassInfo.right);
174        rect.bottom = GET_INT(dirtyRect, gRectClassInfo.bottom);
175    } else {
176        rect.set(Rect(buffer->getWidth(), buffer->getHeight()));
177    }
178
179    void* bits = NULL;
180    status_t status = buffer->lock(LOCK_CANVAS_USAGE, rect, &bits);
181
182    if (status) return JNI_FALSE;
183    if (!bits) {
184        buffer->unlock();
185        return JNI_FALSE;
186    }
187
188    ssize_t bytesCount = buffer->getStride() * bytesPerPixel(buffer->getPixelFormat());
189
190    SkBitmap bitmap;
191    bitmap.setInfo(SkImageInfo::Make(buffer->getWidth(), buffer->getHeight(),
192                                     convertPixelFormat(buffer->getPixelFormat()),
193                                     kPremul_SkAlphaType),
194                   bytesCount);
195
196    if (buffer->getWidth() > 0 && buffer->getHeight() > 0) {
197        bitmap.setPixels(bits);
198    } else {
199        bitmap.setPixels(NULL);
200    }
201
202    SET_INT(canvas, gCanvasClassInfo.mSurfaceFormat, buffer->getPixelFormat());
203    INVOKEV(canvas, gCanvasClassInfo.setNativeBitmap, reinterpret_cast<jlong>(&bitmap));
204
205    SkRect clipRect;
206    clipRect.set(rect.left, rect.top, rect.right, rect.bottom);
207    SkCanvas* nativeCanvas = GraphicsJNI::getNativeCanvas(env, canvas);
208    nativeCanvas->clipRect(clipRect);
209
210    if (dirtyRect) {
211        INVOKEV(dirtyRect, gRectClassInfo.set,
212                int(rect.left), int(rect.top), int(rect.right), int(rect.bottom));
213    }
214
215    return JNI_TRUE;
216}
217
218static jboolean android_view_GraphicBuffer_unlockCanvasAndPost(JNIEnv* env, jobject,
219        jlong wrapperHandle, jobject canvas) {
220
221    GraphicBufferWrapper* wrapper =
222                reinterpret_cast<GraphicBufferWrapper*>(wrapperHandle);
223    INVOKEV(canvas, gCanvasClassInfo.setNativeBitmap, (jlong)0);
224
225    if (wrapper) {
226        status_t status = wrapper->buffer->unlock();
227        return status == 0 ? JNI_TRUE : JNI_FALSE;
228    }
229
230    return JNI_FALSE;
231}
232
233// ----------------------------------------------------------------------------
234// Serialization
235// ----------------------------------------------------------------------------
236
237static void android_view_GraphiceBuffer_write(JNIEnv* env, jobject clazz,
238        jlong wrapperHandle, jobject dest) {
239    GraphicBufferWrapper* wrapper =
240                reinterpret_cast<GraphicBufferWrapper*>(wrapperHandle);
241    Parcel* parcel = parcelForJavaObject(env, dest);
242    if (parcel) {
243        parcel->write(*wrapper->buffer);
244    }
245}
246
247static jlong android_view_GraphiceBuffer_read(JNIEnv* env, jobject clazz,
248        jobject in) {
249
250    Parcel* parcel = parcelForJavaObject(env, in);
251    if (parcel) {
252        sp<GraphicBuffer> buffer = new GraphicBuffer();
253        parcel->read(*buffer);
254        return reinterpret_cast<jlong>(new GraphicBufferWrapper(buffer));
255    }
256
257    return NULL;
258}
259
260// ----------------------------------------------------------------------------
261// External helpers
262// ----------------------------------------------------------------------------
263
264sp<GraphicBuffer> graphicBufferForJavaObject(JNIEnv* env, jobject obj) {
265    if (obj) {
266        jlong nativeObject = env->GetLongField(obj, gGraphicBufferClassInfo.mNativeObject);
267        GraphicBufferWrapper* wrapper = (GraphicBufferWrapper*) nativeObject;
268        if (wrapper != NULL) {
269            sp<GraphicBuffer> buffer(wrapper->buffer);
270            return buffer;
271        }
272    }
273    return NULL;
274}
275
276// ----------------------------------------------------------------------------
277// JNI Glue
278// ----------------------------------------------------------------------------
279
280#define FIND_CLASS(var, className) \
281        var = env->FindClass(className); \
282        LOG_FATAL_IF(! var, "Unable to find class " className);
283
284#define GET_FIELD_ID(var, clazz, fieldName, fieldDescriptor) \
285        var = env->GetFieldID(clazz, fieldName, fieldDescriptor); \
286        LOG_FATAL_IF(! var, "Unable to find field " fieldName);
287
288#define GET_METHOD_ID(var, clazz, methodName, methodDescriptor) \
289        var = env->GetMethodID(clazz, methodName, methodDescriptor); \
290        LOG_FATAL_IF(!var, "Unable to find method " methodName);
291
292const char* const kClassPathName = "android/view/GraphicBuffer";
293
294static JNINativeMethod gMethods[] = {
295    { "nCreateGraphicBuffer",  "(IIII)J", (void*) android_view_GraphiceBuffer_create },
296    { "nDestroyGraphicBuffer", "(J)V",    (void*) android_view_GraphiceBuffer_destroy },
297
298    { "nWriteGraphicBufferToParcel",  "(JLandroid/os/Parcel;)V",
299            (void*) android_view_GraphiceBuffer_write },
300    { "nReadGraphicBufferFromParcel", "(Landroid/os/Parcel;)J",
301            (void*) android_view_GraphiceBuffer_read },
302
303    { "nLockCanvas", "(JLandroid/graphics/Canvas;Landroid/graphics/Rect;)Z",
304            (void*) android_view_GraphicBuffer_lockCanvas },
305    { "nUnlockCanvasAndPost", "(JLandroid/graphics/Canvas;)Z",
306            (void*) android_view_GraphicBuffer_unlockCanvasAndPost },
307};
308
309int register_android_view_GraphicBuffer(JNIEnv* env) {
310    jclass clazz;
311    FIND_CLASS(clazz, "android/view/GraphicBuffer");
312    GET_FIELD_ID(gGraphicBufferClassInfo.mNativeObject, clazz, "mNativeObject", "J");
313
314    FIND_CLASS(clazz, "android/graphics/Rect");
315    GET_METHOD_ID(gRectClassInfo.set, clazz, "set", "(IIII)V");
316    GET_FIELD_ID(gRectClassInfo.left, clazz, "left", "I");
317    GET_FIELD_ID(gRectClassInfo.top, clazz, "top", "I");
318    GET_FIELD_ID(gRectClassInfo.right, clazz, "right", "I");
319    GET_FIELD_ID(gRectClassInfo.bottom, clazz, "bottom", "I");
320
321    FIND_CLASS(clazz, "android/graphics/Canvas");
322    GET_FIELD_ID(gCanvasClassInfo.mSurfaceFormat, clazz, "mSurfaceFormat", "I");
323    GET_METHOD_ID(gCanvasClassInfo.setNativeBitmap, clazz, "setNativeBitmap", "(J)V");
324
325    return AndroidRuntime::registerNativeMethods(env, kClassPathName, gMethods, NELEM(gMethods));
326}
327
328};
329