CanvasContext.cpp revision 4f02bf4eef6af47f35c70c4dda5b7b9523d89ca0
123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck/*
223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck * Copyright (C) 2014 The Android Open Source Project
323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck *
423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck * Licensed under the Apache License, Version 2.0 (the "License");
523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck * you may not use this file except in compliance with the License.
623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck * You may obtain a copy of the License at
723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck *
823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck *      http://www.apache.org/licenses/LICENSE-2.0
923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck *
1023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck * Unless required by applicable law or agreed to in writing, software
1123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck * distributed under the License is distributed on an "AS IS" BASIS,
1223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck * See the License for the specific language governing permissions and
1423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck * limitations under the License.
1523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck */
1623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
1723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck#define LOG_TAG "CanvasContext"
1823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
1923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck#include "CanvasContext.h"
2023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
2123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck#include <cutils/properties.h>
224f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#include <private/hwui/DrawGlInfo.h>
2323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck#include <strings.h>
2423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
254f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#include "RenderThread.h"
2623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck#include "../Caches.h"
274f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#include "../OpenGLRenderer.h"
2823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck#include "../Stencil.h"
2923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
3023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck#define PROPERTY_RENDER_DIRTY_REGIONS "debug.hwui.render_dirty_regions"
3123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck#define GLES_VERSION 2
3223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
334f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#ifdef USE_OPENGL_RENDERER
344f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck// Android-specific addition that is used to show when frames began in systrace
354f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John ReckEGLAPI void EGLAPIENTRY eglBeginFrame(EGLDisplay dpy, EGLSurface surface);
364f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck#endif
374f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
3823b797ab5151eb2474f3bdd679f2f07bfd723042John Recknamespace android {
3923b797ab5151eb2474f3bdd679f2f07bfd723042John Recknamespace uirenderer {
4023b797ab5151eb2474f3bdd679f2f07bfd723042John Recknamespace renderthread {
4123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
4223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck#define ERROR_CASE(x) case x: return #x;
4323b797ab5151eb2474f3bdd679f2f07bfd723042John Reckstatic const char* egl_error_str(EGLint error) {
4423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    switch (error) {
4523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_SUCCESS)
4623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_NOT_INITIALIZED)
4723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_BAD_ACCESS)
4823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_BAD_ALLOC)
4923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_BAD_ATTRIBUTE)
5023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_BAD_CONFIG)
5123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_BAD_CONTEXT)
5223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_BAD_CURRENT_SURFACE)
5323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_BAD_DISPLAY)
5423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_BAD_MATCH)
5523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_BAD_NATIVE_PIXMAP)
5623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_BAD_NATIVE_WINDOW)
5723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_BAD_PARAMETER)
5823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_BAD_SURFACE)
5923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ERROR_CASE(EGL_CONTEXT_LOST)
6023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    default:
6123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        return "Unknown error";
6223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
6323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
6423b797ab5151eb2474f3bdd679f2f07bfd723042John Reckstatic const char* egl_error_str() {
6523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    return egl_error_str(eglGetError());
6623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
6723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
6823b797ab5151eb2474f3bdd679f2f07bfd723042John Reckstatic bool load_dirty_regions_property() {
6923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    char buf[PROPERTY_VALUE_MAX];
7023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    int len = property_get(PROPERTY_RENDER_DIRTY_REGIONS, buf, "true");
7123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    return !strncasecmp("true", buf, len);
7223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
7323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
7423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck// This class contains the shared global EGL objects, such as EGLDisplay
7523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck// and EGLConfig, which are re-used by CanvasContext
7623b797ab5151eb2474f3bdd679f2f07bfd723042John Reckclass GlobalContext {
7723b797ab5151eb2474f3bdd679f2f07bfd723042John Reckpublic:
7823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    static GlobalContext* get();
7923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
804f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    // Returns true on success, false on failure
814f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    void initialize();
8223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
834f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    void usePBufferSurface();
8423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    EGLSurface createSurface(EGLNativeWindowType window);
8523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    void destroySurface(EGLSurface surface);
8623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
8723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    void destroy();
8823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
8923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    bool isCurrent(EGLSurface surface) { return mCurrentSurface == surface; }
904f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    void makeCurrent(EGLSurface surface);
914f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    void beginFrame(EGLSurface surface, EGLint* width, EGLint* height);
924f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    void swapBuffers(EGLSurface surface);
9323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
9423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    bool enableDirtyRegions(EGLSurface surface);
9523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
9623b797ab5151eb2474f3bdd679f2f07bfd723042John Reckprivate:
9723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    GlobalContext();
9823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    // GlobalContext is never destroyed, method is purposely not implemented
9923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    ~GlobalContext();
10023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
1014f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    void loadConfig();
1024f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    void createContext();
1034f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    void initAtlas();
10423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
10523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    static GlobalContext* sContext;
10623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
10723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    EGLDisplay mEglDisplay;
10823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    EGLConfig mEglConfig;
10923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    EGLContext mEglContext;
11023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    EGLSurface mPBufferSurface;
11123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
11223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    const bool mRequestDirtyRegions;
11323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    bool mCanSetDirtyRegions;
11423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
11523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    EGLSurface mCurrentSurface;
11623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck};
11723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
11823b797ab5151eb2474f3bdd679f2f07bfd723042John ReckGlobalContext* GlobalContext::sContext = 0;
11923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
12023b797ab5151eb2474f3bdd679f2f07bfd723042John ReckGlobalContext* GlobalContext::get() {
12123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (!sContext) {
12223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        sContext = new GlobalContext();
12323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
12423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    return sContext;
12523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
12623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
12723b797ab5151eb2474f3bdd679f2f07bfd723042John ReckGlobalContext::GlobalContext()
12823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        : mEglDisplay(EGL_NO_DISPLAY)
12923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        , mEglConfig(0)
13023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        , mEglContext(EGL_NO_CONTEXT)
13123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        , mPBufferSurface(EGL_NO_SURFACE)
13223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        , mRequestDirtyRegions(load_dirty_regions_property())
13323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        , mCurrentSurface(EGL_NO_SURFACE) {
13423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    mCanSetDirtyRegions = mRequestDirtyRegions;
13523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    ALOGD("Render dirty regions requested: %s", mRequestDirtyRegions ? "true" : "false");
13623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
13723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
1384f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid GlobalContext::initialize() {
1394f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (mEglDisplay != EGL_NO_DISPLAY) return;
14023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
14123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    mEglDisplay = eglGetDisplay(EGL_DEFAULT_DISPLAY);
1424f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    LOG_ALWAYS_FATAL_IF(mEglDisplay == EGL_NO_DISPLAY,
1434f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck            "Failed to get EGL_DEFAULT_DISPLAY! err=%s", egl_error_str());
14423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
14523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    EGLint major, minor;
1464f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    LOG_ALWAYS_FATAL_IF(eglInitialize(mEglDisplay, &major, &minor) == EGL_FALSE,
1474f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck            "Failed to initialize display %p! err=%s", mEglDisplay, egl_error_str());
14823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
1494f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    ALOGI("Initialized EGL, version %d.%d", (int)major, (int)minor);
15023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
1514f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    loadConfig();
1524f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    createContext();
1534f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    usePBufferSurface();
1544f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    Caches::getInstance().init();
1554f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    initAtlas();
15623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
15723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
1584f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid GlobalContext::loadConfig() {
15923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    EGLint swapBehavior = mCanSetDirtyRegions ? EGL_SWAP_BEHAVIOR_PRESERVED_BIT : 0;
16023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    EGLint attribs[] = {
16123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
16223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            EGL_RED_SIZE, 8,
16323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            EGL_GREEN_SIZE, 8,
16423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            EGL_BLUE_SIZE, 8,
16523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            EGL_ALPHA_SIZE, 8,
16623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            EGL_DEPTH_SIZE, 0,
16723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            EGL_CONFIG_CAVEAT, EGL_NONE,
16823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            EGL_STENCIL_SIZE, Stencil::getStencilSize(),
16923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            EGL_SURFACE_TYPE, EGL_WINDOW_BIT | swapBehavior,
17023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            EGL_NONE
17123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    };
17223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
17323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    EGLint num_configs = 1;
17423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (!eglChooseConfig(mEglDisplay, attribs, &mEglConfig, num_configs, &num_configs)
17523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            || num_configs != 1) {
17623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        // Failed to get a valid config
17723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        if (mCanSetDirtyRegions) {
17823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            ALOGW("Failed to choose config with EGL_SWAP_BEHAVIOR_PRESERVED, retrying without...");
17923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            // Try again without dirty regions enabled
18023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            mCanSetDirtyRegions = false;
18123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            loadConfig();
18223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        } else {
1834f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck            LOG_ALWAYS_FATAL("Failed to choose config, error = %s", egl_error_str());
18423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        }
18523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
18623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
18723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
1884f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid GlobalContext::createContext() {
18923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    EGLint attribs[] = { EGL_CONTEXT_CLIENT_VERSION, GLES_VERSION, EGL_NONE };
19023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    mEglContext = eglCreateContext(mEglDisplay, mEglConfig, EGL_NO_CONTEXT, attribs);
1914f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    LOG_ALWAYS_FATAL_IF(mEglContext == EGL_NO_CONTEXT,
1924f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        "Failed to create context, error = %s", egl_error_str());
1934f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
1944f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
1954f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid GlobalContext::initAtlas() {
1964f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    // TODO implement
1974f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    // For now just run without an atlas
19823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
19923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
2004f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid GlobalContext::usePBufferSurface() {
2014f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    LOG_ALWAYS_FATAL_IF(mEglDisplay == EGL_NO_DISPLAY,
2024f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck            "usePBufferSurface() called on uninitialized GlobalContext!");
20323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
20423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (mPBufferSurface == EGL_NO_SURFACE) {
20523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        EGLint attribs[] = { EGL_WIDTH, 1, EGL_HEIGHT, 1, EGL_NONE };
20623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        mPBufferSurface = eglCreatePbufferSurface(mEglDisplay, mEglConfig, attribs);
20723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
2084f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    makeCurrent(mPBufferSurface);
20923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
21023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
21123b797ab5151eb2474f3bdd679f2f07bfd723042John ReckEGLSurface GlobalContext::createSurface(EGLNativeWindowType window) {
21223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    initialize();
21323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    return eglCreateWindowSurface(mEglDisplay, mEglConfig, window, NULL);
21423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
21523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
21623b797ab5151eb2474f3bdd679f2f07bfd723042John Reckvoid GlobalContext::destroySurface(EGLSurface surface) {
21723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (isCurrent(surface)) {
21823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        makeCurrent(EGL_NO_SURFACE);
21923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
22023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (!eglDestroySurface(mEglDisplay, surface)) {
22123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ALOGW("Failed to destroy surface %p, error=%s", (void*)surface, egl_error_str());
22223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
22323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
22423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
22523b797ab5151eb2474f3bdd679f2f07bfd723042John Reckvoid GlobalContext::destroy() {
22623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (mEglDisplay == EGL_NO_DISPLAY) return;
22723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
22823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    usePBufferSurface();
22923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (Caches::hasInstance()) {
23023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        Caches::getInstance().terminate();
23123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
23223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
23323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    eglDestroyContext(mEglDisplay, mEglContext);
23423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    eglDestroySurface(mEglDisplay, mPBufferSurface);
23523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    eglMakeCurrent(mEglDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
23623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    eglTerminate(mEglDisplay);
23723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    eglReleaseThread();
23823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
23923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    mEglDisplay = EGL_NO_DISPLAY;
24023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    mEglContext = EGL_NO_CONTEXT;
24123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    mPBufferSurface = EGL_NO_SURFACE;
24223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    mCurrentSurface = EGL_NO_SURFACE;
24323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
24423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
2454f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid GlobalContext::makeCurrent(EGLSurface surface) {
2464f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (isCurrent(surface)) return;
24723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
24823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (surface == EGL_NO_SURFACE) {
24923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        // If we are setting EGL_NO_SURFACE we don't care about any of the potential
25023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        // return errors, which would only happen if mEglDisplay had already been
25123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        // destroyed in which case the current context is already NO_CONTEXT
25223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        eglMakeCurrent(mEglDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
25323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    } else if (!eglMakeCurrent(mEglDisplay, surface, surface, mEglContext)) {
2544f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        LOG_ALWAYS_FATAL("Failed to make current on surface %p, error=%s",
2554f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck                (void*)surface, egl_error_str());
25623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
25723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    mCurrentSurface = surface;
25823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
25923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
2604f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid GlobalContext::beginFrame(EGLSurface surface, EGLint* width, EGLint* height) {
2614f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    LOG_ALWAYS_FATAL_IF(surface == EGL_NO_SURFACE,
2624f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck            "Tried to beginFrame on EGL_NO_SURFACE!");
2634f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    makeCurrent(surface);
2644f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (width) {
2654f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        eglQuerySurface(mEglDisplay, surface, EGL_WIDTH, width);
26623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
2674f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (height) {
2684f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        eglQuerySurface(mEglDisplay, surface, EGL_HEIGHT, height);
2694f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    }
2704f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    eglBeginFrame(mEglDisplay, surface);
2714f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
2724f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
2734f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid GlobalContext::swapBuffers(EGLSurface surface) {
2744f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    eglSwapBuffers(mEglDisplay, surface);
2754f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    EGLint err = eglGetError();
2764f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    // TODO: Check whether we need to special case EGL_CONTEXT_LOST
2774f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    LOG_ALWAYS_FATAL_IF(err != EGL_SUCCESS,
2784f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck            "Encountered EGL error %d %s during rendering", err, egl_error_str(err));
27923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
28023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
28123b797ab5151eb2474f3bdd679f2f07bfd723042John Reckbool GlobalContext::enableDirtyRegions(EGLSurface surface) {
28223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (!mRequestDirtyRegions) return false;
28323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
28423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (mCanSetDirtyRegions) {
28523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        if (!eglSurfaceAttrib(mEglDisplay, surface, EGL_SWAP_BEHAVIOR, EGL_BUFFER_PRESERVED)) {
28623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            ALOGW("Failed to set EGL_SWAP_BEHAVIOR on surface %p, error=%s",
28723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck                    (void*) surface, egl_error_str());
28823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck            return false;
28923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        }
29023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        return true;
29123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
29223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    // Perhaps it is already enabled?
29323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    EGLint value;
29423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (!eglQuerySurface(mEglDisplay, surface, EGL_SWAP_BEHAVIOR, &value)) {
29523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        ALOGW("Failed to query EGL_SWAP_BEHAVIOR on surface %p, error=%p",
29623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck                (void*) surface, egl_error_str());
29723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        return false;
29823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
29923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    return value == EGL_BUFFER_PRESERVED;
30023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
30123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
3024f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John ReckCanvasContext::CanvasContext(bool translucent)
3034f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        : mRenderThread(RenderThread::getInstance())
3044f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        , mEglSurface(EGL_NO_SURFACE)
3054f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        , mDirtyRegionsEnabled(false)
3064f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        , mOpaque(!translucent)
3074f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        , mCanvas(0)
3084f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        , mHaveNewSurface(false)
3094f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        , mInvokeFunctorsPending(false)
3104f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        , mInvokeFunctorsTask(this) {
31123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    mGlobalContext = GlobalContext::get();
31223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
31323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
31423b797ab5151eb2474f3bdd679f2f07bfd723042John ReckCanvasContext::~CanvasContext() {
3154f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    removeFunctorsTask();
3164f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    destroyCanvas();
3174f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
3184f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
3194f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid CanvasContext::destroyCanvas() {
3204f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (mCanvas) {
3214f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        delete mCanvas;
3224f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        mCanvas = 0;
3234f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    }
32423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    setSurface(NULL);
32523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
32623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
3274f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid CanvasContext::setSurface(EGLNativeWindowType window) {
32823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (mEglSurface != EGL_NO_SURFACE) {
32923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        mGlobalContext->destroySurface(mEglSurface);
33023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        mEglSurface = EGL_NO_SURFACE;
33123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
33223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
33323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (window) {
33423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        mEglSurface = mGlobalContext->createSurface(window);
3354f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        LOG_ALWAYS_FATAL_IF(mEglSurface == EGL_NO_SURFACE,
3364f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck                "Failed to create EGLSurface for window %p, eglErr = %s",
3374f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck                (void*) window, egl_error_str());
33823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
33923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
34023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    if (mEglSurface != EGL_NO_SURFACE) {
34123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck        mDirtyRegionsEnabled = mGlobalContext->enableDirtyRegions(mEglSurface);
3424f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        mHaveNewSurface = true;
3434f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    }
3444f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
3454f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
3464f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid CanvasContext::swapBuffers() {
3474f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mGlobalContext->swapBuffers(mEglSurface);
3484f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mHaveNewSurface = false;
3494f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
3504f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
3514f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid CanvasContext::makeCurrent() {
3524f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mGlobalContext->makeCurrent(mEglSurface);
3534f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
3544f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
3554f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckbool CanvasContext::initialize(EGLNativeWindowType window) {
3564f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (mCanvas) return false;
3574f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    setSurface(window);
3584f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    makeCurrent();
3594f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mCanvas = new OpenGLRenderer();
3604f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mCanvas->initProperties();
3614f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    return true;
3624f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
3634f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
3644f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid CanvasContext::updateSurface(EGLNativeWindowType window) {
3654f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    setSurface(window);
3664f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    makeCurrent();
3674f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
3684f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
3694f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid CanvasContext::setup(int width, int height) {
3704f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (!mCanvas) return;
3714f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mCanvas->setViewport(width, height);
3724f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
3734f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
3744f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid CanvasContext::drawDisplayList(DisplayList* displayList, Rect* dirty) {
3754f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    LOG_ALWAYS_FATAL_IF(!mCanvas || mEglSurface == EGL_NO_SURFACE,
3764f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck            "drawDisplayList called on a context with no canvas or surface!");
3774f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
3784f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    EGLint width, height;
3794f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mGlobalContext->beginFrame(mEglSurface, &width, &height);
3804f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (width != mCanvas->getViewportWidth() || height != mCanvas->getViewportHeight()) {
3814f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        mCanvas->setViewport(width, height);
3824f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        dirty = NULL;
3834f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    } else if (!mDirtyRegionsEnabled || mHaveNewSurface) {
3844f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        dirty = NULL;
3854f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    }
3864f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
3874f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    status_t status;
3884f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (dirty) {
3894f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        status = mCanvas->prepareDirty(dirty->left, dirty->top,
3904f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck                dirty->right, dirty->bottom, mOpaque);
3914f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    } else {
3924f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        status = mCanvas->prepare(mOpaque);
3934f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    }
3944f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
3954f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    Rect outBounds;
3964f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    status |= mCanvas->drawDisplayList(displayList, outBounds);
3974f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    handleFunctorStatus(status, outBounds);
3984f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
3994f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    // TODO: Draw debug info
4004f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    // TODO: Performance tracking
4014f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4024f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mCanvas->finish();
4034f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4044f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (status & DrawGlInfo::kStatusDrew) {
4054f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        swapBuffers();
40623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck    }
40723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
40823b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
4094f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid InvokeFunctorsTask::run() {
4104f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mContext->invokeFunctors();
41123b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
41223b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
4134f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid CanvasContext::attachFunctor(Functor* functor) {
4144f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (!mCanvas) return;
4154f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4164f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mCanvas->attachFunctor(functor);
4174f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    removeFunctorsTask();
4184f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    queueFunctorsTask(0);
41923b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
42023b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
4214f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid CanvasContext::detachFunctor(Functor* functor) {
4224f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (!mCanvas) return;
4234f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4244f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mCanvas->detachFunctor(functor);
4254f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
4264f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4274f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid CanvasContext::invokeFunctors() {
4284f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mInvokeFunctorsPending = false;
4294f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4304f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (!mCanvas) return;
4314f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4324f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    makeCurrent();
4334f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    Rect dirty;
4344f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    int status = mCanvas->invokeFunctors(dirty);
4354f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    handleFunctorStatus(status, dirty);
4364f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
4374f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4384f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid CanvasContext::handleFunctorStatus(int status, const Rect& redrawClip) {
4394f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (status & DrawGlInfo::kStatusDraw) {
4404f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        // TODO: Invalidate the redrawClip
4414f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        // Do we need to post to ViewRootImpl like the current renderer?
4424f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        // Can we just enqueue ourselves to re-invoke the same display list?
4434f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        // Something else entirely? Does ChromiumView still want this in a
4444f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        // RenderThread world?
4454f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    }
4464f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4474f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (status & DrawGlInfo::kStatusInvoke) {
4484f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck        queueFunctorsTask();
4494f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    }
4504f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
4514f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4524f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid CanvasContext::removeFunctorsTask() {
4534f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (!mInvokeFunctorsPending) return;
4544f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4554f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mRenderThread.remove(&mInvokeFunctorsTask);
4564f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck}
4574f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4584f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reckvoid CanvasContext::queueFunctorsTask(int delayMs) {
4594f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    if (mInvokeFunctorsPending) return;
4604f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck
4614f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mInvokeFunctorsPending = true;
4624f02bf4eef6af47f35c70c4dda5b7b9523d89ca0John Reck    mRenderThread.queueDelayed(&mInvokeFunctorsTask, delayMs);
46323b797ab5151eb2474f3bdd679f2f07bfd723042John Reck}
46423b797ab5151eb2474f3bdd679f2f07bfd723042John Reck
46523b797ab5151eb2474f3bdd679f2f07bfd723042John Reck} /* namespace renderthread */
46623b797ab5151eb2474f3bdd679f2f07bfd723042John Reck} /* namespace uirenderer */
46723b797ab5151eb2474f3bdd679f2f07bfd723042John Reck} /* namespace android */
468