SurfaceFlinger.cpp revision 0dd593f22352a0863223fa4ea7e37e926b99282e
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>
350dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian#include <binder/PermissionCache.h>
36d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/String8.h>
389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/String16.h>
399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/StopWatch.h>
409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
416950e428feaccc8164b989ef64e771a99948797aMathias Agopian#include <ui/GraphicBufferAllocator.h>
4204262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian#include <ui/GraphicLog.h>
439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <ui/PixelFormat.h>
449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <pixelflinger/pixelflinger.h>
469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <GLES/gl.h>
479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "clz.h"
49781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian#include "GLExtensions.h"
509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "Layer.h"
519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "LayerDim.h"
529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "SurfaceFlinger.h"
539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "DisplayHardware/DisplayHardware.h"
55e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian#include "DisplayHardware/HWComposer.h"
569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
577bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian#include <private/surfaceflinger/SharedBufferStack.h>
587bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian
59627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian/* ideally AID_GRAPHICS would be in a semi-public header
60627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian * or there would be a way to map a user/group name to its id
61627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian */
62627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian#ifndef AID_GRAPHICS
63627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian#define AID_GRAPHICS 1003
64627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian#endif
65627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#define DISPLAY_COUNT       1
679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectnamespace android {
699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
710dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopianconst String16 sHardwareTest("android.permission.HARDWARE_TEST");
720dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopianconst String16 sAccessSurfaceFlinger("android.permission.ACCESS_SURFACE_FLINGER");
730dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopianconst String16 sReadFramebuffer("android.permission.READ_FRAME_BUFFER");
740dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopianconst String16 sDump("android.permission.DUMP");
750dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian
760dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian// ---------------------------------------------------------------------------
770dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian
789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectSurfaceFlinger::SurfaceFlinger()
799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    :   BnSurfaceComposer(), Thread(false),
809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mTransactionFlags(0),
819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mTransactionCount(0),
829779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        mResizeTransationPending(false),
831473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        mLayersRemoved(false),
849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mBootTime(systemTime()),
859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mVisibleRegionsDirty(false),
86e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        mHwWorkListDirty(false),
879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDeferReleaseConsole(false),
889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mFreezeDisplay(false),
89d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        mElectronBeamAnimationMode(0),
909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mFreezeCount(0),
91c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project        mFreezeDisplayTime(0),
929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDebugRegion(0),
939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDebugBackground(0),
946a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        mDebugDisableHWC(0),
95a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers(0),
96a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastSwapBufferTime(0),
97a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInTransaction(0),
98a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastTransactionTime(0),
996950e428feaccc8164b989ef64e771a99948797aMathias Agopian        mBootFinished(false),
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mConsoleSignals(0),
1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mSecureFrameBuffer(0)
1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    init();
1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::init()
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LOGI("SurfaceFlinger is starting");
1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // debugging stuff...
1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    char value[PROPERTY_VALUE_MAX];
1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    property_get("debug.sf.showupdates", value, "0");
1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDebugRegion = atoi(value);
1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    property_get("debug.sf.showbackground", value, "0");
1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDebugBackground = atoi(value);
1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
117e5c0a7b7810092a593ff386843927238c175bdbdMathias Agopian    LOGI_IF(mDebugRegion,       "showupdates enabled");
118e5c0a7b7810092a593ff386843927238c175bdbdMathias Agopian    LOGI_IF(mDebugBackground,   "showbackground enabled");
1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectSurfaceFlinger::~SurfaceFlinger()
1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDeleteTextures(1, &mWormholeTexName);
1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
126d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopiansp<IMemoryHeap> SurfaceFlinger::getCblk() const
1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
128d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    return mServerHeap;
1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
131770492cb2b19f6a36ad748cd05fbedfbb9a67dfaMathias Agopiansp<ISurfaceComposerClient> SurfaceFlinger::createConnection()
1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
133593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<ISurfaceComposerClient> bclient;
134593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<Client> client(new Client(this));
135593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    status_t err = client->initCheck();
136593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (err == NO_ERROR) {
137593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        bclient = client;
1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return bclient;
1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
142f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennissp<IGraphicBufferAlloc> SurfaceFlinger::createGraphicBufferAlloc()
143f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis{
144f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    sp<GraphicBufferAlloc> gba(new GraphicBufferAlloc());
145f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    return gba;
146f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis}
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    mReadyToRunBarrier.open();
2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *  We're now ready to accept clients...
2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
263627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    // start boot animation
264627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    property_set("ctl.start", "bootanim");
265e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
2709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#if 0
2719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark -
2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark Events Handler
2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif
2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::waitForEvent()
2769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
2776ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    while (true) {
2786ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        nsecs_t timeout = -1;
2790e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        const nsecs_t freezeDisplayTimeout = ms2ns(5000);
2806ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        if (UNLIKELY(isFrozen())) {
2816ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            // wait 5 seconds
2826ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            const nsecs_t now = systemTime();
2836ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            if (mFreezeDisplayTime == 0) {
2846ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian                mFreezeDisplayTime = now;
2856ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            }
2866ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            nsecs_t waitTime = freezeDisplayTimeout - (now - mFreezeDisplayTime);
2876ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            timeout = waitTime>0 ? waitTime : 0;
288c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project        }
2896ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian
290898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        sp<MessageBase> msg = mEventQueue.waitMessage(timeout);
2910e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian
2920e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        // see if we timed out
2930e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        if (isFrozen()) {
2940e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            const nsecs_t now = systemTime();
2950e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            nsecs_t frozenTime = (now - mFreezeDisplayTime);
2960e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            if (frozenTime >= freezeDisplayTimeout) {
2979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // we timed out and are still frozen
2989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                LOGW("timeout expired mFreezeDisplay=%d, mFreezeCount=%d",
2999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        mFreezeDisplay, mFreezeCount);
3000e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                mFreezeDisplayTime = 0;
3019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeCount = 0;
302c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project                mFreezeDisplay = false;
3030e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            }
3040e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        }
3050e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian
3060e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        if (msg != 0) {
3070e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            switch (msg->what) {
3080e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                case MessageQueue::INVALIDATE:
3090e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                    // invalidate message, just return to the main loop
3100e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                    return;
3119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
3129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::signalEvent() {
3176ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    mEventQueue.invalidate();
3189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
320d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennisbool SurfaceFlinger::authenticateSurface(const sp<ISurface>& surface) const {
321d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    Mutex::Autolock _l(mStateLock);
322d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    sp<IBinder> surfBinder(surface->asBinder());
323d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis
324d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // Check the visible layer list for the ISurface
325d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    const LayerVector& currentLayers = mCurrentState.layersSortedByZ;
326d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    size_t count = currentLayers.size();
327d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    for (size_t i=0 ; i<count ; i++) {
328d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        const sp<LayerBase>& layer(currentLayers[i]);
329d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        sp<LayerBaseClient> lbc(layer->getLayerBaseClient());
330d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        if (lbc != NULL && lbc->getSurfaceBinder() == surfBinder) {
331d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis            return true;
332d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        }
333d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    }
334d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis
335d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // Check the layers in the purgatory.  This check is here so that if a
336d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // Surface gets destroyed before all the clients are done using it, the
337d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // error will not be reported as "surface XYZ is not authenticated", but
338d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // will instead fail later on when the client tries to use the surface,
339d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // which should be reported as "surface XYZ returned an -ENODEV".  The
340d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // purgatorized layers are no less authentic than the visible ones, so this
341d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // should not cause any harm.
342d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    size_t purgatorySize =  mLayerPurgatory.size();
343d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    for (size_t i=0 ; i<purgatorySize ; i++) {
344d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        const sp<LayerBase>& layer(mLayerPurgatory.itemAt(i));
345d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        sp<LayerBaseClient> lbc(layer->getLayerBaseClient());
346d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        if (lbc != NULL && lbc->getSurfaceBinder() == surfBinder) {
347d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis            return true;
348d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        }
349d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    }
350d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis
351d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    return false;
352d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis}
353d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis
354898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianstatus_t SurfaceFlinger::postMessageAsync(const sp<MessageBase>& msg,
355898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        nsecs_t reltime, uint32_t flags)
3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
357898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    return mEventQueue.postMessage(msg, reltime, flags);
358898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian}
359898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian
360898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianstatus_t SurfaceFlinger::postMessageSync(const sp<MessageBase>& msg,
361898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        nsecs_t reltime, uint32_t flags)
362898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian{
363898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    status_t res = mEventQueue.postMessage(msg, reltime, flags);
364898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    if (res == NO_ERROR) {
365898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        msg->wait();
366898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    }
367898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    return res;
3689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
3719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#if 0
3729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark -
3739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark Main loop
3749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif
3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool SurfaceFlinger::threadLoop()
3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
3789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    waitForEvent();
3799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3806960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // call Layer's destructor
3816960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    handleDestroyLayers();
3826960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian
3839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // check for transactions
3849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(mConsoleSignals)) {
3859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        handleConsoleEvents();
3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (LIKELY(mTransactionCount == 0)) {
3899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // if we're in a global transaction, don't do anything.
3909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const uint32_t mask = eTransactionNeeded | eTraversalNeeded;
3916dcb1557efc546b74323b4367d9b7f674821e1e9Mathias Agopian        uint32_t transactionFlags = peekTransactionFlags(mask);
3929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (LIKELY(transactionFlags)) {
3939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            handleTransaction(transactionFlags);
3949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // post surfaces (if needed)
3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    handlePageFlip();
3999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
400e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    if (UNLIKELY(mHwWorkListDirty)) {
401e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        // build the h/w work list
402e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        handleWorkList();
403e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    }
404e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
4059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw(graphicPlane(0).displayHardware());
40681384bf927c47a4efa653b14273084a13e67e3acMathias Agopian    if (LIKELY(hw.canDraw() && !isFrozen())) {
4079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // repaint the framebuffer (if needed)
40804262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian
40904262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        const int index = hw.getCurrentBufferIndex();
41004262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        GraphicLog& logger(GraphicLog::getInstance());
41104262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian
41204262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_REPAINT, index);
4139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        handleRepaint();
4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
415b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian        // inform the h/w that we're done compositing
41604262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_COMPOSITION_COMPLETE, index);
417b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian        hw.compositionComplete();
418b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian
41904262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_SWAP_BUFFERS, index);
4204c0a4a2b74db9d4a439a0aaa39d80586f7eb1258Antti Hatala        postFramebuffer();
4214c0a4a2b74db9d4a439a0aaa39d80586f7eb1258Antti Hatala
42204262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_REPAINT_DONE, index);
4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // pretend we did the post
425a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian        hw.compositionComplete();
4269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        usleep(16667); // 60 fps period
4279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return true;
4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::postFramebuffer()
4329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (!mInvalidRegion.isEmpty()) {
4349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw(graphicPlane(0).displayHardware());
435a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t now = systemTime();
436a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers = now;
4379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.flip(mInvalidRegion);
438a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastSwapBufferTime = systemTime() - now;
439a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers = 0;
4409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mInvalidRegion.clear();
4419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleConsoleEvents()
4459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // something to do with the console
4479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw = graphicPlane(0).displayHardware();
4489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int what = android_atomic_and(0, &mConsoleSignals);
4509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (what & eConsoleAcquired) {
4519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.acquireScreen();
4522d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // this is a temporary work-around, eventually this should be called
4532d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // by the power-manager
454d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        SurfaceFlinger::turnElectronBeamOn(mElectronBeamAnimationMode);
4559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
457aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (mDeferReleaseConsole && hw.isScreenAcquired()) {
458ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian        // We got the release signal before the acquire signal
4599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDeferReleaseConsole = false;
4609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.releaseScreen();
4619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (what & eConsoleReleased) {
464aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        if (hw.isScreenAcquired()) {
4659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            hw.releaseScreen();
4669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
4679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDeferReleaseConsole = true;
4689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.set(hw.bounds());
4729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleTransaction(uint32_t transactionFlags)
4759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4766960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    Mutex::Autolock _l(mStateLock);
4776960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    const nsecs_t now = systemTime();
4786960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    mDebugInTransaction = now;
4796960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian
4806960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // Here we're guaranteed that some transaction flags are set
4816960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // so we can call handleTransactionLocked() unconditionally.
4826960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // We call getTransactionFlags(), which will also clear the flags,
4836960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // with mStateLock held to guarantee that mCurrentState won't change
4846960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // until the transaction is committed.
4856960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian
4866960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    const uint32_t mask = eTransactionNeeded | eTraversalNeeded;
4876960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    transactionFlags = getTransactionFlags(mask);
4886960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    handleTransactionLocked(transactionFlags);
4896960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian
4906960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    mLastTransactionTime = systemTime() - now;
4916960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    mDebugInTransaction = 0;
4926960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    invalidateHwcGeometry();
4936960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // here the transaction has been committed
4942d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian}
4959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4966960811d15eb66e8469330d3bc574c32c481db05Mathias Agopianvoid SurfaceFlinger::handleTransactionLocked(uint32_t transactionFlags)
4972d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian{
4982d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    const LayerVector& currentLayers(mCurrentState.layersSortedByZ);
4999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const size_t count = currentLayers.size();
5009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Traversal of the children
5039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * (perform the transaction for each of them if needed)
5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const bool layersNeedTransaction = transactionFlags & eTraversalNeeded;
5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (layersNeedTransaction) {
5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (size_t i=0 ; i<count ; i++) {
5091473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian            const sp<LayerBase>& layer = currentLayers[i];
5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            uint32_t trFlags = layer->getTransactionFlags(eTransactionNeeded);
5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (!trFlags) continue;
5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const uint32_t flags = layer->doTransaction(0);
5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (flags & Layer::eVisibleRegion)
5159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mVisibleRegionsDirty = true;
5169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
5209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Perform our own transaction if needed
5219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (transactionFlags & eTransactionNeeded) {
5249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mCurrentState.orientation != mDrawingState.orientation) {
5259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // the orientation has changed, recompute all visible regions
5269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // and invalidate everything.
5279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int dpy = 0;
5299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int orientation = mCurrentState.orientation;
530eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian            const uint32_t type = mCurrentState.orientationType;
5319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            GraphicPlane& plane(graphicPlane(dpy));
5329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            plane.setOrientation(orientation);
5339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // update the shared control block
5359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const DisplayHardware& hw(plane.displayHardware());
5369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            volatile display_cblk_t* dcblk = mServerCblk->displays + dpy;
5379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dcblk->orientation = orientation;
53866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            dcblk->w = plane.getWidth();
53966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            dcblk->h = plane.getHeight();
5409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
5429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(hw.bounds());
5439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mCurrentState.freezeDisplay != mDrawingState.freezeDisplay) {
5469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // freezing or unfreezing the display -> trigger animation if needed
5479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mFreezeDisplay = mCurrentState.freezeDisplay;
54831901cc0b6f0c678be4f629c8c3405700e63c346Mathias Agopian            if (mFreezeDisplay)
54931901cc0b6f0c678be4f629c8c3405700e63c346Mathias Agopian                 mFreezeDisplayTime = 0;
5509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
552a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        if (currentLayers.size() > mDrawingState.layersSortedByZ.size()) {
553a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            // layers have been added
5549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
5559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
557a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        // some layers might have been removed, so
558a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        // we need to update the regions they're exposing.
559a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        if (mLayersRemoved) {
560248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian            mLayersRemoved = false;
5619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
562a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            const LayerVector& previousLayers(mDrawingState.layersSortedByZ);
5632d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian            const size_t count = previousLayers.size();
5642d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian            for (size_t i=0 ; i<count ; i++) {
565a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                const sp<LayerBase>& layer(previousLayers[i]);
566a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                if (currentLayers.indexOf( layer ) < 0) {
567a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                    // this layer is not visible anymore
56833863dd9e1005478d20b70fad42e447ac97f5abfMathias Agopian                    mDirtyRegionRemovedLayer.orSelf(layer->visibleRegionScreen);
569a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                }
570a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            }
5719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    commitTransaction();
5759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
5769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5776960811d15eb66e8469330d3bc574c32c481db05Mathias Agopianvoid SurfaceFlinger::destroyLayer(LayerBase const* layer)
5786960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian{
5796960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    Mutex::Autolock _l(mDestroyedLayerLock);
5806960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    mDestroyedLayers.add(layer);
5816960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    signalEvent();
5826960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian}
5836960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian
5846960811d15eb66e8469330d3bc574c32c481db05Mathias Agopianvoid SurfaceFlinger::handleDestroyLayers()
5856960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian{
5866960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    Vector<LayerBase const *> destroyedLayers;
5876960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian
5886960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    { // scope for the lock
5896960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian        Mutex::Autolock _l(mDestroyedLayerLock);
5906960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian        destroyedLayers = mDestroyedLayers;
5916960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian        mDestroyedLayers.clear();
5926960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    }
5936960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian
5946960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // call destructors without a lock held
5956960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    const size_t count = destroyedLayers.size();
5966960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
5976960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian        //LOGD("destroying %s", destroyedLayers[i]->getName().string());
5986960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian        delete destroyedLayers[i];
5996960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    }
6006960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian}
6016960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian
6029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectsp<FreezeLock> SurfaceFlinger::getFreezeLock() const
6039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
6049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return new FreezeLock(const_cast<SurfaceFlinger *>(this));
6059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
6069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::computeVisibleRegions(
608f0ff906fa427ddc3293dc061e2ee34ce39c1336eMathias Agopian    const LayerVector& currentLayers, Region& dirtyRegion, Region& opaqueRegion)
6099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
6109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(0));
6119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Transform& planeTransform(plane.transform());
6129c041bbd81789c209e2369ba958306979b67614fMathias Agopian    const DisplayHardware& hw(plane.displayHardware());
6139c041bbd81789c209e2369ba958306979b67614fMathias Agopian    const Region screenRegion(hw.bounds());
6149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region aboveOpaqueLayers;
6169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region aboveCoveredLayers;
6179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region dirty;
6189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool secureFrameBuffer = false;
6209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t i = currentLayers.size();
6229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    while (i--) {
6231473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        const sp<LayerBase>& layer = currentLayers[i];
6249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->validateVisibility(planeTransform);
6259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // start with the whole surface at its current location
62712cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        const Layer::State& s(layer->drawingState());
6289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6299c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
6309c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * opaqueRegion: area of a surface that is fully opaque.
6319c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
6329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region opaqueRegion;
6339c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6349c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
6359c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * visibleRegion: area of a surface that is visible on screen
6369c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * and not fully transparent. This is essentially the layer's
6379c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * footprint minus the opaque regions above it.
6389c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * Areas covered by a translucent surface are considered visible.
6399c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
6409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region visibleRegion;
6419c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6429c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
6439c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * coveredRegion: area of a surface that is covered by all
6449c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * visible regions above it (which includes the translucent areas).
6459c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
6469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region coveredRegion;
6479c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6489c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6499c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // handle hidden surfaces by setting the visible region to empty
65012cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        if (LIKELY(!(s.flags & ISurfaceComposer::eLayerHidden) && s.alpha)) {
6517bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian            const bool translucent = !layer->isOpaque();
65212cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian            const Rect bounds(layer->visibleBounds());
6539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            visibleRegion.set(bounds);
6549c041bbd81789c209e2369ba958306979b67614fMathias Agopian            visibleRegion.andSelf(screenRegion);
6559c041bbd81789c209e2369ba958306979b67614fMathias Agopian            if (!visibleRegion.isEmpty()) {
6569c041bbd81789c209e2369ba958306979b67614fMathias Agopian                // Remove the transparent area from the visible region
6579c041bbd81789c209e2369ba958306979b67614fMathias Agopian                if (translucent) {
6589c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    visibleRegion.subtractSelf(layer->transparentRegionScreen);
6599c041bbd81789c209e2369ba958306979b67614fMathias Agopian                }
6609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6619c041bbd81789c209e2369ba958306979b67614fMathias Agopian                // compute the opaque region
6629c041bbd81789c209e2369ba958306979b67614fMathias Agopian                const int32_t layerOrientation = layer->getOrientation();
6639c041bbd81789c209e2369ba958306979b67614fMathias Agopian                if (s.alpha==255 && !translucent &&
6649c041bbd81789c209e2369ba958306979b67614fMathias Agopian                        ((layerOrientation & Transform::ROT_INVALID) == false)) {
6659c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    // the opaque region is the layer's footprint
6669c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    opaqueRegion = visibleRegion;
6679c041bbd81789c209e2369ba958306979b67614fMathias Agopian                }
6689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
6699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6719c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Clip the covered region to the visible region
6729c041bbd81789c209e2369ba958306979b67614fMathias Agopian        coveredRegion = aboveCoveredLayers.intersect(visibleRegion);
6739c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6749c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Update aboveCoveredLayers for next (lower) layer
6759c041bbd81789c209e2369ba958306979b67614fMathias Agopian        aboveCoveredLayers.orSelf(visibleRegion);
6769c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // subtract the opaque region covered by the layers above us
6789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        visibleRegion.subtractSelf(aboveOpaqueLayers);
6799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // compute this layer's dirty region
6819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (layer->contentDirty) {
6829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // we need to invalidate the whole region
6839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dirty = visibleRegion;
6849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // as well, as the old visible region
6859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dirty.orSelf(layer->visibleRegionScreen);
6869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->contentDirty = false;
6879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
6880aed7e97a3efbeec23a4aa301df03036a67f8ea3Mathias Agopian            /* compute the exposed region:
6899c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   the exposed region consists of two components:
6909c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   1) what's VISIBLE now and was COVERED before
6919c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   2) what's EXPOSED now less what was EXPOSED before
6929c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *
6939c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * note that (1) is conservative, we start with the whole
6949c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * visible region but only keep what used to be covered by
6959c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * something -- which mean it may have been exposed.
6969c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *
6979c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * (2) handles areas that were not covered by anything but got
6989c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * exposed because of a resize.
6990aed7e97a3efbeec23a4aa301df03036a67f8ea3Mathias Agopian             */
7009c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region newExposed = visibleRegion - coveredRegion;
7019c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldVisibleRegion = layer->visibleRegionScreen;
7029c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldCoveredRegion = layer->coveredRegionScreen;
7039c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldExposed = oldVisibleRegion - oldCoveredRegion;
7049c041bbd81789c209e2369ba958306979b67614fMathias Agopian            dirty = (visibleRegion&oldCoveredRegion) | (newExposed-oldExposed);
7059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dirty.subtractSelf(aboveOpaqueLayers);
7079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // accumulate to the screen dirty region
7099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dirtyRegion.orSelf(dirty);
7109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7119c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Update aboveOpaqueLayers for next (lower) layer
7129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        aboveOpaqueLayers.orSelf(opaqueRegion);
713e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
7149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Store the visible region is screen space
7159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->setVisibleRegion(visibleRegion);
7169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->setCoveredRegion(coveredRegion);
7179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
71812cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        // If a secure layer is partially visible, lock-down the screen!
7199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (layer->isSecure() && !visibleRegion.isEmpty()) {
7209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            secureFrameBuffer = true;
7219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
72412cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    // invalidate the areas where a layer was removed
72512cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    dirtyRegion.orSelf(mDirtyRegionRemovedLayer);
72612cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    mDirtyRegionRemovedLayer.clear();
72712cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian
7289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mSecureFrameBuffer = secureFrameBuffer;
7299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    opaqueRegion = aboveOpaqueLayers;
7309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::commitTransaction()
7349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDrawingState = mCurrentState;
7369779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian    mResizeTransationPending = false;
7379779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian    mTransactionCV.broadcast();
7389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handlePageFlip()
7419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool visibleRegions = mVisibleRegionsDirty;
743f0ff906fa427ddc3293dc061e2ee34ce39c1336eMathias Agopian    const LayerVector& currentLayers(mDrawingState.layersSortedByZ);
7449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    visibleRegions |= lockPageFlip(currentLayers);
7459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw = graphicPlane(0).displayHardware();
7479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const Region screenRegion(hw.bounds());
7489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (visibleRegions) {
7499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            Region opaqueRegion;
7509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            computeVisibleRegions(currentLayers, mDirtyRegion, opaqueRegion);
751ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian
752ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            /*
753ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian             *  rebuild the visible layer list
754ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian             */
755f0ff906fa427ddc3293dc061e2ee34ce39c1336eMathias Agopian            const size_t count = currentLayers.size();
756ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            mVisibleLayersSortedByZ.clear();
757ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            mVisibleLayersSortedByZ.setCapacity(count);
758ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            for (size_t i=0 ; i<count ; i++) {
759ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian                if (!currentLayers[i]->visibleRegionScreen.isEmpty())
760ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian                    mVisibleLayersSortedByZ.add(currentLayers[i]);
761ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            }
762ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian
7639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mWormholeRegion = screenRegion.subtract(opaqueRegion);
7649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = false;
765fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian            invalidateHwcGeometry();
7669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    unlockPageFlip(currentLayers);
7699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.andSelf(screenRegion);
7709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
772fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopianvoid SurfaceFlinger::invalidateHwcGeometry()
773fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian{
774fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian    mHwWorkListDirty = true;
775fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian}
776fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian
7779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool SurfaceFlinger::lockPageFlip(const LayerVector& currentLayers)
7789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool recomputeVisibleRegions = false;
7809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t count = currentLayers.size();
7811473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBase> const* layers = currentLayers.array();
7829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (size_t i=0 ; i<count ; i++) {
7837623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
7849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->lockPageFlip(recomputeVisibleRegions);
7859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return recomputeVisibleRegions;
7879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::unlockPageFlip(const LayerVector& currentLayers)
7909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(0));
7929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Transform& planeTransform(plane.transform());
7939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t count = currentLayers.size();
7941473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBase> const* layers = currentLayers.array();
7959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (size_t i=0 ; i<count ; i++) {
7967623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
7979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->unlockPageFlip(planeTransform, mDirtyRegion);
7989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
8009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
801e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopianvoid SurfaceFlinger::handleWorkList()
802e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian{
803e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    mHwWorkListDirty = false;
804e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    HWComposer& hwc(graphicPlane(0).displayHardware().getHwComposer());
805e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    if (hwc.initCheck() == NO_ERROR) {
806e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        const Vector< sp<LayerBase> >& currentLayers(mVisibleLayersSortedByZ);
807e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        const size_t count = currentLayers.size();
808e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        hwc.createWorkList(count);
809f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        hwc_layer_t* const cur(hwc.getLayers());
810f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        for (size_t i=0 ; cur && i<count ; i++) {
811f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            currentLayers[i]->setGeometry(&cur[i]);
8126a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian            if (mDebugDisableHWC) {
8136a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                cur[i].compositionType = HWC_FRAMEBUFFER;
8146a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                cur[i].flags |= HWC_SKIP_LAYER;
8156a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian            }
816e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        }
817e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    }
818e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian}
8198c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian
8209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleRepaint()
8219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
8228c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    // compute the invalid region
8238c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    mInvalidRegion.orSelf(mDirtyRegion);
8249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(mDebugRegion)) {
8269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        debugFlashRegions();
8279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8298c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    // set the frame buffer
8308c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
8318c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    glMatrixMode(GL_MODELVIEW);
8328c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    glLoadIdentity();
8339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t flags = hw.getFlags();
835e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber    if ((flags & DisplayHardware::SWAP_RECTANGLE) ||
836e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber        (flags & DisplayHardware::BUFFER_PRESERVED))
8372e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian    {
838ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // we can redraw only what's dirty, but since SWAP_RECTANGLE only
839ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // takes a rectangle, we must make sure to update that whole
840ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // rectangle in that case
841ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        if (flags & DisplayHardware::SWAP_RECTANGLE) {
8427623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            // TODO: we really should be able to pass a region to
843ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // SWAP_RECTANGLE so that we don't have to redraw all this.
844ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            mDirtyRegion.set(mInvalidRegion.bounds());
845ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        } else {
846ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // in the BUFFER_PRESERVED case, obviously, we can update only
847ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // what's needed and nothing more.
848ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // NOTE: this is NOT a common case, as preserving the backbuffer
849ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // is costly and usually involves copying the whole update back.
850ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        }
8519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
852f2d28b74850ea0869aef2ce0727a6abb7b166a5cMathias Agopian        if (flags & DisplayHardware::PARTIAL_UPDATES) {
853ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // We need to redraw the rectangle that will be updated
8542e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian            // (pushed to the framebuffer).
855f2d28b74850ea0869aef2ce0727a6abb7b166a5cMathias Agopian            // This is needed because PARTIAL_UPDATES only takes one
856ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // rectangle instead of a region (see DisplayHardware::flip())
8579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(mInvalidRegion.bounds());
8589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
859ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // we need to redraw everything (the whole screen)
8609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(hw.bounds());
8619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mInvalidRegion = mDirtyRegion;
8629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // compose all surfaces
8669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    composeSurfaces(mDirtyRegion);
8679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // clear the dirty regions
8699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.clear();
8709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
8719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::composeSurfaces(const Region& dirty)
8739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
8749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(!mWormholeRegion.isEmpty())) {
8759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // should never happen unless the window manager has a bug
8769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // draw something...
8779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        drawWormhole();
8789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
879e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
880e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    status_t err = NO_ERROR;
881ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian    const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
882f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    size_t count = layers.size();
883e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
884e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
885e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    HWComposer& hwc(hw.getHwComposer());
886f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    hwc_layer_t* const cur(hwc.getLayers());
887e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
888f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    LOGE_IF(cur && hwc.getNumLayers() != count,
889f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            "HAL number of layers (%d) doesn't match surfaceflinger (%d)",
890f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            hwc.getNumLayers(), count);
891e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
892f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    // just to be extra-safe, use the smallest count
8930cf2f4354bf8952d822fa4ee9b9588a73999c2f4Erik Gilling    if (hwc.initCheck() == NO_ERROR) {
8940cf2f4354bf8952d822fa4ee9b9588a73999c2f4Erik Gilling        count = count < hwc.getNumLayers() ? count : hwc.getNumLayers();
8950cf2f4354bf8952d822fa4ee9b9588a73999c2f4Erik Gilling    }
896e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
897f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    /*
898f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     *  update the per-frame h/w composer data for each layer
899f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     *  and build the transparent region of the FB
900f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     */
901f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    Region transparent;
902f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    if (cur) {
903f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        for (size_t i=0 ; i<count ; i++) {
904f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            const sp<LayerBase>& layer(layers[i]);
905f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            layer->setPerFrameData(&cur[i]);
9069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
907f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        err = hwc.prepare();
908f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        LOGE_IF(err, "HWComposer::prepare failed (%s)", strerror(-err));
909e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
91045491690d8f21d2648cad9247115720fa90046b4Mathias Agopian        if (err == NO_ERROR) {
91145491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            for (size_t i=0 ; i<count ; i++) {
91245491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                if (cur[i].hints & HWC_HINT_CLEAR_FB) {
91345491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    const sp<LayerBase>& layer(layers[i]);
9147bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                    if (layer->isOpaque()) {
91545491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                        transparent.orSelf(layer->visibleRegionScreen);
91645491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    }
91745491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                }
91845491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            }
91945491690d8f21d2648cad9247115720fa90046b4Mathias Agopian
92045491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            /*
92145491690d8f21d2648cad9247115720fa90046b4Mathias Agopian             *  clear the area of the FB that need to be transparent
92245491690d8f21d2648cad9247115720fa90046b4Mathias Agopian             */
92345491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            transparent.andSelf(dirty);
92445491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            if (!transparent.isEmpty()) {
92545491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                glClearColor(0,0,0,0);
92645491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                Region::const_iterator it = transparent.begin();
92745491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                Region::const_iterator const end = transparent.end();
92845491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                const int32_t height = hw.getHeight();
92945491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                while (it != end) {
93045491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    const Rect& r(*it++);
93145491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    const GLint sy = height - (r.top + r.height());
93245491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    glScissor(r.left, sy, r.width(), r.height());
93345491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    glClear(GL_COLOR_BUFFER_BIT);
93445491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                }
93545491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            }
936e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        }
937e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    }
938f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian
939f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian
940f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    /*
941f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     * and then, render the layers targeted at the framebuffer
942f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     */
943f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
944f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        if (cur) {
945982f58bdccce7e4e537cffb2410ad78d73398461Antti Hatala            if ((cur[i].compositionType != HWC_FRAMEBUFFER) &&
946982f58bdccce7e4e537cffb2410ad78d73398461Antti Hatala                !(cur[i].flags & HWC_SKIP_LAYER)) {
947f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian                // skip layers handled by the HAL
948f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian                continue;
949f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            }
950f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        }
9516a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian
952f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
953f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        const Region clip(dirty.intersect(layer->visibleRegionScreen));
954f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        if (!clip.isEmpty()) {
955f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            layer->draw(clip);
956f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        }
957f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    }
9589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
9599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::debugFlashRegions()
9619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
962f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
963f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    const uint32_t flags = hw.getFlags();
964f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
965f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    if (!((flags & DisplayHardware::SWAP_RECTANGLE) ||
966f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian            (flags & DisplayHardware::BUFFER_PRESERVED))) {
967f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        const Region repaint((flags & DisplayHardware::PARTIAL_UPDATES) ?
968f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian                mDirtyRegion.bounds() : hw.bounds());
969f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        composeSurfaces(repaint);
970f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    }
971f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
9729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_BLEND);
9739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
9749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_SCISSOR_TEST);
9759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
976dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    static int toggle = 0;
977dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    toggle = 1 - toggle;
978dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    if (toggle) {
979f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glColor4f(1, 0, 1, 1);
980dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    } else {
981f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glColor4f(1, 1, 0, 1);
982dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    }
9839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9846158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    Region::const_iterator it = mDirtyRegion.begin();
9856158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    Region::const_iterator const end = mDirtyRegion.end();
9866158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    while (it != end) {
9876158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        const Rect& r = *it++;
9889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        GLfloat vertices[][2] = {
9899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.left,  r.top },
9909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.left,  r.bottom },
9919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.right, r.bottom },
9929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.right, r.top }
9939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        };
9949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glVertexPointer(2, GL_FLOAT, 0, vertices);
9959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
9969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
997f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
9988c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    if (mInvalidRegion.isEmpty()) {
9998c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        mDirtyRegion.dump("mDirtyRegion");
10008c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        mInvalidRegion.dump("mInvalidRegion");
10018c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    }
10028c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    hw.flip(mInvalidRegion);
10039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (mDebugRegion > 1)
1005f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        usleep(mDebugRegion * 1000);
10069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glEnable(GL_SCISSOR_TEST);
10089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    //mDirtyRegion.dump("mDirtyRegion");
10099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::drawWormhole() const
10129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Region region(mWormholeRegion.intersect(mDirtyRegion));
10149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (region.isEmpty())
10159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return;
10169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw(graphicPlane(0).displayHardware());
10189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int32_t width = hw.getWidth();
10199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int32_t height = hw.getHeight();
10209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_BLEND);
10229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
10239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (LIKELY(!mDebugBackground)) {
1025f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glClearColor(0,0,0,0);
10266158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator it = region.begin();
10276158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator const end = region.end();
10286158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        while (it != end) {
10296158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian            const Rect& r = *it++;
10309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const GLint sy = height - (r.top + r.height());
10319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glScissor(r.left, sy, r.width(), r.height());
10329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glClear(GL_COLOR_BUFFER_BIT);
10339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
10359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const GLshort vertices[][2] = { { 0, 0 }, { width, 0 },
10369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { width, height }, { 0, height }  };
10379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const GLshort tcoords[][2] = { { 0, 0 }, { 1, 0 },  { 1, 1 }, { 0, 1 } };
10389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glVertexPointer(2, GL_SHORT, 0, vertices);
10399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glTexCoordPointer(2, GL_SHORT, 0, tcoords);
10409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glEnableClientState(GL_TEXTURE_COORD_ARRAY);
1041e20a56d929fc8fedc2b468ea6d1900bd2aa6e81aMichael I. Gold#if defined(GL_OES_EGL_image_external)
1042781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian        if (GLExtensions::getInstance().haveTextureExternal()) {
1043781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian            glDisable(GL_TEXTURE_EXTERNAL_OES);
1044781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian        }
1045f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian#endif
10469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glEnable(GL_TEXTURE_2D);
10479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glBindTexture(GL_TEXTURE_2D, mWormholeTexName);
10489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
10499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glMatrixMode(GL_TEXTURE);
10509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glLoadIdentity();
10519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glScalef(width*(1.0f/32.0f), height*(1.0f/32.0f), 1);
10526158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator it = region.begin();
10536158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator const end = region.end();
10546158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        while (it != end) {
10556158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian            const Rect& r = *it++;
10569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const GLint sy = height - (r.top + r.height());
10579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glScissor(r.left, sy, r.width(), r.height());
10589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
10599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glDisableClientState(GL_TEXTURE_COORD_ARRAY);
10617bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        glDisable(GL_TEXTURE_2D);
106204a709e459db6f15c04b00bcd3b030c90ca52949Mathias Agopian        glLoadIdentity();
106304a709e459db6f15c04b00bcd3b030c90ca52949Mathias Agopian        glMatrixMode(GL_MODELVIEW);
10649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::debugShowFPS() const
10689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int mFrameCount;
10709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int mLastFrameCount = 0;
10719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static nsecs_t mLastFpsTime = 0;
10729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static float mFps = 0;
10739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mFrameCount++;
10749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    nsecs_t now = systemTime();
10759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    nsecs_t diff = now - mLastFpsTime;
10769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (diff > ms2ns(250)) {
10779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mFps =  ((mFrameCount - mLastFrameCount) * float(s2ns(1))) / diff;
10789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mLastFpsTime = now;
10799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mLastFrameCount = mFrameCount;
10809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // XXX: mFPS has the value we want
10829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project }
10839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10841473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::addLayer(const sp<LayerBase>& layer)
10859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
10879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    addLayer_l(layer);
10889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded|eTraversalNeeded);
10899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
10909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10921473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::addLayer_l(const sp<LayerBase>& layer)
10939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10941efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian    ssize_t i = mCurrentState.layersSortedByZ.add(layer);
10959bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian    return (i < 0) ? status_t(i) : status_t(NO_ERROR);
10969bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian}
10979bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1098593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianssize_t SurfaceFlinger::addClientLayer(const sp<Client>& client,
1099593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const sp<LayerBaseClient>& lbc)
11009bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian{
1101593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // attach this layer to the client
11025fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    size_t name = client->attachLayer(lbc);
11035fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian
11045fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    Mutex::Autolock _l(mStateLock);
1105593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
1106593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // add this layer to the current state list
1107593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    addLayer_l(lbc);
1108593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
11095fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    return ssize_t(name);
1110593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian}
1111593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
1112593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::removeLayer(const sp<LayerBase>& layer)
1113593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
1114593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    Mutex::Autolock _l(mStateLock);
1115593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    status_t err = purgatorizeLayer_l(layer);
1116593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (err == NO_ERROR)
1117593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        setTransactionFlags(eTransactionNeeded);
1118593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return err;
11199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11211473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::removeLayer_l(const sp<LayerBase>& layerBase)
11229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11237623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<LayerBaseClient> lbc(layerBase->getLayerBaseClient());
11247623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (lbc != 0) {
1125d35c6667c8233385f31aa203f486b2cb826bf6beMathias Agopian        mLayerMap.removeItem( lbc->getSurfaceBinder() );
11267623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
11279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    ssize_t index = mCurrentState.layersSortedByZ.remove(layerBase);
11289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (index >= 0) {
11291473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        mLayersRemoved = true;
11309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return NO_ERROR;
11319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11322d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    return status_t(index);
11339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11356cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopianstatus_t SurfaceFlinger::purgatorizeLayer_l(const sp<LayerBase>& layerBase)
11366cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian{
1137f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian    // First add the layer to the purgatory list, which makes sure it won't
1138f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian    // go away, then remove it from the main list (through a transaction).
11396cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    ssize_t err = removeLayer_l(layerBase);
1140f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian    if (err >= 0) {
1141f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian        mLayerPurgatory.add(layerBase);
1142f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian    }
11432e4b68d57bb64d7e93139238c5a8be91ff956c2aMathias Agopian
11440c4cec7e4df87181486d280c98fba9c0f4774c37Mathias Agopian    layerBase->onRemoved();
11450c4cec7e4df87181486d280c98fba9c0f4774c37Mathias Agopian
11462d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    // it's possible that we don't find a layer, because it might
11472d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    // have been destroyed already -- this is not technically an error
1148593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // from the user because there is a race between Client::destroySurface(),
1149593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // ~Client() and ~ISurface().
11506cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    return (err == NAME_NOT_FOUND) ? status_t(NO_ERROR) : err;
11516cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian}
11526cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
1153593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::invalidateLayerVisibility(const sp<LayerBase>& layer)
11549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1155593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    layer->forceVisibilityTransaction();
1156593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    setTransactionFlags(eTraversalNeeded);
1157593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return NO_ERROR;
11589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11606dcb1557efc546b74323b4367d9b7f674821e1e9Mathias Agopianuint32_t SurfaceFlinger::peekTransactionFlags(uint32_t flags)
11616dcb1557efc546b74323b4367d9b7f674821e1e9Mathias Agopian{
11626dcb1557efc546b74323b4367d9b7f674821e1e9Mathias Agopian    return android_atomic_release_load(&mTransactionFlags);
11636dcb1557efc546b74323b4367d9b7f674821e1e9Mathias Agopian}
11646dcb1557efc546b74323b4367d9b7f674821e1e9Mathias Agopian
11659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectuint32_t SurfaceFlinger::getTransactionFlags(uint32_t flags)
11669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return android_atomic_and(~flags, &mTransactionFlags) & flags;
11689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1170898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianuint32_t SurfaceFlinger::setTransactionFlags(uint32_t flags)
11719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t old = android_atomic_or(flags, &mTransactionFlags);
11739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if ((old & flags)==0) { // wake the server up
1174898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        signalEvent();
11759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return old;
11779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::openGlobalTransaction()
11809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    android_atomic_inc(&mTransactionCount);
11829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::closeGlobalTransaction()
11859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (android_atomic_dec(&mTransactionCount) == 1) {
11879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        signalEvent();
11889779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian
1189e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber        // if there is a transaction with a resize, wait for it to
11909779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        // take effect before returning.
11919779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        Mutex::Autolock _l(mStateLock);
11929779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        while (mResizeTransationPending) {
119398a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian            status_t err = mTransactionCV.waitRelative(mStateLock, s2ns(5));
119498a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian            if (CC_UNLIKELY(err != NO_ERROR)) {
119598a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                // just in case something goes wrong in SF, return to the
119698a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                // called after a few seconds.
119798a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                LOGW_IF(err == TIMED_OUT, "closeGlobalTransaction timed out!");
119898a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                mResizeTransationPending = false;
119998a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                break;
120098a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian            }
12019779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        }
12029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::freezeDisplay(DisplayID dpy, uint32_t flags)
12069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
12079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
12089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
12099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
12119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mCurrentState.freezeDisplay = 1;
12129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded);
12139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // flags is intended to communicate some sort of animation behavior
1215ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian    // (for instance fading)
12169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
12179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::unfreezeDisplay(DisplayID dpy, uint32_t flags)
12209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
12219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
12229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
12239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
12259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mCurrentState.freezeDisplay = 0;
12269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded);
12279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // flags is intended to communicate some sort of animation behavior
1229ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian    // (for instance fading)
12309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
12319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1233e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huberint SurfaceFlinger::setOrientation(DisplayID dpy,
1234eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian        int orientation, uint32_t flags)
12359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
12369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
12379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
12389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
12409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (mCurrentState.orientation != orientation) {
12419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (uint32_t(orientation)<=eOrientation270 || orientation==42) {
1242eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian            mCurrentState.orientationType = flags;
12439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mCurrentState.orientation = orientation;
12449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            setTransactionFlags(eTransactionNeeded);
12459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mTransactionCV.wait(mStateLock);
12469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
12479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            orientation = BAD_VALUE;
12489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
12499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return orientation;
12519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12539638e5c167be321643bf3f3ee39e3fb45541fb3bMathias Agopiansp<ISurface> SurfaceFlinger::createSurface(
12549638e5c167be321643bf3f3ee39e3fb45541fb3bMathias Agopian        ISurfaceComposerClient::surface_data_t* params,
12559638e5c167be321643bf3f3ee39e3fb45541fb3bMathias Agopian        const String8& name,
12569638e5c167be321643bf3f3ee39e3fb45541fb3bMathias Agopian        const sp<Client>& client,
12579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        DisplayID d, uint32_t w, uint32_t h, PixelFormat format,
12589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        uint32_t flags)
12599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
12601473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBaseClient> layer;
12617bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    sp<ISurface> surfaceHandle;
12624d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian
12634d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian    if (int32_t(w|h) < 0) {
12644d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian        LOGE("createSurface() failed, w or h is negative (w=%d, h=%d)",
12654d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian                int(w), int(h));
12664d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian        return surfaceHandle;
12674d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian    }
1268e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
12699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    //LOGD("createSurface for pid %d (%d x %d)", pid, w, h);
12707623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<Layer> normalLayer;
12719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (flags & eFXSurfaceMask) {
12729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceNormal:
1273d2112306330ce0c162bee4b864991962ca2b655aMathias Agopian            normalLayer = createNormalSurface(client, d, w, h, flags, format);
1274d2112306330ce0c162bee4b864991962ca2b655aMathias Agopian            layer = normalLayer;
12759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
12769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceBlur:
1277c3802d22f11b4e513ba776277d18f315c5c769f7Mathias Agopian            // for now we treat Blur as Dim, until we can implement it
1278c3802d22f11b4e513ba776277d18f315c5c769f7Mathias Agopian            // efficiently.
12799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceDim:
1280593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            layer = createDimSurface(client, d, w, h, flags);
12819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
12829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12841473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    if (layer != 0) {
1285593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        layer->initStates(w, h, flags);
12865d26c1e38dabb3ad8b4b6e1000375f3b1a6b7693Mathias Agopian        layer->setName(name);
1287593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        ssize_t token = addClientLayer(client, layer);
12887623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
12899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        surfaceHandle = layer->getSurface();
1290e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber        if (surfaceHandle != 0) {
1291593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            params->token = token;
12927bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian            params->identity = layer->getIdentity();
129318b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->width = w;
129418b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->height = h;
129518b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->format = format;
12967623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            if (normalLayer != 0) {
12977623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian                Mutex::Autolock _l(mStateLock);
12987bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                mLayerMap.add(layer->getSurfaceBinder(), normalLayer);
12997623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            }
130018b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian        }
13017623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
1302593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        setTransactionFlags(eTransactionNeeded);
13039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
13049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return surfaceHandle;
13069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13087623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<Layer> SurfaceFlinger::createNormalSurface(
13096edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian        const sp<Client>& client, DisplayID display,
1310593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        uint32_t w, uint32_t h, uint32_t flags,
131118b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian        PixelFormat& format)
13129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
13139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize the surfaces
13149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (format) { // TODO: take h/w into account
13159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_TRANSPARENT:
13169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_TRANSLUCENT:
13179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        format = PIXEL_FORMAT_RGBA_8888;
13189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
13199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_OPAQUE:
13204cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#ifdef NO_RGBX_8888
13214cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian        format = PIXEL_FORMAT_RGB_565;
13224cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#else
132359962ce3b0d8b7efec2840accca8fb7a6066f2bdMathias Agopian        format = PIXEL_FORMAT_RGBX_8888;
13244cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#endif
13259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
13269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
13279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13284cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#ifdef NO_RGBX_8888
13294cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian    if (format == PIXEL_FORMAT_RGBX_8888)
13304cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian        format = PIXEL_FORMAT_RGBA_8888;
13314cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#endif
13324cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian
1333593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<Layer> layer = new Layer(this, display, client);
13346edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian    status_t err = layer->setBuffers(w, h, format, flags);
1335593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (LIKELY(err != NO_ERROR)) {
13369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        LOGE("createNormalSurfaceLocked() failed (%s)", strerror(-err));
13371473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        layer.clear();
13389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
13399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return layer;
13409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13427623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<LayerDim> SurfaceFlinger::createDimSurface(
13436edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian        const sp<Client>& client, DisplayID display,
1344593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        uint32_t w, uint32_t h, uint32_t flags)
13459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1346593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<LayerDim> layer = new LayerDim(this, display, client);
13479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    layer->initStates(w, h, flags);
13489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return layer;
13499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1351593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::removeSurface(const sp<Client>& client, SurfaceID sid)
13526cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian{
13536cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    /*
13546cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     * called by the window manager, when a surface should be marked for
13556cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     * destruction.
1356e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber     *
1357a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * The surface is removed from the current and drawing lists, but placed
1358a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * in the purgatory queue, so it's not destroyed right-away (we need
1359a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * to wait for all client's references to go away first).
13606cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     */
13616cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
1362248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian    status_t err = NAME_NOT_FOUND;
1363a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian    Mutex::Autolock _l(mStateLock);
1364593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<LayerBaseClient> layer = client->getLayerUser(sid);
1365248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian    if (layer != 0) {
1366248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        err = purgatorizeLayer_l(layer);
1367248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        if (err == NO_ERROR) {
1368248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian            setTransactionFlags(eTransactionNeeded);
1369248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        }
13706cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    }
13716cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    return err;
13726cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian}
13736cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
13746960811d15eb66e8469330d3bc574c32c481db05Mathias Agopianstatus_t SurfaceFlinger::destroySurface(const wp<LayerBaseClient>& layer)
13759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1376359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian    // called by ~ISurface() when all references are gone
13776960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    status_t err = NO_ERROR;
13786960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    sp<LayerBaseClient> l(layer.promote());
13796960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    if (l != NULL) {
13806960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian        Mutex::Autolock _l(mStateLock);
13816960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian        err = removeLayer_l(l);
13826960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian        if (err == NAME_NOT_FOUND) {
13836960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian            // The surface wasn't in the current list, which means it was
13846960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian            // removed already, which means it is in the purgatory,
13856960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian            // and need to be removed from there.
13866960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian            ssize_t idx = mLayerPurgatory.remove(l);
13876960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian            LOGE_IF(idx < 0,
13886960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian                    "layer=%p is not in the purgatory list", l.get());
13896ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        }
13906960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian        LOGE_IF(err<0 && err != NAME_NOT_FOUND,
13916960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian                "error removing layer=%p (%s)", l.get(), strerror(-err));
13926960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    }
13936960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    return err;
13949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::setClientState(
1397593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const sp<Client>& client,
13989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int32_t count,
13999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const layer_state_t* states)
14009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
14019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
14029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t flags = 0;
14039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (int i=0 ; i<count ; i++) {
1404593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const layer_state_t& s(states[i]);
1405593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        sp<LayerBaseClient> layer(client->getLayerUser(s.surface));
14061473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        if (layer != 0) {
14079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const uint32_t what = s.what;
14089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & ePositionChanged) {
14099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setPosition(s.x, s.y))
14109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eLayerChanged) {
14131efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian                ssize_t idx = mCurrentState.layersSortedByZ.indexOf(layer);
14149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setLayer(s.z)) {
14151efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian                    mCurrentState.layersSortedByZ.removeAt(idx);
14161efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian                    mCurrentState.layersSortedByZ.add(layer);
14179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // we need traversal (state changed)
14189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // AND transaction (list changed)
14199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTransactionNeeded|eTraversalNeeded;
14209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
14219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eSizeChanged) {
14239779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                if (layer->setSize(s.w, s.h)) {
14249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14259779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                    mResizeTransationPending = true;
14269779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                }
14279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eAlphaChanged) {
14299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setAlpha(uint8_t(255.0f*s.alpha+0.5f)))
14309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eMatrixChanged) {
14339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setMatrix(s.matrix))
14349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eTransparentRegionChanged) {
14379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setTransparentRegionHint(s.transparentRegion))
14389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eVisibilityChanged) {
14419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setFlags(s.flags, s.mask))
14429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
14439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
14459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
14469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (flags) {
14479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setTransactionFlags(flags);
14489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
14499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
14509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
14529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::screenReleased(int dpy)
14539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
14549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // this may be called by a signal handler, we can't do too much in here
14559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    android_atomic_or(eConsoleReleased, &mConsoleSignals);
14569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    signalEvent();
14579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
14599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::screenAcquired(int dpy)
14609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
14619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // this may be called by a signal handler, we can't do too much in here
14629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    android_atomic_or(eConsoleAcquired, &mConsoleSignals);
14639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    signalEvent();
14649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
14669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::dump(int fd, const Vector<String16>& args)
14679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
146894720d7ec40d2b5ee200d95a46588cdb0fb30e66Erik Gilling    const size_t SIZE = 4096;
14699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    char buffer[SIZE];
14709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    String8 result;
14710dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian
14720dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian    if (!PermissionCache::checkCallingPermission(sDump)) {
14739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        snprintf(buffer, SIZE, "Permission Denial: "
14749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                "can't dump SurfaceFlinger from pid=%d, uid=%d\n",
14759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                IPCThreadState::self()->getCallingPid(),
14769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                IPCThreadState::self()->getCallingUid());
14779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        result.append(buffer);
14789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
1479a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1480a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // figure out if we're stuck somewhere
1481a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t now = systemTime();
1482a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t inSwapBuffers(mDebugInSwapBuffers);
1483a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t inTransaction(mDebugInTransaction);
1484a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        nsecs_t inSwapBuffersDuration = (inSwapBuffers) ? now-inSwapBuffers : 0;
1485a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        nsecs_t inTransactionDuration = (inTransaction) ? now-inTransaction : 0;
1486a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1487a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // Try to get the main lock, but don't insist if we can't
1488a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // (this would indicate SF is stuck, but we want to be able to
1489a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // print something in dumpsys).
1490a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        int retry = 3;
1491a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        while (mStateLock.tryLock()<0 && --retry>=0) {
1492a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            usleep(1000000);
1493a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
1494a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const bool locked(retry >= 0);
1495a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (!locked) {
1496e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber            snprintf(buffer, SIZE,
1497a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    "SurfaceFlinger appears to be unresponsive, "
1498a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    "dumping anyways (no locks held)\n");
1499a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1500a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
1501a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
150206a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
150306a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump the visible layer list
150406a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
15059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const LayerVector& currentLayers = mCurrentState.layersSortedByZ;
15069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const size_t count = currentLayers.size();
150706a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        snprintf(buffer, SIZE, "Visible layers (count = %d)\n", count);
150806a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        result.append(buffer);
15099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (size_t i=0 ; i<count ; i++) {
15109bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            const sp<LayerBase>& layer(currentLayers[i]);
15119bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            layer->dump(result, buffer, SIZE);
15129bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            const Layer::State& s(layer->drawingState());
15139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            s.transparentRegion.dump(result, "transparentRegion");
15149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->transparentRegionScreen.dump(result, "transparentRegionScreen");
15159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->visibleRegionScreen.dump(result, "visibleRegionScreen");
15169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
15179bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
151806a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
151906a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump the layers in the purgatory
152006a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
152106a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian
152206a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        const size_t purgatorySize =  mLayerPurgatory.size();
152306a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        snprintf(buffer, SIZE, "Purgatory state (%d entries)\n", purgatorySize);
152406a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        result.append(buffer);
152506a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        for (size_t i=0 ; i<purgatorySize ; i++) {
152606a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian            const sp<LayerBase>& layer(mLayerPurgatory.itemAt(i));
152706a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian            layer->shortDump(result, buffer, SIZE);
152806a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        }
152906a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian
153006a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
153106a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump SurfaceFlinger global state
153206a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
153306a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian
153406a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        snprintf(buffer, SIZE, "SurfaceFlinger global state\n");
153506a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        result.append(buffer);
15369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mWormholeRegion.dump(result, "WormholeRegion");
15379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw(graphicPlane(0).displayHardware());
15389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        snprintf(buffer, SIZE,
15399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                "  display frozen: %s, freezeCount=%d, orientation=%d, canDraw=%d\n",
15409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeDisplay?"yes":"no", mFreezeCount,
15419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mCurrentState.orientation, hw.canDraw());
15429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        result.append(buffer);
1543a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        snprintf(buffer, SIZE,
1544a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                "  last eglSwapBuffers() time: %f us\n"
1545a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                "  last transaction time     : %f us\n",
1546a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                mLastSwapBufferTime/1000.0, mLastTransactionTime/1000.0);
1547a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        result.append(buffer);
15489bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1549a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (inSwapBuffersDuration || !locked) {
1550a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            snprintf(buffer, SIZE, "  eglSwapBuffers time: %f us\n",
1551a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    inSwapBuffersDuration/1000.0);
1552a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1553a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
15549bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1555a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (inTransactionDuration || !locked) {
1556a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            snprintf(buffer, SIZE, "  transaction time: %f us\n",
1557a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    inTransactionDuration/1000.0);
1558a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1559a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
15609bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
156106a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
156206a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump HWComposer state
156306a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
15646a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        HWComposer& hwc(hw.getHwComposer());
15656a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        snprintf(buffer, SIZE, "  h/w composer %s and %s\n",
15666a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                hwc.initCheck()==NO_ERROR ? "present" : "not present",
15676a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                mDebugDisableHWC ? "disabled" : "enabled");
15686a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        result.append(buffer);
156990da4dd2a20c11fa2f9b860905c867a64b2b299eMathias Agopian        hwc.dump(result, buffer, SIZE);
15706a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian
157106a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
157206a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump gralloc state
157306a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
15746950e428feaccc8164b989ef64e771a99948797aMathias Agopian        const GraphicBufferAllocator& alloc(GraphicBufferAllocator::get());
15751473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        alloc.dump(result);
157694720d7ec40d2b5ee200d95a46588cdb0fb30e66Erik Gilling        hw.dump(result);
1577a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1578a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (locked) {
1579a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            mStateLock.unlock();
1580a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
15819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
15829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    write(fd, result.string(), result.size());
15839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
15849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
15859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
15869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::onTransact(
15879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
15889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
15899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (code) {
15909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case CREATE_CONNECTION:
15919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case OPEN_GLOBAL_TRANSACTION:
15929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case CLOSE_GLOBAL_TRANSACTION:
15939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case SET_ORIENTATION:
15949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case FREEZE_DISPLAY:
15959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case UNFREEZE_DISPLAY:
15969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case BOOT_FINISHED:
1597aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        case TURN_ELECTRON_BEAM_OFF:
15982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        case TURN_ELECTRON_BEAM_ON:
15999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        {
16009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // codes that require permission check
16019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            IPCThreadState* ipc = IPCThreadState::self();
16029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int pid = ipc->getCallingPid();
1603627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian            const int uid = ipc->getCallingUid();
16040dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian            if ((uid != AID_GRAPHICS) &&
16050dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian                    !PermissionCache::checkPermission(sAccessSurfaceFlinger, pid, uid)) {
1606151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                LOGE("Permission Denial: "
1607151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                        "can't access SurfaceFlinger pid=%d, uid=%d", pid, uid);
1608151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                return PERMISSION_DENIED;
16099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
1610ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            break;
1611ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
1612ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        case CAPTURE_SCREEN:
1613ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        {
1614ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            // codes that require permission check
1615ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            IPCThreadState* ipc = IPCThreadState::self();
1616ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            const int pid = ipc->getCallingPid();
1617ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            const int uid = ipc->getCallingUid();
16180dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian            if ((uid != AID_GRAPHICS) &&
16190dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian                    !PermissionCache::checkPermission(sReadFramebuffer, pid, uid)) {
1620ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                LOGE("Permission Denial: "
1621ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                        "can't read framebuffer pid=%d, uid=%d", pid, uid);
1622ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                return PERMISSION_DENIED;
1623ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            }
1624ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            break;
16259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
16269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1627ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
16289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    status_t err = BnSurfaceComposer::onTransact(code, data, reply, flags);
16299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (err == UNKNOWN_TRANSACTION || err == PERMISSION_DENIED) {
16308c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        CHECK_INTERFACE(ISurfaceComposer, data, reply);
16310dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian        if (UNLIKELY(!PermissionCache::checkCallingPermission(sHardwareTest))) {
1632151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            IPCThreadState* ipc = IPCThreadState::self();
1633151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            const int pid = ipc->getCallingPid();
1634151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            const int uid = ipc->getCallingUid();
1635151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            LOGE("Permission Denial: "
1636151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                    "can't access SurfaceFlinger pid=%d, uid=%d", pid, uid);
16379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return PERMISSION_DENIED;
16389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
16399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int n;
16409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (code) {
164117f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian            case 1000: // SHOW_CPU, NOT SUPPORTED ANYMORE
164204262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            case 1001: // SHOW_FPS, NOT SUPPORTED ANYMORE
16439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1002:  // SHOW_UPDATES
16459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                n = data.readInt32();
16469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDebugRegion = n ? n : (mDebugRegion ? 0 : 1);
16479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1003:  // SHOW_BACKGROUND
16499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                n = data.readInt32();
16509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDebugBackground = n ? 1 : 0;
16519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16526a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian            case 1008:  // toggle use of hw composer
16536a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                n = data.readInt32();
16546a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                mDebugDisableHWC = n ? 1 : 0;
1655fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian                invalidateHwcGeometry();
16566a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                // fall-through...
16579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1004:{ // repaint everything
16589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                Mutex::Autolock _l(mStateLock);
16599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                const DisplayHardware& hw(graphicPlane(0).displayHardware());
16609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDirtyRegion.set(hw.bounds()); // careful that's not thread-safe
16619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                signalEvent();
16629779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                return NO_ERROR;
16639779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian            }
16649779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian            case 1005:{ // force transaction
16659779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                setTransactionFlags(eTransactionNeeded|eTraversalNeeded);
16669779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                return NO_ERROR;
16679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
166804262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            case 1006:{ // enable/disable GraphicLog
166904262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                int enabled = data.readInt32();
167004262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                GraphicLog::getInstance().setEnabled(enabled);
167104262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                return NO_ERROR;
167204262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            }
16739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1007: // set mFreezeCount
16749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeCount = data.readInt32();
16750e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                mFreezeDisplayTime = 0;
16769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1010:  // interrogate.
167817f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian                reply->writeInt32(0);
16799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(0);
16809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(mDebugRegion);
16819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(mDebugBackground);
16829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1013: {
16849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                Mutex::Autolock _l(mStateLock);
16859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                const DisplayHardware& hw(graphicPlane(0).displayHardware());
16869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(hw.getPageFlipCount());
16879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
16889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return NO_ERROR;
16899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
16909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
16919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return err;
16929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
16939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1694aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian// ---------------------------------------------------------------------------
1695aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
16962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::renderScreenToTextureLocked(DisplayID dpy,
16972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        GLuint* textureName, GLfloat* uOut, GLfloat* vOut)
1698aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian{
1699aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
1700aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        return INVALID_OPERATION;
1701aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1702aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // get screen geometry
1703aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const DisplayHardware& hw(graphicPlane(dpy).displayHardware());
1704aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const uint32_t hw_w = hw.getWidth();
1705aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const uint32_t hw_h = hw.getHeight();
1706aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLfloat u = 1;
1707aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLfloat v = 1;
1708aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1709aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // make sure to clear all GL error flags
1710aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    while ( glGetError() != GL_NO_ERROR ) ;
1711aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1712aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // create a FBO
1713aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLuint name, tname;
1714aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glGenTextures(1, &tname);
1715aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glBindTexture(GL_TEXTURE_2D, tname);
17162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB,
17172d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            hw_w, hw_h, 0, GL_RGB, GL_UNSIGNED_BYTE, 0);
1718aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (glGetError() != GL_NO_ERROR) {
1719a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian        while ( glGetError() != GL_NO_ERROR ) ;
1720aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        GLint tw = (2 << (31 - clz(hw_w)));
1721aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        GLint th = (2 << (31 - clz(hw_h)));
17222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB,
17232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian                tw, th, 0, GL_RGB, GL_UNSIGNED_BYTE, 0);
1724aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        u = GLfloat(hw_w) / tw;
1725aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        v = GLfloat(hw_h) / th;
1726aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
1727aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glGenFramebuffersOES(1, &name);
1728aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, name);
17292d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glFramebufferTexture2DOES(GL_FRAMEBUFFER_OES,
17302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            GL_COLOR_ATTACHMENT0_OES, GL_TEXTURE_2D, tname, 0);
1731aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17322d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // redraw the screen entirely...
17332d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glClearColor(0,0,0,1);
17342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glClear(GL_COLOR_BUFFER_BIT);
17352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
17362d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const size_t count = layers.size();
17372d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (size_t i=0 ; i<count ; ++i) {
17382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
17392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        layer->drawForSreenShot();
17402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
1741aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // back to main framebuffer
17432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
17442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_SCISSOR_TEST);
17452d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDeleteFramebuffersOES(1, &name);
1746aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17472d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    *textureName = tname;
17482d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    *uOut = u;
17492d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    *vOut = v;
17502d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
17512d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
1752aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17532d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian// ---------------------------------------------------------------------------
1754aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17552d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::electronBeamOffAnimationImplLocked()
17562d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
17572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    status_t result = PERMISSION_DENIED;
1758aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17592d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
17602d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return INVALID_OPERATION;
1761aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17622d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // get screen geometry
17632d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
17642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_w = hw.getWidth();
17652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_h = hw.getHeight();
17662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const Region screenBounds(hw.bounds());
1767aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat u, v;
17692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLuint tname;
17702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    result = renderScreenToTextureLocked(0, &tname, &u, &v);
17712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (result != NO_ERROR) {
17722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return result;
17732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
1774aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17752d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat vtx[8];
17762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const GLfloat texCoords[4][2] = { {0,v}, {0,0}, {u,0}, {u,v} };
17772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindTexture(GL_TEXTURE_2D, tname);
17782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
17792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
17802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
17812d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexCoordPointer(2, GL_FLOAT, 0, texCoords);
17822d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnableClientState(GL_TEXTURE_COORD_ARRAY);
17832d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glVertexPointer(2, GL_FLOAT, 0, vtx);
17842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
17852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class s_curve_interpolator {
17862d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float nbFrames, s, v;
17872d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
17882d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        s_curve_interpolator(int nbFrames, float s)
17892d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : nbFrames(1.0f / (nbFrames-1)), s(s),
17902d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian          v(1.0f + expf(-s + 0.5f*s)) {
17912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
17922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float operator()(int f) {
17932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const float x = f * nbFrames;
17942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return ((1.0f/(1.0f + expf(-x*s + 0.5f*s))) - 0.5f) * v + 0.5f;
17952d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
17962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
17972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
17982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class v_stretch {
17992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
18002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
18012d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        v_stretch(uint32_t hw_w, uint32_t hw_h)
18022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
18032d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
18042d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
18052d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w + (hw_w * v);
18062d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = hw_h - (hw_h * v);
18072d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
18082d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
18092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
18102d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
18112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
18122d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
18132d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
18142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
18152d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class h_stretch {
18172d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
18182d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
18192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        h_stretch(uint32_t hw_w, uint32_t hw_h)
18202d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
1821aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
18222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
18232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w - (hw_w * v);
18242d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = 1.0f;
18252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
18262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
18272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
18282d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
18292d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
18302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
18312d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
18322d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
18332d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // the full animation is 24 frames
18352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const int nbFrames = 12;
18362d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itr(nbFrames, 7.5f);
18372d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itg(nbFrames, 8.0f);
18382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itb(nbFrames, 8.5f);
18392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    v_stretch vverts(hw_w, hw_h);
18412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_BLEND);
18422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBlendFunc(GL_ONE, GL_ONE);
18432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=0 ; i<nbFrames ; i++) {
18442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float x, y, w, h;
18452d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vr = itr(i);
18462d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vg = itg(i);
18472d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vb = itb(i);
18482d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18492d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // clear screen
18502d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(1,1,1,1);
18512d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
18522d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glEnable(GL_TEXTURE_2D);
1853aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
18542d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the red plane
18552d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vr);
18562d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(1,0,0,1);
18572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
18582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18592d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the green plane
18602d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vg);
18612d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,1,0,1);
18622d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
18632d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the blue plane
18652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vb);
18662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,0,1,1);
18672d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
18682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the white highlight (we use the last vertices)
1870aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glDisable(GL_TEXTURE_2D);
1871aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glColorMask(1,1,1,1);
18722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColor4f(vg, vg, vg, 1);
18732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
18742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
18752d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
18762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    h_stretch hverts(hw_w, hw_h);
18782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_BLEND);
18792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_TEXTURE_2D);
18802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
18812d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=0 ; i<nbFrames ; i++) {
18822d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float v = itg(i);
18832d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hverts(vtx, v);
18842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
18852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColor4f(1-v, 1-v, 1-v, 1);
18862d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
18872d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
18882d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
18892d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18902d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
18912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_SCISSOR_TEST);
18922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisableClientState(GL_TEXTURE_COORD_ARRAY);
18932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDeleteTextures(1, &tname);
18947bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    glDisable(GL_TEXTURE_2D);
18952d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
18962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
18972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::electronBeamOnAnimationImplLocked()
18992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
19002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    status_t result = PERMISSION_DENIED;
19012d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
19032d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return INVALID_OPERATION;
19042d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19052d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19062d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // get screen geometry
19072d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
19082d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_w = hw.getWidth();
19092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_h = hw.getHeight();
19102d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const Region screenBounds(hw.bounds());
19112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19122d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat u, v;
19132d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLuint tname;
19142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    result = renderScreenToTextureLocked(0, &tname, &u, &v);
19152d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (result != NO_ERROR) {
19162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return result;
19172d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
19182d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // back to main framebuffer
19202d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
19212d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_SCISSOR_TEST);
19222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat vtx[8];
19242d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const GLfloat texCoords[4][2] = { {0,v}, {0,0}, {u,0}, {u,v} };
19252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindTexture(GL_TEXTURE_2D, tname);
19262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE);
19272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
19282d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
19292d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexCoordPointer(2, GL_FLOAT, 0, texCoords);
19302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnableClientState(GL_TEXTURE_COORD_ARRAY);
19312d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glVertexPointer(2, GL_FLOAT, 0, vtx);
19322d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19332d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class s_curve_interpolator {
19342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float nbFrames, s, v;
19352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
19362d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        s_curve_interpolator(int nbFrames, float s)
19372d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : nbFrames(1.0f / (nbFrames-1)), s(s),
19382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian          v(1.0f + expf(-s + 0.5f*s)) {
19392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float operator()(int f) {
19412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const float x = f * nbFrames;
19422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return ((1.0f/(1.0f + expf(-x*s + 0.5f*s))) - 0.5f) * v + 0.5f;
19432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
19452d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19462d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class v_stretch {
19472d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
19482d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
19492d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        v_stretch(uint32_t hw_w, uint32_t hw_h)
19502d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
1951aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
19522d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
19532d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w + (hw_w * v);
19542d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = hw_h - (hw_h * v);
19552d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
19562d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
19572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
19582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
19592d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
19602d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
19612d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19622d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
19632d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class h_stretch {
19652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
19662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
19672d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        h_stretch(uint32_t hw_w, uint32_t hw_h)
19682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
19692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
19712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w - (hw_w * v);
19722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = 1.0f;
19732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
19742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
19752d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
19762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
19772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
19782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
19792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
19812d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
1982dfa08fbfd8a5ebd013c122e391282b7f21f6342fMathias Agopian    // the full animation is 12 frames
1983dfa08fbfd8a5ebd013c122e391282b7f21f6342fMathias Agopian    int nbFrames = 8;
19842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itr(nbFrames, 7.5f);
19852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itg(nbFrames, 8.0f);
19862d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itb(nbFrames, 8.5f);
19872d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19882d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    h_stretch hverts(hw_w, hw_h);
19892d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_BLEND);
19902d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_TEXTURE_2D);
19912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
19922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=nbFrames-1 ; i>=0 ; i--) {
19932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float v = itg(i);
19942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hverts(vtx, v);
19952d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
19962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColor4f(1-v, 1-v, 1-v, 1);
19972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
19982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
19992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
20002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
2001dfa08fbfd8a5ebd013c122e391282b7f21f6342fMathias Agopian    nbFrames = 4;
20022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    v_stretch vverts(hw_w, hw_h);
20032d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_BLEND);
20042d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBlendFunc(GL_ONE, GL_ONE);
20052d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=nbFrames-1 ; i>=0 ; i--) {
20062d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float x, y, w, h;
20072d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vr = itr(i);
20082d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vg = itg(i);
20092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vb = itb(i);
2010aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
20112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // clear screen
2012aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glColorMask(1,1,1,1);
20132d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
20142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glEnable(GL_TEXTURE_2D);
20152d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the red plane
20172d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vr);
20182d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(1,0,0,1);
20192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
20202d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20212d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the green plane
20222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vg);
20232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,1,0,1);
20242d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
20252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the blue plane
20272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vb);
20282d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,0,1,1);
20292d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
20302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20312d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
2032aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
2033aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
20342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
20352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_SCISSOR_TEST);
20362d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisableClientState(GL_TEXTURE_COORD_ARRAY);
2037aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glDeleteTextures(1, &tname);
20387bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    glDisable(GL_TEXTURE_2D);
2039aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
20402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
20412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
20422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian// ---------------------------------------------------------------------------
20442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
2045d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOffImplLocked(int32_t mode)
20462d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
20472d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    DisplayHardware& hw(graphicPlane(0).editDisplayHardware());
20482d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (!hw.canDraw()) {
20492d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // we're already off
20502d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return NO_ERROR;
20512d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
2052d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    if (mode & ISurfaceComposer::eElectronBeamAnimationOff) {
2053d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        electronBeamOffAnimationImplLocked();
2054d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    }
2055d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian
2056d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    // always clear the whole screen at the end of the animation
2057d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glClearColor(0,0,0,1);
2058d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glDisable(GL_SCISSOR_TEST);
2059d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glClear(GL_COLOR_BUFFER_BIT);
2060d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glEnable(GL_SCISSOR_TEST);
2061d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    hw.flip( Region(hw.bounds()) );
2062d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian
2063a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    hw.setCanDraw(false);
2064a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    return NO_ERROR;
2065aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
2066aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
2067aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOff(int32_t mode)
2068aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian{
2069aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    class MessageTurnElectronBeamOff : public MessageBase {
2070aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        SurfaceFlinger* flinger;
2071d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        int32_t mode;
2072aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        status_t result;
2073aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    public:
2074d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        MessageTurnElectronBeamOff(SurfaceFlinger* flinger, int32_t mode)
2075d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            : flinger(flinger), mode(mode), result(PERMISSION_DENIED) {
2076aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
2077aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        status_t getResult() const {
2078aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            return result;
2079aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
2080aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        virtual bool handler() {
2081aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
2082d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            result = flinger->turnElectronBeamOffImplLocked(mode);
2083aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            return true;
2084aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
2085aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    };
2086aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
2087d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    sp<MessageBase> msg = new MessageTurnElectronBeamOff(this, mode);
2088aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    status_t res = postMessageSync(msg);
2089aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (res == NO_ERROR) {
2090aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        res = static_cast<MessageTurnElectronBeamOff*>( msg.get() )->getResult();
20912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // work-around: when the power-manager calls us we activate the
20932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // animation. eventually, the "on" animation will be called
20942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // by the power-manager itself
2095d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        mElectronBeamAnimationMode = mode;
2096aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
2097aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    return res;
2098aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
2099aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
21009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
21019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2102d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOnImplLocked(int32_t mode)
21032d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
21042d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    DisplayHardware& hw(graphicPlane(0).editDisplayHardware());
21052d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (hw.canDraw()) {
21062d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // we're already on
21072d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return NO_ERROR;
21082d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
2109d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    if (mode & ISurfaceComposer::eElectronBeamAnimationOn) {
2110d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        electronBeamOnAnimationImplLocked();
2111d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    }
2112a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    hw.setCanDraw(true);
21138b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian
21148b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian    // make sure to redraw the whole screen when the animation is done
21158b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian    mDirtyRegion.set(hw.bounds());
21168b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian    signalEvent();
21178b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian
2118a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    return NO_ERROR;
21192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
21202d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
21212d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOn(int32_t mode)
21222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
21232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class MessageTurnElectronBeamOn : public MessageBase {
21242d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        SurfaceFlinger* flinger;
2125d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        int32_t mode;
21262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        status_t result;
21272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
2128d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        MessageTurnElectronBeamOn(SurfaceFlinger* flinger, int32_t mode)
2129d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            : flinger(flinger), mode(mode), result(PERMISSION_DENIED) {
21302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
21312d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        status_t getResult() const {
21322d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return result;
21332d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
21342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        virtual bool handler() {
21352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
2136d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            result = flinger->turnElectronBeamOnImplLocked(mode);
21372d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return true;
21382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
21392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
21402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
2141d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    postMessageAsync( new MessageTurnElectronBeamOn(this, mode) );
21422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
21432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
21442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
21452d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian// ---------------------------------------------------------------------------
21462d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
214738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopianstatus_t SurfaceFlinger::captureScreenImplLocked(DisplayID dpy,
214838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        sp<IMemoryHeap>* heap,
214938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t* w, uint32_t* h, PixelFormat* f,
21503dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t sw, uint32_t sh,
21513dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t minLayerZ, uint32_t maxLayerZ)
215238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian{
215338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    status_t result = PERMISSION_DENIED;
215438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
215538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // only one display supported for now
215638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
215738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        return BAD_VALUE;
215838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
2159f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis    // make sure none of the layers are protected
2160f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis    const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
2161f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis    const size_t count = layers.size();
2162f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis    for (size_t i=0 ; i<count ; ++i) {
2163f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis        const sp<LayerBase>& layer(layers[i]);
2164f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis        const uint32_t z = layer->drawingState().z;
2165f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis        if (z >= minLayerZ && z <= maxLayerZ) {
2166f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis            if (layer->isProtected()) {
2167f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis                return INVALID_OPERATION;
2168f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis            }
2169f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis        }
2170f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis    }
2171f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis
217238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
217338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        return INVALID_OPERATION;
217438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
217538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // get screen geometry
217638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const DisplayHardware& hw(graphicPlane(dpy).displayHardware());
217738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const uint32_t hw_w = hw.getWidth();
217838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const uint32_t hw_h = hw.getHeight();
217938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
218038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if ((sw > hw_w) || (sh > hw_h))
218138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        return BAD_VALUE;
218238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
218338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    sw = (!sw) ? hw_w : sw;
218438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    sh = (!sh) ? hw_h : sh;
218538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const size_t size = sw * sh * 4;
218638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
218732ae094d87cbb35f8b31acb7b83b430db62d5925Mathias Agopian    //LOGD("screenshot: sw=%d, sh=%d, minZ=%d, maxZ=%d",
218832ae094d87cbb35f8b31acb7b83b430db62d5925Mathias Agopian    //        sw, sh, minLayerZ, maxLayerZ);
2189cd2cfb6a15975af02a02c34783480eb858ee2928Mathias Agopian
219038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // make sure to clear all GL error flags
219138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    while ( glGetError() != GL_NO_ERROR ) ;
219238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
219338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // create a FBO
219438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    GLuint name, tname;
219538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glGenRenderbuffersOES(1, &tname);
219638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glBindRenderbufferOES(GL_RENDERBUFFER_OES, tname);
219738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glRenderbufferStorageOES(GL_RENDERBUFFER_OES, GL_RGBA8_OES, sw, sh);
219838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glGenFramebuffersOES(1, &name);
219938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, name);
220038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES,
220138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            GL_COLOR_ATTACHMENT0_OES, GL_RENDERBUFFER_OES, tname);
220238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
220338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    GLenum status = glCheckFramebufferStatusOES(GL_FRAMEBUFFER_OES);
2204cd2cfb6a15975af02a02c34783480eb858ee2928Mathias Agopian
220538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if (status == GL_FRAMEBUFFER_COMPLETE_OES) {
220638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
220738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // invert everything, b/c glReadPixel() below will invert the FB
220838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glViewport(0, 0, sw, sh);
22091c4e4fc0490c6f07111365cfe21116c31254d5ffMathias Agopian        glScissor(0, 0, sw, sh);
221038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_PROJECTION);
221138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glPushMatrix();
221238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glLoadIdentity();
221338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glOrthof(0, hw_w, 0, hw_h, 0, 1);
221438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_MODELVIEW);
221538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
221638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // redraw the screen entirely...
221738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glClearColor(0,0,0,1);
221838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
22191c4e4fc0490c6f07111365cfe21116c31254d5ffMathias Agopian
222038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        for (size_t i=0 ; i<count ; ++i) {
222138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            const sp<LayerBase>& layer(layers[i]);
22223dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            const uint32_t z = layer->drawingState().z;
22233dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            if (z >= minLayerZ && z <= maxLayerZ) {
22243dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                layer->drawForSreenShot();
22253dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            }
222638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        }
222738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
222838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // XXX: this is needed on tegra
222938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glScissor(0, 0, sw, sh);
223038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
223138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // check for errors and return screen capture
223238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        if (glGetError() != GL_NO_ERROR) {
223338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            // error while rendering
223438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            result = INVALID_OPERATION;
223538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        } else {
223638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            // allocate shared memory large enough to hold the
223738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            // screen capture
223838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            sp<MemoryHeapBase> base(
223938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    new MemoryHeapBase(size, 0, "screen-capture") );
224038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            void* const ptr = base->getBase();
224138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            if (ptr) {
224238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                // capture the screen with glReadPixels()
224338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                glReadPixels(0, 0, sw, sh, GL_RGBA, GL_UNSIGNED_BYTE, ptr);
224438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                if (glGetError() == GL_NO_ERROR) {
224538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *heap = base;
224638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *w = sw;
224738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *h = sh;
224838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *f = PIXEL_FORMAT_RGBA_8888;
224938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    result = NO_ERROR;
225038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                }
225138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            } else {
225238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                result = NO_MEMORY;
225338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            }
225438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        }
225538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glEnable(GL_SCISSOR_TEST);
225638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glViewport(0, 0, hw_w, hw_h);
225738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_PROJECTION);
225838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glPopMatrix();
225938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_MODELVIEW);
226038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    } else {
226138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        result = BAD_VALUE;
226238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    }
226338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
226438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // release FBO resources
226538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
226638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glDeleteRenderbuffersOES(1, &tname);
226738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glDeleteFramebuffersOES(1, &name);
2268a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian
2269a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian    hw.compositionComplete();
2270a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian
227132ae094d87cbb35f8b31acb7b83b430db62d5925Mathias Agopian    // LOGD("screenshot: result = %s", result<0 ? strerror(result) : "OK");
2272cd2cfb6a15975af02a02c34783480eb858ee2928Mathias Agopian
227338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    return result;
227438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian}
227538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
227638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
2277ca5edbeba92b96913291792a4df984e158853b6dMathias Agopianstatus_t SurfaceFlinger::captureScreen(DisplayID dpy,
2278ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        sp<IMemoryHeap>* heap,
227938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t* width, uint32_t* height, PixelFormat* format,
22803dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t sw, uint32_t sh,
22813dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t minLayerZ, uint32_t maxLayerZ)
2282ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian{
2283ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    // only one display supported for now
2284ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
2285ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        return BAD_VALUE;
2286ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2287ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
2288ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        return INVALID_OPERATION;
2289ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2290ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    class MessageCaptureScreen : public MessageBase {
2291ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        SurfaceFlinger* flinger;
2292ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        DisplayID dpy;
2293ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        sp<IMemoryHeap>* heap;
2294ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        uint32_t* w;
2295ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        uint32_t* h;
2296ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        PixelFormat* f;
229738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t sw;
229838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t sh;
22993dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t minLayerZ;
23003dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t maxLayerZ;
2301ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        status_t result;
2302ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    public:
2303ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        MessageCaptureScreen(SurfaceFlinger* flinger, DisplayID dpy,
230438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                sp<IMemoryHeap>* heap, uint32_t* w, uint32_t* h, PixelFormat* f,
23053dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                uint32_t sw, uint32_t sh,
23063dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                uint32_t minLayerZ, uint32_t maxLayerZ)
2307ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            : flinger(flinger), dpy(dpy),
23083dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian              heap(heap), w(w), h(h), f(f), sw(sw), sh(sh),
23093dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian              minLayerZ(minLayerZ), maxLayerZ(maxLayerZ),
23103dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian              result(PERMISSION_DENIED)
2311ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        {
2312ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
2313ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        status_t getResult() const {
2314ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            return result;
2315ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
2316ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        virtual bool handler() {
2317ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
2318ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2319ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            // if we have secure windows, never allow the screen capture
2320ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            if (flinger->mSecureFrameBuffer)
2321ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                return true;
2322ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
232338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            result = flinger->captureScreenImplLocked(dpy,
23243dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                    heap, w, h, f, sw, sh, minLayerZ, maxLayerZ);
2325ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2326ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            return true;
2327ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
2328ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    };
2329ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2330ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    sp<MessageBase> msg = new MessageCaptureScreen(this,
23313dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            dpy, heap, width, height, format, sw, sh, minLayerZ, maxLayerZ);
2332ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    status_t res = postMessageSync(msg);
2333ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (res == NO_ERROR) {
2334ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        res = static_cast<MessageCaptureScreen*>( msg.get() )->getResult();
2335ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    }
2336ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    return res;
2337ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian}
2338ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2339ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian// ---------------------------------------------------------------------------
2340ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
23417623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<Layer> SurfaceFlinger::getLayer(const sp<ISurface>& sur) const
23429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
23437623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<Layer> result;
23447623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    Mutex::Autolock _l(mStateLock);
23457623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    result = mLayerMap.valueFor( sur->asBinder() ).promote();
23467623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return result;
23477623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
2348d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
23497623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian// ---------------------------------------------------------------------------
2350593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
23517623da435e45c7c03ef6a00a43675deb6645f070Mathias AgopianClient::Client(const sp<SurfaceFlinger>& flinger)
23527623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    : mFlinger(flinger), mNameGenerator(1)
23537623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
23549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
23559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2356593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias AgopianClient::~Client()
2357593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
2358593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    const size_t count = mLayers.size();
2359593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
2360593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        sp<LayerBaseClient> layer(mLayers.valueAt(i).promote());
2361593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        if (layer != 0) {
2362593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            mFlinger->removeLayer(layer);
2363593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        }
23649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
23659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
23661473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
2367593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t Client::initCheck() const {
23687623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return NO_ERROR;
23699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
23701473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
23715fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopiansize_t Client::attachLayer(const sp<LayerBaseClient>& layer)
23729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
23735fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    Mutex::Autolock _l(mLock);
23745fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    size_t name = mNameGenerator++;
2375593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    mLayers.add(name, layer);
2376593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return name;
23779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
23789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
23797623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianvoid Client::detachLayer(const LayerBaseClient* layer)
2380593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
23815fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    Mutex::Autolock _l(mLock);
2382593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // we do a linear search here, because this doesn't happen often
2383593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    const size_t count = mLayers.size();
2384593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
2385593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        if (mLayers.valueAt(i) == layer) {
2386593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            mLayers.removeItemsAt(i, 1);
2387593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            break;
2388593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        }
2389593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    }
2390593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian}
23915fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopiansp<LayerBaseClient> Client::getLayerUser(int32_t i) const
23925fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian{
23935fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    Mutex::Autolock _l(mLock);
23941473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBaseClient> lbc;
23955fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    wp<LayerBaseClient> layer(mLayers.valueFor(i));
2396593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (layer != 0) {
2397593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        lbc = layer.promote();
2398593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        LOGE_IF(lbc==0, "getLayerUser(name=%d) is dead", int(i));
23991473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    }
24001473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    return lbc;
24019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
24029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
24037bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian
24047bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopianstatus_t Client::onTransact(
24057bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
24067bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian{
24077bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    // these must be checked
24087bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     IPCThreadState* ipc = IPCThreadState::self();
24097bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     const int pid = ipc->getCallingPid();
24107bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     const int uid = ipc->getCallingUid();
24117bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     const int self_pid = getpid();
24127bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     if (UNLIKELY(pid != self_pid && uid != AID_GRAPHICS && uid != 0)) {
24137bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian         // we're called from a different process, do the real check
24140dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian         if (!PermissionCache::checkCallingPermission(sAccessSurfaceFlinger))
24157bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian         {
24167bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian             LOGE("Permission Denial: "
24177bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                     "can't openGlobalTransaction pid=%d, uid=%d", pid, uid);
24187bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian             return PERMISSION_DENIED;
24197bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian         }
24207bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     }
24217bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     return BnSurfaceComposerClient::onTransact(code, data, reply, flags);
24229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
24237bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian
24247bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian
2425593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopiansp<ISurface> Client::createSurface(
24269638e5c167be321643bf3f3ee39e3fb45541fb3bMathias Agopian        ISurfaceComposerClient::surface_data_t* params,
24277623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const String8& name,
24287623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        DisplayID display, uint32_t w, uint32_t h, PixelFormat format,
24299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        uint32_t flags)
24309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
24317623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    /*
24327bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     * createSurface must be called from the GL thread so that it can
24337bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     * have access to the GL context.
24347623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     */
24357623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24367bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    class MessageCreateSurface : public MessageBase {
24377bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        sp<ISurface> result;
24387bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        SurfaceFlinger* flinger;
24397bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        ISurfaceComposerClient::surface_data_t* params;
24407bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        Client* client;
24417bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        const String8& name;
24427bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        DisplayID display;
24437bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        uint32_t w, h;
24447bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        PixelFormat format;
24457bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        uint32_t flags;
24467bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    public:
24477bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        MessageCreateSurface(SurfaceFlinger* flinger,
24487bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                ISurfaceComposerClient::surface_data_t* params,
24497bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                const String8& name, Client* client,
24507bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                DisplayID display, uint32_t w, uint32_t h, PixelFormat format,
24517bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                uint32_t flags)
24527bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian            : flinger(flinger), params(params), client(client), name(name),
24537bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian              display(display), w(w), h(h), format(format), flags(flags)
24547bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        {
24555e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        }
24567bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        sp<ISurface> getResult() const { return result; }
24577bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        virtual bool handler() {
24587bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian            result = flinger->createSurface(params, name, client,
24597bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                    display, w, h, format, flags);
24607bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian            return true;
24617623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        }
24627bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    };
24637623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24647bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    sp<MessageBase> msg = new MessageCreateSurface(mFlinger.get(),
24657bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian            params, name, this, display, w, h, format, flags);
24667bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    mFlinger->postMessageSync(msg);
24677bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    return static_cast<MessageCreateSurface*>( msg.get() )->getResult();
24687623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24697bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopianstatus_t Client::destroySurface(SurfaceID sid) {
24707bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    return mFlinger->removeSurface(this, sid);
24717623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24727bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopianstatus_t Client::setState(int32_t count, const layer_state_t* states) {
24737bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    return mFlinger->setClientState(this, count, states);
24747623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
24757623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24767623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian// ---------------------------------------------------------------------------
24779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2478f7acf162f8d682c6ebc9af41ca76795b79509193Jamie GennisGraphicBufferAlloc::GraphicBufferAlloc() {}
2479f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis
2480f7acf162f8d682c6ebc9af41ca76795b79509193Jamie GennisGraphicBufferAlloc::~GraphicBufferAlloc() {}
2481f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis
2482f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennissp<GraphicBuffer> GraphicBufferAlloc::createGraphicBuffer(uint32_t w, uint32_t h,
2483f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis        PixelFormat format, uint32_t usage) {
2484f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    sp<GraphicBuffer> graphicBuffer(new GraphicBuffer(w, h, format, usage));
2485f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    status_t err = graphicBuffer->initCheck();
24867bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    if (err != 0 || graphicBuffer->handle == 0) {
24877bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        GraphicBuffer::dumpAllocationsToSystemLog();
24887bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        LOGE("GraphicBufferAlloc::createGraphicBuffer(w=%d, h=%d) "
24897bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian             "failed (%s), handle=%p",
24907bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                w, h, strerror(-err), graphicBuffer->handle);
2491f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis        return 0;
2492f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    }
2493f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    return graphicBuffer;
2494f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis}
2495f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis
2496f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis// ---------------------------------------------------------------------------
2497f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis
24989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane::GraphicPlane()
24999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    : mHw(0)
25009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
25019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane::~GraphicPlane() {
25049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    delete mHw;
25059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool GraphicPlane::initialized() const {
25089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return mHw ? true : false;
25099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
251166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianint GraphicPlane::getWidth() const {
251266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    return mWidth;
25139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
251566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianint GraphicPlane::getHeight() const {
251666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    return mHeight;
251766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian}
251866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
251966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianvoid GraphicPlane::setDisplayHardware(DisplayHardware *hw)
252066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian{
252166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mHw = hw;
252266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
252366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    // initialize the display orientation transform.
252466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    // it's a constant that should come from the display driver.
252566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    int displayOrientation = ISurfaceComposer::eOrientationDefault;
252666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    char property[PROPERTY_VALUE_MAX];
252766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    if (property_get("ro.sf.hwrotation", property, NULL) > 0) {
252866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        //displayOrientation
252966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        switch (atoi(property)) {
253066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        case 90:
253166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            displayOrientation = ISurfaceComposer::eOrientation90;
253266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            break;
253366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        case 270:
253466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            displayOrientation = ISurfaceComposer::eOrientation270;
253566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            break;
253666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        }
253766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    }
253866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
253966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float w = hw->getWidth();
254066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float h = hw->getHeight();
254166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    GraphicPlane::orientationToTransfrom(displayOrientation, w, h,
254266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            &mDisplayTransform);
254366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    if (displayOrientation & ISurfaceComposer::eOrientationSwapMask) {
254466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayWidth = h;
254566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayHeight = w;
254666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    } else {
254766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayWidth = w;
254866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayHeight = h;
254966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    }
255066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
255166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    setOrientation(ISurfaceComposer::eOrientationDefault);
25529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t GraphicPlane::orientationToTransfrom(
25559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int orientation, int w, int h, Transform* tr)
25568c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian{
25578c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    uint32_t flags = 0;
25589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (orientation) {
25599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientationDefault:
25608c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_0;
25618c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        break;
25629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation90:
25638c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_90;
25649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
25659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation180:
25668c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_180;
25679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
25689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation270:
25698c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_270;
25709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
25719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    default:
25729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
25739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
25748c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    tr->set(flags, w, h);
25759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
25769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t GraphicPlane::setOrientation(int orientation)
25799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
25809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // If the rotation can be handled in hardware, this is where
25819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // the magic should happen.
258266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
258366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const DisplayHardware& hw(displayHardware());
258466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float w = mDisplayWidth;
258566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float h = mDisplayHeight;
258666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mWidth = int(w);
258766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mHeight = int(h);
258866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
258966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    Transform orientationTransform;
25908c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    GraphicPlane::orientationToTransfrom(orientation, w, h,
25918c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian            &orientationTransform);
25928c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    if (orientation & ISurfaceComposer::eOrientationSwapMask) {
25938c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        mWidth = int(h);
25948c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        mHeight = int(w);
25959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
25968c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian
25973552f53c8370ced8680951f4ac811a126da02b0eMathias Agopian    mOrientation = orientation;
259866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mGlobalTransform = mDisplayTransform * orientationTransform;
25999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
26009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
26019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
26029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst DisplayHardware& GraphicPlane::displayHardware() const {
26039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return *mHw;
26049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
26059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2606aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias AgopianDisplayHardware& GraphicPlane::editDisplayHardware() {
2607aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    return *mHw;
2608aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
2609aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
26109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst Transform& GraphicPlane::transform() const {
26119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return mGlobalTransform;
26129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
26139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
26141473f46cbc82aa6f0ba744cc896a36923823d55bMathias AgopianEGLDisplay GraphicPlane::getEGLDisplay() const {
26151473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    return mHw->getEGLDisplay();
26161473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian}
26171473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
26189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
26199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
26209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}; // namespace android
2621