/* // Copyright (c) 2014 Intel Corporation // // 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 <HwcTrace.h> #include <SoftVsyncObserver.h> #include <IDisplayDevice.h> extern "C" int clock_nanosleep(clockid_t clock_id, int flags, const struct timespec *request, struct timespec *remain); namespace android { namespace intel { SoftVsyncObserver::SoftVsyncObserver(IDisplayDevice& disp) : mDisplayDevice(disp), mDevice(IDisplayDevice::DEVICE_COUNT), mEnabled(false), mRefreshRate(60), // default 60 frames per second mRefreshPeriod(0), mLock(), mCondition(), mNextFakeVSync(0), mExitThread(false), mInitialized(false) { } SoftVsyncObserver::~SoftVsyncObserver() { WARN_IF_NOT_DEINIT(); } bool SoftVsyncObserver::initialize() { if (mInitialized) { WTRACE("object has been initialized"); return true; } mExitThread = false; mEnabled = false; mRefreshRate = 60; mDevice = mDisplayDevice.getType(); mThread = new VsyncEventPollThread(this); if (!mThread.get()) { DEINIT_AND_RETURN_FALSE("failed to create vsync event poll thread."); } mThread->run("SoftVsyncObserver", PRIORITY_URGENT_DISPLAY); mInitialized = true; return true; } void SoftVsyncObserver::deinitialize() { if (mEnabled) { WTRACE("soft vsync is still enabled"); control(false); } mExitThread = true; mCondition.signal(); if (mThread.get()) { mThread->requestExitAndWait(); mThread = NULL; } mInitialized = false; } void SoftVsyncObserver::setRefreshRate(int rate) { if (mEnabled) { WTRACE("too late to set refresh rate"); } else if (rate < 1 || rate > 120) { WTRACE("invalid refresh rate %d", rate); } else { mRefreshRate = rate; } } bool SoftVsyncObserver::control(bool enabled) { if (enabled == mEnabled) { WTRACE("vsync state %d is not changed", enabled); return true; } if (enabled) { mRefreshPeriod = nsecs_t(1e9 / mRefreshRate); mNextFakeVSync = systemTime(CLOCK_MONOTONIC) + mRefreshPeriod; } mEnabled = enabled; mCondition.signal(); return true; } bool SoftVsyncObserver::threadLoop() { { // scope for lock Mutex::Autolock _l(mLock); while (!mEnabled) { mCondition.wait(mLock); if (mExitThread) { ITRACE("exiting thread loop"); return false; } } } const nsecs_t period = mRefreshPeriod; const nsecs_t now = systemTime(CLOCK_MONOTONIC); nsecs_t next_vsync = mNextFakeVSync; nsecs_t sleep = next_vsync - now; if (sleep < 0) { // we missed, find where the next vsync should be sleep = (period - ((now - next_vsync) % period)); next_vsync = now + sleep; } mNextFakeVSync = next_vsync + period; struct timespec spec; spec.tv_sec = next_vsync / 1000000000; spec.tv_nsec = next_vsync % 1000000000; int err; do { err = clock_nanosleep(CLOCK_MONOTONIC, TIMER_ABSTIME, &spec, NULL); } while (err < 0 && errno == EINTR); if (err == 0) { mDisplayDevice.onVsync(next_vsync); } return true; } } // namespace intel } // namesapce android