/* * Copyright 2018 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "EGL.h" #include <vector> #define LOG_TAG "Swappy::EGL" #include "Log.h" using namespace std::chrono_literals; namespace swappy { std::unique_ptr<EGL> EGL::create(std::chrono::nanoseconds refreshPeriod) { auto eglPresentationTimeANDROID = reinterpret_cast<eglPresentationTimeANDROID_type>( eglGetProcAddress("eglPresentationTimeANDROID")); if (eglPresentationTimeANDROID == nullptr) { ALOGE("Failed to load eglPresentationTimeANDROID"); return nullptr; } auto eglCreateSyncKHR = reinterpret_cast<eglCreateSyncKHR_type>( eglGetProcAddress("eglCreateSyncKHR")); if (eglCreateSyncKHR == nullptr) { ALOGE("Failed to load eglCreateSyncKHR"); return nullptr; } auto eglDestroySyncKHR = reinterpret_cast<eglDestroySyncKHR_type>( eglGetProcAddress("eglDestroySyncKHR")); if (eglDestroySyncKHR == nullptr) { ALOGE("Failed to load eglDestroySyncKHR"); return nullptr; } auto eglGetSyncAttribKHR = reinterpret_cast<eglGetSyncAttribKHR_type>( eglGetProcAddress("eglGetSyncAttribKHR")); if (eglGetSyncAttribKHR == nullptr) { ALOGE("Failed to load eglGetSyncAttribKHR"); return nullptr; } auto eglGetError = reinterpret_cast<eglGetError_type>( eglGetProcAddress("eglGetError")); if (eglGetError == nullptr) { ALOGE("Failed to load eglGetError"); return nullptr; } auto eglSurfaceAttrib = reinterpret_cast<eglSurfaceAttrib_type>( eglGetProcAddress("eglSurfaceAttrib")); if (eglSurfaceAttrib == nullptr) { ALOGE("Failed to load eglSurfaceAttrib"); return nullptr; } // stats may not be supported on all versions auto eglGetNextFrameIdANDROID = reinterpret_cast<eglGetNextFrameIdANDROID_type>( eglGetProcAddress("eglGetNextFrameIdANDROID")); if (eglGetNextFrameIdANDROID == nullptr) { ALOGI("Failed to load eglGetNextFrameIdANDROID"); } auto eglGetFrameTimestampsANDROID = reinterpret_cast<eglGetFrameTimestampsANDROID_type>( eglGetProcAddress("eglGetFrameTimestampsANDROID")); if (eglGetFrameTimestampsANDROID == nullptr) { ALOGI("Failed to load eglGetFrameTimestampsANDROID"); } auto egl = std::make_unique<EGL>(refreshPeriod, ConstructorTag{}); egl->eglPresentationTimeANDROID = eglPresentationTimeANDROID; egl->eglCreateSyncKHR = eglCreateSyncKHR; egl->eglDestroySyncKHR = eglDestroySyncKHR; egl->eglGetSyncAttribKHR = eglGetSyncAttribKHR; egl->eglGetError = eglGetError; egl->eglSurfaceAttrib = eglSurfaceAttrib; egl->eglGetNextFrameIdANDROID = eglGetNextFrameIdANDROID; egl->eglGetFrameTimestampsANDROID = eglGetFrameTimestampsANDROID; return egl; } void EGL::resetSyncFence(EGLDisplay display) { std::lock_guard<std::mutex> lock(mSyncFenceMutex); mFenceWaiter.waitForIdle(); if (mSyncFence != EGL_NO_SYNC_KHR) { EGLBoolean result = eglDestroySyncKHR(display, mSyncFence); if (result == EGL_FALSE) { ALOGE("Failed to destroy sync fence"); } } mSyncFence = eglCreateSyncKHR(display, EGL_SYNC_FENCE_KHR, nullptr); // kick of the thread work to wait for the fence and measure its time mFenceWaiter.onFenceCreation(display, mSyncFence); } bool EGL::lastFrameIsComplete(EGLDisplay display) { std::lock_guard<std::mutex> lock(mSyncFenceMutex); // This will be the case on the first frame if (mSyncFence == EGL_NO_SYNC_KHR) { return true; } EGLint status = 0; EGLBoolean result = eglGetSyncAttribKHR(display, mSyncFence, EGL_SYNC_STATUS_KHR, &status); if (result == EGL_FALSE) { ALOGE("Failed to get sync status"); return true; } if (status == EGL_SIGNALED_KHR) { return true; } else if (status == EGL_UNSIGNALED_KHR) { return false; } else { ALOGE("Unexpected sync status: %d", status); return true; } } bool EGL::setPresentationTime(EGLDisplay display, EGLSurface surface, std::chrono::steady_clock::time_point time) { eglPresentationTimeANDROID(display, surface, time.time_since_epoch().count()); return EGL_TRUE; } bool EGL::statsSupported() { return (eglGetNextFrameIdANDROID != nullptr && eglGetFrameTimestampsANDROID != nullptr); } std::pair<bool,EGLuint64KHR> EGL::getNextFrameId(EGLDisplay dpy, EGLSurface surface) { if (eglGetNextFrameIdANDROID == nullptr) { ALOGE("stats are not supported on this platform"); return {false, 0}; } EGLuint64KHR frameId; EGLBoolean result = eglGetNextFrameIdANDROID(dpy, surface, &frameId); if (result == EGL_FALSE) { ALOGE("Failed to get next frame ID"); return {false, 0}; } return {true, frameId}; } std::unique_ptr<EGL::FrameTimestamps> EGL::getFrameTimestamps(EGLDisplay dpy, EGLSurface surface, EGLuint64KHR frameId) { if (eglGetFrameTimestampsANDROID == nullptr) { ALOGE("stats are not supported on this platform"); return nullptr; } const std::vector<EGLint> timestamps = { EGL_REQUESTED_PRESENT_TIME_ANDROID, EGL_RENDERING_COMPLETE_TIME_ANDROID, EGL_COMPOSITION_LATCH_TIME_ANDROID, EGL_DISPLAY_PRESENT_TIME_ANDROID, }; std::vector<EGLnsecsANDROID> values(timestamps.size()); EGLBoolean result = eglGetFrameTimestampsANDROID(dpy, surface, frameId, timestamps.size(), timestamps.data(), values.data()); if (result == EGL_FALSE) { EGLint reason = eglGetError(); if (reason == EGL_BAD_SURFACE) { eglSurfaceAttrib(dpy, surface, EGL_TIMESTAMPS_ANDROID, EGL_TRUE); } else { ALOGE("Failed to get timestamps for frame %llu", (unsigned long long) frameId); } return nullptr; } // try again if we got some pending stats for (auto i : values) { if (i == EGL_TIMESTAMP_PENDING_ANDROID) return nullptr; } std::unique_ptr<EGL::FrameTimestamps> frameTimestamps = std::make_unique<EGL::FrameTimestamps>(); frameTimestamps->requested = values[0]; frameTimestamps->renderingCompleted = values[1]; frameTimestamps->compositionLatched = values[2]; frameTimestamps->presented = values[3]; return frameTimestamps; } EGL::FenceWaiter::FenceWaiter(): mFenceWaiter(&FenceWaiter::threadMain, this) { std::unique_lock<std::mutex> lock(mFenceWaiterLock); eglClientWaitSyncKHR = reinterpret_cast<eglClientWaitSyncKHR_type>( eglGetProcAddress("eglClientWaitSyncKHR")); if (eglClientWaitSyncKHR == nullptr) ALOGE("Failed to load eglClientWaitSyncKHR"); } EGL::FenceWaiter::~FenceWaiter() { { std::lock_guard<std::mutex> lock(mFenceWaiterLock); mFenceWaiterRunning = false; mFenceWaiterCondition.notify_all(); } mFenceWaiter.join(); } void EGL::FenceWaiter::waitForIdle() { std::lock_guard<std::mutex> lock(mFenceWaiterLock); mFenceWaiterCondition.wait(mFenceWaiterLock, [this]() REQUIRES(mFenceWaiterLock) { return !mFenceWaiterPending; }); } void EGL::FenceWaiter::onFenceCreation(EGLDisplay display, EGLSyncKHR syncFence) { std::lock_guard<std::mutex> lock(mFenceWaiterLock); mDisplay = display; mSyncFence = syncFence; mFenceWaiterPending = true; mFenceWaiterCondition.notify_all(); } void EGL::FenceWaiter::threadMain() { std::lock_guard<std::mutex> lock(mFenceWaiterLock); while (mFenceWaiterRunning) { // wait for new fence object mFenceWaiterCondition.wait(mFenceWaiterLock, [this]() REQUIRES(mFenceWaiterLock) { return mFenceWaiterPending || !mFenceWaiterRunning; }); if (!mFenceWaiterRunning) { break; } const auto startTime = std::chrono::steady_clock::now(); EGLBoolean result = eglClientWaitSyncKHR(mDisplay, mSyncFence, 0, EGL_FOREVER_KHR); if (result == EGL_FALSE) { ALOGE("Failed to wait sync"); } mFencePendingTime = std::chrono::steady_clock::now() - startTime; mFenceWaiterPending = false; mFenceWaiterCondition.notify_all(); } } std::chrono::nanoseconds EGL::FenceWaiter::getFencePendingTime() { // return mFencePendingTime without a lock to avoid blocking the main thread // worst case, the time will be of some previous frame return mFencePendingTime.load(); } } // namespace swappy