RenderProxy.cpp revision 119907cd2575c56b1ebf66348b52e67aaf6a88d8
14f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck/*
24f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck * Copyright (C) 2013 The Android Open Source Project
34f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck *
44f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck * Licensed under the Apache License, Version 2.0 (the "License");
54f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck * you may not use this file except in compliance with the License.
64f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck * You may obtain a copy of the License at
74f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck *
84f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck *      http://www.apache.org/licenses/LICENSE-2.0
94f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck *
104f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck * Unless required by applicable law or agreed to in writing, software
114f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck * distributed under the License is distributed on an "AS IS" BASIS,
124f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
134f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck * See the License for the specific language governing permissions and
144f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck * limitations under the License.
154f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck */
164f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
174f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#include "RenderProxy.h"
184f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
194f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#include "CanvasContext.h"
204f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#include "RenderTask.h"
214f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#include "RenderThread.h"
224f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
2319b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck#include "../DeferredLayerUpdater.h"
244f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#include "../DisplayList.h"
2519b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck#include "../LayerRenderer.h"
264f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#include "../Rect.h"
274f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
284f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Recknamespace android {
294f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Recknamespace uirenderer {
304f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Recknamespace renderthread {
314f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
324f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#define ARGS(method) method ## Args
334f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
3419b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck#define CREATE_BRIDGE0(name) CREATE_BRIDGE(name,,,,,,,,)
354f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#define CREATE_BRIDGE1(name, a1) CREATE_BRIDGE(name, a1,,,,,,,)
364f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#define CREATE_BRIDGE2(name, a1, a2) CREATE_BRIDGE(name, a1,a2,,,,,,)
374f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#define CREATE_BRIDGE3(name, a1, a2, a3) CREATE_BRIDGE(name, a1,a2,a3,,,,,)
384f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#define CREATE_BRIDGE4(name, a1, a2, a3, a4) CREATE_BRIDGE(name, a1,a2,a3,a4,,,,)
39797b95b26bbb7557678af78b9a2a61830158920fChris Craik#define CREATE_BRIDGE5(name, a1, a2, a3, a4, a5) CREATE_BRIDGE(name, a1,a2,a3,a4,a5,,,)
40058fc640017c90120c599d378a4cbc55668b05b7Chris Craik#define CREATE_BRIDGE6(name, a1, a2, a3, a4, a5, a6) CREATE_BRIDGE(name, a1,a2,a3,a4,a5,a6,,)
41058fc640017c90120c599d378a4cbc55668b05b7Chris Craik#define CREATE_BRIDGE7(name, a1, a2, a3, a4, a5, a6, a7) CREATE_BRIDGE(name, a1,a2,a3,a4,a5,a6,a7,)
424f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#define CREATE_BRIDGE(name, a1, a2, a3, a4, a5, a6, a7, a8) \
434f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    typedef struct { \
444f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        a1; a2; a3; a4; a5; a6; a7; a8; \
454f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    } ARGS(name); \
464f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    static void* Bridge_ ## name(ARGS(name)* args)
474f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
484f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#define SETUP_TASK(method) \
494f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    LOG_ALWAYS_FATAL_IF( METHOD_INVOKE_PAYLOAD_SIZE < sizeof(ARGS(method)), \
50546f353e7f562fdbcf59980bcb7dc11567658aa3Mark Salyzyn        "METHOD_INVOKE_PAYLOAD_SIZE %zu is smaller than sizeof(" #method "Args) %zu", \
514f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck                METHOD_INVOKE_PAYLOAD_SIZE, sizeof(ARGS(method))); \
52e2c455264351964bf1ae78da2256c17258f0d3eaJohn Reck    MethodInvokeRenderTask* task = new MethodInvokeRenderTask((RunnableMethod) Bridge_ ## method); \
534f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    ARGS(method) *args = (ARGS(method) *) task->payload()
544f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
55119907cd2575c56b1ebf66348b52e67aaf6a88d8John ReckCREATE_BRIDGE4(createContext, RenderThread* thread, bool translucent,
56119907cd2575c56b1ebf66348b52e67aaf6a88d8John Reck        RenderNode* rootRenderNode, IContextFactory* contextFactory) {
57119907cd2575c56b1ebf66348b52e67aaf6a88d8John Reck    return new CanvasContext(*args->thread, args->translucent,
58119907cd2575c56b1ebf66348b52e67aaf6a88d8John Reck            args->rootRenderNode, args->contextFactory);
594f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
604f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
61119907cd2575c56b1ebf66348b52e67aaf6a88d8John ReckRenderProxy::RenderProxy(bool translucent, RenderNode* rootRenderNode, IContextFactory* contextFactory)
624f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        : mRenderThread(RenderThread::getInstance())
634f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        , mContext(0) {
644f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    SETUP_TASK(createContext);
654f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    args->translucent = translucent;
66e45b1fd03b524d2b57cc6c222d89076a31a08beaJohn Reck    args->rootRenderNode = rootRenderNode;
673b20251a355c88193c439f928a84ae69483fb488John Reck    args->thread = &mRenderThread;
68119907cd2575c56b1ebf66348b52e67aaf6a88d8John Reck    args->contextFactory = contextFactory;
694f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mContext = (CanvasContext*) postAndWait(task);
7018f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn Reck    mDrawFrameTask.setContext(&mRenderThread, mContext);
714f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
724f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
734f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John ReckRenderProxy::~RenderProxy() {
744f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    destroyContext();
754f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
764f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
774f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John ReckCREATE_BRIDGE1(destroyContext, CanvasContext* context) {
784f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    delete args->context;
794f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    return NULL;
804f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
814f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
824f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid RenderProxy::destroyContext() {
834f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (mContext) {
844f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        SETUP_TASK(destroyContext);
854f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        args->context = mContext;
864f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        mContext = 0;
8718f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn Reck        mDrawFrameTask.setContext(NULL, NULL);
88668f0e38ef0277d55d3118af37e17b8c435df85cJohn Reck        // This is also a fence as we need to be certain that there are no
89668f0e38ef0277d55d3118af37e17b8c435df85cJohn Reck        // outstanding mDrawFrame tasks posted before it is destroyed
90668f0e38ef0277d55d3118af37e17b8c435df85cJohn Reck        postAndWait(task);
914f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    }
924f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
934f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
9418f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn ReckCREATE_BRIDGE2(setFrameInterval, RenderThread* thread, nsecs_t frameIntervalNanos) {
9518f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn Reck    args->thread->timeLord().setFrameInterval(args->frameIntervalNanos);
9618f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn Reck    return NULL;
9718f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn Reck}
9818f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn Reck
9918f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn Reckvoid RenderProxy::setFrameInterval(nsecs_t frameIntervalNanos) {
10018f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn Reck    SETUP_TASK(setFrameInterval);
10118f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn Reck    args->thread = &mRenderThread;
10218f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn Reck    args->frameIntervalNanos = frameIntervalNanos;
10318f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn Reck    post(task);
10418f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn Reck}
10518f16e6fba74eda173e1e7c869e6e2e2acc073ffJohn Reck
106fe5e7b7346a54537b980796ceeca66bfdbd05561John ReckCREATE_BRIDGE1(loadSystemProperties, CanvasContext* context) {
107e4280baaa709c74d86cf6a389a4674ca665f5af6John Reck    bool needsRedraw = false;
108e4280baaa709c74d86cf6a389a4674ca665f5af6John Reck    if (Caches::hasInstance()) {
109e4280baaa709c74d86cf6a389a4674ca665f5af6John Reck        needsRedraw = Caches::getInstance().initProperties();
110e4280baaa709c74d86cf6a389a4674ca665f5af6John Reck    }
111fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck    if (args->context->profiler().loadSystemProperties()) {
112fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck        needsRedraw = true;
113fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck    }
114e4280baaa709c74d86cf6a389a4674ca665f5af6John Reck    return (void*) needsRedraw;
115e4280baaa709c74d86cf6a389a4674ca665f5af6John Reck}
116e4280baaa709c74d86cf6a389a4674ca665f5af6John Reck
117e4280baaa709c74d86cf6a389a4674ca665f5af6John Reckbool RenderProxy::loadSystemProperties() {
118e4280baaa709c74d86cf6a389a4674ca665f5af6John Reck    SETUP_TASK(loadSystemProperties);
119fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck    args->context = mContext;
120e4280baaa709c74d86cf6a389a4674ca665f5af6John Reck    return (bool) postAndWait(task);
121e4280baaa709c74d86cf6a389a4674ca665f5af6John Reck}
122e4280baaa709c74d86cf6a389a4674ca665f5af6John Reck
123a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn ReckCREATE_BRIDGE2(initialize, CanvasContext* context, ANativeWindow* window) {
1244f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    return (void*) args->context->initialize(args->window);
1254f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
1264f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
127f7d9c1dc84671d4e99657ef071d275700d85bb11John Reckbool RenderProxy::initialize(const sp<ANativeWindow>& window) {
1284f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    SETUP_TASK(initialize);
1294f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    args->context = mContext;
130f7d9c1dc84671d4e99657ef071d275700d85bb11John Reck    args->window = window.get();
1314f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    return (bool) postAndWait(task);
1324f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
1334f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
134a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn ReckCREATE_BRIDGE2(updateSurface, CanvasContext* context, ANativeWindow* window) {
1354f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    args->context->updateSurface(args->window);
1364f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    return NULL;
1374f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
1384f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
139f7d9c1dc84671d4e99657ef071d275700d85bb11John Reckvoid RenderProxy::updateSurface(const sp<ANativeWindow>& window) {
1404f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    SETUP_TASK(updateSurface);
1414f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    args->context = mContext;
142f7d9c1dc84671d4e99657ef071d275700d85bb11John Reck    args->window = window.get();
143f7d9c1dc84671d4e99657ef071d275700d85bb11John Reck    postAndWait(task);
144f7d9c1dc84671d4e99657ef071d275700d85bb11John Reck}
145f7d9c1dc84671d4e99657ef071d275700d85bb11John Reck
146a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn ReckCREATE_BRIDGE2(pauseSurface, CanvasContext* context, ANativeWindow* window) {
147f7d9c1dc84671d4e99657ef071d275700d85bb11John Reck    args->context->pauseSurface(args->window);
148f7d9c1dc84671d4e99657ef071d275700d85bb11John Reck    return NULL;
149f7d9c1dc84671d4e99657ef071d275700d85bb11John Reck}
150f7d9c1dc84671d4e99657ef071d275700d85bb11John Reck
151f7d9c1dc84671d4e99657ef071d275700d85bb11John Reckvoid RenderProxy::pauseSurface(const sp<ANativeWindow>& window) {
152f7d9c1dc84671d4e99657ef071d275700d85bb11John Reck    SETUP_TASK(pauseSurface);
153f7d9c1dc84671d4e99657ef071d275700d85bb11John Reck    args->context = mContext;
154f7d9c1dc84671d4e99657ef071d275700d85bb11John Reck    args->window = window.get();
155f7d9c1dc84671d4e99657ef071d275700d85bb11John Reck    postAndWait(task);
1564f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
1574f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
158058fc640017c90120c599d378a4cbc55668b05b7Chris CraikCREATE_BRIDGE7(setup, CanvasContext* context, int width, int height,
159058fc640017c90120c599d378a4cbc55668b05b7Chris Craik        Vector3 lightCenter, float lightRadius,
160058fc640017c90120c599d378a4cbc55668b05b7Chris Craik        uint8_t ambientShadowAlpha, uint8_t spotShadowAlpha) {
161058fc640017c90120c599d378a4cbc55668b05b7Chris Craik    args->context->setup(args->width, args->height, args->lightCenter, args->lightRadius,
162058fc640017c90120c599d378a4cbc55668b05b7Chris Craik            args->ambientShadowAlpha, args->spotShadowAlpha);
1634f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    return NULL;
1644f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
1654f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
166058fc640017c90120c599d378a4cbc55668b05b7Chris Craikvoid RenderProxy::setup(int width, int height, const Vector3& lightCenter, float lightRadius,
167058fc640017c90120c599d378a4cbc55668b05b7Chris Craik        uint8_t ambientShadowAlpha, uint8_t spotShadowAlpha) {
1684f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    SETUP_TASK(setup);
1694f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    args->context = mContext;
1704f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    args->width = width;
1714f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    args->height = height;
172797b95b26bbb7557678af78b9a2a61830158920fChris Craik    args->lightCenter = lightCenter;
173797b95b26bbb7557678af78b9a2a61830158920fChris Craik    args->lightRadius = lightRadius;
174058fc640017c90120c599d378a4cbc55668b05b7Chris Craik    args->ambientShadowAlpha = ambientShadowAlpha;
175058fc640017c90120c599d378a4cbc55668b05b7Chris Craik    args->spotShadowAlpha = spotShadowAlpha;
1764f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    post(task);
1774f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
1784f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
17963a06673253914510bbeebd500655008682dade1John ReckCREATE_BRIDGE2(setOpaque, CanvasContext* context, bool opaque) {
18063a06673253914510bbeebd500655008682dade1John Reck    args->context->setOpaque(args->opaque);
18163a06673253914510bbeebd500655008682dade1John Reck    return NULL;
18263a06673253914510bbeebd500655008682dade1John Reck}
18363a06673253914510bbeebd500655008682dade1John Reck
18463a06673253914510bbeebd500655008682dade1John Reckvoid RenderProxy::setOpaque(bool opaque) {
18563a06673253914510bbeebd500655008682dade1John Reck    SETUP_TASK(setOpaque);
18663a06673253914510bbeebd500655008682dade1John Reck    args->context = mContext;
18763a06673253914510bbeebd500655008682dade1John Reck    args->opaque = opaque;
18863a06673253914510bbeebd500655008682dade1John Reck    post(task);
18963a06673253914510bbeebd500655008682dade1John Reck}
19063a06673253914510bbeebd500655008682dade1John Reck
191fe5e7b7346a54537b980796ceeca66bfdbd05561John Reckint RenderProxy::syncAndDrawFrame(nsecs_t frameTimeNanos, nsecs_t recordDurationNanos,
192e4267ea4f20740c37c01bfb6aefcf61fddc4566aJohn Reck        float density) {
193fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck    mDrawFrameTask.setDensity(density);
194fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck    return mDrawFrameTask.drawFrame(frameTimeNanos, recordDurationNanos);
1954f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
1964f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
197fae904d63947fe1687d1d44be29234cc3d538f24John ReckCREATE_BRIDGE1(destroyCanvasAndSurface, CanvasContext* context) {
198fae904d63947fe1687d1d44be29234cc3d538f24John Reck    args->context->destroyCanvasAndSurface();
1994f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    return NULL;
2004f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
2014f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
202fae904d63947fe1687d1d44be29234cc3d538f24John Reckvoid RenderProxy::destroyCanvasAndSurface() {
203fae904d63947fe1687d1d44be29234cc3d538f24John Reck    SETUP_TASK(destroyCanvasAndSurface);
2044f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    args->context = mContext;
205fae904d63947fe1687d1d44be29234cc3d538f24John Reck    // destroyCanvasAndSurface() needs a fence as when it returns the
206fae904d63947fe1687d1d44be29234cc3d538f24John Reck    // underlying BufferQueue is going to be released from under
207fae904d63947fe1687d1d44be29234cc3d538f24John Reck    // the render thread.
208fae904d63947fe1687d1d44be29234cc3d538f24John Reck    postAndWait(task);
2094f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
2104f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
2113b20251a355c88193c439f928a84ae69483fb488John ReckCREATE_BRIDGE2(invokeFunctor, RenderThread* thread, Functor* functor) {
2123b20251a355c88193c439f928a84ae69483fb488John Reck    CanvasContext::invokeFunctor(*args->thread, args->functor);
2130d1f634f4b5e1bb37aa51777efb6a68619488d01John Reck    return NULL;
2140d1f634f4b5e1bb37aa51777efb6a68619488d01John Reck}
2150d1f634f4b5e1bb37aa51777efb6a68619488d01John Reck
2160d1f634f4b5e1bb37aa51777efb6a68619488d01John Reckvoid RenderProxy::invokeFunctor(Functor* functor, bool waitForCompletion) {
217d3d8dafc2f61fb118c060720b52684c59303f3dbJohn Reck    ATRACE_CALL();
2183b20251a355c88193c439f928a84ae69483fb488John Reck    RenderThread& thread = RenderThread::getInstance();
2190d1f634f4b5e1bb37aa51777efb6a68619488d01John Reck    SETUP_TASK(invokeFunctor);
2203b20251a355c88193c439f928a84ae69483fb488John Reck    args->thread = &thread;
2210d1f634f4b5e1bb37aa51777efb6a68619488d01John Reck    args->functor = functor;
2220d1f634f4b5e1bb37aa51777efb6a68619488d01John Reck    if (waitForCompletion) {
2233b20251a355c88193c439f928a84ae69483fb488John Reck        // waitForCompletion = true is expected to be fairly rare and only
2243b20251a355c88193c439f928a84ae69483fb488John Reck        // happen in destruction. Thus it should be fine to temporarily
2253b20251a355c88193c439f928a84ae69483fb488John Reck        // create a Mutex
2263b20251a355c88193c439f928a84ae69483fb488John Reck        Mutex mutex;
2273b20251a355c88193c439f928a84ae69483fb488John Reck        Condition condition;
2283b20251a355c88193c439f928a84ae69483fb488John Reck        SignalingRenderTask syncTask(task, &mutex, &condition);
2293b20251a355c88193c439f928a84ae69483fb488John Reck        AutoMutex _lock(mutex);
2303b20251a355c88193c439f928a84ae69483fb488John Reck        thread.queue(&syncTask);
2313b20251a355c88193c439f928a84ae69483fb488John Reck        condition.wait(mutex);
2320d1f634f4b5e1bb37aa51777efb6a68619488d01John Reck    } else {
2333b20251a355c88193c439f928a84ae69483fb488John Reck        thread.queue(task);
2340d1f634f4b5e1bb37aa51777efb6a68619488d01John Reck    }
2350d1f634f4b5e1bb37aa51777efb6a68619488d01John Reck}
2360d1f634f4b5e1bb37aa51777efb6a68619488d01John Reck
237fc53ef27793a39e9effd829e9cae02a9ca14147eJohn ReckCREATE_BRIDGE2(runWithGlContext, CanvasContext* context, RenderTask* task) {
238fc53ef27793a39e9effd829e9cae02a9ca14147eJohn Reck    args->context->runWithGlContext(args->task);
239fc53ef27793a39e9effd829e9cae02a9ca14147eJohn Reck    return NULL;
240fc53ef27793a39e9effd829e9cae02a9ca14147eJohn Reck}
241fc53ef27793a39e9effd829e9cae02a9ca14147eJohn Reck
242fc53ef27793a39e9effd829e9cae02a9ca14147eJohn Reckvoid RenderProxy::runWithGlContext(RenderTask* gltask) {
243fc53ef27793a39e9effd829e9cae02a9ca14147eJohn Reck    SETUP_TASK(runWithGlContext);
244fc53ef27793a39e9effd829e9cae02a9ca14147eJohn Reck    args->context = mContext;
245fc53ef27793a39e9effd829e9cae02a9ca14147eJohn Reck    args->task = gltask;
246fc53ef27793a39e9effd829e9cae02a9ca14147eJohn Reck    postAndWait(task);
247fc53ef27793a39e9effd829e9cae02a9ca14147eJohn Reck}
248fc53ef27793a39e9effd829e9cae02a9ca14147eJohn Reck
249d72e0a339b54af0c4e731513bbad120dff694723John ReckCREATE_BRIDGE1(destroyLayer, Layer* layer) {
250d72e0a339b54af0c4e731513bbad120dff694723John Reck    LayerRenderer::destroyLayer(args->layer);
251d72e0a339b54af0c4e731513bbad120dff694723John Reck    return NULL;
252d72e0a339b54af0c4e731513bbad120dff694723John Reck}
253d72e0a339b54af0c4e731513bbad120dff694723John Reck
2543b20251a355c88193c439f928a84ae69483fb488John Reckvoid RenderProxy::enqueueDestroyLayer(Layer* layer) {
255d72e0a339b54af0c4e731513bbad120dff694723John Reck    SETUP_TASK(destroyLayer);
256d72e0a339b54af0c4e731513bbad120dff694723John Reck    args->layer = layer;
257d72e0a339b54af0c4e731513bbad120dff694723John Reck    RenderThread::getInstance().queue(task);
258d72e0a339b54af0c4e731513bbad120dff694723John Reck}
259d72e0a339b54af0c4e731513bbad120dff694723John Reck
2601949e7928eeec22cd3f74b5f763a4eb433238453John ReckCREATE_BRIDGE3(createDisplayListLayer, CanvasContext* context, int width, int height) {
2611949e7928eeec22cd3f74b5f763a4eb433238453John Reck    Layer* layer = args->context->createRenderLayer(args->width, args->height);
26219b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    if (!layer) return 0;
2633b20251a355c88193c439f928a84ae69483fb488John Reck    return new DeferredLayerUpdater(layer, RenderProxy::enqueueDestroyLayer);
26419b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck}
26519b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck
26619b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John ReckDeferredLayerUpdater* RenderProxy::createDisplayListLayer(int width, int height) {
26719b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    SETUP_TASK(createDisplayListLayer);
26819b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    args->width = width;
26919b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    args->height = height;
2701949e7928eeec22cd3f74b5f763a4eb433238453John Reck    args->context = mContext;
27119b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    void* retval = postAndWait(task);
27219b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    DeferredLayerUpdater* layer = reinterpret_cast<DeferredLayerUpdater*>(retval);
27319b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    return layer;
27419b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck}
27519b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck
2761949e7928eeec22cd3f74b5f763a4eb433238453John ReckCREATE_BRIDGE1(createTextureLayer, CanvasContext* context) {
2771949e7928eeec22cd3f74b5f763a4eb433238453John Reck    Layer* layer = args->context->createTextureLayer();
27819b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    if (!layer) return 0;
2793b20251a355c88193c439f928a84ae69483fb488John Reck    return new DeferredLayerUpdater(layer, RenderProxy::enqueueDestroyLayer);
28019b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck}
28119b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck
28219b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John ReckDeferredLayerUpdater* RenderProxy::createTextureLayer() {
28319b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    SETUP_TASK(createTextureLayer);
2841949e7928eeec22cd3f74b5f763a4eb433238453John Reck    args->context = mContext;
28519b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    void* retval = postAndWait(task);
28619b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    DeferredLayerUpdater* layer = reinterpret_cast<DeferredLayerUpdater*>(retval);
28719b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    return layer;
28819b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck}
28919b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck
2903e8249568cc428296ac76c7ddce3f0382d40fe5bJohn ReckCREATE_BRIDGE2(buildLayer, CanvasContext* context, RenderNode* node) {
2913e8249568cc428296ac76c7ddce3f0382d40fe5bJohn Reck    args->context->buildLayer(args->node);
2923e8249568cc428296ac76c7ddce3f0382d40fe5bJohn Reck    return NULL;
2933e8249568cc428296ac76c7ddce3f0382d40fe5bJohn Reck}
2943e8249568cc428296ac76c7ddce3f0382d40fe5bJohn Reck
2953e8249568cc428296ac76c7ddce3f0382d40fe5bJohn Reckvoid RenderProxy::buildLayer(RenderNode* node) {
2963e8249568cc428296ac76c7ddce3f0382d40fe5bJohn Reck    SETUP_TASK(buildLayer);
2973e8249568cc428296ac76c7ddce3f0382d40fe5bJohn Reck    args->context = mContext;
2983e8249568cc428296ac76c7ddce3f0382d40fe5bJohn Reck    args->node = node;
2993e8249568cc428296ac76c7ddce3f0382d40fe5bJohn Reck    postAndWait(task);
3003e8249568cc428296ac76c7ddce3f0382d40fe5bJohn Reck}
3013e8249568cc428296ac76c7ddce3f0382d40fe5bJohn Reck
30219b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John ReckCREATE_BRIDGE3(copyLayerInto, CanvasContext* context, DeferredLayerUpdater* layer,
30319b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck        SkBitmap* bitmap) {
30419b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    bool success = args->context->copyLayerInto(args->layer, args->bitmap);
30519b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    return (void*) success;
30619b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck}
30719b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck
30819b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reckbool RenderProxy::copyLayerInto(DeferredLayerUpdater* layer, SkBitmap* bitmap) {
30919b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    SETUP_TASK(copyLayerInto);
31019b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    args->context = mContext;
31119b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    args->layer = layer;
31219b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    args->bitmap = bitmap;
31319b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck    return (bool) postAndWait(task);
31419b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck}
31519b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck
316d72e0a339b54af0c4e731513bbad120dff694723John Reckvoid RenderProxy::pushLayerUpdate(DeferredLayerUpdater* layer) {
317d72e0a339b54af0c4e731513bbad120dff694723John Reck    mDrawFrameTask.pushLayerUpdate(layer);
318d72e0a339b54af0c4e731513bbad120dff694723John Reck}
319d72e0a339b54af0c4e731513bbad120dff694723John Reck
320d72e0a339b54af0c4e731513bbad120dff694723John Reckvoid RenderProxy::cancelLayerUpdate(DeferredLayerUpdater* layer) {
321d72e0a339b54af0c4e731513bbad120dff694723John Reck    mDrawFrameTask.removeLayerUpdate(layer);
322918ad523b2780e0c893f3d2a32d4ec13f2a7e921John Reck}
323918ad523b2780e0c893f3d2a32d4ec13f2a7e921John Reck
324918ad523b2780e0c893f3d2a32d4ec13f2a7e921John ReckCREATE_BRIDGE1(detachSurfaceTexture, DeferredLayerUpdater* layer) {
325918ad523b2780e0c893f3d2a32d4ec13f2a7e921John Reck    args->layer->detachSurfaceTexture();
326918ad523b2780e0c893f3d2a32d4ec13f2a7e921John Reck    return NULL;
327918ad523b2780e0c893f3d2a32d4ec13f2a7e921John Reck}
328918ad523b2780e0c893f3d2a32d4ec13f2a7e921John Reck
329918ad523b2780e0c893f3d2a32d4ec13f2a7e921John Reckvoid RenderProxy::detachSurfaceTexture(DeferredLayerUpdater* layer) {
330918ad523b2780e0c893f3d2a32d4ec13f2a7e921John Reck    SETUP_TASK(detachSurfaceTexture);
331918ad523b2780e0c893f3d2a32d4ec13f2a7e921John Reck    args->layer = layer;
332918ad523b2780e0c893f3d2a32d4ec13f2a7e921John Reck    postAndWait(task);
33319b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck}
33419b6bcfd83eb7fb92ebd06d2fec89e308311f1d0John Reck
335f47a594f5250b1914c36423ee6b371f0b8db09d0John ReckCREATE_BRIDGE1(destroyHardwareResources, CanvasContext* context) {
336f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck    args->context->destroyHardwareResources();
337e1628b7c6fc3822fa83cf02028ce8ad67abb0afeJohn Reck    return NULL;
338e1628b7c6fc3822fa83cf02028ce8ad67abb0afeJohn Reck}
339e1628b7c6fc3822fa83cf02028ce8ad67abb0afeJohn Reck
340f47a594f5250b1914c36423ee6b371f0b8db09d0John Reckvoid RenderProxy::destroyHardwareResources() {
341f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck    SETUP_TASK(destroyHardwareResources);
342e1628b7c6fc3822fa83cf02028ce8ad67abb0afeJohn Reck    args->context = mContext;
343e1628b7c6fc3822fa83cf02028ce8ad67abb0afeJohn Reck    post(task);
344e1628b7c6fc3822fa83cf02028ce8ad67abb0afeJohn Reck}
345e1628b7c6fc3822fa83cf02028ce8ad67abb0afeJohn Reck
346f47a594f5250b1914c36423ee6b371f0b8db09d0John ReckCREATE_BRIDGE2(timMemory, RenderThread* thread, int level) {
347f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck    CanvasContext::trimMemory(*args->thread, args->level);
348f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck    return NULL;
349f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck}
350f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck
351f47a594f5250b1914c36423ee6b371f0b8db09d0John Reckvoid RenderProxy::trimMemory(int level) {
352cd3a22cfec09c065d0667dd044c0788912e82465John Reck    // Avoid creating a RenderThread to do a trimMemory.
353cd3a22cfec09c065d0667dd044c0788912e82465John Reck    if (RenderThread::hasInstance()) {
354cd3a22cfec09c065d0667dd044c0788912e82465John Reck        RenderThread& thread = RenderThread::getInstance();
355cd3a22cfec09c065d0667dd044c0788912e82465John Reck        SETUP_TASK(timMemory);
356cd3a22cfec09c065d0667dd044c0788912e82465John Reck        args->thread = &thread;
357cd3a22cfec09c065d0667dd044c0788912e82465John Reck        args->level = level;
358cd3a22cfec09c065d0667dd044c0788912e82465John Reck        thread.queue(task);
359cd3a22cfec09c065d0667dd044c0788912e82465John Reck    }
360f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck}
361f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck
36228ad7b52e038ef0cdd89f753d9839444a434b299John ReckCREATE_BRIDGE0(fence) {
36328ad7b52e038ef0cdd89f753d9839444a434b299John Reck    // Intentionally empty
36428ad7b52e038ef0cdd89f753d9839444a434b299John Reck    return NULL;
36528ad7b52e038ef0cdd89f753d9839444a434b299John Reck}
36628ad7b52e038ef0cdd89f753d9839444a434b299John Reck
36728ad7b52e038ef0cdd89f753d9839444a434b299John Reckvoid RenderProxy::fence() {
36828ad7b52e038ef0cdd89f753d9839444a434b299John Reck    SETUP_TASK(fence);
36928ad7b52e038ef0cdd89f753d9839444a434b299John Reck    postAndWait(task);
37028ad7b52e038ef0cdd89f753d9839444a434b299John Reck}
37128ad7b52e038ef0cdd89f753d9839444a434b299John Reck
372f47a594f5250b1914c36423ee6b371f0b8db09d0John ReckCREATE_BRIDGE1(stopDrawing, CanvasContext* context) {
373f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck    args->context->stopDrawing();
374f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck    return NULL;
375f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck}
376f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck
377f47a594f5250b1914c36423ee6b371f0b8db09d0John Reckvoid RenderProxy::stopDrawing() {
378f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck    SETUP_TASK(stopDrawing);
379f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck    args->context = mContext;
380f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck    postAndWait(task);
381f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck}
382f47a594f5250b1914c36423ee6b371f0b8db09d0John Reck
383a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn ReckCREATE_BRIDGE1(notifyFramePending, CanvasContext* context) {
384a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn Reck    args->context->notifyFramePending();
385a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn Reck    return NULL;
386a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn Reck}
387a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn Reck
388a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn Reckvoid RenderProxy::notifyFramePending() {
389a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn Reck    SETUP_TASK(notifyFramePending);
390a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn Reck    args->context = mContext;
391a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn Reck    mRenderThread.queueAtFront(task);
392a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn Reck}
393a5dda645da738da7b4ae15e28fa7d93d3b04b94fJohn Reck
394fe5e7b7346a54537b980796ceeca66bfdbd05561John ReckCREATE_BRIDGE2(dumpProfileInfo, CanvasContext* context, int fd) {
395fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck    args->context->profiler().dumpData(args->fd);
396fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck    return NULL;
397fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck}
398fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck
399fe5e7b7346a54537b980796ceeca66bfdbd05561John Reckvoid RenderProxy::dumpProfileInfo(int fd) {
400fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck    SETUP_TASK(dumpProfileInfo);
401fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck    args->context = mContext;
402fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck    args->fd = fd;
403fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck    postAndWait(task);
404fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck}
405fe5e7b7346a54537b980796ceeca66bfdbd05561John Reck
4063b20251a355c88193c439f928a84ae69483fb488John ReckCREATE_BRIDGE4(setTextureAtlas, RenderThread* thread, GraphicBuffer* buffer, int64_t* map, size_t size) {
4073b20251a355c88193c439f928a84ae69483fb488John Reck    CanvasContext::setTextureAtlas(*args->thread, args->buffer, args->map, args->size);
4083b20251a355c88193c439f928a84ae69483fb488John Reck    args->buffer->decStrong(0);
4093b20251a355c88193c439f928a84ae69483fb488John Reck    return NULL;
4103b20251a355c88193c439f928a84ae69483fb488John Reck}
4113b20251a355c88193c439f928a84ae69483fb488John Reck
4123b20251a355c88193c439f928a84ae69483fb488John Reckvoid RenderProxy::setTextureAtlas(const sp<GraphicBuffer>& buffer, int64_t* map, size_t size) {
4133b20251a355c88193c439f928a84ae69483fb488John Reck    SETUP_TASK(setTextureAtlas);
4143b20251a355c88193c439f928a84ae69483fb488John Reck    args->thread = &mRenderThread;
4153b20251a355c88193c439f928a84ae69483fb488John Reck    args->buffer = buffer.get();
4163b20251a355c88193c439f928a84ae69483fb488John Reck    args->buffer->incStrong(0);
4173b20251a355c88193c439f928a84ae69483fb488John Reck    args->map = map;
4183b20251a355c88193c439f928a84ae69483fb488John Reck    args->size = size;
4193b20251a355c88193c439f928a84ae69483fb488John Reck    post(task);
4203b20251a355c88193c439f928a84ae69483fb488John Reck}
4213b20251a355c88193c439f928a84ae69483fb488John Reck
4224f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid RenderProxy::post(RenderTask* task) {
4234f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mRenderThread.queue(task);
4244f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
4254f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4264f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid* RenderProxy::postAndWait(MethodInvokeRenderTask* task) {
4274f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    void* retval;
4284f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    task->setReturnPtr(&retval);
4294f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    SignalingRenderTask syncTask(task, &mSyncMutex, &mSyncCondition);
4304f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    AutoMutex _lock(mSyncMutex);
431738ec3aace180018560998d1c2cdeb9ddde5fbfaChris Craik    mRenderThread.queue(&syncTask);
432738ec3aace180018560998d1c2cdeb9ddde5fbfaChris Craik    mSyncCondition.wait(mSyncMutex);
4334f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    return retval;
4344f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
4354f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4364f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck} /* namespace renderthread */
4374f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck} /* namespace uirenderer */
4384f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck} /* namespace android */
439