1d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase/*
2d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase * Copyright (C) 2012 The Android Open Source Project
3d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase *
4d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase * Licensed under the Apache License, Version 2.0 (the "License");
5d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase * you may not use this file except in compliance with the License.
6d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase * You may obtain a copy of the License at
7d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase *
8d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase *      http://www.apache.org/licenses/LICENSE-2.0
9d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase *
10d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase * Unless required by applicable law or agreed to in writing, software
11d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase * distributed under the License is distributed on an "AS IS" BASIS,
12d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase * See the License for the specific language governing permissions and
14d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase * limitations under the License.
15d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase */
16d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase
17d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase#define LOG_TAG "OpenGLRenderer"
18d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase
19d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase#include <utils/Log.h>
20d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase
2196885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy#include "DisplayList.h"
2296885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy#include "DeferredDisplayList.h"
23d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase#include "Layer.h"
2498d3a64ffa13596e3ea9125bbff40c51ec96bd8dChet Haase#include "LayerRenderer.h"
25d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase#include "OpenGLRenderer.h"
26d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase#include "Caches.h"
27d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase
28d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haasenamespace android {
29d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haasenamespace uirenderer {
30d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase
31603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet HaaseLayer::Layer(const uint32_t layerWidth, const uint32_t layerHeight) {
32603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    mesh = NULL;
33603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    meshIndices = NULL;
34603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    meshElementCount = 0;
35603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    cacheable = true;
367c25aab491707f7324f9941b8cfa9bd2b4b97e76Romain Guy    dirty = false;
37603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    textureLayer = false;
38603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    renderTarget = GL_TEXTURE_2D;
39603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    texture.width = layerWidth;
40603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    texture.height = layerHeight;
41603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    colorFilter = NULL;
42603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    deferredUpdateScheduled = false;
43603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    renderer = NULL;
44603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    displayList = NULL;
45603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    fbo = 0;
463bbacf27c0be1bae4e4483577fc89ae3113abe5dRomain Guy    stencil = NULL;
475bb3c730f5ebd2a0db1b02a8981c6fdbea6c1a2eRomain Guy    debugDrawUpdate = false;
4834416eaa1c07b3d7a139e780cea8f8d73219650eChris Craik    hasDrawnSinceUpdate = false;
4996885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    deferredList = NULL;
50603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase    Caches::getInstance().resourceCache.incrementRefcount(this);
51603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase}
52603f6de35f21d74ae242d52d501f4f5c25ff4f4cChet Haase
53d15ebf25c595b855f6978d0600218e3ea5f31e92Chet HaaseLayer::~Layer() {
54d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase    if (colorFilter) Caches::getInstance().resourceCache.decrementRefcount(colorFilter);
5598d3a64ffa13596e3ea9125bbff40c51ec96bd8dChet Haase    removeFbo();
568a1374946a928fcba7495c87ff6adda327fdfb9fRomain Guy    deleteTexture();
5796885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy
5896885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    delete[] mesh;
5996885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    delete[] meshIndices;
6096885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    delete deferredList;
6197dc9172b0e58979c63de0dedbab656399a62281Romain Guy}
6297dc9172b0e58979c63de0dedbab656399a62281Romain Guy
632055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guyuint32_t Layer::computeIdealWidth(uint32_t layerWidth) {
642055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    return uint32_t(ceilf(layerWidth / float(LAYER_SIZE)) * LAYER_SIZE);
652055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy}
662055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy
672055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guyuint32_t Layer::computeIdealHeight(uint32_t layerHeight) {
682055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    return uint32_t(ceilf(layerHeight / float(LAYER_SIZE)) * LAYER_SIZE);
692055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy}
702055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy
712055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guybool Layer::resize(const uint32_t width, const uint32_t height) {
722055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    uint32_t desiredWidth = computeIdealWidth(width);
732055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    uint32_t desiredHeight = computeIdealWidth(height);
742055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy
752055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    if (desiredWidth <= getWidth() && desiredHeight <= getHeight()) {
762055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy        return true;
772055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    }
782055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy
79ce4a7dfc516ee61301e9af91fad17ca1320efaabRomain Guy    const uint32_t maxTextureSize = Caches::getInstance().maxTextureSize;
80ce4a7dfc516ee61301e9af91fad17ca1320efaabRomain Guy    if (desiredWidth > maxTextureSize || desiredHeight > maxTextureSize) {
81ce4a7dfc516ee61301e9af91fad17ca1320efaabRomain Guy        ALOGW("Layer exceeds max. dimensions supported by the GPU (%dx%d, max=%dx%d)",
82ce4a7dfc516ee61301e9af91fad17ca1320efaabRomain Guy                desiredWidth, desiredHeight, maxTextureSize, maxTextureSize);
83ce4a7dfc516ee61301e9af91fad17ca1320efaabRomain Guy        return false;
84ce4a7dfc516ee61301e9af91fad17ca1320efaabRomain Guy    }
85ce4a7dfc516ee61301e9af91fad17ca1320efaabRomain Guy
862055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    uint32_t oldWidth = getWidth();
872055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    uint32_t oldHeight = getHeight();
882055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy
892055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    setSize(desiredWidth, desiredHeight);
902055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy
912055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    if (fbo) {
922055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy        Caches::getInstance().activeTexture(0);
932055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy        bindTexture();
940908764b2b3cf5075df4178a5f0a8547dcb7b317Romain Guy        allocateTexture();
952055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy
962055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy        if (glGetError() != GL_NO_ERROR) {
972055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy            setSize(oldWidth, oldHeight);
982055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy            return false;
992055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy        }
1002055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    }
1012055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy
1022055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    if (stencil) {
1033bbacf27c0be1bae4e4483577fc89ae3113abe5dRomain Guy        stencil->bind();
1043bbacf27c0be1bae4e4483577fc89ae3113abe5dRomain Guy        stencil->resize(desiredWidth, desiredHeight);
1052055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy
1062055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy        if (glGetError() != GL_NO_ERROR) {
1072055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy            setSize(oldWidth, oldHeight);
1082055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy            return false;
1092055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy        }
1102055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    }
1112055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy
1122055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy    return true;
1132055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy}
1142055abaa0a590c35e27e1ae2e7d7cfccdfb98b59Romain Guy
1158ce00301a023eecaeb8891ce906f67b513ebb42aRomain Guyvoid Layer::removeFbo(bool flush) {
1168ce00301a023eecaeb8891ce906f67b513ebb42aRomain Guy    if (stencil) {
1178ce00301a023eecaeb8891ce906f67b513ebb42aRomain Guy        GLuint previousFbo;
1188ce00301a023eecaeb8891ce906f67b513ebb42aRomain Guy        glGetIntegerv(GL_FRAMEBUFFER_BINDING, (GLint*) &previousFbo);
1198ce00301a023eecaeb8891ce906f67b513ebb42aRomain Guy        if (fbo != previousFbo) glBindFramebuffer(GL_FRAMEBUFFER, fbo);
1208ce00301a023eecaeb8891ce906f67b513ebb42aRomain Guy        glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_STENCIL_ATTACHMENT, GL_RENDERBUFFER, 0);
1218ce00301a023eecaeb8891ce906f67b513ebb42aRomain Guy        if (fbo != previousFbo) glBindFramebuffer(GL_FRAMEBUFFER, previousFbo);
1228ce00301a023eecaeb8891ce906f67b513ebb42aRomain Guy
1238d4aeb7111afac0c3c7e56d4ad5d92f9cfce2ffdRomain Guy        Caches::getInstance().renderBufferCache.put(stencil);
1243bbacf27c0be1bae4e4483577fc89ae3113abe5dRomain Guy        stencil = NULL;
1258ce00301a023eecaeb8891ce906f67b513ebb42aRomain Guy    }
1268ce00301a023eecaeb8891ce906f67b513ebb42aRomain Guy
12798d3a64ffa13596e3ea9125bbff40c51ec96bd8dChet Haase    if (fbo) {
1288ce00301a023eecaeb8891ce906f67b513ebb42aRomain Guy        if (flush) LayerRenderer::flushLayer(this);
1298ce00301a023eecaeb8891ce906f67b513ebb42aRomain Guy        // If put fails the cache will delete the FBO
13098d3a64ffa13596e3ea9125bbff40c51ec96bd8dChet Haase        Caches::getInstance().fboCache.put(fbo);
13198d3a64ffa13596e3ea9125bbff40c51ec96bd8dChet Haase        fbo = 0;
13256257aff8a55c847be72be9924c392033fd8151dDave Burke    }
13356257aff8a55c847be72be9924c392033fd8151dDave Burke}
13456257aff8a55c847be72be9924c392033fd8151dDave Burke
135d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haasevoid Layer::setPaint(SkPaint* paint) {
136d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase    OpenGLRenderer::getAlphaAndModeDirect(paint, &alpha, &mode);
137d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase}
138d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase
139d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haasevoid Layer::setColorFilter(SkiaColorFilter* filter) {
140d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase    if (colorFilter) {
141d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase        Caches::getInstance().resourceCache.decrementRefcount(colorFilter);
142d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase    }
143d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase    colorFilter = filter;
144d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase    if (colorFilter) {
145d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase        Caches::getInstance().resourceCache.incrementRefcount(colorFilter);
146d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase    }
147d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase}
148d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase
14996885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guyvoid Layer::defer() {
15096885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    if (!deferredList) {
15196885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy        deferredList = new DeferredDisplayList;
15296885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    }
15396885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    DeferStateStruct deferredState(*deferredList, *renderer,
15496885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy            DisplayList::kReplayFlag_ClipChildren);
15596885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy
15696885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    const float width = layer.getWidth();
15796885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    const float height = layer.getHeight();
15896885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy
15996885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    if (dirtyRect.isEmpty() || (dirtyRect.left <= 0 && dirtyRect.top <= 0 &&
16096885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy            dirtyRect.right >= width && dirtyRect.bottom >= height)) {
16196885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy        dirtyRect.set(0, 0, width, height);
16296885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    }
16396885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy
16496885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    renderer->initViewport(width, height);
16596885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    renderer->setupFrameState(dirtyRect.left, dirtyRect.top,
16696885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy            dirtyRect.right, dirtyRect.bottom, !isBlend());
16796885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy
16896885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    displayList->defer(deferredState, 0);
16902b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy
17002b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy    deferredUpdateScheduled = false;
17196885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy}
17296885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy
17396885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guyvoid Layer::flush() {
174696dcf84a382fcc5ef7041e0bbf3af20b0ecb99fRomain Guy    if (deferredList) {
17596885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy        renderer->setViewport(layer.getWidth(), layer.getHeight());
17696885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy        renderer->prepareDirty(dirtyRect.left, dirtyRect.top, dirtyRect.right, dirtyRect.bottom,
17796885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy                !isBlend());
17896885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy
17996885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy        deferredList->flush(*renderer, dirtyRect);
18096885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy
18196885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy        renderer->finish();
18296885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy        renderer = NULL;
18396885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy
18496885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy        dirtyRect.setEmpty();
1851206b9bba91f7ed899c5c87427cce725fe5aadfcChris Craik        displayList = NULL;
18696885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy    }
18796885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy}
18896885eb480c5e0526fe2f77d30f6e551f3f3ceabRomain Guy
18902b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guyvoid Layer::render() {
19002b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy    renderer->setViewport(layer.getWidth(), layer.getHeight());
19102b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy    renderer->prepareDirty(dirtyRect.left, dirtyRect.top, dirtyRect.right, dirtyRect.bottom,
19202b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy            !isBlend());
19302b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy
19402b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy    renderer->drawDisplayList(displayList, dirtyRect, DisplayList::kReplayFlag_ClipChildren);
19502b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy
19602b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy    renderer->finish();
19702b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy    renderer = NULL;
19802b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy
19902b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy    dirtyRect.setEmpty();
20002b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy
20102b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy    deferredUpdateScheduled = false;
20202b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy    displayList = NULL;
20302b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy}
20402b49b70ede0b9eb760ff334823aee1d9520ed85Romain Guy
205d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase}; // namespace uirenderer
206d15ebf25c595b855f6978d0600218e3ea5f31e92Chet Haase}; // namespace android
207