SurfaceFlingerConsumer.cpp revision c1c05de415854eb7a13a16b7e22a22de8515123a
1bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden/*
2bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden * Copyright (C) 2012 The Android Open Source Project
3bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden *
4bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden * Licensed under the Apache License, Version 2.0 (the "License");
5bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden * you may not use this file except in compliance with the License.
6bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden * You may obtain a copy of the License at
7bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden *
8bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden *      http://www.apache.org/licenses/LICENSE-2.0
9bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden *
10bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden * Unless required by applicable law or agreed to in writing, software
11bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden * distributed under the License is distributed on an "AS IS" BASIS,
12bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden * See the License for the specific language governing permissions and
14bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden * limitations under the License.
15bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden */
16bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
17bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden#define ATRACE_TAG ATRACE_TAG_GRAPHICS
181df8c345854155cbbcb9f80de9d12d66ea70ac08Jamie Gennis//#define LOG_NDEBUG 0
19bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
20bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden#include "SurfaceFlingerConsumer.h"
21bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
22ca08833d5ea99130797e10ad68a651b50e99da74Mathias Agopian#include <private/gui/SyncFeatures.h>
23ca08833d5ea99130797e10ad68a651b50e99da74Mathias Agopian
24bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden#include <utils/Trace.h>
25bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden#include <utils/Errors.h>
26bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
27bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFaddennamespace android {
28bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
29bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden// ---------------------------------------------------------------------------
30bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
31bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFaddenstatus_t SurfaceFlingerConsumer::updateTexImage(BufferRejecter* rejecter)
32bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden{
33bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    ATRACE_CALL();
34bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    ALOGV("updateTexImage");
35bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    Mutex::Autolock lock(mMutex);
36bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
37bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    if (mAbandoned) {
382adaf04fab35cf47c824d74d901b54094e01ccd3Andy McFadden        ALOGE("updateTexImage: GLConsumer is abandoned!");
39bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden        return NO_INIT;
40bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    }
41bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
42bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    // Make sure the EGL state is the same as in previous calls.
43bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    status_t err = checkAndUpdateEglStateLocked();
44bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    if (err != NO_ERROR) {
45bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden        return err;
46bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    }
47bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
48bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    BufferQueue::BufferItem item;
49bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
50bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    // Acquire the next buffer.
51bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    // In asynchronous mode the list is guaranteed to be one buffer
52bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    // deep, while in synchronous mode we use the oldest buffer.
531585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    err = acquireBufferLocked(&item, computeExpectedPresent());
54bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    if (err != NO_ERROR) {
55bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden        if (err == BufferQueue::NO_BUFFER_AVAILABLE) {
56bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden            err = NO_ERROR;
571585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden        } else if (err == BufferQueue::PRESENT_LATER) {
581585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden            // return the error, without logging
59bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden        } else {
60bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden            ALOGE("updateTexImage: acquire failed: %s (%d)",
61bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden                strerror(-err), err);
62bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden        }
63bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden        return err;
64bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    }
65bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
66bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
67bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    // We call the rejecter here, in case the caller has a reason to
68bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    // not accept this buffer.  This is used by SurfaceFlinger to
69bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    // reject buffers which have the wrong size
70bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    int buf = item.mBuf;
71bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    if (rejecter && rejecter->reject(mSlots[buf].mGraphicBuffer, item)) {
7298d3d6ec1250248574f2e8d72e5c934fecbcd734Lajos Molnar        releaseBufferLocked(buf, mSlots[buf].mGraphicBuffer, EGL_NO_SYNC_KHR);
73bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden        return NO_ERROR;
74bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    }
75bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
76bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    // Release the previous buffer.
77ad678e18b66f495efa78dc3b9ab99b579945c9e2Mathias Agopian    err = updateAndReleaseLocked(item);
78bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    if (err != NO_ERROR) {
79bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden        return err;
80bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden    }
81bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
82ca08833d5ea99130797e10ad68a651b50e99da74Mathias Agopian    if (!SyncFeatures::getInstance().useNativeFenceSync()) {
8397eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden        // Bind the new buffer to the GL texture.
8497eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden        //
8597eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden        // Older devices require the "implicit" synchronization provided
8697eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden        // by glEGLImageTargetTexture2DOES, which this method calls.  Newer
8797eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden        // devices will either call this in Layer::onDraw, or (if it's not
8897eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden        // a GL-composited layer) not at all.
8997eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden        err = bindTextureImageLocked();
9097eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden    }
9197eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden
9297eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden    return err;
9397eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden}
9497eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden
9597eba8904c2f221c42a9473407223a4c3a213f75Andy McFaddenstatus_t SurfaceFlingerConsumer::bindTextureImage()
9697eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden{
9797eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden    Mutex::Autolock lock(mMutex);
9897eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden
9997eba8904c2f221c42a9473407223a4c3a213f75Andy McFadden    return bindTextureImageLocked();
100bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden}
101bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
102c1c05de415854eb7a13a16b7e22a22de8515123aMathias Agopianstatus_t SurfaceFlingerConsumer::acquireBufferLocked(
103c1c05de415854eb7a13a16b7e22a22de8515123aMathias Agopian        BufferQueue::BufferItem *item, nsecs_t presentWhen) {
104c1c05de415854eb7a13a16b7e22a22de8515123aMathias Agopian    status_t result = GLConsumer::acquireBufferLocked(item, presentWhen);
105c1c05de415854eb7a13a16b7e22a22de8515123aMathias Agopian    if (result == NO_ERROR) {
106c1c05de415854eb7a13a16b7e22a22de8515123aMathias Agopian        mTransformToDisplayInverse = item->mTransformToDisplayInverse;
107c1c05de415854eb7a13a16b7e22a22de8515123aMathias Agopian    }
108c1c05de415854eb7a13a16b7e22a22de8515123aMathias Agopian    return result;
109c1c05de415854eb7a13a16b7e22a22de8515123aMathias Agopian}
110c1c05de415854eb7a13a16b7e22a22de8515123aMathias Agopian
111c1c05de415854eb7a13a16b7e22a22de8515123aMathias Agopianbool SurfaceFlingerConsumer::getTransformToDisplayInverse() const {
112c1c05de415854eb7a13a16b7e22a22de8515123aMathias Agopian    return mTransformToDisplayInverse;
113c1c05de415854eb7a13a16b7e22a22de8515123aMathias Agopian}
114c1c05de415854eb7a13a16b7e22a22de8515123aMathias Agopian
1151585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden// We need to determine the time when a buffer acquired now will be
1161585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden// displayed.  This can be calculated:
1171585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden//   time when previous buffer's actual-present fence was signaled
1181585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden//    + current display refresh rate * HWC latency
1191585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden//    + a little extra padding
1201585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden//
1211585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden// Buffer producers are expected to set their desired presentation time
1221585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden// based on choreographer time stamps, which (coming from vsync events)
1231585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden// will be slightly later then the actual-present timing.  If we get a
1241585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden// desired-present time that is unintentionally a hair after the next
1251585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden// vsync, we'll hold the frame when we really want to display it.  We
1261585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden// want to use an expected-presentation time that is slightly late to
1271585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden// avoid this sort of edge case.
1281585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFaddennsecs_t SurfaceFlingerConsumer::computeExpectedPresent()
1291585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden{
1301585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // Don't yet have an easy way to get actual buffer flip time for
1311585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // the specific display, so use the current time.  This is typically
1321585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // 1.3ms past the vsync event time.
1331585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    const nsecs_t prevVsync = systemTime(CLOCK_MONOTONIC);
1341585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden
1351585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // Given a SurfaceFlinger reference, and information about what display
1361585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // we're destined for, we could query the HWC for the refresh rate.  This
1371585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // could change over time, e.g. we could switch to 24fps for a movie.
1381585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // For now, assume 60fps.
1391585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    //const nsecs_t vsyncPeriod =
1401585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    //        getHwComposer().getRefreshPeriod(HWC_DISPLAY_PRIMARY);
1411585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    const nsecs_t vsyncPeriod = 16700000;
1421585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden
1431585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // The HWC doesn't currently have a way to report additional latency.
1441585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // Assume that whatever we submit now will appear on the next flip,
1451585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // i.e. 1 frame of latency w.r.t. the previous flip.
1461585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    const uint32_t hwcLatency = 1;
1471585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden
1481585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // A little extra padding to compensate for slack between actual vsync
1491585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // time and vsync event receipt.  Currently not needed since we're
1501585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // using "now" instead of a vsync time.
1511585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    const nsecs_t extraPadding = 0;
1521585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden
1531585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    // Total it up.
1541585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden    return prevVsync + hwcLatency * vsyncPeriod + extraPadding;
1551585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden}
1561585c4d9fbbba3ba70ae625923b85cd02cb8a0fdAndy McFadden
157bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden// ---------------------------------------------------------------------------
158bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden}; // namespace android
159bf974abe92f7495529916fe0f483f3b56e7c30e3Andy McFadden
160