blob: 2c10212f5d0a1cb8afc0aacbe1048c0ff99cedcd [file] [log] [blame]
John Reckcec24ae2013-11-05 13:27:50 -08001/*
2 * Copyright (C) 2010 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
John Reck23b797a2014-01-03 18:08:34 -080017#define LOG_TAG "ThreadedRenderer"
John Reckcec24ae2013-11-05 13:27:50 -080018
John Recke45b1fd2014-04-15 09:50:16 -070019#include <algorithm>
20
John Reckcec24ae2013-11-05 13:27:50 -080021#include "jni.h"
22#include <nativehelper/JNIHelp.h>
23#include <android_runtime/AndroidRuntime.h>
24
John Reck4f02bf42014-01-03 18:09:17 -080025#include <utils/StrongPointer.h>
26#include <android_runtime/android_view_Surface.h>
27#include <system/window.h>
28
John Reck66f0be62014-05-13 13:39:31 -070029#include "android_view_GraphicBuffer.h"
30
John Reck52244ff2014-05-01 21:27:37 -070031#include <Animator.h>
John Recke45b1fd2014-04-15 09:50:16 -070032#include <RenderNode.h>
John Reck66f0be62014-05-13 13:39:31 -070033#include <renderthread/CanvasContext.h>
John Reck4f02bf42014-01-03 18:09:17 -080034#include <renderthread/RenderProxy.h>
John Reckcec24ae2013-11-05 13:27:50 -080035#include <renderthread/RenderTask.h>
36#include <renderthread/RenderThread.h>
37
38namespace android {
39
40#ifdef USE_OPENGL_RENDERER
41
John Reck4f02bf42014-01-03 18:09:17 -080042using namespace android::uirenderer;
43using namespace android::uirenderer::renderthread;
John Reckcec24ae2013-11-05 13:27:50 -080044
45static jmethodID gRunnableMethod;
46
John Reck4f02bf42014-01-03 18:09:17 -080047class JavaTask : public RenderTask {
John Reckcec24ae2013-11-05 13:27:50 -080048public:
49 JavaTask(JNIEnv* env, jobject jrunnable) {
50 env->GetJavaVM(&mVm);
51 mRunnable = env->NewGlobalRef(jrunnable);
52 }
53
John Reckcec24ae2013-11-05 13:27:50 -080054 virtual void run() {
55 env()->CallVoidMethod(mRunnable, gRunnableMethod);
John Reck4f02bf42014-01-03 18:09:17 -080056 env()->DeleteGlobalRef(mRunnable);
57 delete this;
John Reckcec24ae2013-11-05 13:27:50 -080058 };
59
60private:
61 JNIEnv* env() {
62 JNIEnv* env;
63 if (mVm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6) != JNI_OK) {
64 return 0;
65 }
66 return env;
67 }
68
69 JavaVM* mVm;
70 jobject mRunnable;
71};
72
John Reck66f0be62014-05-13 13:39:31 -070073class SetAtlasTask : public RenderTask {
74public:
75 SetAtlasTask(const sp<GraphicBuffer>& buffer, int64_t* map, size_t size)
76 : mBuffer(buffer)
77 , mMap(map)
78 , mMapSize(size) {
79 }
80
81 virtual void run() {
82 CanvasContext::setTextureAtlas(mBuffer, mMap, mMapSize);
83 mMap = 0;
84 delete this;
85 }
86
87private:
88 sp<GraphicBuffer> mBuffer;
89 int64_t* mMap;
90 size_t mMapSize;
91};
92
John Reck52244ff2014-05-01 21:27:37 -070093class OnFinishedEvent {
94public:
John Reckff941dc2014-05-14 16:34:14 -070095 OnFinishedEvent(BaseRenderNodeAnimator* animator, AnimationListener* listener)
John Reck52244ff2014-05-01 21:27:37 -070096 : animator(animator), listener(listener) {}
John Reckff941dc2014-05-14 16:34:14 -070097 sp<BaseRenderNodeAnimator> animator;
John Reck52244ff2014-05-01 21:27:37 -070098 sp<AnimationListener> listener;
99};
100
John Recke45b1fd2014-04-15 09:50:16 -0700101class InvokeAnimationListeners : public MessageHandler {
102public:
John Reck52244ff2014-05-01 21:27:37 -0700103 InvokeAnimationListeners(std::vector<OnFinishedEvent>& events) {
104 mOnFinishedEvents.swap(events);
John Recke45b1fd2014-04-15 09:50:16 -0700105 }
106
John Reck52244ff2014-05-01 21:27:37 -0700107 static void callOnFinished(OnFinishedEvent& event) {
108 event.listener->onAnimationFinished(event.animator.get());
John Recke45b1fd2014-04-15 09:50:16 -0700109 }
110
111 virtual void handleMessage(const Message& message) {
John Reck52244ff2014-05-01 21:27:37 -0700112 std::for_each(mOnFinishedEvents.begin(), mOnFinishedEvents.end(), callOnFinished);
113 mOnFinishedEvents.clear();
John Recke45b1fd2014-04-15 09:50:16 -0700114 }
115
116private:
John Reck52244ff2014-05-01 21:27:37 -0700117 std::vector<OnFinishedEvent> mOnFinishedEvents;
John Recke45b1fd2014-04-15 09:50:16 -0700118};
119
John Reck52244ff2014-05-01 21:27:37 -0700120class RootRenderNode : public RenderNode, public AnimationHook {
John Recke45b1fd2014-04-15 09:50:16 -0700121public:
122 RootRenderNode() : RenderNode() {
123 mLooper = Looper::getForThread();
124 LOG_ALWAYS_FATAL_IF(!mLooper.get(),
125 "Must create RootRenderNode on a thread with a looper!");
126 }
127
128 virtual ~RootRenderNode() {}
129
John Reckff941dc2014-05-14 16:34:14 -0700130 virtual void callOnFinished(BaseRenderNodeAnimator* animator, AnimationListener* listener) {
John Reck52244ff2014-05-01 21:27:37 -0700131 OnFinishedEvent event(animator, listener);
132 mOnFinishedEvents.push_back(event);
John Recke45b1fd2014-04-15 09:50:16 -0700133 }
134
135 virtual void prepareTree(TreeInfo& info) {
John Reck52244ff2014-05-01 21:27:37 -0700136 info.animationHook = this;
John Recke45b1fd2014-04-15 09:50:16 -0700137 RenderNode::prepareTree(info);
John Reck52244ff2014-05-01 21:27:37 -0700138 info.animationHook = NULL;
John Recke45b1fd2014-04-15 09:50:16 -0700139
140 // post all the finished stuff
John Reck52244ff2014-05-01 21:27:37 -0700141 if (mOnFinishedEvents.size()) {
John Recke45b1fd2014-04-15 09:50:16 -0700142 sp<InvokeAnimationListeners> message
John Reck52244ff2014-05-01 21:27:37 -0700143 = new InvokeAnimationListeners(mOnFinishedEvents);
John Recke45b1fd2014-04-15 09:50:16 -0700144 mLooper->sendMessage(message, 0);
145 }
146 }
147
148private:
149 sp<Looper> mLooper;
John Reck52244ff2014-05-01 21:27:37 -0700150 std::vector<OnFinishedEvent> mOnFinishedEvents;
John Recke45b1fd2014-04-15 09:50:16 -0700151};
152
John Reck66f0be62014-05-13 13:39:31 -0700153static void android_view_ThreadedRenderer_setAtlas(JNIEnv* env, jobject clazz,
154 jobject graphicBuffer, jlongArray atlasMapArray) {
155 sp<GraphicBuffer> buffer = graphicBufferForJavaObject(env, graphicBuffer);
156 jsize len = env->GetArrayLength(atlasMapArray);
157 if (len <= 0) {
158 ALOGW("Failed to initialize atlas, invalid map length: %d", len);
159 return;
160 }
161 int64_t* map = new int64_t[len];
162 env->GetLongArrayRegion(atlasMapArray, 0, len, map);
163
164 SetAtlasTask* task = new SetAtlasTask(buffer, map, len);
John Reck4f02bf42014-01-03 18:09:17 -0800165 RenderThread::getInstance().queue(task);
166}
167
John Recke45b1fd2014-04-15 09:50:16 -0700168static jlong android_view_ThreadedRenderer_createRootRenderNode(JNIEnv* env, jobject clazz) {
169 RootRenderNode* node = new RootRenderNode();
170 node->incStrong(0);
171 node->setName("RootRenderNode");
172 return reinterpret_cast<jlong>(node);
173}
174
John Reck4f02bf42014-01-03 18:09:17 -0800175static jlong android_view_ThreadedRenderer_createProxy(JNIEnv* env, jobject clazz,
John Recke45b1fd2014-04-15 09:50:16 -0700176 jboolean translucent, jlong rootRenderNodePtr) {
177 RenderNode* rootRenderNode = reinterpret_cast<RenderNode*>(rootRenderNodePtr);
178 return (jlong) new RenderProxy(translucent, rootRenderNode);
John Reck4f02bf42014-01-03 18:09:17 -0800179}
180
181static void android_view_ThreadedRenderer_deleteProxy(JNIEnv* env, jobject clazz,
182 jlong proxyPtr) {
John Reck19b6bcf2014-02-14 20:03:38 -0800183 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reck4f02bf42014-01-03 18:09:17 -0800184 delete proxy;
185}
186
John Reck18f16e62014-05-02 16:46:41 -0700187static void android_view_ThreadedRenderer_setFrameInterval(JNIEnv* env, jobject clazz,
188 jlong proxyPtr, jlong frameIntervalNanos) {
189 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
190 proxy->setFrameInterval(frameIntervalNanos);
191}
192
John Recke4280ba2014-05-05 16:39:37 -0700193static jboolean android_view_ThreadedRenderer_loadSystemProperties(JNIEnv* env, jobject clazz,
194 jlong proxyPtr) {
195 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
196 return proxy->loadSystemProperties();
197}
198
John Reck4f02bf42014-01-03 18:09:17 -0800199static jboolean android_view_ThreadedRenderer_initialize(JNIEnv* env, jobject clazz,
200 jlong proxyPtr, jobject jsurface) {
John Reck19b6bcf2014-02-14 20:03:38 -0800201 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reck4f02bf42014-01-03 18:09:17 -0800202 sp<ANativeWindow> window = android_view_Surface_getNativeWindow(env, jsurface);
John Reckf7d9c1d2014-04-09 10:01:03 -0700203 return proxy->initialize(window);
John Reck4f02bf42014-01-03 18:09:17 -0800204}
205
206static void android_view_ThreadedRenderer_updateSurface(JNIEnv* env, jobject clazz,
207 jlong proxyPtr, jobject jsurface) {
John Reck19b6bcf2014-02-14 20:03:38 -0800208 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reck4f02bf42014-01-03 18:09:17 -0800209 sp<ANativeWindow> window;
210 if (jsurface) {
211 window = android_view_Surface_getNativeWindow(env, jsurface);
212 }
John Reckf7d9c1d2014-04-09 10:01:03 -0700213 proxy->updateSurface(window);
214}
215
216static void android_view_ThreadedRenderer_pauseSurface(JNIEnv* env, jobject clazz,
217 jlong proxyPtr, jobject jsurface) {
218 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
219 sp<ANativeWindow> window;
220 if (jsurface) {
221 window = android_view_Surface_getNativeWindow(env, jsurface);
222 }
223 proxy->pauseSurface(window);
John Reck4f02bf42014-01-03 18:09:17 -0800224}
225
226static void android_view_ThreadedRenderer_setup(JNIEnv* env, jobject clazz,
227 jlong proxyPtr, jint width, jint height) {
John Reck19b6bcf2014-02-14 20:03:38 -0800228 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reck4f02bf42014-01-03 18:09:17 -0800229 proxy->setup(width, height);
230}
231
John Reck63a06672014-05-07 13:45:54 -0700232static void android_view_ThreadedRenderer_setOpaque(JNIEnv* env, jobject clazz,
233 jlong proxyPtr, jboolean opaque) {
234 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
235 proxy->setOpaque(opaque);
236}
237
John Reckf9be7792014-05-02 18:21:16 -0700238static int android_view_ThreadedRenderer_syncAndDrawFrame(JNIEnv* env, jobject clazz,
John Reck18f16e62014-05-02 16:46:41 -0700239 jlong proxyPtr, jlong frameTimeNanos, jint dirtyLeft, jint dirtyTop,
John Reck4f02bf42014-01-03 18:09:17 -0800240 jint dirtyRight, jint dirtyBottom) {
John Reck19b6bcf2014-02-14 20:03:38 -0800241 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reckf9be7792014-05-02 18:21:16 -0700242 return proxy->syncAndDrawFrame(frameTimeNanos, dirtyLeft, dirtyTop, dirtyRight, dirtyBottom);
John Reck4f02bf42014-01-03 18:09:17 -0800243}
244
John Reckfae904d2014-04-14 11:01:57 -0700245static void android_view_ThreadedRenderer_destroyCanvasAndSurface(JNIEnv* env, jobject clazz,
John Reck4f02bf42014-01-03 18:09:17 -0800246 jlong proxyPtr) {
John Reck19b6bcf2014-02-14 20:03:38 -0800247 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reckfae904d2014-04-14 11:01:57 -0700248 proxy->destroyCanvasAndSurface();
John Reck4f02bf42014-01-03 18:09:17 -0800249}
250
John Reck0d1f6342014-03-28 20:30:27 -0700251static void android_view_ThreadedRenderer_invokeFunctor(JNIEnv* env, jobject clazz,
252 jlong proxyPtr, jlong functorPtr, jboolean waitForCompletion) {
253 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
254 Functor* functor = reinterpret_cast<Functor*>(functorPtr);
255 proxy->invokeFunctor(functor, waitForCompletion);
256}
257
John Reckfc53ef272014-02-11 10:40:25 -0800258static void android_view_ThreadedRenderer_runWithGlContext(JNIEnv* env, jobject clazz,
259 jlong proxyPtr, jobject jrunnable) {
John Reck19b6bcf2014-02-14 20:03:38 -0800260 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reckfc53ef272014-02-11 10:40:25 -0800261 RenderTask* task = new JavaTask(env, jrunnable);
262 proxy->runWithGlContext(task);
263}
264
John Reck19b6bcf2014-02-14 20:03:38 -0800265static jlong android_view_ThreadedRenderer_createDisplayListLayer(JNIEnv* env, jobject clazz,
266 jlong proxyPtr, jint width, jint height) {
267 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
268 DeferredLayerUpdater* layer = proxy->createDisplayListLayer(width, height);
269 return reinterpret_cast<jlong>(layer);
270}
271
272static jlong android_view_ThreadedRenderer_createTextureLayer(JNIEnv* env, jobject clazz,
273 jlong proxyPtr) {
274 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
275 DeferredLayerUpdater* layer = proxy->createTextureLayer();
276 return reinterpret_cast<jlong>(layer);
277}
278
279static jboolean android_view_ThreadedRenderer_copyLayerInto(JNIEnv* env, jobject clazz,
280 jlong proxyPtr, jlong layerPtr, jlong bitmapPtr) {
281 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
282 DeferredLayerUpdater* layer = reinterpret_cast<DeferredLayerUpdater*>(layerPtr);
283 SkBitmap* bitmap = reinterpret_cast<SkBitmap*>(bitmapPtr);
284 return proxy->copyLayerInto(layer, bitmap);
285}
286
287static void android_view_ThreadedRenderer_destroyLayer(JNIEnv* env, jobject clazz,
288 jlong proxyPtr, jlong layerPtr) {
289 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
290 DeferredLayerUpdater* layer = reinterpret_cast<DeferredLayerUpdater*>(layerPtr);
291 proxy->destroyLayer(layer);
292}
293
John Reck28ad7b52014-04-07 16:59:25 -0700294static void android_view_ThreadedRenderer_fence(JNIEnv* env, jobject clazz,
295 jlong proxyPtr) {
296 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
297 proxy->fence();
298}
299
John Reckcec24ae2013-11-05 13:27:50 -0800300#endif
301
302// ----------------------------------------------------------------------------
303// JNI Glue
304// ----------------------------------------------------------------------------
305
306const char* const kClassPathName = "android/view/ThreadedRenderer";
307
308static JNINativeMethod gMethods[] = {
309#ifdef USE_OPENGL_RENDERER
John Reck66f0be62014-05-13 13:39:31 -0700310 { "nSetAtlas", "(Landroid/view/GraphicBuffer;[J)V", (void*) android_view_ThreadedRenderer_setAtlas },
John Recke45b1fd2014-04-15 09:50:16 -0700311 { "nCreateRootRenderNode", "()J", (void*) android_view_ThreadedRenderer_createRootRenderNode },
312 { "nCreateProxy", "(ZJ)J", (void*) android_view_ThreadedRenderer_createProxy },
John Reck4f02bf42014-01-03 18:09:17 -0800313 { "nDeleteProxy", "(J)V", (void*) android_view_ThreadedRenderer_deleteProxy },
John Reck18f16e62014-05-02 16:46:41 -0700314 { "nSetFrameInterval", "(JJ)V", (void*) android_view_ThreadedRenderer_setFrameInterval },
John Recke4280ba2014-05-05 16:39:37 -0700315 { "nLoadSystemProperties", "(J)Z", (void*) android_view_ThreadedRenderer_loadSystemProperties },
John Reck4f02bf42014-01-03 18:09:17 -0800316 { "nInitialize", "(JLandroid/view/Surface;)Z", (void*) android_view_ThreadedRenderer_initialize },
317 { "nUpdateSurface", "(JLandroid/view/Surface;)V", (void*) android_view_ThreadedRenderer_updateSurface },
John Reckf7d9c1d2014-04-09 10:01:03 -0700318 { "nPauseSurface", "(JLandroid/view/Surface;)V", (void*) android_view_ThreadedRenderer_pauseSurface },
John Reck4f02bf42014-01-03 18:09:17 -0800319 { "nSetup", "(JII)V", (void*) android_view_ThreadedRenderer_setup },
John Reck63a06672014-05-07 13:45:54 -0700320 { "nSetOpaque", "(JZ)V", (void*) android_view_ThreadedRenderer_setOpaque },
John Reckf9be7792014-05-02 18:21:16 -0700321 { "nSyncAndDrawFrame", "(JJIIII)I", (void*) android_view_ThreadedRenderer_syncAndDrawFrame },
John Reckfae904d2014-04-14 11:01:57 -0700322 { "nDestroyCanvasAndSurface", "(J)V", (void*) android_view_ThreadedRenderer_destroyCanvasAndSurface },
John Reck0d1f6342014-03-28 20:30:27 -0700323 { "nInvokeFunctor", "(JJZ)V", (void*) android_view_ThreadedRenderer_invokeFunctor },
John Reckfc53ef272014-02-11 10:40:25 -0800324 { "nRunWithGlContext", "(JLjava/lang/Runnable;)V", (void*) android_view_ThreadedRenderer_runWithGlContext },
John Reck19b6bcf2014-02-14 20:03:38 -0800325 { "nCreateDisplayListLayer", "(JII)J", (void*) android_view_ThreadedRenderer_createDisplayListLayer },
326 { "nCreateTextureLayer", "(J)J", (void*) android_view_ThreadedRenderer_createTextureLayer },
327 { "nCopyLayerInto", "(JJJ)Z", (void*) android_view_ThreadedRenderer_copyLayerInto },
328 { "nDestroyLayer", "(JJ)V", (void*) android_view_ThreadedRenderer_destroyLayer },
John Reck28ad7b52014-04-07 16:59:25 -0700329 { "nFence", "(J)V", (void*) android_view_ThreadedRenderer_fence },
John Reckcec24ae2013-11-05 13:27:50 -0800330#endif
331};
332
333int register_android_view_ThreadedRenderer(JNIEnv* env) {
334#ifdef USE_OPENGL_RENDERER
335 jclass cls = env->FindClass("java/lang/Runnable");
336 gRunnableMethod = env->GetMethodID(cls, "run", "()V");
337 env->DeleteLocalRef(cls);
338#endif
339 return AndroidRuntime::registerNativeMethods(env, kClassPathName, gMethods, NELEM(gMethods));
340}
341
342}; // namespace android