Home | History | Annotate | Download | only in gl
      1 // Copyright 2013 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "ui/gl/sync_control_vsync_provider.h"
      6 
      7 #include <math.h>
      8 
      9 #include "base/logging.h"
     10 #include "base/time/time.h"
     11 
     12 #if defined(OS_LINUX)
     13 // These constants define a reasonable range for a calculated refresh interval.
     14 // Calculating refreshes out of this range will be considered a fatal error.
     15 const int64 kMinVsyncIntervalUs = base::Time::kMicrosecondsPerSecond / 400;
     16 const int64 kMaxVsyncIntervalUs = base::Time::kMicrosecondsPerSecond / 10;
     17 
     18 // How much noise we'll tolerate between successive computed intervals before
     19 // we think the latest computed interval is invalid (noisey due to
     20 // monitor configuration change, moving a window between monitors, etc.).
     21 const double kRelativeIntervalDifferenceThreshold = 0.05;
     22 #endif
     23 
     24 namespace gfx {
     25 
     26 SyncControlVSyncProvider::SyncControlVSyncProvider()
     27     : VSyncProvider(), last_media_stream_counter_(0), invalid_msc_(false) {
     28   // On platforms where we can't get an accurate reading on the refresh
     29   // rate we fall back to the assumption that we're displaying 60 frames
     30   // per second.
     31   last_good_interval_ = base::TimeDelta::FromSeconds(1) / 60;
     32 }
     33 
     34 SyncControlVSyncProvider::~SyncControlVSyncProvider() {}
     35 
     36 void SyncControlVSyncProvider::GetVSyncParameters(
     37     const UpdateVSyncCallback& callback) {
     38 #if defined(OS_LINUX)
     39   base::TimeTicks timebase;
     40 
     41   // The actual clock used for the system time returned by glXGetSyncValuesOML
     42   // is unspecified. In practice, the clock used is likely to be either
     43   // CLOCK_REALTIME or CLOCK_MONOTONIC, so we compare the returned time to the
     44   // current time according to both clocks, and assume that the returned time
     45   // was produced by the clock whose current time is closest to it, subject
     46   // to the restriction that the returned time must not be in the future
     47   // (since it is the time of a vblank that has already occurred).
     48   int64 system_time;
     49   int64 media_stream_counter;
     50   int64 swap_buffer_counter;
     51   if (!GetSyncValues(&system_time, &media_stream_counter, &swap_buffer_counter))
     52     return;
     53 
     54   // Both Intel and Mali drivers will return TRUE for GetSyncValues
     55   // but a value of 0 for MSC if they cannot access the CRTC data structure
     56   // associated with the surface. crbug.com/231945
     57   bool prev_invalid_msc = invalid_msc_;
     58   invalid_msc_ = (media_stream_counter == 0);
     59   if (invalid_msc_) {
     60     LOG_IF(ERROR, !prev_invalid_msc) << "glXGetSyncValuesOML "
     61         "should not return TRUE with a media stream counter of 0.";
     62     return;
     63   }
     64 
     65   struct timespec real_time;
     66   struct timespec monotonic_time;
     67   clock_gettime(CLOCK_REALTIME, &real_time);
     68   clock_gettime(CLOCK_MONOTONIC, &monotonic_time);
     69 
     70   int64 real_time_in_microseconds =
     71       real_time.tv_sec * base::Time::kMicrosecondsPerSecond +
     72       real_time.tv_nsec / base::Time::kNanosecondsPerMicrosecond;
     73   int64 monotonic_time_in_microseconds =
     74       monotonic_time.tv_sec * base::Time::kMicrosecondsPerSecond +
     75       monotonic_time.tv_nsec / base::Time::kNanosecondsPerMicrosecond;
     76 
     77   // We need the time according to CLOCK_MONOTONIC, so if we've been given
     78   // a time from CLOCK_REALTIME, we need to convert.
     79   bool time_conversion_needed =
     80       llabs(system_time - real_time_in_microseconds) <
     81       llabs(system_time - monotonic_time_in_microseconds);
     82 
     83   if (time_conversion_needed)
     84     system_time += monotonic_time_in_microseconds - real_time_in_microseconds;
     85 
     86   // Return if |system_time| is more than 1 frames in the future.
     87   int64 interval_in_microseconds = last_good_interval_.InMicroseconds();
     88   if (system_time > monotonic_time_in_microseconds + interval_in_microseconds)
     89     return;
     90 
     91   // If |system_time| is slightly in the future, adjust it to the previous
     92   // frame and use the last frame counter to prevent issues in the callback.
     93   if (system_time > monotonic_time_in_microseconds) {
     94     system_time -= interval_in_microseconds;
     95     media_stream_counter--;
     96   }
     97   if (monotonic_time_in_microseconds - system_time >
     98       base::Time::kMicrosecondsPerSecond)
     99     return;
    100 
    101   timebase = base::TimeTicks::FromInternalValue(system_time);
    102 
    103   // Only need the previous calculated interval for our filtering.
    104   while (last_computed_intervals_.size() > 1)
    105     last_computed_intervals_.pop();
    106 
    107   int32 numerator, denominator;
    108   if (GetMscRate(&numerator, &denominator)) {
    109     last_computed_intervals_.push(base::TimeDelta::FromSeconds(denominator) /
    110                                   numerator);
    111   } else if (!last_timebase_.is_null()) {
    112     base::TimeDelta timebase_diff = timebase - last_timebase_;
    113     int64 counter_diff = media_stream_counter - last_media_stream_counter_;
    114     if (counter_diff > 0 && timebase > last_timebase_)
    115       last_computed_intervals_.push(timebase_diff / counter_diff);
    116   }
    117 
    118   if (last_computed_intervals_.size() == 2) {
    119     const base::TimeDelta& old_interval = last_computed_intervals_.front();
    120     const base::TimeDelta& new_interval = last_computed_intervals_.back();
    121 
    122     double relative_change =
    123         fabs(old_interval.InMillisecondsF() - new_interval.InMillisecondsF()) /
    124         new_interval.InMillisecondsF();
    125     if (relative_change < kRelativeIntervalDifferenceThreshold) {
    126       if (new_interval.InMicroseconds() < kMinVsyncIntervalUs ||
    127           new_interval.InMicroseconds() > kMaxVsyncIntervalUs) {
    128 #if defined(USE_ASH)
    129         // On ash platforms (ChromeOS essentially), the real refresh interval is
    130         // queried from XRandR, regardless of the value calculated here, and
    131         // this value is overriden by ui::CompositorVSyncManager.  The log
    132         // should not be fatal in this case. Reconsider all this when XRandR
    133         // support is added to non-ash platforms.
    134         // http://crbug.com/340851
    135         LOG(ERROR)
    136 #else
    137         LOG(FATAL)
    138 #endif  // USE_ASH
    139             << "Calculated bogus refresh interval="
    140             << new_interval.InMicroseconds()
    141             << " us., last_timebase_=" << last_timebase_.ToInternalValue()
    142             << " us., timebase=" << timebase.ToInternalValue()
    143             << " us., last_media_stream_counter_=" << last_media_stream_counter_
    144             << ", media_stream_counter=" << media_stream_counter;
    145       } else {
    146         last_good_interval_ = new_interval;
    147       }
    148     }
    149   }
    150 
    151   last_timebase_ = timebase;
    152   last_media_stream_counter_ = media_stream_counter;
    153   callback.Run(timebase, last_good_interval_);
    154 #endif  // defined(OS_LINUX)
    155 }
    156 
    157 }  // namespace gfx
    158