blob: dd0e45636ee759ed860198c5ce06063bcfc36ba6 [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>
Andreas Gampeed6b9df2014-11-20 22:02:20 -080023#include "core_jni_helpers.h"
John Reck3731dc22015-04-13 15:20:29 -070024#include <GraphicsJNI.h>
John Reckedc524c2015-03-18 15:24:33 -070025#include <ScopedPrimitiveArray.h>
John Reckcec24ae2013-11-05 13:27:50 -080026
John Reck84a4c882014-05-30 14:34:03 -070027#include <EGL/egl.h>
28#include <EGL/eglext.h>
29#include <EGL/egl_cache.h>
Michael Lentine03d8f7682016-01-31 15:37:11 -060030#include <vulkan/vulkan_loader_data.h>
John Reck84a4c882014-05-30 14:34:03 -070031
Andres Morales06f5bc72015-12-15 15:21:31 -080032#include <utils/Looper.h>
33#include <utils/RefBase.h>
John Reck4f02bf42014-01-03 18:09:17 -080034#include <utils/StrongPointer.h>
35#include <android_runtime/android_view_Surface.h>
36#include <system/window.h>
37
John Reck66f0be62014-05-13 13:39:31 -070038#include "android_view_GraphicBuffer.h"
Andres Morales06f5bc72015-12-15 15:21:31 -080039#include "android_os_MessageQueue.h"
John Reck66f0be62014-05-13 13:39:31 -070040
John Reck52244ff2014-05-01 21:27:37 -070041#include <Animator.h>
John Reck119907c2014-08-14 09:02:01 -070042#include <AnimationContext.h>
Andres Morales06f5bc72015-12-15 15:21:31 -080043#include <FrameInfo.h>
Andres Morales910beb82016-02-02 16:19:40 -080044#include <FrameMetricsObserver.h>
John Reck119907c2014-08-14 09:02:01 -070045#include <IContextFactory.h>
John Reckedc524c2015-03-18 15:24:33 -070046#include <JankTracker.h>
John Recke45b1fd2014-04-15 09:50:16 -070047#include <RenderNode.h>
John Reck66f0be62014-05-13 13:39:31 -070048#include <renderthread/CanvasContext.h>
John Reck4f02bf42014-01-03 18:09:17 -080049#include <renderthread/RenderProxy.h>
John Reckcec24ae2013-11-05 13:27:50 -080050#include <renderthread/RenderTask.h>
51#include <renderthread/RenderThread.h>
Chris Craik797b95b2014-05-20 18:10:25 -070052#include <Vector.h>
John Reckcec24ae2013-11-05 13:27:50 -080053
54namespace android {
55
John Reck4f02bf42014-01-03 18:09:17 -080056using namespace android::uirenderer;
57using namespace android::uirenderer::renderthread;
John Reckcec24ae2013-11-05 13:27:50 -080058
Andres Morales06f5bc72015-12-15 15:21:31 -080059struct {
Andres Morales910beb82016-02-02 16:19:40 -080060 jfieldID frameMetrics;
61 jfieldID timingDataBuffer;
Andres Morales06f5bc72015-12-15 15:21:31 -080062 jfieldID messageQueue;
Andres Morales910beb82016-02-02 16:19:40 -080063 jmethodID callback;
64} gFrameMetricsObserverClassInfo;
Andres Morales06f5bc72015-12-15 15:21:31 -080065
John Reckc25e5062014-06-18 14:21:29 -070066static JNIEnv* getenv(JavaVM* vm) {
67 JNIEnv* env;
68 if (vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6) != JNI_OK) {
69 LOG_ALWAYS_FATAL("Failed to get JNIEnv for JavaVM: %p", vm);
70 }
71 return env;
72}
73
John Reck52244ff2014-05-01 21:27:37 -070074class OnFinishedEvent {
75public:
John Reckff941dc2014-05-14 16:34:14 -070076 OnFinishedEvent(BaseRenderNodeAnimator* animator, AnimationListener* listener)
John Reck52244ff2014-05-01 21:27:37 -070077 : animator(animator), listener(listener) {}
John Reckff941dc2014-05-14 16:34:14 -070078 sp<BaseRenderNodeAnimator> animator;
John Reck52244ff2014-05-01 21:27:37 -070079 sp<AnimationListener> listener;
80};
81
John Recke45b1fd2014-04-15 09:50:16 -070082class InvokeAnimationListeners : public MessageHandler {
83public:
John Reck52244ff2014-05-01 21:27:37 -070084 InvokeAnimationListeners(std::vector<OnFinishedEvent>& events) {
85 mOnFinishedEvents.swap(events);
John Recke45b1fd2014-04-15 09:50:16 -070086 }
87
John Reck52244ff2014-05-01 21:27:37 -070088 static void callOnFinished(OnFinishedEvent& event) {
89 event.listener->onAnimationFinished(event.animator.get());
John Recke45b1fd2014-04-15 09:50:16 -070090 }
91
92 virtual void handleMessage(const Message& message) {
John Reck52244ff2014-05-01 21:27:37 -070093 std::for_each(mOnFinishedEvents.begin(), mOnFinishedEvents.end(), callOnFinished);
94 mOnFinishedEvents.clear();
John Recke45b1fd2014-04-15 09:50:16 -070095 }
96
97private:
John Reck52244ff2014-05-01 21:27:37 -070098 std::vector<OnFinishedEvent> mOnFinishedEvents;
John Recke45b1fd2014-04-15 09:50:16 -070099};
100
John Reckc25e5062014-06-18 14:21:29 -0700101class RenderingException : public MessageHandler {
John Recke45b1fd2014-04-15 09:50:16 -0700102public:
John Reckc25e5062014-06-18 14:21:29 -0700103 RenderingException(JavaVM* vm, const std::string& message)
104 : mVm(vm)
105 , mMessage(message) {
106 }
107
108 virtual void handleMessage(const Message&) {
109 throwException(mVm, mMessage);
110 }
111
112 static void throwException(JavaVM* vm, const std::string& message) {
113 JNIEnv* env = getenv(vm);
114 jniThrowException(env, "java/lang/IllegalStateException", message.c_str());
115 }
116
117private:
118 JavaVM* mVm;
119 std::string mMessage;
120};
121
John Reck119907c2014-08-14 09:02:01 -0700122class RootRenderNode : public RenderNode, ErrorHandler {
John Reckc25e5062014-06-18 14:21:29 -0700123public:
124 RootRenderNode(JNIEnv* env) : RenderNode() {
John Recke45b1fd2014-04-15 09:50:16 -0700125 mLooper = Looper::getForThread();
126 LOG_ALWAYS_FATAL_IF(!mLooper.get(),
127 "Must create RootRenderNode on a thread with a looper!");
John Reckc25e5062014-06-18 14:21:29 -0700128 env->GetJavaVM(&mVm);
John Recke45b1fd2014-04-15 09:50:16 -0700129 }
130
131 virtual ~RootRenderNode() {}
132
John Reckc25e5062014-06-18 14:21:29 -0700133 virtual void onError(const std::string& message) {
134 mLooper->sendMessage(new RenderingException(mVm, message), 0);
135 }
136
John Recke45b1fd2014-04-15 09:50:16 -0700137 virtual void prepareTree(TreeInfo& info) {
John Reckc25e5062014-06-18 14:21:29 -0700138 info.errorHandler = this;
John Reckf6481082016-02-02 15:18:23 -0800139 // TODO: This is hacky
140 info.windowInsetLeft = -stagingProperties().getLeft();
141 info.windowInsetTop = -stagingProperties().getTop();
142 info.updateWindowPositions = true;
John Recke45b1fd2014-04-15 09:50:16 -0700143 RenderNode::prepareTree(info);
John Reckf6481082016-02-02 15:18:23 -0800144 info.updateWindowPositions = false;
145 info.windowInsetLeft = 0;
146 info.windowInsetTop = 0;
John Reckc25e5062014-06-18 14:21:29 -0700147 info.errorHandler = NULL;
John Reck119907c2014-08-14 09:02:01 -0700148 }
John Recke45b1fd2014-04-15 09:50:16 -0700149
John Reck119907c2014-08-14 09:02:01 -0700150 void sendMessage(const sp<MessageHandler>& handler) {
151 mLooper->sendMessage(handler, 0);
152 }
153
154 void attachAnimatingNode(RenderNode* animatingNode) {
155 mPendingAnimatingRenderNodes.push_back(animatingNode);
156 }
157
158 void doAttachAnimatingNodes(AnimationContext* context) {
159 for (size_t i = 0; i < mPendingAnimatingRenderNodes.size(); i++) {
160 RenderNode* node = mPendingAnimatingRenderNodes[i].get();
161 context->addAnimatingRenderNode(*node);
John Recke45b1fd2014-04-15 09:50:16 -0700162 }
John Reck119907c2014-08-14 09:02:01 -0700163 mPendingAnimatingRenderNodes.clear();
John Recke45b1fd2014-04-15 09:50:16 -0700164 }
165
166private:
167 sp<Looper> mLooper;
John Reckc25e5062014-06-18 14:21:29 -0700168 JavaVM* mVm;
John Reck119907c2014-08-14 09:02:01 -0700169 std::vector< sp<RenderNode> > mPendingAnimatingRenderNodes;
170};
171
172class AnimationContextBridge : public AnimationContext {
173public:
174 AnimationContextBridge(renderthread::TimeLord& clock, RootRenderNode* rootNode)
175 : AnimationContext(clock), mRootNode(rootNode) {
176 }
177
178 virtual ~AnimationContextBridge() {}
179
180 // Marks the start of a frame, which will update the frame time and move all
181 // next frame animations into the current frame
John Reckec845a22014-09-05 15:23:38 -0700182 virtual void startFrame(TreeInfo::TraversalMode mode) {
183 if (mode == TreeInfo::MODE_FULL) {
184 mRootNode->doAttachAnimatingNodes(this);
185 }
186 AnimationContext::startFrame(mode);
John Reck119907c2014-08-14 09:02:01 -0700187 }
188
189 // Runs any animations still left in mCurrentFrameAnimations
190 virtual void runRemainingAnimations(TreeInfo& info) {
191 AnimationContext::runRemainingAnimations(info);
John Recke2478d42014-09-03 16:46:05 -0700192 postOnFinishedEvents();
John Reck119907c2014-08-14 09:02:01 -0700193 }
194
195 virtual void callOnFinished(BaseRenderNodeAnimator* animator, AnimationListener* listener) {
196 OnFinishedEvent event(animator, listener);
197 mOnFinishedEvents.push_back(event);
198 }
199
John Recke2478d42014-09-03 16:46:05 -0700200 virtual void destroy() {
201 AnimationContext::destroy();
202 postOnFinishedEvents();
203 }
204
John Reck119907c2014-08-14 09:02:01 -0700205private:
206 sp<RootRenderNode> mRootNode;
207 std::vector<OnFinishedEvent> mOnFinishedEvents;
John Recke2478d42014-09-03 16:46:05 -0700208
209 void postOnFinishedEvents() {
210 if (mOnFinishedEvents.size()) {
211 sp<InvokeAnimationListeners> message
212 = new InvokeAnimationListeners(mOnFinishedEvents);
213 mRootNode->sendMessage(message);
214 }
215 }
John Reck119907c2014-08-14 09:02:01 -0700216};
217
218class ContextFactoryImpl : public IContextFactory {
219public:
220 ContextFactoryImpl(RootRenderNode* rootNode) : mRootNode(rootNode) {}
221
222 virtual AnimationContext* createAnimationContext(renderthread::TimeLord& clock) {
223 return new AnimationContextBridge(clock, mRootNode);
224 }
225
226private:
227 RootRenderNode* mRootNode;
John Recke45b1fd2014-04-15 09:50:16 -0700228};
229
Andres Morales06f5bc72015-12-15 15:21:31 -0800230class ObserverProxy;
231
232class NotifyHandler : public MessageHandler {
233public:
234 NotifyHandler(JavaVM* vm) : mVm(vm) {}
235
236 void setObserver(ObserverProxy* observer) {
237 mObserver = observer;
238 }
239
240 void setBuffer(BufferPool::Buffer* buffer) {
241 mBuffer = buffer;
242 }
243
Andres Morales910beb82016-02-02 16:19:40 -0800244 void setDropCount(int dropCount) {
245 mDropCount = dropCount;
246 }
247
Andres Morales06f5bc72015-12-15 15:21:31 -0800248 virtual void handleMessage(const Message& message);
249
250private:
251 JavaVM* mVm;
252
253 sp<ObserverProxy> mObserver;
Andres Morales910beb82016-02-02 16:19:40 -0800254 BufferPool::Buffer* mBuffer = nullptr;
255 int mDropCount = 0;
Andres Morales06f5bc72015-12-15 15:21:31 -0800256};
257
Andres Morales910beb82016-02-02 16:19:40 -0800258static jlongArray get_metrics_buffer(JNIEnv* env, jobject observer) {
259 jobject frameMetrics = env->GetObjectField(
260 observer, gFrameMetricsObserverClassInfo.frameMetrics);
261 LOG_ALWAYS_FATAL_IF(frameMetrics == nullptr, "unable to retrieve data sink object");
262 jobject buffer = env->GetObjectField(
263 frameMetrics, gFrameMetricsObserverClassInfo.timingDataBuffer);
264 LOG_ALWAYS_FATAL_IF(buffer == nullptr, "unable to retrieve data sink buffer");
265 return reinterpret_cast<jlongArray>(buffer);
266}
267
268class ObserverProxy : public FrameMetricsObserver {
Andres Morales06f5bc72015-12-15 15:21:31 -0800269public:
Andres Morales910beb82016-02-02 16:19:40 -0800270 ObserverProxy(JavaVM *vm, jobject observer) : mVm(vm) {
Andres Morales06f5bc72015-12-15 15:21:31 -0800271 JNIEnv* env = getenv(mVm);
272
Andres Morales910beb82016-02-02 16:19:40 -0800273 mObserverWeak = env->NewWeakGlobalRef(observer);
274 LOG_ALWAYS_FATAL_IF(mObserverWeak == nullptr,
Andres Morales06f5bc72015-12-15 15:21:31 -0800275 "unable to create frame stats observer reference");
276
Andres Morales910beb82016-02-02 16:19:40 -0800277 jlongArray buffer = get_metrics_buffer(env, observer);
278 jsize bufferSize = env->GetArrayLength(reinterpret_cast<jarray>(buffer));
279 LOG_ALWAYS_FATAL_IF(bufferSize != kBufferSize,
280 "Mismatched Java/Native FrameMetrics data format.");
281
282 jobject messageQueueLocal = env->GetObjectField(
283 observer, gFrameMetricsObserverClassInfo.messageQueue);
Andres Morales06f5bc72015-12-15 15:21:31 -0800284 mMessageQueue = android_os_MessageQueue_getMessageQueue(env, messageQueueLocal);
285 LOG_ALWAYS_FATAL_IF(mMessageQueue == nullptr, "message queue not available");
286
287 mMessageHandler = new NotifyHandler(mVm);
288 LOG_ALWAYS_FATAL_IF(mMessageHandler == nullptr,
289 "OOM: unable to allocate NotifyHandler");
290 }
291
292 ~ObserverProxy() {
293 JNIEnv* env = getenv(mVm);
Andres Morales910beb82016-02-02 16:19:40 -0800294 env->DeleteWeakGlobalRef(mObserverWeak);
Andres Morales06f5bc72015-12-15 15:21:31 -0800295 }
296
Andres Morales910beb82016-02-02 16:19:40 -0800297 jweak getObserverReference() {
298 return mObserverWeak;
Andres Morales06f5bc72015-12-15 15:21:31 -0800299 }
300
Andres Morales910beb82016-02-02 16:19:40 -0800301 virtual void notify(BufferPool::Buffer* buffer, int dropCount) {
Andres Morales06f5bc72015-12-15 15:21:31 -0800302 buffer->incRef();
303 mMessageHandler->setBuffer(buffer);
304 mMessageHandler->setObserver(this);
Andres Morales910beb82016-02-02 16:19:40 -0800305 mMessageHandler->setDropCount(dropCount);
Andres Morales06f5bc72015-12-15 15:21:31 -0800306 mMessageQueue->getLooper()->sendMessage(mMessageHandler, mMessage);
307 }
308
309private:
310 static const int kBufferSize = static_cast<int>(FrameInfoIndex::NumIndexes);
311
312 JavaVM* mVm;
Andres Morales910beb82016-02-02 16:19:40 -0800313 jweak mObserverWeak;
314 jobject mJavaBufferGlobal;
Andres Morales06f5bc72015-12-15 15:21:31 -0800315
316 sp<MessageQueue> mMessageQueue;
317 sp<NotifyHandler> mMessageHandler;
318 Message mMessage;
Andres Morales910beb82016-02-02 16:19:40 -0800319
Andres Morales06f5bc72015-12-15 15:21:31 -0800320};
321
322void NotifyHandler::handleMessage(const Message& message) {
323 JNIEnv* env = getenv(mVm);
324
Andres Morales910beb82016-02-02 16:19:40 -0800325 jobject target = env->NewLocalRef(mObserver->getObserverReference());
Andres Morales06f5bc72015-12-15 15:21:31 -0800326
327 if (target != nullptr) {
Andres Morales910beb82016-02-02 16:19:40 -0800328 jlongArray javaBuffer = get_metrics_buffer(env, target);
329 env->SetLongArrayRegion(javaBuffer,
330 0, mBuffer->getSize(), mBuffer->getBuffer());
331 env->CallVoidMethod(target, gFrameMetricsObserverClassInfo.callback,
332 mDropCount);
333 env->DeleteLocalRef(target);
Andres Morales06f5bc72015-12-15 15:21:31 -0800334 }
335
336 mBuffer->release();
337 mObserver.clear();
338}
339
John Reck66f0be62014-05-13 13:39:31 -0700340static void android_view_ThreadedRenderer_setAtlas(JNIEnv* env, jobject clazz,
John Reck3b202512014-06-23 13:13:08 -0700341 jlong proxyPtr, jobject graphicBuffer, jlongArray atlasMapArray) {
John Reck66f0be62014-05-13 13:39:31 -0700342 sp<GraphicBuffer> buffer = graphicBufferForJavaObject(env, graphicBuffer);
343 jsize len = env->GetArrayLength(atlasMapArray);
344 if (len <= 0) {
345 ALOGW("Failed to initialize atlas, invalid map length: %d", len);
346 return;
347 }
348 int64_t* map = new int64_t[len];
349 env->GetLongArrayRegion(atlasMapArray, 0, len, map);
350
John Reck3b202512014-06-23 13:13:08 -0700351 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
352 proxy->setTextureAtlas(buffer, map, len);
John Reck4f02bf42014-01-03 18:09:17 -0800353}
354
John Reckedc524c2015-03-18 15:24:33 -0700355static void android_view_ThreadedRenderer_setProcessStatsBuffer(JNIEnv* env, jobject clazz,
356 jlong proxyPtr, jint fd) {
357 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
358 proxy->setProcessStatsBuffer(fd);
359}
360
John Recke45b1fd2014-04-15 09:50:16 -0700361static jlong android_view_ThreadedRenderer_createRootRenderNode(JNIEnv* env, jobject clazz) {
John Reckc25e5062014-06-18 14:21:29 -0700362 RootRenderNode* node = new RootRenderNode(env);
John Recke45b1fd2014-04-15 09:50:16 -0700363 node->incStrong(0);
364 node->setName("RootRenderNode");
365 return reinterpret_cast<jlong>(node);
366}
367
John Reck4f02bf42014-01-03 18:09:17 -0800368static jlong android_view_ThreadedRenderer_createProxy(JNIEnv* env, jobject clazz,
John Recke45b1fd2014-04-15 09:50:16 -0700369 jboolean translucent, jlong rootRenderNodePtr) {
John Reck119907c2014-08-14 09:02:01 -0700370 RootRenderNode* rootRenderNode = reinterpret_cast<RootRenderNode*>(rootRenderNodePtr);
371 ContextFactoryImpl factory(rootRenderNode);
372 return (jlong) new RenderProxy(translucent, rootRenderNode, &factory);
John Reck4f02bf42014-01-03 18:09:17 -0800373}
374
375static void android_view_ThreadedRenderer_deleteProxy(JNIEnv* env, jobject clazz,
376 jlong proxyPtr) {
John Reck19b6bcf2014-02-14 20:03:38 -0800377 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reck4f02bf42014-01-03 18:09:17 -0800378 delete proxy;
379}
380
John Recke4280ba2014-05-05 16:39:37 -0700381static jboolean android_view_ThreadedRenderer_loadSystemProperties(JNIEnv* env, jobject clazz,
382 jlong proxyPtr) {
383 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
384 return proxy->loadSystemProperties();
385}
386
John Reckb36016c2015-03-11 08:50:53 -0700387static void android_view_ThreadedRenderer_setName(JNIEnv* env, jobject clazz,
388 jlong proxyPtr, jstring jname) {
389 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
390 const char* name = env->GetStringUTFChars(jname, NULL);
391 proxy->setName(name);
392 env->ReleaseStringUTFChars(jname, name);
393}
394
Thomas Buhot0bcd0cb2015-12-04 12:18:03 +0100395static void android_view_ThreadedRenderer_initialize(JNIEnv* env, jobject clazz,
John Reck4f02bf42014-01-03 18:09:17 -0800396 jlong proxyPtr, jobject jsurface) {
John Reck19b6bcf2014-02-14 20:03:38 -0800397 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reckf6481082016-02-02 15:18:23 -0800398 sp<Surface> surface = android_view_Surface_getSurface(env, jsurface);
399 proxy->initialize(surface);
John Reck4f02bf42014-01-03 18:09:17 -0800400}
401
402static void android_view_ThreadedRenderer_updateSurface(JNIEnv* env, jobject clazz,
403 jlong proxyPtr, jobject jsurface) {
John Reck19b6bcf2014-02-14 20:03:38 -0800404 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reckf6481082016-02-02 15:18:23 -0800405 sp<Surface> surface;
John Reck4f02bf42014-01-03 18:09:17 -0800406 if (jsurface) {
John Reckf6481082016-02-02 15:18:23 -0800407 surface = android_view_Surface_getSurface(env, jsurface);
John Reck4f02bf42014-01-03 18:09:17 -0800408 }
John Reckf6481082016-02-02 15:18:23 -0800409 proxy->updateSurface(surface);
John Reckf7d9c1d2014-04-09 10:01:03 -0700410}
411
John Reck01a5ea32014-12-03 13:01:07 -0800412static jboolean android_view_ThreadedRenderer_pauseSurface(JNIEnv* env, jobject clazz,
John Reckf7d9c1d2014-04-09 10:01:03 -0700413 jlong proxyPtr, jobject jsurface) {
414 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reckf6481082016-02-02 15:18:23 -0800415 sp<Surface> surface;
John Reckf7d9c1d2014-04-09 10:01:03 -0700416 if (jsurface) {
John Reckf6481082016-02-02 15:18:23 -0800417 surface = android_view_Surface_getSurface(env, jsurface);
John Reckf7d9c1d2014-04-09 10:01:03 -0700418 }
John Reckf6481082016-02-02 15:18:23 -0800419 return proxy->pauseSurface(surface);
John Reck4f02bf42014-01-03 18:09:17 -0800420}
421
Chris Craik797b95b2014-05-20 18:10:25 -0700422static void android_view_ThreadedRenderer_setup(JNIEnv* env, jobject clazz, jlong proxyPtr,
Alan Viverette50210d92015-05-14 18:05:36 -0700423 jint width, jint height, jfloat lightRadius, jint ambientShadowAlpha, jint spotShadowAlpha) {
John Reck19b6bcf2014-02-14 20:03:38 -0800424 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
Alan Viverette50210d92015-05-14 18:05:36 -0700425 proxy->setup(width, height, lightRadius, ambientShadowAlpha, spotShadowAlpha);
426}
427
428static void android_view_ThreadedRenderer_setLightCenter(JNIEnv* env, jobject clazz,
429 jlong proxyPtr, jfloat lightX, jfloat lightY, jfloat lightZ) {
430 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
431 proxy->setLightCenter((Vector3){lightX, lightY, lightZ});
John Reck4f02bf42014-01-03 18:09:17 -0800432}
433
John Reck63a06672014-05-07 13:45:54 -0700434static void android_view_ThreadedRenderer_setOpaque(JNIEnv* env, jobject clazz,
435 jlong proxyPtr, jboolean opaque) {
436 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
437 proxy->setOpaque(opaque);
438}
439
John Reckf9be7792014-05-02 18:21:16 -0700440static int android_view_ThreadedRenderer_syncAndDrawFrame(JNIEnv* env, jobject clazz,
John Reckba6adf62015-02-19 14:36:50 -0800441 jlong proxyPtr, jlongArray frameInfo, jint frameInfoSize) {
442 LOG_ALWAYS_FATAL_IF(frameInfoSize != UI_THREAD_FRAME_INFO_SIZE,
443 "Mismatched size expectations, given %d expected %d",
444 frameInfoSize, UI_THREAD_FRAME_INFO_SIZE);
John Reck19b6bcf2014-02-14 20:03:38 -0800445 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reckba6adf62015-02-19 14:36:50 -0800446 env->GetLongArrayRegion(frameInfo, 0, frameInfoSize, proxy->frameInfo());
447 return proxy->syncAndDrawFrame();
John Reck4f02bf42014-01-03 18:09:17 -0800448}
449
John Reck17035b02014-09-03 07:39:53 -0700450static void android_view_ThreadedRenderer_destroy(JNIEnv* env, jobject clazz,
John Reck4f02bf42014-01-03 18:09:17 -0800451 jlong proxyPtr) {
John Reck19b6bcf2014-02-14 20:03:38 -0800452 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reck17035b02014-09-03 07:39:53 -0700453 proxy->destroy();
John Reck4f02bf42014-01-03 18:09:17 -0800454}
455
John Reck119907c2014-08-14 09:02:01 -0700456static void android_view_ThreadedRenderer_registerAnimatingRenderNode(JNIEnv* env, jobject clazz,
457 jlong rootNodePtr, jlong animatingNodePtr) {
458 RootRenderNode* rootRenderNode = reinterpret_cast<RootRenderNode*>(rootNodePtr);
459 RenderNode* animatingNode = reinterpret_cast<RenderNode*>(animatingNodePtr);
460 rootRenderNode->attachAnimatingNode(animatingNode);
461}
462
John Reck0d1f6342014-03-28 20:30:27 -0700463static void android_view_ThreadedRenderer_invokeFunctor(JNIEnv* env, jobject clazz,
John Reck3b202512014-06-23 13:13:08 -0700464 jlong functorPtr, jboolean waitForCompletion) {
John Reck0d1f6342014-03-28 20:30:27 -0700465 Functor* functor = reinterpret_cast<Functor*>(functorPtr);
John Reck3b202512014-06-23 13:13:08 -0700466 RenderProxy::invokeFunctor(functor, waitForCompletion);
John Reck0d1f6342014-03-28 20:30:27 -0700467}
468
John Reck19b6bcf2014-02-14 20:03:38 -0800469static jlong android_view_ThreadedRenderer_createTextureLayer(JNIEnv* env, jobject clazz,
470 jlong proxyPtr) {
471 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
472 DeferredLayerUpdater* layer = proxy->createTextureLayer();
473 return reinterpret_cast<jlong>(layer);
474}
475
John Reck3e824952014-08-20 10:08:39 -0700476static void android_view_ThreadedRenderer_buildLayer(JNIEnv* env, jobject clazz,
477 jlong proxyPtr, jlong nodePtr) {
478 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
479 RenderNode* node = reinterpret_cast<RenderNode*>(nodePtr);
480 proxy->buildLayer(node);
481}
482
John Reck19b6bcf2014-02-14 20:03:38 -0800483static jboolean android_view_ThreadedRenderer_copyLayerInto(JNIEnv* env, jobject clazz,
John Reck3731dc22015-04-13 15:20:29 -0700484 jlong proxyPtr, jlong layerPtr, jobject jbitmap) {
John Reck19b6bcf2014-02-14 20:03:38 -0800485 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
486 DeferredLayerUpdater* layer = reinterpret_cast<DeferredLayerUpdater*>(layerPtr);
John Reck3731dc22015-04-13 15:20:29 -0700487 SkBitmap bitmap;
488 GraphicsJNI::getSkBitmap(env, jbitmap, &bitmap);
John Reck19b6bcf2014-02-14 20:03:38 -0800489 return proxy->copyLayerInto(layer, bitmap);
490}
491
John Reckd72e0a32014-05-29 18:56:11 -0700492static void android_view_ThreadedRenderer_pushLayerUpdate(JNIEnv* env, jobject clazz,
John Reck19b6bcf2014-02-14 20:03:38 -0800493 jlong proxyPtr, jlong layerPtr) {
494 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
495 DeferredLayerUpdater* layer = reinterpret_cast<DeferredLayerUpdater*>(layerPtr);
John Reckd72e0a32014-05-29 18:56:11 -0700496 proxy->pushLayerUpdate(layer);
497}
498
499static void android_view_ThreadedRenderer_cancelLayerUpdate(JNIEnv* env, jobject clazz,
500 jlong proxyPtr, jlong layerPtr) {
501 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
502 DeferredLayerUpdater* layer = reinterpret_cast<DeferredLayerUpdater*>(layerPtr);
503 proxy->cancelLayerUpdate(layer);
John Reck19b6bcf2014-02-14 20:03:38 -0800504}
505
John Reck918ad522014-06-27 14:45:25 -0700506static void android_view_ThreadedRenderer_detachSurfaceTexture(JNIEnv* env, jobject clazz,
507 jlong proxyPtr, jlong layerPtr) {
508 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
509 DeferredLayerUpdater* layer = reinterpret_cast<DeferredLayerUpdater*>(layerPtr);
510 proxy->detachSurfaceTexture(layer);
511}
512
John Reckf47a5942014-06-30 16:20:04 -0700513static void android_view_ThreadedRenderer_destroyHardwareResources(JNIEnv* env, jobject clazz,
514 jlong proxyPtr) {
John Recke1628b72014-05-23 15:11:19 -0700515 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
John Reckf47a5942014-06-30 16:20:04 -0700516 proxy->destroyHardwareResources();
517}
518
519static void android_view_ThreadedRenderer_trimMemory(JNIEnv* env, jobject clazz,
520 jint level) {
521 RenderProxy::trimMemory(level);
John Recke1628b72014-05-23 15:11:19 -0700522}
523
Chris Craik2507c342015-05-04 14:36:49 -0700524static void android_view_ThreadedRenderer_overrideProperty(JNIEnv* env, jobject clazz,
525 jstring name, jstring value) {
526 const char* nameCharArray = env->GetStringUTFChars(name, NULL);
527 const char* valueCharArray = env->GetStringUTFChars(value, NULL);
528 RenderProxy::overrideProperty(nameCharArray, valueCharArray);
529 env->ReleaseStringUTFChars(name, nameCharArray);
530 env->ReleaseStringUTFChars(name, valueCharArray);
531}
532
John Reck28ad7b52014-04-07 16:59:25 -0700533static void android_view_ThreadedRenderer_fence(JNIEnv* env, jobject clazz,
534 jlong proxyPtr) {
535 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
536 proxy->fence();
537}
538
John Reckf47a5942014-06-30 16:20:04 -0700539static void android_view_ThreadedRenderer_stopDrawing(JNIEnv* env, jobject clazz,
540 jlong proxyPtr) {
541 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
542 proxy->stopDrawing();
543}
544
John Recka5dda642014-05-22 15:43:54 -0700545static void android_view_ThreadedRenderer_notifyFramePending(JNIEnv* env, jobject clazz,
546 jlong proxyPtr) {
547 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
548 proxy->notifyFramePending();
549}
550
John Recke248bd12015-08-05 13:53:53 -0700551static void android_view_ThreadedRenderer_serializeDisplayListTree(JNIEnv* env, jobject clazz,
552 jlong proxyPtr) {
553 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
554 proxy->serializeDisplayListTree();
555}
556
John Reckfe5e7b72014-05-23 17:42:28 -0700557static void android_view_ThreadedRenderer_dumpProfileInfo(JNIEnv* env, jobject clazz,
John Reckba6adf62015-02-19 14:36:50 -0800558 jlong proxyPtr, jobject javaFileDescriptor, jint dumpFlags) {
John Reckfe5e7b72014-05-23 17:42:28 -0700559 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
560 int fd = jniGetFDFromFileDescriptor(env, javaFileDescriptor);
John Reckba6adf62015-02-19 14:36:50 -0800561 proxy->dumpProfileInfo(fd, dumpFlags);
John Reckfe5e7b72014-05-23 17:42:28 -0700562}
563
John Reckedc524c2015-03-18 15:24:33 -0700564static void android_view_ThreadedRenderer_dumpProfileData(JNIEnv* env, jobject clazz,
565 jbyteArray jdata, jobject javaFileDescriptor) {
566 int fd = jniGetFDFromFileDescriptor(env, javaFileDescriptor);
567 ScopedByteArrayRO buffer(env, jdata);
568 if (buffer.get()) {
569 JankTracker::dumpBuffer(buffer.get(), buffer.size(), fd);
570 }
571}
572
Skuhneea7a7fb2015-08-28 07:10:31 -0700573static void android_view_ThreadedRenderer_addRenderNode(JNIEnv* env, jobject clazz,
574 jlong proxyPtr, jlong renderNodePtr, jboolean placeFront) {
575 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
576 RenderNode* renderNode = reinterpret_cast<RenderNode*>(renderNodePtr);
577 proxy->addRenderNode(renderNode, placeFront);
578}
579
580static void android_view_ThreadedRenderer_removeRenderNode(JNIEnv* env, jobject clazz,
581 jlong proxyPtr, jlong renderNodePtr) {
582 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
583 RenderNode* renderNode = reinterpret_cast<RenderNode*>(renderNodePtr);
584 proxy->removeRenderNode(renderNode);
585}
586
587static void android_view_ThreadedRendererd_drawRenderNode(JNIEnv* env, jobject clazz,
588 jlong proxyPtr, jlong renderNodePtr) {
589 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
590 RenderNode* renderNode = reinterpret_cast<RenderNode*>(renderNodePtr);
591 proxy->drawRenderNode(renderNode);
592}
593
Skuhneb8160872015-09-22 09:51:39 -0700594static void android_view_ThreadedRenderer_setContentDrawBounds(JNIEnv* env,
Skuhneea7a7fb2015-08-28 07:10:31 -0700595 jobject clazz, jlong proxyPtr, jint left, jint top, jint right, jint bottom) {
596 RenderProxy* proxy = reinterpret_cast<RenderProxy*>(proxyPtr);
Skuhneb8160872015-09-22 09:51:39 -0700597 proxy->setContentDrawBounds(left, top, right, bottom);
Skuhneea7a7fb2015-08-28 07:10:31 -0700598}
John Reckedc524c2015-03-18 15:24:33 -0700599
John Reckcec24ae2013-11-05 13:27:50 -0800600// ----------------------------------------------------------------------------
Andres Morales910beb82016-02-02 16:19:40 -0800601// FrameMetricsObserver
Andres Morales06f5bc72015-12-15 15:21:31 -0800602// ----------------------------------------------------------------------------
603
Andres Morales910beb82016-02-02 16:19:40 -0800604static jlong android_view_ThreadedRenderer_addFrameMetricsObserver(JNIEnv* env,
Andres Morales06f5bc72015-12-15 15:21:31 -0800605 jclass clazz, jlong proxyPtr, jobject fso) {
606 JavaVM* vm = nullptr;
607 if (env->GetJavaVM(&vm) != JNI_OK) {
608 LOG_ALWAYS_FATAL("Unable to get Java VM");
609 return 0;
610 }
611
612 renderthread::RenderProxy* renderProxy =
613 reinterpret_cast<renderthread::RenderProxy*>(proxyPtr);
614
Andres Morales910beb82016-02-02 16:19:40 -0800615 FrameMetricsObserver* observer = new ObserverProxy(vm, fso);
616 renderProxy->addFrameMetricsObserver(observer);
Andres Morales06f5bc72015-12-15 15:21:31 -0800617 return reinterpret_cast<jlong>(observer);
618}
619
Andres Morales910beb82016-02-02 16:19:40 -0800620static void android_view_ThreadedRenderer_removeFrameMetricsObserver(JNIEnv* env, jclass clazz,
Andres Morales06f5bc72015-12-15 15:21:31 -0800621 jlong proxyPtr, jlong observerPtr) {
Andres Morales910beb82016-02-02 16:19:40 -0800622 FrameMetricsObserver* observer = reinterpret_cast<FrameMetricsObserver*>(observerPtr);
Andres Morales06f5bc72015-12-15 15:21:31 -0800623 renderthread::RenderProxy* renderProxy =
624 reinterpret_cast<renderthread::RenderProxy*>(proxyPtr);
625
Andres Morales910beb82016-02-02 16:19:40 -0800626 renderProxy->removeFrameMetricsObserver(observer);
Andres Morales06f5bc72015-12-15 15:21:31 -0800627}
628
629// ----------------------------------------------------------------------------
John Reck84a4c882014-05-30 14:34:03 -0700630// Shaders
631// ----------------------------------------------------------------------------
632
633static void android_view_ThreadedRenderer_setupShadersDiskCache(JNIEnv* env, jobject clazz,
634 jstring diskCachePath) {
John Reck84a4c882014-05-30 14:34:03 -0700635 const char* cacheArray = env->GetStringUTFChars(diskCachePath, NULL);
636 egl_cache_t::get()->setCacheFilename(cacheArray);
637 env->ReleaseStringUTFChars(diskCachePath, cacheArray);
638}
639
640// ----------------------------------------------------------------------------
Michael Lentine03d8f7682016-01-31 15:37:11 -0600641// Layers
642// ----------------------------------------------------------------------------
643
644static void android_view_ThreadedRenderer_setupVulkanLayerPath(JNIEnv* env, jobject clazz,
645 jstring layerPath) {
646
647 const char* layerArray = env->GetStringUTFChars(layerPath, NULL);
648 vulkan::LoaderData::GetInstance().layer_path = layerArray;
649 env->ReleaseStringUTFChars(layerPath, layerArray);
650}
651
652// ----------------------------------------------------------------------------
John Reckcec24ae2013-11-05 13:27:50 -0800653// JNI Glue
654// ----------------------------------------------------------------------------
655
656const char* const kClassPathName = "android/view/ThreadedRenderer";
657
Daniel Micay76f6a862015-09-19 17:31:01 -0400658static const JNINativeMethod gMethods[] = {
John Reck3b202512014-06-23 13:13:08 -0700659 { "nSetAtlas", "(JLandroid/view/GraphicBuffer;[J)V", (void*) android_view_ThreadedRenderer_setAtlas },
John Reckedc524c2015-03-18 15:24:33 -0700660 { "nSetProcessStatsBuffer", "(JI)V", (void*) android_view_ThreadedRenderer_setProcessStatsBuffer },
John Recke45b1fd2014-04-15 09:50:16 -0700661 { "nCreateRootRenderNode", "()J", (void*) android_view_ThreadedRenderer_createRootRenderNode },
662 { "nCreateProxy", "(ZJ)J", (void*) android_view_ThreadedRenderer_createProxy },
John Reck4f02bf42014-01-03 18:09:17 -0800663 { "nDeleteProxy", "(J)V", (void*) android_view_ThreadedRenderer_deleteProxy },
John Recke4280ba2014-05-05 16:39:37 -0700664 { "nLoadSystemProperties", "(J)Z", (void*) android_view_ThreadedRenderer_loadSystemProperties },
John Reckb36016c2015-03-11 08:50:53 -0700665 { "nSetName", "(JLjava/lang/String;)V", (void*) android_view_ThreadedRenderer_setName },
Thomas Buhot0bcd0cb2015-12-04 12:18:03 +0100666 { "nInitialize", "(JLandroid/view/Surface;)V", (void*) android_view_ThreadedRenderer_initialize },
John Reck4f02bf42014-01-03 18:09:17 -0800667 { "nUpdateSurface", "(JLandroid/view/Surface;)V", (void*) android_view_ThreadedRenderer_updateSurface },
John Reck01a5ea32014-12-03 13:01:07 -0800668 { "nPauseSurface", "(JLandroid/view/Surface;)Z", (void*) android_view_ThreadedRenderer_pauseSurface },
Alan Viverette50210d92015-05-14 18:05:36 -0700669 { "nSetup", "(JIIFII)V", (void*) android_view_ThreadedRenderer_setup },
670 { "nSetLightCenter", "(JFFF)V", (void*) android_view_ThreadedRenderer_setLightCenter },
John Reck63a06672014-05-07 13:45:54 -0700671 { "nSetOpaque", "(JZ)V", (void*) android_view_ThreadedRenderer_setOpaque },
John Reckba6adf62015-02-19 14:36:50 -0800672 { "nSyncAndDrawFrame", "(J[JI)I", (void*) android_view_ThreadedRenderer_syncAndDrawFrame },
John Reck17035b02014-09-03 07:39:53 -0700673 { "nDestroy", "(J)V", (void*) android_view_ThreadedRenderer_destroy },
John Reck119907c2014-08-14 09:02:01 -0700674 { "nRegisterAnimatingRenderNode", "(JJ)V", (void*) android_view_ThreadedRenderer_registerAnimatingRenderNode },
John Reck3b202512014-06-23 13:13:08 -0700675 { "nInvokeFunctor", "(JZ)V", (void*) android_view_ThreadedRenderer_invokeFunctor },
John Reck19b6bcf2014-02-14 20:03:38 -0800676 { "nCreateTextureLayer", "(J)J", (void*) android_view_ThreadedRenderer_createTextureLayer },
John Reck3e824952014-08-20 10:08:39 -0700677 { "nBuildLayer", "(JJ)V", (void*) android_view_ThreadedRenderer_buildLayer },
John Reck3731dc22015-04-13 15:20:29 -0700678 { "nCopyLayerInto", "(JJLandroid/graphics/Bitmap;)Z", (void*) android_view_ThreadedRenderer_copyLayerInto },
John Reckd72e0a32014-05-29 18:56:11 -0700679 { "nPushLayerUpdate", "(JJ)V", (void*) android_view_ThreadedRenderer_pushLayerUpdate },
680 { "nCancelLayerUpdate", "(JJ)V", (void*) android_view_ThreadedRenderer_cancelLayerUpdate },
John Reck918ad522014-06-27 14:45:25 -0700681 { "nDetachSurfaceTexture", "(JJ)V", (void*) android_view_ThreadedRenderer_detachSurfaceTexture },
John Reckf47a5942014-06-30 16:20:04 -0700682 { "nDestroyHardwareResources", "(J)V", (void*) android_view_ThreadedRenderer_destroyHardwareResources },
683 { "nTrimMemory", "(I)V", (void*) android_view_ThreadedRenderer_trimMemory },
Chris Craik2507c342015-05-04 14:36:49 -0700684 { "nOverrideProperty", "(Ljava/lang/String;Ljava/lang/String;)V", (void*) android_view_ThreadedRenderer_overrideProperty },
John Reck28ad7b52014-04-07 16:59:25 -0700685 { "nFence", "(J)V", (void*) android_view_ThreadedRenderer_fence },
John Reckf47a5942014-06-30 16:20:04 -0700686 { "nStopDrawing", "(J)V", (void*) android_view_ThreadedRenderer_stopDrawing },
John Recka5dda642014-05-22 15:43:54 -0700687 { "nNotifyFramePending", "(J)V", (void*) android_view_ThreadedRenderer_notifyFramePending },
John Recke248bd12015-08-05 13:53:53 -0700688 { "nSerializeDisplayListTree", "(J)V", (void*) android_view_ThreadedRenderer_serializeDisplayListTree },
John Reckba6adf62015-02-19 14:36:50 -0800689 { "nDumpProfileInfo", "(JLjava/io/FileDescriptor;I)V", (void*) android_view_ThreadedRenderer_dumpProfileInfo },
John Reckedc524c2015-03-18 15:24:33 -0700690 { "nDumpProfileData", "([BLjava/io/FileDescriptor;)V", (void*) android_view_ThreadedRenderer_dumpProfileData },
John Reck84a4c882014-05-30 14:34:03 -0700691 { "setupShadersDiskCache", "(Ljava/lang/String;)V",
692 (void*) android_view_ThreadedRenderer_setupShadersDiskCache },
Michael Lentine03d8f7682016-01-31 15:37:11 -0600693 { "setupVulkanLayerPath", "(Ljava/lang/String;)V",
694 (void*) android_view_ThreadedRenderer_setupVulkanLayerPath },
Skuhneea7a7fb2015-08-28 07:10:31 -0700695 { "nAddRenderNode", "(JJZ)V", (void*) android_view_ThreadedRenderer_addRenderNode},
696 { "nRemoveRenderNode", "(JJ)V", (void*) android_view_ThreadedRenderer_removeRenderNode},
697 { "nDrawRenderNode", "(JJ)V", (void*) android_view_ThreadedRendererd_drawRenderNode},
Skuhneb8160872015-09-22 09:51:39 -0700698 { "nSetContentDrawBounds", "(JIIII)V", (void*)android_view_ThreadedRenderer_setContentDrawBounds},
Andres Morales910beb82016-02-02 16:19:40 -0800699 { "nAddFrameMetricsObserver",
700 "(JLandroid/view/FrameMetricsObserver;)J",
701 (void*)android_view_ThreadedRenderer_addFrameMetricsObserver },
702 { "nRemoveFrameMetricsObserver",
Andres Morales06f5bc72015-12-15 15:21:31 -0800703 "(JJ)V",
Andres Morales910beb82016-02-02 16:19:40 -0800704 (void*)android_view_ThreadedRenderer_removeFrameMetricsObserver },
John Reckcec24ae2013-11-05 13:27:50 -0800705};
706
707int register_android_view_ThreadedRenderer(JNIEnv* env) {
Andres Morales910beb82016-02-02 16:19:40 -0800708 jclass observerClass = FindClassOrDie(env, "android/view/FrameMetricsObserver");
709 gFrameMetricsObserverClassInfo.frameMetrics = GetFieldIDOrDie(
710 env, observerClass, "mFrameMetrics", "Landroid/view/FrameMetrics;");
711 gFrameMetricsObserverClassInfo.messageQueue = GetFieldIDOrDie(
712 env, observerClass, "mMessageQueue", "Landroid/os/MessageQueue;");
713 gFrameMetricsObserverClassInfo.callback = GetMethodIDOrDie(
714 env, observerClass, "notifyDataAvailable", "(I)V");
715
716 jclass metricsClass = FindClassOrDie(env, "android/view/FrameMetrics");
717 gFrameMetricsObserverClassInfo.timingDataBuffer = GetFieldIDOrDie(
718 env, metricsClass, "mTimingData", "[J");
Andres Morales06f5bc72015-12-15 15:21:31 -0800719
Andreas Gampeed6b9df2014-11-20 22:02:20 -0800720 return RegisterMethodsOrDie(env, kClassPathName, gMethods, NELEM(gMethods));
John Reckcec24ae2013-11-05 13:27:50 -0800721}
722
723}; // namespace android