| Index: cc/scheduler/frame_source.cc | 
| diff --git a/cc/scheduler/frame_source.cc b/cc/scheduler/frame_source.cc | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..2a709fea6fd0eb20ab91cd3c06b4e84192f44456 | 
| --- /dev/null | 
| +++ b/cc/scheduler/frame_source.cc | 
| @@ -0,0 +1,365 @@ | 
| +// Copyright 2014 The Chromium Authors. All rights reserved. | 
| +// Use of this source code is governed by a BSD-style license that can be | 
| +// found in the LICENSE file. | 
| + | 
| +#include "cc/scheduler/frame_source.h" | 
| + | 
| +#include "base/auto_reset.h" | 
| +#include "base/debug/trace_event.h" | 
| +#include "base/debug/trace_event_argument.h" | 
| +#include "base/logging.h" | 
| +#include "base/strings/string_number_conversions.h" | 
| +#include "cc/scheduler/delay_based_time_source.h" | 
| +#include "cc/scheduler/scheduler.h" | 
| +#include "ui/gfx/frame_time.h" | 
| + | 
| +namespace cc { | 
| + | 
| +BeginFrameSource::BeginFrameSource() | 
| +    : observer_(NULL), inside_as_value_into_(false) { | 
| +  DCHECK(!observer_); | 
| +  DCHECK_EQ(inside_as_value_into_, false); | 
| +} | 
| + | 
| +void BeginFrameSource::AddObserver(BeginFrameObserver* obs) { | 
| +  DCHECK(!observer_); | 
| +  observer_ = obs; | 
| +} | 
| + | 
| +void BeginFrameSource::RemoveObserver(BeginFrameObserver* obs) { | 
| +  DCHECK_EQ(observer_, obs); | 
| +  observer_ = NULL; | 
| +} | 
| + | 
| +void BeginFrameSource::CallOnBeginFrame(const BeginFrameArgs& args) { | 
| +  if (observer_) { | 
| +    observer_->OnBeginFrame(args); | 
| +  } | 
| +} | 
| + | 
| +// Tracing support | 
| +void BeginFrameSource::AsValueInto(base::debug::TracedValue* dict) const { | 
| +  // As the observer might try to trace the source, prevent an infinte loop | 
| +  // from occuring. | 
| +  if (inside_as_value_into_) { | 
| +    dict->SetString("observer", "<loop detected>"); | 
| +    return; | 
| +  } | 
| + | 
| +  if (observer_) { | 
| +    base::AutoReset<bool> prevent_loops( | 
| +        const_cast<bool*>(&inside_as_value_into_), true); | 
| +    dict->BeginDictionary("observer"); | 
| +    observer_->AsValueInto(dict); | 
| +    dict->EndDictionary(); | 
| +  } else { | 
| +    dict->SetString("observer", "NULL"); | 
| +  } | 
| +} | 
| + | 
| +/*************************************************************************/ | 
| +scoped_ptr<BackToBackBeginFrameSource> BackToBackBeginFrameSource::Create( | 
| +    base::SingleThreadTaskRunner* task_runner) { | 
| +  return make_scoped_ptr(new BackToBackBeginFrameSource(task_runner)); | 
| +} | 
| + | 
| +BackToBackBeginFrameSource::BackToBackBeginFrameSource( | 
| +    base::SingleThreadTaskRunner* task_runner) | 
| +    : BeginFrameSource(), | 
| +      weak_factory_(this), | 
| +      task_runner_(task_runner), | 
| +      needs_begin_frames_(false), | 
| +      send_begin_frame_posted_(false) { | 
| +  DCHECK(task_runner); | 
| +  DCHECK_EQ(needs_begin_frames_, false); | 
| +  DCHECK_EQ(send_begin_frame_posted_, false); | 
| +} | 
| + | 
| +BackToBackBeginFrameSource::~BackToBackBeginFrameSource() { | 
| +} | 
| + | 
| +base::TimeTicks BackToBackBeginFrameSource::Now() { | 
| +  return gfx::FrameTime::Now(); | 
| +} | 
| + | 
| +void BackToBackBeginFrameSource::ScheduleBeginFrame() { | 
| +  if (!needs_begin_frames_) | 
| +    return; | 
| + | 
| +  if (send_begin_frame_posted_) | 
| +    return; | 
| + | 
| +  send_begin_frame_posted_ = true; | 
| +  task_runner_->PostTask(FROM_HERE, | 
| +                         base::Bind(&BackToBackBeginFrameSource::BeginFrame, | 
| +                                    weak_factory_.GetWeakPtr())); | 
| +} | 
| + | 
| +void BackToBackBeginFrameSource::BeginFrame() { | 
| +  send_begin_frame_posted_ = false; | 
| + | 
| +  if (!needs_begin_frames_) { | 
| +    return; | 
| +  } | 
| + | 
| +  base::TimeTicks now = Now(); | 
| +  BeginFrameArgs args = | 
| +      BeginFrameArgs::Create(now, | 
| +                             now + BeginFrameArgs::DefaultInterval(), | 
| +                             BeginFrameArgs::DefaultInterval()); | 
| +  CallOnBeginFrame(args); | 
| +} | 
| + | 
| +// BeginFrameSource ------------------------------------------------------ | 
| +bool BackToBackBeginFrameSource::NeedsBeginFrames() const { | 
| +  return needs_begin_frames_; | 
| +} | 
| + | 
| +void BackToBackBeginFrameSource::SetNeedsBeginFrames(bool needs_begin_frames) { | 
| +  needs_begin_frames_ = needs_begin_frames; | 
| +  ScheduleBeginFrame(); | 
| +} | 
| + | 
| +void BackToBackBeginFrameSource::DidFinishFrame(size_t remaining_frames) { | 
| +  if (remaining_frames == 0) { | 
| +    ScheduleBeginFrame(); | 
| +  } | 
| +} | 
| + | 
| +// Tracing | 
| +void BackToBackBeginFrameSource::AsValueInto( | 
| +    base::debug::TracedValue* dict) const { | 
| +  dict->SetString("type", "BackToBackBeginFrameSource"); | 
| +  BeginFrameSource::AsValueInto(dict); | 
| +  dict->SetBoolean("send_begin_frame_posted_", send_begin_frame_posted_); | 
| +  dict->SetBoolean("needs_begin_frames", needs_begin_frames_); | 
| +} | 
| + | 
| +/*************************************************************************/ | 
| + | 
| +scoped_ptr<SyntheticBeginFrameSource> SyntheticBeginFrameSource::Create( | 
| +    base::SingleThreadTaskRunner* task_runner, | 
| +    base::TimeTicks initial_vsync_timebase, | 
| +    base::TimeDelta initial_vsync_interval) { | 
| +  scoped_refptr<DelayBasedTimeSource> time_source; | 
| +  if (gfx::FrameTime::TimestampsAreHighRes()) { | 
| +    time_source = DelayBasedTimeSourceHighRes::Create(initial_vsync_interval, | 
| +                                                      task_runner); | 
| +  } else { | 
| +    time_source = | 
| +        DelayBasedTimeSource::Create(initial_vsync_interval, task_runner); | 
| +  } | 
| + | 
| +  return make_scoped_ptr(new SyntheticBeginFrameSource(time_source)); | 
| +} | 
| + | 
| +SyntheticBeginFrameSource::SyntheticBeginFrameSource( | 
| +    scoped_refptr<DelayBasedTimeSource> time_source) | 
| +    : BeginFrameSource(), time_source_(time_source) { | 
| +  time_source_->SetActive(false); | 
| +  time_source_->SetClient(this); | 
| +} | 
| + | 
| +SyntheticBeginFrameSource::~SyntheticBeginFrameSource() { | 
| +  if (NeedsBeginFrames()) | 
| +    time_source_->SetActive(false); | 
| +} | 
| + | 
| +void SyntheticBeginFrameSource::OnUpdateVSyncParameters( | 
| +    base::TimeTicks new_vsync_timebase, | 
| +    base::TimeDelta new_vsync_interval) { | 
| +  time_source_->SetTimebaseAndInterval(new_vsync_timebase, new_vsync_interval); | 
| +} | 
| + | 
| +void SyntheticBeginFrameSource::SendBeginFrameFromTick( | 
| +    base::TimeTicks frame_time) { | 
| +  base::TimeTicks deadline = time_source_->NextTickTime(); | 
| +  CallOnBeginFrame( | 
| +      BeginFrameArgs::Create(frame_time, deadline, time_source_->Interval())); | 
| +} | 
| + | 
| +// TimeSourceClient | 
| +void SyntheticBeginFrameSource::OnTimerTick() { | 
| +  SendBeginFrameFromTick(time_source_->LastTickTime()); | 
| +} | 
| + | 
| +// BeginFrameSource | 
| +void SyntheticBeginFrameSource::SetNeedsBeginFrames(bool needs_begin_frames) { | 
| +  if (needs_begin_frames == NeedsBeginFrames()) | 
| +    return; | 
| + | 
| +  base::TimeTicks missed_tick_time = | 
| +      time_source_->SetActive(needs_begin_frames); | 
| +  if (!missed_tick_time.is_null()) { | 
| +    SendBeginFrameFromTick(missed_tick_time); | 
| +  } | 
| +} | 
| + | 
| +bool SyntheticBeginFrameSource::NeedsBeginFrames() const { | 
| +  return time_source_->Active(); | 
| +} | 
| + | 
| +// Tracing | 
| +void SyntheticBeginFrameSource::AsValueInto( | 
| +    base::debug::TracedValue* dict) const { | 
| +  dict->SetString("type", "SyntheticBeginFrameSource"); | 
| +  BeginFrameSource::AsValueInto(dict); | 
| + | 
| +  dict->BeginDictionary("last_frame_args"); | 
| +  time_source_->AsValueInto(dict); | 
| +  dict->EndDictionary(); | 
| +} | 
| + | 
| +/*************************************************************************/ | 
| + | 
| +scoped_ptr<BeginFrameSourceMultiplexer> BeginFrameSourceMultiplexer::Create() { | 
| +  return make_scoped_ptr(new BeginFrameSourceMultiplexer()); | 
| +} | 
| + | 
| +BeginFrameSourceMultiplexer::BeginFrameSourceMultiplexer() | 
| +    : BeginFrameSource(), | 
| +      minimum_interval_(base::TimeDelta()), | 
| +      active_source_(NULL), | 
| +      source_list_() { | 
| +} | 
| + | 
| +BeginFrameSourceMultiplexer::BeginFrameSourceMultiplexer( | 
| +    base::TimeDelta minimum_interval) | 
| +    : BeginFrameSource(), | 
| +      minimum_interval_(minimum_interval), | 
| +      active_source_(NULL), | 
| +      source_list_() { | 
| +} | 
| + | 
| +BeginFrameSourceMultiplexer::~BeginFrameSourceMultiplexer() { | 
| +} | 
| + | 
| +void BeginFrameSourceMultiplexer::SetActiveSource( | 
| +    BeginFrameSource* new_source) { | 
| +  DCHECK(HasSource(new_source) || new_source == NULL); | 
| + | 
| +  bool needs_begin_frames = NeedsBeginFrames(); | 
| +  if (active_source_) { | 
| +    if (needs_begin_frames) | 
| +      SetNeedsBeginFrames(false); | 
| + | 
| +    // Technically we shouldn't need to remove observation, but this prevents | 
| +    // the case where SetNeedsBeginFrames message gets to the source after a | 
| +    // message has already been sent. | 
| +    active_source_->RemoveObserver(this); | 
| +    active_source_ = NULL; | 
| +  } | 
| +  DCHECK(!active_source_); | 
| +  active_source_ = new_source; | 
| + | 
| +  if (active_source_) { | 
| +    active_source_->AddObserver(this); | 
| + | 
| +    if (needs_begin_frames) { | 
| +      SetNeedsBeginFrames(true); | 
| +    } | 
| +  } | 
| +} | 
| + | 
| +const BeginFrameSource* BeginFrameSourceMultiplexer::ActiveSource() { | 
| +  return active_source_; | 
| +} | 
| + | 
| +bool BeginFrameSourceMultiplexer::HasSource(BeginFrameSource* source) { | 
| +  return (source_list_.find(source) != source_list_.end()); | 
| +} | 
| + | 
| +// ThrottledBeginFrameSource | 
| +void BeginFrameSourceMultiplexer::AddSource(BeginFrameSource* new_source) { | 
| +  DCHECK(new_source); | 
| +  DCHECK(!HasSource(new_source)); | 
| + | 
| +  source_list_.insert(new_source); | 
| + | 
| +  // If there is no active source, set the new one as the active one. | 
| +  if (!active_source_) | 
| +    SetActiveSource(new_source); | 
| +} | 
| + | 
| +void BeginFrameSourceMultiplexer::RemoveSource( | 
| +    BeginFrameSource* existing_source) { | 
| +  DCHECK(existing_source); | 
| +  DCHECK(HasSource(existing_source)); | 
| +  DCHECK_NE(existing_source, active_source_); | 
| +  source_list_.erase(existing_source); | 
| +} | 
| + | 
| +void BeginFrameSourceMultiplexer::SetMinimumInterval( | 
| +    base::TimeDelta new_minimum_interval) { | 
| +  DCHECK_GE(new_minimum_interval.ToInternalValue(), 0); | 
| +  minimum_interval_ = new_minimum_interval; | 
| +} | 
| + | 
| +// BeginFrameObserver | 
| +void BeginFrameSourceMultiplexer::OnBeginFrame(const BeginFrameArgs& args) { | 
| +  if (!observer_) | 
| +    return; | 
| + | 
| +  if (observer_->LastBeginFrameArgs().IsValid() && | 
| +      observer_->LastBeginFrameArgs().frame_time + minimum_interval_ > | 
| +          args.frame_time) | 
| +    return; | 
| +  CallOnBeginFrame(args); | 
| +} | 
| + | 
| +const BeginFrameArgs& BeginFrameSourceMultiplexer::LastBeginFrameArgs() const { | 
| +  return observer_->LastBeginFrameArgs(); | 
| +} | 
| + | 
| +// BeginFrameSource | 
| +bool BeginFrameSourceMultiplexer::NeedsBeginFrames() const { | 
| +  if (active_source_) { | 
| +    return active_source_->NeedsBeginFrames(); | 
| +  } else { | 
| +    return false; | 
| +  } | 
| +} | 
| + | 
| +void BeginFrameSourceMultiplexer::SetNeedsBeginFrames(bool needs_begin_frames) { | 
| +  if (active_source_) { | 
| +    active_source_->SetNeedsBeginFrames(needs_begin_frames); | 
| +  } else { | 
| +    DCHECK(!needs_begin_frames); | 
| +  } | 
| +} | 
| + | 
| +void BeginFrameSourceMultiplexer::DidFinishFrame(size_t remaining_frames) { | 
| +  active_source_->DidFinishFrame(remaining_frames); | 
| +} | 
| + | 
| +// Tracing | 
| +void BeginFrameSourceMultiplexer::AsValueInto( | 
| +    base::debug::TracedValue* dict) const { | 
| +  dict->SetString("type", "BeginFrameSourceMultiplexer"); | 
| + | 
| +  dict->SetInteger("minimum_interval_us", minimum_interval_.InMicroseconds()); | 
| +  if (observer_) { | 
| +    dict->BeginDictionary("last_begin_frame_args"); | 
| +    observer_->LastBeginFrameArgs().AsValueInto(dict); | 
| +    dict->EndDictionary(); | 
| +  } | 
| + | 
| +  if (active_source_) { | 
| +    dict->BeginDictionary("active_source"); | 
| +    active_source_->AsValueInto(dict); | 
| +    dict->EndDictionary(); | 
| +  } else { | 
| +    dict->SetString("active_source", "NULL"); | 
| +  } | 
| + | 
| +  for (std::set<BeginFrameSource*>::const_iterator it = source_list_.begin(); | 
| +       it != source_list_.end(); | 
| +       ++it) { | 
| +    dict->BeginDictionary( | 
| +        base::SizeTToString(std::distance(source_list_.begin(), it)).c_str()); | 
| +    (*it)->AsValueInto(dict); | 
| +    dict->EndDictionary(); | 
| +  } | 
| +} | 
| + | 
| +}  // namespace cc | 
|  |