SurfaceFlinger.cpp revision 32ae094d87cbb35f8b31acb7b83b430db62d5925
19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/*
29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2007 The Android Open Source Project
39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License.
69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at
79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and
149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License.
159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <stdlib.h>
189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <stdio.h>
199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <stdint.h>
209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <unistd.h>
219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <fcntl.h>
229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <errno.h>
239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <math.h>
24207637327511c960805713971e2e7dd11cb9c290Jean-Baptiste Queru#include <limits.h>
259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <sys/types.h>
269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <sys/stat.h>
279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <sys/ioctl.h>
289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <cutils/log.h>
309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <cutils/properties.h>
319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
320795272aa226f4e965968a03daddc53ce30b7cdaMathias Agopian#include <binder/IPCThreadState.h>
330795272aa226f4e965968a03daddc53ce30b7cdaMathias Agopian#include <binder/IServiceManager.h>
34d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian#include <binder/MemoryHeapBase.h>
35d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/String8.h>
379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/String16.h>
389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/StopWatch.h>
399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
406950e428feaccc8164b989ef64e771a99948797aMathias Agopian#include <ui/GraphicBufferAllocator.h>
4104262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian#include <ui/GraphicLog.h>
429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <ui/PixelFormat.h>
439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <pixelflinger/pixelflinger.h>
459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <GLES/gl.h>
469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "clz.h"
48781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian#include "GLExtensions.h"
499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "Layer.h"
509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "LayerDim.h"
519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "SurfaceFlinger.h"
529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "DisplayHardware/DisplayHardware.h"
54e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian#include "DisplayHardware/HWComposer.h"
559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
56627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian/* ideally AID_GRAPHICS would be in a semi-public header
57627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian * or there would be a way to map a user/group name to its id
58627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian */
59627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian#ifndef AID_GRAPHICS
60627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian#define AID_GRAPHICS 1003
61627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian#endif
62627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#define DISPLAY_COUNT       1
649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectnamespace android {
669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectSurfaceFlinger::SurfaceFlinger()
699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    :   BnSurfaceComposer(), Thread(false),
709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mTransactionFlags(0),
719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mTransactionCount(0),
729779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        mResizeTransationPending(false),
731473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        mLayersRemoved(false),
749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mBootTime(systemTime()),
75151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian        mHardwareTest("android.permission.HARDWARE_TEST"),
76151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian        mAccessSurfaceFlinger("android.permission.ACCESS_SURFACE_FLINGER"),
77ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        mReadFramebuffer("android.permission.READ_FRAME_BUFFER"),
78151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian        mDump("android.permission.DUMP"),
799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mVisibleRegionsDirty(false),
80e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        mHwWorkListDirty(false),
819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDeferReleaseConsole(false),
829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mFreezeDisplay(false),
83d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        mElectronBeamAnimationMode(0),
849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mFreezeCount(0),
85c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project        mFreezeDisplayTime(0),
869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDebugRegion(0),
879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDebugBackground(0),
886a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        mDebugDisableHWC(0),
89a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers(0),
90a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastSwapBufferTime(0),
91a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInTransaction(0),
92a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastTransactionTime(0),
936950e428feaccc8164b989ef64e771a99948797aMathias Agopian        mBootFinished(false),
949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mConsoleSignals(0),
959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mSecureFrameBuffer(0)
969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    init();
989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::init()
1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LOGI("SurfaceFlinger is starting");
1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // debugging stuff...
1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    char value[PROPERTY_VALUE_MAX];
1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    property_get("debug.sf.showupdates", value, "0");
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDebugRegion = atoi(value);
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    property_get("debug.sf.showbackground", value, "0");
1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDebugBackground = atoi(value);
1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
111e5c0a7b7810092a593ff386843927238c175bdbdMathias Agopian    LOGI_IF(mDebugRegion,       "showupdates enabled");
112e5c0a7b7810092a593ff386843927238c175bdbdMathias Agopian    LOGI_IF(mDebugBackground,   "showbackground enabled");
1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectSurfaceFlinger::~SurfaceFlinger()
1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDeleteTextures(1, &mWormholeTexName);
1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
120d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopiansp<IMemoryHeap> SurfaceFlinger::getCblk() const
1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
122d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    return mServerHeap;
1239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
125770492cb2b19f6a36ad748cd05fbedfbb9a67dfaMathias Agopiansp<ISurfaceComposerClient> SurfaceFlinger::createConnection()
1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
127593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<ISurfaceComposerClient> bclient;
128593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<Client> client(new Client(this));
129593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    status_t err = client->initCheck();
130593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (err == NO_ERROR) {
131593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        bclient = client;
1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return bclient;
1349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1367623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<ISurfaceComposerClient> SurfaceFlinger::createClientConnection()
1377623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
1387623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<ISurfaceComposerClient> bclient;
1397623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<UserClient> client(new UserClient(this));
1407623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    status_t err = client->initCheck();
1417623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (err == NO_ERROR) {
1427623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        bclient = client;
1437623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
1447623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return bclient;
1457623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
1467623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
147f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennissp<IGraphicBufferAlloc> SurfaceFlinger::createGraphicBufferAlloc()
148f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis{
149f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    sp<GraphicBufferAlloc> gba(new GraphicBufferAlloc());
150f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    return gba;
151f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis}
1527623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst GraphicPlane& SurfaceFlinger::graphicPlane(int dpy) const
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LOGE_IF(uint32_t(dpy) >= DISPLAY_COUNT, "Invalid DisplayID %d", dpy);
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(mGraphicPlanes[dpy]);
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return plane;
1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane& SurfaceFlinger::graphicPlane(int dpy)
1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return const_cast<GraphicPlane&>(
1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const_cast<SurfaceFlinger const *>(this)->graphicPlane(dpy));
1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::bootFinished()
1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const nsecs_t now = systemTime();
1699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const nsecs_t duration = now - mBootTime;
170e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber    LOGI("Boot is finished (%ld ms)", long(ns2ms(duration)) );
1716950e428feaccc8164b989ef64e771a99948797aMathias Agopian    mBootFinished = true;
172627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    property_set("ctl.stop", "bootanim");
1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::onFirstRef()
1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    run("SurfaceFlinger", PRIORITY_URGENT_DISPLAY);
1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // Wait for the main thread to be done with its initialization
1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mReadyToRunBarrier.wait();
1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic inline uint16_t pack565(int r, int g, int b) {
1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return (r<<11)|(g<<5)|b;
1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::readyToRun()
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LOGI(   "SurfaceFlinger's main thread ready to run. "
1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            "Initializing graphics H/W...");
1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // we only support one display currently
1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int dpy = 0;
1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    {
1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // initialize the main display
1979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        GraphicPlane& plane(graphicPlane(dpy));
1989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        DisplayHardware* const hw = new DisplayHardware(this, dpy);
1999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        plane.setDisplayHardware(hw);
2009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
202d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    // create the shared control-block
203d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    mServerHeap = new MemoryHeapBase(4096,
204d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian            MemoryHeapBase::READ_ONLY, "SurfaceFlinger read-only heap");
205d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    LOGE_IF(mServerHeap==0, "can't create shared memory dealer");
206e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
207d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    mServerCblk = static_cast<surface_flinger_cblk_t*>(mServerHeap->getBase());
208d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    LOGE_IF(mServerCblk==0, "can't get to shared control block's address");
209e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
210d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    new(mServerCblk) surface_flinger_cblk_t;
211d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize primary screen
2139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // (other display should be initialized in the same manner, but
2149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // asynchronously, as they could come and go. None of this is supported
2159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // yet).
2169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(dpy));
2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw = plane.displayHardware();
2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint32_t w = hw.getWidth();
2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint32_t h = hw.getHeight();
2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint32_t f = hw.getFormat();
2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    hw.makeCurrent();
2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize the shared control block
2249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mServerCblk->connected |= 1<<dpy;
2259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    display_cblk_t* dcblk = mServerCblk->displays + dpy;
2269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    memset(dcblk, 0, sizeof(display_cblk_t));
22766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    dcblk->w            = plane.getWidth();
22866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    dcblk->h            = plane.getHeight();
2299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->format       = f;
2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->orientation  = ISurfaceComposer::eOrientationDefault;
2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->xdpi         = hw.getDpiX();
2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->ydpi         = hw.getDpiY();
2339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->fps          = hw.getRefreshRate();
2349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->density      = hw.getDensity();
2359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // Initialize OpenGL|ES
2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glPixelStorei(GL_UNPACK_ALIGNMENT, 4);
238e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber    glPixelStorei(GL_PACK_ALIGNMENT, 4);
2399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glEnableClientState(GL_VERTEX_ARRAY);
2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glEnable(GL_SCISSOR_TEST);
2419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glShadeModel(GL_FLAT);
2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_CULL_FACE);
2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint16_t g0 = pack565(0x0F,0x1F,0x0F);
2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint16_t g1 = pack565(0x17,0x2f,0x17);
2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint16_t textureData[4] = { g0, g1, g1, g0 };
2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glGenTextures(1, &mWormholeTexName);
2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glBindTexture(GL_TEXTURE_2D, mWormholeTexName);
2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, 2, 2, 0,
2559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            GL_RGB, GL_UNSIGNED_SHORT_5_6_5, textureData);
2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glViewport(0, 0, w, h);
2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glMatrixMode(GL_PROJECTION);
2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glLoadIdentity();
2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glOrthof(0, w, h, 0, 0, 1);
2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mReadyToRunBarrier.open();
2639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
2659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *  We're now ready to accept clients...
2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
268627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    // start boot animation
269627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    property_set("ctl.start", "bootanim");
270e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
2719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#if 0
2769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark -
2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark Events Handler
2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif
2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::waitForEvent()
2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
2826ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    while (true) {
2836ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        nsecs_t timeout = -1;
2840e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        const nsecs_t freezeDisplayTimeout = ms2ns(5000);
2856ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        if (UNLIKELY(isFrozen())) {
2866ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            // wait 5 seconds
2876ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            const nsecs_t now = systemTime();
2886ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            if (mFreezeDisplayTime == 0) {
2896ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian                mFreezeDisplayTime = now;
2906ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            }
2916ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            nsecs_t waitTime = freezeDisplayTimeout - (now - mFreezeDisplayTime);
2926ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            timeout = waitTime>0 ? waitTime : 0;
293c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project        }
2946ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian
295898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        sp<MessageBase> msg = mEventQueue.waitMessage(timeout);
2960e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian
2970e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        // see if we timed out
2980e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        if (isFrozen()) {
2990e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            const nsecs_t now = systemTime();
3000e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            nsecs_t frozenTime = (now - mFreezeDisplayTime);
3010e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            if (frozenTime >= freezeDisplayTimeout) {
3029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // we timed out and are still frozen
3039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                LOGW("timeout expired mFreezeDisplay=%d, mFreezeCount=%d",
3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        mFreezeDisplay, mFreezeCount);
3050e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                mFreezeDisplayTime = 0;
3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeCount = 0;
307c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project                mFreezeDisplay = false;
3080e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            }
3090e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        }
3100e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian
3110e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        if (msg != 0) {
3120e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            switch (msg->what) {
3130e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                case MessageQueue::INVALIDATE:
3140e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                    // invalidate message, just return to the main loop
3150e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                    return;
3169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
3179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::signalEvent() {
3226ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    mEventQueue.invalidate();
3239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::signal() const {
3266ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    // this is the IPC call
3276ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    const_cast<SurfaceFlinger*>(this)->signalEvent();
3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
330898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianstatus_t SurfaceFlinger::postMessageAsync(const sp<MessageBase>& msg,
331898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        nsecs_t reltime, uint32_t flags)
3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
333898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    return mEventQueue.postMessage(msg, reltime, flags);
334898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian}
335898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian
336898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianstatus_t SurfaceFlinger::postMessageSync(const sp<MessageBase>& msg,
337898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        nsecs_t reltime, uint32_t flags)
338898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian{
339898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    status_t res = mEventQueue.postMessage(msg, reltime, flags);
340898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    if (res == NO_ERROR) {
341898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        msg->wait();
342898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    }
343898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    return res;
3449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
3479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#if 0
3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark -
3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark Main loop
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif
3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool SurfaceFlinger::threadLoop()
3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
3549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    waitForEvent();
3559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // check for transactions
3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(mConsoleSignals)) {
3589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        handleConsoleEvents();
3599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (LIKELY(mTransactionCount == 0)) {
3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // if we're in a global transaction, don't do anything.
3639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const uint32_t mask = eTransactionNeeded | eTraversalNeeded;
3649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        uint32_t transactionFlags = getTransactionFlags(mask);
3659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (LIKELY(transactionFlags)) {
3669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            handleTransaction(transactionFlags);
3679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // post surfaces (if needed)
3719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    handlePageFlip();
3729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
373e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    if (UNLIKELY(mHwWorkListDirty)) {
374e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        // build the h/w work list
375e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        handleWorkList();
376e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    }
377e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
3789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw(graphicPlane(0).displayHardware());
37981384bf927c47a4efa653b14273084a13e67e3acMathias Agopian    if (LIKELY(hw.canDraw() && !isFrozen())) {
3809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // repaint the framebuffer (if needed)
38104262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian
38204262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        const int index = hw.getCurrentBufferIndex();
38304262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        GraphicLog& logger(GraphicLog::getInstance());
38404262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian
38504262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_REPAINT, index);
3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        handleRepaint();
3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
388b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian        // inform the h/w that we're done compositing
38904262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_COMPOSITION_COMPLETE, index);
390b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian        hw.compositionComplete();
391b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian
39204262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_SWAP_BUFFERS, index);
3934c0a4a2b74db9d4a439a0aaa39d80586f7eb1258Antti Hatala        postFramebuffer();
3944c0a4a2b74db9d4a439a0aaa39d80586f7eb1258Antti Hatala
39504262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_REPAINT_DONE, index);
3969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
3979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // pretend we did the post
398a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian        hw.compositionComplete();
3999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        usleep(16667); // 60 fps period
4009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return true;
4029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::postFramebuffer()
4059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (!mInvalidRegion.isEmpty()) {
4079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw(graphicPlane(0).displayHardware());
408a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t now = systemTime();
409a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers = now;
4109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.flip(mInvalidRegion);
411a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastSwapBufferTime = systemTime() - now;
412a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers = 0;
4139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mInvalidRegion.clear();
4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleConsoleEvents()
4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // something to do with the console
4209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw = graphicPlane(0).displayHardware();
4219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int what = android_atomic_and(0, &mConsoleSignals);
4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (what & eConsoleAcquired) {
4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.acquireScreen();
4252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // this is a temporary work-around, eventually this should be called
4262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // by the power-manager
427d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        SurfaceFlinger::turnElectronBeamOn(mElectronBeamAnimationMode);
4289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
430aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (mDeferReleaseConsole && hw.isScreenAcquired()) {
431ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian        // We got the release signal before the acquire signal
4329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDeferReleaseConsole = false;
4339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.releaseScreen();
4349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (what & eConsoleReleased) {
437aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        if (hw.isScreenAcquired()) {
4389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            hw.releaseScreen();
4399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
4409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDeferReleaseConsole = true;
4419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.set(hw.bounds());
4459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleTransaction(uint32_t transactionFlags)
4489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4492d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    Vector< sp<LayerBase> > ditchedLayers;
4502d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian
451ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian    /*
452ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     * Perform and commit the transaction
453ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     */
454ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian
4552d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    { // scope for the lock
4562d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian        Mutex::Autolock _l(mStateLock);
457a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t now = systemTime();
458a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInTransaction = now;
4592d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian        handleTransactionLocked(transactionFlags, ditchedLayers);
460a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastTransactionTime = systemTime() - now;
461a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInTransaction = 0;
462fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian        invalidateHwcGeometry();
463ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian        // here the transaction has been committed
4642d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    }
4652d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian
466ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian    /*
467ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     * Clean-up all layers that went away
468ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     * (do this without the lock held)
469ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     */
470e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
4712d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    const size_t count = ditchedLayers.size();
4722d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
473ffae4fcc78d0f280da6052d102b11962fb8041b7Mathias Agopian        if (ditchedLayers[i] != 0) {
474ffae4fcc78d0f280da6052d102b11962fb8041b7Mathias Agopian            //LOGD("ditching layer %p", ditchedLayers[i].get());
475ffae4fcc78d0f280da6052d102b11962fb8041b7Mathias Agopian            ditchedLayers[i]->ditch();
476ffae4fcc78d0f280da6052d102b11962fb8041b7Mathias Agopian        }
4772d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    }
4782d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian}
4799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4802d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopianvoid SurfaceFlinger::handleTransactionLocked(
4812d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian        uint32_t transactionFlags, Vector< sp<LayerBase> >& ditchedLayers)
4822d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian{
4832d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    const LayerVector& currentLayers(mCurrentState.layersSortedByZ);
4849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const size_t count = currentLayers.size();
4859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
4879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Traversal of the children
4889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * (perform the transaction for each of them if needed)
4899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
4909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const bool layersNeedTransaction = transactionFlags & eTraversalNeeded;
4929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (layersNeedTransaction) {
4939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (size_t i=0 ; i<count ; i++) {
4941473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian            const sp<LayerBase>& layer = currentLayers[i];
4959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            uint32_t trFlags = layer->getTransactionFlags(eTransactionNeeded);
4969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (!trFlags) continue;
4979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const uint32_t flags = layer->doTransaction(0);
4999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (flags & Layer::eVisibleRegion)
5009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mVisibleRegionsDirty = true;
5019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Perform our own transaction if needed
5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (transactionFlags & eTransactionNeeded) {
5099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mCurrentState.orientation != mDrawingState.orientation) {
5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // the orientation has changed, recompute all visible regions
5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // and invalidate everything.
5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int dpy = 0;
5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int orientation = mCurrentState.orientation;
515eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian            const uint32_t type = mCurrentState.orientationType;
5169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            GraphicPlane& plane(graphicPlane(dpy));
5179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            plane.setOrientation(orientation);
5189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // update the shared control block
5209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const DisplayHardware& hw(plane.displayHardware());
5219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            volatile display_cblk_t* dcblk = mServerCblk->displays + dpy;
5229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dcblk->orientation = orientation;
52366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            dcblk->w = plane.getWidth();
52466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            dcblk->h = plane.getHeight();
5259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
5279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(hw.bounds());
5289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mCurrentState.freezeDisplay != mDrawingState.freezeDisplay) {
5319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // freezing or unfreezing the display -> trigger animation if needed
5329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mFreezeDisplay = mCurrentState.freezeDisplay;
53331901cc0b6f0c678be4f629c8c3405700e63c346Mathias Agopian            if (mFreezeDisplay)
53431901cc0b6f0c678be4f629c8c3405700e63c346Mathias Agopian                 mFreezeDisplayTime = 0;
5359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
537a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        if (currentLayers.size() > mDrawingState.layersSortedByZ.size()) {
538a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            // layers have been added
5399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
5409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
542a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        // some layers might have been removed, so
543a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        // we need to update the regions they're exposing.
544a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        if (mLayersRemoved) {
545248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian            mLayersRemoved = false;
5469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
547a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            const LayerVector& previousLayers(mDrawingState.layersSortedByZ);
5482d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian            const size_t count = previousLayers.size();
5492d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian            for (size_t i=0 ; i<count ; i++) {
550a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                const sp<LayerBase>& layer(previousLayers[i]);
551a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                if (currentLayers.indexOf( layer ) < 0) {
552a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                    // this layer is not visible anymore
5532d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian                    ditchedLayers.add(layer);
55433863dd9e1005478d20b70fad42e447ac97f5abfMathias Agopian                    mDirtyRegionRemovedLayer.orSelf(layer->visibleRegionScreen);
555a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                }
556a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            }
5579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    commitTransaction();
5619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
5629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectsp<FreezeLock> SurfaceFlinger::getFreezeLock() const
5649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
5659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return new FreezeLock(const_cast<SurfaceFlinger *>(this));
5669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
5679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::computeVisibleRegions(
5699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LayerVector& currentLayers, Region& dirtyRegion, Region& opaqueRegion)
5709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
5719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(0));
5729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Transform& planeTransform(plane.transform());
5739c041bbd81789c209e2369ba958306979b67614fMathias Agopian    const DisplayHardware& hw(plane.displayHardware());
5749c041bbd81789c209e2369ba958306979b67614fMathias Agopian    const Region screenRegion(hw.bounds());
5759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region aboveOpaqueLayers;
5779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region aboveCoveredLayers;
5789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region dirty;
5799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool secureFrameBuffer = false;
5819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t i = currentLayers.size();
5839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    while (i--) {
5841473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        const sp<LayerBase>& layer = currentLayers[i];
5859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->validateVisibility(planeTransform);
5869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // start with the whole surface at its current location
58812cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        const Layer::State& s(layer->drawingState());
5899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5909c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
5919c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * opaqueRegion: area of a surface that is fully opaque.
5929c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
5939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region opaqueRegion;
5949c041bbd81789c209e2369ba958306979b67614fMathias Agopian
5959c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
5969c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * visibleRegion: area of a surface that is visible on screen
5979c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * and not fully transparent. This is essentially the layer's
5989c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * footprint minus the opaque regions above it.
5999c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * Areas covered by a translucent surface are considered visible.
6009c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
6019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region visibleRegion;
6029c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6039c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
6049c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * coveredRegion: area of a surface that is covered by all
6059c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * visible regions above it (which includes the translucent areas).
6069c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
6079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region coveredRegion;
6089c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6099c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6109c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // handle hidden surfaces by setting the visible region to empty
61112cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        if (LIKELY(!(s.flags & ISurfaceComposer::eLayerHidden) && s.alpha)) {
6129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const bool translucent = layer->needsBlending();
61312cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian            const Rect bounds(layer->visibleBounds());
6149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            visibleRegion.set(bounds);
6159c041bbd81789c209e2369ba958306979b67614fMathias Agopian            visibleRegion.andSelf(screenRegion);
6169c041bbd81789c209e2369ba958306979b67614fMathias Agopian            if (!visibleRegion.isEmpty()) {
6179c041bbd81789c209e2369ba958306979b67614fMathias Agopian                // Remove the transparent area from the visible region
6189c041bbd81789c209e2369ba958306979b67614fMathias Agopian                if (translucent) {
6199c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    visibleRegion.subtractSelf(layer->transparentRegionScreen);
6209c041bbd81789c209e2369ba958306979b67614fMathias Agopian                }
6219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6229c041bbd81789c209e2369ba958306979b67614fMathias Agopian                // compute the opaque region
6239c041bbd81789c209e2369ba958306979b67614fMathias Agopian                const int32_t layerOrientation = layer->getOrientation();
6249c041bbd81789c209e2369ba958306979b67614fMathias Agopian                if (s.alpha==255 && !translucent &&
6259c041bbd81789c209e2369ba958306979b67614fMathias Agopian                        ((layerOrientation & Transform::ROT_INVALID) == false)) {
6269c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    // the opaque region is the layer's footprint
6279c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    opaqueRegion = visibleRegion;
6289c041bbd81789c209e2369ba958306979b67614fMathias Agopian                }
6299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
6309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6329c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Clip the covered region to the visible region
6339c041bbd81789c209e2369ba958306979b67614fMathias Agopian        coveredRegion = aboveCoveredLayers.intersect(visibleRegion);
6349c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6359c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Update aboveCoveredLayers for next (lower) layer
6369c041bbd81789c209e2369ba958306979b67614fMathias Agopian        aboveCoveredLayers.orSelf(visibleRegion);
6379c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // subtract the opaque region covered by the layers above us
6399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        visibleRegion.subtractSelf(aboveOpaqueLayers);
6409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // compute this layer's dirty region
6429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (layer->contentDirty) {
6439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // we need to invalidate the whole region
6449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dirty = visibleRegion;
6459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // as well, as the old visible region
6469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dirty.orSelf(layer->visibleRegionScreen);
6479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->contentDirty = false;
6489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
6490aed7e97a3efbeec23a4aa301df03036a67f8ea3Mathias Agopian            /* compute the exposed region:
6509c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   the exposed region consists of two components:
6519c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   1) what's VISIBLE now and was COVERED before
6529c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   2) what's EXPOSED now less what was EXPOSED before
6539c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *
6549c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * note that (1) is conservative, we start with the whole
6559c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * visible region but only keep what used to be covered by
6569c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * something -- which mean it may have been exposed.
6579c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *
6589c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * (2) handles areas that were not covered by anything but got
6599c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * exposed because of a resize.
6600aed7e97a3efbeec23a4aa301df03036a67f8ea3Mathias Agopian             */
6619c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region newExposed = visibleRegion - coveredRegion;
6629c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldVisibleRegion = layer->visibleRegionScreen;
6639c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldCoveredRegion = layer->coveredRegionScreen;
6649c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldExposed = oldVisibleRegion - oldCoveredRegion;
6659c041bbd81789c209e2369ba958306979b67614fMathias Agopian            dirty = (visibleRegion&oldCoveredRegion) | (newExposed-oldExposed);
6669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dirty.subtractSelf(aboveOpaqueLayers);
6689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // accumulate to the screen dirty region
6709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dirtyRegion.orSelf(dirty);
6719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6729c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Update aboveOpaqueLayers for next (lower) layer
6739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        aboveOpaqueLayers.orSelf(opaqueRegion);
674e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
6759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Store the visible region is screen space
6769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->setVisibleRegion(visibleRegion);
6779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->setCoveredRegion(coveredRegion);
6789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
67912cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        // If a secure layer is partially visible, lock-down the screen!
6809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (layer->isSecure() && !visibleRegion.isEmpty()) {
6819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            secureFrameBuffer = true;
6829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
68512cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    // invalidate the areas where a layer was removed
68612cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    dirtyRegion.orSelf(mDirtyRegionRemovedLayer);
68712cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    mDirtyRegionRemovedLayer.clear();
68812cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian
6899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mSecureFrameBuffer = secureFrameBuffer;
6909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    opaqueRegion = aboveOpaqueLayers;
6919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
6929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::commitTransaction()
6959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
6969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDrawingState = mCurrentState;
6979779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian    mResizeTransationPending = false;
6989779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian    mTransactionCV.broadcast();
6999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handlePageFlip()
7029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool visibleRegions = mVisibleRegionsDirty;
704e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    LayerVector& currentLayers(
705e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian            const_cast<LayerVector&>(mDrawingState.layersSortedByZ));
7069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    visibleRegions |= lockPageFlip(currentLayers);
7079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw = graphicPlane(0).displayHardware();
7099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const Region screenRegion(hw.bounds());
7109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (visibleRegions) {
7119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            Region opaqueRegion;
7129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            computeVisibleRegions(currentLayers, mDirtyRegion, opaqueRegion);
713ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian
714ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            /*
715ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian             *  rebuild the visible layer list
716ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian             */
717ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            mVisibleLayersSortedByZ.clear();
718ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            const LayerVector& currentLayers(mDrawingState.layersSortedByZ);
719ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            size_t count = currentLayers.size();
720ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            mVisibleLayersSortedByZ.setCapacity(count);
721ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            for (size_t i=0 ; i<count ; i++) {
722ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian                if (!currentLayers[i]->visibleRegionScreen.isEmpty())
723ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian                    mVisibleLayersSortedByZ.add(currentLayers[i]);
724ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            }
725ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian
7269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mWormholeRegion = screenRegion.subtract(opaqueRegion);
7279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = false;
728fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian            invalidateHwcGeometry();
7299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    unlockPageFlip(currentLayers);
7329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.andSelf(screenRegion);
7339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
735fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopianvoid SurfaceFlinger::invalidateHwcGeometry()
736fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian{
737fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian    mHwWorkListDirty = true;
738fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian}
739fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian
7409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool SurfaceFlinger::lockPageFlip(const LayerVector& currentLayers)
7419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool recomputeVisibleRegions = false;
7439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t count = currentLayers.size();
7441473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBase> const* layers = currentLayers.array();
7459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (size_t i=0 ; i<count ; i++) {
7467623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
7479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->lockPageFlip(recomputeVisibleRegions);
7489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return recomputeVisibleRegions;
7509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::unlockPageFlip(const LayerVector& currentLayers)
7539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(0));
7559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Transform& planeTransform(plane.transform());
7569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t count = currentLayers.size();
7571473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBase> const* layers = currentLayers.array();
7589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (size_t i=0 ; i<count ; i++) {
7597623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
7609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->unlockPageFlip(planeTransform, mDirtyRegion);
7619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
764e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopianvoid SurfaceFlinger::handleWorkList()
765e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian{
766e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    mHwWorkListDirty = false;
767e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    HWComposer& hwc(graphicPlane(0).displayHardware().getHwComposer());
768e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    if (hwc.initCheck() == NO_ERROR) {
769e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        const Vector< sp<LayerBase> >& currentLayers(mVisibleLayersSortedByZ);
770e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        const size_t count = currentLayers.size();
771e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        hwc.createWorkList(count);
772f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        hwc_layer_t* const cur(hwc.getLayers());
773f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        for (size_t i=0 ; cur && i<count ; i++) {
774f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            currentLayers[i]->setGeometry(&cur[i]);
7756a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian            if (mDebugDisableHWC) {
7766a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                cur[i].compositionType = HWC_FRAMEBUFFER;
7776a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                cur[i].flags |= HWC_SKIP_LAYER;
7786a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian            }
779e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        }
780e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    }
781e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian}
7828c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian
7839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleRepaint()
7849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7858c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    // compute the invalid region
7868c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    mInvalidRegion.orSelf(mDirtyRegion);
7879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(mDebugRegion)) {
7899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        debugFlashRegions();
7909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7928c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    // set the frame buffer
7938c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
7948c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    glMatrixMode(GL_MODELVIEW);
7958c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    glLoadIdentity();
7969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t flags = hw.getFlags();
798e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber    if ((flags & DisplayHardware::SWAP_RECTANGLE) ||
799e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber        (flags & DisplayHardware::BUFFER_PRESERVED))
8002e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian    {
801ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // we can redraw only what's dirty, but since SWAP_RECTANGLE only
802ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // takes a rectangle, we must make sure to update that whole
803ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // rectangle in that case
804ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        if (flags & DisplayHardware::SWAP_RECTANGLE) {
8057623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            // TODO: we really should be able to pass a region to
806ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // SWAP_RECTANGLE so that we don't have to redraw all this.
807ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            mDirtyRegion.set(mInvalidRegion.bounds());
808ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        } else {
809ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // in the BUFFER_PRESERVED case, obviously, we can update only
810ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // what's needed and nothing more.
811ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // NOTE: this is NOT a common case, as preserving the backbuffer
812ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // is costly and usually involves copying the whole update back.
813ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        }
8149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
815f2d28b74850ea0869aef2ce0727a6abb7b166a5cMathias Agopian        if (flags & DisplayHardware::PARTIAL_UPDATES) {
816ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // We need to redraw the rectangle that will be updated
8172e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian            // (pushed to the framebuffer).
818f2d28b74850ea0869aef2ce0727a6abb7b166a5cMathias Agopian            // This is needed because PARTIAL_UPDATES only takes one
819ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // rectangle instead of a region (see DisplayHardware::flip())
8209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(mInvalidRegion.bounds());
8219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
822ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // we need to redraw everything (the whole screen)
8239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(hw.bounds());
8249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mInvalidRegion = mDirtyRegion;
8259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // compose all surfaces
8299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    composeSurfaces(mDirtyRegion);
8309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // clear the dirty regions
8329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.clear();
8339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
8349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::composeSurfaces(const Region& dirty)
8369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
8379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(!mWormholeRegion.isEmpty())) {
8389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // should never happen unless the window manager has a bug
8399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // draw something...
8409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        drawWormhole();
8419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
842e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
843e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    status_t err = NO_ERROR;
844ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian    const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
845f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    size_t count = layers.size();
846e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
847e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
848e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    HWComposer& hwc(hw.getHwComposer());
849f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    hwc_layer_t* const cur(hwc.getLayers());
850e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
851f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    LOGE_IF(cur && hwc.getNumLayers() != count,
852f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            "HAL number of layers (%d) doesn't match surfaceflinger (%d)",
853f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            hwc.getNumLayers(), count);
854e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
855f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    // just to be extra-safe, use the smallest count
8560cf2f4354bf8952d822fa4ee9b9588a73999c2f4Erik Gilling    if (hwc.initCheck() == NO_ERROR) {
8570cf2f4354bf8952d822fa4ee9b9588a73999c2f4Erik Gilling        count = count < hwc.getNumLayers() ? count : hwc.getNumLayers();
8580cf2f4354bf8952d822fa4ee9b9588a73999c2f4Erik Gilling    }
859e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
860f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    /*
861f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     *  update the per-frame h/w composer data for each layer
862f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     *  and build the transparent region of the FB
863f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     */
864f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    Region transparent;
865f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    if (cur) {
866f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        for (size_t i=0 ; i<count ; i++) {
867f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            const sp<LayerBase>& layer(layers[i]);
868f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            layer->setPerFrameData(&cur[i]);
8699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
870f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        err = hwc.prepare();
871f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        LOGE_IF(err, "HWComposer::prepare failed (%s)", strerror(-err));
872e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
87345491690d8f21d2648cad9247115720fa90046b4Mathias Agopian        if (err == NO_ERROR) {
87445491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            for (size_t i=0 ; i<count ; i++) {
87545491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                if (cur[i].hints & HWC_HINT_CLEAR_FB) {
87645491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    const sp<LayerBase>& layer(layers[i]);
87745491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    if (!(layer->needsBlending())) {
87845491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                        transparent.orSelf(layer->visibleRegionScreen);
87945491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    }
88045491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                }
88145491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            }
88245491690d8f21d2648cad9247115720fa90046b4Mathias Agopian
88345491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            /*
88445491690d8f21d2648cad9247115720fa90046b4Mathias Agopian             *  clear the area of the FB that need to be transparent
88545491690d8f21d2648cad9247115720fa90046b4Mathias Agopian             */
88645491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            transparent.andSelf(dirty);
88745491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            if (!transparent.isEmpty()) {
88845491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                glClearColor(0,0,0,0);
88945491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                Region::const_iterator it = transparent.begin();
89045491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                Region::const_iterator const end = transparent.end();
89145491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                const int32_t height = hw.getHeight();
89245491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                while (it != end) {
89345491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    const Rect& r(*it++);
89445491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    const GLint sy = height - (r.top + r.height());
89545491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    glScissor(r.left, sy, r.width(), r.height());
89645491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    glClear(GL_COLOR_BUFFER_BIT);
89745491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                }
89845491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            }
899e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        }
900e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    }
901f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian
902f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian
903f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    /*
904f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     * and then, render the layers targeted at the framebuffer
905f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     */
906f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
907f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        if (cur) {
908982f58bdccce7e4e537cffb2410ad78d73398461Antti Hatala            if ((cur[i].compositionType != HWC_FRAMEBUFFER) &&
909982f58bdccce7e4e537cffb2410ad78d73398461Antti Hatala                !(cur[i].flags & HWC_SKIP_LAYER)) {
910f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian                // skip layers handled by the HAL
911f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian                continue;
912f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            }
913f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        }
9146a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian
915f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
916f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        const Region clip(dirty.intersect(layer->visibleRegionScreen));
917f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        if (!clip.isEmpty()) {
918f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            layer->draw(clip);
919f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        }
920f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    }
9219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
9229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::debugFlashRegions()
9249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
925f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
926f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    const uint32_t flags = hw.getFlags();
927f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
928f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    if (!((flags & DisplayHardware::SWAP_RECTANGLE) ||
929f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian            (flags & DisplayHardware::BUFFER_PRESERVED))) {
930f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        const Region repaint((flags & DisplayHardware::PARTIAL_UPDATES) ?
931f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian                mDirtyRegion.bounds() : hw.bounds());
932f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        composeSurfaces(repaint);
933f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    }
934f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
935f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    TextureManager::deactivateTextures();
9362e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian
9379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_BLEND);
9389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
9399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_SCISSOR_TEST);
9409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
941dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    static int toggle = 0;
942dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    toggle = 1 - toggle;
943dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    if (toggle) {
944f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glColor4f(1, 0, 1, 1);
945dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    } else {
946f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glColor4f(1, 1, 0, 1);
947dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    }
9489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9496158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    Region::const_iterator it = mDirtyRegion.begin();
9506158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    Region::const_iterator const end = mDirtyRegion.end();
9516158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    while (it != end) {
9526158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        const Rect& r = *it++;
9539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        GLfloat vertices[][2] = {
9549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.left,  r.top },
9559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.left,  r.bottom },
9569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.right, r.bottom },
9579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.right, r.top }
9589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        };
9599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glVertexPointer(2, GL_FLOAT, 0, vertices);
9609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
9619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
962f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
9638c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    if (mInvalidRegion.isEmpty()) {
9648c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        mDirtyRegion.dump("mDirtyRegion");
9658c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        mInvalidRegion.dump("mInvalidRegion");
9668c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    }
9678c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    hw.flip(mInvalidRegion);
9689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (mDebugRegion > 1)
970f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        usleep(mDebugRegion * 1000);
9719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glEnable(GL_SCISSOR_TEST);
9739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    //mDirtyRegion.dump("mDirtyRegion");
9749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
9759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::drawWormhole() const
9779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
9789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Region region(mWormholeRegion.intersect(mDirtyRegion));
9799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (region.isEmpty())
9809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return;
9819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw(graphicPlane(0).displayHardware());
9839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int32_t width = hw.getWidth();
9849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int32_t height = hw.getHeight();
9859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_BLEND);
9879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
9889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (LIKELY(!mDebugBackground)) {
990f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glClearColor(0,0,0,0);
9916158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator it = region.begin();
9926158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator const end = region.end();
9936158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        while (it != end) {
9946158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian            const Rect& r = *it++;
9959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const GLint sy = height - (r.top + r.height());
9969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glScissor(r.left, sy, r.width(), r.height());
9979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glClear(GL_COLOR_BUFFER_BIT);
9989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
10009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const GLshort vertices[][2] = { { 0, 0 }, { width, 0 },
10019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { width, height }, { 0, height }  };
10029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const GLshort tcoords[][2] = { { 0, 0 }, { 1, 0 },  { 1, 1 }, { 0, 1 } };
10039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glVertexPointer(2, GL_SHORT, 0, vertices);
10049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glTexCoordPointer(2, GL_SHORT, 0, tcoords);
10059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glEnableClientState(GL_TEXTURE_COORD_ARRAY);
1006e20a56d929fc8fedc2b468ea6d1900bd2aa6e81aMichael I. Gold#if defined(GL_OES_EGL_image_external)
1007781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian        if (GLExtensions::getInstance().haveTextureExternal()) {
1008781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian            glDisable(GL_TEXTURE_EXTERNAL_OES);
1009781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian        }
1010f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian#endif
10119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glEnable(GL_TEXTURE_2D);
10129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glBindTexture(GL_TEXTURE_2D, mWormholeTexName);
10139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
10149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glMatrixMode(GL_TEXTURE);
10159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glLoadIdentity();
10169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glScalef(width*(1.0f/32.0f), height*(1.0f/32.0f), 1);
10176158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator it = region.begin();
10186158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator const end = region.end();
10196158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        while (it != end) {
10206158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian            const Rect& r = *it++;
10219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const GLint sy = height - (r.top + r.height());
10229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glScissor(r.left, sy, r.width(), r.height());
10239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
10249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glDisableClientState(GL_TEXTURE_COORD_ARRAY);
102604a709e459db6f15c04b00bcd3b030c90ca52949Mathias Agopian        glLoadIdentity();
102704a709e459db6f15c04b00bcd3b030c90ca52949Mathias Agopian        glMatrixMode(GL_MODELVIEW);
10289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::debugShowFPS() const
10329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int mFrameCount;
10349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int mLastFrameCount = 0;
10359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static nsecs_t mLastFpsTime = 0;
10369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static float mFps = 0;
10379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mFrameCount++;
10389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    nsecs_t now = systemTime();
10399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    nsecs_t diff = now - mLastFpsTime;
10409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (diff > ms2ns(250)) {
10419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mFps =  ((mFrameCount - mLastFrameCount) * float(s2ns(1))) / diff;
10429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mLastFpsTime = now;
10439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mLastFrameCount = mFrameCount;
10449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // XXX: mFPS has the value we want
10469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project }
10479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10481473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::addLayer(const sp<LayerBase>& layer)
10499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
10519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    addLayer_l(layer);
10529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded|eTraversalNeeded);
10539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
10549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10561473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::addLayer_l(const sp<LayerBase>& layer)
10579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10581efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian    ssize_t i = mCurrentState.layersSortedByZ.add(layer);
10599bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian    return (i < 0) ? status_t(i) : status_t(NO_ERROR);
10609bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian}
10619bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1062593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianssize_t SurfaceFlinger::addClientLayer(const sp<Client>& client,
1063593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const sp<LayerBaseClient>& lbc)
10649bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian{
1065593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    Mutex::Autolock _l(mStateLock);
1066593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
1067593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // attach this layer to the client
1068593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    ssize_t name = client->attachLayer(lbc);
1069593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
1070593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // add this layer to the current state list
1071593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    addLayer_l(lbc);
1072593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
1073593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return name;
1074593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian}
1075593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
1076593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::removeLayer(const sp<LayerBase>& layer)
1077593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
1078593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    Mutex::Autolock _l(mStateLock);
1079593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    status_t err = purgatorizeLayer_l(layer);
1080593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (err == NO_ERROR)
1081593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        setTransactionFlags(eTransactionNeeded);
1082593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return err;
10839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10851473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::removeLayer_l(const sp<LayerBase>& layerBase)
10869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10877623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<LayerBaseClient> lbc(layerBase->getLayerBaseClient());
10887623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (lbc != 0) {
1089d35c6667c8233385f31aa203f486b2cb826bf6beMathias Agopian        mLayerMap.removeItem( lbc->getSurfaceBinder() );
10907623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
10919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    ssize_t index = mCurrentState.layersSortedByZ.remove(layerBase);
10929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (index >= 0) {
10931473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        mLayersRemoved = true;
10949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return NO_ERROR;
10959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10962d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    return status_t(index);
10979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10996cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopianstatus_t SurfaceFlinger::purgatorizeLayer_l(const sp<LayerBase>& layerBase)
11006cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian{
1101f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian    // First add the layer to the purgatory list, which makes sure it won't
1102f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian    // go away, then remove it from the main list (through a transaction).
11036cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    ssize_t err = removeLayer_l(layerBase);
1104f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian    if (err >= 0) {
1105f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian        mLayerPurgatory.add(layerBase);
1106f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian    }
11072e4b68d57bb64d7e93139238c5a8be91ff956c2aMathias Agopian
11080c4cec7e4df87181486d280c98fba9c0f4774c37Mathias Agopian    layerBase->onRemoved();
11090c4cec7e4df87181486d280c98fba9c0f4774c37Mathias Agopian
11102d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    // it's possible that we don't find a layer, because it might
11112d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    // have been destroyed already -- this is not technically an error
1112593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // from the user because there is a race between Client::destroySurface(),
1113593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // ~Client() and ~ISurface().
11146cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    return (err == NAME_NOT_FOUND) ? status_t(NO_ERROR) : err;
11156cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian}
11166cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
1117593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::invalidateLayerVisibility(const sp<LayerBase>& layer)
11189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1119593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    layer->forceVisibilityTransaction();
1120593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    setTransactionFlags(eTraversalNeeded);
1121593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return NO_ERROR;
11229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectuint32_t SurfaceFlinger::getTransactionFlags(uint32_t flags)
11259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return android_atomic_and(~flags, &mTransactionFlags) & flags;
11279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1129898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianuint32_t SurfaceFlinger::setTransactionFlags(uint32_t flags)
11309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t old = android_atomic_or(flags, &mTransactionFlags);
11329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if ((old & flags)==0) { // wake the server up
1133898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        signalEvent();
11349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return old;
11369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::openGlobalTransaction()
11399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    android_atomic_inc(&mTransactionCount);
11419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::closeGlobalTransaction()
11449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (android_atomic_dec(&mTransactionCount) == 1) {
11469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        signalEvent();
11479779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian
1148e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber        // if there is a transaction with a resize, wait for it to
11499779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        // take effect before returning.
11509779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        Mutex::Autolock _l(mStateLock);
11519779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        while (mResizeTransationPending) {
115298a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian            status_t err = mTransactionCV.waitRelative(mStateLock, s2ns(5));
115398a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian            if (CC_UNLIKELY(err != NO_ERROR)) {
115498a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                // just in case something goes wrong in SF, return to the
115598a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                // called after a few seconds.
115698a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                LOGW_IF(err == TIMED_OUT, "closeGlobalTransaction timed out!");
115798a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                mResizeTransationPending = false;
115898a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                break;
115998a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian            }
11609779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        }
11619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::freezeDisplay(DisplayID dpy, uint32_t flags)
11659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
11679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
11689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
11709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mCurrentState.freezeDisplay = 1;
11719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded);
11729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // flags is intended to communicate some sort of animation behavior
1174ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian    // (for instance fading)
11759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
11769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::unfreezeDisplay(DisplayID dpy, uint32_t flags)
11799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
11819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
11829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
11849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mCurrentState.freezeDisplay = 0;
11859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded);
11869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // flags is intended to communicate some sort of animation behavior
1188ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian    // (for instance fading)
11899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
11909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1192e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huberint SurfaceFlinger::setOrientation(DisplayID dpy,
1193eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian        int orientation, uint32_t flags)
11949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
11969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
11979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
11999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (mCurrentState.orientation != orientation) {
12009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (uint32_t(orientation)<=eOrientation270 || orientation==42) {
1201eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian            mCurrentState.orientationType = flags;
12029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mCurrentState.orientation = orientation;
12039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            setTransactionFlags(eTransactionNeeded);
12049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mTransactionCV.wait(mStateLock);
12059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
12069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            orientation = BAD_VALUE;
12079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
12089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return orientation;
12109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1212593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopiansp<ISurface> SurfaceFlinger::createSurface(const sp<Client>& client, int pid,
1213770492cb2b19f6a36ad748cd05fbedfbb9a67dfaMathias Agopian        const String8& name, ISurfaceComposerClient::surface_data_t* params,
12149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        DisplayID d, uint32_t w, uint32_t h, PixelFormat format,
12159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        uint32_t flags)
12169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
12171473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBaseClient> layer;
12189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    sp<LayerBaseClient::Surface> surfaceHandle;
12194d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian
12204d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian    if (int32_t(w|h) < 0) {
12214d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian        LOGE("createSurface() failed, w or h is negative (w=%d, h=%d)",
12224d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian                int(w), int(h));
12234d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian        return surfaceHandle;
12244d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian    }
1225e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
12269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    //LOGD("createSurface for pid %d (%d x %d)", pid, w, h);
12277623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<Layer> normalLayer;
12289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (flags & eFXSurfaceMask) {
12299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceNormal:
1230d2112306330ce0c162bee4b864991962ca2b655aMathias Agopian            normalLayer = createNormalSurface(client, d, w, h, flags, format);
1231d2112306330ce0c162bee4b864991962ca2b655aMathias Agopian            layer = normalLayer;
12329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
12339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceBlur:
1234c3802d22f11b4e513ba776277d18f315c5c769f7Mathias Agopian            // for now we treat Blur as Dim, until we can implement it
1235c3802d22f11b4e513ba776277d18f315c5c769f7Mathias Agopian            // efficiently.
12369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceDim:
1237593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            layer = createDimSurface(client, d, w, h, flags);
12389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
12399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12411473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    if (layer != 0) {
1242593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        layer->initStates(w, h, flags);
12435d26c1e38dabb3ad8b4b6e1000375f3b1a6b7693Mathias Agopian        layer->setName(name);
1244593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        ssize_t token = addClientLayer(client, layer);
12457623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
12469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        surfaceHandle = layer->getSurface();
1247e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber        if (surfaceHandle != 0) {
1248593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            params->token = token;
124918b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->identity = surfaceHandle->getIdentity();
125018b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->width = w;
125118b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->height = h;
125218b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->format = format;
12537623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            if (normalLayer != 0) {
12547623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian                Mutex::Autolock _l(mStateLock);
12557623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian                mLayerMap.add(surfaceHandle->asBinder(), normalLayer);
12567623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            }
125718b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian        }
12587623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
1259593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        setTransactionFlags(eTransactionNeeded);
12609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return surfaceHandle;
12639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12657623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<Layer> SurfaceFlinger::createNormalSurface(
12666edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian        const sp<Client>& client, DisplayID display,
1267593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        uint32_t w, uint32_t h, uint32_t flags,
126818b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian        PixelFormat& format)
12699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
12709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize the surfaces
12719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (format) { // TODO: take h/w into account
12729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_TRANSPARENT:
12739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_TRANSLUCENT:
12749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        format = PIXEL_FORMAT_RGBA_8888;
12759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
12769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_OPAQUE:
12774cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#ifdef NO_RGBX_8888
12784cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian        format = PIXEL_FORMAT_RGB_565;
12794cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#else
128059962ce3b0d8b7efec2840accca8fb7a6066f2bdMathias Agopian        format = PIXEL_FORMAT_RGBX_8888;
12814cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#endif
12829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
12839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12854cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#ifdef NO_RGBX_8888
12864cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian    if (format == PIXEL_FORMAT_RGBX_8888)
12874cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian        format = PIXEL_FORMAT_RGBA_8888;
12884cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#endif
12894cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian
1290593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<Layer> layer = new Layer(this, display, client);
12916edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian    status_t err = layer->setBuffers(w, h, format, flags);
1292593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (LIKELY(err != NO_ERROR)) {
12939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        LOGE("createNormalSurfaceLocked() failed (%s)", strerror(-err));
12941473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        layer.clear();
12959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return layer;
12979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12997623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<LayerDim> SurfaceFlinger::createDimSurface(
13006edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian        const sp<Client>& client, DisplayID display,
1301593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        uint32_t w, uint32_t h, uint32_t flags)
13029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1303593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<LayerDim> layer = new LayerDim(this, display, client);
13049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    layer->initStates(w, h, flags);
13059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return layer;
13069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1308593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::removeSurface(const sp<Client>& client, SurfaceID sid)
13096cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian{
13106cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    /*
13116cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     * called by the window manager, when a surface should be marked for
13126cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     * destruction.
1313e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber     *
1314a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * The surface is removed from the current and drawing lists, but placed
1315a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * in the purgatory queue, so it's not destroyed right-away (we need
1316a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * to wait for all client's references to go away first).
13176cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     */
13186cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
1319248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian    status_t err = NAME_NOT_FOUND;
1320a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian    Mutex::Autolock _l(mStateLock);
1321593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<LayerBaseClient> layer = client->getLayerUser(sid);
1322248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian    if (layer != 0) {
1323248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        err = purgatorizeLayer_l(layer);
1324248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        if (err == NO_ERROR) {
1325248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian            setTransactionFlags(eTransactionNeeded);
1326248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        }
13276cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    }
13286cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    return err;
13296cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian}
13306cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
13316cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopianstatus_t SurfaceFlinger::destroySurface(const sp<LayerBaseClient>& layer)
13329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1333359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian    // called by ~ISurface() when all references are gone
1334e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
13356ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    class MessageDestroySurface : public MessageBase {
1336a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        SurfaceFlinger* flinger;
13376ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        sp<LayerBaseClient> layer;
13386ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    public:
1339a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        MessageDestroySurface(
1340a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                SurfaceFlinger* flinger, const sp<LayerBaseClient>& layer)
1341a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            : flinger(flinger), layer(layer) { }
13426ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        virtual bool handler() {
1343c8fb5b1979da4829e1486e6a1008c06c979b94b0Mathias Agopian            sp<LayerBaseClient> l(layer);
1344c8fb5b1979da4829e1486e6a1008c06c979b94b0Mathias Agopian            layer.clear(); // clear it outside of the lock;
13456ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
1346359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian            /*
1347e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber             * remove the layer from the current list -- chances are that it's
1348e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber             * not in the list anyway, because it should have been removed
1349e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber             * already upon request of the client (eg: window manager).
1350359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             * However, a buggy client could have not done that.
1351359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             * Since we know we don't have any more clients, we don't need
1352359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             * to use the purgatory.
1353359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             */
1354c8fb5b1979da4829e1486e6a1008c06c979b94b0Mathias Agopian            status_t err = flinger->removeLayer_l(l);
1355f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian            if (err == NAME_NOT_FOUND) {
1356f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian                // The surface wasn't in the current list, which means it was
1357f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian                // removed already, which means it is in the purgatory,
1358f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian                // and need to be removed from there.
1359f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian                // This needs to happen from the main thread since its dtor
1360f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian                // must run from there (b/c of OpenGL ES). Additionally, we
1361f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian                // can't really acquire our internal lock from
1362f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian                // destroySurface() -- see postMessage() below.
1363f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian                ssize_t idx = flinger->mLayerPurgatory.remove(l);
1364f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian                LOGE_IF(idx < 0,
1365f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian                        "layer=%p is not in the purgatory list", l.get());
1366f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian            }
1367f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian
13682e4b68d57bb64d7e93139238c5a8be91ff956c2aMathias Agopian            LOGE_IF(err<0 && err != NAME_NOT_FOUND,
13692e4b68d57bb64d7e93139238c5a8be91ff956c2aMathias Agopian                    "error removing layer=%p (%s)", l.get(), strerror(-err));
13706ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            return true;
13716ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        }
13726ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    };
13732d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian
1374898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    postMessageAsync( new MessageDestroySurface(this, layer) );
13759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
13769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::setClientState(
1379593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const sp<Client>& client,
13809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int32_t count,
13819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const layer_state_t* states)
13829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
13839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
13849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t flags = 0;
13859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (int i=0 ; i<count ; i++) {
1386593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const layer_state_t& s(states[i]);
1387593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        sp<LayerBaseClient> layer(client->getLayerUser(s.surface));
13881473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        if (layer != 0) {
13899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const uint32_t what = s.what;
13909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & ePositionChanged) {
13919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setPosition(s.x, s.y))
13929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
13939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
13949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eLayerChanged) {
13951efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian                ssize_t idx = mCurrentState.layersSortedByZ.indexOf(layer);
13969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setLayer(s.z)) {
13971efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian                    mCurrentState.layersSortedByZ.removeAt(idx);
13981efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian                    mCurrentState.layersSortedByZ.add(layer);
13999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // we need traversal (state changed)
14009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // AND transaction (list changed)
14019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTransactionNeeded|eTraversalNeeded;
14029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
14039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eSizeChanged) {
14059779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                if (layer->setSize(s.w, s.h)) {
14069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14079779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                    mResizeTransationPending = true;
14089779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                }
14099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eAlphaChanged) {
14119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setAlpha(uint8_t(255.0f*s.alpha+0.5f)))
14129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eMatrixChanged) {
14159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setMatrix(s.matrix))
14169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eTransparentRegionChanged) {
14199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setTransparentRegionHint(s.transparentRegion))
14209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eVisibilityChanged) {
14239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setFlags(s.flags, s.mask))
14249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
14279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
14289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (flags) {
14299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setTransactionFlags(flags);
14309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
14319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
14329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
14349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::screenReleased(int dpy)
14359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
14369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // this may be called by a signal handler, we can't do too much in here
14379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    android_atomic_or(eConsoleReleased, &mConsoleSignals);
14389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    signalEvent();
14399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
14419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::screenAcquired(int dpy)
14429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
14439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // this may be called by a signal handler, we can't do too much in here
14449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    android_atomic_or(eConsoleAcquired, &mConsoleSignals);
14459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    signalEvent();
14469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
14489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::dump(int fd, const Vector<String16>& args)
14499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
145094720d7ec40d2b5ee200d95a46588cdb0fb30e66Erik Gilling    const size_t SIZE = 4096;
14519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    char buffer[SIZE];
14529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    String8 result;
1453151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian    if (!mDump.checkCalling()) {
14549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        snprintf(buffer, SIZE, "Permission Denial: "
14559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                "can't dump SurfaceFlinger from pid=%d, uid=%d\n",
14569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                IPCThreadState::self()->getCallingPid(),
14579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                IPCThreadState::self()->getCallingUid());
14589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        result.append(buffer);
14599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
1460a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1461a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // figure out if we're stuck somewhere
1462a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t now = systemTime();
1463a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t inSwapBuffers(mDebugInSwapBuffers);
1464a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t inTransaction(mDebugInTransaction);
1465a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        nsecs_t inSwapBuffersDuration = (inSwapBuffers) ? now-inSwapBuffers : 0;
1466a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        nsecs_t inTransactionDuration = (inTransaction) ? now-inTransaction : 0;
1467a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1468a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // Try to get the main lock, but don't insist if we can't
1469a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // (this would indicate SF is stuck, but we want to be able to
1470a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // print something in dumpsys).
1471a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        int retry = 3;
1472a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        while (mStateLock.tryLock()<0 && --retry>=0) {
1473a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            usleep(1000000);
1474a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
1475a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const bool locked(retry >= 0);
1476a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (!locked) {
1477e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber            snprintf(buffer, SIZE,
1478a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    "SurfaceFlinger appears to be unresponsive, "
1479a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    "dumping anyways (no locks held)\n");
1480a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1481a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
1482a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
148306a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
148406a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump the visible layer list
148506a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
14869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const LayerVector& currentLayers = mCurrentState.layersSortedByZ;
14879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const size_t count = currentLayers.size();
148806a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        snprintf(buffer, SIZE, "Visible layers (count = %d)\n", count);
148906a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        result.append(buffer);
14909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (size_t i=0 ; i<count ; i++) {
14919bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            const sp<LayerBase>& layer(currentLayers[i]);
14929bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            layer->dump(result, buffer, SIZE);
14939bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            const Layer::State& s(layer->drawingState());
14949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            s.transparentRegion.dump(result, "transparentRegion");
14959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->transparentRegionScreen.dump(result, "transparentRegionScreen");
14969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->visibleRegionScreen.dump(result, "visibleRegionScreen");
14979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
14989bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
149906a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
150006a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump the layers in the purgatory
150106a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
150206a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian
150306a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        const size_t purgatorySize =  mLayerPurgatory.size();
150406a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        snprintf(buffer, SIZE, "Purgatory state (%d entries)\n", purgatorySize);
150506a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        result.append(buffer);
150606a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        for (size_t i=0 ; i<purgatorySize ; i++) {
150706a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian            const sp<LayerBase>& layer(mLayerPurgatory.itemAt(i));
150806a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian            layer->shortDump(result, buffer, SIZE);
150906a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        }
151006a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian
151106a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
151206a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump SurfaceFlinger global state
151306a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
151406a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian
151506a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        snprintf(buffer, SIZE, "SurfaceFlinger global state\n");
151606a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        result.append(buffer);
15179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mWormholeRegion.dump(result, "WormholeRegion");
15189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw(graphicPlane(0).displayHardware());
15199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        snprintf(buffer, SIZE,
15209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                "  display frozen: %s, freezeCount=%d, orientation=%d, canDraw=%d\n",
15219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeDisplay?"yes":"no", mFreezeCount,
15229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mCurrentState.orientation, hw.canDraw());
15239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        result.append(buffer);
1524a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        snprintf(buffer, SIZE,
1525a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                "  last eglSwapBuffers() time: %f us\n"
1526a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                "  last transaction time     : %f us\n",
1527a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                mLastSwapBufferTime/1000.0, mLastTransactionTime/1000.0);
1528a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        result.append(buffer);
15299bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1530a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (inSwapBuffersDuration || !locked) {
1531a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            snprintf(buffer, SIZE, "  eglSwapBuffers time: %f us\n",
1532a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    inSwapBuffersDuration/1000.0);
1533a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1534a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
15359bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1536a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (inTransactionDuration || !locked) {
1537a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            snprintf(buffer, SIZE, "  transaction time: %f us\n",
1538a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    inTransactionDuration/1000.0);
1539a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1540a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
15419bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
154206a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
154306a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump HWComposer state
154406a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
15456a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        HWComposer& hwc(hw.getHwComposer());
15466a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        snprintf(buffer, SIZE, "  h/w composer %s and %s\n",
15476a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                hwc.initCheck()==NO_ERROR ? "present" : "not present",
15486a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                mDebugDisableHWC ? "disabled" : "enabled");
15496a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        result.append(buffer);
155090da4dd2a20c11fa2f9b860905c867a64b2b299eMathias Agopian        hwc.dump(result, buffer, SIZE);
15516a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian
155206a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
155306a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump gralloc state
155406a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
15556950e428feaccc8164b989ef64e771a99948797aMathias Agopian        const GraphicBufferAllocator& alloc(GraphicBufferAllocator::get());
15561473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        alloc.dump(result);
155794720d7ec40d2b5ee200d95a46588cdb0fb30e66Erik Gilling        hw.dump(result);
1558a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1559a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (locked) {
1560a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            mStateLock.unlock();
1561a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
15629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
15639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    write(fd, result.string(), result.size());
15649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
15659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
15669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
15679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::onTransact(
15689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
15699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
15709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (code) {
15719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case CREATE_CONNECTION:
15729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case OPEN_GLOBAL_TRANSACTION:
15739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case CLOSE_GLOBAL_TRANSACTION:
15749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case SET_ORIENTATION:
15759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case FREEZE_DISPLAY:
15769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case UNFREEZE_DISPLAY:
15779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case BOOT_FINISHED:
1578aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        case TURN_ELECTRON_BEAM_OFF:
15792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        case TURN_ELECTRON_BEAM_ON:
15809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        {
15819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // codes that require permission check
15829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            IPCThreadState* ipc = IPCThreadState::self();
15839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int pid = ipc->getCallingPid();
1584627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian            const int uid = ipc->getCallingUid();
1585151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            if ((uid != AID_GRAPHICS) && !mAccessSurfaceFlinger.check(pid, uid)) {
1586151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                LOGE("Permission Denial: "
1587151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                        "can't access SurfaceFlinger pid=%d, uid=%d", pid, uid);
1588151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                return PERMISSION_DENIED;
15899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
1590ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            break;
1591ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
1592ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        case CAPTURE_SCREEN:
1593ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        {
1594ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            // codes that require permission check
1595ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            IPCThreadState* ipc = IPCThreadState::self();
1596ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            const int pid = ipc->getCallingPid();
1597ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            const int uid = ipc->getCallingUid();
1598ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            if ((uid != AID_GRAPHICS) && !mReadFramebuffer.check(pid, uid)) {
1599ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                LOGE("Permission Denial: "
1600ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                        "can't read framebuffer pid=%d, uid=%d", pid, uid);
1601ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                return PERMISSION_DENIED;
1602ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            }
1603ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            break;
16049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
16059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1606ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
16079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    status_t err = BnSurfaceComposer::onTransact(code, data, reply, flags);
16089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (err == UNKNOWN_TRANSACTION || err == PERMISSION_DENIED) {
16098c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        CHECK_INTERFACE(ISurfaceComposer, data, reply);
1610151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian        if (UNLIKELY(!mHardwareTest.checkCalling())) {
1611151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            IPCThreadState* ipc = IPCThreadState::self();
1612151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            const int pid = ipc->getCallingPid();
1613151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            const int uid = ipc->getCallingUid();
1614151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            LOGE("Permission Denial: "
1615151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                    "can't access SurfaceFlinger pid=%d, uid=%d", pid, uid);
16169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return PERMISSION_DENIED;
16179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
16189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int n;
16199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (code) {
162017f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian            case 1000: // SHOW_CPU, NOT SUPPORTED ANYMORE
162104262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            case 1001: // SHOW_FPS, NOT SUPPORTED ANYMORE
16229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1002:  // SHOW_UPDATES
16249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                n = data.readInt32();
16259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDebugRegion = n ? n : (mDebugRegion ? 0 : 1);
16269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1003:  // SHOW_BACKGROUND
16289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                n = data.readInt32();
16299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDebugBackground = n ? 1 : 0;
16309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16316a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian            case 1008:  // toggle use of hw composer
16326a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                n = data.readInt32();
16336a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                mDebugDisableHWC = n ? 1 : 0;
1634fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian                invalidateHwcGeometry();
16356a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                // fall-through...
16369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1004:{ // repaint everything
16379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                Mutex::Autolock _l(mStateLock);
16389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                const DisplayHardware& hw(graphicPlane(0).displayHardware());
16399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDirtyRegion.set(hw.bounds()); // careful that's not thread-safe
16409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                signalEvent();
16419779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                return NO_ERROR;
16429779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian            }
16439779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian            case 1005:{ // force transaction
16449779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                setTransactionFlags(eTransactionNeeded|eTraversalNeeded);
16459779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                return NO_ERROR;
16469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
164704262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            case 1006:{ // enable/disable GraphicLog
164804262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                int enabled = data.readInt32();
164904262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                GraphicLog::getInstance().setEnabled(enabled);
165004262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                return NO_ERROR;
165104262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            }
16529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1007: // set mFreezeCount
16539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeCount = data.readInt32();
16540e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                mFreezeDisplayTime = 0;
16559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1010:  // interrogate.
165717f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian                reply->writeInt32(0);
16589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(0);
16599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(mDebugRegion);
16609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(mDebugBackground);
16619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1013: {
16639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                Mutex::Autolock _l(mStateLock);
16649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                const DisplayHardware& hw(graphicPlane(0).displayHardware());
16659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(hw.getPageFlipCount());
16669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
16679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return NO_ERROR;
16689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
16699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
16709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return err;
16719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
16729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1673aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian// ---------------------------------------------------------------------------
1674aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
16752d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::renderScreenToTextureLocked(DisplayID dpy,
16762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        GLuint* textureName, GLfloat* uOut, GLfloat* vOut)
1677aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian{
1678aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
1679aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        return INVALID_OPERATION;
1680aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1681aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // get screen geometry
1682aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const DisplayHardware& hw(graphicPlane(dpy).displayHardware());
1683aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const uint32_t hw_w = hw.getWidth();
1684aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const uint32_t hw_h = hw.getHeight();
1685aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLfloat u = 1;
1686aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLfloat v = 1;
1687aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1688aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // make sure to clear all GL error flags
1689aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    while ( glGetError() != GL_NO_ERROR ) ;
1690aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1691aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // create a FBO
1692aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLuint name, tname;
1693aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glGenTextures(1, &tname);
1694aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glBindTexture(GL_TEXTURE_2D, tname);
16952d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB,
16962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            hw_w, hw_h, 0, GL_RGB, GL_UNSIGNED_BYTE, 0);
1697aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (glGetError() != GL_NO_ERROR) {
1698a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian        while ( glGetError() != GL_NO_ERROR ) ;
1699aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        GLint tw = (2 << (31 - clz(hw_w)));
1700aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        GLint th = (2 << (31 - clz(hw_h)));
17012d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB,
17022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian                tw, th, 0, GL_RGB, GL_UNSIGNED_BYTE, 0);
1703aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        u = GLfloat(hw_w) / tw;
1704aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        v = GLfloat(hw_h) / th;
1705aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
1706aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glGenFramebuffersOES(1, &name);
1707aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, name);
17082d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glFramebufferTexture2DOES(GL_FRAMEBUFFER_OES,
17092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            GL_COLOR_ATTACHMENT0_OES, GL_TEXTURE_2D, tname, 0);
1710aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // redraw the screen entirely...
17122d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glClearColor(0,0,0,1);
17132d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glClear(GL_COLOR_BUFFER_BIT);
17142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
17152d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const size_t count = layers.size();
17162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (size_t i=0 ; i<count ; ++i) {
17172d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
17182d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        layer->drawForSreenShot();
17192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
1720aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17212d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // back to main framebuffer
17222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
17232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_SCISSOR_TEST);
17242d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDeleteFramebuffersOES(1, &name);
1725aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    *textureName = tname;
17272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    *uOut = u;
17282d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    *vOut = v;
17292d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
17302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
1731aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17322d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian// ---------------------------------------------------------------------------
1733aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::electronBeamOffAnimationImplLocked()
17352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
17362d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    status_t result = PERMISSION_DENIED;
1737aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
17392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return INVALID_OPERATION;
1740aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // get screen geometry
17422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
17432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_w = hw.getWidth();
17442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_h = hw.getHeight();
17452d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const Region screenBounds(hw.bounds());
1746aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17472d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat u, v;
17482d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLuint tname;
17492d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    result = renderScreenToTextureLocked(0, &tname, &u, &v);
17502d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (result != NO_ERROR) {
17512d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return result;
17522d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
1753aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17542d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat vtx[8];
17552d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const GLfloat texCoords[4][2] = { {0,v}, {0,0}, {u,0}, {u,v} };
17562d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_TEXTURE_2D);
17572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindTexture(GL_TEXTURE_2D, tname);
17582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
17592d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
17602d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
17612d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexCoordPointer(2, GL_FLOAT, 0, texCoords);
17622d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnableClientState(GL_TEXTURE_COORD_ARRAY);
17632d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glVertexPointer(2, GL_FLOAT, 0, vtx);
17642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
17652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class s_curve_interpolator {
17662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float nbFrames, s, v;
17672d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
17682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        s_curve_interpolator(int nbFrames, float s)
17692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : nbFrames(1.0f / (nbFrames-1)), s(s),
17702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian          v(1.0f + expf(-s + 0.5f*s)) {
17712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
17722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float operator()(int f) {
17732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const float x = f * nbFrames;
17742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return ((1.0f/(1.0f + expf(-x*s + 0.5f*s))) - 0.5f) * v + 0.5f;
17752d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
17762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
17772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
17782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class v_stretch {
17792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
17802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
17812d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        v_stretch(uint32_t hw_w, uint32_t hw_h)
17822d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
17832d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
17842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
17852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w + (hw_w * v);
17862d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = hw_h - (hw_h * v);
17872d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
17882d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
17892d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
17902d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
17912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
17922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
17932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
17942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
17952d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
17962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class h_stretch {
17972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
17982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
17992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        h_stretch(uint32_t hw_w, uint32_t hw_h)
18002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
1801aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
18022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
18032d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w - (hw_w * v);
18042d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = 1.0f;
18052d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
18062d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
18072d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
18082d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
18092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
18102d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
18112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
18122d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
18132d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // the full animation is 24 frames
18152d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const int nbFrames = 12;
18162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itr(nbFrames, 7.5f);
18172d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itg(nbFrames, 8.0f);
18182d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itb(nbFrames, 8.5f);
18192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18202d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    v_stretch vverts(hw_w, hw_h);
18212d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_BLEND);
18222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBlendFunc(GL_ONE, GL_ONE);
18232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=0 ; i<nbFrames ; i++) {
18242d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float x, y, w, h;
18252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vr = itr(i);
18262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vg = itg(i);
18272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vb = itb(i);
18282d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18292d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // clear screen
18302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(1,1,1,1);
18312d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
18322d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glEnable(GL_TEXTURE_2D);
1833aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
18342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the red plane
18352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vr);
18362d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(1,0,0,1);
18372d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
18382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the green plane
18402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vg);
18412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,1,0,1);
18422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
18432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the blue plane
18452d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vb);
18462d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,0,1,1);
18472d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
18482d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18492d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the white highlight (we use the last vertices)
1850aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glDisable(GL_TEXTURE_2D);
1851aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glColorMask(1,1,1,1);
18522d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColor4f(vg, vg, vg, 1);
18532d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
18542d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
18552d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
18562d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    h_stretch hverts(hw_w, hw_h);
18582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_BLEND);
18592d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_TEXTURE_2D);
18602d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
18612d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=0 ; i<nbFrames ; i++) {
18622d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float v = itg(i);
18632d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hverts(vtx, v);
18642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
18652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColor4f(1-v, 1-v, 1-v, 1);
18662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
18672d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
18682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
18692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
18712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_SCISSOR_TEST);
18722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisableClientState(GL_TEXTURE_COORD_ARRAY);
18732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDeleteTextures(1, &tname);
18742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
18752d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
18762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::electronBeamOnAnimationImplLocked()
18782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
18792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    status_t result = PERMISSION_DENIED;
18802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18812d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
18822d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return INVALID_OPERATION;
18832d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // get screen geometry
18862d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
18872d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_w = hw.getWidth();
18882d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_h = hw.getHeight();
18892d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const Region screenBounds(hw.bounds());
18902d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat u, v;
18922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLuint tname;
18932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    result = renderScreenToTextureLocked(0, &tname, &u, &v);
18942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (result != NO_ERROR) {
18952d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return result;
18962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
18972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // back to main framebuffer
18992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
19002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_SCISSOR_TEST);
19012d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat vtx[8];
19032d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const GLfloat texCoords[4][2] = { {0,v}, {0,0}, {u,0}, {u,v} };
19042d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_TEXTURE_2D);
19052d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindTexture(GL_TEXTURE_2D, tname);
19062d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE);
19072d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
19082d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
19092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexCoordPointer(2, GL_FLOAT, 0, texCoords);
19102d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnableClientState(GL_TEXTURE_COORD_ARRAY);
19112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glVertexPointer(2, GL_FLOAT, 0, vtx);
19122d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19132d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class s_curve_interpolator {
19142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float nbFrames, s, v;
19152d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
19162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        s_curve_interpolator(int nbFrames, float s)
19172d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : nbFrames(1.0f / (nbFrames-1)), s(s),
19182d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian          v(1.0f + expf(-s + 0.5f*s)) {
19192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19202d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float operator()(int f) {
19212d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const float x = f * nbFrames;
19222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return ((1.0f/(1.0f + expf(-x*s + 0.5f*s))) - 0.5f) * v + 0.5f;
19232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19242d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
19252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class v_stretch {
19272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
19282d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
19292d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        v_stretch(uint32_t hw_w, uint32_t hw_h)
19302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
1931aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
19322d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
19332d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w + (hw_w * v);
19342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = hw_h - (hw_h * v);
19352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
19362d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
19372d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
19382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
19392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
19402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
19412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
19432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class h_stretch {
19452d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
19462d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
19472d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        h_stretch(uint32_t hw_w, uint32_t hw_h)
19482d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
19492d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19502d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
19512d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w - (hw_w * v);
19522d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = 1.0f;
19532d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
19542d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
19552d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
19562d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
19572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
19582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
19592d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19602d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
19612d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
1962dfa08fbfd8a5ebd013c122e391282b7f21f6342fMathias Agopian    // the full animation is 12 frames
1963dfa08fbfd8a5ebd013c122e391282b7f21f6342fMathias Agopian    int nbFrames = 8;
19642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itr(nbFrames, 7.5f);
19652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itg(nbFrames, 8.0f);
19662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itb(nbFrames, 8.5f);
19672d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    h_stretch hverts(hw_w, hw_h);
19692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_BLEND);
19702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_TEXTURE_2D);
19712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
19722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=nbFrames-1 ; i>=0 ; i--) {
19732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float v = itg(i);
19742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hverts(vtx, v);
19752d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
19762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColor4f(1-v, 1-v, 1-v, 1);
19772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
19782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
19792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
19802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
1981dfa08fbfd8a5ebd013c122e391282b7f21f6342fMathias Agopian    nbFrames = 4;
19822d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    v_stretch vverts(hw_w, hw_h);
19832d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_BLEND);
19842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBlendFunc(GL_ONE, GL_ONE);
19852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=nbFrames-1 ; i>=0 ; i--) {
19862d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float x, y, w, h;
19872d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vr = itr(i);
19882d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vg = itg(i);
19892d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vb = itb(i);
1990aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
19912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // clear screen
1992aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glColorMask(1,1,1,1);
19932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
19942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glEnable(GL_TEXTURE_2D);
19952d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the red plane
19972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vr);
19982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(1,0,0,1);
19992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
20002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20012d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the green plane
20022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vg);
20032d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,1,0,1);
20042d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
20052d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20062d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the blue plane
20072d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vb);
20082d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,0,1,1);
20092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
20102d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
2012aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
2013aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
20142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
20152d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_SCISSOR_TEST);
20162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisableClientState(GL_TEXTURE_COORD_ARRAY);
2017aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glDeleteTextures(1, &tname);
2018aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
20192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
20202d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
20212d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian// ---------------------------------------------------------------------------
20232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
2024d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOffImplLocked(int32_t mode)
20252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
20262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    DisplayHardware& hw(graphicPlane(0).editDisplayHardware());
20272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (!hw.canDraw()) {
20282d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // we're already off
20292d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return NO_ERROR;
20302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
2031d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    if (mode & ISurfaceComposer::eElectronBeamAnimationOff) {
2032d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        electronBeamOffAnimationImplLocked();
2033d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    }
2034d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian
2035d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    // always clear the whole screen at the end of the animation
2036d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glClearColor(0,0,0,1);
2037d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glDisable(GL_SCISSOR_TEST);
2038d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glClear(GL_COLOR_BUFFER_BIT);
2039d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glEnable(GL_SCISSOR_TEST);
2040d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    hw.flip( Region(hw.bounds()) );
2041d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian
2042a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    hw.setCanDraw(false);
2043a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    return NO_ERROR;
2044aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
2045aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
2046aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOff(int32_t mode)
2047aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian{
2048aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    class MessageTurnElectronBeamOff : public MessageBase {
2049aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        SurfaceFlinger* flinger;
2050d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        int32_t mode;
2051aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        status_t result;
2052aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    public:
2053d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        MessageTurnElectronBeamOff(SurfaceFlinger* flinger, int32_t mode)
2054d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            : flinger(flinger), mode(mode), result(PERMISSION_DENIED) {
2055aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
2056aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        status_t getResult() const {
2057aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            return result;
2058aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
2059aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        virtual bool handler() {
2060aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
2061d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            result = flinger->turnElectronBeamOffImplLocked(mode);
2062aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            return true;
2063aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
2064aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    };
2065aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
2066d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    sp<MessageBase> msg = new MessageTurnElectronBeamOff(this, mode);
2067aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    status_t res = postMessageSync(msg);
2068aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (res == NO_ERROR) {
2069aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        res = static_cast<MessageTurnElectronBeamOff*>( msg.get() )->getResult();
20702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // work-around: when the power-manager calls us we activate the
20722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // animation. eventually, the "on" animation will be called
20732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // by the power-manager itself
2074d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        mElectronBeamAnimationMode = mode;
2075aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
2076aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    return res;
2077aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
2078aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
20799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
20809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2081d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOnImplLocked(int32_t mode)
20822d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
20832d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    DisplayHardware& hw(graphicPlane(0).editDisplayHardware());
20842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (hw.canDraw()) {
20852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // we're already on
20862d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return NO_ERROR;
20872d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
2088d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    if (mode & ISurfaceComposer::eElectronBeamAnimationOn) {
2089d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        electronBeamOnAnimationImplLocked();
2090d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    }
2091a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    hw.setCanDraw(true);
20928b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian
20938b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian    // make sure to redraw the whole screen when the animation is done
20948b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian    mDirtyRegion.set(hw.bounds());
20958b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian    signalEvent();
20968b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian
2097a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    return NO_ERROR;
20982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
20992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
21002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOn(int32_t mode)
21012d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
21022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class MessageTurnElectronBeamOn : public MessageBase {
21032d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        SurfaceFlinger* flinger;
2104d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        int32_t mode;
21052d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        status_t result;
21062d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
2107d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        MessageTurnElectronBeamOn(SurfaceFlinger* flinger, int32_t mode)
2108d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            : flinger(flinger), mode(mode), result(PERMISSION_DENIED) {
21092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
21102d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        status_t getResult() const {
21112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return result;
21122d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
21132d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        virtual bool handler() {
21142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
2115d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            result = flinger->turnElectronBeamOnImplLocked(mode);
21162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return true;
21172d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
21182d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
21192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
2120d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    postMessageAsync( new MessageTurnElectronBeamOn(this, mode) );
21212d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
21222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
21232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
21242d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian// ---------------------------------------------------------------------------
21252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
212638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopianstatus_t SurfaceFlinger::captureScreenImplLocked(DisplayID dpy,
212738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        sp<IMemoryHeap>* heap,
212838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t* w, uint32_t* h, PixelFormat* f,
21293dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t sw, uint32_t sh,
21303dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t minLayerZ, uint32_t maxLayerZ)
213138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian{
213238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    status_t result = PERMISSION_DENIED;
213338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
213438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // only one display supported for now
213538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
213638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        return BAD_VALUE;
213738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
213838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
213938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        return INVALID_OPERATION;
214038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
214138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // get screen geometry
214238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const DisplayHardware& hw(graphicPlane(dpy).displayHardware());
214338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const uint32_t hw_w = hw.getWidth();
214438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const uint32_t hw_h = hw.getHeight();
214538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
214638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if ((sw > hw_w) || (sh > hw_h))
214738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        return BAD_VALUE;
214838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
214938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    sw = (!sw) ? hw_w : sw;
215038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    sh = (!sh) ? hw_h : sh;
215138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const size_t size = sw * sh * 4;
215238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
215332ae094d87cbb35f8b31acb7b83b430db62d5925Mathias Agopian    //LOGD("screenshot: sw=%d, sh=%d, minZ=%d, maxZ=%d",
215432ae094d87cbb35f8b31acb7b83b430db62d5925Mathias Agopian    //        sw, sh, minLayerZ, maxLayerZ);
2155cd2cfb6a15975af02a02c34783480eb858ee2928Mathias Agopian
215638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // make sure to clear all GL error flags
215738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    while ( glGetError() != GL_NO_ERROR ) ;
215838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
215938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // create a FBO
216038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    GLuint name, tname;
216138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glGenRenderbuffersOES(1, &tname);
216238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glBindRenderbufferOES(GL_RENDERBUFFER_OES, tname);
216338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glRenderbufferStorageOES(GL_RENDERBUFFER_OES, GL_RGBA8_OES, sw, sh);
216438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glGenFramebuffersOES(1, &name);
216538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, name);
216638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES,
216738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            GL_COLOR_ATTACHMENT0_OES, GL_RENDERBUFFER_OES, tname);
216838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
216938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    GLenum status = glCheckFramebufferStatusOES(GL_FRAMEBUFFER_OES);
2170cd2cfb6a15975af02a02c34783480eb858ee2928Mathias Agopian
217138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if (status == GL_FRAMEBUFFER_COMPLETE_OES) {
217238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
217338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // invert everything, b/c glReadPixel() below will invert the FB
217438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glViewport(0, 0, sw, sh);
21751c4e4fc0490c6f07111365cfe21116c31254d5ffMathias Agopian        glScissor(0, 0, sw, sh);
217638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_PROJECTION);
217738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glPushMatrix();
217838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glLoadIdentity();
217938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glOrthof(0, hw_w, 0, hw_h, 0, 1);
218038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_MODELVIEW);
218138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
218238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // redraw the screen entirely...
218338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glClearColor(0,0,0,1);
218438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
21851c4e4fc0490c6f07111365cfe21116c31254d5ffMathias Agopian
218638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
218738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        const size_t count = layers.size();
218838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        for (size_t i=0 ; i<count ; ++i) {
218938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            const sp<LayerBase>& layer(layers[i]);
21903dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            const uint32_t z = layer->drawingState().z;
21913dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            if (z >= minLayerZ && z <= maxLayerZ) {
21923dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                layer->drawForSreenShot();
21933dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            }
219438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        }
219538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
219638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // XXX: this is needed on tegra
219738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glScissor(0, 0, sw, sh);
219838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
219938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // check for errors and return screen capture
220038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        if (glGetError() != GL_NO_ERROR) {
220138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            // error while rendering
220238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            result = INVALID_OPERATION;
220338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        } else {
220438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            // allocate shared memory large enough to hold the
220538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            // screen capture
220638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            sp<MemoryHeapBase> base(
220738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    new MemoryHeapBase(size, 0, "screen-capture") );
220838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            void* const ptr = base->getBase();
220938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            if (ptr) {
221038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                // capture the screen with glReadPixels()
221138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                glReadPixels(0, 0, sw, sh, GL_RGBA, GL_UNSIGNED_BYTE, ptr);
221238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                if (glGetError() == GL_NO_ERROR) {
221338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *heap = base;
221438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *w = sw;
221538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *h = sh;
221638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *f = PIXEL_FORMAT_RGBA_8888;
221738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    result = NO_ERROR;
221838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                }
221938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            } else {
222038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                result = NO_MEMORY;
222138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            }
222238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        }
222338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glEnable(GL_SCISSOR_TEST);
222438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glViewport(0, 0, hw_w, hw_h);
222538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_PROJECTION);
222638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glPopMatrix();
222738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_MODELVIEW);
222838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    } else {
222938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        result = BAD_VALUE;
223038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    }
223138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
223238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // release FBO resources
223338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
223438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glDeleteRenderbuffersOES(1, &tname);
223538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glDeleteFramebuffersOES(1, &name);
2236a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian
2237a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian    hw.compositionComplete();
2238a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian
223932ae094d87cbb35f8b31acb7b83b430db62d5925Mathias Agopian    // LOGD("screenshot: result = %s", result<0 ? strerror(result) : "OK");
2240cd2cfb6a15975af02a02c34783480eb858ee2928Mathias Agopian
224138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    return result;
224238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian}
224338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
224438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
2245ca5edbeba92b96913291792a4df984e158853b6dMathias Agopianstatus_t SurfaceFlinger::captureScreen(DisplayID dpy,
2246ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        sp<IMemoryHeap>* heap,
224738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t* width, uint32_t* height, PixelFormat* format,
22483dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t sw, uint32_t sh,
22493dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t minLayerZ, uint32_t maxLayerZ)
2250ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian{
2251ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    // only one display supported for now
2252ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
2253ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        return BAD_VALUE;
2254ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2255ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
2256ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        return INVALID_OPERATION;
2257ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2258ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    class MessageCaptureScreen : public MessageBase {
2259ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        SurfaceFlinger* flinger;
2260ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        DisplayID dpy;
2261ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        sp<IMemoryHeap>* heap;
2262ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        uint32_t* w;
2263ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        uint32_t* h;
2264ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        PixelFormat* f;
226538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t sw;
226638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t sh;
22673dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t minLayerZ;
22683dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t maxLayerZ;
2269ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        status_t result;
2270ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    public:
2271ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        MessageCaptureScreen(SurfaceFlinger* flinger, DisplayID dpy,
227238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                sp<IMemoryHeap>* heap, uint32_t* w, uint32_t* h, PixelFormat* f,
22733dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                uint32_t sw, uint32_t sh,
22743dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                uint32_t minLayerZ, uint32_t maxLayerZ)
2275ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            : flinger(flinger), dpy(dpy),
22763dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian              heap(heap), w(w), h(h), f(f), sw(sw), sh(sh),
22773dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian              minLayerZ(minLayerZ), maxLayerZ(maxLayerZ),
22783dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian              result(PERMISSION_DENIED)
2279ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        {
2280ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
2281ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        status_t getResult() const {
2282ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            return result;
2283ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
2284ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        virtual bool handler() {
2285ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
2286ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2287ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            // if we have secure windows, never allow the screen capture
2288ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            if (flinger->mSecureFrameBuffer)
2289ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                return true;
2290ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
229138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            result = flinger->captureScreenImplLocked(dpy,
22923dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                    heap, w, h, f, sw, sh, minLayerZ, maxLayerZ);
2293ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2294ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            return true;
2295ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
2296ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    };
2297ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2298ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    sp<MessageBase> msg = new MessageCaptureScreen(this,
22993dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            dpy, heap, width, height, format, sw, sh, minLayerZ, maxLayerZ);
2300ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    status_t res = postMessageSync(msg);
2301ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (res == NO_ERROR) {
2302ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        res = static_cast<MessageCaptureScreen*>( msg.get() )->getResult();
2303ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    }
2304ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    return res;
2305ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian}
2306ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2307ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian// ---------------------------------------------------------------------------
2308ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
23097623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<Layer> SurfaceFlinger::getLayer(const sp<ISurface>& sur) const
23109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
23117623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<Layer> result;
23127623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    Mutex::Autolock _l(mStateLock);
23137623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    result = mLayerMap.valueFor( sur->asBinder() ).promote();
23147623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return result;
23157623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
2316d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
23177623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian// ---------------------------------------------------------------------------
2318593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
23197623da435e45c7c03ef6a00a43675deb6645f070Mathias AgopianClient::Client(const sp<SurfaceFlinger>& flinger)
23207623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    : mFlinger(flinger), mNameGenerator(1)
23217623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
23229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
23239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2324593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias AgopianClient::~Client()
2325593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
2326593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    const size_t count = mLayers.size();
2327593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
2328593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        sp<LayerBaseClient> layer(mLayers.valueAt(i).promote());
2329593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        if (layer != 0) {
2330593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            mFlinger->removeLayer(layer);
2331593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        }
23329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
23339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
23341473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
2335593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t Client::initCheck() const {
23367623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return NO_ERROR;
23379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
23381473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
2339593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianssize_t Client::attachLayer(const sp<LayerBaseClient>& layer)
23409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
23417623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    int32_t name = android_atomic_inc(&mNameGenerator);
2342593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    mLayers.add(name, layer);
2343593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return name;
23449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
23459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
23467623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianvoid Client::detachLayer(const LayerBaseClient* layer)
2347593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
2348593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // we do a linear search here, because this doesn't happen often
2349593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    const size_t count = mLayers.size();
2350593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
2351593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        if (mLayers.valueAt(i) == layer) {
2352593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            mLayers.removeItemsAt(i, 1);
2353593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            break;
2354593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        }
2355593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    }
2356593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian}
23571473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopiansp<LayerBaseClient> Client::getLayerUser(int32_t i) const {
23581473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBaseClient> lbc;
2359593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    const wp<LayerBaseClient>& layer(mLayers.valueFor(i));
2360593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (layer != 0) {
2361593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        lbc = layer.promote();
2362593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        LOGE_IF(lbc==0, "getLayerUser(name=%d) is dead", int(i));
23631473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    }
23641473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    return lbc;
23659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
23669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2367593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopiansp<IMemoryHeap> Client::getControlBlock() const {
23687623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return 0;
23697623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
23707623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianssize_t Client::getTokenForSurface(const sp<ISurface>& sur) const {
23717623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return -1;
23729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2373593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopiansp<ISurface> Client::createSurface(
23747623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        ISurfaceComposerClient::surface_data_t* params, int pid,
23757623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const String8& name,
23767623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        DisplayID display, uint32_t w, uint32_t h, PixelFormat format,
23779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        uint32_t flags)
23789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
2379593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return mFlinger->createSurface(this, pid, name, params,
2380593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            display, w, h, format, flags);
23819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2382593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t Client::destroySurface(SurfaceID sid) {
2383593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return mFlinger->removeSurface(this, sid);
23849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2385593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t Client::setState(int32_t count, const layer_state_t* states) {
2386593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return mFlinger->setClientState(this, count, states);
23879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
23889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
23899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
23907623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
23917623da435e45c7c03ef6a00a43675deb6645f070Mathias AgopianUserClient::UserClient(const sp<SurfaceFlinger>& flinger)
23927623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    : ctrlblk(0), mBitmap(0), mFlinger(flinger)
23937623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
23947623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    const int pgsize = getpagesize();
23957623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    const int cblksize = ((sizeof(SharedClient)+(pgsize-1))&~(pgsize-1));
23967623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
23977623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    mCblkHeap = new MemoryHeapBase(cblksize, 0,
23987623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            "SurfaceFlinger Client control-block");
23997623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24007623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    ctrlblk = static_cast<SharedClient *>(mCblkHeap->getBase());
24017623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (ctrlblk) { // construct the shared structure in-place.
24027623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        new(ctrlblk) SharedClient;
24037623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
24047623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24057623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24067623da435e45c7c03ef6a00a43675deb6645f070Mathias AgopianUserClient::~UserClient()
24077623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
24087623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (ctrlblk) {
24097623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        ctrlblk->~SharedClient();  // destroy our shared-structure.
24107623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
24117623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24127623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    /*
24137623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * When a UserClient dies, it's unclear what to do exactly.
24147623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * We could go ahead and destroy all surfaces linked to that client
24157623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * however, it wouldn't be fair to the main Client
24167623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * (usually the the window-manager), which might want to re-target
24177623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * the layer to another UserClient.
24187623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * I think the best is to do nothing, or not much; in most cases the
24197623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * WM itself will go ahead and clean things up when it detects a client of
24207623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * his has died.
24217623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * The remaining question is what to display? currently we keep
24227623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * just keep the current buffer.
24237623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     */
24247623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24257623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24267623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianstatus_t UserClient::initCheck() const {
24277623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return ctrlblk == 0 ? NO_INIT : NO_ERROR;
24287623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24297623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24307623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianvoid UserClient::detachLayer(const Layer* layer)
24317623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
24327623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    int32_t name = layer->getToken();
24337623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (name >= 0) {
24345e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        int32_t mask = 1LU<<name;
24355e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        if ((android_atomic_and(~mask, &mBitmap) & mask) == 0) {
24365e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian            LOGW("token %d wasn't marked as used %08x", name, int(mBitmap));
24375e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        }
24387623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
24397623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24407623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24417623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<IMemoryHeap> UserClient::getControlBlock() const {
24427623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return mCblkHeap;
24437623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24447623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24457623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianssize_t UserClient::getTokenForSurface(const sp<ISurface>& sur) const
24467623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
24477623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    int32_t name = NAME_NOT_FOUND;
24487623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<Layer> layer(mFlinger->getLayer(sur));
244985cfdd011241a5f2fb7fabc65b5943a39af7e1deJamie Gennis    if (layer == 0) {
245085cfdd011241a5f2fb7fabc65b5943a39af7e1deJamie Gennis        return name;
245185cfdd011241a5f2fb7fabc65b5943a39af7e1deJamie Gennis    }
24527623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24535e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian    // if this layer already has a token, just return it
24547623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    name = layer->getToken();
245585cfdd011241a5f2fb7fabc65b5943a39af7e1deJamie Gennis    if ((name >= 0) && (layer->getClient() == this)) {
24565e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        return name;
245785cfdd011241a5f2fb7fabc65b5943a39af7e1deJamie Gennis    }
24587623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24597623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    name = 0;
24607623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    do {
24617623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        int32_t mask = 1LU<<name;
24627623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        if ((android_atomic_or(mask, &mBitmap) & mask) == 0) {
24637623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            // we found and locked that name
24645e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian            status_t err = layer->setToken(
24655e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian                    const_cast<UserClient*>(this), ctrlblk, name);
24665e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian            if (err != NO_ERROR) {
24675e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian                // free the name
24685e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian                android_atomic_and(~mask, &mBitmap);
24695e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian                name = err;
24705e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian            }
24717623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            break;
24727623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        }
2473c86d904b71be950ad973d803d1a19c6cd13be669Jamie Gennis        if (++name >= SharedBufferStack::NUM_LAYERS_MAX)
24747623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            name = NO_MEMORY;
24757623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    } while(name >= 0);
24767623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24771debc66521f699bbf0a8eb80cababaef8bc63607Mathias Agopian    //LOGD("getTokenForSurface(%p) => %d (client=%p, bitmap=%08lx)",
24781debc66521f699bbf0a8eb80cababaef8bc63607Mathias Agopian    //        sur->asBinder().get(), name, this, mBitmap);
24797623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return name;
24807623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24817623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24827623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<ISurface> UserClient::createSurface(
24837623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        ISurfaceComposerClient::surface_data_t* params, int pid,
24847623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const String8& name,
24857623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        DisplayID display, uint32_t w, uint32_t h, PixelFormat format,
24867623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        uint32_t flags) {
24877623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return 0;
24887623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24897623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianstatus_t UserClient::destroySurface(SurfaceID sid) {
24907623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return INVALID_OPERATION;
24917623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24927623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianstatus_t UserClient::setState(int32_t count, const layer_state_t* states) {
24937623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return INVALID_OPERATION;
24947623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24957623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24967623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian// ---------------------------------------------------------------------------
24979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2498f7acf162f8d682c6ebc9af41ca76795b79509193Jamie GennisGraphicBufferAlloc::GraphicBufferAlloc() {}
2499f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis
2500f7acf162f8d682c6ebc9af41ca76795b79509193Jamie GennisGraphicBufferAlloc::~GraphicBufferAlloc() {}
2501f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis
2502f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennissp<GraphicBuffer> GraphicBufferAlloc::createGraphicBuffer(uint32_t w, uint32_t h,
2503f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis        PixelFormat format, uint32_t usage) {
2504f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    sp<GraphicBuffer> graphicBuffer(new GraphicBuffer(w, h, format, usage));
2505f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    status_t err = graphicBuffer->initCheck();
2506f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    if (err != 0) {
2507f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis        LOGE("createGraphicBuffer: init check failed: %d", err);
2508f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis        return 0;
2509f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    } else if (graphicBuffer->handle == 0) {
2510f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis        LOGE("createGraphicBuffer: unable to create GraphicBuffer");
2511f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis        return 0;
2512f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    }
2513f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    Mutex::Autolock _l(mLock);
2514f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    mBuffers.add(graphicBuffer);
2515f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    return graphicBuffer;
2516f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis}
2517f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis
2518f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennisvoid GraphicBufferAlloc::freeAllGraphicBuffersExcept(int bufIdx) {
2519f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    Mutex::Autolock _l(mLock);
2520f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    if (0 <= bufIdx && bufIdx < mBuffers.size()) {
2521f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis        sp<GraphicBuffer> b(mBuffers[bufIdx]);
2522f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis        mBuffers.clear();
2523f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis        mBuffers.add(b);
2524f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    } else {
2525f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis        mBuffers.clear();
2526f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    }
2527f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis}
2528f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis
2529f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis// ---------------------------------------------------------------------------
2530f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis
25319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane::GraphicPlane()
25329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    : mHw(0)
25339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
25349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane::~GraphicPlane() {
25379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    delete mHw;
25389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool GraphicPlane::initialized() const {
25419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return mHw ? true : false;
25429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
254466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianint GraphicPlane::getWidth() const {
254566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    return mWidth;
25469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
254866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianint GraphicPlane::getHeight() const {
254966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    return mHeight;
255066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian}
255166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
255266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianvoid GraphicPlane::setDisplayHardware(DisplayHardware *hw)
255366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian{
255466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mHw = hw;
255566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
255666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    // initialize the display orientation transform.
255766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    // it's a constant that should come from the display driver.
255866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    int displayOrientation = ISurfaceComposer::eOrientationDefault;
255966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    char property[PROPERTY_VALUE_MAX];
256066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    if (property_get("ro.sf.hwrotation", property, NULL) > 0) {
256166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        //displayOrientation
256266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        switch (atoi(property)) {
256366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        case 90:
256466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            displayOrientation = ISurfaceComposer::eOrientation90;
256566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            break;
256666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        case 270:
256766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            displayOrientation = ISurfaceComposer::eOrientation270;
256866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            break;
256966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        }
257066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    }
257166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
257266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float w = hw->getWidth();
257366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float h = hw->getHeight();
257466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    GraphicPlane::orientationToTransfrom(displayOrientation, w, h,
257566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            &mDisplayTransform);
257666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    if (displayOrientation & ISurfaceComposer::eOrientationSwapMask) {
257766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayWidth = h;
257866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayHeight = w;
257966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    } else {
258066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayWidth = w;
258166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayHeight = h;
258266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    }
258366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
258466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    setOrientation(ISurfaceComposer::eOrientationDefault);
25859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t GraphicPlane::orientationToTransfrom(
25889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int orientation, int w, int h, Transform* tr)
25898c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian{
25908c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    uint32_t flags = 0;
25919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (orientation) {
25929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientationDefault:
25938c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_0;
25948c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        break;
25959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation90:
25968c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_90;
25979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
25989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation180:
25998c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_180;
26009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
26019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation270:
26028c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_270;
26039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
26049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    default:
26059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
26069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
26078c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    tr->set(flags, w, h);
26089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
26099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
26109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
26119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t GraphicPlane::setOrientation(int orientation)
26129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
26139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // If the rotation can be handled in hardware, this is where
26149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // the magic should happen.
261566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
261666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const DisplayHardware& hw(displayHardware());
261766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float w = mDisplayWidth;
261866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float h = mDisplayHeight;
261966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mWidth = int(w);
262066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mHeight = int(h);
262166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
262266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    Transform orientationTransform;
26238c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    GraphicPlane::orientationToTransfrom(orientation, w, h,
26248c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian            &orientationTransform);
26258c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    if (orientation & ISurfaceComposer::eOrientationSwapMask) {
26268c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        mWidth = int(h);
26278c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        mHeight = int(w);
26289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
26298c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian
26303552f53c8370ced8680951f4ac811a126da02b0eMathias Agopian    mOrientation = orientation;
263166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mGlobalTransform = mDisplayTransform * orientationTransform;
26329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
26339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
26349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
26359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst DisplayHardware& GraphicPlane::displayHardware() const {
26369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return *mHw;
26379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
26389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2639aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias AgopianDisplayHardware& GraphicPlane::editDisplayHardware() {
2640aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    return *mHw;
2641aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
2642aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
26439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst Transform& GraphicPlane::transform() const {
26449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return mGlobalTransform;
26459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
26469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
26471473f46cbc82aa6f0ba744cc896a36923823d55bMathias AgopianEGLDisplay GraphicPlane::getEGLDisplay() const {
26481473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    return mHw->getEGLDisplay();
26491473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian}
26501473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
26519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
26529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
26539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}; // namespace android
2654