SurfaceFlinger.cpp revision ec49d8970125b3da422f93bf6f6b32ac62230a83
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"
5093d75ec3714d829a41947a29b5c4028b282622c4Mathias Agopian#include "DdmConnection.h"
519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "Layer.h"
529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "LayerDim.h"
539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "SurfaceFlinger.h"
549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "DisplayHardware/DisplayHardware.h"
56e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian#include "DisplayHardware/HWComposer.h"
579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
587bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian#include <private/surfaceflinger/SharedBufferStack.h>
597bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian
60627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian/* ideally AID_GRAPHICS would be in a semi-public header
61627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian * or there would be a way to map a user/group name to its id
62627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian */
63627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian#ifndef AID_GRAPHICS
64627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian#define AID_GRAPHICS 1003
65627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian#endif
66627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#define DISPLAY_COUNT       1
689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectnamespace android {
709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
720dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopianconst String16 sHardwareTest("android.permission.HARDWARE_TEST");
730dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopianconst String16 sAccessSurfaceFlinger("android.permission.ACCESS_SURFACE_FLINGER");
740dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopianconst String16 sReadFramebuffer("android.permission.READ_FRAME_BUFFER");
750dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopianconst String16 sDump("android.permission.DUMP");
760dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian
770dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian// ---------------------------------------------------------------------------
780dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian
799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectSurfaceFlinger::SurfaceFlinger()
809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    :   BnSurfaceComposer(), Thread(false),
819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mTransactionFlags(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),
9493d75ec3714d829a41947a29b5c4028b282622c4Mathias Agopian        mDebugDDMS(0),
956a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        mDebugDisableHWC(0),
962143fe05e3a1aeae641ca126e76db82d17e8b8e6Mathias Agopian        mDebugDisableTransformHint(0),
97a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers(0),
98a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastSwapBufferTime(0),
99a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInTransaction(0),
100a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastTransactionTime(0),
1016950e428feaccc8164b989ef64e771a99948797aMathias Agopian        mBootFinished(false),
1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mConsoleSignals(0),
1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mSecureFrameBuffer(0)
1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    init();
1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::init()
1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LOGI("SurfaceFlinger is starting");
1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // debugging stuff...
1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    char value[PROPERTY_VALUE_MAX];
11493d75ec3714d829a41947a29b5c4028b282622c4Mathias Agopian
1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    property_get("debug.sf.showupdates", value, "0");
1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDebugRegion = atoi(value);
11793d75ec3714d829a41947a29b5c4028b282622c4Mathias Agopian
1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    property_get("debug.sf.showbackground", value, "0");
1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDebugBackground = atoi(value);
1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12193d75ec3714d829a41947a29b5c4028b282622c4Mathias Agopian    property_get("debug.sf.ddms", value, "0");
12293d75ec3714d829a41947a29b5c4028b282622c4Mathias Agopian    mDebugDDMS = atoi(value);
12393d75ec3714d829a41947a29b5c4028b282622c4Mathias Agopian    if (mDebugDDMS) {
12493d75ec3714d829a41947a29b5c4028b282622c4Mathias Agopian        DdmConnection::start(getServiceName());
12593d75ec3714d829a41947a29b5c4028b282622c4Mathias Agopian    }
12693d75ec3714d829a41947a29b5c4028b282622c4Mathias Agopian
127e5c0a7b7810092a593ff386843927238c175bdbdMathias Agopian    LOGI_IF(mDebugRegion,       "showupdates enabled");
128e5c0a7b7810092a593ff386843927238c175bdbdMathias Agopian    LOGI_IF(mDebugBackground,   "showbackground enabled");
12993d75ec3714d829a41947a29b5c4028b282622c4Mathias Agopian    LOGI_IF(mDebugDDMS,         "DDMS debugging enabled");
1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectSurfaceFlinger::~SurfaceFlinger()
1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDeleteTextures(1, &mWormholeTexName);
1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
137d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopiansp<IMemoryHeap> SurfaceFlinger::getCblk() const
1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
139d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    return mServerHeap;
1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
142770492cb2b19f6a36ad748cd05fbedfbb9a67dfaMathias Agopiansp<ISurfaceComposerClient> SurfaceFlinger::createConnection()
1439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
144593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<ISurfaceComposerClient> bclient;
145593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<Client> client(new Client(this));
146593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    status_t err = client->initCheck();
147593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (err == NO_ERROR) {
148593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        bclient = client;
1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return bclient;
1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
153f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennissp<IGraphicBufferAlloc> SurfaceFlinger::createGraphicBufferAlloc()
154f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis{
155f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    sp<GraphicBufferAlloc> gba(new GraphicBufferAlloc());
156f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    return gba;
157f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis}
1587623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst GraphicPlane& SurfaceFlinger::graphicPlane(int dpy) const
1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LOGE_IF(uint32_t(dpy) >= DISPLAY_COUNT, "Invalid DisplayID %d", dpy);
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(mGraphicPlanes[dpy]);
1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return plane;
1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane& SurfaceFlinger::graphicPlane(int dpy)
1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return const_cast<GraphicPlane&>(
1699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const_cast<SurfaceFlinger const *>(this)->graphicPlane(dpy));
1709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::bootFinished()
1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const nsecs_t now = systemTime();
1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const nsecs_t duration = now - mBootTime;
176e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber    LOGI("Boot is finished (%ld ms)", long(ns2ms(duration)) );
1776950e428feaccc8164b989ef64e771a99948797aMathias Agopian    mBootFinished = true;
1780c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian
1790c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    // wait patiently for the window manager death
1800c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    const String16 name("window");
1810c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    sp<IBinder> window(defaultServiceManager()->getService(name));
1820c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    if (window != 0) {
1830c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian        window->linkToDeath(this);
1840c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    }
1850c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian
1860c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    // stop boot animation
187627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    property_set("ctl.stop", "bootanim");
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1900c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopianvoid SurfaceFlinger::binderDied(const wp<IBinder>& who)
1910c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian{
1920c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    // the window manager died on us. prepare its eulogy.
1930c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian
1940c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    // unfreeze the screen in case it was... frozen
1950c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    mFreezeDisplayTime = 0;
1960c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    mFreezeCount = 0;
1970c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    mFreezeDisplay = false;
1980c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian
1990c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    // reset screen orientation
2000c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    setOrientation(0, eOrientationDefault, 0);
2010c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian
2020c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    // restart the boot-animation
2030c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian    property_set("ctl.start", "bootanim");
2040c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian}
2050c63ef5091def41e91a2202560c57b2f6caabfbeMathias Agopian
2069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::onFirstRef()
2079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
2089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    run("SurfaceFlinger", PRIORITY_URGENT_DISPLAY);
2099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // Wait for the main thread to be done with its initialization
2119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mReadyToRunBarrier.wait();
2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic inline uint16_t pack565(int r, int g, int b) {
2159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return (r<<11)|(g<<5)|b;
2169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::readyToRun()
2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LOGI(   "SurfaceFlinger's main thread ready to run. "
2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            "Initializing graphics H/W...");
2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // we only support one display currently
2249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int dpy = 0;
2259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    {
2279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // initialize the main display
2289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        GraphicPlane& plane(graphicPlane(dpy));
2299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        DisplayHardware* const hw = new DisplayHardware(this, dpy);
2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        plane.setDisplayHardware(hw);
2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
233d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    // create the shared control-block
234d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    mServerHeap = new MemoryHeapBase(4096,
235d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian            MemoryHeapBase::READ_ONLY, "SurfaceFlinger read-only heap");
236d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    LOGE_IF(mServerHeap==0, "can't create shared memory dealer");
237e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
238d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    mServerCblk = static_cast<surface_flinger_cblk_t*>(mServerHeap->getBase());
239d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    LOGE_IF(mServerCblk==0, "can't get to shared control block's address");
240e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
241d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    new(mServerCblk) surface_flinger_cblk_t;
242d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize primary screen
2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // (other display should be initialized in the same manner, but
2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // asynchronously, as they could come and go. None of this is supported
2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // yet).
2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(dpy));
2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw = plane.displayHardware();
2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint32_t w = hw.getWidth();
2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint32_t h = hw.getHeight();
2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint32_t f = hw.getFormat();
2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    hw.makeCurrent();
2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize the shared control block
2559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mServerCblk->connected |= 1<<dpy;
2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    display_cblk_t* dcblk = mServerCblk->displays + dpy;
2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    memset(dcblk, 0, sizeof(display_cblk_t));
25866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    dcblk->w            = plane.getWidth();
25966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    dcblk->h            = plane.getHeight();
2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->format       = f;
2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->orientation  = ISurfaceComposer::eOrientationDefault;
2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->xdpi         = hw.getDpiX();
2639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->ydpi         = hw.getDpiY();
2649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->fps          = hw.getRefreshRate();
2659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->density      = hw.getDensity();
2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // Initialize OpenGL|ES
2689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glPixelStorei(GL_UNPACK_ALIGNMENT, 4);
269e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber    glPixelStorei(GL_PACK_ALIGNMENT, 4);
2709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glEnableClientState(GL_VERTEX_ARRAY);
2719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glEnable(GL_SCISSOR_TEST);
2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glShadeModel(GL_FLAT);
2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_CULL_FACE);
2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint16_t g0 = pack565(0x0F,0x1F,0x0F);
2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint16_t g1 = pack565(0x17,0x2f,0x17);
2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint16_t textureData[4] = { g0, g1, g1, g0 };
2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glGenTextures(1, &mWormholeTexName);
2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glBindTexture(GL_TEXTURE_2D, mWormholeTexName);
2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
2829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
2839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
2849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
2859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, 2, 2, 0,
2869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            GL_RGB, GL_UNSIGNED_SHORT_5_6_5, textureData);
2879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glViewport(0, 0, w, h);
2899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glMatrixMode(GL_PROJECTION);
2909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glLoadIdentity();
2913a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian    // put the origin in the left-bottom corner
2923a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian    glOrthof(0, w, 0, h, 0, 1); // l=0, r=w ; b=0, t=h
2939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mReadyToRunBarrier.open();
2959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
2979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *  We're now ready to accept clients...
2989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
300627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    // start boot animation
301627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    property_set("ctl.start", "bootanim");
302e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
3039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
3079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#if 0
3089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark -
3099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark Events Handler
3109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif
3119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::waitForEvent()
3139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
3146ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    while (true) {
3156ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        nsecs_t timeout = -1;
3160e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        const nsecs_t freezeDisplayTimeout = ms2ns(5000);
3176ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        if (UNLIKELY(isFrozen())) {
3186ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            // wait 5 seconds
3196ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            const nsecs_t now = systemTime();
3206ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            if (mFreezeDisplayTime == 0) {
3216ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian                mFreezeDisplayTime = now;
3226ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            }
3236ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            nsecs_t waitTime = freezeDisplayTimeout - (now - mFreezeDisplayTime);
3246ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            timeout = waitTime>0 ? waitTime : 0;
325c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project        }
3266ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian
327898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        sp<MessageBase> msg = mEventQueue.waitMessage(timeout);
3280e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian
3290e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        // see if we timed out
3300e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        if (isFrozen()) {
3310e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            const nsecs_t now = systemTime();
3320e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            nsecs_t frozenTime = (now - mFreezeDisplayTime);
3330e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            if (frozenTime >= freezeDisplayTimeout) {
3349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // we timed out and are still frozen
3359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                LOGW("timeout expired mFreezeDisplay=%d, mFreezeCount=%d",
3369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        mFreezeDisplay, mFreezeCount);
3370e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                mFreezeDisplayTime = 0;
3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeCount = 0;
339c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project                mFreezeDisplay = false;
3400e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            }
3410e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        }
3420e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian
3430e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        if (msg != 0) {
3440e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            switch (msg->what) {
3450e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                case MessageQueue::INVALIDATE:
3460e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                    // invalidate message, just return to the main loop
3470e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                    return;
3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::signalEvent() {
3546ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    mEventQueue.invalidate();
3559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3579b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennisbool SurfaceFlinger::authenticateSurfaceTexture(
3589b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis        const sp<ISurfaceTexture>& surfaceTexture) const {
359d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    Mutex::Autolock _l(mStateLock);
3609b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis    sp<IBinder> surfaceTextureBinder(surfaceTexture->asBinder());
361d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis
362d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // Check the visible layer list for the ISurface
363d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    const LayerVector& currentLayers = mCurrentState.layersSortedByZ;
364d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    size_t count = currentLayers.size();
365d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    for (size_t i=0 ; i<count ; i++) {
366d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        const sp<LayerBase>& layer(currentLayers[i]);
367d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        sp<LayerBaseClient> lbc(layer->getLayerBaseClient());
3689b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis        if (lbc != NULL) {
3699b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis            wp<IBinder> lbcBinder = lbc->getSurfaceTextureBinder();
3709b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis            if (lbcBinder == surfaceTextureBinder) {
3719b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis                return true;
3729b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis            }
373d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        }
374d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    }
375d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis
376d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // Check the layers in the purgatory.  This check is here so that if a
3779b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis    // SurfaceTexture gets destroyed before all the clients are done using it,
3789b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis    // the error will not be reported as "surface XYZ is not authenticated", but
379d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // will instead fail later on when the client tries to use the surface,
380d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // which should be reported as "surface XYZ returned an -ENODEV".  The
381d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // purgatorized layers are no less authentic than the visible ones, so this
382d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    // should not cause any harm.
383d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    size_t purgatorySize =  mLayerPurgatory.size();
384d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    for (size_t i=0 ; i<purgatorySize ; i++) {
385d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        const sp<LayerBase>& layer(mLayerPurgatory.itemAt(i));
386d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        sp<LayerBaseClient> lbc(layer->getLayerBaseClient());
3879b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis        if (lbc != NULL) {
3889b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis            wp<IBinder> lbcBinder = lbc->getSurfaceTextureBinder();
3899b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis            if (lbcBinder == surfaceTextureBinder) {
3909b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis                return true;
3919b8fc65b0353db3a962ab52dc556c356d556fcc1Jamie Gennis            }
392d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis        }
393d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    }
394d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis
395d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis    return false;
396d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis}
397d2acedf18c9a3527f6e3dc5d578a85fdc0ad4338Jamie Gennis
398898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianstatus_t SurfaceFlinger::postMessageAsync(const sp<MessageBase>& msg,
399898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        nsecs_t reltime, uint32_t flags)
4009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
401898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    return mEventQueue.postMessage(msg, reltime, flags);
402898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian}
403898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian
404898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianstatus_t SurfaceFlinger::postMessageSync(const sp<MessageBase>& msg,
405898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        nsecs_t reltime, uint32_t flags)
406898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian{
407898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    status_t res = mEventQueue.postMessage(msg, reltime, flags);
408898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    if (res == NO_ERROR) {
409898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        msg->wait();
410898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    }
411898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    return res;
4129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
4159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#if 0
4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark -
4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark Main loop
4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif
4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool SurfaceFlinger::threadLoop()
4219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    waitForEvent();
4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // check for transactions
4259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(mConsoleSignals)) {
4269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        handleConsoleEvents();
4279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
429439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    // if we're in a global transaction, don't do anything.
430439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    const uint32_t mask = eTransactionNeeded | eTraversalNeeded;
431439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    uint32_t transactionFlags = peekTransactionFlags(mask);
432439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    if (UNLIKELY(transactionFlags)) {
433439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        handleTransaction(transactionFlags);
4349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // post surfaces (if needed)
4379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    handlePageFlip();
4389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
439e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    if (UNLIKELY(mHwWorkListDirty)) {
440e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        // build the h/w work list
441e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        handleWorkList();
442e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    }
443e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw(graphicPlane(0).displayHardware());
44581384bf927c47a4efa653b14273084a13e67e3acMathias Agopian    if (LIKELY(hw.canDraw() && !isFrozen())) {
4469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // repaint the framebuffer (if needed)
44704262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian
44804262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        const int index = hw.getCurrentBufferIndex();
44904262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        GraphicLog& logger(GraphicLog::getInstance());
45004262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian
45104262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_REPAINT, index);
4529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        handleRepaint();
4539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
454b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian        // inform the h/w that we're done compositing
45504262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_COMPOSITION_COMPLETE, index);
456b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian        hw.compositionComplete();
457b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian
45804262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_SWAP_BUFFERS, index);
4594c0a4a2b74db9d4a439a0aaa39d80586f7eb1258Antti Hatala        postFramebuffer();
4604c0a4a2b74db9d4a439a0aaa39d80586f7eb1258Antti Hatala
46104262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_REPAINT_DONE, index);
4629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
4639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // pretend we did the post
464a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian        hw.compositionComplete();
4659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        usleep(16667); // 60 fps period
4669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return true;
4689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::postFramebuffer()
4719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (!mInvalidRegion.isEmpty()) {
4739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw(graphicPlane(0).displayHardware());
474a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t now = systemTime();
475a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers = now;
4769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.flip(mInvalidRegion);
477a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastSwapBufferTime = systemTime() - now;
478a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers = 0;
4799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mInvalidRegion.clear();
4809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleConsoleEvents()
4849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // something to do with the console
4869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw = graphicPlane(0).displayHardware();
4879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int what = android_atomic_and(0, &mConsoleSignals);
4899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (what & eConsoleAcquired) {
4909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.acquireScreen();
4912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // this is a temporary work-around, eventually this should be called
4922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // by the power-manager
493d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        SurfaceFlinger::turnElectronBeamOn(mElectronBeamAnimationMode);
4949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
496aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (mDeferReleaseConsole && hw.isScreenAcquired()) {
497ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian        // We got the release signal before the acquire signal
4989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDeferReleaseConsole = false;
4999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.releaseScreen();
5009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (what & eConsoleReleased) {
503aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        if (hw.isScreenAcquired()) {
5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            hw.releaseScreen();
5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDeferReleaseConsole = true;
5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.set(hw.bounds());
5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleTransaction(uint32_t transactionFlags)
5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
5156960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    Mutex::Autolock _l(mStateLock);
5166960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    const nsecs_t now = systemTime();
5176960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    mDebugInTransaction = now;
5186960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian
5196960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // Here we're guaranteed that some transaction flags are set
5206960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // so we can call handleTransactionLocked() unconditionally.
5216960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // We call getTransactionFlags(), which will also clear the flags,
5226960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // with mStateLock held to guarantee that mCurrentState won't change
5236960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // until the transaction is committed.
5246960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian
5256960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    const uint32_t mask = eTransactionNeeded | eTraversalNeeded;
5266960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    transactionFlags = getTransactionFlags(mask);
5276960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    handleTransactionLocked(transactionFlags);
5286960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian
5296960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    mLastTransactionTime = systemTime() - now;
5306960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    mDebugInTransaction = 0;
5316960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    invalidateHwcGeometry();
5326960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    // here the transaction has been committed
5332d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian}
5349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5356960811d15eb66e8469330d3bc574c32c481db05Mathias Agopianvoid SurfaceFlinger::handleTransactionLocked(uint32_t transactionFlags)
5362d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian{
5372d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    const LayerVector& currentLayers(mCurrentState.layersSortedByZ);
5389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const size_t count = currentLayers.size();
5399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
5419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Traversal of the children
5429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * (perform the transaction for each of them if needed)
5439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const bool layersNeedTransaction = transactionFlags & eTraversalNeeded;
5469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (layersNeedTransaction) {
5479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (size_t i=0 ; i<count ; i++) {
5481473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian            const sp<LayerBase>& layer = currentLayers[i];
5499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            uint32_t trFlags = layer->getTransactionFlags(eTransactionNeeded);
5509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (!trFlags) continue;
5519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const uint32_t flags = layer->doTransaction(0);
5539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (flags & Layer::eVisibleRegion)
5549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mVisibleRegionsDirty = true;
5559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
5599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Perform our own transaction if needed
5609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (transactionFlags & eTransactionNeeded) {
5639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mCurrentState.orientation != mDrawingState.orientation) {
5649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // the orientation has changed, recompute all visible regions
5659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // and invalidate everything.
5669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int dpy = 0;
5689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int orientation = mCurrentState.orientation;
569eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian            const uint32_t type = mCurrentState.orientationType;
5709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            GraphicPlane& plane(graphicPlane(dpy));
5719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            plane.setOrientation(orientation);
5729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // update the shared control block
5749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const DisplayHardware& hw(plane.displayHardware());
5759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            volatile display_cblk_t* dcblk = mServerCblk->displays + dpy;
5769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dcblk->orientation = orientation;
57766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            dcblk->w = plane.getWidth();
57866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            dcblk->h = plane.getHeight();
5799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
5819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(hw.bounds());
5829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mCurrentState.freezeDisplay != mDrawingState.freezeDisplay) {
5859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // freezing or unfreezing the display -> trigger animation if needed
5869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mFreezeDisplay = mCurrentState.freezeDisplay;
58731901cc0b6f0c678be4f629c8c3405700e63c346Mathias Agopian            if (mFreezeDisplay)
58831901cc0b6f0c678be4f629c8c3405700e63c346Mathias Agopian                 mFreezeDisplayTime = 0;
5899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
591a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        if (currentLayers.size() > mDrawingState.layersSortedByZ.size()) {
592a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            // layers have been added
5939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
5949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
596a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        // some layers might have been removed, so
597a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        // we need to update the regions they're exposing.
598a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        if (mLayersRemoved) {
599248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian            mLayersRemoved = false;
6009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
601a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            const LayerVector& previousLayers(mDrawingState.layersSortedByZ);
6022d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian            const size_t count = previousLayers.size();
6032d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian            for (size_t i=0 ; i<count ; i++) {
604a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                const sp<LayerBase>& layer(previousLayers[i]);
605a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                if (currentLayers.indexOf( layer ) < 0) {
606a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                    // this layer is not visible anymore
60733863dd9e1005478d20b70fad42e447ac97f5abfMathias Agopian                    mDirtyRegionRemovedLayer.orSelf(layer->visibleRegionScreen);
608a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                }
609a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            }
6109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    commitTransaction();
6149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
6159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectsp<FreezeLock> SurfaceFlinger::getFreezeLock() const
6179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
6189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return new FreezeLock(const_cast<SurfaceFlinger *>(this));
6199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
6209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::computeVisibleRegions(
622f0ff906fa427ddc3293dc061e2ee34ce39c1336eMathias Agopian    const LayerVector& currentLayers, Region& dirtyRegion, Region& opaqueRegion)
6239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
6249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(0));
6259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Transform& planeTransform(plane.transform());
6269c041bbd81789c209e2369ba958306979b67614fMathias Agopian    const DisplayHardware& hw(plane.displayHardware());
6279c041bbd81789c209e2369ba958306979b67614fMathias Agopian    const Region screenRegion(hw.bounds());
6289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region aboveOpaqueLayers;
6309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region aboveCoveredLayers;
6319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region dirty;
6329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool secureFrameBuffer = false;
6349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t i = currentLayers.size();
6369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    while (i--) {
6371473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        const sp<LayerBase>& layer = currentLayers[i];
6389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->validateVisibility(planeTransform);
6399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // start with the whole surface at its current location
64112cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        const Layer::State& s(layer->drawingState());
6429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6439c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
6449c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * opaqueRegion: area of a surface that is fully opaque.
6459c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
6469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region opaqueRegion;
6479c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6489c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
6499c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * visibleRegion: area of a surface that is visible on screen
6509c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * and not fully transparent. This is essentially the layer's
6519c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * footprint minus the opaque regions above it.
6529c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * Areas covered by a translucent surface are considered visible.
6539c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
6549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region visibleRegion;
6559c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6569c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
6579c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * coveredRegion: area of a surface that is covered by all
6589c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * visible regions above it (which includes the translucent areas).
6599c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
6609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region coveredRegion;
6619c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6629c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6639c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // handle hidden surfaces by setting the visible region to empty
66412cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        if (LIKELY(!(s.flags & ISurfaceComposer::eLayerHidden) && s.alpha)) {
6657bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian            const bool translucent = !layer->isOpaque();
66612cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian            const Rect bounds(layer->visibleBounds());
6679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            visibleRegion.set(bounds);
6689c041bbd81789c209e2369ba958306979b67614fMathias Agopian            visibleRegion.andSelf(screenRegion);
6699c041bbd81789c209e2369ba958306979b67614fMathias Agopian            if (!visibleRegion.isEmpty()) {
6709c041bbd81789c209e2369ba958306979b67614fMathias Agopian                // Remove the transparent area from the visible region
6719c041bbd81789c209e2369ba958306979b67614fMathias Agopian                if (translucent) {
6729c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    visibleRegion.subtractSelf(layer->transparentRegionScreen);
6739c041bbd81789c209e2369ba958306979b67614fMathias Agopian                }
6749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6759c041bbd81789c209e2369ba958306979b67614fMathias Agopian                // compute the opaque region
6769c041bbd81789c209e2369ba958306979b67614fMathias Agopian                const int32_t layerOrientation = layer->getOrientation();
6779c041bbd81789c209e2369ba958306979b67614fMathias Agopian                if (s.alpha==255 && !translucent &&
6789c041bbd81789c209e2369ba958306979b67614fMathias Agopian                        ((layerOrientation & Transform::ROT_INVALID) == false)) {
6799c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    // the opaque region is the layer's footprint
6809c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    opaqueRegion = visibleRegion;
6819c041bbd81789c209e2369ba958306979b67614fMathias Agopian                }
6829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
6839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6859c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Clip the covered region to the visible region
6869c041bbd81789c209e2369ba958306979b67614fMathias Agopian        coveredRegion = aboveCoveredLayers.intersect(visibleRegion);
6879c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6889c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Update aboveCoveredLayers for next (lower) layer
6899c041bbd81789c209e2369ba958306979b67614fMathias Agopian        aboveCoveredLayers.orSelf(visibleRegion);
6909c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // subtract the opaque region covered by the layers above us
6929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        visibleRegion.subtractSelf(aboveOpaqueLayers);
6939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // compute this layer's dirty region
6959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (layer->contentDirty) {
6969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // we need to invalidate the whole region
6979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dirty = visibleRegion;
6989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // as well, as the old visible region
6999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dirty.orSelf(layer->visibleRegionScreen);
7009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->contentDirty = false;
7019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
7020aed7e97a3efbeec23a4aa301df03036a67f8ea3Mathias Agopian            /* compute the exposed region:
7039c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   the exposed region consists of two components:
7049c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   1) what's VISIBLE now and was COVERED before
7059c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   2) what's EXPOSED now less what was EXPOSED before
7069c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *
7079c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * note that (1) is conservative, we start with the whole
7089c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * visible region but only keep what used to be covered by
7099c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * something -- which mean it may have been exposed.
7109c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *
7119c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * (2) handles areas that were not covered by anything but got
7129c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * exposed because of a resize.
7130aed7e97a3efbeec23a4aa301df03036a67f8ea3Mathias Agopian             */
7149c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region newExposed = visibleRegion - coveredRegion;
7159c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldVisibleRegion = layer->visibleRegionScreen;
7169c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldCoveredRegion = layer->coveredRegionScreen;
7179c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldExposed = oldVisibleRegion - oldCoveredRegion;
7189c041bbd81789c209e2369ba958306979b67614fMathias Agopian            dirty = (visibleRegion&oldCoveredRegion) | (newExposed-oldExposed);
7199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dirty.subtractSelf(aboveOpaqueLayers);
7219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // accumulate to the screen dirty region
7239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dirtyRegion.orSelf(dirty);
7249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7259c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Update aboveOpaqueLayers for next (lower) layer
7269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        aboveOpaqueLayers.orSelf(opaqueRegion);
727e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
7289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Store the visible region is screen space
7299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->setVisibleRegion(visibleRegion);
7309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->setCoveredRegion(coveredRegion);
7319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
73212cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        // If a secure layer is partially visible, lock-down the screen!
7339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (layer->isSecure() && !visibleRegion.isEmpty()) {
7349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            secureFrameBuffer = true;
7359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
73812cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    // invalidate the areas where a layer was removed
73912cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    dirtyRegion.orSelf(mDirtyRegionRemovedLayer);
74012cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    mDirtyRegionRemovedLayer.clear();
74112cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian
7429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mSecureFrameBuffer = secureFrameBuffer;
7439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    opaqueRegion = aboveOpaqueLayers;
7449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::commitTransaction()
7489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDrawingState = mCurrentState;
7509779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian    mResizeTransationPending = false;
7519779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian    mTransactionCV.broadcast();
7529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handlePageFlip()
7559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool visibleRegions = mVisibleRegionsDirty;
757f0ff906fa427ddc3293dc061e2ee34ce39c1336eMathias Agopian    const LayerVector& currentLayers(mDrawingState.layersSortedByZ);
7589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    visibleRegions |= lockPageFlip(currentLayers);
7599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw = graphicPlane(0).displayHardware();
7619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const Region screenRegion(hw.bounds());
7629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (visibleRegions) {
7639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            Region opaqueRegion;
7649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            computeVisibleRegions(currentLayers, mDirtyRegion, opaqueRegion);
765ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian
766ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            /*
767ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian             *  rebuild the visible layer list
768ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian             */
769f0ff906fa427ddc3293dc061e2ee34ce39c1336eMathias Agopian            const size_t count = currentLayers.size();
770ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            mVisibleLayersSortedByZ.clear();
771ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            mVisibleLayersSortedByZ.setCapacity(count);
772ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            for (size_t i=0 ; i<count ; i++) {
773ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian                if (!currentLayers[i]->visibleRegionScreen.isEmpty())
774ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian                    mVisibleLayersSortedByZ.add(currentLayers[i]);
775ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            }
776ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian
7779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mWormholeRegion = screenRegion.subtract(opaqueRegion);
7789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = false;
779fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian            invalidateHwcGeometry();
7809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    unlockPageFlip(currentLayers);
7839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.andSelf(screenRegion);
7849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
786fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopianvoid SurfaceFlinger::invalidateHwcGeometry()
787fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian{
788fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian    mHwWorkListDirty = true;
789fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian}
790fb4dcb1a679000c029bb8e76fea7066d002235d5Mathias Agopian
7919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool SurfaceFlinger::lockPageFlip(const LayerVector& currentLayers)
7929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool recomputeVisibleRegions = false;
7949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t count = currentLayers.size();
7951473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBase> const* layers = currentLayers.array();
7969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (size_t i=0 ; i<count ; i++) {
7977623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
7989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->lockPageFlip(recomputeVisibleRegions);
7999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return recomputeVisibleRegions;
8019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
8029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::unlockPageFlip(const LayerVector& currentLayers)
8049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
8059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(0));
8069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Transform& planeTransform(plane.transform());
8079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t count = currentLayers.size();
8081473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBase> const* layers = currentLayers.array();
8099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (size_t i=0 ; i<count ; i++) {
8107623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
8119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->unlockPageFlip(planeTransform, mDirtyRegion);
8129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
8149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
815e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopianvoid SurfaceFlinger::handleWorkList()
816e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian{
817e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    mHwWorkListDirty = false;
818e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    HWComposer& hwc(graphicPlane(0).displayHardware().getHwComposer());
819e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    if (hwc.initCheck() == NO_ERROR) {
820e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        const Vector< sp<LayerBase> >& currentLayers(mVisibleLayersSortedByZ);
821e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        const size_t count = currentLayers.size();
822e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        hwc.createWorkList(count);
823f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        hwc_layer_t* const cur(hwc.getLayers());
824f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        for (size_t i=0 ; cur && i<count ; i++) {
825f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            currentLayers[i]->setGeometry(&cur[i]);
8267f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian            if (mDebugDisableHWC || mDebugRegion) {
8276a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                cur[i].compositionType = HWC_FRAMEBUFFER;
8286a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                cur[i].flags |= HWC_SKIP_LAYER;
8296a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian            }
830e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        }
831e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    }
832e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian}
8338c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian
8349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleRepaint()
8359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
8368c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    // compute the invalid region
8378c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    mInvalidRegion.orSelf(mDirtyRegion);
8389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(mDebugRegion)) {
8409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        debugFlashRegions();
8419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8438c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    // set the frame buffer
8448c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
8458c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    glMatrixMode(GL_MODELVIEW);
8468c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    glLoadIdentity();
8479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t flags = hw.getFlags();
849e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber    if ((flags & DisplayHardware::SWAP_RECTANGLE) ||
850e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber        (flags & DisplayHardware::BUFFER_PRESERVED))
8512e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian    {
852ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // we can redraw only what's dirty, but since SWAP_RECTANGLE only
853ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // takes a rectangle, we must make sure to update that whole
854ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // rectangle in that case
855ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        if (flags & DisplayHardware::SWAP_RECTANGLE) {
8567623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            // TODO: we really should be able to pass a region to
857ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // SWAP_RECTANGLE so that we don't have to redraw all this.
858ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            mDirtyRegion.set(mInvalidRegion.bounds());
859ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        } else {
860ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // in the BUFFER_PRESERVED case, obviously, we can update only
861ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // what's needed and nothing more.
862ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // NOTE: this is NOT a common case, as preserving the backbuffer
863ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // is costly and usually involves copying the whole update back.
864ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        }
8659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
866f2d28b74850ea0869aef2ce0727a6abb7b166a5cMathias Agopian        if (flags & DisplayHardware::PARTIAL_UPDATES) {
867ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // We need to redraw the rectangle that will be updated
8682e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian            // (pushed to the framebuffer).
869f2d28b74850ea0869aef2ce0727a6abb7b166a5cMathias Agopian            // This is needed because PARTIAL_UPDATES only takes one
870ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // rectangle instead of a region (see DisplayHardware::flip())
8719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(mInvalidRegion.bounds());
8729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
873ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // we need to redraw everything (the whole screen)
8749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(hw.bounds());
8759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mInvalidRegion = mDirtyRegion;
8769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // compose all surfaces
8809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    composeSurfaces(mDirtyRegion);
8819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // clear the dirty regions
8839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.clear();
8849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
8859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::composeSurfaces(const Region& dirty)
8879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
8889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(!mWormholeRegion.isEmpty())) {
8899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // should never happen unless the window manager has a bug
8909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // draw something...
8919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        drawWormhole();
8929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
893e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
894e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    status_t err = NO_ERROR;
895ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian    const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
896f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    size_t count = layers.size();
897e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
898e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
899e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    HWComposer& hwc(hw.getHwComposer());
900f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    hwc_layer_t* const cur(hwc.getLayers());
901e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
902f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    LOGE_IF(cur && hwc.getNumLayers() != count,
903f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            "HAL number of layers (%d) doesn't match surfaceflinger (%d)",
904f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            hwc.getNumLayers(), count);
905e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
906f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    // just to be extra-safe, use the smallest count
9070cf2f4354bf8952d822fa4ee9b9588a73999c2f4Erik Gilling    if (hwc.initCheck() == NO_ERROR) {
9080cf2f4354bf8952d822fa4ee9b9588a73999c2f4Erik Gilling        count = count < hwc.getNumLayers() ? count : hwc.getNumLayers();
9090cf2f4354bf8952d822fa4ee9b9588a73999c2f4Erik Gilling    }
910e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
911f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    /*
912f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     *  update the per-frame h/w composer data for each layer
913f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     *  and build the transparent region of the FB
914f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     */
915f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    Region transparent;
916f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    if (cur) {
917f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        for (size_t i=0 ; i<count ; i++) {
918f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            const sp<LayerBase>& layer(layers[i]);
919f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            layer->setPerFrameData(&cur[i]);
9209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
921f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        err = hwc.prepare();
922f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        LOGE_IF(err, "HWComposer::prepare failed (%s)", strerror(-err));
923e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian
92445491690d8f21d2648cad9247115720fa90046b4Mathias Agopian        if (err == NO_ERROR) {
92545491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            for (size_t i=0 ; i<count ; i++) {
92645491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                if (cur[i].hints & HWC_HINT_CLEAR_FB) {
92745491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    const sp<LayerBase>& layer(layers[i]);
9287bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                    if (layer->isOpaque()) {
92945491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                        transparent.orSelf(layer->visibleRegionScreen);
93045491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    }
93145491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                }
93245491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            }
93345491690d8f21d2648cad9247115720fa90046b4Mathias Agopian
93445491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            /*
93545491690d8f21d2648cad9247115720fa90046b4Mathias Agopian             *  clear the area of the FB that need to be transparent
93645491690d8f21d2648cad9247115720fa90046b4Mathias Agopian             */
93745491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            transparent.andSelf(dirty);
93845491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            if (!transparent.isEmpty()) {
93945491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                glClearColor(0,0,0,0);
94045491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                Region::const_iterator it = transparent.begin();
94145491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                Region::const_iterator const end = transparent.end();
94245491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                const int32_t height = hw.getHeight();
94345491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                while (it != end) {
94445491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    const Rect& r(*it++);
94545491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    const GLint sy = height - (r.top + r.height());
94645491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    glScissor(r.left, sy, r.width(), r.height());
94745491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                    glClear(GL_COLOR_BUFFER_BIT);
94845491690d8f21d2648cad9247115720fa90046b4Mathias Agopian                }
94945491690d8f21d2648cad9247115720fa90046b4Mathias Agopian            }
950e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian        }
951e0d5f5bcf5a8b26f4ad75f549cbf380b2c9faf20Mathias Agopian    }
952f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian
953f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian
954f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    /*
955f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     * and then, render the layers targeted at the framebuffer
956f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian     */
957f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
958f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        if (cur) {
959982f58bdccce7e4e537cffb2410ad78d73398461Antti Hatala            if ((cur[i].compositionType != HWC_FRAMEBUFFER) &&
960982f58bdccce7e4e537cffb2410ad78d73398461Antti Hatala                !(cur[i].flags & HWC_SKIP_LAYER)) {
961f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian                // skip layers handled by the HAL
962f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian                continue;
963f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            }
964f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        }
9656a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian
966f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
967f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        const Region clip(dirty.intersect(layer->visibleRegionScreen));
968f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        if (!clip.isEmpty()) {
969f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            layer->draw(clip);
970f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        }
971f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    }
9729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
9739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::debugFlashRegions()
9759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
976f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
977f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    const uint32_t flags = hw.getFlags();
9787f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian    const int32_t height = hw.getHeight();
9797f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian    if (mInvalidRegion.isEmpty()) {
9807f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian        return;
9817f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian    }
982f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
983f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    if (!((flags & DisplayHardware::SWAP_RECTANGLE) ||
984f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian            (flags & DisplayHardware::BUFFER_PRESERVED))) {
985f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        const Region repaint((flags & DisplayHardware::PARTIAL_UPDATES) ?
986f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian                mDirtyRegion.bounds() : hw.bounds());
987f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        composeSurfaces(repaint);
988f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    }
989f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
9909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_BLEND);
9919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
9929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_SCISSOR_TEST);
9939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
994dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    static int toggle = 0;
995dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    toggle = 1 - toggle;
996dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    if (toggle) {
997f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glColor4f(1, 0, 1, 1);
998dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    } else {
999f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glColor4f(1, 1, 0, 1);
1000dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    }
10019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10026158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    Region::const_iterator it = mDirtyRegion.begin();
10036158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    Region::const_iterator const end = mDirtyRegion.end();
10046158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    while (it != end) {
10056158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        const Rect& r = *it++;
10069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        GLfloat vertices[][2] = {
10077f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian                { r.left,  height - r.top },
10087f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian                { r.left,  height - r.bottom },
10097f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian                { r.right, height - r.bottom },
10107f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian                { r.right, height - r.top }
10119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        };
10129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glVertexPointer(2, GL_FLOAT, 0, vertices);
10139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
10149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1015f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
10168c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    hw.flip(mInvalidRegion);
10179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (mDebugRegion > 1)
1019f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        usleep(mDebugRegion * 1000);
10209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glEnable(GL_SCISSOR_TEST);
10229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::drawWormhole() const
10259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Region region(mWormholeRegion.intersect(mDirtyRegion));
10279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (region.isEmpty())
10289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return;
10299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw(graphicPlane(0).displayHardware());
10319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int32_t width = hw.getWidth();
10329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int32_t height = hw.getHeight();
10339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_BLEND);
10359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
10369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (LIKELY(!mDebugBackground)) {
1038f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glClearColor(0,0,0,0);
10396158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator it = region.begin();
10406158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator const end = region.end();
10416158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        while (it != end) {
10426158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian            const Rect& r = *it++;
10439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const GLint sy = height - (r.top + r.height());
10449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glScissor(r.left, sy, r.width(), r.height());
10459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glClear(GL_COLOR_BUFFER_BIT);
10469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
10489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const GLshort vertices[][2] = { { 0, 0 }, { width, 0 },
10499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { width, height }, { 0, height }  };
10509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const GLshort tcoords[][2] = { { 0, 0 }, { 1, 0 },  { 1, 1 }, { 0, 1 } };
10519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glVertexPointer(2, GL_SHORT, 0, vertices);
10529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glTexCoordPointer(2, GL_SHORT, 0, tcoords);
10539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glEnableClientState(GL_TEXTURE_COORD_ARRAY);
1054e20a56d929fc8fedc2b468ea6d1900bd2aa6e81aMichael I. Gold#if defined(GL_OES_EGL_image_external)
1055781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian        if (GLExtensions::getInstance().haveTextureExternal()) {
1056781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian            glDisable(GL_TEXTURE_EXTERNAL_OES);
1057781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian        }
1058f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian#endif
10599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glEnable(GL_TEXTURE_2D);
10609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glBindTexture(GL_TEXTURE_2D, mWormholeTexName);
10619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
10629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glMatrixMode(GL_TEXTURE);
10639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glLoadIdentity();
10649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glScalef(width*(1.0f/32.0f), height*(1.0f/32.0f), 1);
10656158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator it = region.begin();
10666158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator const end = region.end();
10676158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        while (it != end) {
10686158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian            const Rect& r = *it++;
10699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const GLint sy = height - (r.top + r.height());
10709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glScissor(r.left, sy, r.width(), r.height());
10719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
10729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glDisableClientState(GL_TEXTURE_COORD_ARRAY);
10747bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        glDisable(GL_TEXTURE_2D);
107504a709e459db6f15c04b00bcd3b030c90ca52949Mathias Agopian        glLoadIdentity();
107604a709e459db6f15c04b00bcd3b030c90ca52949Mathias Agopian        glMatrixMode(GL_MODELVIEW);
10779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::debugShowFPS() const
10819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int mFrameCount;
10839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int mLastFrameCount = 0;
10849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static nsecs_t mLastFpsTime = 0;
10859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static float mFps = 0;
10869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mFrameCount++;
10879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    nsecs_t now = systemTime();
10889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    nsecs_t diff = now - mLastFpsTime;
10899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (diff > ms2ns(250)) {
10909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mFps =  ((mFrameCount - mLastFrameCount) * float(s2ns(1))) / diff;
10919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mLastFpsTime = now;
10929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mLastFrameCount = mFrameCount;
10939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // XXX: mFPS has the value we want
10959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project }
10969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10971473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::addLayer(const sp<LayerBase>& layer)
10989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
11009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    addLayer_l(layer);
11019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded|eTraversalNeeded);
11029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
11039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11051473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::addLayer_l(const sp<LayerBase>& layer)
11069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11071efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian    ssize_t i = mCurrentState.layersSortedByZ.add(layer);
11089bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian    return (i < 0) ? status_t(i) : status_t(NO_ERROR);
11099bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian}
11109bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1111593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianssize_t SurfaceFlinger::addClientLayer(const sp<Client>& client,
1112593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const sp<LayerBaseClient>& lbc)
11139bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian{
1114593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // attach this layer to the client
11155fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    size_t name = client->attachLayer(lbc);
11165fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian
11175fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    Mutex::Autolock _l(mStateLock);
1118593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
1119593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // add this layer to the current state list
1120593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    addLayer_l(lbc);
1121593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
11225fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    return ssize_t(name);
1123593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian}
1124593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
1125593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::removeLayer(const sp<LayerBase>& layer)
1126593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
1127593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    Mutex::Autolock _l(mStateLock);
1128593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    status_t err = purgatorizeLayer_l(layer);
1129593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (err == NO_ERROR)
1130593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        setTransactionFlags(eTransactionNeeded);
1131593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return err;
11329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11341473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::removeLayer_l(const sp<LayerBase>& layerBase)
11359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11367623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<LayerBaseClient> lbc(layerBase->getLayerBaseClient());
11377623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (lbc != 0) {
1138d35c6667c8233385f31aa203f486b2cb826bf6beMathias Agopian        mLayerMap.removeItem( lbc->getSurfaceBinder() );
11397623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
11409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    ssize_t index = mCurrentState.layersSortedByZ.remove(layerBase);
11419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (index >= 0) {
11421473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        mLayersRemoved = true;
11439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return NO_ERROR;
11449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11452d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    return status_t(index);
11469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11486cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopianstatus_t SurfaceFlinger::purgatorizeLayer_l(const sp<LayerBase>& layerBase)
11496cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian{
1150f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian    // First add the layer to the purgatory list, which makes sure it won't
1151f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian    // go away, then remove it from the main list (through a transaction).
11526cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    ssize_t err = removeLayer_l(layerBase);
1153f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian    if (err >= 0) {
1154f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian        mLayerPurgatory.add(layerBase);
1155f4dfe1bee6223add052b6f0ebc4cdb1f46606accMathias Agopian    }
11562e4b68d57bb64d7e93139238c5a8be91ff956c2aMathias Agopian
11570c4cec7e4df87181486d280c98fba9c0f4774c37Mathias Agopian    layerBase->onRemoved();
11580c4cec7e4df87181486d280c98fba9c0f4774c37Mathias Agopian
11592d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    // it's possible that we don't find a layer, because it might
11602d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    // have been destroyed already -- this is not technically an error
1161593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // from the user because there is a race between Client::destroySurface(),
1162593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // ~Client() and ~ISurface().
11636cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    return (err == NAME_NOT_FOUND) ? status_t(NO_ERROR) : err;
11646cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian}
11656cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
1166593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::invalidateLayerVisibility(const sp<LayerBase>& layer)
11679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1168593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    layer->forceVisibilityTransaction();
1169593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    setTransactionFlags(eTraversalNeeded);
1170593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return NO_ERROR;
11719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11736dcb1557efc546b74323b4367d9b7f674821e1e9Mathias Agopianuint32_t SurfaceFlinger::peekTransactionFlags(uint32_t flags)
11746dcb1557efc546b74323b4367d9b7f674821e1e9Mathias Agopian{
11756dcb1557efc546b74323b4367d9b7f674821e1e9Mathias Agopian    return android_atomic_release_load(&mTransactionFlags);
11766dcb1557efc546b74323b4367d9b7f674821e1e9Mathias Agopian}
11776dcb1557efc546b74323b4367d9b7f674821e1e9Mathias Agopian
11789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectuint32_t SurfaceFlinger::getTransactionFlags(uint32_t flags)
11799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return android_atomic_and(~flags, &mTransactionFlags) & flags;
11819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1183898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianuint32_t SurfaceFlinger::setTransactionFlags(uint32_t flags)
11849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t old = android_atomic_or(flags, &mTransactionFlags);
11869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if ((old & flags)==0) { // wake the server up
1187898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        signalEvent();
11889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return old;
11909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1193439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopianvoid SurfaceFlinger::setTransactionState(const Vector<ComposerState>& state) {
1194439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    Mutex::Autolock _l(mStateLock);
11959779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian
1196439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    uint32_t flags = 0;
1197439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    const size_t count = state.size();
1198439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
1199439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        const ComposerState& s(state[i]);
1200439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        sp<Client> client( static_cast<Client *>(s.client.get()) );
1201439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        flags |= setClientStateLocked(client, s.state);
1202439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    }
1203439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    if (flags) {
1204439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        setTransactionFlags(flags);
1205439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    }
1206439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian
1207439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    signalEvent();
1208439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian
1209439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    // if there is a transaction with a resize, wait for it to
1210439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    // take effect before returning.
1211439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    while (mResizeTransationPending) {
1212439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        status_t err = mTransactionCV.waitRelative(mStateLock, s2ns(5));
1213439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        if (CC_UNLIKELY(err != NO_ERROR)) {
1214439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian            // just in case something goes wrong in SF, return to the
1215439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian            // called after a few seconds.
1216439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian            LOGW_IF(err == TIMED_OUT, "closeGlobalTransaction timed out!");
1217439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian            mResizeTransationPending = false;
1218439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian            break;
12199779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        }
12209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::freezeDisplay(DisplayID dpy, uint32_t flags)
12249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
12259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
12269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
12279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
12299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mCurrentState.freezeDisplay = 1;
12309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded);
12319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // flags is intended to communicate some sort of animation behavior
1233ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian    // (for instance fading)
12349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
12359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::unfreezeDisplay(DisplayID dpy, uint32_t flags)
12389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
12399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
12409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
12419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
12439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mCurrentState.freezeDisplay = 0;
12449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded);
12459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // flags is intended to communicate some sort of animation behavior
1247ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian    // (for instance fading)
12489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
12499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1251e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huberint SurfaceFlinger::setOrientation(DisplayID dpy,
1252eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian        int orientation, uint32_t flags)
12539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
12549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
12559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
12569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
12589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (mCurrentState.orientation != orientation) {
12599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (uint32_t(orientation)<=eOrientation270 || orientation==42) {
1260eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian            mCurrentState.orientationType = flags;
12619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mCurrentState.orientation = orientation;
12629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            setTransactionFlags(eTransactionNeeded);
12639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mTransactionCV.wait(mStateLock);
12649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
12659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            orientation = BAD_VALUE;
12669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
12679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return orientation;
12699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12719638e5c167be321643bf3f3ee39e3fb45541fb3bMathias Agopiansp<ISurface> SurfaceFlinger::createSurface(
12729638e5c167be321643bf3f3ee39e3fb45541fb3bMathias Agopian        ISurfaceComposerClient::surface_data_t* params,
12739638e5c167be321643bf3f3ee39e3fb45541fb3bMathias Agopian        const String8& name,
12749638e5c167be321643bf3f3ee39e3fb45541fb3bMathias Agopian        const sp<Client>& client,
12759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        DisplayID d, uint32_t w, uint32_t h, PixelFormat format,
12769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        uint32_t flags)
12779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
12781473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBaseClient> layer;
12797bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    sp<ISurface> surfaceHandle;
12804d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian
12814d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian    if (int32_t(w|h) < 0) {
12824d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian        LOGE("createSurface() failed, w or h is negative (w=%d, h=%d)",
12834d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian                int(w), int(h));
12844d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian        return surfaceHandle;
12854d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian    }
1286e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber
12879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    //LOGD("createSurface for pid %d (%d x %d)", pid, w, h);
12887623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<Layer> normalLayer;
12899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (flags & eFXSurfaceMask) {
12909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceNormal:
1291d2112306330ce0c162bee4b864991962ca2b655aMathias Agopian            normalLayer = createNormalSurface(client, d, w, h, flags, format);
1292d2112306330ce0c162bee4b864991962ca2b655aMathias Agopian            layer = normalLayer;
12939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
12949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceBlur:
1295c3802d22f11b4e513ba776277d18f315c5c769f7Mathias Agopian            // for now we treat Blur as Dim, until we can implement it
1296c3802d22f11b4e513ba776277d18f315c5c769f7Mathias Agopian            // efficiently.
12979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceDim:
1298593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            layer = createDimSurface(client, d, w, h, flags);
12999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
13009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
13019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13021473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    if (layer != 0) {
1303593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        layer->initStates(w, h, flags);
13045d26c1e38dabb3ad8b4b6e1000375f3b1a6b7693Mathias Agopian        layer->setName(name);
1305593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        ssize_t token = addClientLayer(client, layer);
13067623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
13079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        surfaceHandle = layer->getSurface();
1308e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber        if (surfaceHandle != 0) {
1309593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            params->token = token;
13107bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian            params->identity = layer->getIdentity();
13117623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            if (normalLayer != 0) {
13127623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian                Mutex::Autolock _l(mStateLock);
13137bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                mLayerMap.add(layer->getSurfaceBinder(), normalLayer);
13147623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            }
131518b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian        }
13167623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
1317593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        setTransactionFlags(eTransactionNeeded);
13189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
13199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return surfaceHandle;
13219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13237623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<Layer> SurfaceFlinger::createNormalSurface(
13246edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian        const sp<Client>& client, DisplayID display,
1325593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        uint32_t w, uint32_t h, uint32_t flags,
132618b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian        PixelFormat& format)
13279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
13289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize the surfaces
13299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (format) { // TODO: take h/w into account
13309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_TRANSPARENT:
13319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_TRANSLUCENT:
13329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        format = PIXEL_FORMAT_RGBA_8888;
13339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
13349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_OPAQUE:
13354cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#ifdef NO_RGBX_8888
13364cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian        format = PIXEL_FORMAT_RGB_565;
13374cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#else
133859962ce3b0d8b7efec2840accca8fb7a6066f2bdMathias Agopian        format = PIXEL_FORMAT_RGBX_8888;
13394cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#endif
13409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
13419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
13429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13434cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#ifdef NO_RGBX_8888
13444cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian    if (format == PIXEL_FORMAT_RGBX_8888)
13454cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian        format = PIXEL_FORMAT_RGBA_8888;
13464cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#endif
13474cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian
1348593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<Layer> layer = new Layer(this, display, client);
13496edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian    status_t err = layer->setBuffers(w, h, format, flags);
1350593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (LIKELY(err != NO_ERROR)) {
13519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        LOGE("createNormalSurfaceLocked() failed (%s)", strerror(-err));
13521473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        layer.clear();
13539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
13549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return layer;
13559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13577623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<LayerDim> SurfaceFlinger::createDimSurface(
13586edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian        const sp<Client>& client, DisplayID display,
1359593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        uint32_t w, uint32_t h, uint32_t flags)
13609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1361593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<LayerDim> layer = new LayerDim(this, display, client);
13629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    layer->initStates(w, h, flags);
13639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return layer;
13649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1366593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::removeSurface(const sp<Client>& client, SurfaceID sid)
13676cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian{
13686cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    /*
13696cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     * called by the window manager, when a surface should be marked for
13706cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     * destruction.
1371e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber     *
1372a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * The surface is removed from the current and drawing lists, but placed
1373a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * in the purgatory queue, so it's not destroyed right-away (we need
1374a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * to wait for all client's references to go away first).
13756cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     */
13766cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
1377248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian    status_t err = NAME_NOT_FOUND;
1378a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian    Mutex::Autolock _l(mStateLock);
1379593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<LayerBaseClient> layer = client->getLayerUser(sid);
1380248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian    if (layer != 0) {
1381248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        err = purgatorizeLayer_l(layer);
1382248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        if (err == NO_ERROR) {
1383248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian            setTransactionFlags(eTransactionNeeded);
1384248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        }
13856cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    }
13866cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    return err;
13876cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian}
13886cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
13896960811d15eb66e8469330d3bc574c32c481db05Mathias Agopianstatus_t SurfaceFlinger::destroySurface(const wp<LayerBaseClient>& layer)
13909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1391359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian    // called by ~ISurface() when all references are gone
13926960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    status_t err = NO_ERROR;
13936960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    sp<LayerBaseClient> l(layer.promote());
13946960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    if (l != NULL) {
13956960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian        Mutex::Autolock _l(mStateLock);
13966960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian        err = removeLayer_l(l);
13976960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian        if (err == NAME_NOT_FOUND) {
13986960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian            // The surface wasn't in the current list, which means it was
13996960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian            // removed already, which means it is in the purgatory,
14006960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian            // and need to be removed from there.
14016960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian            ssize_t idx = mLayerPurgatory.remove(l);
14026960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian            LOGE_IF(idx < 0,
14036960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian                    "layer=%p is not in the purgatory list", l.get());
14046ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        }
14056960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian        LOGE_IF(err<0 && err != NAME_NOT_FOUND,
14066960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian                "error removing layer=%p (%s)", l.get(), strerror(-err));
14076960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    }
14086960811d15eb66e8469330d3bc574c32c481db05Mathias Agopian    return err;
14099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1411439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopianuint32_t SurfaceFlinger::setClientStateLocked(
1412593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const sp<Client>& client,
1413439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        const layer_state_t& s)
14149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
14159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t flags = 0;
1416439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    sp<LayerBaseClient> layer(client->getLayerUser(s.surface));
1417439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    if (layer != 0) {
1418439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        const uint32_t what = s.what;
1419439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        if (what & ePositionChanged) {
1420439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian            if (layer->setPosition(s.x, s.y))
1421439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian                flags |= eTraversalNeeded;
1422439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        }
1423439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        if (what & eLayerChanged) {
1424439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian            ssize_t idx = mCurrentState.layersSortedByZ.indexOf(layer);
1425439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian            if (layer->setLayer(s.z)) {
1426439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian                mCurrentState.layersSortedByZ.removeAt(idx);
1427439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian                mCurrentState.layersSortedByZ.add(layer);
1428439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian                // we need traversal (state changed)
1429439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian                // AND transaction (list changed)
1430439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian                flags |= eTransactionNeeded|eTraversalNeeded;
14319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
1432439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        }
1433439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        if (what & eSizeChanged) {
1434439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian            if (layer->setSize(s.w, s.h)) {
1435439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian                flags |= eTraversalNeeded;
1436439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian                mResizeTransationPending = true;
14379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
14389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1439439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        if (what & eAlphaChanged) {
1440439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian            if (layer->setAlpha(uint8_t(255.0f*s.alpha+0.5f)))
1441439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian                flags |= eTraversalNeeded;
1442439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        }
1443439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        if (what & eMatrixChanged) {
1444439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian            if (layer->setMatrix(s.matrix))
1445439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian                flags |= eTraversalNeeded;
1446439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        }
1447439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        if (what & eTransparentRegionChanged) {
1448439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian            if (layer->setTransparentRegionHint(s.transparentRegion))
1449439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian                flags |= eTraversalNeeded;
1450439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        }
1451439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        if (what & eVisibilityChanged) {
1452439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian            if (layer->setFlags(s.flags, s.mask))
1453439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian                flags |= eTraversalNeeded;
1454439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        }
14559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1456439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian    return flags;
14579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
14599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::screenReleased(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(eConsoleReleased, &mConsoleSignals);
14639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    signalEvent();
14649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
14669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::screenAcquired(int dpy)
14679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
14689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // this may be called by a signal handler, we can't do too much in here
14699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    android_atomic_or(eConsoleAcquired, &mConsoleSignals);
14709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    signalEvent();
14719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
14739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::dump(int fd, const Vector<String16>& args)
14749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
147594720d7ec40d2b5ee200d95a46588cdb0fb30e66Erik Gilling    const size_t SIZE = 4096;
14769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    char buffer[SIZE];
14779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    String8 result;
14780dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian
14790dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian    if (!PermissionCache::checkCallingPermission(sDump)) {
14809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        snprintf(buffer, SIZE, "Permission Denial: "
14819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                "can't dump SurfaceFlinger from pid=%d, uid=%d\n",
14829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                IPCThreadState::self()->getCallingPid(),
14839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                IPCThreadState::self()->getCallingUid());
14849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        result.append(buffer);
14859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
1486a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1487a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // figure out if we're stuck somewhere
1488a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t now = systemTime();
1489a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t inSwapBuffers(mDebugInSwapBuffers);
1490a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t inTransaction(mDebugInTransaction);
1491a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        nsecs_t inSwapBuffersDuration = (inSwapBuffers) ? now-inSwapBuffers : 0;
1492a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        nsecs_t inTransactionDuration = (inTransaction) ? now-inTransaction : 0;
1493a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1494a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // Try to get the main lock, but don't insist if we can't
1495a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // (this would indicate SF is stuck, but we want to be able to
1496a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // print something in dumpsys).
1497a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        int retry = 3;
1498a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        while (mStateLock.tryLock()<0 && --retry>=0) {
1499a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            usleep(1000000);
1500a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
1501a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const bool locked(retry >= 0);
1502a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (!locked) {
1503e3c01832fc741e9908f047d86cd40db1ea5d78c8Andreas Huber            snprintf(buffer, SIZE,
1504a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    "SurfaceFlinger appears to be unresponsive, "
1505a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    "dumping anyways (no locks held)\n");
1506a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1507a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
1508a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
150906a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
151006a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump the visible layer list
151106a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
15129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const LayerVector& currentLayers = mCurrentState.layersSortedByZ;
15139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const size_t count = currentLayers.size();
151406a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        snprintf(buffer, SIZE, "Visible layers (count = %d)\n", count);
151506a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        result.append(buffer);
15169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (size_t i=0 ; i<count ; i++) {
15179bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            const sp<LayerBase>& layer(currentLayers[i]);
15189bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            layer->dump(result, buffer, SIZE);
15199bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            const Layer::State& s(layer->drawingState());
15209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            s.transparentRegion.dump(result, "transparentRegion");
15219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->transparentRegionScreen.dump(result, "transparentRegionScreen");
15229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->visibleRegionScreen.dump(result, "visibleRegionScreen");
15239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
15249bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
152506a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
152606a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump the layers in the purgatory
152706a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
152806a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian
152906a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        const size_t purgatorySize =  mLayerPurgatory.size();
153006a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        snprintf(buffer, SIZE, "Purgatory state (%d entries)\n", purgatorySize);
153106a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        result.append(buffer);
153206a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        for (size_t i=0 ; i<purgatorySize ; i++) {
153306a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian            const sp<LayerBase>& layer(mLayerPurgatory.itemAt(i));
153406a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian            layer->shortDump(result, buffer, SIZE);
153506a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        }
153606a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian
153706a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
153806a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump SurfaceFlinger global state
153906a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
154006a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian
1541ab95117be00ea8a207aaeefb4cb328db7d0f8848Mathias Agopian        snprintf(buffer, SIZE, "SurfaceFlinger global state:\n");
154206a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        result.append(buffer);
1543ab95117be00ea8a207aaeefb4cb328db7d0f8848Mathias Agopian
1544ab95117be00ea8a207aaeefb4cb328db7d0f8848Mathias Agopian        const GLExtensions& extensions(GLExtensions::getInstance());
1545ab95117be00ea8a207aaeefb4cb328db7d0f8848Mathias Agopian        snprintf(buffer, SIZE, "GLES: %s, %s, %s\n",
1546ab95117be00ea8a207aaeefb4cb328db7d0f8848Mathias Agopian                extensions.getVendor(),
1547ab95117be00ea8a207aaeefb4cb328db7d0f8848Mathias Agopian                extensions.getRenderer(),
1548ab95117be00ea8a207aaeefb4cb328db7d0f8848Mathias Agopian                extensions.getVersion());
1549ab95117be00ea8a207aaeefb4cb328db7d0f8848Mathias Agopian        result.append(buffer);
1550ab95117be00ea8a207aaeefb4cb328db7d0f8848Mathias Agopian        snprintf(buffer, SIZE, "EXTS: %s\n", extensions.getExtension());
1551ab95117be00ea8a207aaeefb4cb328db7d0f8848Mathias Agopian        result.append(buffer);
1552ab95117be00ea8a207aaeefb4cb328db7d0f8848Mathias Agopian
15539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mWormholeRegion.dump(result, "WormholeRegion");
15549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw(graphicPlane(0).displayHardware());
15559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        snprintf(buffer, SIZE,
15569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                "  display frozen: %s, freezeCount=%d, orientation=%d, canDraw=%d\n",
15579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeDisplay?"yes":"no", mFreezeCount,
15589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mCurrentState.orientation, hw.canDraw());
15599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        result.append(buffer);
1560a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        snprintf(buffer, SIZE,
1561a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                "  last eglSwapBuffers() time: %f us\n"
1562a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                "  last transaction time     : %f us\n",
1563a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                mLastSwapBufferTime/1000.0, mLastTransactionTime/1000.0);
1564a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        result.append(buffer);
15659bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1566a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (inSwapBuffersDuration || !locked) {
1567a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            snprintf(buffer, SIZE, "  eglSwapBuffers time: %f us\n",
1568a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    inSwapBuffersDuration/1000.0);
1569a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1570a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
15719bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1572a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (inTransactionDuration || !locked) {
1573a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            snprintf(buffer, SIZE, "  transaction time: %f us\n",
1574a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    inTransactionDuration/1000.0);
1575a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1576a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
15779bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
157806a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
157906a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump HWComposer state
158006a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
15816a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        HWComposer& hwc(hw.getHwComposer());
15826a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        snprintf(buffer, SIZE, "  h/w composer %s and %s\n",
15836a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian                hwc.initCheck()==NO_ERROR ? "present" : "not present",
15847f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian                (mDebugDisableHWC || mDebugRegion) ? "disabled" : "enabled");
15856a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian        result.append(buffer);
158690da4dd2a20c11fa2f9b860905c867a64b2b299eMathias Agopian        hwc.dump(result, buffer, SIZE);
15876a9692424a88bf87b80870915946af401fd8dc1fMathias Agopian
158806a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian        /*
158906a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         * Dump gralloc state
159006a61e2fa830fcd66c12741a52cc5d9b4b214f94Mathias Agopian         */
15916950e428feaccc8164b989ef64e771a99948797aMathias Agopian        const GraphicBufferAllocator& alloc(GraphicBufferAllocator::get());
15921473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        alloc.dump(result);
159394720d7ec40d2b5ee200d95a46588cdb0fb30e66Erik Gilling        hw.dump(result);
1594a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1595a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (locked) {
1596a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            mStateLock.unlock();
1597a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
15989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
15999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    write(fd, result.string(), result.size());
16009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
16019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
16029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
16039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::onTransact(
16049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
16059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
16069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (code) {
16079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case CREATE_CONNECTION:
1608439863f3b3e725b5de1cba4940a21900369961c0Mathias Agopian        case SET_TRANSACTION_STATE:
16099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case SET_ORIENTATION:
16109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case FREEZE_DISPLAY:
16119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case UNFREEZE_DISPLAY:
16129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case BOOT_FINISHED:
1613aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        case TURN_ELECTRON_BEAM_OFF:
16142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        case TURN_ELECTRON_BEAM_ON:
16159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        {
16169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // codes that require permission check
16179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            IPCThreadState* ipc = IPCThreadState::self();
16189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int pid = ipc->getCallingPid();
1619627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian            const int uid = ipc->getCallingUid();
16200dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian            if ((uid != AID_GRAPHICS) &&
16210dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian                    !PermissionCache::checkPermission(sAccessSurfaceFlinger, pid, uid)) {
1622151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                LOGE("Permission Denial: "
1623151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                        "can't access SurfaceFlinger pid=%d, uid=%d", pid, uid);
1624151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                return PERMISSION_DENIED;
16259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
1626ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            break;
1627ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
1628ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        case CAPTURE_SCREEN:
1629ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        {
1630ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            // codes that require permission check
1631ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            IPCThreadState* ipc = IPCThreadState::self();
1632ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            const int pid = ipc->getCallingPid();
1633ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            const int uid = ipc->getCallingUid();
16340dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian            if ((uid != AID_GRAPHICS) &&
16350dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian                    !PermissionCache::checkPermission(sReadFramebuffer, pid, uid)) {
1636ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                LOGE("Permission Denial: "
1637ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                        "can't read framebuffer pid=%d, uid=%d", pid, uid);
1638ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                return PERMISSION_DENIED;
1639ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            }
1640ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            break;
16419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
16429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1643ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
16449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    status_t err = BnSurfaceComposer::onTransact(code, data, reply, flags);
16459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (err == UNKNOWN_TRANSACTION || err == PERMISSION_DENIED) {
16468c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        CHECK_INTERFACE(ISurfaceComposer, data, reply);
16470dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian        if (UNLIKELY(!PermissionCache::checkCallingPermission(sHardwareTest))) {
1648151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            IPCThreadState* ipc = IPCThreadState::self();
1649151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            const int pid = ipc->getCallingPid();
1650151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            const int uid = ipc->getCallingUid();
1651151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            LOGE("Permission Denial: "
1652151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                    "can't access SurfaceFlinger pid=%d, uid=%d", pid, uid);
16539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return PERMISSION_DENIED;
16549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
16559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int n;
16569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (code) {
165717f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian            case 1000: // SHOW_CPU, NOT SUPPORTED ANYMORE
165804262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            case 1001: // SHOW_FPS, NOT SUPPORTED ANYMORE
16599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1002:  // SHOW_UPDATES
16619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                n = data.readInt32();
16629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDebugRegion = n ? n : (mDebugRegion ? 0 : 1);
16637f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian                invalidateHwcGeometry();
16647f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian                repaintEverything();
16659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1003:  // SHOW_BACKGROUND
16679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                n = data.readInt32();
16689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDebugBackground = n ? 1 : 0;
16699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1004:{ // repaint everything
16717f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian                repaintEverything();
16729779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                return NO_ERROR;
16739779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian            }
16749779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian            case 1005:{ // force transaction
16759779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                setTransactionFlags(eTransactionNeeded|eTraversalNeeded);
16769779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                return NO_ERROR;
16779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
167804262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            case 1006:{ // enable/disable GraphicLog
167904262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                int enabled = data.readInt32();
168004262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                GraphicLog::getInstance().setEnabled(enabled);
168104262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                return NO_ERROR;
168204262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            }
16839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1007: // set mFreezeCount
16849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeCount = data.readInt32();
16850e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                mFreezeDisplayTime = 0;
16869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
16877f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian            case 1008:  // toggle use of hw composer
16887f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian                n = data.readInt32();
16897f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian                mDebugDisableHWC = n ? 1 : 0;
16907f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian                invalidateHwcGeometry();
16917f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian                repaintEverything();
16927f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian                return NO_ERROR;
16932143fe05e3a1aeae641ca126e76db82d17e8b8e6Mathias Agopian            case 1009:  // toggle use of transform hint
16942143fe05e3a1aeae641ca126e76db82d17e8b8e6Mathias Agopian                n = data.readInt32();
16952143fe05e3a1aeae641ca126e76db82d17e8b8e6Mathias Agopian                mDebugDisableTransformHint = n ? 1 : 0;
16962143fe05e3a1aeae641ca126e76db82d17e8b8e6Mathias Agopian                invalidateHwcGeometry();
16972143fe05e3a1aeae641ca126e76db82d17e8b8e6Mathias Agopian                repaintEverything();
16982143fe05e3a1aeae641ca126e76db82d17e8b8e6Mathias Agopian                return NO_ERROR;
16999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1010:  // interrogate.
170017f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian                reply->writeInt32(0);
17019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(0);
17029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(mDebugRegion);
17039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(mDebugBackground);
17049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
17059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1013: {
17069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                Mutex::Autolock _l(mStateLock);
17079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                const DisplayHardware& hw(graphicPlane(0).displayHardware());
17089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(hw.getPageFlipCount());
17099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
17109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return NO_ERROR;
17119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
17129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
17139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return err;
17149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
17159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
17167f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopianvoid SurfaceFlinger::repaintEverything() {
17177f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian    Mutex::Autolock _l(mStateLock);
17187f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
17197f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian    mDirtyRegion.set(hw.bounds()); // careful that's not thread-safe
17207f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian    signalEvent();
17217f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian}
17227f76a3cf667b95caccb3e6d3f5cf160180717340Mathias Agopian
1723aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian// ---------------------------------------------------------------------------
1724aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::renderScreenToTextureLocked(DisplayID dpy,
17262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        GLuint* textureName, GLfloat* uOut, GLfloat* vOut)
1727aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian{
1728aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
1729aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        return INVALID_OPERATION;
1730aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1731aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // get screen geometry
1732aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const DisplayHardware& hw(graphicPlane(dpy).displayHardware());
1733aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const uint32_t hw_w = hw.getWidth();
1734aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const uint32_t hw_h = hw.getHeight();
1735aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLfloat u = 1;
1736aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLfloat v = 1;
1737aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1738aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // make sure to clear all GL error flags
1739aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    while ( glGetError() != GL_NO_ERROR ) ;
1740aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1741aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // create a FBO
1742aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLuint name, tname;
1743aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glGenTextures(1, &tname);
1744aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glBindTexture(GL_TEXTURE_2D, tname);
17452d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB,
17462d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            hw_w, hw_h, 0, GL_RGB, GL_UNSIGNED_BYTE, 0);
1747aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (glGetError() != GL_NO_ERROR) {
1748a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian        while ( glGetError() != GL_NO_ERROR ) ;
1749aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        GLint tw = (2 << (31 - clz(hw_w)));
1750aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        GLint th = (2 << (31 - clz(hw_h)));
17512d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB,
17522d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian                tw, th, 0, GL_RGB, GL_UNSIGNED_BYTE, 0);
1753aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        u = GLfloat(hw_w) / tw;
1754aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        v = GLfloat(hw_h) / th;
1755aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
1756aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glGenFramebuffersOES(1, &name);
1757aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, name);
17582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glFramebufferTexture2DOES(GL_FRAMEBUFFER_OES,
17592d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            GL_COLOR_ATTACHMENT0_OES, GL_TEXTURE_2D, tname, 0);
1760aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17612d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // redraw the screen entirely...
17622d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glClearColor(0,0,0,1);
17632d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glClear(GL_COLOR_BUFFER_BIT);
17642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
17652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const size_t count = layers.size();
17662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (size_t i=0 ; i<count ; ++i) {
17672d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
17682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        layer->drawForSreenShot();
17692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
1770aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // back to main framebuffer
17722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
17732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_SCISSOR_TEST);
17742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDeleteFramebuffersOES(1, &name);
1775aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    *textureName = tname;
17772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    *uOut = u;
17782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    *vOut = v;
17792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
17802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
1781aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17822d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian// ---------------------------------------------------------------------------
1783aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::electronBeamOffAnimationImplLocked()
17852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
17862d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    status_t result = PERMISSION_DENIED;
1787aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17882d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
17892d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return INVALID_OPERATION;
1790aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // get screen geometry
17922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
17932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_w = hw.getWidth();
17942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_h = hw.getHeight();
17952d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const Region screenBounds(hw.bounds());
1796aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat u, v;
17982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLuint tname;
17992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    result = renderScreenToTextureLocked(0, &tname, &u, &v);
18002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (result != NO_ERROR) {
18012d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return result;
18022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
1803aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
18042d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat vtx[8];
18053a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian    const GLfloat texCoords[4][2] = { {0,1}, {0,1-v}, {u,1-v}, {u,1} };
18062d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindTexture(GL_TEXTURE_2D, tname);
18072d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
18082d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
18092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
18102d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexCoordPointer(2, GL_FLOAT, 0, texCoords);
18112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnableClientState(GL_TEXTURE_COORD_ARRAY);
18122d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glVertexPointer(2, GL_FLOAT, 0, vtx);
18132d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18143a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian    /*
18153a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     * Texture coordinate mapping
18163a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     *
18173a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     *                 u
18183a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     *    1 +----------+---+
18193a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     *      |     |    |   |  image is inverted
18203a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     *      |     V    |   |  w.r.t. the texture
18213a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     *  1-v +----------+   |  coordinates
18223a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     *      |              |
18233a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     *      |              |
18243a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     *      |              |
18253a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     *    0 +--------------+
18263a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     *      0              1
18273a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     *
18283a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian     */
18293a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian
18302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class s_curve_interpolator {
18312d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float nbFrames, s, v;
18322d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
18332d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        s_curve_interpolator(int nbFrames, float s)
18342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : nbFrames(1.0f / (nbFrames-1)), s(s),
18352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian          v(1.0f + expf(-s + 0.5f*s)) {
18362d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
18372d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float operator()(int f) {
18382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const float x = f * nbFrames;
18392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return ((1.0f/(1.0f + expf(-x*s + 0.5f*s))) - 0.5f) * v + 0.5f;
18402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
18412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
18422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class v_stretch {
18442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
18452d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
18462d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        v_stretch(uint32_t hw_w, uint32_t hw_h)
18472d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
18482d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
18492d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
18502d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w + (hw_w * v);
18512d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = hw_h - (hw_h * v);
18522d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
18532d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
18543a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian            vtx[0] = x;         vtx[1] = y;
18553a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
18563a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
18573a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
18582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
18592d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
18602d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18612d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class h_stretch {
18622d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
18632d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
18642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        h_stretch(uint32_t hw_w, uint32_t hw_h)
18652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
1866aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
18672d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
18682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w - (hw_w * v);
18692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = 1.0f;
18702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
18712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
18723a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian            vtx[0] = x;         vtx[1] = y;
18733a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
18743a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
18753a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
18762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
18772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
18782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // the full animation is 24 frames
18803a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian    char value[PROPERTY_VALUE_MAX];
18813a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian    property_get("debug.sf.electron_frames", value, "24");
18823a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian    int nbFrames = (atoi(value) + 1) >> 1;
18833a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian    if (nbFrames <= 0) // just in case
18843a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian        nbFrames = 24;
18853a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian
18862d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itr(nbFrames, 7.5f);
18872d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itg(nbFrames, 8.0f);
18882d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itb(nbFrames, 8.5f);
18892d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18902d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    v_stretch vverts(hw_w, hw_h);
18912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_BLEND);
18922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBlendFunc(GL_ONE, GL_ONE);
18932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=0 ; i<nbFrames ; i++) {
18942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float x, y, w, h;
18952d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vr = itr(i);
18962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vg = itg(i);
18972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vb = itb(i);
18982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
18992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // clear screen
19002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(1,1,1,1);
19012d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
19022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glEnable(GL_TEXTURE_2D);
1903aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
19042d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the red plane
19052d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vr);
19062d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(1,0,0,1);
19072d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
19082d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the green plane
19102d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vg);
19112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,1,0,1);
19122d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
19132d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the blue plane
19152d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vb);
19162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,0,1,1);
19172d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
19182d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the white highlight (we use the last vertices)
1920aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glDisable(GL_TEXTURE_2D);
1921aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glColorMask(1,1,1,1);
19222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColor4f(vg, vg, vg, 1);
19232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
19242d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
19252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
19262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    h_stretch hverts(hw_w, hw_h);
19282d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_BLEND);
19292d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_TEXTURE_2D);
19302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
19312d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=0 ; i<nbFrames ; i++) {
19322d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float v = itg(i);
19332d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hverts(vtx, v);
19342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
19352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColor4f(1-v, 1-v, 1-v, 1);
19362d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
19372d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
19382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
19392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
19412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_SCISSOR_TEST);
19422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisableClientState(GL_TEXTURE_COORD_ARRAY);
19432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDeleteTextures(1, &tname);
19447bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    glDisable(GL_TEXTURE_2D);
19452d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
19462d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
19472d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19482d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::electronBeamOnAnimationImplLocked()
19492d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
19502d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    status_t result = PERMISSION_DENIED;
19512d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19522d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
19532d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return INVALID_OPERATION;
19542d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19552d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19562d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // get screen geometry
19572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
19582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_w = hw.getWidth();
19592d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const uint32_t hw_h = hw.getHeight();
19602d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const Region screenBounds(hw.bounds());
19612d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19622d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat u, v;
19632d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLuint tname;
19642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    result = renderScreenToTextureLocked(0, &tname, &u, &v);
19652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (result != NO_ERROR) {
19662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return result;
19672d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
19682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    // back to main framebuffer
19702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
19712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_SCISSOR_TEST);
19722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    GLfloat vtx[8];
19742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    const GLfloat texCoords[4][2] = { {0,v}, {0,0}, {u,0}, {u,v} };
19752d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBindTexture(GL_TEXTURE_2D, tname);
19762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE);
19772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
19782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
19792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glTexCoordPointer(2, GL_FLOAT, 0, texCoords);
19802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnableClientState(GL_TEXTURE_COORD_ARRAY);
19812d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glVertexPointer(2, GL_FLOAT, 0, vtx);
19822d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19832d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class s_curve_interpolator {
19842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float nbFrames, s, v;
19852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
19862d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        s_curve_interpolator(int nbFrames, float s)
19872d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : nbFrames(1.0f / (nbFrames-1)), s(s),
19882d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian          v(1.0f + expf(-s + 0.5f*s)) {
19892d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19902d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float operator()(int f) {
19912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const float x = f * nbFrames;
19922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return ((1.0f/(1.0f + expf(-x*s + 0.5f*s))) - 0.5f) * v + 0.5f;
19932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
19942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
19952d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
19962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class v_stretch {
19972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
19982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
19992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        v_stretch(uint32_t hw_w, uint32_t hw_h)
20002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
2001aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
20022d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
20032d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w + (hw_w * v);
20042d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = hw_h - (hw_h * v);
20052d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
20062d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
20072d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
20082d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
20092d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
20102d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
20112d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
20122d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
20132d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20142d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class h_stretch {
20152d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const GLfloat hw_w, hw_h;
20162d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
20172d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        h_stretch(uint32_t hw_w, uint32_t hw_h)
20182d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        : hw_w(hw_w), hw_h(hw_h) {
20192d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
20202d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        void operator()(GLfloat* vtx, float v) {
20212d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat w = hw_w - (hw_w * v);
20222d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat h = 1.0f;
20232d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat x = (hw_w - w) * 0.5f;
20242d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            const GLfloat y = (hw_h - h) * 0.5f;
20252d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[0] = x;         vtx[1] = y;
20262d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[2] = x;         vtx[3] = y + h;
20272d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[4] = x + w;     vtx[5] = y + h;
20282d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            vtx[6] = x + w;     vtx[7] = y;
20292d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
20302d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
20312d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
2032dfa08fbfd8a5ebd013c122e391282b7f21f6342fMathias Agopian    // the full animation is 12 frames
2033dfa08fbfd8a5ebd013c122e391282b7f21f6342fMathias Agopian    int nbFrames = 8;
20342d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itr(nbFrames, 7.5f);
20352d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itg(nbFrames, 8.0f);
20362d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    s_curve_interpolator itb(nbFrames, 8.5f);
20372d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20382d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    h_stretch hverts(hw_w, hw_h);
20392d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_BLEND);
20402d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisable(GL_TEXTURE_2D);
20412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
20422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=nbFrames-1 ; i>=0 ; i--) {
20432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float v = itg(i);
20442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hverts(vtx, v);
20452d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
20462d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColor4f(1-v, 1-v, 1-v, 1);
20472d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
20482d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
20492d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
20502d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
2051dfa08fbfd8a5ebd013c122e391282b7f21f6342fMathias Agopian    nbFrames = 4;
20522d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    v_stretch vverts(hw_w, hw_h);
20532d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_BLEND);
20542d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glBlendFunc(GL_ONE, GL_ONE);
20552d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    for (int i=nbFrames-1 ; i>=0 ; i--) {
20562d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        float x, y, w, h;
20572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vr = itr(i);
20582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vg = itg(i);
20592d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        const float vb = itb(i);
2060aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
20612d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // clear screen
2062aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glColorMask(1,1,1,1);
20632d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
20642d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glEnable(GL_TEXTURE_2D);
20652d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20662d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the red plane
20672d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vr);
20682d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(1,0,0,1);
20692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
20702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the green plane
20722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vg);
20732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,1,0,1);
20742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
20752d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // draw the blue plane
20772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        vverts(vtx, vb);
20782d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glColorMask(0,0,1,1);
20792d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
20802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20812d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        hw.flip(screenBounds);
2082aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
2083aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
20842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glColorMask(1,1,1,1);
20852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glEnable(GL_SCISSOR_TEST);
20862d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    glDisableClientState(GL_TEXTURE_COORD_ARRAY);
2087aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glDeleteTextures(1, &tname);
20887bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    glDisable(GL_TEXTURE_2D);
2089aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
20902d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
20912d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
20922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
20932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian// ---------------------------------------------------------------------------
20942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
2095d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOffImplLocked(int32_t mode)
20962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
20972d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    DisplayHardware& hw(graphicPlane(0).editDisplayHardware());
20982d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (!hw.canDraw()) {
20992d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // we're already off
21002d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return NO_ERROR;
21012d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
2102d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    if (mode & ISurfaceComposer::eElectronBeamAnimationOff) {
2103d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        electronBeamOffAnimationImplLocked();
2104d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    }
2105d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian
2106d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    // always clear the whole screen at the end of the animation
2107d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glClearColor(0,0,0,1);
2108d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glDisable(GL_SCISSOR_TEST);
2109d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glClear(GL_COLOR_BUFFER_BIT);
2110d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    glEnable(GL_SCISSOR_TEST);
2111d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    hw.flip( Region(hw.bounds()) );
2112d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian
2113a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    hw.setCanDraw(false);
2114a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    return NO_ERROR;
2115aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
2116aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
2117aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOff(int32_t mode)
2118aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian{
2119aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    class MessageTurnElectronBeamOff : public MessageBase {
2120aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        SurfaceFlinger* flinger;
2121d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        int32_t mode;
2122aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        status_t result;
2123aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    public:
2124d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        MessageTurnElectronBeamOff(SurfaceFlinger* flinger, int32_t mode)
2125d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            : flinger(flinger), mode(mode), result(PERMISSION_DENIED) {
2126aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
2127aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        status_t getResult() const {
2128aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            return result;
2129aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
2130aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        virtual bool handler() {
2131aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
2132d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            result = flinger->turnElectronBeamOffImplLocked(mode);
2133aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            return true;
2134aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
2135aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    };
2136aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
2137d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    sp<MessageBase> msg = new MessageTurnElectronBeamOff(this, mode);
2138aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    status_t res = postMessageSync(msg);
2139aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (res == NO_ERROR) {
2140aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        res = static_cast<MessageTurnElectronBeamOff*>( msg.get() )->getResult();
21412d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
21422d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // work-around: when the power-manager calls us we activate the
21432d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // animation. eventually, the "on" animation will be called
21442d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // by the power-manager itself
2145d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        mElectronBeamAnimationMode = mode;
2146aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
2147aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    return res;
2148aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
2149aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
21509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
21519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2152d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOnImplLocked(int32_t mode)
21532d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
21542d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    DisplayHardware& hw(graphicPlane(0).editDisplayHardware());
21552d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    if (hw.canDraw()) {
21562d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        // we're already on
21572d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        return NO_ERROR;
21582d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    }
2159d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    if (mode & ISurfaceComposer::eElectronBeamAnimationOn) {
2160d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        electronBeamOnAnimationImplLocked();
2161d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    }
2162a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    hw.setCanDraw(true);
21638b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian
21648b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian    // make sure to redraw the whole screen when the animation is done
21658b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian    mDirtyRegion.set(hw.bounds());
21668b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian    signalEvent();
21678b6a0545adb968f15ba8bb8c7587ff9ba356db13Mathias Agopian
2168a6cd6d310473e9896c5148946fe9a5fc57db173bMathias Agopian    return NO_ERROR;
21692d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
21702d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
21712d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOn(int32_t mode)
21722d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian{
21732d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    class MessageTurnElectronBeamOn : public MessageBase {
21742d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        SurfaceFlinger* flinger;
2175d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        int32_t mode;
21762d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        status_t result;
21772d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    public:
2178d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian        MessageTurnElectronBeamOn(SurfaceFlinger* flinger, int32_t mode)
2179d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            : flinger(flinger), mode(mode), result(PERMISSION_DENIED) {
21802d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
21812d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        status_t getResult() const {
21822d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return result;
21832d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
21842d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        virtual bool handler() {
21852d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
2186d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian            result = flinger->turnElectronBeamOnImplLocked(mode);
21872d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian            return true;
21882d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian        }
21892d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    };
21902d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
2191d4e03f37423bee383d17f7292753a5f67e497a28Mathias Agopian    postMessageAsync( new MessageTurnElectronBeamOn(this, mode) );
21922d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian    return NO_ERROR;
21932d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian}
21942d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
21952d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian// ---------------------------------------------------------------------------
21962d2b803a92ba531cf6c4bb7042326653255b8780Mathias Agopian
219738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopianstatus_t SurfaceFlinger::captureScreenImplLocked(DisplayID dpy,
219838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        sp<IMemoryHeap>* heap,
219938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t* w, uint32_t* h, PixelFormat* f,
22003dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t sw, uint32_t sh,
22013dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t minLayerZ, uint32_t maxLayerZ)
220238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian{
220338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    status_t result = PERMISSION_DENIED;
220438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
220538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // only one display supported for now
220638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
220738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        return BAD_VALUE;
220838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
2209f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis    // make sure none of the layers are protected
2210e328134b8fa4e5bfe0277a1fa55a9fbd99d27ef0Mathias Agopian    const LayerVector& layers(mDrawingState.layersSortedByZ);
2211f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis    const size_t count = layers.size();
2212f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis    for (size_t i=0 ; i<count ; ++i) {
2213f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis        const sp<LayerBase>& layer(layers[i]);
2214ec49d8970125b3da422f93bf6f6b32ac62230a83Mathias Agopian        const uint32_t flags = layer->drawingState().flags;
2215ec49d8970125b3da422f93bf6f6b32ac62230a83Mathias Agopian        if (!(flags & ISurfaceComposer::eLayerHidden)) {
2216ec49d8970125b3da422f93bf6f6b32ac62230a83Mathias Agopian            const uint32_t z = layer->drawingState().z;
2217ec49d8970125b3da422f93bf6f6b32ac62230a83Mathias Agopian            if (z >= minLayerZ && z <= maxLayerZ) {
2218ec49d8970125b3da422f93bf6f6b32ac62230a83Mathias Agopian                if (layer->isProtected()) {
2219ec49d8970125b3da422f93bf6f6b32ac62230a83Mathias Agopian                    return INVALID_OPERATION;
2220ec49d8970125b3da422f93bf6f6b32ac62230a83Mathias Agopian                }
2221f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis            }
2222f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis        }
2223f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis    }
2224f72606ce3ea8cb787e5c71325f08b1898e0718d9Jamie Gennis
222538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
222638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        return INVALID_OPERATION;
222738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
222838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // get screen geometry
222938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const DisplayHardware& hw(graphicPlane(dpy).displayHardware());
223038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const uint32_t hw_w = hw.getWidth();
223138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const uint32_t hw_h = hw.getHeight();
223238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
223338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if ((sw > hw_w) || (sh > hw_h))
223438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        return BAD_VALUE;
223538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
223638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    sw = (!sw) ? hw_w : sw;
223738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    sh = (!sh) ? hw_h : sh;
223838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    const size_t size = sw * sh * 4;
223938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
224032ae094d87cbb35f8b31acb7b83b430db62d5925Mathias Agopian    //LOGD("screenshot: sw=%d, sh=%d, minZ=%d, maxZ=%d",
224132ae094d87cbb35f8b31acb7b83b430db62d5925Mathias Agopian    //        sw, sh, minLayerZ, maxLayerZ);
2242cd2cfb6a15975af02a02c34783480eb858ee2928Mathias Agopian
224338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // make sure to clear all GL error flags
224438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    while ( glGetError() != GL_NO_ERROR ) ;
224538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
224638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // create a FBO
224738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    GLuint name, tname;
224838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glGenRenderbuffersOES(1, &tname);
224938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glBindRenderbufferOES(GL_RENDERBUFFER_OES, tname);
225038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glRenderbufferStorageOES(GL_RENDERBUFFER_OES, GL_RGBA8_OES, sw, sh);
225138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glGenFramebuffersOES(1, &name);
225238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, name);
225338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES,
225438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            GL_COLOR_ATTACHMENT0_OES, GL_RENDERBUFFER_OES, tname);
225538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
225638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    GLenum status = glCheckFramebufferStatusOES(GL_FRAMEBUFFER_OES);
2257cd2cfb6a15975af02a02c34783480eb858ee2928Mathias Agopian
225838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    if (status == GL_FRAMEBUFFER_COMPLETE_OES) {
225938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
226038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // invert everything, b/c glReadPixel() below will invert the FB
226138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glViewport(0, 0, sw, sh);
22621c4e4fc0490c6f07111365cfe21116c31254d5ffMathias Agopian        glScissor(0, 0, sw, sh);
22633a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian        glEnable(GL_SCISSOR_TEST);
226438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_PROJECTION);
226538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glPushMatrix();
226638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glLoadIdentity();
22673a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian        glOrthof(0, hw_w, hw_h, 0, 0, 1);
226838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_MODELVIEW);
226938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
227038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // redraw the screen entirely...
227138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glClearColor(0,0,0,1);
227238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
22731c4e4fc0490c6f07111365cfe21116c31254d5ffMathias Agopian
227438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        for (size_t i=0 ; i<count ; ++i) {
227538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            const sp<LayerBase>& layer(layers[i]);
2276ec49d8970125b3da422f93bf6f6b32ac62230a83Mathias Agopian            const uint32_t flags = layer->drawingState().flags;
2277ec49d8970125b3da422f93bf6f6b32ac62230a83Mathias Agopian            if (!(flags & ISurfaceComposer::eLayerHidden)) {
2278ec49d8970125b3da422f93bf6f6b32ac62230a83Mathias Agopian                const uint32_t z = layer->drawingState().z;
2279ec49d8970125b3da422f93bf6f6b32ac62230a83Mathias Agopian                if (z >= minLayerZ && z <= maxLayerZ) {
2280ec49d8970125b3da422f93bf6f6b32ac62230a83Mathias Agopian                    layer->drawForSreenShot();
2281ec49d8970125b3da422f93bf6f6b32ac62230a83Mathias Agopian                }
22823dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            }
228338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        }
228438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
228538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // XXX: this is needed on tegra
22863a831d24e9147d737c6aaed72d84c1126c232368Mathias Agopian        glEnable(GL_SCISSOR_TEST);
228738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glScissor(0, 0, sw, sh);
228838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
228938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        // check for errors and return screen capture
229038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        if (glGetError() != GL_NO_ERROR) {
229138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            // error while rendering
229238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            result = INVALID_OPERATION;
229338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        } else {
229438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            // allocate shared memory large enough to hold the
229538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            // screen capture
229638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            sp<MemoryHeapBase> base(
229738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    new MemoryHeapBase(size, 0, "screen-capture") );
229838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            void* const ptr = base->getBase();
229938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            if (ptr) {
230038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                // capture the screen with glReadPixels()
230138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                glReadPixels(0, 0, sw, sh, GL_RGBA, GL_UNSIGNED_BYTE, ptr);
230238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                if (glGetError() == GL_NO_ERROR) {
230338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *heap = base;
230438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *w = sw;
230538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *h = sh;
230638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    *f = PIXEL_FORMAT_RGBA_8888;
230738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                    result = NO_ERROR;
230838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                }
230938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            } else {
231038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                result = NO_MEMORY;
231138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            }
231238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        }
231338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glEnable(GL_SCISSOR_TEST);
231438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glViewport(0, 0, hw_w, hw_h);
231538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_PROJECTION);
231638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glPopMatrix();
231738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        glMatrixMode(GL_MODELVIEW);
231838ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    } else {
231938ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        result = BAD_VALUE;
232038ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    }
232138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
232238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    // release FBO resources
232338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
232438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glDeleteRenderbuffersOES(1, &tname);
232538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    glDeleteFramebuffersOES(1, &name);
2326a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian
2327a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian    hw.compositionComplete();
2328a6b8c1c3bd1d9b84bbeb1820c0239742f50d3edbMathias Agopian
232932ae094d87cbb35f8b31acb7b83b430db62d5925Mathias Agopian    // LOGD("screenshot: result = %s", result<0 ? strerror(result) : "OK");
2330cd2cfb6a15975af02a02c34783480eb858ee2928Mathias Agopian
233138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian    return result;
233238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian}
233338ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
233438ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian
2335ca5edbeba92b96913291792a4df984e158853b6dMathias Agopianstatus_t SurfaceFlinger::captureScreen(DisplayID dpy,
2336ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        sp<IMemoryHeap>* heap,
233738ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t* width, uint32_t* height, PixelFormat* format,
23383dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t sw, uint32_t sh,
23393dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t minLayerZ, uint32_t maxLayerZ)
2340ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian{
2341ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    // only one display supported for now
2342ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
2343ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        return BAD_VALUE;
2344ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2345ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
2346ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        return INVALID_OPERATION;
2347ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2348ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    class MessageCaptureScreen : public MessageBase {
2349ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        SurfaceFlinger* flinger;
2350ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        DisplayID dpy;
2351ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        sp<IMemoryHeap>* heap;
2352ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        uint32_t* w;
2353ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        uint32_t* h;
2354ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        PixelFormat* f;
235538ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t sw;
235638ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian        uint32_t sh;
23573dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t minLayerZ;
23583dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian        uint32_t maxLayerZ;
2359ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        status_t result;
2360ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    public:
2361ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        MessageCaptureScreen(SurfaceFlinger* flinger, DisplayID dpy,
236238ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian                sp<IMemoryHeap>* heap, uint32_t* w, uint32_t* h, PixelFormat* f,
23633dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                uint32_t sw, uint32_t sh,
23643dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                uint32_t minLayerZ, uint32_t maxLayerZ)
2365ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            : flinger(flinger), dpy(dpy),
23663dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian              heap(heap), w(w), h(h), f(f), sw(sw), sh(sh),
23673dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian              minLayerZ(minLayerZ), maxLayerZ(maxLayerZ),
23683dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian              result(PERMISSION_DENIED)
2369ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        {
2370ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
2371ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        status_t getResult() const {
2372ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            return result;
2373ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
2374ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        virtual bool handler() {
2375ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
2376ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2377ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            // if we have secure windows, never allow the screen capture
2378ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            if (flinger->mSecureFrameBuffer)
2379ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                return true;
2380ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
238138ed2e39c54a42dda8f00620f960788f569a3698Mathias Agopian            result = flinger->captureScreenImplLocked(dpy,
23823dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian                    heap, w, h, f, sw, sh, minLayerZ, maxLayerZ);
2383ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2384ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            return true;
2385ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
2386ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    };
2387ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2388ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    sp<MessageBase> msg = new MessageCaptureScreen(this,
23893dd25a6bf71bd535bf9dbbe16234229ff45414a0Mathias Agopian            dpy, heap, width, height, format, sw, sh, minLayerZ, maxLayerZ);
2390ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    status_t res = postMessageSync(msg);
2391ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (res == NO_ERROR) {
2392ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        res = static_cast<MessageCaptureScreen*>( msg.get() )->getResult();
2393ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    }
2394ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    return res;
2395ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian}
2396ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
2397ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian// ---------------------------------------------------------------------------
2398ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
23997623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<Layer> SurfaceFlinger::getLayer(const sp<ISurface>& sur) const
24009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
24017623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<Layer> result;
24027623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    Mutex::Autolock _l(mStateLock);
24037623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    result = mLayerMap.valueFor( sur->asBinder() ).promote();
24047623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return result;
24057623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
2406d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
24077623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian// ---------------------------------------------------------------------------
2408593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
24097623da435e45c7c03ef6a00a43675deb6645f070Mathias AgopianClient::Client(const sp<SurfaceFlinger>& flinger)
24107623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    : mFlinger(flinger), mNameGenerator(1)
24117623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
24129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
24139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2414593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias AgopianClient::~Client()
2415593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
2416593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    const size_t count = mLayers.size();
2417593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
2418593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        sp<LayerBaseClient> layer(mLayers.valueAt(i).promote());
2419593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        if (layer != 0) {
2420593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            mFlinger->removeLayer(layer);
2421593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        }
24229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
24239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
24241473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
2425593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t Client::initCheck() const {
24267623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return NO_ERROR;
24279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
24281473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
24295fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopiansize_t Client::attachLayer(const sp<LayerBaseClient>& layer)
24309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
24315fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    Mutex::Autolock _l(mLock);
24325fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    size_t name = mNameGenerator++;
2433593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    mLayers.add(name, layer);
2434593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return name;
24359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
24369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
24377623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianvoid Client::detachLayer(const LayerBaseClient* layer)
2438593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
24395fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    Mutex::Autolock _l(mLock);
2440593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // we do a linear search here, because this doesn't happen often
2441593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    const size_t count = mLayers.size();
2442593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
2443593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        if (mLayers.valueAt(i) == layer) {
2444593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            mLayers.removeItemsAt(i, 1);
2445593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            break;
2446593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        }
2447593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    }
2448593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian}
24495fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopiansp<LayerBaseClient> Client::getLayerUser(int32_t i) const
24505fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian{
24515fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    Mutex::Autolock _l(mLock);
24521473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBaseClient> lbc;
24535fa7ad6fa1316036819ec506205888eac5512f68Mathias Agopian    wp<LayerBaseClient> layer(mLayers.valueFor(i));
2454593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (layer != 0) {
2455593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        lbc = layer.promote();
2456593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        LOGE_IF(lbc==0, "getLayerUser(name=%d) is dead", int(i));
24571473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    }
24581473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    return lbc;
24599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
24609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
24617bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian
24627bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopianstatus_t Client::onTransact(
24637bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
24647bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian{
24657bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    // these must be checked
24667bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     IPCThreadState* ipc = IPCThreadState::self();
24677bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     const int pid = ipc->getCallingPid();
24687bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     const int uid = ipc->getCallingUid();
24697bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     const int self_pid = getpid();
24707bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     if (UNLIKELY(pid != self_pid && uid != AID_GRAPHICS && uid != 0)) {
24717bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian         // we're called from a different process, do the real check
24720dd593f22352a0863223fa4ea7e37e926b99282eMathias Agopian         if (!PermissionCache::checkCallingPermission(sAccessSurfaceFlinger))
24737bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian         {
24747bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian             LOGE("Permission Denial: "
24757bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                     "can't openGlobalTransaction pid=%d, uid=%d", pid, uid);
24767bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian             return PERMISSION_DENIED;
24777bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian         }
24787bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     }
24797bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     return BnSurfaceComposerClient::onTransact(code, data, reply, flags);
24809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
24817bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian
24827bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian
2483593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopiansp<ISurface> Client::createSurface(
24849638e5c167be321643bf3f3ee39e3fb45541fb3bMathias Agopian        ISurfaceComposerClient::surface_data_t* params,
24857623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const String8& name,
24867623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        DisplayID display, uint32_t w, uint32_t h, PixelFormat format,
24879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        uint32_t flags)
24889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
24897623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    /*
24907bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     * createSurface must be called from the GL thread so that it can
24917bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian     * have access to the GL context.
24927623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     */
24937623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
24947bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    class MessageCreateSurface : public MessageBase {
24957bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        sp<ISurface> result;
24967bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        SurfaceFlinger* flinger;
24977bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        ISurfaceComposerClient::surface_data_t* params;
24987bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        Client* client;
24997bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        const String8& name;
25007bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        DisplayID display;
25017bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        uint32_t w, h;
25027bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        PixelFormat format;
25037bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        uint32_t flags;
25047bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    public:
25057bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        MessageCreateSurface(SurfaceFlinger* flinger,
25067bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                ISurfaceComposerClient::surface_data_t* params,
25077bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                const String8& name, Client* client,
25087bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                DisplayID display, uint32_t w, uint32_t h, PixelFormat format,
25097bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                uint32_t flags)
25107bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian            : flinger(flinger), params(params), client(client), name(name),
25117bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian              display(display), w(w), h(h), format(format), flags(flags)
25127bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        {
25135e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        }
25147bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        sp<ISurface> getResult() const { return result; }
25157bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        virtual bool handler() {
25167bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian            result = flinger->createSurface(params, name, client,
25177bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                    display, w, h, format, flags);
25187bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian            return true;
25197623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        }
25207bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    };
25217623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
25227bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    sp<MessageBase> msg = new MessageCreateSurface(mFlinger.get(),
25237bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian            params, name, this, display, w, h, format, flags);
25247bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    mFlinger->postMessageSync(msg);
25257bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    return static_cast<MessageCreateSurface*>( msg.get() )->getResult();
25267623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
25277bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopianstatus_t Client::destroySurface(SurfaceID sid) {
25287bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    return mFlinger->removeSurface(this, sid);
25297623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
25307623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
25317623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian// ---------------------------------------------------------------------------
25329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2533f7acf162f8d682c6ebc9af41ca76795b79509193Jamie GennisGraphicBufferAlloc::GraphicBufferAlloc() {}
2534f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis
2535f7acf162f8d682c6ebc9af41ca76795b79509193Jamie GennisGraphicBufferAlloc::~GraphicBufferAlloc() {}
2536f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis
2537f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennissp<GraphicBuffer> GraphicBufferAlloc::createGraphicBuffer(uint32_t w, uint32_t h,
2538eec0f7ebac85d3d1b1151e62b2ed0f25c138d447Mathias Agopian        PixelFormat format, uint32_t usage, status_t* error) {
2539f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    sp<GraphicBuffer> graphicBuffer(new GraphicBuffer(w, h, format, usage));
2540f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    status_t err = graphicBuffer->initCheck();
2541eec0f7ebac85d3d1b1151e62b2ed0f25c138d447Mathias Agopian    *error = err;
25427bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian    if (err != 0 || graphicBuffer->handle == 0) {
2543eec0f7ebac85d3d1b1151e62b2ed0f25c138d447Mathias Agopian        if (err == NO_MEMORY) {
2544eec0f7ebac85d3d1b1151e62b2ed0f25c138d447Mathias Agopian            GraphicBuffer::dumpAllocationsToSystemLog();
2545eec0f7ebac85d3d1b1151e62b2ed0f25c138d447Mathias Agopian        }
25467bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian        LOGE("GraphicBufferAlloc::createGraphicBuffer(w=%d, h=%d) "
25477bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian             "failed (%s), handle=%p",
25487bb843ca0777111dae7daf8f1b0705817cf523c4Mathias Agopian                w, h, strerror(-err), graphicBuffer->handle);
2549f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis        return 0;
2550f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    }
2551f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis    return graphicBuffer;
2552f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis}
2553f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis
2554f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis// ---------------------------------------------------------------------------
2555f7acf162f8d682c6ebc9af41ca76795b79509193Jamie Gennis
25569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane::GraphicPlane()
25579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    : mHw(0)
25589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
25599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane::~GraphicPlane() {
25629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    delete mHw;
25639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool GraphicPlane::initialized() const {
25669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return mHw ? true : false;
25679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
256966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianint GraphicPlane::getWidth() const {
257066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    return mWidth;
25719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
25729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
257366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianint GraphicPlane::getHeight() const {
257466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    return mHeight;
257566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian}
257666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
257766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianvoid GraphicPlane::setDisplayHardware(DisplayHardware *hw)
257866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian{
257966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mHw = hw;
258066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
258166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    // initialize the display orientation transform.
258266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    // it's a constant that should come from the display driver.
258366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    int displayOrientation = ISurfaceComposer::eOrientationDefault;
258466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    char property[PROPERTY_VALUE_MAX];
258566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    if (property_get("ro.sf.hwrotation", property, NULL) > 0) {
258666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        //displayOrientation
258766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        switch (atoi(property)) {
258866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        case 90:
258966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            displayOrientation = ISurfaceComposer::eOrientation90;
259066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            break;
259166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        case 270:
259266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            displayOrientation = ISurfaceComposer::eOrientation270;
259366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            break;
259466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        }
259566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    }
259666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
259766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float w = hw->getWidth();
259866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float h = hw->getHeight();
259966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    GraphicPlane::orientationToTransfrom(displayOrientation, w, h,
260066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            &mDisplayTransform);
260166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    if (displayOrientation & ISurfaceComposer::eOrientationSwapMask) {
260266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayWidth = h;
260366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayHeight = w;
260466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    } else {
260566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayWidth = w;
260666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayHeight = h;
260766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    }
260866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
260966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    setOrientation(ISurfaceComposer::eOrientationDefault);
26109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
26119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
26129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t GraphicPlane::orientationToTransfrom(
26139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int orientation, int w, int h, Transform* tr)
26148c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian{
26158c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    uint32_t flags = 0;
26169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (orientation) {
26179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientationDefault:
26188c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_0;
26198c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        break;
26209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation90:
26218c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_90;
26229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
26239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation180:
26248c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_180;
26259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
26269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation270:
26278c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_270;
26289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
26299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    default:
26309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
26319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
26328c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    tr->set(flags, w, h);
26339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
26349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
26359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
26369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t GraphicPlane::setOrientation(int orientation)
26379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
26389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // If the rotation can be handled in hardware, this is where
26399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // the magic should happen.
264066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
264166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const DisplayHardware& hw(displayHardware());
264266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float w = mDisplayWidth;
264366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float h = mDisplayHeight;
264466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mWidth = int(w);
264566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mHeight = int(h);
264666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
264766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    Transform orientationTransform;
26488c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    GraphicPlane::orientationToTransfrom(orientation, w, h,
26498c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian            &orientationTransform);
26508c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    if (orientation & ISurfaceComposer::eOrientationSwapMask) {
26518c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        mWidth = int(h);
26528c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        mHeight = int(w);
26539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
26548c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian
26553552f53c8370ced8680951f4ac811a126da02b0eMathias Agopian    mOrientation = orientation;
265666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mGlobalTransform = mDisplayTransform * orientationTransform;
26579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
26589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
26599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
26609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst DisplayHardware& GraphicPlane::displayHardware() const {
26619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return *mHw;
26629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
26639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2664aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias AgopianDisplayHardware& GraphicPlane::editDisplayHardware() {
2665aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    return *mHw;
2666aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
2667aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
26689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst Transform& GraphicPlane::transform() const {
26699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return mGlobalTransform;
26709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
26719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
26721473f46cbc82aa6f0ba744cc896a36923823d55bMathias AgopianEGLDisplay GraphicPlane::getEGLDisplay() const {
26731473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    return mHw->getEGLDisplay();
26741473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian}
26751473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
26769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
26779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
26789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}; // namespace android
2679