blob: 58fc1e1278e53e07b89d53984c78338e081cca4b [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 Recke45b1fd2014-04-15 09:50:16 -070029#include "android_view_RenderNodeAnimator.h"
30#include <RenderNode.h>
John Reck4f02bf42014-01-03 18:09:17 -080031#include <renderthread/RenderProxy.h>
John Reckcec24ae2013-11-05 13:27:50 -080032#include <renderthread/RenderTask.h>
33#include <renderthread/RenderThread.h>
34
35namespace android {
36
37#ifdef USE_OPENGL_RENDERER
38
John Reck4f02bf42014-01-03 18:09:17 -080039using namespace android::uirenderer;
40using namespace android::uirenderer::renderthread;
John Reckcec24ae2013-11-05 13:27:50 -080041
42static jmethodID gRunnableMethod;
43
John Reck4f02bf42014-01-03 18:09:17 -080044class JavaTask : public RenderTask {
John Reckcec24ae2013-11-05 13:27:50 -080045public:
46 JavaTask(JNIEnv* env, jobject jrunnable) {
47 env->GetJavaVM(&mVm);
48 mRunnable = env->NewGlobalRef(jrunnable);
49 }
50
John Reckcec24ae2013-11-05 13:27:50 -080051 virtual void run() {
52 env()->CallVoidMethod(mRunnable, gRunnableMethod);
John Reck4f02bf42014-01-03 18:09:17 -080053 env()->DeleteGlobalRef(mRunnable);
54 delete this;
John Reckcec24ae2013-11-05 13:27:50 -080055 };
56
57private:
58 JNIEnv* env() {
59 JNIEnv* env;
60 if (mVm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6) != JNI_OK) {
61 return 0;
62 }
63 return env;
64 }
65
66 JavaVM* mVm;
67 jobject mRunnable;
68};
69
John Recke45b1fd2014-04-15 09:50:16 -070070class InvokeAnimationListeners : public MessageHandler {
71public:
72 InvokeAnimationListeners(std::vector< sp<RenderNodeAnimator> >& animators) {
73 mAnimators.swap(animators);
74 }
75
76 static void callOnFinished(const sp<RenderNodeAnimator>& animator) {
77 animator->callOnFinished();
78 }
79
80 virtual void handleMessage(const Message& message) {
81 std::for_each(mAnimators.begin(), mAnimators.end(), callOnFinished);
82 mAnimators.clear();
83 }
84
85private:
86 std::vector< sp<RenderNodeAnimator> > mAnimators;
87};
88
89class RootRenderNode : public RenderNode, public AnimationListener {
90public:
91 RootRenderNode() : RenderNode() {
92 mLooper = Looper::getForThread();
93 LOG_ALWAYS_FATAL_IF(!mLooper.get(),
94 "Must create RootRenderNode on a thread with a looper!");
95 }
96
97 virtual ~RootRenderNode() {}
98
99 void onAnimationFinished(const sp<RenderPropertyAnimator>& animator) {
100 mFinishedAnimators.push_back(
101 reinterpret_cast<RenderNodeAnimator*>(animator.get()));
102 }
103
104 virtual void prepareTree(TreeInfo& info) {
105 info.animationListener = this;
106 RenderNode::prepareTree(info);
107 info.animationListener = NULL;
108
109 // post all the finished stuff
110 if (mFinishedAnimators.size()) {
111 sp<InvokeAnimationListeners> message
112 = new InvokeAnimationListeners(mFinishedAnimators);
113 mLooper->sendMessage(message, 0);
114 }
115 }
116
117private:
118 sp<Looper> mLooper;
119 std::vector< sp<RenderNodeAnimator> > mFinishedAnimators;
120};
121
John Reckcec24ae2013-11-05 13:27:50 -0800122static void android_view_ThreadedRenderer_postToRenderThread(JNIEnv* env, jobject clazz,
123 jobject jrunnable) {
John Reck4f02bf42014-01-03 18:09:17 -0800124 RenderTask* task = new JavaTask(env, jrunnable);
125 RenderThread::getInstance().queue(task);
126}
127
John Recke45b1fd2014-04-15 09:50:16 -0700128static jlong android_view_ThreadedRenderer_createRootRenderNode(JNIEnv* env, jobject clazz) {
129 RootRenderNode* node = new RootRenderNode();
130 node->incStrong(0);
131 node->setName("RootRenderNode");
132 return reinterpret_cast<jlong>(node);
133}
134
John Reck4f02bf42014-01-03 18:09:17 -0800135static jlong android_view_ThreadedRenderer_createProxy(JNIEnv* env, jobject clazz,
John Recke45b1fd2014-04-15 09:50:16 -0700136 jboolean translucent, jlong rootRenderNodePtr) {
137 RenderNode* rootRenderNode = reinterpret_cast<RenderNode*>(rootRenderNodePtr);
138 return (jlong) new RenderProxy(translucent, rootRenderNode);
John Reck4f02bf42014-01-03 18:09:17 -0800139}
140
141static void android_view_ThreadedRenderer_deleteProxy(JNIEnv* env, jobject clazz,
142 jlong proxyPtr) {
John Reck19b6bcf2014-02-14 20:03:38 -0800143 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reck4f02bf42014-01-03 18:09:17 -0800144 delete proxy;
145}
146
147static jboolean android_view_ThreadedRenderer_initialize(JNIEnv* env, jobject clazz,
148 jlong proxyPtr, jobject jsurface) {
John Reck19b6bcf2014-02-14 20:03:38 -0800149 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reck4f02bf42014-01-03 18:09:17 -0800150 sp<ANativeWindow> window = android_view_Surface_getNativeWindow(env, jsurface);
John Reckf7d9c1d2014-04-09 10:01:03 -0700151 return proxy->initialize(window);
John Reck4f02bf42014-01-03 18:09:17 -0800152}
153
154static void android_view_ThreadedRenderer_updateSurface(JNIEnv* env, jobject clazz,
155 jlong proxyPtr, jobject jsurface) {
John Reck19b6bcf2014-02-14 20:03:38 -0800156 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reck4f02bf42014-01-03 18:09:17 -0800157 sp<ANativeWindow> window;
158 if (jsurface) {
159 window = android_view_Surface_getNativeWindow(env, jsurface);
160 }
John Reckf7d9c1d2014-04-09 10:01:03 -0700161 proxy->updateSurface(window);
162}
163
164static void android_view_ThreadedRenderer_pauseSurface(JNIEnv* env, jobject clazz,
165 jlong proxyPtr, jobject jsurface) {
166 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
167 sp<ANativeWindow> window;
168 if (jsurface) {
169 window = android_view_Surface_getNativeWindow(env, jsurface);
170 }
171 proxy->pauseSurface(window);
John Reck4f02bf42014-01-03 18:09:17 -0800172}
173
174static void android_view_ThreadedRenderer_setup(JNIEnv* env, jobject clazz,
175 jlong proxyPtr, jint width, jint height) {
John Reck19b6bcf2014-02-14 20:03:38 -0800176 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reck4f02bf42014-01-03 18:09:17 -0800177 proxy->setup(width, height);
178}
179
John Recke45b1fd2014-04-15 09:50:16 -0700180static void android_view_ThreadedRenderer_syncAndDrawFrame(JNIEnv* env, jobject clazz,
181 jlong proxyPtr, jint dirtyLeft, jint dirtyTop,
John Reck4f02bf42014-01-03 18:09:17 -0800182 jint dirtyRight, jint dirtyBottom) {
John Reck19b6bcf2014-02-14 20:03:38 -0800183 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Recke45b1fd2014-04-15 09:50:16 -0700184 proxy->syncAndDrawFrame(dirtyLeft, dirtyTop, dirtyRight, dirtyBottom);
John Reck4f02bf42014-01-03 18:09:17 -0800185}
186
John Reckfae904d2014-04-14 11:01:57 -0700187static void android_view_ThreadedRenderer_destroyCanvasAndSurface(JNIEnv* env, jobject clazz,
John Reck4f02bf42014-01-03 18:09:17 -0800188 jlong proxyPtr) {
John Reck19b6bcf2014-02-14 20:03:38 -0800189 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reckfae904d2014-04-14 11:01:57 -0700190 proxy->destroyCanvasAndSurface();
John Reck4f02bf42014-01-03 18:09:17 -0800191}
192
John Reck0d1f6342014-03-28 20:30:27 -0700193static void android_view_ThreadedRenderer_invokeFunctor(JNIEnv* env, jobject clazz,
194 jlong proxyPtr, jlong functorPtr, jboolean waitForCompletion) {
195 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
196 Functor* functor = reinterpret_cast<Functor*>(functorPtr);
197 proxy->invokeFunctor(functor, waitForCompletion);
198}
199
John Reckfc53ef272014-02-11 10:40:25 -0800200static void android_view_ThreadedRenderer_runWithGlContext(JNIEnv* env, jobject clazz,
201 jlong proxyPtr, jobject jrunnable) {
John Reck19b6bcf2014-02-14 20:03:38 -0800202 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reckfc53ef272014-02-11 10:40:25 -0800203 RenderTask* task = new JavaTask(env, jrunnable);
204 proxy->runWithGlContext(task);
205}
206
John Reck19b6bcf2014-02-14 20:03:38 -0800207static jlong android_view_ThreadedRenderer_createDisplayListLayer(JNIEnv* env, jobject clazz,
208 jlong proxyPtr, jint width, jint height) {
209 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
210 DeferredLayerUpdater* layer = proxy->createDisplayListLayer(width, height);
211 return reinterpret_cast<jlong>(layer);
212}
213
214static jlong android_view_ThreadedRenderer_createTextureLayer(JNIEnv* env, jobject clazz,
215 jlong proxyPtr) {
216 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
217 DeferredLayerUpdater* layer = proxy->createTextureLayer();
218 return reinterpret_cast<jlong>(layer);
219}
220
221static jboolean android_view_ThreadedRenderer_copyLayerInto(JNIEnv* env, jobject clazz,
222 jlong proxyPtr, jlong layerPtr, jlong bitmapPtr) {
223 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
224 DeferredLayerUpdater* layer = reinterpret_cast<DeferredLayerUpdater*>(layerPtr);
225 SkBitmap* bitmap = reinterpret_cast<SkBitmap*>(bitmapPtr);
226 return proxy->copyLayerInto(layer, bitmap);
227}
228
229static void android_view_ThreadedRenderer_destroyLayer(JNIEnv* env, jobject clazz,
230 jlong proxyPtr, jlong layerPtr) {
231 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
232 DeferredLayerUpdater* layer = reinterpret_cast<DeferredLayerUpdater*>(layerPtr);
233 proxy->destroyLayer(layer);
234}
235
John Reck28ad7b52014-04-07 16:59:25 -0700236static void android_view_ThreadedRenderer_fence(JNIEnv* env, jobject clazz,
237 jlong proxyPtr) {
238 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
239 proxy->fence();
240}
241
John Reckcec24ae2013-11-05 13:27:50 -0800242#endif
243
244// ----------------------------------------------------------------------------
245// JNI Glue
246// ----------------------------------------------------------------------------
247
248const char* const kClassPathName = "android/view/ThreadedRenderer";
249
250static JNINativeMethod gMethods[] = {
251#ifdef USE_OPENGL_RENDERER
252 { "postToRenderThread", "(Ljava/lang/Runnable;)V", (void*) android_view_ThreadedRenderer_postToRenderThread },
John Recke45b1fd2014-04-15 09:50:16 -0700253 { "nCreateRootRenderNode", "()J", (void*) android_view_ThreadedRenderer_createRootRenderNode },
254 { "nCreateProxy", "(ZJ)J", (void*) android_view_ThreadedRenderer_createProxy },
John Reck4f02bf42014-01-03 18:09:17 -0800255 { "nDeleteProxy", "(J)V", (void*) android_view_ThreadedRenderer_deleteProxy },
256 { "nInitialize", "(JLandroid/view/Surface;)Z", (void*) android_view_ThreadedRenderer_initialize },
257 { "nUpdateSurface", "(JLandroid/view/Surface;)V", (void*) android_view_ThreadedRenderer_updateSurface },
John Reckf7d9c1d2014-04-09 10:01:03 -0700258 { "nPauseSurface", "(JLandroid/view/Surface;)V", (void*) android_view_ThreadedRenderer_pauseSurface },
John Reck4f02bf42014-01-03 18:09:17 -0800259 { "nSetup", "(JII)V", (void*) android_view_ThreadedRenderer_setup },
John Recke45b1fd2014-04-15 09:50:16 -0700260 { "nSyncAndDrawFrame", "(JIIII)V", (void*) android_view_ThreadedRenderer_syncAndDrawFrame },
John Reckfae904d2014-04-14 11:01:57 -0700261 { "nDestroyCanvasAndSurface", "(J)V", (void*) android_view_ThreadedRenderer_destroyCanvasAndSurface },
John Reck0d1f6342014-03-28 20:30:27 -0700262 { "nInvokeFunctor", "(JJZ)V", (void*) android_view_ThreadedRenderer_invokeFunctor },
John Reckfc53ef272014-02-11 10:40:25 -0800263 { "nRunWithGlContext", "(JLjava/lang/Runnable;)V", (void*) android_view_ThreadedRenderer_runWithGlContext },
John Reck19b6bcf2014-02-14 20:03:38 -0800264 { "nCreateDisplayListLayer", "(JII)J", (void*) android_view_ThreadedRenderer_createDisplayListLayer },
265 { "nCreateTextureLayer", "(J)J", (void*) android_view_ThreadedRenderer_createTextureLayer },
266 { "nCopyLayerInto", "(JJJ)Z", (void*) android_view_ThreadedRenderer_copyLayerInto },
267 { "nDestroyLayer", "(JJ)V", (void*) android_view_ThreadedRenderer_destroyLayer },
John Reck28ad7b52014-04-07 16:59:25 -0700268 { "nFence", "(J)V", (void*) android_view_ThreadedRenderer_fence },
John Reckcec24ae2013-11-05 13:27:50 -0800269#endif
270};
271
272int register_android_view_ThreadedRenderer(JNIEnv* env) {
273#ifdef USE_OPENGL_RENDERER
274 jclass cls = env->FindClass("java/lang/Runnable");
275 gRunnableMethod = env->GetMethodID(cls, "run", "()V");
276 env->DeleteLocalRef(cls);
277#endif
278 return AndroidRuntime::registerNativeMethods(env, kClassPathName, gMethods, NELEM(gMethods));
279}
280
281}; // namespace android