/* * Copyright (C) 2012 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define LOG_TAG "OpenGLRenderer" #include #include "DisplayList.h" #include "DeferredDisplayList.h" #include "Layer.h" #include "LayerRenderer.h" #include "OpenGLRenderer.h" #include "Caches.h" namespace android { namespace uirenderer { Layer::Layer(const uint32_t layerWidth, const uint32_t layerHeight) { mesh = NULL; meshIndices = NULL; meshElementCount = 0; cacheable = true; dirty = false; textureLayer = false; renderTarget = GL_TEXTURE_2D; texture.width = layerWidth; texture.height = layerHeight; colorFilter = NULL; deferredUpdateScheduled = false; renderer = NULL; displayList = NULL; fbo = 0; stencil = NULL; debugDrawUpdate = false; deferredList = NULL; Caches::getInstance().resourceCache.incrementRefcount(this); } Layer::~Layer() { if (colorFilter) Caches::getInstance().resourceCache.decrementRefcount(colorFilter); removeFbo(); deleteTexture(); delete[] mesh; delete[] meshIndices; delete deferredList; } uint32_t Layer::computeIdealWidth(uint32_t layerWidth) { return uint32_t(ceilf(layerWidth / float(LAYER_SIZE)) * LAYER_SIZE); } uint32_t Layer::computeIdealHeight(uint32_t layerHeight) { return uint32_t(ceilf(layerHeight / float(LAYER_SIZE)) * LAYER_SIZE); } bool Layer::resize(const uint32_t width, const uint32_t height) { uint32_t desiredWidth = computeIdealWidth(width); uint32_t desiredHeight = computeIdealWidth(height); if (desiredWidth <= getWidth() && desiredHeight <= getHeight()) { return true; } const uint32_t maxTextureSize = Caches::getInstance().maxTextureSize; if (desiredWidth > maxTextureSize || desiredHeight > maxTextureSize) { ALOGW("Layer exceeds max. dimensions supported by the GPU (%dx%d, max=%dx%d)", desiredWidth, desiredHeight, maxTextureSize, maxTextureSize); return false; } uint32_t oldWidth = getWidth(); uint32_t oldHeight = getHeight(); setSize(desiredWidth, desiredHeight); if (fbo) { Caches::getInstance().activeTexture(0); bindTexture(); allocateTexture(GL_RGBA, GL_UNSIGNED_BYTE); if (glGetError() != GL_NO_ERROR) { setSize(oldWidth, oldHeight); return false; } } if (stencil) { stencil->bind(); stencil->resize(desiredWidth, desiredHeight); if (glGetError() != GL_NO_ERROR) { setSize(oldWidth, oldHeight); return false; } } return true; } void Layer::removeFbo(bool flush) { if (stencil) { GLuint previousFbo; glGetIntegerv(GL_FRAMEBUFFER_BINDING, (GLint*) &previousFbo); if (fbo != previousFbo) glBindFramebuffer(GL_FRAMEBUFFER, fbo); glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_STENCIL_ATTACHMENT, GL_RENDERBUFFER, 0); if (fbo != previousFbo) glBindFramebuffer(GL_FRAMEBUFFER, previousFbo); Caches::getInstance().renderBufferCache.put(stencil); stencil = NULL; } if (fbo) { if (flush) LayerRenderer::flushLayer(this); // If put fails the cache will delete the FBO Caches::getInstance().fboCache.put(fbo); fbo = 0; } } void Layer::setPaint(SkPaint* paint) { OpenGLRenderer::getAlphaAndModeDirect(paint, &alpha, &mode); } void Layer::setColorFilter(SkiaColorFilter* filter) { if (colorFilter) { Caches::getInstance().resourceCache.decrementRefcount(colorFilter); } colorFilter = filter; if (colorFilter) { Caches::getInstance().resourceCache.incrementRefcount(colorFilter); } } void Layer::defer() { if (!deferredList) { deferredList = new DeferredDisplayList; } DeferStateStruct deferredState(*deferredList, *renderer, DisplayList::kReplayFlag_ClipChildren); const float width = layer.getWidth(); const float height = layer.getHeight(); if (dirtyRect.isEmpty() || (dirtyRect.left <= 0 && dirtyRect.top <= 0 && dirtyRect.right >= width && dirtyRect.bottom >= height)) { dirtyRect.set(0, 0, width, height); } renderer->initViewport(width, height); renderer->setupFrameState(dirtyRect.left, dirtyRect.top, dirtyRect.right, dirtyRect.bottom, !isBlend()); displayList->defer(deferredState, 0); deferredUpdateScheduled = false; displayList = NULL; } void Layer::flush() { if (deferredList) { renderer->setViewport(layer.getWidth(), layer.getHeight()); renderer->prepareDirty(dirtyRect.left, dirtyRect.top, dirtyRect.right, dirtyRect.bottom, !isBlend()); deferredList->flush(*renderer, dirtyRect); renderer->finish(); renderer = NULL; dirtyRect.setEmpty(); deferredList->clear(); } } void Layer::render() { renderer->setViewport(layer.getWidth(), layer.getHeight()); renderer->prepareDirty(dirtyRect.left, dirtyRect.top, dirtyRect.right, dirtyRect.bottom, !isBlend()); renderer->drawDisplayList(displayList, dirtyRect, DisplayList::kReplayFlag_ClipChildren); renderer->finish(); renderer = NULL; dirtyRect.setEmpty(); deferredUpdateScheduled = false; displayList = NULL; } }; // namespace uirenderer }; // namespace android