SurfaceFlinger.cpp revision 1c4e4fc0490c6f07111365cfe21116c31254d5ff
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
1477623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
1489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst GraphicPlane& SurfaceFlinger::graphicPlane(int dpy) const
1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LOGE_IF(uint32_t(dpy) >= DISPLAY_COUNT, "Invalid DisplayID %d", dpy);
1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(mGraphicPlanes[dpy]);
1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return plane;
1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane& SurfaceFlinger::graphicPlane(int dpy)
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return const_cast<GraphicPlane&>(
1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const_cast<SurfaceFlinger const *>(this)->graphicPlane(dpy));
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::bootFinished()
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const nsecs_t now = systemTime();
1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const nsecs_t duration = now - mBootTime;
165e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber    LOGI("Boot is finished (%ld ms)", long(ns2ms(duration)) );
1666950e428feaccc8164b989ef64e771a99948797aMathias Agopian    mBootFinished = true;
167627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    property_set("ctl.stop", "bootanim");
1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::onFirstRef()
1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    run("SurfaceFlinger", PRIORITY_URGENT_DISPLAY);
1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // Wait for the main thread to be done with its initialization
1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mReadyToRunBarrier.wait();
1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic inline uint16_t pack565(int r, int g, int b) {
1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return (r<<11)|(g<<5)|b;
1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::readyToRun()
1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LOGI(   "SurfaceFlinger's main thread ready to run. "
1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            "Initializing graphics H/W...");
1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // we only support one display currently
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int dpy = 0;
1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    {
1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // initialize the main display
1929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        GraphicPlane& plane(graphicPlane(dpy));
1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        DisplayHardware* const hw = new DisplayHardware(this, dpy);
1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        plane.setDisplayHardware(hw);
1959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
197d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    // create the shared control-block
198d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    mServerHeap = new MemoryHeapBase(4096,
199d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian            MemoryHeapBase::READ_ONLY, "SurfaceFlinger read-only heap");
200d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    LOGE_IF(mServerHeap==0, "can't create shared memory dealer");
201e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
202d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    mServerCblk = static_cast<surface_flinger_cblk_t*>(mServerHeap->getBase());
203d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    LOGE_IF(mServerCblk==0, "can't get to shared control block's address");
204e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
205d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    new(mServerCblk) surface_flinger_cblk_t;
206d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
2079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize primary screen
2089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // (other display should be initialized in the same manner, but
2099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // asynchronously, as they could come and go. None of this is supported
2109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // yet).
2119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(dpy));
2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw = plane.displayHardware();
2139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint32_t w = hw.getWidth();
2149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint32_t h = hw.getHeight();
2159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint32_t f = hw.getFormat();
2169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    hw.makeCurrent();
2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize the shared control block
2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mServerCblk->connected |= 1<<dpy;
2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    display_cblk_t* dcblk = mServerCblk->displays + dpy;
2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    memset(dcblk, 0, sizeof(display_cblk_t));
22266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    dcblk->w            = plane.getWidth();
22366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    dcblk->h            = plane.getHeight();
2249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->format       = f;
2259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->orientation  = ISurfaceComposer::eOrientationDefault;
2269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->xdpi         = hw.getDpiX();
2279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->ydpi         = hw.getDpiY();
2289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->fps          = hw.getRefreshRate();
2299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->density      = hw.getDensity();
2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // Initialize OpenGL|ES
2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glPixelStorei(GL_UNPACK_ALIGNMENT, 4);
233e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber    glPixelStorei(GL_PACK_ALIGNMENT, 4);
2349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glEnableClientState(GL_VERTEX_ARRAY);
2359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glEnable(GL_SCISSOR_TEST);
2369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glShadeModel(GL_FLAT);
2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
2389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_CULL_FACE);
2399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint16_t g0 = pack565(0x0F,0x1F,0x0F);
2419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint16_t g1 = pack565(0x17,0x2f,0x17);
2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint16_t textureData[4] = { g0, g1, g1, g0 };
2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glGenTextures(1, &mWormholeTexName);
2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glBindTexture(GL_TEXTURE_2D, mWormholeTexName);
2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, 2, 2, 0,
2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            GL_RGB, GL_UNSIGNED_SHORT_5_6_5, textureData);
2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glViewport(0, 0, w, h);
2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glMatrixMode(GL_PROJECTION);
2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glLoadIdentity();
2559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glOrthof(0, w, h, 0, 0, 1);
2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project   LayerDim::initDimmer(this, w, h);
2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mReadyToRunBarrier.open();
2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *  We're now ready to accept clients...
2639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
265627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    // start boot animation
266627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    property_set("ctl.start", "bootanim");
267e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
2689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
2699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#if 0
2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark -
2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark Events Handler
2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif
2769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::waitForEvent()
2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
2796ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    while (true) {
2806ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        nsecs_t timeout = -1;
2810e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        const nsecs_t freezeDisplayTimeout = ms2ns(5000);
2826ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        if (UNLIKELY(isFrozen())) {
2836ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            // wait 5 seconds
2846ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            const nsecs_t now = systemTime();
2856ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            if (mFreezeDisplayTime == 0) {
2866ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian                mFreezeDisplayTime = now;
2876ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            }
2886ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            nsecs_t waitTime = freezeDisplayTimeout - (now - mFreezeDisplayTime);
2896ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            timeout = waitTime>0 ? waitTime : 0;
290c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project        }
2916ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian
292898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        sp<MessageBase> msg = mEventQueue.waitMessage(timeout);
2930e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian
2940e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        // see if we timed out
2950e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        if (isFrozen()) {
2960e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            const nsecs_t now = systemTime();
2970e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            nsecs_t frozenTime = (now - mFreezeDisplayTime);
2980e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            if (frozenTime >= freezeDisplayTimeout) {
2999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // we timed out and are still frozen
3009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                LOGW("timeout expired mFreezeDisplay=%d, mFreezeCount=%d",
3019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        mFreezeDisplay, mFreezeCount);
3020e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                mFreezeDisplayTime = 0;
3039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeCount = 0;
304c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project                mFreezeDisplay = false;
3050e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            }
3060e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        }
3070e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian
3080e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        if (msg != 0) {
3090e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            switch (msg->what) {
3100e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                case MessageQueue::INVALIDATE:
3110e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                    // invalidate message, just return to the main loop
3120e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                    return;
3139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
3149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::signalEvent() {
3196ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    mEventQueue.invalidate();
3209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::signal() const {
3236ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    // this is the IPC call
3246ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    const_cast<SurfaceFlinger*>(this)->signalEvent();
3259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
327898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianstatus_t SurfaceFlinger::postMessageAsync(const sp<MessageBase>& msg,
328898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        nsecs_t reltime, uint32_t flags)
3299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
330898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    return mEventQueue.postMessage(msg, reltime, flags);
331898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian}
332898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian
333898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianstatus_t SurfaceFlinger::postMessageSync(const sp<MessageBase>& msg,
334898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        nsecs_t reltime, uint32_t flags)
335898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian{
336898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    status_t res = mEventQueue.postMessage(msg, reltime, flags);
337898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    if (res == NO_ERROR) {
338898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        msg->wait();
339898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    }
340898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    return res;
3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
3449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#if 0
3459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark -
3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark Main loop
3479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif
3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool SurfaceFlinger::threadLoop()
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    waitForEvent();
3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // check for transactions
3549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(mConsoleSignals)) {
3559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        handleConsoleEvents();
3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (LIKELY(mTransactionCount == 0)) {
3599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // if we're in a global transaction, don't do anything.
3609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const uint32_t mask = eTransactionNeeded | eTraversalNeeded;
3619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        uint32_t transactionFlags = getTransactionFlags(mask);
3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (LIKELY(transactionFlags)) {
3639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            handleTransaction(transactionFlags);
3649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // post surfaces (if needed)
3689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    handlePageFlip();
3699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
370e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    if (UNLIKELY(mHwWorkListDirty)) {
371e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        // build the h/w work list
372e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        handleWorkList();
373e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    }
374e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw(graphicPlane(0).displayHardware());
37681384bf927c47a4efa653b14273084a13e67e3acMathias Agopian    if (LIKELY(hw.canDraw() && !isFrozen())) {
3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // repaint the framebuffer (if needed)
37804262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian
37904262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        const int index = hw.getCurrentBufferIndex();
38004262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        GraphicLog& logger(GraphicLog::getInstance());
38104262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian
38204262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_REPAINT, index);
3839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        handleRepaint();
3849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
385b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian        // inform the h/w that we're done compositing
38604262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_COMPOSITION_COMPLETE, index);
387b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian        hw.compositionComplete();
388b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian
38904262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_SWAP_BUFFERS, index);
3904c0a4a2b74db9d4a439a0aaa39d80586f7eb1258Antti Hatala        postFramebuffer();
3914c0a4a2b74db9d4a439a0aaa39d80586f7eb1258Antti Hatala
392a5ab8ce602ecfd897805e185c89a6eff0c78aabeMathias Agopian        logger.log(GraphicLog::SF_UNLOCK_CLIENTS, index);
3939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        unlockClients();
3949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
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        unlockClients();
4009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        usleep(16667); // 60 fps period
4019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return true;
4039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::postFramebuffer()
4069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (!mInvalidRegion.isEmpty()) {
4089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw(graphicPlane(0).displayHardware());
409a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t now = systemTime();
410a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers = now;
4119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.flip(mInvalidRegion);
412a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastSwapBufferTime = systemTime() - now;
413a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers = 0;
4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mInvalidRegion.clear();
4159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleConsoleEvents()
4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // something to do with the console
4219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw = graphicPlane(0).displayHardware();
4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int what = android_atomic_and(0, &mConsoleSignals);
4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (what & eConsoleAcquired) {
4259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.acquireScreen();
4262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // this is a temporary work-around, eventually this should be called
4272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // by the power-manager
428d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        SurfaceFlinger::turnElectronBeamOn(mElectronBeamAnimationMode);
4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
431aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (mDeferReleaseConsole && hw.isScreenAcquired()) {
432ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian        // We got the release signal before the acquire signal
4339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDeferReleaseConsole = false;
4349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.releaseScreen();
4359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (what & eConsoleReleased) {
438aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        if (hw.isScreenAcquired()) {
4399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            hw.releaseScreen();
4409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
4419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDeferReleaseConsole = true;
4429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.set(hw.bounds());
4469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleTransaction(uint32_t transactionFlags)
4499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4502d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    Vector< sp<LayerBase> > ditchedLayers;
4512d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian
452ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian    /*
453ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     * Perform and commit the transaction
454ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     */
455ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian
4562d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    { // scope for the lock
4572d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian        Mutex::Autolock _l(mStateLock);
458a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t now = systemTime();
459a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInTransaction = now;
4602d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian        handleTransactionLocked(transactionFlags, ditchedLayers);
461a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastTransactionTime = systemTime() - now;
462a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInTransaction = 0;
463e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        mHwWorkListDirty = true;
464ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian        // here the transaction has been committed
4652d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    }
4662d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian
467ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian    /*
468ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     * Clean-up all layers that went away
469ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     * (do this without the lock held)
470ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     */
471e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
4722d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    const size_t count = ditchedLayers.size();
4732d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
474ffae4fcc78d0f280da6052d102b11962fb8041b7Mathias Agopian        if (ditchedLayers[i] != 0) {
475ffae4fcc78d0f280da6052d102b11962fb8041b7Mathias Agopian            //LOGD("ditching layer %p", ditchedLayers[i].get());
476ffae4fcc78d0f280da6052d102b11962fb8041b7Mathias Agopian            ditchedLayers[i]->ditch();
477ffae4fcc78d0f280da6052d102b11962fb8041b7Mathias Agopian        }
4782d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    }
4792d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian}
4809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4812d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopianvoid SurfaceFlinger::handleTransactionLocked(
4822d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian        uint32_t transactionFlags, Vector< sp<LayerBase> >& ditchedLayers)
4832d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian{
4842d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    const LayerVector& currentLayers(mCurrentState.layersSortedByZ);
4859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const size_t count = currentLayers.size();
4869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
4889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Traversal of the children
4899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * (perform the transaction for each of them if needed)
4909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
4919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const bool layersNeedTransaction = transactionFlags & eTraversalNeeded;
4939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (layersNeedTransaction) {
4949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (size_t i=0 ; i<count ; i++) {
4951473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian            const sp<LayerBase>& layer = currentLayers[i];
4969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            uint32_t trFlags = layer->getTransactionFlags(eTransactionNeeded);
4979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (!trFlags) continue;
4989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const uint32_t flags = layer->doTransaction(0);
5009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (flags & Layer::eVisibleRegion)
5019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mVisibleRegionsDirty = true;
5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Perform our own transaction if needed
5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (transactionFlags & eTransactionNeeded) {
5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mCurrentState.orientation != mDrawingState.orientation) {
5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // the orientation has changed, recompute all visible regions
5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // and invalidate everything.
5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int dpy = 0;
5159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int orientation = mCurrentState.orientation;
516eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian            const uint32_t type = mCurrentState.orientationType;
5179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            GraphicPlane& plane(graphicPlane(dpy));
5189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            plane.setOrientation(orientation);
5199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // update the shared control block
5219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const DisplayHardware& hw(plane.displayHardware());
5229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            volatile display_cblk_t* dcblk = mServerCblk->displays + dpy;
5239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dcblk->orientation = orientation;
52466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            dcblk->w = plane.getWidth();
52566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            dcblk->h = plane.getHeight();
5269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
5289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(hw.bounds());
5299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mCurrentState.freezeDisplay != mDrawingState.freezeDisplay) {
5329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // freezing or unfreezing the display -> trigger animation if needed
5339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mFreezeDisplay = mCurrentState.freezeDisplay;
53431901cc0b6f0c678be4f629c8c3405700e63c346Mathias Agopian            if (mFreezeDisplay)
53531901cc0b6f0c678be4f629c8c3405700e63c346Mathias Agopian                 mFreezeDisplayTime = 0;
5369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
538a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        if (currentLayers.size() > mDrawingState.layersSortedByZ.size()) {
539a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            // layers have been added
5409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
5419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
543a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        // some layers might have been removed, so
544a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        // we need to update the regions they're exposing.
545a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        if (mLayersRemoved) {
546248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian            mLayersRemoved = false;
5479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
548a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            const LayerVector& previousLayers(mDrawingState.layersSortedByZ);
5492d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian            const size_t count = previousLayers.size();
5502d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian            for (size_t i=0 ; i<count ; i++) {
551a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                const sp<LayerBase>& layer(previousLayers[i]);
552a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                if (currentLayers.indexOf( layer ) < 0) {
553a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                    // this layer is not visible anymore
5542d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian                    ditchedLayers.add(layer);
55533863dd9e1005478d20b70fad42e447ac97f5abfMathias Agopian                    mDirtyRegionRemovedLayer.orSelf(layer->visibleRegionScreen);
556a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                }
557a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            }
5589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    commitTransaction();
5629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
5639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectsp<FreezeLock> SurfaceFlinger::getFreezeLock() const
5659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
5669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return new FreezeLock(const_cast<SurfaceFlinger *>(this));
5679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
5689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::computeVisibleRegions(
5709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LayerVector& currentLayers, Region& dirtyRegion, Region& opaqueRegion)
5719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
5729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(0));
5739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Transform& planeTransform(plane.transform());
5749c041bbd81789c209e2369ba958306979b67614fMathias Agopian    const DisplayHardware& hw(plane.displayHardware());
5759c041bbd81789c209e2369ba958306979b67614fMathias Agopian    const Region screenRegion(hw.bounds());
5769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region aboveOpaqueLayers;
5789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region aboveCoveredLayers;
5799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region dirty;
5809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool secureFrameBuffer = false;
5829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t i = currentLayers.size();
5849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    while (i--) {
5851473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        const sp<LayerBase>& layer = currentLayers[i];
5869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->validateVisibility(planeTransform);
5879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // start with the whole surface at its current location
58912cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        const Layer::State& s(layer->drawingState());
5909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5919c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
5929c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * opaqueRegion: area of a surface that is fully opaque.
5939c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
5949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region opaqueRegion;
5959c041bbd81789c209e2369ba958306979b67614fMathias Agopian
5969c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
5979c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * visibleRegion: area of a surface that is visible on screen
5989c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * and not fully transparent. This is essentially the layer's
5999c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * footprint minus the opaque regions above it.
6009c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * Areas covered by a translucent surface are considered visible.
6019c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
6029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region visibleRegion;
6039c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6049c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
6059c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * coveredRegion: area of a surface that is covered by all
6069c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * visible regions above it (which includes the translucent areas).
6079c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
6089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region coveredRegion;
6099c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6109c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6119c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // handle hidden surfaces by setting the visible region to empty
61212cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        if (LIKELY(!(s.flags & ISurfaceComposer::eLayerHidden) && s.alpha)) {
6139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const bool translucent = layer->needsBlending();
61412cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian            const Rect bounds(layer->visibleBounds());
6159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            visibleRegion.set(bounds);
6169c041bbd81789c209e2369ba958306979b67614fMathias Agopian            visibleRegion.andSelf(screenRegion);
6179c041bbd81789c209e2369ba958306979b67614fMathias Agopian            if (!visibleRegion.isEmpty()) {
6189c041bbd81789c209e2369ba958306979b67614fMathias Agopian                // Remove the transparent area from the visible region
6199c041bbd81789c209e2369ba958306979b67614fMathias Agopian                if (translucent) {
6209c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    visibleRegion.subtractSelf(layer->transparentRegionScreen);
6219c041bbd81789c209e2369ba958306979b67614fMathias Agopian                }
6229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6239c041bbd81789c209e2369ba958306979b67614fMathias Agopian                // compute the opaque region
6249c041bbd81789c209e2369ba958306979b67614fMathias Agopian                const int32_t layerOrientation = layer->getOrientation();
6259c041bbd81789c209e2369ba958306979b67614fMathias Agopian                if (s.alpha==255 && !translucent &&
6269c041bbd81789c209e2369ba958306979b67614fMathias Agopian                        ((layerOrientation & Transform::ROT_INVALID) == false)) {
6279c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    // the opaque region is the layer's footprint
6289c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    opaqueRegion = visibleRegion;
6299c041bbd81789c209e2369ba958306979b67614fMathias Agopian                }
6309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
6319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6339c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Clip the covered region to the visible region
6349c041bbd81789c209e2369ba958306979b67614fMathias Agopian        coveredRegion = aboveCoveredLayers.intersect(visibleRegion);
6359c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6369c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Update aboveCoveredLayers for next (lower) layer
6379c041bbd81789c209e2369ba958306979b67614fMathias Agopian        aboveCoveredLayers.orSelf(visibleRegion);
6389c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // subtract the opaque region covered by the layers above us
6409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        visibleRegion.subtractSelf(aboveOpaqueLayers);
6419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // compute this layer's dirty region
6439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (layer->contentDirty) {
6449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // we need to invalidate the whole region
6459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dirty = visibleRegion;
6469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // as well, as the old visible region
6479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dirty.orSelf(layer->visibleRegionScreen);
6489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->contentDirty = false;
6499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
6500aed7e97a3efbeec23a4aa301df03036a67f8ea3Mathias Agopian            /* compute the exposed region:
6519c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   the exposed region consists of two components:
6529c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   1) what's VISIBLE now and was COVERED before
6539c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   2) what's EXPOSED now less what was EXPOSED before
6549c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *
6559c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * note that (1) is conservative, we start with the whole
6569c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * visible region but only keep what used to be covered by
6579c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * something -- which mean it may have been exposed.
6589c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *
6599c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * (2) handles areas that were not covered by anything but got
6609c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * exposed because of a resize.
6610aed7e97a3efbeec23a4aa301df03036a67f8ea3Mathias Agopian             */
6629c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region newExposed = visibleRegion - coveredRegion;
6639c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldVisibleRegion = layer->visibleRegionScreen;
6649c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldCoveredRegion = layer->coveredRegionScreen;
6659c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldExposed = oldVisibleRegion - oldCoveredRegion;
6669c041bbd81789c209e2369ba958306979b67614fMathias Agopian            dirty = (visibleRegion&oldCoveredRegion) | (newExposed-oldExposed);
6679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dirty.subtractSelf(aboveOpaqueLayers);
6699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // accumulate to the screen dirty region
6719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dirtyRegion.orSelf(dirty);
6729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6739c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Update aboveOpaqueLayers for next (lower) layer
6749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        aboveOpaqueLayers.orSelf(opaqueRegion);
675e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
6769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Store the visible region is screen space
6779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->setVisibleRegion(visibleRegion);
6789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->setCoveredRegion(coveredRegion);
6799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
68012cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        // If a secure layer is partially visible, lock-down the screen!
6819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (layer->isSecure() && !visibleRegion.isEmpty()) {
6829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            secureFrameBuffer = true;
6839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
68612cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    // invalidate the areas where a layer was removed
68712cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    dirtyRegion.orSelf(mDirtyRegionRemovedLayer);
68812cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    mDirtyRegionRemovedLayer.clear();
68912cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian
6909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mSecureFrameBuffer = secureFrameBuffer;
6919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    opaqueRegion = aboveOpaqueLayers;
6929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
6939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::commitTransaction()
6969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
6979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDrawingState = mCurrentState;
6989779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian    mResizeTransationPending = false;
6999779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian    mTransactionCV.broadcast();
7009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handlePageFlip()
7039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool visibleRegions = mVisibleRegionsDirty;
705e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    LayerVector& currentLayers(
706e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian            const_cast<LayerVector&>(mDrawingState.layersSortedByZ));
7079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    visibleRegions |= lockPageFlip(currentLayers);
7089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw = graphicPlane(0).displayHardware();
7109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const Region screenRegion(hw.bounds());
7119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (visibleRegions) {
7129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            Region opaqueRegion;
7139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            computeVisibleRegions(currentLayers, mDirtyRegion, opaqueRegion);
714ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian
715ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            /*
716ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian             *  rebuild the visible layer list
717ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian             */
718ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            mVisibleLayersSortedByZ.clear();
719ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            const LayerVector& currentLayers(mDrawingState.layersSortedByZ);
720ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            size_t count = currentLayers.size();
721ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            mVisibleLayersSortedByZ.setCapacity(count);
722ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            for (size_t i=0 ; i<count ; i++) {
723ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian                if (!currentLayers[i]->visibleRegionScreen.isEmpty())
724ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian                    mVisibleLayersSortedByZ.add(currentLayers[i]);
725ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            }
726ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian
7279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mWormholeRegion = screenRegion.subtract(opaqueRegion);
7289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = false;
729e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian            mHwWorkListDirty = true;
7309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    unlockPageFlip(currentLayers);
7339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.andSelf(screenRegion);
7349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool SurfaceFlinger::lockPageFlip(const LayerVector& currentLayers)
7379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool recomputeVisibleRegions = false;
7399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t count = currentLayers.size();
7401473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBase> const* layers = currentLayers.array();
7419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (size_t i=0 ; i<count ; i++) {
7427623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
7439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->lockPageFlip(recomputeVisibleRegions);
7449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return recomputeVisibleRegions;
7469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::unlockPageFlip(const LayerVector& currentLayers)
7499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(0));
7519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Transform& planeTransform(plane.transform());
7529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t count = currentLayers.size();
7531473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBase> const* layers = currentLayers.array();
7549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (size_t i=0 ; i<count ; i++) {
7557623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
7569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->unlockPageFlip(planeTransform, mDirtyRegion);
7579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
760e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopianvoid SurfaceFlinger::handleWorkList()
761e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian{
762e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    mHwWorkListDirty = false;
763e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    HWComposer& hwc(graphicPlane(0).displayHardware().getHwComposer());
764e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    if (hwc.initCheck() == NO_ERROR) {
765e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        const Vector< sp<LayerBase> >& currentLayers(mVisibleLayersSortedByZ);
766e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        const size_t count = currentLayers.size();
767e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        hwc.createWorkList(count);
768f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        hwc_layer_t* const cur(hwc.getLayers());
769f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        for (size_t i=0 ; cur && i<count ; i++) {
770f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            currentLayers[i]->setGeometry(&cur[i]);
7716a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian            if (mDebugDisableHWC) {
7726a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                cur[i].compositionType = HWC_FRAMEBUFFER;
7736a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                cur[i].flags |= HWC_SKIP_LAYER;
7746a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian            }
775e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        }
776e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    }
777e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian}
7788c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian
7799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleRepaint()
7809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7818c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    // compute the invalid region
7828c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    mInvalidRegion.orSelf(mDirtyRegion);
7839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(mDebugRegion)) {
7859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        debugFlashRegions();
7869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7888c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    // set the frame buffer
7898c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
7908c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    glMatrixMode(GL_MODELVIEW);
7918c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    glLoadIdentity();
7929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t flags = hw.getFlags();
794e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber    if ((flags & DisplayHardware::SWAP_RECTANGLE) ||
795e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber        (flags & DisplayHardware::BUFFER_PRESERVED))
7962e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian    {
797ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // we can redraw only what's dirty, but since SWAP_RECTANGLE only
798ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // takes a rectangle, we must make sure to update that whole
799ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // rectangle in that case
800ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        if (flags & DisplayHardware::SWAP_RECTANGLE) {
8017623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            // TODO: we really should be able to pass a region to
802ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // SWAP_RECTANGLE so that we don't have to redraw all this.
803ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            mDirtyRegion.set(mInvalidRegion.bounds());
804ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        } else {
805ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // in the BUFFER_PRESERVED case, obviously, we can update only
806ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // what's needed and nothing more.
807ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // NOTE: this is NOT a common case, as preserving the backbuffer
808ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // is costly and usually involves copying the whole update back.
809ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        }
8109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
811f2d28b74850ea0869aef2ce0727a6abb7b166a5cMathias Agopian        if (flags & DisplayHardware::PARTIAL_UPDATES) {
812ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // We need to redraw the rectangle that will be updated
8132e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian            // (pushed to the framebuffer).
814f2d28b74850ea0869aef2ce0727a6abb7b166a5cMathias Agopian            // This is needed because PARTIAL_UPDATES only takes one
815ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // rectangle instead of a region (see DisplayHardware::flip())
8169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(mInvalidRegion.bounds());
8179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
818ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // we need to redraw everything (the whole screen)
8199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(hw.bounds());
8209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mInvalidRegion = mDirtyRegion;
8219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // compose all surfaces
8259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    composeSurfaces(mDirtyRegion);
8269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // clear the dirty regions
8289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.clear();
8299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
8309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::composeSurfaces(const Region& dirty)
8329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
8339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(!mWormholeRegion.isEmpty())) {
8349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // should never happen unless the window manager has a bug
8359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // draw something...
8369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        drawWormhole();
8379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
838e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
839e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    status_t err = NO_ERROR;
840ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian    const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
841f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    size_t count = layers.size();
842e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
843e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
844e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    HWComposer& hwc(hw.getHwComposer());
845f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    hwc_layer_t* const cur(hwc.getLayers());
846e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
847f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    LOGE_IF(cur && hwc.getNumLayers() != count,
848f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            "HAL number of layers (%d) doesn't match surfaceflinger (%d)",
849f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            hwc.getNumLayers(), count);
850e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
851f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    // just to be extra-safe, use the smallest count
8520cf2f4354bf8952d822fa4ee9b9588a73999c2f4Erik Gilling    if (hwc.initCheck() == NO_ERROR) {
8530cf2f4354bf8952d822fa4ee9b9588a73999c2f4Erik Gilling        count = count < hwc.getNumLayers() ? count : hwc.getNumLayers();
8540cf2f4354bf8952d822fa4ee9b9588a73999c2f4Erik Gilling    }
855e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
856f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    /*
857f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     *  update the per-frame h/w composer data for each layer
858f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     *  and build the transparent region of the FB
859f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     */
860f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    Region transparent;
861f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    if (cur) {
862f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        for (size_t i=0 ; i<count ; i++) {
863f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            const sp<LayerBase>& layer(layers[i]);
864f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            layer->setPerFrameData(&cur[i]);
865f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            if (cur[i].hints & HWC_HINT_CLEAR_FB) {
866f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian                if (!(layer->needsBlending())) {
867f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian                    transparent.orSelf(layer->visibleRegionScreen);
868f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian                }
869e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian            }
8709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
871f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        err = hwc.prepare();
872f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        LOGE_IF(err, "HWComposer::prepare failed (%s)", strerror(-err));
8739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
874e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
875f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    /*
876f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     *  clear the area of the FB that need to be transparent
877f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     */
878e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    transparent.andSelf(dirty);
879e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    if (!transparent.isEmpty()) {
880e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        glClearColor(0,0,0,0);
881e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        Region::const_iterator it = transparent.begin();
882e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        Region::const_iterator const end = transparent.end();
883e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        const int32_t height = hw.getHeight();
884e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        while (it != end) {
885e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian            const Rect& r(*it++);
886e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian            const GLint sy = height - (r.top + r.height());
887e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian            glScissor(r.left, sy, r.width(), r.height());
888e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian            glClear(GL_COLOR_BUFFER_BIT);
889e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        }
890e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    }
891f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian
892f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian
893f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    /*
894f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     * and then, render the layers targeted at the framebuffer
895f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     */
896f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
897f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        if (cur) {
898982f58bdccce7e4e537cffb2410ad78d73398461Antti Hatala            if ((cur[i].compositionType != HWC_FRAMEBUFFER) &&
899982f58bdccce7e4e537cffb2410ad78d73398461Antti Hatala                !(cur[i].flags & HWC_SKIP_LAYER)) {
900f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian                // skip layers handled by the HAL
901f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian                continue;
902f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            }
903f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        }
9046a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian
905f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
906f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        const Region clip(dirty.intersect(layer->visibleRegionScreen));
907f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        if (!clip.isEmpty()) {
908f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            layer->draw(clip);
909f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        }
910f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    }
9119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
9129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::unlockClients()
9149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
9159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const LayerVector& drawingLayers(mDrawingState.layersSortedByZ);
9169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const size_t count = drawingLayers.size();
9171473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBase> const* const layers = drawingLayers.array();
9189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (size_t i=0 ; i<count ; ++i) {
9191473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        const sp<LayerBase>& layer = layers[i];
9209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->finishPageFlip();
9219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
9239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::debugFlashRegions()
9259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
926f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
927f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    const uint32_t flags = hw.getFlags();
928f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
929f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    if (!((flags & DisplayHardware::SWAP_RECTANGLE) ||
930f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian            (flags & DisplayHardware::BUFFER_PRESERVED))) {
931f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        const Region repaint((flags & DisplayHardware::PARTIAL_UPDATES) ?
932f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian                mDirtyRegion.bounds() : hw.bounds());
933f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        composeSurfaces(repaint);
934f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    }
935f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
936f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    TextureManager::deactivateTextures();
9372e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian
9389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_BLEND);
9399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
9409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_SCISSOR_TEST);
9419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
942dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    static int toggle = 0;
943dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    toggle = 1 - toggle;
944dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    if (toggle) {
945f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glColor4f(1, 0, 1, 1);
946dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    } else {
947f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glColor4f(1, 1, 0, 1);
948dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    }
9499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9506158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    Region::const_iterator it = mDirtyRegion.begin();
9516158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    Region::const_iterator const end = mDirtyRegion.end();
9526158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    while (it != end) {
9536158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        const Rect& r = *it++;
9549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        GLfloat vertices[][2] = {
9559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.left,  r.top },
9569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.left,  r.bottom },
9579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.right, r.bottom },
9589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.right, r.top }
9599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        };
9609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glVertexPointer(2, GL_FLOAT, 0, vertices);
9619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
9629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
963f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
9648c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    if (mInvalidRegion.isEmpty()) {
9658c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        mDirtyRegion.dump("mDirtyRegion");
9668c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        mInvalidRegion.dump("mInvalidRegion");
9678c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    }
9688c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    hw.flip(mInvalidRegion);
9699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (mDebugRegion > 1)
971f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        usleep(mDebugRegion * 1000);
9729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glEnable(GL_SCISSOR_TEST);
9749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    //mDirtyRegion.dump("mDirtyRegion");
9759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
9769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::drawWormhole() const
9789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
9799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Region region(mWormholeRegion.intersect(mDirtyRegion));
9809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (region.isEmpty())
9819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return;
9829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw(graphicPlane(0).displayHardware());
9849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int32_t width = hw.getWidth();
9859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int32_t height = hw.getHeight();
9869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_BLEND);
9889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
9899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (LIKELY(!mDebugBackground)) {
991f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glClearColor(0,0,0,0);
9926158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator it = region.begin();
9936158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator const end = region.end();
9946158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        while (it != end) {
9956158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian            const Rect& r = *it++;
9969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const GLint sy = height - (r.top + r.height());
9979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glScissor(r.left, sy, r.width(), r.height());
9989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glClear(GL_COLOR_BUFFER_BIT);
9999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
10019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const GLshort vertices[][2] = { { 0, 0 }, { width, 0 },
10029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { width, height }, { 0, height }  };
10039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const GLshort tcoords[][2] = { { 0, 0 }, { 1, 0 },  { 1, 1 }, { 0, 1 } };
10049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glVertexPointer(2, GL_SHORT, 0, vertices);
10059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glTexCoordPointer(2, GL_SHORT, 0, tcoords);
10069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glEnableClientState(GL_TEXTURE_COORD_ARRAY);
1007e20a56d929fc8fedc2b468ea6d1900bd2aa6e81aMichael I. Gold#if defined(GL_OES_EGL_image_external)
1008781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian        if (GLExtensions::getInstance().haveTextureExternal()) {
1009781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian            glDisable(GL_TEXTURE_EXTERNAL_OES);
1010781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian        }
1011f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian#endif
10129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glEnable(GL_TEXTURE_2D);
10139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glBindTexture(GL_TEXTURE_2D, mWormholeTexName);
10149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
10159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glMatrixMode(GL_TEXTURE);
10169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glLoadIdentity();
10179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glScalef(width*(1.0f/32.0f), height*(1.0f/32.0f), 1);
10186158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator it = region.begin();
10196158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator const end = region.end();
10206158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        while (it != end) {
10216158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian            const Rect& r = *it++;
10229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const GLint sy = height - (r.top + r.height());
10239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glScissor(r.left, sy, r.width(), r.height());
10249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
10259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glDisableClientState(GL_TEXTURE_COORD_ARRAY);
102704a709e459db6f15c04b00bcd3b030c90ca52949Mathias Agopian        glLoadIdentity();
102804a709e459db6f15c04b00bcd3b030c90ca52949Mathias Agopian        glMatrixMode(GL_MODELVIEW);
10299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::debugShowFPS() const
10339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int mFrameCount;
10359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int mLastFrameCount = 0;
10369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static nsecs_t mLastFpsTime = 0;
10379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static float mFps = 0;
10389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mFrameCount++;
10399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    nsecs_t now = systemTime();
10409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    nsecs_t diff = now - mLastFpsTime;
10419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (diff > ms2ns(250)) {
10429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mFps =  ((mFrameCount - mLastFrameCount) * float(s2ns(1))) / diff;
10439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mLastFpsTime = now;
10449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mLastFrameCount = mFrameCount;
10459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // XXX: mFPS has the value we want
10479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project }
10489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10491473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::addLayer(const sp<LayerBase>& layer)
10509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
10529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    addLayer_l(layer);
10539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded|eTraversalNeeded);
10549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
10559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10571473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::addLayer_l(const sp<LayerBase>& layer)
10589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10591efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian    ssize_t i = mCurrentState.layersSortedByZ.add(layer);
10609bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian    return (i < 0) ? status_t(i) : status_t(NO_ERROR);
10619bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian}
10629bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1063593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianssize_t SurfaceFlinger::addClientLayer(const sp<Client>& client,
1064593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const sp<LayerBaseClient>& lbc)
10659bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian{
1066593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    Mutex::Autolock _l(mStateLock);
1067593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
1068593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // attach this layer to the client
1069593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    ssize_t name = client->attachLayer(lbc);
1070593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
1071593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // add this layer to the current state list
1072593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    addLayer_l(lbc);
1073593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
1074593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return name;
1075593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian}
1076593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
1077593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::removeLayer(const sp<LayerBase>& layer)
1078593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
1079593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    Mutex::Autolock _l(mStateLock);
1080593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    status_t err = purgatorizeLayer_l(layer);
1081593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (err == NO_ERROR)
1082593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        setTransactionFlags(eTransactionNeeded);
1083593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return err;
10849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10861473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::removeLayer_l(const sp<LayerBase>& layerBase)
10879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10887623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<LayerBaseClient> lbc(layerBase->getLayerBaseClient());
10897623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (lbc != 0) {
10907623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        mLayerMap.removeItem( lbc->getSurface()->asBinder() );
10917623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
10929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    ssize_t index = mCurrentState.layersSortedByZ.remove(layerBase);
10939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (index >= 0) {
10941473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        mLayersRemoved = true;
10959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return NO_ERROR;
10969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10972d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    return status_t(index);
10989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11006cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopianstatus_t SurfaceFlinger::purgatorizeLayer_l(const sp<LayerBase>& layerBase)
11016cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian{
11022e4b68d57bb64d7e93139238c5a8be91ff956c2aMathias Agopian    // remove the layer from the main list (through a transaction).
11036cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    ssize_t err = removeLayer_l(layerBase);
11042e4b68d57bb64d7e93139238c5a8be91ff956c2aMathias Agopian
11050c4cec7e4df87181486d280c98fba9c0f4774c37Mathias Agopian    layerBase->onRemoved();
11060c4cec7e4df87181486d280c98fba9c0f4774c37Mathias Agopian
11072d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    // it's possible that we don't find a layer, because it might
11082d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    // have been destroyed already -- this is not technically an error
1109593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // from the user because there is a race between Client::destroySurface(),
1110593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // ~Client() and ~ISurface().
11116cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    return (err == NAME_NOT_FOUND) ? status_t(NO_ERROR) : err;
11126cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian}
11136cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
1114593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::invalidateLayerVisibility(const sp<LayerBase>& layer)
11159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1116593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    layer->forceVisibilityTransaction();
1117593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    setTransactionFlags(eTraversalNeeded);
1118593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return NO_ERROR;
11199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectuint32_t SurfaceFlinger::getTransactionFlags(uint32_t flags)
11229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return android_atomic_and(~flags, &mTransactionFlags) & flags;
11249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1126898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianuint32_t SurfaceFlinger::setTransactionFlags(uint32_t flags)
11279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t old = android_atomic_or(flags, &mTransactionFlags);
11299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if ((old & flags)==0) { // wake the server up
1130898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        signalEvent();
11319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return old;
11339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::openGlobalTransaction()
11369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    android_atomic_inc(&mTransactionCount);
11389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::closeGlobalTransaction()
11419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (android_atomic_dec(&mTransactionCount) == 1) {
11439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        signalEvent();
11449779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian
1145e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber        // if there is a transaction with a resize, wait for it to
11469779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        // take effect before returning.
11479779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        Mutex::Autolock _l(mStateLock);
11489779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        while (mResizeTransationPending) {
114998a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian            status_t err = mTransactionCV.waitRelative(mStateLock, s2ns(5));
115098a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian            if (CC_UNLIKELY(err != NO_ERROR)) {
115198a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                // just in case something goes wrong in SF, return to the
115298a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                // called after a few seconds.
115398a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                LOGW_IF(err == TIMED_OUT, "closeGlobalTransaction timed out!");
115498a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                mResizeTransationPending = false;
115598a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                break;
115698a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian            }
11579779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        }
11589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::freezeDisplay(DisplayID dpy, uint32_t flags)
11629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
11649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
11659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
11679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mCurrentState.freezeDisplay = 1;
11689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded);
11699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // flags is intended to communicate some sort of animation behavior
1171ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian    // (for instance fading)
11729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
11739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::unfreezeDisplay(DisplayID dpy, uint32_t flags)
11769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
11789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
11799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
11819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mCurrentState.freezeDisplay = 0;
11829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded);
11839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // flags is intended to communicate some sort of animation behavior
1185ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian    // (for instance fading)
11869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
11879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1189e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huberint SurfaceFlinger::setOrientation(DisplayID dpy,
1190eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian        int orientation, uint32_t flags)
11919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
11939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
11949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
11969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (mCurrentState.orientation != orientation) {
11979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (uint32_t(orientation)<=eOrientation270 || orientation==42) {
1198eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian            mCurrentState.orientationType = flags;
11999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mCurrentState.orientation = orientation;
12009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            setTransactionFlags(eTransactionNeeded);
12019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mTransactionCV.wait(mStateLock);
12029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
12039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            orientation = BAD_VALUE;
12049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
12059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return orientation;
12079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1209593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopiansp<ISurface> SurfaceFlinger::createSurface(const sp<Client>& client, int pid,
1210770492cb2b19f6a36ad748cd05fbedfbb9a67dfaMathias Agopian        const String8& name, ISurfaceComposerClient::surface_data_t* params,
12119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        DisplayID d, uint32_t w, uint32_t h, PixelFormat format,
12129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        uint32_t flags)
12139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
12141473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBaseClient> layer;
12159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    sp<LayerBaseClient::Surface> surfaceHandle;
12164d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian
12174d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian    if (int32_t(w|h) < 0) {
12184d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian        LOGE("createSurface() failed, w or h is negative (w=%d, h=%d)",
12194d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian                int(w), int(h));
12204d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian        return surfaceHandle;
12214d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian    }
1222e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
12239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    //LOGD("createSurface for pid %d (%d x %d)", pid, w, h);
12247623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<Layer> normalLayer;
12259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (flags & eFXSurfaceMask) {
12269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceNormal:
1227d2112306330ce0c162bee4b864991962ca2b655aMathias Agopian            normalLayer = createNormalSurface(client, d, w, h, flags, format);
1228d2112306330ce0c162bee4b864991962ca2b655aMathias Agopian            layer = normalLayer;
12299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
12309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceBlur:
1231c3802d22f11b4e513ba776277d18f315c5c769f7Mathias Agopian            // for now we treat Blur as Dim, until we can implement it
1232c3802d22f11b4e513ba776277d18f315c5c769f7Mathias Agopian            // efficiently.
12339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceDim:
1234593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            layer = createDimSurface(client, d, w, h, flags);
12359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
12369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12381473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    if (layer != 0) {
1239593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        layer->initStates(w, h, flags);
12405d26c1e38dabb3ad8b4b6e1000375f3b1a6b7693Mathias Agopian        layer->setName(name);
1241593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        ssize_t token = addClientLayer(client, layer);
12427623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
12439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        surfaceHandle = layer->getSurface();
1244e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber        if (surfaceHandle != 0) {
1245593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            params->token = token;
124618b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->identity = surfaceHandle->getIdentity();
124718b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->width = w;
124818b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->height = h;
124918b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->format = format;
12507623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            if (normalLayer != 0) {
12517623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian                Mutex::Autolock _l(mStateLock);
12527623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian                mLayerMap.add(surfaceHandle->asBinder(), normalLayer);
12537623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            }
125418b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian        }
12557623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
1256593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        setTransactionFlags(eTransactionNeeded);
12579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return surfaceHandle;
12609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12627623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<Layer> SurfaceFlinger::createNormalSurface(
12636edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian        const sp<Client>& client, DisplayID display,
1264593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        uint32_t w, uint32_t h, uint32_t flags,
126518b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian        PixelFormat& format)
12669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
12679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize the surfaces
12689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (format) { // TODO: take h/w into account
12699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_TRANSPARENT:
12709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_TRANSLUCENT:
12719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        format = PIXEL_FORMAT_RGBA_8888;
12729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
12739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_OPAQUE:
12744cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#ifdef NO_RGBX_8888
12754cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian        format = PIXEL_FORMAT_RGB_565;
12764cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#else
127759962ce3b0d8b7efec2840accca8fb7a6066f2bdMathias Agopian        format = PIXEL_FORMAT_RGBX_8888;
12784cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#endif
12799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
12809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12824cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#ifdef NO_RGBX_8888
12834cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian    if (format == PIXEL_FORMAT_RGBX_8888)
12844cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian        format = PIXEL_FORMAT_RGBA_8888;
12854cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#endif
12864cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian
1287593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<Layer> layer = new Layer(this, display, client);
12886edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian    status_t err = layer->setBuffers(w, h, format, flags);
1289593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (LIKELY(err != NO_ERROR)) {
12909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        LOGE("createNormalSurfaceLocked() failed (%s)", strerror(-err));
12911473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        layer.clear();
12929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return layer;
12949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12967623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<LayerDim> SurfaceFlinger::createDimSurface(
12976edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian        const sp<Client>& client, DisplayID display,
1298593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        uint32_t w, uint32_t h, uint32_t flags)
12999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1300593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<LayerDim> layer = new LayerDim(this, display, client);
13019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    layer->initStates(w, h, flags);
13029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return layer;
13039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1305593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::removeSurface(const sp<Client>& client, SurfaceID sid)
13066cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian{
13076cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    /*
13086cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     * called by the window manager, when a surface should be marked for
13096cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     * destruction.
1310e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber     *
1311a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * The surface is removed from the current and drawing lists, but placed
1312a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * in the purgatory queue, so it's not destroyed right-away (we need
1313a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * to wait for all client's references to go away first).
13146cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     */
13156cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
1316248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian    status_t err = NAME_NOT_FOUND;
1317a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian    Mutex::Autolock _l(mStateLock);
1318593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<LayerBaseClient> layer = client->getLayerUser(sid);
1319248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian    if (layer != 0) {
1320248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        err = purgatorizeLayer_l(layer);
1321248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        if (err == NO_ERROR) {
1322248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian            setTransactionFlags(eTransactionNeeded);
1323248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        }
13246cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    }
13256cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    return err;
13266cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian}
13276cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
13286cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopianstatus_t SurfaceFlinger::destroySurface(const sp<LayerBaseClient>& layer)
13299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1330359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian    // called by ~ISurface() when all references are gone
1331e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
13326ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    class MessageDestroySurface : public MessageBase {
1333a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        SurfaceFlinger* flinger;
13346ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        sp<LayerBaseClient> layer;
13356ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    public:
1336a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        MessageDestroySurface(
1337a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                SurfaceFlinger* flinger, const sp<LayerBaseClient>& layer)
1338a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            : flinger(flinger), layer(layer) { }
13396ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        virtual bool handler() {
1340c8fb5b1979da4829e1486e6a1008c06c979b94b0Mathias Agopian            sp<LayerBaseClient> l(layer);
1341c8fb5b1979da4829e1486e6a1008c06c979b94b0Mathias Agopian            layer.clear(); // clear it outside of the lock;
13426ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
1343359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian            /*
1344e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber             * remove the layer from the current list -- chances are that it's
1345e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber             * not in the list anyway, because it should have been removed
1346e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber             * already upon request of the client (eg: window manager).
1347359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             * However, a buggy client could have not done that.
1348359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             * Since we know we don't have any more clients, we don't need
1349359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             * to use the purgatory.
1350359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             */
1351c8fb5b1979da4829e1486e6a1008c06c979b94b0Mathias Agopian            status_t err = flinger->removeLayer_l(l);
13522e4b68d57bb64d7e93139238c5a8be91ff956c2aMathias Agopian            LOGE_IF(err<0 && err != NAME_NOT_FOUND,
13532e4b68d57bb64d7e93139238c5a8be91ff956c2aMathias Agopian                    "error removing layer=%p (%s)", l.get(), strerror(-err));
13546ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            return true;
13556ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        }
13566ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    };
13572d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian
1358898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    postMessageAsync( new MessageDestroySurface(this, layer) );
13599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
13609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::setClientState(
1363593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const sp<Client>& client,
13649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int32_t count,
13659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const layer_state_t* states)
13669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
13679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
13689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t flags = 0;
13699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (int i=0 ; i<count ; i++) {
1370593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const layer_state_t& s(states[i]);
1371593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        sp<LayerBaseClient> layer(client->getLayerUser(s.surface));
13721473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        if (layer != 0) {
13739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const uint32_t what = s.what;
13749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & ePositionChanged) {
13759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setPosition(s.x, s.y))
13769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
13779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
13789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eLayerChanged) {
13791efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian                ssize_t idx = mCurrentState.layersSortedByZ.indexOf(layer);
13809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setLayer(s.z)) {
13811efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian                    mCurrentState.layersSortedByZ.removeAt(idx);
13821efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian                    mCurrentState.layersSortedByZ.add(layer);
13839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // we need traversal (state changed)
13849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // AND transaction (list changed)
13859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTransactionNeeded|eTraversalNeeded;
13869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
13879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
13889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eSizeChanged) {
13899779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                if (layer->setSize(s.w, s.h)) {
13909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
13919779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                    mResizeTransationPending = true;
13929779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                }
13939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
13949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eAlphaChanged) {
13959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setAlpha(uint8_t(255.0f*s.alpha+0.5f)))
13969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
13979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
13989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eMatrixChanged) {
13999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setMatrix(s.matrix))
14009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eTransparentRegionChanged) {
14039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setTransparentRegionHint(s.transparentRegion))
14049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eVisibilityChanged) {
14079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setFlags(s.flags, s.mask))
14089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
14119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
14129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (flags) {
14139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setTransactionFlags(flags);
14149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
14159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
14169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
14189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::screenReleased(int dpy)
14199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
14209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // this may be called by a signal handler, we can't do too much in here
14219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    android_atomic_or(eConsoleReleased, &mConsoleSignals);
14229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    signalEvent();
14239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
14259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::screenAcquired(int dpy)
14269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
14279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // this may be called by a signal handler, we can't do too much in here
14289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    android_atomic_or(eConsoleAcquired, &mConsoleSignals);
14299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    signalEvent();
14309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
14329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::dump(int fd, const Vector<String16>& args)
14339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
143494720d7ec40d2b5ee200d95a46588cdb0fb30e66Erik Gilling    const size_t SIZE = 4096;
14359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    char buffer[SIZE];
14369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    String8 result;
1437151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian    if (!mDump.checkCalling()) {
14389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        snprintf(buffer, SIZE, "Permission Denial: "
14399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                "can't dump SurfaceFlinger from pid=%d, uid=%d\n",
14409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                IPCThreadState::self()->getCallingPid(),
14419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                IPCThreadState::self()->getCallingUid());
14429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        result.append(buffer);
14439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
1444a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1445a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // figure out if we're stuck somewhere
1446a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t now = systemTime();
1447a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t inSwapBuffers(mDebugInSwapBuffers);
1448a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t inTransaction(mDebugInTransaction);
1449a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        nsecs_t inSwapBuffersDuration = (inSwapBuffers) ? now-inSwapBuffers : 0;
1450a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        nsecs_t inTransactionDuration = (inTransaction) ? now-inTransaction : 0;
1451a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1452a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // Try to get the main lock, but don't insist if we can't
1453a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // (this would indicate SF is stuck, but we want to be able to
1454a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // print something in dumpsys).
1455a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        int retry = 3;
1456a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        while (mStateLock.tryLock()<0 && --retry>=0) {
1457a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            usleep(1000000);
1458a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
1459a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const bool locked(retry >= 0);
1460a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (!locked) {
1461e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber            snprintf(buffer, SIZE,
1462a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    "SurfaceFlinger appears to be unresponsive, "
1463a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    "dumping anyways (no locks held)\n");
1464a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1465a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
1466a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
14679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const LayerVector& currentLayers = mCurrentState.layersSortedByZ;
14689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const size_t count = currentLayers.size();
14699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (size_t i=0 ; i<count ; i++) {
14709bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            const sp<LayerBase>& layer(currentLayers[i]);
14719bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            layer->dump(result, buffer, SIZE);
14729bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            const Layer::State& s(layer->drawingState());
14739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            s.transparentRegion.dump(result, "transparentRegion");
14749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->transparentRegionScreen.dump(result, "transparentRegionScreen");
14759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->visibleRegionScreen.dump(result, "visibleRegionScreen");
14769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
14779bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
14789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mWormholeRegion.dump(result, "WormholeRegion");
14799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw(graphicPlane(0).displayHardware());
14809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        snprintf(buffer, SIZE,
14819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                "  display frozen: %s, freezeCount=%d, orientation=%d, canDraw=%d\n",
14829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeDisplay?"yes":"no", mFreezeCount,
14839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mCurrentState.orientation, hw.canDraw());
14849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        result.append(buffer);
1485a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        snprintf(buffer, SIZE,
1486a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                "  last eglSwapBuffers() time: %f us\n"
1487a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                "  last transaction time     : %f us\n",
1488a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                mLastSwapBufferTime/1000.0, mLastTransactionTime/1000.0);
1489a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        result.append(buffer);
14909bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1491a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (inSwapBuffersDuration || !locked) {
1492a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            snprintf(buffer, SIZE, "  eglSwapBuffers time: %f us\n",
1493a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    inSwapBuffersDuration/1000.0);
1494a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1495a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
14969bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1497a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (inTransactionDuration || !locked) {
1498a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            snprintf(buffer, SIZE, "  transaction time: %f us\n",
1499a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    inTransactionDuration/1000.0);
1500a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1501a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
15029bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
15036a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        HWComposer& hwc(hw.getHwComposer());
15046a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        snprintf(buffer, SIZE, "  h/w composer %s and %s\n",
15056a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                hwc.initCheck()==NO_ERROR ? "present" : "not present",
15066a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                mDebugDisableHWC ? "disabled" : "enabled");
15076a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        result.append(buffer);
150890da4dd2a20c11fa2f9b860905c867a64b2b299eMathias Agopian        hwc.dump(result, buffer, SIZE);
15096a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian
15106950e428feaccc8164b989ef64e771a99948797aMathias Agopian        const GraphicBufferAllocator& alloc(GraphicBufferAllocator::get());
15111473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        alloc.dump(result);
151294720d7ec40d2b5ee200d95a46588cdb0fb30e66Erik Gilling        hw.dump(result);
1513a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1514a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (locked) {
1515a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            mStateLock.unlock();
1516a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
15179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
15189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    write(fd, result.string(), result.size());
15199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
15209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
15219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
15229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::onTransact(
15239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
15249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
15259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (code) {
15269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case CREATE_CONNECTION:
15279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case OPEN_GLOBAL_TRANSACTION:
15289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case CLOSE_GLOBAL_TRANSACTION:
15299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case SET_ORIENTATION:
15309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case FREEZE_DISPLAY:
15319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case UNFREEZE_DISPLAY:
15329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case BOOT_FINISHED:
1533aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        case TURN_ELECTRON_BEAM_OFF:
15342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        case TURN_ELECTRON_BEAM_ON:
15359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        {
15369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // codes that require permission check
15379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            IPCThreadState* ipc = IPCThreadState::self();
15389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int pid = ipc->getCallingPid();
1539627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian            const int uid = ipc->getCallingUid();
1540151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            if ((uid != AID_GRAPHICS) && !mAccessSurfaceFlinger.check(pid, uid)) {
1541151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                LOGE("Permission Denial: "
1542151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                        "can't access SurfaceFlinger pid=%d, uid=%d", pid, uid);
1543151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                return PERMISSION_DENIED;
15449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
1545ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            break;
1546ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
1547ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        case CAPTURE_SCREEN:
1548ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        {
1549ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            // codes that require permission check
1550ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            IPCThreadState* ipc = IPCThreadState::self();
1551ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            const int pid = ipc->getCallingPid();
1552ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            const int uid = ipc->getCallingUid();
1553ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            if ((uid != AID_GRAPHICS) && !mReadFramebuffer.check(pid, uid)) {
1554ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                LOGE("Permission Denial: "
1555ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                        "can't read framebuffer pid=%d, uid=%d", pid, uid);
1556ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                return PERMISSION_DENIED;
1557ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            }
1558ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            break;
15599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
15609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1561ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
15629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    status_t err = BnSurfaceComposer::onTransact(code, data, reply, flags);
15639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (err == UNKNOWN_TRANSACTION || err == PERMISSION_DENIED) {
15648c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        CHECK_INTERFACE(ISurfaceComposer, data, reply);
1565151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian        if (UNLIKELY(!mHardwareTest.checkCalling())) {
1566151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            IPCThreadState* ipc = IPCThreadState::self();
1567151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            const int pid = ipc->getCallingPid();
1568151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            const int uid = ipc->getCallingUid();
1569151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            LOGE("Permission Denial: "
1570151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                    "can't access SurfaceFlinger pid=%d, uid=%d", pid, uid);
15719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return PERMISSION_DENIED;
15729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
15739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int n;
15749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (code) {
157517f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian            case 1000: // SHOW_CPU, NOT SUPPORTED ANYMORE
157604262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            case 1001: // SHOW_FPS, NOT SUPPORTED ANYMORE
15779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
15789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1002:  // SHOW_UPDATES
15799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                n = data.readInt32();
15809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDebugRegion = n ? n : (mDebugRegion ? 0 : 1);
15819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
15829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1003:  // SHOW_BACKGROUND
15839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                n = data.readInt32();
15849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDebugBackground = n ? 1 : 0;
15859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
15866a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian            case 1008:  // toggle use of hw composer
15876a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                n = data.readInt32();
15886a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                mDebugDisableHWC = n ? 1 : 0;
15896a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                mHwWorkListDirty = true;
15906a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                // fall-through...
15919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1004:{ // repaint everything
15929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                Mutex::Autolock _l(mStateLock);
15939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                const DisplayHardware& hw(graphicPlane(0).displayHardware());
15949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDirtyRegion.set(hw.bounds()); // careful that's not thread-safe
15959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                signalEvent();
15969779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                return NO_ERROR;
15979779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian            }
15989779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian            case 1005:{ // force transaction
15999779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                setTransactionFlags(eTransactionNeeded|eTraversalNeeded);
16009779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                return NO_ERROR;
16019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
160204262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            case 1006:{ // enable/disable GraphicLog
160304262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                int enabled = data.readInt32();
160404262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                GraphicLog::getInstance().setEnabled(enabled);
160504262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                return NO_ERROR;
160604262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            }
16079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1007: // set mFreezeCount
16089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeCount = data.readInt32();
16090e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                mFreezeDisplayTime = 0;
16109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1010:  // interrogate.
161217f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian                reply->writeInt32(0);
16139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(0);
16149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(mDebugRegion);
16159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(mDebugBackground);
16169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1013: {
16189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                Mutex::Autolock _l(mStateLock);
16199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                const DisplayHardware& hw(graphicPlane(0).displayHardware());
16209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(hw.getPageFlipCount());
16219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
16229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return NO_ERROR;
16239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
16249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
16259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return err;
16269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
16279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1628aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian// ---------------------------------------------------------------------------
1629aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
16302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::renderScreenToTextureLocked(DisplayID dpy,
16312d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        GLuint* textureName, GLfloat* uOut, GLfloat* vOut)
1632aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian{
1633aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
1634aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        return INVALID_OPERATION;
1635aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1636aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // get screen geometry
1637aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const DisplayHardware& hw(graphicPlane(dpy).displayHardware());
1638aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const uint32_t hw_w = hw.getWidth();
1639aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const uint32_t hw_h = hw.getHeight();
1640aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLfloat u = 1;
1641aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLfloat v = 1;
1642aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1643aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // make sure to clear all GL error flags
1644aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    while ( glGetError() != GL_NO_ERROR ) ;
1645aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1646aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // create a FBO
1647aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLuint name, tname;
1648aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glGenTextures(1, &tname);
1649aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glBindTexture(GL_TEXTURE_2D, tname);
16502d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB,
16512d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            hw_w, hw_h, 0, GL_RGB, GL_UNSIGNED_BYTE, 0);
1652aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (glGetError() != GL_NO_ERROR) {
1653a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian        while ( glGetError() != GL_NO_ERROR ) ;
1654aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        GLint tw = (2 << (31 - clz(hw_w)));
1655aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        GLint th = (2 << (31 - clz(hw_h)));
16562d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB,
16572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian                tw, th, 0, GL_RGB, GL_UNSIGNED_BYTE, 0);
1658aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        u = GLfloat(hw_w) / tw;
1659aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        v = GLfloat(hw_h) / th;
1660aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
1661aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glGenFramebuffersOES(1, &name);
1662aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, name);
16632d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glFramebufferTexture2DOES(GL_FRAMEBUFFER_OES,
16642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            GL_COLOR_ATTACHMENT0_OES, GL_TEXTURE_2D, tname, 0);
1665aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
16662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // redraw the screen entirely...
16672d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glClearColor(0,0,0,1);
16682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glClear(GL_COLOR_BUFFER_BIT);
16692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
16702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const size_t count = layers.size();
16712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (size_t i=0 ; i<count ; ++i) {
16722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
16732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        layer->drawForSreenShot();
16742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
1675aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
16762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // back to main framebuffer
16772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
16782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_SCISSOR_TEST);
16792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDeleteFramebuffersOES(1, &name);
1680aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
16812d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    *textureName = tname;
16822d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    *uOut = u;
16832d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    *vOut = v;
16842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
16852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
1686aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
16872d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian// ---------------------------------------------------------------------------
1688aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
16892d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::electronBeamOffAnimationImplLocked()
16902d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
16912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    status_t result = PERMISSION_DENIED;
1692aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
16932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
16942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return INVALID_OPERATION;
1695aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
16962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // get screen geometry
16972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
16982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_w = hw.getWidth();
16992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_h = hw.getHeight();
17002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const Region screenBounds(hw.bounds());
1701aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat u, v;
17032d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLuint tname;
17042d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    result = renderScreenToTextureLocked(0, &tname, &u, &v);
17052d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (result != NO_ERROR) {
17062d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return result;
17072d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
1708aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat vtx[8];
17102d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const GLfloat texCoords[4][2] = { {0,v}, {0,0}, {u,0}, {u,v} };
17112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_TEXTURE_2D);
17122d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindTexture(GL_TEXTURE_2D, tname);
17132d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
17142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
17152d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
17162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexCoordPointer(2, GL_FLOAT, 0, texCoords);
17172d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnableClientState(GL_TEXTURE_COORD_ARRAY);
17182d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glVertexPointer(2, GL_FLOAT, 0, vtx);
17192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
17202d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class s_curve_interpolator {
17212d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float nbFrames, s, v;
17222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
17232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        s_curve_interpolator(int nbFrames, float s)
17242d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : nbFrames(1.0f / (nbFrames-1)), s(s),
17252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian          v(1.0f + expf(-s + 0.5f*s)) {
17262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
17272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float operator()(int f) {
17282d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const float x = f * nbFrames;
17292d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return ((1.0f/(1.0f + expf(-x*s + 0.5f*s))) - 0.5f) * v + 0.5f;
17302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
17312d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
17322d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
17332d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class v_stretch {
17342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
17352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
17362d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        v_stretch(uint32_t hw_w, uint32_t hw_h)
17372d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
17382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
17392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
17402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w + (hw_w * v);
17412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = hw_h - (hw_h * v);
17422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
17432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
17442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
17452d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
17462d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
17472d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
17482d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
17492d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
17502d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
17512d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class h_stretch {
17522d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
17532d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
17542d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        h_stretch(uint32_t hw_w, uint32_t hw_h)
17552d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
1756aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
17572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
17582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w - (hw_w * v);
17592d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = 1.0f;
17602d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
17612d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
17622d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
17632d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
17642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
17652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
17662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
17672d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
17682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
17692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // the full animation is 24 frames
17702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const int nbFrames = 12;
17712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itr(nbFrames, 7.5f);
17722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itg(nbFrames, 8.0f);
17732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itb(nbFrames, 8.5f);
17742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
17752d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    v_stretch vverts(hw_w, hw_h);
17762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_BLEND);
17772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBlendFunc(GL_ONE, GL_ONE);
17782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=0 ; i<nbFrames ; i++) {
17792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float x, y, w, h;
17802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vr = itr(i);
17812d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vg = itg(i);
17822d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vb = itb(i);
17832d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
17842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // clear screen
17852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(1,1,1,1);
17862d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
17872d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glEnable(GL_TEXTURE_2D);
1788aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17892d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the red plane
17902d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vr);
17912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(1,0,0,1);
17922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
17932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
17942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the green plane
17952d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vg);
17962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,1,0,1);
17972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
17982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
17992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the blue plane
18002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vb);
18012d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,0,1,1);
18022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
18032d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18042d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the white highlight (we use the last vertices)
1805aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glDisable(GL_TEXTURE_2D);
1806aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glColorMask(1,1,1,1);
18072d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColor4f(vg, vg, vg, 1);
18082d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
18092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
18102d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
18112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18122d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    h_stretch hverts(hw_w, hw_h);
18132d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_BLEND);
18142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_TEXTURE_2D);
18152d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
18162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=0 ; i<nbFrames ; i++) {
18172d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float v = itg(i);
18182d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hverts(vtx, v);
18192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
18202d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColor4f(1-v, 1-v, 1-v, 1);
18212d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
18222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
18232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
18242d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
18262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_SCISSOR_TEST);
18272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisableClientState(GL_TEXTURE_COORD_ARRAY);
18282d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDeleteTextures(1, &tname);
18292d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
18302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
18312d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18322d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::electronBeamOnAnimationImplLocked()
18332d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
18342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    status_t result = PERMISSION_DENIED;
18352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18362d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
18372d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return INVALID_OPERATION;
18382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // get screen geometry
18412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
18422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_w = hw.getWidth();
18432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_h = hw.getHeight();
18442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const Region screenBounds(hw.bounds());
18452d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18462d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat u, v;
18472d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLuint tname;
18482d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    result = renderScreenToTextureLocked(0, &tname, &u, &v);
18492d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (result != NO_ERROR) {
18502d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return result;
18512d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
18522d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18532d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // back to main framebuffer
18542d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
18552d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_SCISSOR_TEST);
18562d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat vtx[8];
18582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const GLfloat texCoords[4][2] = { {0,v}, {0,0}, {u,0}, {u,v} };
18592d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_TEXTURE_2D);
18602d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindTexture(GL_TEXTURE_2D, tname);
18612d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE);
18622d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
18632d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
18642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexCoordPointer(2, GL_FLOAT, 0, texCoords);
18652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnableClientState(GL_TEXTURE_COORD_ARRAY);
18662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glVertexPointer(2, GL_FLOAT, 0, vtx);
18672d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class s_curve_interpolator {
18692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float nbFrames, s, v;
18702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
18712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        s_curve_interpolator(int nbFrames, float s)
18722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : nbFrames(1.0f / (nbFrames-1)), s(s),
18732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian          v(1.0f + expf(-s + 0.5f*s)) {
18742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
18752d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float operator()(int f) {
18762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const float x = f * nbFrames;
18772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return ((1.0f/(1.0f + expf(-x*s + 0.5f*s))) - 0.5f) * v + 0.5f;
18782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
18792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
18802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18812d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class v_stretch {
18822d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
18832d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
18842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        v_stretch(uint32_t hw_w, uint32_t hw_h)
18852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
1886aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
18872d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
18882d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w + (hw_w * v);
18892d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = hw_h - (hw_h * v);
18902d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
18912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
18922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
18932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
18942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
18952d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
18962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
18972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
18982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class h_stretch {
19002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
19012d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
19022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        h_stretch(uint32_t hw_w, uint32_t hw_h)
19032d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
19042d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19052d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
19062d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w - (hw_w * v);
19072d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = 1.0f;
19082d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
19092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
19102d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
19112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
19122d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
19132d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
19142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19152d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
19162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
1917dfa08fbfd8a5ebd013c122e391282b7f21f6342fMathias Agopian    // the full animation is 12 frames
1918dfa08fbfd8a5ebd013c122e391282b7f21f6342fMathias Agopian    int nbFrames = 8;
19192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itr(nbFrames, 7.5f);
19202d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itg(nbFrames, 8.0f);
19212d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itb(nbFrames, 8.5f);
19222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    h_stretch hverts(hw_w, hw_h);
19242d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_BLEND);
19252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_TEXTURE_2D);
19262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
19272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=nbFrames-1 ; i>=0 ; i--) {
19282d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float v = itg(i);
19292d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hverts(vtx, v);
19302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
19312d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColor4f(1-v, 1-v, 1-v, 1);
19322d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
19332d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
19342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
19352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
1936dfa08fbfd8a5ebd013c122e391282b7f21f6342fMathias Agopian    nbFrames = 4;
19372d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    v_stretch vverts(hw_w, hw_h);
19382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_BLEND);
19392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBlendFunc(GL_ONE, GL_ONE);
19402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=nbFrames-1 ; i>=0 ; i--) {
19412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float x, y, w, h;
19422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vr = itr(i);
19432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vg = itg(i);
19442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vb = itb(i);
1945aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
19462d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // clear screen
1947aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glColorMask(1,1,1,1);
19482d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
19492d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glEnable(GL_TEXTURE_2D);
19502d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19512d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the red plane
19522d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vr);
19532d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(1,0,0,1);
19542d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
19552d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19562d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the green plane
19572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vg);
19582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,1,0,1);
19592d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
19602d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19612d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the blue plane
19622d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vb);
19632d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,0,1,1);
19642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
19652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
1967aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
1968aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
19692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
19702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_SCISSOR_TEST);
19712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisableClientState(GL_TEXTURE_COORD_ARRAY);
1972aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glDeleteTextures(1, &tname);
1973aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
19742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
19752d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
19762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian// ---------------------------------------------------------------------------
19782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
1979d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOffImplLocked(int32_t mode)
19802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
19812d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    DisplayHardware& hw(graphicPlane(0).editDisplayHardware());
19822d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (!hw.canDraw()) {
19832d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // we're already off
19842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return NO_ERROR;
19852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
1986d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    if (mode & ISurfaceComposer::eElectronBeamAnimationOff) {
1987d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        electronBeamOffAnimationImplLocked();
1988d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    }
1989d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian
1990d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    // always clear the whole screen at the end of the animation
1991d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glClearColor(0,0,0,1);
1992d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glDisable(GL_SCISSOR_TEST);
1993d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glClear(GL_COLOR_BUFFER_BIT);
1994d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glEnable(GL_SCISSOR_TEST);
1995d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    hw.flip( Region(hw.bounds()) );
1996d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian
1997a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    hw.setCanDraw(false);
1998a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    return NO_ERROR;
1999aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
2000aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
2001aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOff(int32_t mode)
2002aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian{
2003aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    class MessageTurnElectronBeamOff : public MessageBase {
2004aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        SurfaceFlinger* flinger;
2005d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        int32_t mode;
2006aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        status_t result;
2007aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    public:
2008d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        MessageTurnElectronBeamOff(SurfaceFlinger* flinger, int32_t mode)
2009d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            : flinger(flinger), mode(mode), result(PERMISSION_DENIED) {
2010aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
2011aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        status_t getResult() const {
2012aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            return result;
2013aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
2014aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        virtual bool handler() {
2015aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
2016d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            result = flinger->turnElectronBeamOffImplLocked(mode);
2017aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            return true;
2018aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
2019aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    };
2020aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
2021d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    sp<MessageBase> msg = new MessageTurnElectronBeamOff(this, mode);
2022aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    status_t res = postMessageSync(msg);
2023aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (res == NO_ERROR) {
2024aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        res = static_cast<MessageTurnElectronBeamOff*>( msg.get() )->getResult();
20252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // work-around: when the power-manager calls us we activate the
20272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // animation. eventually, the "on" animation will be called
20282d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // by the power-manager itself
2029d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        mElectronBeamAnimationMode = mode;
2030aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
2031aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    return res;
2032aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
2033aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
20349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
20359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2036d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOnImplLocked(int32_t mode)
20372d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
20382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    DisplayHardware& hw(graphicPlane(0).editDisplayHardware());
20392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (hw.canDraw()) {
20402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // we're already on
20412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return NO_ERROR;
20422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
2043d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    if (mode & ISurfaceComposer::eElectronBeamAnimationOn) {
2044d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        electronBeamOnAnimationImplLocked();
2045d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    }
2046a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    hw.setCanDraw(true);
20478b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian
20488b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian    // make sure to redraw the whole screen when the animation is done
20498b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian    mDirtyRegion.set(hw.bounds());
20508b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian    signalEvent();
20518b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian
2052a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    return NO_ERROR;
20532d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
20542d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20552d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOn(int32_t mode)
20562d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
20572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class MessageTurnElectronBeamOn : public MessageBase {
20582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        SurfaceFlinger* flinger;
2059d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        int32_t mode;
20602d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        status_t result;
20612d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
2062d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        MessageTurnElectronBeamOn(SurfaceFlinger* flinger, int32_t mode)
2063d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            : flinger(flinger), mode(mode), result(PERMISSION_DENIED) {
20642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
20652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        status_t getResult() const {
20662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return result;
20672d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
20682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        virtual bool handler() {
20692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
2070d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            result = flinger->turnElectronBeamOnImplLocked(mode);
20712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return true;
20722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
20732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
20742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
2075d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    postMessageAsync( new MessageTurnElectronBeamOn(this, mode) );
20762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
20772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
20782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian// ---------------------------------------------------------------------------
20802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
208138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopianstatus_t SurfaceFlinger::captureScreenImplLocked(DisplayID dpy,
208238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        sp<IMemoryHeap>* heap,
208338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t* w, uint32_t* h, PixelFormat* f,
20843dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t sw, uint32_t sh,
20853dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t minLayerZ, uint32_t maxLayerZ)
208638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian{
208738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    status_t result = PERMISSION_DENIED;
208838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
208938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // only one display supported for now
209038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
209138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        return BAD_VALUE;
209238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
209338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
209438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        return INVALID_OPERATION;
209538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
209638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // get screen geometry
209738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const DisplayHardware& hw(graphicPlane(dpy).displayHardware());
209838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const uint32_t hw_w = hw.getWidth();
209938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const uint32_t hw_h = hw.getHeight();
210038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
210138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if ((sw > hw_w) || (sh > hw_h))
210238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        return BAD_VALUE;
210338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
210438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    sw = (!sw) ? hw_w : sw;
210538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    sh = (!sh) ? hw_h : sh;
210638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const size_t size = sw * sh * 4;
210738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
210838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // make sure to clear all GL error flags
210938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    while ( glGetError() != GL_NO_ERROR ) ;
211038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
211138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // create a FBO
211238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    GLuint name, tname;
211338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glGenRenderbuffersOES(1, &tname);
211438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glBindRenderbufferOES(GL_RENDERBUFFER_OES, tname);
211538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glRenderbufferStorageOES(GL_RENDERBUFFER_OES, GL_RGBA8_OES, sw, sh);
211638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glGenFramebuffersOES(1, &name);
211738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, name);
211838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES,
211938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            GL_COLOR_ATTACHMENT0_OES, GL_RENDERBUFFER_OES, tname);
212038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
212138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    GLenum status = glCheckFramebufferStatusOES(GL_FRAMEBUFFER_OES);
212238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if (status == GL_FRAMEBUFFER_COMPLETE_OES) {
212338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
212438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // invert everything, b/c glReadPixel() below will invert the FB
212538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glViewport(0, 0, sw, sh);
21261c4e4fc0490c6f07111365cfe21116c31254d5ffMathias Agopian        glScissor(0, 0, sw, sh);
212738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_PROJECTION);
212838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glPushMatrix();
212938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glLoadIdentity();
213038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glOrthof(0, hw_w, 0, hw_h, 0, 1);
213138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_MODELVIEW);
213238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
213338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // redraw the screen entirely...
213438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glClearColor(0,0,0,1);
213538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
21361c4e4fc0490c6f07111365cfe21116c31254d5ffMathias Agopian
213738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
213838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        const size_t count = layers.size();
213938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        for (size_t i=0 ; i<count ; ++i) {
214038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            const sp<LayerBase>& layer(layers[i]);
21413dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            const uint32_t z = layer->drawingState().z;
21423dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            if (z >= minLayerZ && z <= maxLayerZ) {
21433dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                layer->drawForSreenShot();
21443dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            }
214538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        }
214638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
214738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // XXX: this is needed on tegra
214838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glScissor(0, 0, sw, sh);
214938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
215038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // check for errors and return screen capture
215138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        if (glGetError() != GL_NO_ERROR) {
215238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            // error while rendering
215338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            result = INVALID_OPERATION;
215438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        } else {
215538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            // allocate shared memory large enough to hold the
215638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            // screen capture
215738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            sp<MemoryHeapBase> base(
215838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    new MemoryHeapBase(size, 0, "screen-capture") );
215938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            void* const ptr = base->getBase();
216038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            if (ptr) {
216138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                // capture the screen with glReadPixels()
216238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                glReadPixels(0, 0, sw, sh, GL_RGBA, GL_UNSIGNED_BYTE, ptr);
216338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                if (glGetError() == GL_NO_ERROR) {
216438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *heap = base;
216538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *w = sw;
216638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *h = sh;
216738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *f = PIXEL_FORMAT_RGBA_8888;
216838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    result = NO_ERROR;
216938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                }
217038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            } else {
217138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                result = NO_MEMORY;
217238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            }
217338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        }
217438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glEnable(GL_SCISSOR_TEST);
217538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glViewport(0, 0, hw_w, hw_h);
217638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_PROJECTION);
217738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glPopMatrix();
217838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_MODELVIEW);
217938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
218038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
218138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    } else {
218238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        result = BAD_VALUE;
218338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    }
218438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
218538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // release FBO resources
218638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
218738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glDeleteRenderbuffersOES(1, &tname);
218838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glDeleteFramebuffersOES(1, &name);
2189a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian
2190a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian    hw.compositionComplete();
2191a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian
219238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    return result;
219338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian}
219438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
219538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
2196ca5edbeba92b96913291792a4df984e158853b6dMathias Agopianstatus_t SurfaceFlinger::captureScreen(DisplayID dpy,
2197ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        sp<IMemoryHeap>* heap,
219838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t* width, uint32_t* height, PixelFormat* format,
21993dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t sw, uint32_t sh,
22003dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t minLayerZ, uint32_t maxLayerZ)
2201ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian{
2202ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    // only one display supported for now
2203ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
2204ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        return BAD_VALUE;
2205ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2206ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
2207ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        return INVALID_OPERATION;
2208ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2209ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    class MessageCaptureScreen : public MessageBase {
2210ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        SurfaceFlinger* flinger;
2211ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        DisplayID dpy;
2212ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        sp<IMemoryHeap>* heap;
2213ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        uint32_t* w;
2214ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        uint32_t* h;
2215ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        PixelFormat* f;
221638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t sw;
221738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t sh;
22183dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t minLayerZ;
22193dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t maxLayerZ;
2220ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        status_t result;
2221ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    public:
2222ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        MessageCaptureScreen(SurfaceFlinger* flinger, DisplayID dpy,
222338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                sp<IMemoryHeap>* heap, uint32_t* w, uint32_t* h, PixelFormat* f,
22243dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                uint32_t sw, uint32_t sh,
22253dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                uint32_t minLayerZ, uint32_t maxLayerZ)
2226ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            : flinger(flinger), dpy(dpy),
22273dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian              heap(heap), w(w), h(h), f(f), sw(sw), sh(sh),
22283dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian              minLayerZ(minLayerZ), maxLayerZ(maxLayerZ),
22293dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian              result(PERMISSION_DENIED)
2230ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        {
2231ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
2232ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        status_t getResult() const {
2233ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            return result;
2234ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
2235ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        virtual bool handler() {
2236ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
2237ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2238ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            // if we have secure windows, never allow the screen capture
2239ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            if (flinger->mSecureFrameBuffer)
2240ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                return true;
2241ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
224238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            result = flinger->captureScreenImplLocked(dpy,
22433dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                    heap, w, h, f, sw, sh, minLayerZ, maxLayerZ);
2244ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2245ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            return true;
2246ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
2247ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    };
2248ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2249ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    sp<MessageBase> msg = new MessageCaptureScreen(this,
22503dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            dpy, heap, width, height, format, sw, sh, minLayerZ, maxLayerZ);
2251ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    status_t res = postMessageSync(msg);
2252ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (res == NO_ERROR) {
2253ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        res = static_cast<MessageCaptureScreen*>( msg.get() )->getResult();
2254ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    }
2255ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    return res;
2256ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian}
2257ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2258ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian// ---------------------------------------------------------------------------
2259ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
22607623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<Layer> SurfaceFlinger::getLayer(const sp<ISurface>& sur) const
22619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
22627623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<Layer> result;
22637623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    Mutex::Autolock _l(mStateLock);
22647623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    result = mLayerMap.valueFor( sur->asBinder() ).promote();
22657623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return result;
22667623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
2267d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
22687623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian// ---------------------------------------------------------------------------
2269593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
22707623da435e45c7c03ef6a00a43675deb6645f070Mathias AgopianClient::Client(const sp<SurfaceFlinger>& flinger)
22717623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    : mFlinger(flinger), mNameGenerator(1)
22727623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
22739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
22749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2275593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias AgopianClient::~Client()
2276593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
2277593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    const size_t count = mLayers.size();
2278593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
2279593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        sp<LayerBaseClient> layer(mLayers.valueAt(i).promote());
2280593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        if (layer != 0) {
2281593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            mFlinger->removeLayer(layer);
2282593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        }
22839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
22849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
22851473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
2286593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t Client::initCheck() const {
22877623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return NO_ERROR;
22889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
22891473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
2290593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianssize_t Client::attachLayer(const sp<LayerBaseClient>& layer)
22919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
22927623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    int32_t name = android_atomic_inc(&mNameGenerator);
2293593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    mLayers.add(name, layer);
2294593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return name;
22959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
22969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
22977623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianvoid Client::detachLayer(const LayerBaseClient* layer)
2298593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
2299593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // we do a linear search here, because this doesn't happen often
2300593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    const size_t count = mLayers.size();
2301593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
2302593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        if (mLayers.valueAt(i) == layer) {
2303593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            mLayers.removeItemsAt(i, 1);
2304593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            break;
2305593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        }
2306593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    }
2307593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian}
23081473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopiansp<LayerBaseClient> Client::getLayerUser(int32_t i) const {
23091473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBaseClient> lbc;
2310593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    const wp<LayerBaseClient>& layer(mLayers.valueFor(i));
2311593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (layer != 0) {
2312593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        lbc = layer.promote();
2313593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        LOGE_IF(lbc==0, "getLayerUser(name=%d) is dead", int(i));
23141473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    }
23151473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    return lbc;
23169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
23179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2318593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopiansp<IMemoryHeap> Client::getControlBlock() const {
23197623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return 0;
23207623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
23217623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianssize_t Client::getTokenForSurface(const sp<ISurface>& sur) const {
23227623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return -1;
23239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2324593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopiansp<ISurface> Client::createSurface(
23257623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        ISurfaceComposerClient::surface_data_t* params, int pid,
23267623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const String8& name,
23277623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        DisplayID display, uint32_t w, uint32_t h, PixelFormat format,
23289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        uint32_t flags)
23299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
2330593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return mFlinger->createSurface(this, pid, name, params,
2331593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            display, w, h, format, flags);
23329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2333593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t Client::destroySurface(SurfaceID sid) {
2334593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return mFlinger->removeSurface(this, sid);
23359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2336593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t Client::setState(int32_t count, const layer_state_t* states) {
2337593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return mFlinger->setClientState(this, count, states);
23389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
23399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
23409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
23417623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
23427623da435e45c7c03ef6a00a43675deb6645f070Mathias AgopianUserClient::UserClient(const sp<SurfaceFlinger>& flinger)
23437623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    : ctrlblk(0), mBitmap(0), mFlinger(flinger)
23447623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
23457623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    const int pgsize = getpagesize();
23467623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    const int cblksize = ((sizeof(SharedClient)+(pgsize-1))&~(pgsize-1));
23477623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
23487623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    mCblkHeap = new MemoryHeapBase(cblksize, 0,
23497623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            "SurfaceFlinger Client control-block");
23507623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
23517623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    ctrlblk = static_cast<SharedClient *>(mCblkHeap->getBase());
23527623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (ctrlblk) { // construct the shared structure in-place.
23537623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        new(ctrlblk) SharedClient;
23547623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
23557623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
23567623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
23577623da435e45c7c03ef6a00a43675deb6645f070Mathias AgopianUserClient::~UserClient()
23587623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
23597623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (ctrlblk) {
23607623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        ctrlblk->~SharedClient();  // destroy our shared-structure.
23617623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
23627623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
23637623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    /*
23647623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * When a UserClient dies, it's unclear what to do exactly.
23657623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * We could go ahead and destroy all surfaces linked to that client
23667623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * however, it wouldn't be fair to the main Client
23677623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * (usually the the window-manager), which might want to re-target
23687623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * the layer to another UserClient.
23697623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * I think the best is to do nothing, or not much; in most cases the
23707623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * WM itself will go ahead and clean things up when it detects a client of
23717623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * his has died.
23727623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * The remaining question is what to display? currently we keep
23737623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * just keep the current buffer.
23747623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     */
23757623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
23767623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
23777623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianstatus_t UserClient::initCheck() const {
23787623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return ctrlblk == 0 ? NO_INIT : NO_ERROR;
23797623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
23807623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
23817623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianvoid UserClient::detachLayer(const Layer* layer)
23827623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
23837623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    int32_t name = layer->getToken();
23847623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (name >= 0) {
23855e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        int32_t mask = 1LU<<name;
23865e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        if ((android_atomic_and(~mask, &mBitmap) & mask) == 0) {
23875e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian            LOGW("token %d wasn't marked as used %08x", name, int(mBitmap));
23885e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        }
23897623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
23907623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
23917623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
23927623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<IMemoryHeap> UserClient::getControlBlock() const {
23937623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return mCblkHeap;
23947623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
23957623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
23967623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianssize_t UserClient::getTokenForSurface(const sp<ISurface>& sur) const
23977623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
23987623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    int32_t name = NAME_NOT_FOUND;
23997623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<Layer> layer(mFlinger->getLayer(sur));
240085cfdd011241a5f2fb7fabc65b5943a39af7e1deJamie Gennis    if (layer == 0) {
240185cfdd011241a5f2fb7fabc65b5943a39af7e1deJamie Gennis        return name;
240285cfdd011241a5f2fb7fabc65b5943a39af7e1deJamie Gennis    }
24037623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24045e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian    // if this layer already has a token, just return it
24057623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    name = layer->getToken();
240685cfdd011241a5f2fb7fabc65b5943a39af7e1deJamie Gennis    if ((name >= 0) && (layer->getClient() == this)) {
24075e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        return name;
240885cfdd011241a5f2fb7fabc65b5943a39af7e1deJamie Gennis    }
24097623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24107623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    name = 0;
24117623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    do {
24127623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        int32_t mask = 1LU<<name;
24137623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        if ((android_atomic_or(mask, &mBitmap) & mask) == 0) {
24147623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            // we found and locked that name
24155e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian            status_t err = layer->setToken(
24165e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian                    const_cast<UserClient*>(this), ctrlblk, name);
24175e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian            if (err != NO_ERROR) {
24185e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian                // free the name
24195e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian                android_atomic_and(~mask, &mBitmap);
24205e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian                name = err;
24215e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian            }
24227623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            break;
24237623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        }
24247623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        if (++name > 31)
24257623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            name = NO_MEMORY;
24267623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    } while(name >= 0);
24277623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24281debc66521f699bbf0a8eb80cababaef8bc63607Mathias Agopian    //LOGD("getTokenForSurface(%p) => %d (client=%p, bitmap=%08lx)",
24291debc66521f699bbf0a8eb80cababaef8bc63607Mathias Agopian    //        sur->asBinder().get(), name, this, mBitmap);
24307623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return name;
24317623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24327623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24337623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<ISurface> UserClient::createSurface(
24347623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        ISurfaceComposerClient::surface_data_t* params, int pid,
24357623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const String8& name,
24367623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        DisplayID display, uint32_t w, uint32_t h, PixelFormat format,
24377623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        uint32_t flags) {
24387623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return 0;
24397623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24407623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianstatus_t UserClient::destroySurface(SurfaceID sid) {
24417623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return INVALID_OPERATION;
24427623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24437623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianstatus_t UserClient::setState(int32_t count, const layer_state_t* states) {
24447623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return INVALID_OPERATION;
24457623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24467623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24477623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian// ---------------------------------------------------------------------------
24489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
24499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane::GraphicPlane()
24509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    : mHw(0)
24519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
24529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
24539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
24549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane::~GraphicPlane() {
24559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    delete mHw;
24569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
24579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
24589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool GraphicPlane::initialized() const {
24599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return mHw ? true : false;
24609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
24619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
246266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianint GraphicPlane::getWidth() const {
246366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    return mWidth;
24649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
24659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
246666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianint GraphicPlane::getHeight() const {
246766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    return mHeight;
246866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian}
246966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
247066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianvoid GraphicPlane::setDisplayHardware(DisplayHardware *hw)
247166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian{
247266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mHw = hw;
247366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
247466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    // initialize the display orientation transform.
247566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    // it's a constant that should come from the display driver.
247666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    int displayOrientation = ISurfaceComposer::eOrientationDefault;
247766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    char property[PROPERTY_VALUE_MAX];
247866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    if (property_get("ro.sf.hwrotation", property, NULL) > 0) {
247966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        //displayOrientation
248066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        switch (atoi(property)) {
248166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        case 90:
248266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            displayOrientation = ISurfaceComposer::eOrientation90;
248366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            break;
248466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        case 270:
248566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            displayOrientation = ISurfaceComposer::eOrientation270;
248666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            break;
248766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        }
248866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    }
248966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
249066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float w = hw->getWidth();
249166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float h = hw->getHeight();
249266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    GraphicPlane::orientationToTransfrom(displayOrientation, w, h,
249366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            &mDisplayTransform);
249466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    if (displayOrientation & ISurfaceComposer::eOrientationSwapMask) {
249566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayWidth = h;
249666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayHeight = w;
249766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    } else {
249866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayWidth = w;
249966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayHeight = h;
250066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    }
250166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
250266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    setOrientation(ISurfaceComposer::eOrientationDefault);
25039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t GraphicPlane::orientationToTransfrom(
25069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int orientation, int w, int h, Transform* tr)
25078c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian{
25088c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    uint32_t flags = 0;
25099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (orientation) {
25109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientationDefault:
25118c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_0;
25128c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        break;
25139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation90:
25148c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_90;
25159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
25169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation180:
25178c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_180;
25189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
25199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation270:
25208c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_270;
25219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
25229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    default:
25239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
25249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
25258c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    tr->set(flags, w, h);
25269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
25279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t GraphicPlane::setOrientation(int orientation)
25309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
25319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // If the rotation can be handled in hardware, this is where
25329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // the magic should happen.
253366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
253466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const DisplayHardware& hw(displayHardware());
253566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float w = mDisplayWidth;
253666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float h = mDisplayHeight;
253766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mWidth = int(w);
253866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mHeight = int(h);
253966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
254066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    Transform orientationTransform;
25418c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    GraphicPlane::orientationToTransfrom(orientation, w, h,
25428c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian            &orientationTransform);
25438c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    if (orientation & ISurfaceComposer::eOrientationSwapMask) {
25448c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        mWidth = int(h);
25458c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        mHeight = int(w);
25469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
25478c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian
25483552f53c8370ced8680951f4ac811a126da02b0eMathias Agopian    mOrientation = orientation;
254966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mGlobalTransform = mDisplayTransform * orientationTransform;
25509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
25519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst DisplayHardware& GraphicPlane::displayHardware() const {
25549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return *mHw;
25559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2557aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias AgopianDisplayHardware& GraphicPlane::editDisplayHardware() {
2558aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    return *mHw;
2559aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
2560aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
25619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst Transform& GraphicPlane::transform() const {
25629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return mGlobalTransform;
25639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25651473f46cbc82aa6f0ba744cc896a36923823d55bMathias AgopianEGLDisplay GraphicPlane::getEGLDisplay() const {
25661473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    return mHw->getEGLDisplay();
25671473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian}
25681473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
25699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
25709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}; // namespace android
2572