SurfaceFlinger.cpp revision aab758e87991d1460ca94d4a5f22c0ef34641e2d
19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/*
29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2007 The Android Open Source Project
39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License.
69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at
79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and
149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License.
159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <stdlib.h>
189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <stdio.h>
199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <stdint.h>
209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <unistd.h>
219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <fcntl.h>
229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <errno.h>
239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <math.h>
24207637327511c960805713971e2e7dd11cb9c290Jean-Baptiste Queru#include <limits.h>
259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <sys/types.h>
269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <sys/stat.h>
279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <sys/ioctl.h>
289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <cutils/log.h>
309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <cutils/properties.h>
319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
320795272aa226f4e965968a03daddc53ce30b7cdaMathias Agopian#include <binder/IPCThreadState.h>
330795272aa226f4e965968a03daddc53ce30b7cdaMathias Agopian#include <binder/IServiceManager.h>
34d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian#include <binder/MemoryHeapBase.h>
35d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/String8.h>
379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/String16.h>
389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <utils/StopWatch.h>
399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
406950e428feaccc8164b989ef64e771a99948797aMathias Agopian#include <ui/GraphicBufferAllocator.h>
4104262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian#include <ui/GraphicLog.h>
429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <ui/PixelFormat.h>
439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <pixelflinger/pixelflinger.h>
459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <GLES/gl.h>
469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "clz.h"
48781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian#include "GLExtensions.h"
499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "Layer.h"
509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "LayerBlur.h"
519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "LayerBuffer.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"
569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
57627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian/* ideally AID_GRAPHICS would be in a semi-public header
58627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian * or there would be a way to map a user/group name to its id
59627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian */
60627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian#ifndef AID_GRAPHICS
61627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian#define AID_GRAPHICS 1003
62627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian#endif
63627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#define DISPLAY_COUNT       1
659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectnamespace android {
679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectSurfaceFlinger::SurfaceFlinger()
709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    :   BnSurfaceComposer(), Thread(false),
719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mTransactionFlags(0),
729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mTransactionCount(0),
739779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        mResizeTransationPending(false),
741473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        mLayersRemoved(false),
759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mBootTime(systemTime()),
76151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian        mHardwareTest("android.permission.HARDWARE_TEST"),
77151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian        mAccessSurfaceFlinger("android.permission.ACCESS_SURFACE_FLINGER"),
78ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        mReadFramebuffer("android.permission.READ_FRAME_BUFFER"),
79151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian        mDump("android.permission.DUMP"),
809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mVisibleRegionsDirty(false),
819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDeferReleaseConsole(false),
829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mFreezeDisplay(false),
839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mFreezeCount(0),
84c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project        mFreezeDisplayTime(0),
859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDebugRegion(0),
869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDebugBackground(0),
87a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers(0),
88a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastSwapBufferTime(0),
89a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInTransaction(0),
90a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastTransactionTime(0),
916950e428feaccc8164b989ef64e771a99948797aMathias Agopian        mBootFinished(false),
929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mConsoleSignals(0),
939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mSecureFrameBuffer(0)
949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    init();
969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::init()
999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LOGI("SurfaceFlinger is starting");
1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // debugging stuff...
1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    char value[PROPERTY_VALUE_MAX];
1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    property_get("debug.sf.showupdates", value, "0");
1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDebugRegion = atoi(value);
1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    property_get("debug.sf.showbackground", value, "0");
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDebugBackground = atoi(value);
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
109e5c0a7b7810092a593ff386843927238c175bdbdMathias Agopian    LOGI_IF(mDebugRegion,       "showupdates enabled");
110e5c0a7b7810092a593ff386843927238c175bdbdMathias Agopian    LOGI_IF(mDebugBackground,   "showbackground enabled");
1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectSurfaceFlinger::~SurfaceFlinger()
1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDeleteTextures(1, &mWormholeTexName);
1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectoverlay_control_device_t* SurfaceFlinger::getOverlayEngine() const
1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return graphicPlane(0).displayHardware().getOverlayEngine();
1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
123d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopiansp<IMemoryHeap> SurfaceFlinger::getCblk() const
1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
125d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    return mServerHeap;
1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
128770492cb2b19f6a36ad748cd05fbedfbb9a67dfaMathias Agopiansp<ISurfaceComposerClient> SurfaceFlinger::createConnection()
1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
130593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<ISurfaceComposerClient> bclient;
131593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<Client> client(new Client(this));
132593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    status_t err = client->initCheck();
133593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (err == NO_ERROR) {
134593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        bclient = client;
1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return bclient;
1379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1397623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<ISurfaceComposerClient> SurfaceFlinger::createClientConnection()
1407623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
1417623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<ISurfaceComposerClient> bclient;
1427623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<UserClient> client(new UserClient(this));
1437623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    status_t err = client->initCheck();
1447623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (err == NO_ERROR) {
1457623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        bclient = client;
1467623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
1477623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return bclient;
1487623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
1497623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
1507623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst GraphicPlane& SurfaceFlinger::graphicPlane(int dpy) const
1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LOGE_IF(uint32_t(dpy) >= DISPLAY_COUNT, "Invalid DisplayID %d", dpy);
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(mGraphicPlanes[dpy]);
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return plane;
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane& SurfaceFlinger::graphicPlane(int dpy)
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return const_cast<GraphicPlane&>(
1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const_cast<SurfaceFlinger const *>(this)->graphicPlane(dpy));
1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::bootFinished()
1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const nsecs_t now = systemTime();
1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const nsecs_t duration = now - mBootTime;
168627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    LOGI("Boot is finished (%ld ms)", long(ns2ms(duration)) );
1696950e428feaccc8164b989ef64e771a99948797aMathias Agopian    mBootFinished = true;
170627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    property_set("ctl.stop", "bootanim");
1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::onFirstRef()
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    run("SurfaceFlinger", PRIORITY_URGENT_DISPLAY);
1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // Wait for the main thread to be done with its initialization
1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mReadyToRunBarrier.wait();
1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic inline uint16_t pack565(int r, int g, int b) {
1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return (r<<11)|(g<<5)|b;
1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::readyToRun()
1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LOGI(   "SurfaceFlinger's main thread ready to run. "
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            "Initializing graphics H/W...");
1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // we only support one display currently
1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int dpy = 0;
1929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    {
1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // initialize the main display
1959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        GraphicPlane& plane(graphicPlane(dpy));
1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        DisplayHardware* const hw = new DisplayHardware(this, dpy);
1979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        plane.setDisplayHardware(hw);
1989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
200d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    // create the shared control-block
201d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    mServerHeap = new MemoryHeapBase(4096,
202d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian            MemoryHeapBase::READ_ONLY, "SurfaceFlinger read-only heap");
203d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    LOGE_IF(mServerHeap==0, "can't create shared memory dealer");
204d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
205d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    mServerCblk = static_cast<surface_flinger_cblk_t*>(mServerHeap->getBase());
206d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    LOGE_IF(mServerCblk==0, "can't get to shared control block's address");
207d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
208d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian    new(mServerCblk) surface_flinger_cblk_t;
209d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
2109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize primary screen
2119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // (other display should be initialized in the same manner, but
2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // asynchronously, as they could come and go. None of this is supported
2139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // yet).
2149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(dpy));
2159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw = plane.displayHardware();
2169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint32_t w = hw.getWidth();
2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint32_t h = hw.getHeight();
2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint32_t f = hw.getFormat();
2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    hw.makeCurrent();
2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize the shared control block
2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mServerCblk->connected |= 1<<dpy;
2239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    display_cblk_t* dcblk = mServerCblk->displays + dpy;
2249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    memset(dcblk, 0, sizeof(display_cblk_t));
22566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    dcblk->w            = plane.getWidth();
22666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    dcblk->h            = plane.getHeight();
2279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->format       = f;
2289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->orientation  = ISurfaceComposer::eOrientationDefault;
2299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->xdpi         = hw.getDpiX();
2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->ydpi         = hw.getDpiY();
2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->fps          = hw.getRefreshRate();
2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    dcblk->density      = hw.getDensity();
2339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // Initialize OpenGL|ES
2359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glPixelStorei(GL_UNPACK_ALIGNMENT, 4);
2369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glPixelStorei(GL_PACK_ALIGNMENT, 4);
2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glEnableClientState(GL_VERTEX_ARRAY);
2389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glEnable(GL_SCISSOR_TEST);
2399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glShadeModel(GL_FLAT);
2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
2419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_CULL_FACE);
2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint16_t g0 = pack565(0x0F,0x1F,0x0F);
2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint16_t g1 = pack565(0x17,0x2f,0x17);
2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const uint16_t textureData[4] = { g0, g1, g1, g0 };
2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glGenTextures(1, &mWormholeTexName);
2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glBindTexture(GL_TEXTURE_2D, mWormholeTexName);
2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, 2, 2, 0,
2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            GL_RGB, GL_UNSIGNED_SHORT_5_6_5, textureData);
2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glViewport(0, 0, w, h);
2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glMatrixMode(GL_PROJECTION);
2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glLoadIdentity();
2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glOrthof(0, w, h, 0, 0, 1);
2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project   LayerDim::initDimmer(this, w, h);
2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mReadyToRunBarrier.open();
2639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
2659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *  We're now ready to accept clients...
2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
268627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    // start boot animation
269627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian    property_set("ctl.start", "bootanim");
270627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian
2719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#if 0
2769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark -
2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark Events Handler
2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif
2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::waitForEvent()
2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
2826ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    while (true) {
2836ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        nsecs_t timeout = -1;
2840e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        const nsecs_t freezeDisplayTimeout = ms2ns(5000);
2856ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        if (UNLIKELY(isFrozen())) {
2866ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            // wait 5 seconds
2876ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            const nsecs_t now = systemTime();
2886ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            if (mFreezeDisplayTime == 0) {
2896ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian                mFreezeDisplayTime = now;
2906ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            }
2916ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            nsecs_t waitTime = freezeDisplayTimeout - (now - mFreezeDisplayTime);
2926ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            timeout = waitTime>0 ? waitTime : 0;
293c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project        }
2946ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian
295898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        sp<MessageBase> msg = mEventQueue.waitMessage(timeout);
2960e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian
2970e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        // see if we timed out
2980e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        if (isFrozen()) {
2990e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            const nsecs_t now = systemTime();
3000e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            nsecs_t frozenTime = (now - mFreezeDisplayTime);
3010e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            if (frozenTime >= freezeDisplayTimeout) {
3029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // we timed out and are still frozen
3039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                LOGW("timeout expired mFreezeDisplay=%d, mFreezeCount=%d",
3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        mFreezeDisplay, mFreezeCount);
3050e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                mFreezeDisplayTime = 0;
3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeCount = 0;
307c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project                mFreezeDisplay = false;
3080e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            }
3090e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        }
3100e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian
3110e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian        if (msg != 0) {
3120e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian            switch (msg->what) {
3130e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                case MessageQueue::INVALIDATE:
3140e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                    // invalidate message, just return to the main loop
3150e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                    return;
3169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
3179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::signalEvent() {
3226ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    mEventQueue.invalidate();
3239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::signal() const {
3266ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    // this is the IPC call
3276ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    const_cast<SurfaceFlinger*>(this)->signalEvent();
3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
330898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianstatus_t SurfaceFlinger::postMessageAsync(const sp<MessageBase>& msg,
331898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        nsecs_t reltime, uint32_t flags)
3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
333898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    return mEventQueue.postMessage(msg, reltime, flags);
334898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian}
335898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian
336898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianstatus_t SurfaceFlinger::postMessageSync(const sp<MessageBase>& msg,
337898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        nsecs_t reltime, uint32_t flags)
338898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian{
339898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    status_t res = mEventQueue.postMessage(msg, reltime, flags);
340898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    if (res == NO_ERROR) {
341898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        msg->wait();
342898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    }
343898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    return res;
3449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
3459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ----------------------------------------------------------------------------
3479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#if 0
3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark -
3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#pragma mark Main loop
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#endif
3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool SurfaceFlinger::threadLoop()
3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
3549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    waitForEvent();
3559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // check for transactions
3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(mConsoleSignals)) {
3589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        handleConsoleEvents();
3599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (LIKELY(mTransactionCount == 0)) {
3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // if we're in a global transaction, don't do anything.
3639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const uint32_t mask = eTransactionNeeded | eTraversalNeeded;
3649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        uint32_t transactionFlags = getTransactionFlags(mask);
3659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (LIKELY(transactionFlags)) {
3669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            handleTransaction(transactionFlags);
3679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // post surfaces (if needed)
3719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    handlePageFlip();
3729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw(graphicPlane(0).displayHardware());
37481384bf927c47a4efa653b14273084a13e67e3acMathias Agopian    if (LIKELY(hw.canDraw() && !isFrozen())) {
3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // repaint the framebuffer (if needed)
37604262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian
37704262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        const int index = hw.getCurrentBufferIndex();
37804262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        GraphicLog& logger(GraphicLog::getInstance());
37904262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian
38004262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_REPAINT, index);
3819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        handleRepaint();
3829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
383b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian        // inform the h/w that we're done compositing
38404262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_COMPOSITION_COMPLETE, index);
385b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian        hw.compositionComplete();
386b1a1874625a6aca967601fcda53a82d8d712df45Mathias Agopian
3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // release the clients before we flip ('cause flip might block)
38804262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_UNLOCK_CLIENTS, index);
3899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        unlockClients();
3909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
39104262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_SWAP_BUFFERS, index);
3929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        postFramebuffer();
39304262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian
39404262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian        logger.log(GraphicLog::SF_REPAINT_DONE, index);
3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
3969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // pretend we did the post
3979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        unlockClients();
3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        usleep(16667); // 60 fps period
3999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return true;
4019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::postFramebuffer()
4049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (!mInvalidRegion.isEmpty()) {
4069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw(graphicPlane(0).displayHardware());
407a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t now = systemTime();
408a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers = now;
4099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.flip(mInvalidRegion);
410a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastSwapBufferTime = systemTime() - now;
411a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInSwapBuffers = 0;
4129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mInvalidRegion.clear();
4139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleConsoleEvents()
4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // something to do with the console
4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw = graphicPlane(0).displayHardware();
4209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int what = android_atomic_and(0, &mConsoleSignals);
4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (what & eConsoleAcquired) {
4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.acquireScreen();
4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
426aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (mDeferReleaseConsole && hw.isScreenAcquired()) {
427ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian        // We got the release signal before the acquire signal
4289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mDeferReleaseConsole = false;
4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        hw.releaseScreen();
4309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (what & eConsoleReleased) {
433aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        if (hw.isScreenAcquired()) {
4349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            hw.releaseScreen();
4359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
4369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDeferReleaseConsole = true;
4379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.set(hw.bounds());
4419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
4429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleTransaction(uint32_t transactionFlags)
4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
4452d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    Vector< sp<LayerBase> > ditchedLayers;
4462d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian
447ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian    /*
448ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     * Perform and commit the transaction
449ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     */
450ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian
4512d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    { // scope for the lock
4522d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian        Mutex::Autolock _l(mStateLock);
453a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t now = systemTime();
454a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInTransaction = now;
4552d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian        handleTransactionLocked(transactionFlags, ditchedLayers);
456a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mLastTransactionTime = systemTime() - now;
457a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        mDebugInTransaction = 0;
458ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian        // here the transaction has been committed
4592d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    }
4602d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian
461ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian    /*
462ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     * Clean-up all layers that went away
463ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     * (do this without the lock held)
464ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian     */
4652d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    const size_t count = ditchedLayers.size();
4662d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
467ffae4fcc78d0f280da6052d102b11962fb8041b7Mathias Agopian        if (ditchedLayers[i] != 0) {
468ffae4fcc78d0f280da6052d102b11962fb8041b7Mathias Agopian            //LOGD("ditching layer %p", ditchedLayers[i].get());
469ffae4fcc78d0f280da6052d102b11962fb8041b7Mathias Agopian            ditchedLayers[i]->ditch();
470ffae4fcc78d0f280da6052d102b11962fb8041b7Mathias Agopian        }
4712d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    }
4722d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian}
4739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4742d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopianvoid SurfaceFlinger::handleTransactionLocked(
4752d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian        uint32_t transactionFlags, Vector< sp<LayerBase> >& ditchedLayers)
4762d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian{
4772d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    const LayerVector& currentLayers(mCurrentState.layersSortedByZ);
4789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const size_t count = currentLayers.size();
4799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
4819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Traversal of the children
4829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * (perform the transaction for each of them if needed)
4839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
4849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const bool layersNeedTransaction = transactionFlags & eTraversalNeeded;
4869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (layersNeedTransaction) {
4879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (size_t i=0 ; i<count ; i++) {
4881473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian            const sp<LayerBase>& layer = currentLayers[i];
4899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            uint32_t trFlags = layer->getTransactionFlags(eTransactionNeeded);
4909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (!trFlags) continue;
4919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const uint32_t flags = layer->doTransaction(0);
4939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (flags & Layer::eVisibleRegion)
4949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mVisibleRegionsDirty = true;
4959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /*
4999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Perform our own transaction if needed
5009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (transactionFlags & eTransactionNeeded) {
5039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mCurrentState.orientation != mDrawingState.orientation) {
5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // the orientation has changed, recompute all visible regions
5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // and invalidate everything.
5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int dpy = 0;
5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int orientation = mCurrentState.orientation;
509eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian            const uint32_t type = mCurrentState.orientationType;
5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            GraphicPlane& plane(graphicPlane(dpy));
5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            plane.setOrientation(orientation);
5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // update the shared control block
5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const DisplayHardware& hw(plane.displayHardware());
5159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            volatile display_cblk_t* dcblk = mServerCblk->displays + dpy;
5169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dcblk->orientation = orientation;
51766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            dcblk->w = plane.getWidth();
51866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            dcblk->h = plane.getHeight();
5199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
5219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(hw.bounds());
5229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mCurrentState.freezeDisplay != mDrawingState.freezeDisplay) {
5259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // freezing or unfreezing the display -> trigger animation if needed
5269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mFreezeDisplay = mCurrentState.freezeDisplay;
52731901cc0b6f0c678be4f629c8c3405700e63c346Mathias Agopian            if (mFreezeDisplay)
52831901cc0b6f0c678be4f629c8c3405700e63c346Mathias Agopian                 mFreezeDisplayTime = 0;
5299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
531a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        if (currentLayers.size() > mDrawingState.layersSortedByZ.size()) {
532a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            // layers have been added
5339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
5349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
536a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        // some layers might have been removed, so
537a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        // we need to update the regions they're exposing.
538a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        if (mLayersRemoved) {
539248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian            mLayersRemoved = false;
5409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = true;
541a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            const LayerVector& previousLayers(mDrawingState.layersSortedByZ);
5422d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian            const size_t count = previousLayers.size();
5432d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian            for (size_t i=0 ; i<count ; i++) {
544a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                const sp<LayerBase>& layer(previousLayers[i]);
545a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                if (currentLayers.indexOf( layer ) < 0) {
546a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                    // this layer is not visible anymore
5472d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian                    ditchedLayers.add(layer);
54833863dd9e1005478d20b70fad42e447ac97f5abfMathias Agopian                    mDirtyRegionRemovedLayer.orSelf(layer->visibleRegionScreen);
549a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                }
550a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            }
5519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    commitTransaction();
5559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
5569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectsp<FreezeLock> SurfaceFlinger::getFreezeLock() const
5589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
5599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return new FreezeLock(const_cast<SurfaceFlinger *>(this));
5609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
5619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::computeVisibleRegions(
5639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    LayerVector& currentLayers, Region& dirtyRegion, Region& opaqueRegion)
5649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
5659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(0));
5669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Transform& planeTransform(plane.transform());
5679c041bbd81789c209e2369ba958306979b67614fMathias Agopian    const DisplayHardware& hw(plane.displayHardware());
5689c041bbd81789c209e2369ba958306979b67614fMathias Agopian    const Region screenRegion(hw.bounds());
5699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region aboveOpaqueLayers;
5719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region aboveCoveredLayers;
5729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Region dirty;
5739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool secureFrameBuffer = false;
5759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t i = currentLayers.size();
5779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    while (i--) {
5781473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        const sp<LayerBase>& layer = currentLayers[i];
5799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->validateVisibility(planeTransform);
5809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // start with the whole surface at its current location
58212cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        const Layer::State& s(layer->drawingState());
5839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5849c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
5859c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * opaqueRegion: area of a surface that is fully opaque.
5869c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
5879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region opaqueRegion;
5889c041bbd81789c209e2369ba958306979b67614fMathias Agopian
5899c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
5909c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * visibleRegion: area of a surface that is visible on screen
5919c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * and not fully transparent. This is essentially the layer's
5929c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * footprint minus the opaque regions above it.
5939c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * Areas covered by a translucent surface are considered visible.
5949c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
5959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region visibleRegion;
5969c041bbd81789c209e2369ba958306979b67614fMathias Agopian
5979c041bbd81789c209e2369ba958306979b67614fMathias Agopian        /*
5989c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * coveredRegion: area of a surface that is covered by all
5999c041bbd81789c209e2369ba958306979b67614fMathias Agopian         * visible regions above it (which includes the translucent areas).
6009c041bbd81789c209e2369ba958306979b67614fMathias Agopian         */
6019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Region coveredRegion;
6029c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6039c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6049c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // handle hidden surfaces by setting the visible region to empty
60512cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        if (LIKELY(!(s.flags & ISurfaceComposer::eLayerHidden) && s.alpha)) {
6069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const bool translucent = layer->needsBlending();
60712cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian            const Rect bounds(layer->visibleBounds());
6089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            visibleRegion.set(bounds);
6099c041bbd81789c209e2369ba958306979b67614fMathias Agopian            visibleRegion.andSelf(screenRegion);
6109c041bbd81789c209e2369ba958306979b67614fMathias Agopian            if (!visibleRegion.isEmpty()) {
6119c041bbd81789c209e2369ba958306979b67614fMathias Agopian                // Remove the transparent area from the visible region
6129c041bbd81789c209e2369ba958306979b67614fMathias Agopian                if (translucent) {
6139c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    visibleRegion.subtractSelf(layer->transparentRegionScreen);
6149c041bbd81789c209e2369ba958306979b67614fMathias Agopian                }
6159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6169c041bbd81789c209e2369ba958306979b67614fMathias Agopian                // compute the opaque region
6179c041bbd81789c209e2369ba958306979b67614fMathias Agopian                const int32_t layerOrientation = layer->getOrientation();
6189c041bbd81789c209e2369ba958306979b67614fMathias Agopian                if (s.alpha==255 && !translucent &&
6199c041bbd81789c209e2369ba958306979b67614fMathias Agopian                        ((layerOrientation & Transform::ROT_INVALID) == false)) {
6209c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    // the opaque region is the layer's footprint
6219c041bbd81789c209e2369ba958306979b67614fMathias Agopian                    opaqueRegion = visibleRegion;
6229c041bbd81789c209e2369ba958306979b67614fMathias Agopian                }
6239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
6249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6269c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Clip the covered region to the visible region
6279c041bbd81789c209e2369ba958306979b67614fMathias Agopian        coveredRegion = aboveCoveredLayers.intersect(visibleRegion);
6289c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6299c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Update aboveCoveredLayers for next (lower) layer
6309c041bbd81789c209e2369ba958306979b67614fMathias Agopian        aboveCoveredLayers.orSelf(visibleRegion);
6319c041bbd81789c209e2369ba958306979b67614fMathias Agopian
6329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // subtract the opaque region covered by the layers above us
6339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        visibleRegion.subtractSelf(aboveOpaqueLayers);
6349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // compute this layer's dirty region
6369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (layer->contentDirty) {
6379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // we need to invalidate the whole region
6389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dirty = visibleRegion;
6399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // as well, as the old visible region
6409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            dirty.orSelf(layer->visibleRegionScreen);
6419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->contentDirty = false;
6429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
6430aed7e97a3efbeec23a4aa301df03036a67f8ea3Mathias Agopian            /* compute the exposed region:
6449c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   the exposed region consists of two components:
6459c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   1) what's VISIBLE now and was COVERED before
6469c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *   2) what's EXPOSED now less what was EXPOSED before
6479c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *
6489c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * note that (1) is conservative, we start with the whole
6499c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * visible region but only keep what used to be covered by
6509c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * something -- which mean it may have been exposed.
6519c041bbd81789c209e2369ba958306979b67614fMathias Agopian             *
6529c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * (2) handles areas that were not covered by anything but got
6539c041bbd81789c209e2369ba958306979b67614fMathias Agopian             * exposed because of a resize.
6540aed7e97a3efbeec23a4aa301df03036a67f8ea3Mathias Agopian             */
6559c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region newExposed = visibleRegion - coveredRegion;
6569c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldVisibleRegion = layer->visibleRegionScreen;
6579c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldCoveredRegion = layer->coveredRegionScreen;
6589c041bbd81789c209e2369ba958306979b67614fMathias Agopian            const Region oldExposed = oldVisibleRegion - oldCoveredRegion;
6599c041bbd81789c209e2369ba958306979b67614fMathias Agopian            dirty = (visibleRegion&oldCoveredRegion) | (newExposed-oldExposed);
6609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dirty.subtractSelf(aboveOpaqueLayers);
6629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // accumulate to the screen dirty region
6649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        dirtyRegion.orSelf(dirty);
6659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6669c041bbd81789c209e2369ba958306979b67614fMathias Agopian        // Update aboveOpaqueLayers for next (lower) layer
6679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        aboveOpaqueLayers.orSelf(opaqueRegion);
6689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Store the visible region is screen space
6709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->setVisibleRegion(visibleRegion);
6719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->setCoveredRegion(coveredRegion);
6729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
67312cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian        // If a secure layer is partially visible, lock-down the screen!
6749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (layer->isSecure() && !visibleRegion.isEmpty()) {
6759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            secureFrameBuffer = true;
6769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
67912cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    // invalidate the areas where a layer was removed
68012cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    dirtyRegion.orSelf(mDirtyRegionRemovedLayer);
68112cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian    mDirtyRegionRemovedLayer.clear();
68212cedff50ca229a5d025e1011f1518c32e858138Mathias Agopian
6839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mSecureFrameBuffer = secureFrameBuffer;
6849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    opaqueRegion = aboveOpaqueLayers;
6859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
6869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::commitTransaction()
6899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
6909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDrawingState = mCurrentState;
6919779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian    mResizeTransationPending = false;
6929779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian    mTransactionCV.broadcast();
6939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
6949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handlePageFlip()
6969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
6979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool visibleRegions = mVisibleRegionsDirty;
698cfa275908a220c5e1cf496f7fdde1c04e24e95daMathias Agopian    LayerVector& currentLayers = const_cast<LayerVector&>(
699cfa275908a220c5e1cf496f7fdde1c04e24e95daMathias Agopian            mDrawingState.layersSortedByZ);
7009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    visibleRegions |= lockPageFlip(currentLayers);
7019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw = graphicPlane(0).displayHardware();
7039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const Region screenRegion(hw.bounds());
7049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (visibleRegions) {
7059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            Region opaqueRegion;
7069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            computeVisibleRegions(currentLayers, mDirtyRegion, opaqueRegion);
707ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian
708ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            /*
709ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian             *  rebuild the visible layer list
710ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian             */
711ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            mVisibleLayersSortedByZ.clear();
712ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            const LayerVector& currentLayers(mDrawingState.layersSortedByZ);
713ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            size_t count = currentLayers.size();
714ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            mVisibleLayersSortedByZ.setCapacity(count);
715ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            for (size_t i=0 ; i<count ; i++) {
716ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian                if (!currentLayers[i]->visibleRegionScreen.isEmpty())
717ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian                    mVisibleLayersSortedByZ.add(currentLayers[i]);
718ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian            }
719ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian
7209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mWormholeRegion = screenRegion.subtract(opaqueRegion);
7219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mVisibleRegionsDirty = false;
7229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    unlockPageFlip(currentLayers);
7259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.andSelf(screenRegion);
7269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool SurfaceFlinger::lockPageFlip(const LayerVector& currentLayers)
7299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    bool recomputeVisibleRegions = false;
7319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t count = currentLayers.size();
7321473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBase> const* layers = currentLayers.array();
7339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (size_t i=0 ; i<count ; i++) {
7347623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
7359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->lockPageFlip(recomputeVisibleRegions);
7369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return recomputeVisibleRegions;
7389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::unlockPageFlip(const LayerVector& currentLayers)
7419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const GraphicPlane& plane(graphicPlane(0));
7439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Transform& planeTransform(plane.transform());
7449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    size_t count = currentLayers.size();
7451473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBase> const* layers = currentLayers.array();
7469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (size_t i=0 ; i<count ; i++) {
7477623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
7489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->unlockPageFlip(planeTransform, mDirtyRegion);
7499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
7519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7528c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian
7539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::handleRepaint()
7549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
7558c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    // compute the invalid region
7568c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    mInvalidRegion.orSelf(mDirtyRegion);
7578c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    if (mInvalidRegion.isEmpty()) {
7588c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        // nothing to do
7598c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        return;
7608c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    }
7619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(mDebugRegion)) {
7639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        debugFlashRegions();
7649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
7659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7668c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    // set the frame buffer
7678c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
7688c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    glMatrixMode(GL_MODELVIEW);
7698c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    glLoadIdentity();
7709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t flags = hw.getFlags();
7722e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian    if ((flags & DisplayHardware::SWAP_RECTANGLE) ||
7732e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian        (flags & DisplayHardware::BUFFER_PRESERVED))
7742e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian    {
775ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // we can redraw only what's dirty, but since SWAP_RECTANGLE only
776ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // takes a rectangle, we must make sure to update that whole
777ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        // rectangle in that case
778ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        if (flags & DisplayHardware::SWAP_RECTANGLE) {
7797623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            // TODO: we really should be able to pass a region to
780ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // SWAP_RECTANGLE so that we don't have to redraw all this.
781ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            mDirtyRegion.set(mInvalidRegion.bounds());
782ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        } else {
783ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // in the BUFFER_PRESERVED case, obviously, we can update only
784ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // what's needed and nothing more.
785ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // NOTE: this is NOT a common case, as preserving the backbuffer
786ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // is costly and usually involves copying the whole update back.
787ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian        }
7889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
789f2d28b74850ea0869aef2ce0727a6abb7b166a5cMathias Agopian        if (flags & DisplayHardware::PARTIAL_UPDATES) {
790ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // We need to redraw the rectangle that will be updated
7912e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian            // (pushed to the framebuffer).
792f2d28b74850ea0869aef2ce0727a6abb7b166a5cMathias Agopian            // This is needed because PARTIAL_UPDATES only takes one
793ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // rectangle instead of a region (see DisplayHardware::flip())
7949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(mInvalidRegion.bounds());
7959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
796ecfa7ccd4fc3066a4cfd4d84d7de286d95581c78Mathias Agopian            // we need to redraw everything (the whole screen)
7979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mDirtyRegion.set(hw.bounds());
7989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mInvalidRegion = mDirtyRegion;
7999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // compose all surfaces
8039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    composeSurfaces(mDirtyRegion);
8049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // clear the dirty regions
8069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mDirtyRegion.clear();
8079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
8089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::composeSurfaces(const Region& dirty)
8109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
8119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(!mWormholeRegion.isEmpty())) {
8129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // should never happen unless the window manager has a bug
8139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // draw something...
8149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        drawWormhole();
8159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
816ff1d4102882ae9641a61fc4a3937866521bb72dcMathias Agopian    const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
817cfa275908a220c5e1cf496f7fdde1c04e24e95daMathias Agopian    const size_t count = layers.size();
818cfa275908a220c5e1cf496f7fdde1c04e24e95daMathias Agopian    for (size_t i=0 ; i<count ; ++i) {
819f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        const sp<LayerBase>& layer(layers[i]);
820f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        const Region clip(dirty.intersect(layer->visibleRegionScreen));
821f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        if (!clip.isEmpty()) {
822f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian            layer->draw(clip);
823f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian        }
824f8e705dea48f77f1c2532fdbadd4997dd1851af0Mathias Agopian    }
8259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
8269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::unlockClients()
8289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
8299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const LayerVector& drawingLayers(mDrawingState.layersSortedByZ);
8309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const size_t count = drawingLayers.size();
8311473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBase> const* const layers = drawingLayers.array();
8329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (size_t i=0 ; i<count ; ++i) {
8331473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        const sp<LayerBase>& layer = layers[i];
8349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        layer->finishPageFlip();
8359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
8379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::debugFlashRegions()
8399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
840f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    const DisplayHardware& hw(graphicPlane(0).displayHardware());
841f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    const uint32_t flags = hw.getFlags();
842f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
843f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    if (!((flags & DisplayHardware::SWAP_RECTANGLE) ||
844f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian            (flags & DisplayHardware::BUFFER_PRESERVED))) {
845f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        const Region repaint((flags & DisplayHardware::PARTIAL_UPDATES) ?
846f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian                mDirtyRegion.bounds() : hw.bounds());
847f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        composeSurfaces(repaint);
848f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    }
849f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
850f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian    TextureManager::deactivateTextures();
8512e20bffbab8084fedce39d14d7dd17b08f6e9ba2Mathias Agopian
8529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_BLEND);
8539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
8549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_SCISSOR_TEST);
8559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
856dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    static int toggle = 0;
857dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    toggle = 1 - toggle;
858dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    if (toggle) {
859f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glColor4f(1, 0, 1, 1);
860dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    } else {
861f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glColor4f(1, 1, 0, 1);
862dff8e58d47ede6e748c0b02e128ca33b42a4f362Mathias Agopian    }
8639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8646158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    Region::const_iterator it = mDirtyRegion.begin();
8656158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    Region::const_iterator const end = mDirtyRegion.end();
8666158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian    while (it != end) {
8676158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        const Rect& r = *it++;
8689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        GLfloat vertices[][2] = {
8699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.left,  r.top },
8709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.left,  r.bottom },
8719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.right, r.bottom },
8729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { r.right, r.top }
8739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        };
8749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glVertexPointer(2, GL_FLOAT, 0, vertices);
8759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
8769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
877f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian
8788c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    if (mInvalidRegion.isEmpty()) {
8798c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        mDirtyRegion.dump("mDirtyRegion");
8808c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        mInvalidRegion.dump("mInvalidRegion");
8818c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    }
8828c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian    hw.flip(mInvalidRegion);
8839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (mDebugRegion > 1)
885f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        usleep(mDebugRegion * 1000);
8869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glEnable(GL_SCISSOR_TEST);
8889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    //mDirtyRegion.dump("mDirtyRegion");
8899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
8909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::drawWormhole() const
8929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
8939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const Region region(mWormholeRegion.intersect(mDirtyRegion));
8949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (region.isEmpty())
8959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return;
8969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const DisplayHardware& hw(graphicPlane(0).displayHardware());
8989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int32_t width = hw.getWidth();
8999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const int32_t height = hw.getHeight();
9009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_BLEND);
9029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    glDisable(GL_DITHER);
9039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (LIKELY(!mDebugBackground)) {
905f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian        glClearColor(0,0,0,0);
9066158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator it = region.begin();
9076158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator const end = region.end();
9086158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        while (it != end) {
9096158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian            const Rect& r = *it++;
9109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const GLint sy = height - (r.top + r.height());
9119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glScissor(r.left, sy, r.width(), r.height());
9129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glClear(GL_COLOR_BUFFER_BIT);
9139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
9159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const GLshort vertices[][2] = { { 0, 0 }, { width, 0 },
9169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                { width, height }, { 0, height }  };
9179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const GLshort tcoords[][2] = { { 0, 0 }, { 1, 0 },  { 1, 1 }, { 0, 1 } };
9189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glVertexPointer(2, GL_SHORT, 0, vertices);
9199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glTexCoordPointer(2, GL_SHORT, 0, tcoords);
9209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glEnableClientState(GL_TEXTURE_COORD_ARRAY);
921e20a56d929fc8fedc2b468ea6d1900bd2aa6e81aMichael I. Gold#if defined(GL_OES_EGL_image_external)
922781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian        if (GLExtensions::getInstance().haveTextureExternal()) {
923781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian            glDisable(GL_TEXTURE_EXTERNAL_OES);
924781953d62dc17d761e39540f0480e5ca7451cdbeMathias Agopian        }
925f8b4b4408cb864bf604608221eafa9d37323d348Mathias Agopian#endif
9269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glEnable(GL_TEXTURE_2D);
9279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glBindTexture(GL_TEXTURE_2D, mWormholeTexName);
9289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
9299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glMatrixMode(GL_TEXTURE);
9309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glLoadIdentity();
9319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glScalef(width*(1.0f/32.0f), height*(1.0f/32.0f), 1);
9326158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator it = region.begin();
9336158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        Region::const_iterator const end = region.end();
9346158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian        while (it != end) {
9356158b1bf0364da1582468a98ec09d004ba99deecMathias Agopian            const Rect& r = *it++;
9369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const GLint sy = height - (r.top + r.height());
9379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glScissor(r.left, sy, r.width(), r.height());
9389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
9399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        glDisableClientState(GL_TEXTURE_COORD_ARRAY);
9419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
9439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::debugShowFPS() const
9459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
9469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int mFrameCount;
9479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int mLastFrameCount = 0;
9489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static nsecs_t mLastFpsTime = 0;
9499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static float mFps = 0;
9509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mFrameCount++;
9519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    nsecs_t now = systemTime();
9529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    nsecs_t diff = now - mLastFpsTime;
9539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (diff > ms2ns(250)) {
9549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mFps =  ((mFrameCount - mLastFrameCount) * float(s2ns(1))) / diff;
9559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mLastFpsTime = now;
9569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mLastFrameCount = mFrameCount;
9579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // XXX: mFPS has the value we want
9599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project }
9609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9611473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::addLayer(const sp<LayerBase>& layer)
9629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
9639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
9649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    addLayer_l(layer);
9659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded|eTraversalNeeded);
9669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
9679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
9689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9691473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::addLayer_l(const sp<LayerBase>& layer)
9709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
9711efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian    ssize_t i = mCurrentState.layersSortedByZ.add(layer);
9729bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian    return (i < 0) ? status_t(i) : status_t(NO_ERROR);
9739bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian}
9749bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
975593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianssize_t SurfaceFlinger::addClientLayer(const sp<Client>& client,
976593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const sp<LayerBaseClient>& lbc)
9779bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian{
978593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    Mutex::Autolock _l(mStateLock);
979593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
980593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // attach this layer to the client
981593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    ssize_t name = client->attachLayer(lbc);
982593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
983593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // add this layer to the current state list
984593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    addLayer_l(lbc);
985593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
986593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return name;
987593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian}
988593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
989593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::removeLayer(const sp<LayerBase>& layer)
990593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
991593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    Mutex::Autolock _l(mStateLock);
992593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    status_t err = purgatorizeLayer_l(layer);
993593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (err == NO_ERROR)
994593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        setTransactionFlags(eTransactionNeeded);
995593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return err;
9969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
9979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9981473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopianstatus_t SurfaceFlinger::removeLayer_l(const sp<LayerBase>& layerBase)
9999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10007623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<LayerBaseClient> lbc(layerBase->getLayerBaseClient());
10017623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (lbc != 0) {
10027623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        mLayerMap.removeItem( lbc->getSurface()->asBinder() );
10037623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
10049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    ssize_t index = mCurrentState.layersSortedByZ.remove(layerBase);
10059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (index >= 0) {
10061473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        mLayersRemoved = true;
10079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return NO_ERROR;
10089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10092d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    return status_t(index);
10109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10126cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopianstatus_t SurfaceFlinger::purgatorizeLayer_l(const sp<LayerBase>& layerBase)
10136cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian{
10142e4b68d57bb64d7e93139238c5a8be91ff956c2aMathias Agopian    // remove the layer from the main list (through a transaction).
10156cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    ssize_t err = removeLayer_l(layerBase);
10162e4b68d57bb64d7e93139238c5a8be91ff956c2aMathias Agopian
10170c4cec7e4df87181486d280c98fba9c0f4774c37Mathias Agopian    layerBase->onRemoved();
10180c4cec7e4df87181486d280c98fba9c0f4774c37Mathias Agopian
10192d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    // it's possible that we don't find a layer, because it might
10202d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian    // have been destroyed already -- this is not technically an error
1021593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // from the user because there is a race between Client::destroySurface(),
1022593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // ~Client() and ~ISurface().
10236cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    return (err == NAME_NOT_FOUND) ? status_t(NO_ERROR) : err;
10246cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian}
10256cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
1026593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::invalidateLayerVisibility(const sp<LayerBase>& layer)
10279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1028593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    layer->forceVisibilityTransaction();
1029593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    setTransactionFlags(eTraversalNeeded);
1030593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return NO_ERROR;
10319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectuint32_t SurfaceFlinger::getTransactionFlags(uint32_t flags)
10349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return android_atomic_and(~flags, &mTransactionFlags) & flags;
10369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1038898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopianuint32_t SurfaceFlinger::setTransactionFlags(uint32_t flags)
10399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t old = android_atomic_or(flags, &mTransactionFlags);
10419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if ((old & flags)==0) { // wake the server up
1042898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian        signalEvent();
10439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return old;
10459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::openGlobalTransaction()
10489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    android_atomic_inc(&mTransactionCount);
10509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::closeGlobalTransaction()
10539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (android_atomic_dec(&mTransactionCount) == 1) {
10559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        signalEvent();
10569779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian
10579779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        // if there is a transaction with a resize, wait for it to
10589779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        // take effect before returning.
10599779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        Mutex::Autolock _l(mStateLock);
10609779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        while (mResizeTransationPending) {
106198a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian            status_t err = mTransactionCV.waitRelative(mStateLock, s2ns(5));
106298a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian            if (CC_UNLIKELY(err != NO_ERROR)) {
106398a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                // just in case something goes wrong in SF, return to the
106498a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                // called after a few seconds.
106598a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                LOGW_IF(err == TIMED_OUT, "closeGlobalTransaction timed out!");
106698a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                mResizeTransationPending = false;
106798a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian                break;
106898a9c56acf1857adaf1773bd91c88a8364fd3fa8Mathias Agopian            }
10699779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian        }
10709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::freezeDisplay(DisplayID dpy, uint32_t flags)
10749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
10769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
10779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
10799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mCurrentState.freezeDisplay = 1;
10809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded);
10819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // flags is intended to communicate some sort of animation behavior
1083ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian    // (for instance fading)
10849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
10859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::unfreezeDisplay(DisplayID dpy, uint32_t flags)
10889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
10899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
10909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
10919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
10939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    mCurrentState.freezeDisplay = 0;
10949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    setTransactionFlags(eTransactionNeeded);
10959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // flags is intended to communicate some sort of animation behavior
1097ed81f2216a0f200e279de61ec96fbedf5c21f0c7Mathias Agopian    // (for instance fading)
10989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
10999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1101eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopianint SurfaceFlinger::setOrientation(DisplayID dpy,
1102eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian        int orientation, uint32_t flags)
11039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
11059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
11069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
11089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (mCurrentState.orientation != orientation) {
11099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (uint32_t(orientation)<=eOrientation270 || orientation==42) {
1110eb0c86e18b7d620b679ff2a45a0233867a53a334Mathias Agopian            mCurrentState.orientationType = flags;
11119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mCurrentState.orientation = orientation;
11129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            setTransactionFlags(eTransactionNeeded);
11139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mTransactionCV.wait(mStateLock);
11149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
11159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            orientation = BAD_VALUE;
11169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
11179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return orientation;
11199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1121593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopiansp<ISurface> SurfaceFlinger::createSurface(const sp<Client>& client, int pid,
1122770492cb2b19f6a36ad748cd05fbedfbb9a67dfaMathias Agopian        const String8& name, ISurfaceComposerClient::surface_data_t* params,
11239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        DisplayID d, uint32_t w, uint32_t h, PixelFormat format,
11249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        uint32_t flags)
11259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11261473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBaseClient> layer;
11279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    sp<LayerBaseClient::Surface> surfaceHandle;
11284d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian
11294d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian    if (int32_t(w|h) < 0) {
11304d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian        LOGE("createSurface() failed, w or h is negative (w=%d, h=%d)",
11314d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian                int(w), int(h));
11324d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian        return surfaceHandle;
11334d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian    }
11344d2dbebf3d08209f751585d8cc367369e2f6e32fMathias Agopian
11359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    //LOGD("createSurface for pid %d (%d x %d)", pid, w, h);
11367623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<Layer> normalLayer;
11379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (flags & eFXSurfaceMask) {
11389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceNormal:
11399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (UNLIKELY(flags & ePushBuffers)) {
1140593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian                layer = createPushBuffersSurface(client, d, w, h, flags);
11419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else {
11427623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian                normalLayer = createNormalSurface(client, d, w, h, flags, format);
11437623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian                layer = normalLayer;
11449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
11459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
11469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceBlur:
1147593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            layer = createBlurSurface(client, d, w, h, flags);
11489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
11499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case eFXSurfaceDim:
1150593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            layer = createDimSurface(client, d, w, h, flags);
11519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            break;
11529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11541473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    if (layer != 0) {
1155593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        layer->initStates(w, h, flags);
11565d26c1e38dabb3ad8b4b6e1000375f3b1a6b7693Mathias Agopian        layer->setName(name);
1157593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        ssize_t token = addClientLayer(client, layer);
11587623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
11599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        surfaceHandle = layer->getSurface();
116018b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian        if (surfaceHandle != 0) {
1161593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            params->token = token;
116218b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->identity = surfaceHandle->getIdentity();
116318b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->width = w;
116418b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->height = h;
116518b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian            params->format = format;
11667623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            if (normalLayer != 0) {
11677623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian                Mutex::Autolock _l(mStateLock);
11687623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian                mLayerMap.add(surfaceHandle->asBinder(), normalLayer);
11697623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            }
117018b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian        }
11717623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
1172593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        setTransactionFlags(eTransactionNeeded);
11739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return surfaceHandle;
11769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
11779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11787623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<Layer> SurfaceFlinger::createNormalSurface(
11796edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian        const sp<Client>& client, DisplayID display,
1180593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        uint32_t w, uint32_t h, uint32_t flags,
118118b6b49ea5235fb6c0802db9a0cc2c6dd20646cbMathias Agopian        PixelFormat& format)
11829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
11839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // initialize the surfaces
11849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (format) { // TODO: take h/w into account
11859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_TRANSPARENT:
11869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_TRANSLUCENT:
11879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        format = PIXEL_FORMAT_RGBA_8888;
11889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
11899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case PIXEL_FORMAT_OPAQUE:
11904cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#ifdef NO_RGBX_8888
11914cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian        format = PIXEL_FORMAT_RGB_565;
11924cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#else
119359962ce3b0d8b7efec2840accca8fb7a6066f2bdMathias Agopian        format = PIXEL_FORMAT_RGBX_8888;
11944cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#endif
11959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
11969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
11979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11984cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#ifdef NO_RGBX_8888
11994cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian    if (format == PIXEL_FORMAT_RGBX_8888)
12004cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian        format = PIXEL_FORMAT_RGBA_8888;
12014cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian#endif
12024cfb3a69d2906fc2539dd30508aa6ba8d39065a9Mathias Agopian
1203593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<Layer> layer = new Layer(this, display, client);
12046edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian    status_t err = layer->setBuffers(w, h, format, flags);
1205593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (LIKELY(err != NO_ERROR)) {
12069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        LOGE("createNormalSurfaceLocked() failed (%s)", strerror(-err));
12071473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        layer.clear();
12089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
12099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return layer;
12109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12127623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<LayerBlur> SurfaceFlinger::createBlurSurface(
12136edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian        const sp<Client>& client, DisplayID display,
1214593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        uint32_t w, uint32_t h, uint32_t flags)
12159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1216593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<LayerBlur> layer = new LayerBlur(this, display, client);
12179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    layer->initStates(w, h, flags);
12189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return layer;
12199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12217623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<LayerDim> SurfaceFlinger::createDimSurface(
12226edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian        const sp<Client>& client, DisplayID display,
1223593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        uint32_t w, uint32_t h, uint32_t flags)
12249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1225593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<LayerDim> layer = new LayerDim(this, display, client);
12269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    layer->initStates(w, h, flags);
12279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return layer;
12289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12307623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<LayerBuffer> SurfaceFlinger::createPushBuffersSurface(
12316edf5af578c1ab1fcd44b7c08ca371456e4b7430Mathias Agopian        const sp<Client>& client, DisplayID display,
1232593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        uint32_t w, uint32_t h, uint32_t flags)
12339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1234593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<LayerBuffer> layer = new LayerBuffer(this, display, client);
12359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    layer->initStates(w, h, flags);
12369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return layer;
12379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1239593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t SurfaceFlinger::removeSurface(const sp<Client>& client, SurfaceID sid)
12406cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian{
12416cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    /*
12426cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     * called by the window manager, when a surface should be marked for
12436cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     * destruction.
1244a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     *
1245a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * The surface is removed from the current and drawing lists, but placed
1246a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * in the purgatory queue, so it's not destroyed right-away (we need
1247a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian     * to wait for all client's references to go away first).
12486cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian     */
12496cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
1250248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian    status_t err = NAME_NOT_FOUND;
1251a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian    Mutex::Autolock _l(mStateLock);
1252593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    sp<LayerBaseClient> layer = client->getLayerUser(sid);
1253248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian    if (layer != 0) {
1254248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        err = purgatorizeLayer_l(layer);
1255248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        if (err == NO_ERROR) {
1256248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian            setTransactionFlags(eTransactionNeeded);
1257248b5bd51e325107f8119b564db6a06ac51c232aMathias Agopian        }
12586cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    }
12596cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian    return err;
12606cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian}
12616cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
12626cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopianstatus_t SurfaceFlinger::destroySurface(const sp<LayerBaseClient>& layer)
12639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
1264359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian    // called by ~ISurface() when all references are gone
12656cf0db228ca275dfcda57d79c55e5fa306809632Mathias Agopian
12666ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    class MessageDestroySurface : public MessageBase {
1267a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        SurfaceFlinger* flinger;
12686ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        sp<LayerBaseClient> layer;
12696ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    public:
1270a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian        MessageDestroySurface(
1271a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian                SurfaceFlinger* flinger, const sp<LayerBaseClient>& layer)
1272a3aa6c9aa96873a70e2ff3170218a275f503520eMathias Agopian            : flinger(flinger), layer(layer) { }
12736ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        virtual bool handler() {
1274c8fb5b1979da4829e1486e6a1008c06c979b94b0Mathias Agopian            sp<LayerBaseClient> l(layer);
1275c8fb5b1979da4829e1486e6a1008c06c979b94b0Mathias Agopian            layer.clear(); // clear it outside of the lock;
12766ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
1277359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian            /*
1278359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             * remove the layer from the current list -- chances are that it's
1279359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             * not in the list anyway, because it should have been removed
1280359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             * already upon request of the client (eg: window manager).
1281359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             * However, a buggy client could have not done that.
1282359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             * Since we know we don't have any more clients, we don't need
1283359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             * to use the purgatory.
1284359140c171f67b9b29a1beae9743b49d0759414bMathias Agopian             */
1285c8fb5b1979da4829e1486e6a1008c06c979b94b0Mathias Agopian            status_t err = flinger->removeLayer_l(l);
12862e4b68d57bb64d7e93139238c5a8be91ff956c2aMathias Agopian            LOGE_IF(err<0 && err != NAME_NOT_FOUND,
12872e4b68d57bb64d7e93139238c5a8be91ff956c2aMathias Agopian                    "error removing layer=%p (%s)", l.get(), strerror(-err));
12886ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian            return true;
12896ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian        }
12906ead5d9f140529edfb744584fa5427b84b4dc13aMathias Agopian    };
12912d5ee25556cb363700dc5127f88ca05fb9493e14Mathias Agopian
1292898c4c91be8e11b6d5388c623ae80f12ac25fd27Mathias Agopian    postMessageAsync( new MessageDestroySurface(this, layer) );
12939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
12949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
12959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::setClientState(
1297593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const sp<Client>& client,
12989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int32_t count,
12999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const layer_state_t* states)
13009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
13019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    Mutex::Autolock _l(mStateLock);
13029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t flags = 0;
13039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    for (int i=0 ; i<count ; i++) {
1304593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        const layer_state_t& s(states[i]);
1305593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        sp<LayerBaseClient> layer(client->getLayerUser(s.surface));
13061473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        if (layer != 0) {
13079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const uint32_t what = s.what;
13089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & ePositionChanged) {
13099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setPosition(s.x, s.y))
13109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
13119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
13129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eLayerChanged) {
13131efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian                ssize_t idx = mCurrentState.layersSortedByZ.indexOf(layer);
13149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setLayer(s.z)) {
13151efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian                    mCurrentState.layersSortedByZ.removeAt(idx);
13161efba9a61716356014f4c452fd9e099c5ebe9bd0Mathias Agopian                    mCurrentState.layersSortedByZ.add(layer);
13179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // we need traversal (state changed)
13189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // AND transaction (list changed)
13199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTransactionNeeded|eTraversalNeeded;
13209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
13219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
13229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eSizeChanged) {
13239779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                if (layer->setSize(s.w, s.h)) {
13249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
13259779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                    mResizeTransationPending = true;
13269779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                }
13279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
13289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eAlphaChanged) {
13299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setAlpha(uint8_t(255.0f*s.alpha+0.5f)))
13309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
13319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
13329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eMatrixChanged) {
13339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setMatrix(s.matrix))
13349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
13359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
13369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eTransparentRegionChanged) {
13379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setTransparentRegionHint(s.transparentRegion))
13389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
13399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
13409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (what & eVisibilityChanged) {
13419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (layer->setFlags(s.flags, s.mask))
13429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    flags |= eTraversalNeeded;
13439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
13449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
13459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
13469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (flags) {
13479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setTransactionFlags(flags);
13489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
13499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
13509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::screenReleased(int dpy)
13539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
13549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // this may be called by a signal handler, we can't do too much in here
13559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    android_atomic_or(eConsoleReleased, &mConsoleSignals);
13569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    signalEvent();
13579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectvoid SurfaceFlinger::screenAcquired(int dpy)
13609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
13619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // this may be called by a signal handler, we can't do too much in here
13629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    android_atomic_or(eConsoleAcquired, &mConsoleSignals);
13639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    signalEvent();
13649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
13659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::dump(int fd, const Vector<String16>& args)
13679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
13689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    const size_t SIZE = 1024;
13699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    char buffer[SIZE];
13709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    String8 result;
1371151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian    if (!mDump.checkCalling()) {
13729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        snprintf(buffer, SIZE, "Permission Denial: "
13739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                "can't dump SurfaceFlinger from pid=%d, uid=%d\n",
13749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                IPCThreadState::self()->getCallingPid(),
13759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                IPCThreadState::self()->getCallingUid());
13769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        result.append(buffer);
13779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    } else {
1378a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1379a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // figure out if we're stuck somewhere
1380a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t now = systemTime();
1381a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t inSwapBuffers(mDebugInSwapBuffers);
1382a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const nsecs_t inTransaction(mDebugInTransaction);
1383a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        nsecs_t inSwapBuffersDuration = (inSwapBuffers) ? now-inSwapBuffers : 0;
1384a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        nsecs_t inTransactionDuration = (inTransaction) ? now-inTransaction : 0;
1385a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1386a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // Try to get the main lock, but don't insist if we can't
1387a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // (this would indicate SF is stuck, but we want to be able to
1388a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        // print something in dumpsys).
1389a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        int retry = 3;
1390a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        while (mStateLock.tryLock()<0 && --retry>=0) {
1391a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            usleep(1000000);
1392a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
1393a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        const bool locked(retry >= 0);
1394a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (!locked) {
1395a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            snprintf(buffer, SIZE,
1396a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    "SurfaceFlinger appears to be unresponsive, "
1397a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    "dumping anyways (no locks held)\n");
1398a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1399a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
1400a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
14019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const LayerVector& currentLayers = mCurrentState.layersSortedByZ;
14029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const size_t count = currentLayers.size();
14039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (size_t i=0 ; i<count ; i++) {
14049bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            const sp<LayerBase>& layer(currentLayers[i]);
14059bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            layer->dump(result, buffer, SIZE);
14069bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian            const Layer::State& s(layer->drawingState());
14079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            s.transparentRegion.dump(result, "transparentRegion");
14089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->transparentRegionScreen.dump(result, "transparentRegionScreen");
14099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            layer->visibleRegionScreen.dump(result, "visibleRegionScreen");
14109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
14119bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
14129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mWormholeRegion.dump(result, "WormholeRegion");
14139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const DisplayHardware& hw(graphicPlane(0).displayHardware());
14149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        snprintf(buffer, SIZE,
14159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                "  display frozen: %s, freezeCount=%d, orientation=%d, canDraw=%d\n",
14169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeDisplay?"yes":"no", mFreezeCount,
14179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mCurrentState.orientation, hw.canDraw());
14189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        result.append(buffer);
1419a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        snprintf(buffer, SIZE,
1420a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                "  last eglSwapBuffers() time: %f us\n"
1421a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                "  last transaction time     : %f us\n",
1422a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                mLastSwapBufferTime/1000.0, mLastTransactionTime/1000.0);
1423a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        result.append(buffer);
14249bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1425a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (inSwapBuffersDuration || !locked) {
1426a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            snprintf(buffer, SIZE, "  eglSwapBuffers time: %f us\n",
1427a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    inSwapBuffersDuration/1000.0);
1428a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1429a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
14309bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
1431a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (inTransactionDuration || !locked) {
1432a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            snprintf(buffer, SIZE, "  transaction time: %f us\n",
1433a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian                    inTransactionDuration/1000.0);
1434a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            result.append(buffer);
1435a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
14369bce8737f2f7581ade57445286aa150de051ff89Mathias Agopian
14376950e428feaccc8164b989ef64e771a99948797aMathias Agopian        const GraphicBufferAllocator& alloc(GraphicBufferAllocator::get());
14381473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian        alloc.dump(result);
1439a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian
1440a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        if (locked) {
1441a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian            mStateLock.unlock();
1442a8d49178f94e025be8be00460739315b9c273e4cMathias Agopian        }
14439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
14449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    write(fd, result.string(), result.size());
14459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
14469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
14479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
14489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t SurfaceFlinger::onTransact(
14499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
14509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
14519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (code) {
14529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case CREATE_CONNECTION:
14539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case OPEN_GLOBAL_TRANSACTION:
14549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case CLOSE_GLOBAL_TRANSACTION:
14559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case SET_ORIENTATION:
14569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case FREEZE_DISPLAY:
14579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case UNFREEZE_DISPLAY:
14589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        case BOOT_FINISHED:
1459aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        case TURN_ELECTRON_BEAM_OFF:
14609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        {
14619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // codes that require permission check
14629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            IPCThreadState* ipc = IPCThreadState::self();
14639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            const int pid = ipc->getCallingPid();
1464627e7b50be41e4fdee758a1bfad3a55de56b4e27Mathias Agopian            const int uid = ipc->getCallingUid();
1465151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            if ((uid != AID_GRAPHICS) && !mAccessSurfaceFlinger.check(pid, uid)) {
1466151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                LOGE("Permission Denial: "
1467151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                        "can't access SurfaceFlinger pid=%d, uid=%d", pid, uid);
1468151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                return PERMISSION_DENIED;
14699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
1470ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            break;
1471ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
1472ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        case CAPTURE_SCREEN:
1473ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        {
1474ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            // codes that require permission check
1475ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            IPCThreadState* ipc = IPCThreadState::self();
1476ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            const int pid = ipc->getCallingPid();
1477ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            const int uid = ipc->getCallingUid();
1478ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            if ((uid != AID_GRAPHICS) && !mReadFramebuffer.check(pid, uid)) {
1479ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                LOGE("Permission Denial: "
1480ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                        "can't read framebuffer pid=%d, uid=%d", pid, uid);
1481ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                return PERMISSION_DENIED;
1482ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            }
1483ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            break;
14849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
14859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1486ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
14879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    status_t err = BnSurfaceComposer::onTransact(code, data, reply, flags);
14889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    if (err == UNKNOWN_TRANSACTION || err == PERMISSION_DENIED) {
14898c9687ae291a1e0a63d4236069066284f56192cdMathias Agopian        CHECK_INTERFACE(ISurfaceComposer, data, reply);
1490151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian        if (UNLIKELY(!mHardwareTest.checkCalling())) {
1491151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            IPCThreadState* ipc = IPCThreadState::self();
1492151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            const int pid = ipc->getCallingPid();
1493151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            const int uid = ipc->getCallingUid();
1494151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian            LOGE("Permission Denial: "
1495151e859e0fc3a930bdf6d270d275e69e9eba0cbfMathias Agopian                    "can't access SurfaceFlinger pid=%d, uid=%d", pid, uid);
14969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return PERMISSION_DENIED;
14979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
14989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int n;
14999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (code) {
150017f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian            case 1000: // SHOW_CPU, NOT SUPPORTED ANYMORE
150104262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            case 1001: // SHOW_FPS, NOT SUPPORTED ANYMORE
15029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
15039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1002:  // SHOW_UPDATES
15049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                n = data.readInt32();
15059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDebugRegion = n ? n : (mDebugRegion ? 0 : 1);
15069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
15079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1003:  // SHOW_BACKGROUND
15089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                n = data.readInt32();
15099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDebugBackground = n ? 1 : 0;
15109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
15119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1004:{ // repaint everything
15129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                Mutex::Autolock _l(mStateLock);
15139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                const DisplayHardware& hw(graphicPlane(0).displayHardware());
15149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mDirtyRegion.set(hw.bounds()); // careful that's not thread-safe
15159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                signalEvent();
15169779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                return NO_ERROR;
15179779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian            }
15189779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian            case 1005:{ // force transaction
15199779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                setTransactionFlags(eTransactionNeeded|eTraversalNeeded);
15209779b221e999583ff89e0dfc40e56398737adbb3Mathias Agopian                return NO_ERROR;
15219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
152204262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            case 1006:{ // enable/disable GraphicLog
152304262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                int enabled = data.readInt32();
152404262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                GraphicLog::getInstance().setEnabled(enabled);
152504262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian                return NO_ERROR;
152604262e9f842edf20168399b6a70f0d67e518fe69Mathias Agopian            }
15279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1007: // set mFreezeCount
15289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mFreezeCount = data.readInt32();
15290e44976aa425b0c2a0dcf6ec8db918e6a6f51b1aMathias Agopian                mFreezeDisplayTime = 0;
15309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
15319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1010:  // interrogate.
153217f638b39f2e8b610ecfa1290e5bc42ab7700c98Mathias Agopian                reply->writeInt32(0);
15339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(0);
15349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(mDebugRegion);
15359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(mDebugBackground);
15369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return NO_ERROR;
15379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case 1013: {
15389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                Mutex::Autolock _l(mStateLock);
15399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                const DisplayHardware& hw(graphicPlane(0).displayHardware());
15409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                reply->writeInt32(hw.getPageFlipCount());
15419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
15429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return NO_ERROR;
15439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
15449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
15459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return err;
15469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
15479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1548aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1549aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian// ---------------------------------------------------------------------------
1550aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1551aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOffImplLocked()
1552aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian{
1553aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    status_t result = PERMISSION_DENIED;
1554aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1555aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
1556aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        return INVALID_OPERATION;
1557aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1558aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // get screen geometry
1559aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const int dpy = 0;
1560aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const DisplayHardware& hw(graphicPlane(dpy).displayHardware());
1561aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (!hw.canDraw()) {
1562aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        // we're already off
1563aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        return NO_ERROR;
1564aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
1565aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1566aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const uint32_t hw_w = hw.getWidth();
1567aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const uint32_t hw_h = hw.getHeight();
1568aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    const Region screenBounds(hw.bounds());
1569aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLfloat u = 1;
1570aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLfloat v = 1;
1571aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1572aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // make sure to clear all GL error flags
1573aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    while ( glGetError() != GL_NO_ERROR ) ;
1574aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1575aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    // create a FBO
1576aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLuint name, tname;
1577aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glGenTextures(1, &tname);
1578aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glBindTexture(GL_TEXTURE_2D, tname);
1579aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
1580aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glTexParameterx(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
1581aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, hw_w, hw_h, 0, GL_RGB, GL_UNSIGNED_BYTE, 0);
1582aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (glGetError() != GL_NO_ERROR) {
1583aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        GLint tw = (2 << (31 - clz(hw_w)));
1584aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        GLint th = (2 << (31 - clz(hw_h)));
1585aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, tw, th, 0, GL_RGB, GL_UNSIGNED_BYTE, 0);
1586aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        u = GLfloat(hw_w) / tw;
1587aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        v = GLfloat(hw_h) / th;
1588aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
1589aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glGenFramebuffersOES(1, &name);
1590aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, name);
1591aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glFramebufferTexture2DOES(GL_FRAMEBUFFER_OES, GL_COLOR_ATTACHMENT0_OES, GL_TEXTURE_2D, tname, 0);
1592aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1593aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    GLenum status = glCheckFramebufferStatusOES(GL_FRAMEBUFFER_OES);
1594aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (status == GL_FRAMEBUFFER_COMPLETE_OES) {
1595aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        // redraw the screen entirely...
1596aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glClearColor(0,0,0,1);
1597aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
1598aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
1599aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        const size_t count = layers.size();
1600aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        for (size_t i=0 ; i<count ; ++i) {
1601aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            const sp<LayerBase>& layer(layers[i]);
1602aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            layer->drawForSreenShot();
1603aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
1604aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        // back to main framebuffer
1605aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
1606aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glDisable(GL_SCISSOR_TEST);
1607aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1608aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        GLfloat vtx[8];
1609aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        const GLfloat texCoords[4][2] = { {0,v}, {0,0}, {u,0}, {u,v} };
1610aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glEnable(GL_TEXTURE_2D);
1611aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glBindTexture(GL_TEXTURE_2D, tname);
1612aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glTexEnvx(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
1613aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glTexCoordPointer(2, GL_FLOAT, 0, texCoords);
1614aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glEnableClientState(GL_TEXTURE_COORD_ARRAY);
1615aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glVertexPointer(2, GL_FLOAT, 0, vtx);
1616aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1617aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        class s_curve_interpolator {
1618aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            const float nbFrames, s, v;
1619aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        public:
1620aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            s_curve_interpolator(int nbFrames, float s)
1621aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                : nbFrames(1.0f / (nbFrames-1)), s(s),
1622aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                  v(1.0f + expf(-s + 0.5f*s)) {
1623aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            }
1624aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            float operator()(int f) {
1625aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                const float x = f * nbFrames;
1626aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                return ((1.0f/(1.0f + expf(-x*s + 0.5f*s))) - 0.5f) * v + 0.5f;
1627aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            }
1628aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        };
1629aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1630aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        class v_stretch {
1631aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            const GLfloat hw_w, hw_h;
1632aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        public:
1633aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            v_stretch(uint32_t hw_w, uint32_t hw_h)
1634aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                : hw_w(hw_w), hw_h(hw_h) {
1635aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            }
1636aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            void operator()(GLfloat* vtx, float v) {
1637aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                const GLfloat w = hw_w + (hw_w * v);
1638aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                const GLfloat h = hw_h - (hw_h * v);
1639aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                const GLfloat x = (hw_w - w) * 0.5f;
1640aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                const GLfloat y = (hw_h - h) * 0.5f;
1641aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                vtx[0] = x;         vtx[1] = y;
1642aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                vtx[2] = x;         vtx[3] = y + h;
1643aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                vtx[4] = x + w;     vtx[5] = y + h;
1644aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                vtx[6] = x + w;     vtx[7] = y;
1645aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            }
1646aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        };
1647aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1648aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        class h_stretch {
1649aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            const GLfloat hw_w, hw_h;
1650aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        public:
1651aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            h_stretch(uint32_t hw_w, uint32_t hw_h)
1652aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                : hw_w(hw_w), hw_h(hw_h) {
1653aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            }
1654aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            void operator()(GLfloat* vtx, float v) {
1655aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                const GLfloat w = hw_w - (hw_w * v);
1656aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                const GLfloat h = 1.0f;
1657aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                const GLfloat x = (hw_w - w) * 0.5f;
1658aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                const GLfloat y = (hw_h - h) * 0.5f;
1659aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                vtx[0] = x;         vtx[1] = y;
1660aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                vtx[2] = x;         vtx[3] = y + h;
1661aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                vtx[4] = x + w;     vtx[5] = y + h;
1662aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian                vtx[6] = x + w;     vtx[7] = y;
1663aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            }
1664aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        };
1665aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1666aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        // the full animation is 24 frames
1667aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        const int nbFrames = 12;
1668aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1669aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        v_stretch vverts(hw_w, hw_h);
1670aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        s_curve_interpolator itr(nbFrames, 7.5f);
1671aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        s_curve_interpolator itg(nbFrames, 8.0f);
1672aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        s_curve_interpolator itb(nbFrames, 8.5f);
1673aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glEnable(GL_BLEND);
1674aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glBlendFunc(GL_ONE, GL_ONE);
1675aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        for (int i=0 ; i<nbFrames ; i++) {
1676aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            float x, y, w, h;
1677aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            const float vr = itr(i);
1678aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            const float vg = itg(i);
1679aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            const float vb = itb(i);
1680aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1681aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            // clear screen
1682aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glColorMask(1,1,1,1);
1683aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glClear(GL_COLOR_BUFFER_BIT);
1684aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glEnable(GL_TEXTURE_2D);
1685aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1686aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            // draw the red plane
1687aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            vverts(vtx, vr);
1688aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glColorMask(1,0,0,1);
1689aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
1690aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1691aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            // draw the green plane
1692aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            vverts(vtx, vg);
1693aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glColorMask(0,1,0,1);
1694aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
1695aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1696aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            // draw the blue plane
1697aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            vverts(vtx, vb);
1698aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glColorMask(0,0,1,1);
1699aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
1700aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1701aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            // draw the white highlight (we use the last vertices)
1702aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glDisable(GL_TEXTURE_2D);
1703aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glColorMask(1,1,1,1);
1704aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glColor4f(vg, vg, vg, 1);
1705aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
1706aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            hw.flip(screenBounds);
1707aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
1708aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1709aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        h_stretch hverts(hw_w, hw_h);
1710aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glDisable(GL_BLEND);
1711aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glDisable(GL_TEXTURE_2D);
1712aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glColorMask(1,1,1,1);
1713aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        for (int i=0 ; i<nbFrames ; i++) {
1714aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            const float v = itg(i);
1715aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            hverts(vtx, v);
1716aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glClear(GL_COLOR_BUFFER_BIT);
1717aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glColor4f(1-v, 1-v, 1-v, 1);
1718aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            glDrawArrays(GL_TRIANGLE_FAN, 0, 4);
1719aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            hw.flip(screenBounds);
1720aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
1721aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1722aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glColorMask(1,1,1,1);
1723aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glEnable(GL_SCISSOR_TEST);
1724aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glDisableClientState(GL_TEXTURE_COORD_ARRAY);
1725aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        result = NO_ERROR;
1726aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    } else {
1727aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        // release FBO resources
1728aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
1729aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        result = BAD_VALUE;
1730aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
1731aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1732aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glDeleteFramebuffersOES(1, &name);
1733aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    glDeleteTextures(1, &tname);
1734aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1735aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (result == NO_ERROR) {
1736aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        DisplayHardware& hw(graphicPlane(dpy).editDisplayHardware());
1737aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        hw.setCanDraw(false);
1738aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
1739aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1740aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    return result;
1741aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
1742aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1743aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopianstatus_t SurfaceFlinger::turnElectronBeamOff(int32_t mode)
1744aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian{
1745aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
1746aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        return INVALID_OPERATION;
1747aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1748aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    class MessageTurnElectronBeamOff : public MessageBase {
1749aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        SurfaceFlinger* flinger;
1750aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        status_t result;
1751aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    public:
1752aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        MessageTurnElectronBeamOff(SurfaceFlinger* flinger)
1753aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            : flinger(flinger), result(PERMISSION_DENIED) {
1754aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
1755aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        status_t getResult() const {
1756aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            return result;
1757aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
1758aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        virtual bool handler() {
1759aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
1760aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            result = flinger->turnElectronBeamOffImplLocked();
1761aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian            return true;
1762aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        }
1763aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    };
1764aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
1765aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    sp<MessageBase> msg = new MessageTurnElectronBeamOff(this);
1766aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    status_t res = postMessageSync(msg);
1767aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    if (res == NO_ERROR) {
1768aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian        res = static_cast<MessageTurnElectronBeamOff*>( msg.get() )->getResult();
1769aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    }
1770aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    return res;
1771aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
1772aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
17739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
17749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1775597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopianstatus_t SurfaceFlinger::captureScreenImplLocked(DisplayID dpy,
1776597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        sp<IMemoryHeap>* heap,
1777597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        uint32_t* w, uint32_t* h, PixelFormat* f,
1778597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        uint32_t sw, uint32_t sh)
1779597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian{
1780597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    status_t result = PERMISSION_DENIED;
1781597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1782597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    // only one display supported for now
1783597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
1784597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        return BAD_VALUE;
1785597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1786597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
1787597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        return INVALID_OPERATION;
1788597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1789597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    // get screen geometry
1790597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    const DisplayHardware& hw(graphicPlane(dpy).displayHardware());
1791597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    const uint32_t hw_w = hw.getWidth();
1792597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    const uint32_t hw_h = hw.getHeight();
1793597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1794597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    if ((sw > hw_w) || (sh > hw_h))
1795597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        return BAD_VALUE;
1796597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1797597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    sw = (!sw) ? hw_w : sw;
1798597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    sh = (!sh) ? hw_h : sh;
1799597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    const size_t size = sw * sh * 4;
1800597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1801597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    // make sure to clear all GL error flags
1802597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    while ( glGetError() != GL_NO_ERROR ) ;
1803597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1804597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    // create a FBO
1805597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    GLuint name, tname;
1806597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    glGenRenderbuffersOES(1, &tname);
1807597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    glBindRenderbufferOES(GL_RENDERBUFFER_OES, tname);
1808597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    glRenderbufferStorageOES(GL_RENDERBUFFER_OES, GL_RGBA8_OES, sw, sh);
1809597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    glGenFramebuffersOES(1, &name);
1810597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, name);
1811597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES,
1812597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            GL_COLOR_ATTACHMENT0_OES, GL_RENDERBUFFER_OES, tname);
1813597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1814597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    GLenum status = glCheckFramebufferStatusOES(GL_FRAMEBUFFER_OES);
1815597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    if (status == GL_FRAMEBUFFER_COMPLETE_OES) {
1816597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1817597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        // invert everything, b/c glReadPixel() below will invert the FB
1818597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glViewport(0, 0, sw, sh);
1819597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glMatrixMode(GL_PROJECTION);
1820597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glPushMatrix();
1821597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glLoadIdentity();
1822597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glOrthof(0, hw_w, 0, hw_h, 0, 1);
1823597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glMatrixMode(GL_MODELVIEW);
1824597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1825597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        // redraw the screen entirely...
1826597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glClearColor(0,0,0,1);
1827597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glClear(GL_COLOR_BUFFER_BIT);
1828597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
1829597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        const size_t count = layers.size();
1830597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        for (size_t i=0 ; i<count ; ++i) {
1831597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            const sp<LayerBase>& layer(layers[i]);
1832597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            layer->drawForSreenShot();
1833597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        }
1834597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1835597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        // XXX: this is needed on tegra
1836597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glScissor(0, 0, sw, sh);
1837597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1838597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        // check for errors and return screen capture
1839597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        if (glGetError() != GL_NO_ERROR) {
1840597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            // error while rendering
1841597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            result = INVALID_OPERATION;
1842597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        } else {
1843597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            // allocate shared memory large enough to hold the
1844597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            // screen capture
1845597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            sp<MemoryHeapBase> base(
1846597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                    new MemoryHeapBase(size, 0, "screen-capture") );
1847597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            void* const ptr = base->getBase();
1848597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            if (ptr) {
1849597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                // capture the screen with glReadPixels()
1850597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                glReadPixels(0, 0, sw, sh, GL_RGBA, GL_UNSIGNED_BYTE, ptr);
1851597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                if (glGetError() == GL_NO_ERROR) {
1852597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                    *heap = base;
1853597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                    *w = sw;
1854597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                    *h = sh;
1855597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                    *f = PIXEL_FORMAT_RGBA_8888;
1856597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                    result = NO_ERROR;
1857597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                }
1858597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            } else {
1859597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                result = NO_MEMORY;
1860597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            }
1861597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        }
1862597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1863597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glEnable(GL_SCISSOR_TEST);
1864597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glViewport(0, 0, hw_w, hw_h);
1865597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glMatrixMode(GL_PROJECTION);
1866597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glPopMatrix();
1867597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        glMatrixMode(GL_MODELVIEW);
1868597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1869597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1870597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    } else {
1871597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        result = BAD_VALUE;
1872597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    }
1873597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1874597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    // release FBO resources
1875597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
1876597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    glDeleteRenderbuffersOES(1, &tname);
1877597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    glDeleteFramebuffersOES(1, &name);
1878597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian    return result;
1879597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian}
1880597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1881597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian
1882ca5edbeba92b96913291792a4df984e158853b6dMathias Agopianstatus_t SurfaceFlinger::captureScreen(DisplayID dpy,
1883ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        sp<IMemoryHeap>* heap,
1884597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        uint32_t* width, uint32_t* height, PixelFormat* format,
1885597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        uint32_t sw, uint32_t sh)
1886ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian{
1887ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    // only one display supported for now
1888ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
1889ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        return BAD_VALUE;
1890ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
1891ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (!GLExtensions::getInstance().haveFramebufferObject())
1892ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        return INVALID_OPERATION;
1893ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
1894ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    class MessageCaptureScreen : public MessageBase {
1895ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        SurfaceFlinger* flinger;
1896ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        DisplayID dpy;
1897ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        sp<IMemoryHeap>* heap;
1898ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        uint32_t* w;
1899ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        uint32_t* h;
1900ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        PixelFormat* f;
1901597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        uint32_t sw;
1902597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian        uint32_t sh;
1903ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        status_t result;
1904ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    public:
1905ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        MessageCaptureScreen(SurfaceFlinger* flinger, DisplayID dpy,
1906597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                sp<IMemoryHeap>* heap, uint32_t* w, uint32_t* h, PixelFormat* f,
1907597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                uint32_t sw, uint32_t sh)
1908ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            : flinger(flinger), dpy(dpy),
1909597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian              heap(heap), w(w), h(h), f(f), sw(sw), sh(sh), result(PERMISSION_DENIED)
1910ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        {
1911ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
1912ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        status_t getResult() const {
1913ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            return result;
1914ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
1915ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        virtual bool handler() {
1916ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            Mutex::Autolock _l(flinger->mStateLock);
1917ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
1918ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            // if we have secure windows, never allow the screen capture
1919ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            if (flinger->mSecureFrameBuffer)
1920ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian                return true;
1921ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
1922597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            result = flinger->captureScreenImplLocked(dpy,
1923597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian                    heap, w, h, f, sw, sh);
1924ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
1925ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian            return true;
1926ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        }
1927ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    };
1928ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
1929ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    sp<MessageBase> msg = new MessageCaptureScreen(this,
1930597c7f67b5f2491c6098a1de241a3f0fd274688aMathias Agopian            dpy, heap, width, height, format, sw, sh);
1931ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    status_t res = postMessageSync(msg);
1932ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    if (res == NO_ERROR) {
1933ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian        res = static_cast<MessageCaptureScreen*>( msg.get() )->getResult();
1934ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    }
1935ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian    return res;
1936ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian}
1937ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
1938ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian// ---------------------------------------------------------------------------
1939ca5edbeba92b96913291792a4df984e158853b6dMathias Agopian
19407623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<Layer> SurfaceFlinger::getLayer(const sp<ISurface>& sur) const
19419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
19427623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<Layer> result;
19437623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    Mutex::Autolock _l(mStateLock);
19447623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    result = mLayerMap.valueFor( sur->asBinder() ).promote();
19457623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return result;
19467623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
1947d763b5d9197e01662cafa376e1227e53e0463a3cMathias Agopian
19487623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian// ---------------------------------------------------------------------------
1949593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian
19507623da435e45c7c03ef6a00a43675deb6645f070Mathias AgopianClient::Client(const sp<SurfaceFlinger>& flinger)
19517623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    : mFlinger(flinger), mNameGenerator(1)
19527623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
19539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
19549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1955593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias AgopianClient::~Client()
1956593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
1957593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    const size_t count = mLayers.size();
1958593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
1959593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        sp<LayerBaseClient> layer(mLayers.valueAt(i).promote());
1960593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        if (layer != 0) {
1961593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            mFlinger->removeLayer(layer);
1962593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        }
19639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
19649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
19651473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
1966593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t Client::initCheck() const {
19677623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return NO_ERROR;
19689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
19691473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
1970593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianssize_t Client::attachLayer(const sp<LayerBaseClient>& layer)
19719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
19727623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    int32_t name = android_atomic_inc(&mNameGenerator);
1973593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    mLayers.add(name, layer);
1974593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return name;
19759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
19769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
19777623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianvoid Client::detachLayer(const LayerBaseClient* layer)
1978593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian{
1979593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    // we do a linear search here, because this doesn't happen often
1980593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    const size_t count = mLayers.size();
1981593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
1982593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        if (mLayers.valueAt(i) == layer) {
1983593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            mLayers.removeItemsAt(i, 1);
1984593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            break;
1985593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        }
1986593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    }
1987593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian}
19881473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopiansp<LayerBaseClient> Client::getLayerUser(int32_t i) const {
19891473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    sp<LayerBaseClient> lbc;
1990593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    const wp<LayerBaseClient>& layer(mLayers.valueFor(i));
1991593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    if (layer != 0) {
1992593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        lbc = layer.promote();
1993593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian        LOGE_IF(lbc==0, "getLayerUser(name=%d) is dead", int(i));
19941473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    }
19951473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    return lbc;
19969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
19979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1998593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopiansp<IMemoryHeap> Client::getControlBlock() const {
19997623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return 0;
20007623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
20017623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianssize_t Client::getTokenForSurface(const sp<ISurface>& sur) const {
20027623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return -1;
20039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2004593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopiansp<ISurface> Client::createSurface(
20057623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        ISurfaceComposerClient::surface_data_t* params, int pid,
20067623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const String8& name,
20077623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        DisplayID display, uint32_t w, uint32_t h, PixelFormat format,
20089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        uint32_t flags)
20099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
2010593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return mFlinger->createSurface(this, pid, name, params,
2011593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian            display, w, h, format, flags);
20129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2013593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t Client::destroySurface(SurfaceID sid) {
2014593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return mFlinger->removeSurface(this, sid);
20159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
2016593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopianstatus_t Client::setState(int32_t count, const layer_state_t* states) {
2017593c05ce7bcf7b6d94bd8c50da2b818cf05116b1Mathias Agopian    return mFlinger->setClientState(this, count, states);
20189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
20199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
20209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
20217623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
20227623da435e45c7c03ef6a00a43675deb6645f070Mathias AgopianUserClient::UserClient(const sp<SurfaceFlinger>& flinger)
20237623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    : ctrlblk(0), mBitmap(0), mFlinger(flinger)
20247623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
20257623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    const int pgsize = getpagesize();
20267623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    const int cblksize = ((sizeof(SharedClient)+(pgsize-1))&~(pgsize-1));
20277623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
20287623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    mCblkHeap = new MemoryHeapBase(cblksize, 0,
20297623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            "SurfaceFlinger Client control-block");
20307623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
20317623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    ctrlblk = static_cast<SharedClient *>(mCblkHeap->getBase());
20327623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (ctrlblk) { // construct the shared structure in-place.
20337623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        new(ctrlblk) SharedClient;
20347623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
20357623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
20367623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
20377623da435e45c7c03ef6a00a43675deb6645f070Mathias AgopianUserClient::~UserClient()
20387623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
20397623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (ctrlblk) {
20407623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        ctrlblk->~SharedClient();  // destroy our shared-structure.
20417623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
20427623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
20437623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    /*
20447623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * When a UserClient dies, it's unclear what to do exactly.
20457623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * We could go ahead and destroy all surfaces linked to that client
20467623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * however, it wouldn't be fair to the main Client
20477623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * (usually the the window-manager), which might want to re-target
20487623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * the layer to another UserClient.
20497623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * I think the best is to do nothing, or not much; in most cases the
20507623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * WM itself will go ahead and clean things up when it detects a client of
20517623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * his has died.
20527623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * The remaining question is what to display? currently we keep
20537623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     * just keep the current buffer.
20547623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian     */
20557623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
20567623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
20577623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianstatus_t UserClient::initCheck() const {
20587623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return ctrlblk == 0 ? NO_INIT : NO_ERROR;
20597623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
20607623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
20617623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianvoid UserClient::detachLayer(const Layer* layer)
20627623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
20637623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    int32_t name = layer->getToken();
20647623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (name >= 0) {
20655e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        int32_t mask = 1LU<<name;
20665e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        if ((android_atomic_and(~mask, &mBitmap) & mask) == 0) {
20675e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian            LOGW("token %d wasn't marked as used %08x", name, int(mBitmap));
20685e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        }
20697623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    }
20707623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
20717623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
20727623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<IMemoryHeap> UserClient::getControlBlock() const {
20737623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return mCblkHeap;
20747623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
20757623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
20767623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianssize_t UserClient::getTokenForSurface(const sp<ISurface>& sur) const
20777623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian{
20787623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    int32_t name = NAME_NOT_FOUND;
20797623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    sp<Layer> layer(mFlinger->getLayer(sur));
20807623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    if (layer == 0) return name;
20817623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
20825e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian    // if this layer already has a token, just return it
20837623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    name = layer->getToken();
20845e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian    if ((name >= 0) && (layer->getClient() == this))
20855e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian        return name;
20867623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
20877623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    name = 0;
20887623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    do {
20897623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        int32_t mask = 1LU<<name;
20907623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        if ((android_atomic_or(mask, &mBitmap) & mask) == 0) {
20917623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            // we found and locked that name
20925e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian            status_t err = layer->setToken(
20935e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian                    const_cast<UserClient*>(this), ctrlblk, name);
20945e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian            if (err != NO_ERROR) {
20955e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian                // free the name
20965e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian                android_atomic_and(~mask, &mBitmap);
20975e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian                name = err;
20985e14010b1fc066dfcbc0a577d59492687c99667dMathias Agopian            }
20997623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            break;
21007623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        }
21017623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        if (++name > 31)
21027623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian            name = NO_MEMORY;
21037623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    } while(name >= 0);
21047623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
21051debc66521f699bbf0a8eb80cababaef8bc63607Mathias Agopian    //LOGD("getTokenForSurface(%p) => %d (client=%p, bitmap=%08lx)",
21061debc66521f699bbf0a8eb80cababaef8bc63607Mathias Agopian    //        sur->asBinder().get(), name, this, mBitmap);
21077623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return name;
21087623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
21097623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
21107623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopiansp<ISurface> UserClient::createSurface(
21117623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        ISurfaceComposerClient::surface_data_t* params, int pid,
21127623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        const String8& name,
21137623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        DisplayID display, uint32_t w, uint32_t h, PixelFormat format,
21147623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian        uint32_t flags) {
21157623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return 0;
21167623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
21177623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianstatus_t UserClient::destroySurface(SurfaceID sid) {
21187623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return INVALID_OPERATION;
21197623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
21207623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopianstatus_t UserClient::setState(int32_t count, const layer_state_t* states) {
21217623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian    return INVALID_OPERATION;
21227623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian}
21237623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian
21247623da435e45c7c03ef6a00a43675deb6645f070Mathias Agopian// ---------------------------------------------------------------------------
21259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
21269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane::GraphicPlane()
21279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    : mHw(0)
21289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
21299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
21309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
21319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source ProjectGraphicPlane::~GraphicPlane() {
21329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    delete mHw;
21339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
21349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
21359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool GraphicPlane::initialized() const {
21369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return mHw ? true : false;
21379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
21389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
213966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianint GraphicPlane::getWidth() const {
214066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    return mWidth;
21419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
21429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
214366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianint GraphicPlane::getHeight() const {
214466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    return mHeight;
214566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian}
214666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
214766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopianvoid GraphicPlane::setDisplayHardware(DisplayHardware *hw)
214866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian{
214966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mHw = hw;
215066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
215166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    // initialize the display orientation transform.
215266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    // it's a constant that should come from the display driver.
215366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    int displayOrientation = ISurfaceComposer::eOrientationDefault;
215466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    char property[PROPERTY_VALUE_MAX];
215566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    if (property_get("ro.sf.hwrotation", property, NULL) > 0) {
215666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        //displayOrientation
215766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        switch (atoi(property)) {
215866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        case 90:
215966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            displayOrientation = ISurfaceComposer::eOrientation90;
216066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            break;
216166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        case 270:
216266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            displayOrientation = ISurfaceComposer::eOrientation270;
216366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            break;
216466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        }
216566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    }
216666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
216766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float w = hw->getWidth();
216866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float h = hw->getHeight();
216966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    GraphicPlane::orientationToTransfrom(displayOrientation, w, h,
217066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian            &mDisplayTransform);
217166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    if (displayOrientation & ISurfaceComposer::eOrientationSwapMask) {
217266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayWidth = h;
217366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayHeight = w;
217466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    } else {
217566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayWidth = w;
217666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian        mDisplayHeight = h;
217766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    }
217866c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
217966c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    setOrientation(ISurfaceComposer::eOrientationDefault);
21809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
21819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
21829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t GraphicPlane::orientationToTransfrom(
21839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int orientation, int w, int h, Transform* tr)
21848c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian{
21858c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    uint32_t flags = 0;
21869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    switch (orientation) {
21879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientationDefault:
21888c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_0;
21898c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        break;
21909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation90:
21918c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_90;
21929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
21939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation180:
21948c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_180;
21959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
21969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    case ISurfaceComposer::eOrientation270:
21978c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        flags = Transform::ROT_270;
21989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        break;
21999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    default:
22009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return BAD_VALUE;
22019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
22028c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    tr->set(flags, w, h);
22039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
22049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
22059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
22069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t GraphicPlane::setOrientation(int orientation)
22079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{
22089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // If the rotation can be handled in hardware, this is where
22099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // the magic should happen.
221066c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
221166c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const DisplayHardware& hw(displayHardware());
221266c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float w = mDisplayWidth;
221366c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    const float h = mDisplayHeight;
221466c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mWidth = int(w);
221566c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mHeight = int(h);
221666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian
221766c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    Transform orientationTransform;
22188c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    GraphicPlane::orientationToTransfrom(orientation, w, h,
22198c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian            &orientationTransform);
22208c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian    if (orientation & ISurfaceComposer::eOrientationSwapMask) {
22218c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        mWidth = int(h);
22228c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian        mHeight = int(w);
22239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
22248c20ca39c1248bc876ab164b37ae36261c86ad98Mathias Agopian
22253552f53c8370ced8680951f4ac811a126da02b0eMathias Agopian    mOrientation = orientation;
222666c77a5b42517b15c933431e12445b856d804ce5Mathias Agopian    mGlobalTransform = mDisplayTransform * orientationTransform;
22279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return NO_ERROR;
22289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
22299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
22309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst DisplayHardware& GraphicPlane::displayHardware() const {
22319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return *mHw;
22329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
22339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2234aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias AgopianDisplayHardware& GraphicPlane::editDisplayHardware() {
2235aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian    return *mHw;
2236aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian}
2237aab758e87991d1460ca94d4a5f22c0ef34641e2dMathias Agopian
22389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectconst Transform& GraphicPlane::transform() const {
22399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return mGlobalTransform;
22409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
22419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
22421473f46cbc82aa6f0ba744cc896a36923823d55bMathias AgopianEGLDisplay GraphicPlane::getEGLDisplay() const {
22431473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian    return mHw->getEGLDisplay();
22441473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian}
22451473f46cbc82aa6f0ba744cc896a36923823d55bMathias Agopian
22469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// ---------------------------------------------------------------------------
22479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
22489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}; // namespace android
2249