• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2017 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 #undef ANDROID_UTILS_REF_BASE_DISABLE_IMPLICIT_CONSTRUCTION // TODO:remove this and fix code
17 
18 #define LOG_TAG "HardwareBuffer"
19 
20 #include "jni.h"
21 #include <nativehelper/JNIHelp.h>
22 
23 #include "android_os_Parcel.h"
24 
25 #include <android/hardware_buffer.h>
26 #include <android_runtime/android_graphics_GraphicBuffer.h>
27 #include <android_runtime/android_hardware_HardwareBuffer.h>
28 #include <android_runtime/AndroidRuntime.h>
29 #include <android_runtime/Log.h>
30 #include <private/android/AHardwareBufferHelpers.h>
31 
32 #include <binder/Parcel.h>
33 
34 #include <private/gui/ComposerService.h>
35 #include <ui/PixelFormat.h>
36 
37 #include <hardware/gralloc1.h>
38 #include <grallocusage/GrallocUsageConversion.h>
39 
40 #include "core_jni_helpers.h"
41 
42 using namespace android;
43 
44 // ----------------------------------------------------------------------------
45 // Defines
46 // ----------------------------------------------------------------------------
47 
48 // Debug
49 static constexpr bool kDebugGraphicBuffer = false;
50 
51 // ----------------------------------------------------------------------------
52 // Types
53 // ----------------------------------------------------------------------------
54 
55 static struct {
56     jclass clazz;
57     jfieldID mNativeObject;
58     jmethodID ctor;
59 } gHardwareBufferClassInfo;
60 
61 class GraphicBufferWrapper {
62 public:
GraphicBufferWrapper(const sp<GraphicBuffer> & buffer)63     explicit GraphicBufferWrapper(const sp<GraphicBuffer>& buffer)
64             : buffer(buffer) {}
65 
66     sp<GraphicBuffer> buffer;
67 };
68 
69 // ----------------------------------------------------------------------------
70 // HardwareBuffer lifecycle
71 // ----------------------------------------------------------------------------
72 
android_hardware_HardwareBuffer_create(JNIEnv * env,jobject clazz,jint width,jint height,jint format,jint layers,jlong usage)73 static jlong android_hardware_HardwareBuffer_create(JNIEnv* env, jobject clazz,
74         jint width, jint height, jint format, jint layers, jlong usage) {
75 
76     // TODO: update createGraphicBuffer to take two 64-bit values.
77     int pixelFormat = android_hardware_HardwareBuffer_convertToPixelFormat(format);
78     if (pixelFormat == 0) {
79         if (kDebugGraphicBuffer) {
80             ALOGW("createGraphicBufferAlloc() invalid pixel format in HardwareBuffer.create()");
81         }
82         return NULL;
83     }
84 
85     uint64_t grallocUsage = AHardwareBuffer_convertToGrallocUsageBits(usage);
86     sp<GraphicBuffer> buffer = new GraphicBuffer(width, height, pixelFormat, layers,
87             grallocUsage, std::string("HardwareBuffer pid [") + std::to_string(getpid()) +"]");
88     status_t error = buffer->initCheck();
89     if (error != OK) {
90         if (kDebugGraphicBuffer) {
91             ALOGW("createGraphicBuffer() failed in HardwareBuffer.create()");
92         }
93         return NULL;
94     }
95 
96     GraphicBufferWrapper* wrapper = new GraphicBufferWrapper(buffer);
97     return reinterpret_cast<jlong>(wrapper);
98 }
99 
android_hardware_HardwareBuffer_createFromGraphicBuffer(JNIEnv * env,jobject clazz,jobject graphicBuffer)100 static jlong android_hardware_HardwareBuffer_createFromGraphicBuffer(JNIEnv* env, jobject clazz, jobject graphicBuffer) {
101     sp<GraphicBuffer> buffer(android_graphics_GraphicBuffer_getNativeGraphicsBuffer(env,
102                                                                                     graphicBuffer));
103     GraphicBufferWrapper* wrapper = new GraphicBufferWrapper(buffer);
104     return reinterpret_cast<jlong>(wrapper);
105 }
106 
destroyWrapper(GraphicBufferWrapper * wrapper)107 static void destroyWrapper(GraphicBufferWrapper* wrapper) {
108     delete wrapper;
109 }
110 
android_hardware_HardwareBuffer_getNativeFinalizer(JNIEnv * env,jobject clazz)111 static jlong android_hardware_HardwareBuffer_getNativeFinalizer(JNIEnv* env, jobject clazz) {
112     return static_cast<jlong>(reinterpret_cast<uintptr_t>(&destroyWrapper));
113 }
114 
android_hardware_HardwareBuffer_isSupported(JNIEnv * env,jobject clazz,jint width,jint height,jint format,jint layers,jlong usage)115 static jboolean android_hardware_HardwareBuffer_isSupported(JNIEnv* env, jobject clazz,
116         jint width, jint height, jint format, jint layers, jlong usage) {
117 
118     AHardwareBuffer_Desc desc;
119     desc.width = width;
120     desc.height = height;
121     desc.format = format;
122     desc.layers = layers;
123     desc.usage = usage;
124     desc.stride = 0;
125     desc.rfu0 = 0;
126     desc.rfu1 = 0;
127     return AHardwareBuffer_isSupported(&desc);
128 }
129 
130 //----------------------------------------------------------------------------
131 // Accessors
132 // ----------------------------------------------------------------------------
133 
GraphicBufferWrapper_to_GraphicBuffer(jlong nativeObject)134 static inline GraphicBuffer* GraphicBufferWrapper_to_GraphicBuffer(
135         jlong nativeObject) {
136     return reinterpret_cast<GraphicBufferWrapper*>(nativeObject)->buffer.get();
137 }
138 
android_hardware_HardwareBuffer_getWidth(JNIEnv * env,jobject clazz,jlong nativeObject)139 static jint android_hardware_HardwareBuffer_getWidth(JNIEnv* env, jobject clazz,
140     jlong nativeObject) {
141     GraphicBuffer* buffer = GraphicBufferWrapper_to_GraphicBuffer(nativeObject);
142     return static_cast<jint>(buffer->getWidth());
143 }
144 
android_hardware_HardwareBuffer_getHeight(JNIEnv * env,jobject clazz,jlong nativeObject)145 static jint android_hardware_HardwareBuffer_getHeight(JNIEnv* env,
146     jobject clazz, jlong nativeObject) {
147     GraphicBuffer* buffer = GraphicBufferWrapper_to_GraphicBuffer(nativeObject);
148     return static_cast<jint>(buffer->getHeight());
149 }
150 
android_hardware_HardwareBuffer_getFormat(JNIEnv * env,jobject clazz,jlong nativeObject)151 static jint android_hardware_HardwareBuffer_getFormat(JNIEnv* env,
152     jobject clazz, jlong nativeObject) {
153     GraphicBuffer* buffer = GraphicBufferWrapper_to_GraphicBuffer(nativeObject);
154     return static_cast<jint>(android_hardware_HardwareBuffer_convertFromPixelFormat(
155             buffer->getPixelFormat()));
156 }
157 
android_hardware_HardwareBuffer_getLayers(JNIEnv * env,jobject clazz,jlong nativeObject)158 static jint android_hardware_HardwareBuffer_getLayers(JNIEnv* env,
159     jobject clazz, jlong nativeObject) {
160     GraphicBuffer* buffer = GraphicBufferWrapper_to_GraphicBuffer(nativeObject);
161     return static_cast<jint>(buffer->getLayerCount());
162 }
163 
android_hardware_HardwareBuffer_getUsage(JNIEnv * env,jobject clazz,jlong nativeObject)164 static jlong android_hardware_HardwareBuffer_getUsage(JNIEnv* env,
165     jobject clazz, jlong nativeObject) {
166     GraphicBuffer* buffer = GraphicBufferWrapper_to_GraphicBuffer(nativeObject);
167     return static_cast<jlong>(AHardwareBuffer_convertFromGrallocUsageBits(buffer->getUsage()));
168 }
169 
android_hardware_HardwareBuffer_estimateSize(jlong nativeObject)170 static jlong android_hardware_HardwareBuffer_estimateSize(jlong nativeObject) {
171     GraphicBuffer* buffer = GraphicBufferWrapper_to_GraphicBuffer(nativeObject);
172 
173     uint32_t bpp = bytesPerPixel(buffer->getPixelFormat());
174     if (bpp == 0) {
175         // If the pixel format is not recognized, use 1 as default.
176         bpp = 1;
177     }
178 
179     const uint32_t bufferStride =
180             buffer->getStride() > 0 ? buffer->getStride() : buffer->getWidth();
181     return static_cast<jlong>(static_cast<uint64_t>(buffer->getHeight() * bufferStride * bpp));
182 }
183 
android_hardware_HardwareBuffer_getId(jlong nativeObject)184 static jlong android_hardware_HardwareBuffer_getId(jlong nativeObject) {
185     GraphicBuffer* buffer = GraphicBufferWrapper_to_GraphicBuffer(nativeObject);
186     return static_cast<jlong>(buffer->getId());
187 }
188 
189 // ----------------------------------------------------------------------------
190 // Serialization
191 // ----------------------------------------------------------------------------
192 
android_hardware_HardwareBuffer_write(JNIEnv * env,jobject clazz,jlong nativeObject,jobject dest)193 static void android_hardware_HardwareBuffer_write(JNIEnv* env, jobject clazz,
194         jlong nativeObject, jobject dest) {
195     GraphicBuffer* buffer = GraphicBufferWrapper_to_GraphicBuffer(nativeObject);
196     Parcel* parcel = parcelForJavaObject(env, dest);
197     if (parcel) {
198         parcel->write(*buffer);
199     }
200 }
201 
android_hardware_HardwareBuffer_read(JNIEnv * env,jobject clazz,jobject in)202 static jlong android_hardware_HardwareBuffer_read(JNIEnv* env, jobject clazz,
203         jobject in) {
204     Parcel* parcel = parcelForJavaObject(env, in);
205     if (parcel) {
206         sp<GraphicBuffer> buffer = new GraphicBuffer();
207         if (parcel->read(*buffer) == STATUS_OK) {
208             return reinterpret_cast<jlong>(new GraphicBufferWrapper(buffer));
209         }
210     }
211     return NULL;
212 }
213 
214 // ----------------------------------------------------------------------------
215 // Public functions
216 // ----------------------------------------------------------------------------
217 
218 namespace android {
219 
android_hardware_HardwareBuffer_getNativeHardwareBuffer(JNIEnv * env,jobject hardwareBufferObj)220 AHardwareBuffer* android_hardware_HardwareBuffer_getNativeHardwareBuffer(
221         JNIEnv* env, jobject hardwareBufferObj) {
222     if (env->IsInstanceOf(hardwareBufferObj, gHardwareBufferClassInfo.clazz)) {
223         GraphicBuffer* buffer = GraphicBufferWrapper_to_GraphicBuffer(
224                 env->GetLongField(hardwareBufferObj, gHardwareBufferClassInfo.mNativeObject));
225         return AHardwareBuffer_from_GraphicBuffer(buffer);
226 
227     } else {
228         return nullptr;
229     }
230 }
231 
android_hardware_HardwareBuffer_createFromAHardwareBuffer(JNIEnv * env,AHardwareBuffer * hardwareBuffer)232 jobject android_hardware_HardwareBuffer_createFromAHardwareBuffer(
233         JNIEnv* env, AHardwareBuffer* hardwareBuffer) {
234     GraphicBuffer* buffer = AHardwareBuffer_to_GraphicBuffer(hardwareBuffer);
235     GraphicBufferWrapper* wrapper = new GraphicBufferWrapper(buffer);
236     jobject hardwareBufferObj = env->NewObject(gHardwareBufferClassInfo.clazz,
237             gHardwareBufferClassInfo.ctor, reinterpret_cast<jlong>(wrapper));
238     if (hardwareBufferObj == NULL) {
239         delete wrapper;
240         if (env->ExceptionCheck()) {
241             ALOGE("Could not create instance of HardwareBuffer from AHardwareBuffer.");
242             LOGE_EX(env);
243             env->ExceptionClear();
244         }
245         return nullptr;
246     }
247     return hardwareBufferObj;
248 }
249 
android_hardware_HardwareBuffer_convertFromPixelFormat(uint32_t format)250 uint32_t android_hardware_HardwareBuffer_convertFromPixelFormat(uint32_t format) {
251     return AHardwareBuffer_convertFromPixelFormat(format);
252 }
253 
android_hardware_HardwareBuffer_convertToPixelFormat(uint32_t format)254 uint32_t android_hardware_HardwareBuffer_convertToPixelFormat(uint32_t format) {
255     return AHardwareBuffer_convertToPixelFormat(format);
256 }
257 
android_hardware_HardwareBuffer_convertToGrallocUsageBits(uint64_t usage)258 uint64_t android_hardware_HardwareBuffer_convertToGrallocUsageBits(uint64_t usage) {
259     return AHardwareBuffer_convertToGrallocUsageBits(usage);
260 }
261 
262 }  // namespace android
263 
264 // ----------------------------------------------------------------------------
265 // JNI Glue
266 // ----------------------------------------------------------------------------
267 
268 const char* const kClassPathName = "android/hardware/HardwareBuffer";
269 
270 // clang-format off
271 static const JNINativeMethod gMethods[] = {
nCreateHardwareBuffer(IIIIJ)272     { "nCreateHardwareBuffer",  "(IIIIJ)J",
273             (void*) android_hardware_HardwareBuffer_create },
nCreateFromGraphicBuffer(Landroid/graphics/GraphicBuffer;)274     { "nCreateFromGraphicBuffer", "(Landroid/graphics/GraphicBuffer;)J",
275             (void*) android_hardware_HardwareBuffer_createFromGraphicBuffer },
nGetNativeFinalizer()276     { "nGetNativeFinalizer", "()J",
277             (void*) android_hardware_HardwareBuffer_getNativeFinalizer },
nWriteHardwareBufferToParcel(JLandroid/os/Parcel;)278     { "nWriteHardwareBufferToParcel",  "(JLandroid/os/Parcel;)V",
279             (void*) android_hardware_HardwareBuffer_write },
nReadHardwareBufferFromParcel(Landroid/os/Parcel;)280     { "nReadHardwareBufferFromParcel", "(Landroid/os/Parcel;)J",
281             (void*) android_hardware_HardwareBuffer_read },
nIsSupported(IIIIJ)282     { "nIsSupported",  "(IIIIJ)Z",
283             (void*) android_hardware_HardwareBuffer_isSupported },
284 
285     // --------------- @FastNative ----------------------
nGetWidth(J)286     { "nGetWidth", "(J)I",      (void*) android_hardware_HardwareBuffer_getWidth },
nGetHeight(J)287     { "nGetHeight", "(J)I",     (void*) android_hardware_HardwareBuffer_getHeight },
nGetFormat(J)288     { "nGetFormat", "(J)I",     (void*) android_hardware_HardwareBuffer_getFormat },
nGetLayers(J)289     { "nGetLayers", "(J)I",     (void*) android_hardware_HardwareBuffer_getLayers },
nGetUsage(J)290     { "nGetUsage", "(J)J",      (void*) android_hardware_HardwareBuffer_getUsage },
291 
292     // --------------- @CriticalNative ----------------------
nEstimateSize(J)293     { "nEstimateSize", "(J)J",  (void*) android_hardware_HardwareBuffer_estimateSize },
nGetId(J)294     { "nGetId", "(J)J",  (void*) android_hardware_HardwareBuffer_getId },
295 };
296 // clang-format on
297 
register_android_hardware_HardwareBuffer(JNIEnv * env)298 int register_android_hardware_HardwareBuffer(JNIEnv* env) {
299     int err = RegisterMethodsOrDie(env, kClassPathName, gMethods,
300             NELEM(gMethods));
301 
302     jclass clazz = FindClassOrDie(env, "android/hardware/HardwareBuffer");
303     gHardwareBufferClassInfo.clazz = MakeGlobalRefOrDie(env, clazz);
304     gHardwareBufferClassInfo.mNativeObject = GetFieldIDOrDie(env,
305             gHardwareBufferClassInfo.clazz, "mNativeObject", "J");
306     gHardwareBufferClassInfo.ctor = GetMethodIDOrDie(env,
307             gHardwareBufferClassInfo.clazz, "<init>", "(J)V");
308 
309     return err;
310 }
311