DisplayDevice.cpp revision 851cfe834295224cd64bdd499872b95b19c4de8c
1edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project/*
2edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project * Copyright (C) 2007 The Android Open Source Project
3edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project *
4edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
5edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project * you may not use this file except in compliance with the License.
6edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project * You may obtain a copy of the License at
7edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project *
8edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
9edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project *
10edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
11edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
12edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project * See the License for the specific language governing permissions and
14edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project * limitations under the License.
15edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project */
16edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
17edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project#include <stdlib.h>
18edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project#include <stdio.h>
19edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project#include <string.h>
20edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project#include <math.h>
21edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
22edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project#include <cutils/properties.h>
23edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
24076b1cc3a9b90aa5b381a1ed268ca0b548444c9bMathias Agopian#include <utils/RefBase.h>
25edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project#include <utils/Log.h>
26edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
27c666cae2d5995097ec49a87e375e2afdd92802b7Mathias Agopian#include <ui/DisplayInfo.h>
28076b1cc3a9b90aa5b381a1ed268ca0b548444c9bMathias Agopian#include <ui/PixelFormat.h>
29edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
30e3c697fb929c856b59fa56a8e05a2a7eba187c3dMathias Agopian#include <gui/Surface.h>
311a4d883dcc1725892bfb5c28dec255a233186524Jamie Gennis
32edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project#include <GLES/gl.h>
33076b1cc3a9b90aa5b381a1ed268ca0b548444c9bMathias Agopian#include <EGL/egl.h>
34edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project#include <EGL/eglext.h>
35edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
36076b1cc3a9b90aa5b381a1ed268ca0b548444c9bMathias Agopian#include <hardware/gralloc.h>
37edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
3899c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall#include "DisplayHardware/DisplaySurface.h"
391b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian#include "DisplayHardware/HWComposer.h"
401b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian
41da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias Agopian#include "clz.h"
420f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias Agopian#include "DisplayDevice.h"
431f7bec634f19c123410a5155c8d282e177c01930Mathias Agopian#include "GLExtensions.h"
44c7d14e247117392fbd44aa454622778a25c076aeMathias Agopian#include "SurfaceFlinger.h"
4513127d8921356dff794250e04208c3ed60b3a3dfMathias Agopian#include "Layer.h"
461f7bec634f19c123410a5155c8d282e177c01930Mathias Agopian
47a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian// ----------------------------------------------------------------------------
48edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Projectusing namespace android;
49a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian// ----------------------------------------------------------------------------
50edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
51edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Projectstatic __attribute__((noinline))
52edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Projectvoid checkGLErrors()
53edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project{
54cbb288bfe89f585bf48371bd31b2d4aafa32f32eMathias Agopian    do {
55cbb288bfe89f585bf48371bd31b2d4aafa32f32eMathias Agopian        // there could be more than one error flag
56cbb288bfe89f585bf48371bd31b2d4aafa32f32eMathias Agopian        GLenum error = glGetError();
57cbb288bfe89f585bf48371bd31b2d4aafa32f32eMathias Agopian        if (error == GL_NO_ERROR)
58cbb288bfe89f585bf48371bd31b2d4aafa32f32eMathias Agopian            break;
59e6f43ddce78d6846af12550ff9193c5c6fe5844bSteve Block        ALOGE("GL error 0x%04x", int(error));
60cbb288bfe89f585bf48371bd31b2d4aafa32f32eMathias Agopian    } while(true);
61edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project}
62edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
63a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian// ----------------------------------------------------------------------------
64a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian
65edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project/*
66edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project * Initialize the display to the specified values.
67edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project *
68edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project */
69edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
700f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias AgopianDisplayDevice::DisplayDevice(
71edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project        const sp<SurfaceFlinger>& flinger,
72dd3cb84cfbe8068790c6233b5829fae9c4a0ee93Jamie Gennis        DisplayType type,
73dd3cb84cfbe8068790c6233b5829fae9c4a0ee93Jamie Gennis        bool isSecure,
74dd3cb84cfbe8068790c6233b5829fae9c4a0ee93Jamie Gennis        const wp<IBinder>& displayToken,
7599c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall        const sp<DisplaySurface>& displaySurface,
76a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian        EGLConfig config)
7792a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian    : mFlinger(flinger),
783ee454a7bef8bd3d1c9cdd9d17108eb80ebadf2aMathias Agopian      mType(type), mHwcDisplayId(-1),
7927e2562868dcd3ad26f9b4677b64ae272941704eChih-Wei Huang      mDisplayToken(displayToken),
8099c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall      mDisplaySurface(displaySurface),
8192a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian      mDisplay(EGL_NO_DISPLAY),
8292a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian      mSurface(EGL_NO_SURFACE),
8392a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian      mContext(EGL_NO_CONTEXT),
8492a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian      mDisplayWidth(), mDisplayHeight(), mFormat(),
8592a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian      mFlags(),
8692a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian      mPageFlipCount(),
87dd3cb84cfbe8068790c6233b5829fae9c4a0ee93Jamie Gennis      mIsSecure(isSecure),
8892a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian      mSecureLayerVisible(false),
8992a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian      mScreenAcquired(false),
9001e29054e672301e4adbbca15b3562a59a20f267Jesse Hall      mLayerStack(NO_LAYER_STACK),
91da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias Agopian      mOrientation()
92edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project{
9399c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall    mNativeWindow = new Surface(mDisplaySurface->getIGraphicBufferProducer());
94a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian    init(config);
95edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project}
96edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
970f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias AgopianDisplayDevice::~DisplayDevice() {
9892a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian    if (mSurface != EGL_NO_SURFACE) {
9992a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian        eglDestroySurface(mDisplay, mSurface);
10092a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian        mSurface = EGL_NO_SURFACE;
10192a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian    }
10292a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian}
10392a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian
10492a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopianbool DisplayDevice::isValid() const {
10592a979a92c34b7de609ce2b1662c73bb8a2728b9Mathias Agopian    return mFlinger != NULL;
106edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project}
107edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
1080f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias Agopianint DisplayDevice::getWidth() const {
109a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian    return mDisplayWidth;
1106163091a2c366f8311fc3ee627dc7deb9681236eMathias Agopian}
1116163091a2c366f8311fc3ee627dc7deb9681236eMathias Agopian
1120f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias Agopianint DisplayDevice::getHeight() const {
113a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian    return mDisplayHeight;
114a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian}
115a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian
1160f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias AgopianPixelFormat DisplayDevice::getFormat() const {
117a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian    return mFormat;
118a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian}
119a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian
1200f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias AgopianEGLSurface DisplayDevice::getEGLSurface() const {
121a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian    return mSurface;
122a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian}
1236163091a2c366f8311fc3ee627dc7deb9681236eMathias Agopian
1240f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias Agopianvoid DisplayDevice::init(EGLConfig config)
125edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project{
126a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian    ANativeWindow* const window = mNativeWindow.get();
127a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian
1286163091a2c366f8311fc3ee627dc7deb9681236eMathias Agopian    int format;
1296163091a2c366f8311fc3ee627dc7deb9681236eMathias Agopian    window->query(window, NATIVE_WINDOW_FORMAT, &format);
130b5dd9c0fee3b3d6d35035dfb992951ebea3e0e4eMathias Agopian
131a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian    /*
132a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian     * Create our display's surface
133b5dd9c0fee3b3d6d35035dfb992951ebea3e0e4eMathias Agopian     */
134385977f6d6c4e76379df384d50695a10cb3757f2Mathias Agopian
1351f7bec634f19c123410a5155c8d282e177c01930Mathias Agopian    EGLSurface surface;
136a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian    EGLint w, h;
137edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project    EGLDisplay display = eglGetDisplay(EGL_DEFAULT_DISPLAY);
138a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian    surface = eglCreateWindowSurface(display, config, window, NULL);
1391b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian    eglQuerySurface(display, surface, EGL_WIDTH,  &mDisplayWidth);
1401b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian    eglQuerySurface(display, surface, EGL_HEIGHT, &mDisplayHeight);
141edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
1421f7bec634f19c123410a5155c8d282e177c01930Mathias Agopian    mDisplay = display;
1431f7bec634f19c123410a5155c8d282e177c01930Mathias Agopian    mSurface = surface;
144a49126087b4494f4ef50873f3a3f6727265f6621Mathias Agopian    mFormat  = format;
1451f7bec634f19c123410a5155c8d282e177c01930Mathias Agopian    mPageFlipCount = 0;
146da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias Agopian    mViewport.makeInvalid();
147da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias Agopian    mFrame.makeInvalid();
1481f7bec634f19c123410a5155c8d282e177c01930Mathias Agopian
1495f20e2d4462da3471f59152b32cd8640fa4a21daMathias Agopian    // external displays are always considered enabled
1503ee454a7bef8bd3d1c9cdd9d17108eb80ebadf2aMathias Agopian    mScreenAcquired = (mType >= DisplayDevice::NUM_DISPLAY_TYPES);
1513ee454a7bef8bd3d1c9cdd9d17108eb80ebadf2aMathias Agopian
1523ee454a7bef8bd3d1c9cdd9d17108eb80ebadf2aMathias Agopian    // get an h/w composer ID
1533ee454a7bef8bd3d1c9cdd9d17108eb80ebadf2aMathias Agopian    mHwcDisplayId = mFlinger->allocateHwcDisplayId(mType);
1545f20e2d4462da3471f59152b32cd8640fa4a21daMathias Agopian
1558dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden    // Name the display.  The name will be replaced shortly if the display
1568dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden    // was created with createDisplay().
1578dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden    switch (mType) {
1588dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden        case DISPLAY_PRIMARY:
1598dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden            mDisplayName = "Built-in Screen";
1608dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden            break;
1618dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden        case DISPLAY_EXTERNAL:
1628dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden            mDisplayName = "HDMI Screen";
1638dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden            break;
1648dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden        default:
1658dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden            mDisplayName = "Virtual Screen";    // e.g. Overlay #n
1668dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden            break;
1678dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden    }
1688dfa92fef9759a881e96ee58d59875d35023aab9Andy McFadden
16998a121aa916eb7acbf11df0e3e31a6fede6fc9ddMathias Agopian    // initialize the display orientation transform.
17000e8c7a88a5b9c4104a71013a713acd3e4d3b77bMathias Agopian    setProjection(DisplayState::eOrientationDefault, mViewport, mFrame);
171a350ff98692b3a50cad5cc93f9f83221242ca86aMathias Agopian}
172a350ff98692b3a50cad5cc93f9f83221242ca86aMathias Agopian
1739e2463e71796964cfaa06bf09a880875ac3537bcMathias Agopianvoid DisplayDevice::setDisplayName(const String8& displayName) {
1749e2463e71796964cfaa06bf09a880875ac3537bcMathias Agopian    if (!displayName.isEmpty()) {
1759e2463e71796964cfaa06bf09a880875ac3537bcMathias Agopian        // never override the name with an empty name
1769e2463e71796964cfaa06bf09a880875ac3537bcMathias Agopian        mDisplayName = displayName;
1779e2463e71796964cfaa06bf09a880875ac3537bcMathias Agopian    }
1789e2463e71796964cfaa06bf09a880875ac3537bcMathias Agopian}
1799e2463e71796964cfaa06bf09a880875ac3537bcMathias Agopian
1800f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias Agopianuint32_t DisplayDevice::getPageFlipCount() const {
181076b1cc3a9b90aa5b381a1ed268ca0b548444c9bMathias Agopian    return mPageFlipCount;
182edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project}
183edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
1840f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias Agopianstatus_t DisplayDevice::compositionComplete() const {
18599c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall    return mDisplaySurface->compositionComplete();
18674faca212e2675aa55a30235c77cb6403471a4b9Mathias Agopian}
187edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
1880f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias Agopianvoid DisplayDevice::flip(const Region& dirty) const
189edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project{
190edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project    checkGLErrors();
191edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
192edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project    EGLDisplay dpy = mDisplay;
193edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project    EGLSurface surface = mSurface;
194edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
19501e29054e672301e4adbbca15b3562a59a20f267Jesse Hall#ifdef EGL_ANDROID_swap_rectangle
196df3ca30bf663cb8eed88ee3f16fb5e9a65dc00fcMathias Agopian    if (mFlags & SWAP_RECTANGLE) {
197b8a5560e1303cb10f5cd482af466fc04d2bdfcabMathias Agopian        const Region newDirty(dirty.intersect(bounds()));
198b8a5560e1303cb10f5cd482af466fc04d2bdfcabMathias Agopian        const Rect b(newDirty.getBounds());
199df3ca30bf663cb8eed88ee3f16fb5e9a65dc00fcMathias Agopian        eglSetSwapRectangleANDROID(dpy, surface,
200df3ca30bf663cb8eed88ee3f16fb5e9a65dc00fcMathias Agopian                b.left, b.top, b.width(), b.height());
20101e29054e672301e4adbbca15b3562a59a20f267Jesse Hall    }
2025e78e0965169790111f01354e78b0f8d34c94840Mathias Agopian#endif
203d870703d5566490cfdfb389d9336b2b8d3c6cc7aMathias Agopian
204076b1cc3a9b90aa5b381a1ed268ca0b548444c9bMathias Agopian    mPageFlipCount++;
205edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project}
206edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
207da27af9832a0170f1fc40ef3f21371c4d30d21b3Mathias Agopianvoid DisplayDevice::swapBuffers(HWComposer& hwc) const {
20899c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall    // We need to call eglSwapBuffers() unless:
20999c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall    // (a) there was no GLES composition this frame, or
21099c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall    // (b) we're using a legacy HWC with no framebuffer target support (in
21199c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall    //     which case HWComposer::commit() handles things).
21299c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall    if (hwc.initCheck() != NO_ERROR ||
21399c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall            (hwc.hasGlesComposition(mHwcDisplayId) &&
21499c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall             hwc.supportsFramebufferTarget())) {
21599c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall        EGLBoolean success = eglSwapBuffers(mDisplay, mSurface);
21699c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall        if (!success) {
21799c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall            EGLint error = eglGetError();
21899c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall            if (error == EGL_CONTEXT_LOST ||
21999c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall                    mType == DisplayDevice::DISPLAY_PRIMARY) {
22099c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall                LOG_ALWAYS_FATAL("eglSwapBuffers(%p, %p) failed with 0x%08x",
22199c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall                        mDisplay, mSurface, error);
22299c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall            } else {
22399c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall                ALOGE("eglSwapBuffers(%p, %p) failed with 0x%08x",
22499c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall                        mDisplay, mSurface, error);
225da27af9832a0170f1fc40ef3f21371c4d30d21b3Mathias Agopian            }
226da27af9832a0170f1fc40ef3f21371c4d30d21b3Mathias Agopian        }
227da27af9832a0170f1fc40ef3f21371c4d30d21b3Mathias Agopian    }
22852e21483fa48baeb4a88372d75e083bca2e92923Mathias Agopian
22999c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall    status_t result = mDisplaySurface->advanceFrame();
23099c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall    if (result != NO_ERROR) {
23199c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall        ALOGE("[%s] failed pushing new frame to HWC: %d",
23299c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall                mDisplayName.string(), result);
23332341381c9493d7953e40f7f79653cfc52868863Mathias Agopian    }
234da27af9832a0170f1fc40ef3f21371c4d30d21b3Mathias Agopian}
235da27af9832a0170f1fc40ef3f21371c4d30d21b3Mathias Agopian
236da27af9832a0170f1fc40ef3f21371c4d30d21b3Mathias Agopianvoid DisplayDevice::onSwapBuffersCompleted(HWComposer& hwc) const {
237da27af9832a0170f1fc40ef3f21371c4d30d21b3Mathias Agopian    if (hwc.initCheck() == NO_ERROR) {
238851cfe834295224cd64bdd499872b95b19c4de8cJesse Hall        mDisplaySurface->onFrameCommitted();
239da27af9832a0170f1fc40ef3f21371c4d30d21b3Mathias Agopian    }
240da27af9832a0170f1fc40ef3f21371c4d30d21b3Mathias Agopian}
241da27af9832a0170f1fc40ef3f21371c4d30d21b3Mathias Agopian
2420f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias Agopianuint32_t DisplayDevice::getFlags() const
243edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project{
244edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project    return mFlags;
245edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project}
246edbf3b6af777b721cd2a1ef461947e51e88241e1The Android Open Source Project
247da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias AgopianEGLBoolean DisplayDevice::makeCurrent(EGLDisplay dpy,
248da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias Agopian        const sp<const DisplayDevice>& hw, EGLContext ctx) {
249da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias Agopian    EGLBoolean result = EGL_TRUE;
25052bbb1ae239c8a4d05543a23fa8c08467d09c3b2Mathias Agopian    EGLSurface sur = eglGetCurrentSurface(EGL_DRAW);
2514297734c1156fd8ede7e9c61b1e439f9e1c18cd9Mathias Agopian    if (sur != hw->mSurface) {
252da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias Agopian        result = eglMakeCurrent(dpy, hw->mSurface, hw->mSurface, ctx);
253da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias Agopian        if (result == EGL_TRUE) {
254bae92d0d605e99a14731add4f11b72413b2835e5Mathias Agopian            setViewportAndProjection(hw);
255da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias Agopian        }
25652bbb1ae239c8a4d05543a23fa8c08467d09c3b2Mathias Agopian    }
257da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias Agopian    return result;
25852bbb1ae239c8a4d05543a23fa8c08467d09c3b2Mathias Agopian}
25952bbb1ae239c8a4d05543a23fa8c08467d09c3b2Mathias Agopian
260bae92d0d605e99a14731add4f11b72413b2835e5Mathias Agopianvoid DisplayDevice::setViewportAndProjection(const sp<const DisplayDevice>& hw) {
261bae92d0d605e99a14731add4f11b72413b2835e5Mathias Agopian    GLsizei w = hw->mDisplayWidth;
262bae92d0d605e99a14731add4f11b72413b2835e5Mathias Agopian    GLsizei h = hw->mDisplayHeight;
263bae92d0d605e99a14731add4f11b72413b2835e5Mathias Agopian    glViewport(0, 0, w, h);
264bae92d0d605e99a14731add4f11b72413b2835e5Mathias Agopian    glMatrixMode(GL_PROJECTION);
265bae92d0d605e99a14731add4f11b72413b2835e5Mathias Agopian    glLoadIdentity();
266bae92d0d605e99a14731add4f11b72413b2835e5Mathias Agopian    // put the origin in the left-bottom corner
267bae92d0d605e99a14731add4f11b72413b2835e5Mathias Agopian    glOrthof(0, w, 0, h, 0, 1); // l=0, r=w ; b=0, t=h
268135e5899f70a67e62baaf6dbec7ba2ce611ca16aMathias Agopian    glMatrixMode(GL_MODELVIEW);
269bae92d0d605e99a14731add4f11b72413b2835e5Mathias Agopian}
270bae92d0d605e99a14731add4f11b72413b2835e5Mathias Agopian
2711b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian// ----------------------------------------------------------------------------
2721b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian
27313127d8921356dff794250e04208c3ed60b3a3dfMathias Agopianvoid DisplayDevice::setVisibleLayersSortedByZ(const Vector< sp<Layer> >& layers) {
2743b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian    mVisibleLayersSortedByZ = layers;
275ef7b9c7eac036cc1230c64821039d18f8cbd2c1cMathias Agopian    mSecureLayerVisible = false;
2763b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian    size_t count = layers.size();
2773b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian    for (size_t i=0 ; i<count ; i++) {
27813127d8921356dff794250e04208c3ed60b3a3dfMathias Agopian        const sp<Layer>& layer(layers[i]);
279f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        if (layer->isSecure()) {
2803b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian            mSecureLayerVisible = true;
2813b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian        }
2823b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian    }
2833b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian}
2843b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian
28513127d8921356dff794250e04208c3ed60b3a3dfMathias Agopianconst Vector< sp<Layer> >& DisplayDevice::getVisibleLayersSortedByZ() const {
2863b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian    return mVisibleLayersSortedByZ;
2873b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian}
2883b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian
2890f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias Agopianbool DisplayDevice::getSecureLayerVisible() const {
2903b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian    return mSecureLayerVisible;
2913b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian}
2923b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian
293cd60f99aba9e750700a967db30b74a29145739cfMathias AgopianRegion DisplayDevice::getDirtyRegion(bool repaintEverything) const {
294cd60f99aba9e750700a967db30b74a29145739cfMathias Agopian    Region dirty;
295cd60f99aba9e750700a967db30b74a29145739cfMathias Agopian    if (repaintEverything) {
296cd60f99aba9e750700a967db30b74a29145739cfMathias Agopian        dirty.set(getBounds());
297cd60f99aba9e750700a967db30b74a29145739cfMathias Agopian    } else {
298da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias Agopian        const Transform& planeTransform(mGlobalTransform);
299cd60f99aba9e750700a967db30b74a29145739cfMathias Agopian        dirty = planeTransform.transform(this->dirtyRegion);
300cd60f99aba9e750700a967db30b74a29145739cfMathias Agopian        dirty.andSelf(getBounds());
301cd60f99aba9e750700a967db30b74a29145739cfMathias Agopian    }
302cd60f99aba9e750700a967db30b74a29145739cfMathias Agopian    return dirty;
303cd60f99aba9e750700a967db30b74a29145739cfMathias Agopian}
304cd60f99aba9e750700a967db30b74a29145739cfMathias Agopian
3053b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian// ----------------------------------------------------------------------------
306d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian
307d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopianbool DisplayDevice::canDraw() const {
308d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian    return mScreenAcquired;
309d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian}
310d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian
311d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopianvoid DisplayDevice::releaseScreen() const {
312d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian    mScreenAcquired = false;
313d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian}
314d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian
315d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopianvoid DisplayDevice::acquireScreen() const {
316d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian    mScreenAcquired = true;
317d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian}
318d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian
319d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopianbool DisplayDevice::isScreenAcquired() const {
320d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian    return mScreenAcquired;
321d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian}
322d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian
323d3ee231eddce0b69ec5e35188dbd0f4a2c3b9ac3Mathias Agopian// ----------------------------------------------------------------------------
3243b1d2b6b2bbfb5df46b1059ec52360974e6f1428Mathias Agopian
32528947d7fbf9f486539322e8e12dd057568e180c2Mathias Agopianvoid DisplayDevice::setLayerStack(uint32_t stack) {
32628947d7fbf9f486539322e8e12dd057568e180c2Mathias Agopian    mLayerStack = stack;
32728947d7fbf9f486539322e8e12dd057568e180c2Mathias Agopian    dirtyRegion.set(bounds());
32828947d7fbf9f486539322e8e12dd057568e180c2Mathias Agopian}
32928947d7fbf9f486539322e8e12dd057568e180c2Mathias Agopian
33028947d7fbf9f486539322e8e12dd057568e180c2Mathias Agopian// ----------------------------------------------------------------------------
33128947d7fbf9f486539322e8e12dd057568e180c2Mathias Agopian
3320f2f5ff75b7b48ceb64270655ee6b62d09bf4d00Mathias Agopianstatus_t DisplayDevice::orientationToTransfrom(
3331b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian        int orientation, int w, int h, Transform* tr)
3341b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian{
3351b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian    uint32_t flags = 0;
3361b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian    switch (orientation) {
3373165cc21cfea781988407b19bd83292b19f05f55Mathias Agopian    case DisplayState::eOrientationDefault:
3381b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian        flags = Transform::ROT_0;
3391b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian        break;
3403165cc21cfea781988407b19bd83292b19f05f55Mathias Agopian    case DisplayState::eOrientation90:
3411b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian        flags = Transform::ROT_90;
3421b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian        break;
3433165cc21cfea781988407b19bd83292b19f05f55Mathias Agopian    case DisplayState::eOrientation180:
3441b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian        flags = Transform::ROT_180;
3451b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian        break;
3463165cc21cfea781988407b19bd83292b19f05f55Mathias Agopian    case DisplayState::eOrientation270:
3471b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian        flags = Transform::ROT_270;
3481b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian        break;
3491b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian    default:
3501b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian        return BAD_VALUE;
3511b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian    }
3521b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian    tr->set(flags, w, h);
3531b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian    return NO_ERROR;
3541b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian}
3551b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian
35600e8c7a88a5b9c4104a71013a713acd3e4d3b77bMathias Agopianvoid DisplayDevice::setProjection(int orientation,
357f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        const Rect& newViewport, const Rect& newFrame) {
358f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    Rect viewport(newViewport);
359f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    Rect frame(newFrame);
360da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias Agopian
361f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    const int w = mDisplayWidth;
362f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    const int h = mDisplayHeight;
363da8d0a5c0cf9d41915d3b106cad4aaec3e767c11Mathias Agopian
364f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    Transform R;
365f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    DisplayDevice::orientationToTransfrom(orientation, w, h, &R);
3661b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian
367f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    if (!frame.isValid()) {
368f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        // the destination frame can be invalid if it has never been set,
369f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        // in that case we assume the whole display frame.
370f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        frame = Rect(w, h);
371f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    }
3726360ec42d414b1351ecb6c5fc4b8afa30d8f4ebfJesse Hall
373f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    if (viewport.isEmpty()) {
374f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        // viewport can be invalid if it has never been set, in that case
375f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        // we assume the whole display size.
376f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        // it's also invalid to have an empty viewport, so we handle that
377f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        // case in the same way.
378f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        viewport = Rect(w, h);
379f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        if (R.getOrientation() & Transform::ROT_90) {
380f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian            // viewport is always specified in the logical orientation
381f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian            // of the display (ie: post-rotation).
382f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian            swap(viewport.right, viewport.bottom);
383766dc49c17dda977bf7b93a5fd8da41c0b737611Mathias Agopian        }
3841b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian    }
385f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian
386f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    dirtyRegion.set(getBounds());
387f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian
388f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    Transform TL, TP, S;
389f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    float src_width  = viewport.width();
390f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    float src_height = viewport.height();
391f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    float dst_width  = frame.width();
392f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    float dst_height = frame.height();
393f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    if (src_width != dst_width || src_height != dst_height) {
394f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        float sx = dst_width  / src_width;
395f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        float sy = dst_height / src_height;
396f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        S.set(sx, 0, 0, sy);
397f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    }
398f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian
399f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    float src_x = viewport.left;
400f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    float src_y = viewport.top;
401f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    float dst_x = frame.left;
402f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    float dst_y = frame.top;
403f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    TL.set(-src_x, -src_y);
404f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    TP.set(dst_x, dst_y);
405f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian
406f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    // The viewport and frame are both in the logical orientation.
407f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    // Apply the logical translation, scale to physical size, apply the
408f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    // physical translation and finally rotate to the physical orientation.
409f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    mGlobalTransform = R * TP * S * TL;
410f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian
411f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    const uint8_t type = mGlobalTransform.getType();
412f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    mNeedsFiltering = (!mGlobalTransform.preserveRects() ||
413f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian            (type >= Transform::SCALE));
414f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian
415f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    mScissor = mGlobalTransform.transform(viewport);
416f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    if (mScissor.isEmpty()) {
417f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian        mScissor.set(getBounds());
418f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    }
419f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian
420f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    mOrientation = orientation;
421f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    mViewport = viewport;
422f5f714aa188884098aaecbe39d0bc61b40311c0dMathias Agopian    mFrame = frame;
4231b03149f3533db04e72e088d3fdd09d0087ca594Mathias Agopian}
4241d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian
4251d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopianvoid DisplayDevice::dump(String8& result, char* buffer, size_t SIZE) const {
4261d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian    const Transform& tr(mGlobalTransform);
4271d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian    snprintf(buffer, SIZE,
4281d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian        "+ DisplayDevice: %s\n"
4291d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian        "   type=%x, layerStack=%u, (%4dx%4d), ANativeWindow=%p, orient=%2d (type=%08x), "
430dd3cb84cfbe8068790c6233b5829fae9c4a0ee93Jamie Gennis        "flips=%u, isSecure=%d, secureVis=%d, acquired=%d, numLayers=%u\n"
431766dc49c17dda977bf7b93a5fd8da41c0b737611Mathias Agopian        "   v:[%d,%d,%d,%d], f:[%d,%d,%d,%d], s:[%d,%d,%d,%d],"
4321d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian        "transform:[[%0.3f,%0.3f,%0.3f][%0.3f,%0.3f,%0.3f][%0.3f,%0.3f,%0.3f]]\n",
433d56eff2f49e64bb71863ffb866af8c3f686b96e9Mathias Agopian        mDisplayName.string(), mType,
4341d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian        mLayerStack, mDisplayWidth, mDisplayHeight, mNativeWindow.get(),
4351d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian        mOrientation, tr.getType(), getPageFlipCount(),
436dd3cb84cfbe8068790c6233b5829fae9c4a0ee93Jamie Gennis        mIsSecure, mSecureLayerVisible, mScreenAcquired, mVisibleLayersSortedByZ.size(),
4371d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian        mViewport.left, mViewport.top, mViewport.right, mViewport.bottom,
4381d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian        mFrame.left, mFrame.top, mFrame.right, mFrame.bottom,
439766dc49c17dda977bf7b93a5fd8da41c0b737611Mathias Agopian        mScissor.left, mScissor.top, mScissor.right, mScissor.bottom,
4401d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian        tr[0][0], tr[1][0], tr[2][0],
4411d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian        tr[0][1], tr[1][1], tr[2][1],
4421d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian        tr[0][2], tr[1][2], tr[2][2]);
4431d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian
4441d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian    result.append(buffer);
4451d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian
44699c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall    String8 surfaceDump;
44799c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall    mDisplaySurface->dump(surfaceDump);
44899c7dbb24994df2f3e175f7b25dd2c9dd92a72f0Jesse Hall    result.append(surfaceDump);
4491d12d8a8e61163b35cf42c51c558a67138014e82Mathias Agopian}
450