|  | /* | 
|  | * Copyright (C) 2014 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 "TimeLord.h" | 
|  | #include <limits> | 
|  | #include "FrameInfo.h" | 
|  |  | 
|  | namespace android { | 
|  | namespace uirenderer { | 
|  | namespace renderthread { | 
|  |  | 
|  | TimeLord::TimeLord() | 
|  | : mFrameIntervalNanos(milliseconds_to_nanoseconds(16)) | 
|  | , mFrameTimeNanos(0) | 
|  | , mFrameIntendedTimeNanos(0) | 
|  | , mFrameVsyncId(UiFrameInfoBuilder::INVALID_VSYNC_ID) | 
|  | , mFrameDeadline(std::numeric_limits<int64_t>::max()) {} | 
|  |  | 
|  | bool TimeLord::vsyncReceived(nsecs_t vsync, nsecs_t intendedVsync, int64_t vsyncId, | 
|  | int64_t frameDeadline, nsecs_t frameInterval) { | 
|  | if (intendedVsync > mFrameIntendedTimeNanos) { | 
|  | mFrameIntendedTimeNanos = intendedVsync; | 
|  |  | 
|  | // The intendedVsync might have been advanced to account for scheduling | 
|  | // jitter. Since we don't have a way to advance the vsync id we just | 
|  | // reset it. | 
|  | mFrameVsyncId = (vsyncId > mFrameVsyncId) ? vsyncId : UiFrameInfoBuilder::INVALID_VSYNC_ID; | 
|  | mFrameDeadline = frameDeadline; | 
|  | if (frameInterval > 0) { | 
|  | mFrameIntervalNanos = frameInterval; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (vsync > mFrameTimeNanos) { | 
|  | mFrameTimeNanos = vsync; | 
|  | return true; | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | nsecs_t TimeLord::computeFrameTimeNanos() { | 
|  | // Logic copied from Choreographer.java | 
|  | nsecs_t now = systemTime(SYSTEM_TIME_MONOTONIC); | 
|  | nsecs_t jitterNanos = now - mFrameTimeNanos; | 
|  | if (jitterNanos >= mFrameIntervalNanos) { | 
|  | nsecs_t lastFrameOffset = jitterNanos % mFrameIntervalNanos; | 
|  | mFrameTimeNanos = now - lastFrameOffset; | 
|  | // mFrameVsyncId is not adjusted here as we still want to send | 
|  | // the vsync id that started this frame to the Surface Composer | 
|  | } | 
|  | return mFrameTimeNanos; | 
|  | } | 
|  |  | 
|  | } /* namespace renderthread */ | 
|  | } /* namespace uirenderer */ | 
|  | } /* namespace android */ |