blob: 3b8a576c4ed9fa8363aeea15a2bdef772355234e [file] [log] [blame]
// 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.
#ifndef CC_SCHEDULER_BEGIN_FRAME_SOURCE_H_
#define CC_SCHEDULER_BEGIN_FRAME_SOURCE_H_
#include <set>
#include <string>
#include "base/debug/trace_event.h"
#include "base/logging.h"
#include "cc/output/begin_frame_args.h"
#include "cc/output/vsync_parameter_observer.h"
#include "cc/scheduler/delay_based_time_source.h"
namespace cc {
// (Pure) Interface for observing BeginFrame messages from BeginFrameSource
// objects.
class CC_EXPORT BeginFrameObserver {
public:
virtual ~BeginFrameObserver() {}
// The |args| given to OnBeginFrame is guaranteed to have
// |args|.IsValid()==true and have |args|.frame_time
// field be strictly greater than the previous call.
//
// Side effects: This function can (and most of the time *will*) change the
// return value of the LastUsedBeginFrameArgs method. See the documentation
// on that method for more information.
virtual void OnBeginFrame(const BeginFrameArgs& args) = 0;
// Returns the last BeginFrameArgs used by the observer. This method's return
// value is affected by the OnBeginFrame method!
//
// - Before the first call of OnBeginFrame, this method should return a
// BeginFrameArgs on which IsValid() returns false.
//
// - If the |args| passed to OnBeginFrame is (or *will be*) used, then
// LastUsedBeginFrameArgs return value should become the |args| given to
// OnBeginFrame.
//
// - If the |args| passed to OnBeginFrame is dropped, then
// LastUsedBeginFrameArgs return value should *not* change.
//
// These requirements are designed to allow chaining and nesting of
// BeginFrameObservers which filter the incoming BeginFrame messages while
// preventing "double dropping" and other bad side effects.
virtual const BeginFrameArgs LastUsedBeginFrameArgs() const = 0;
// Tracing support
virtual void AsValueInto(base::debug::TracedValue* dict) const = 0;
};
// Simple mix in which implements a BeginFrameObserver which checks the
// incoming values meet the BeginFrameObserver requirements and implements the
// required LastUsedBeginFrameArgs behaviour.
//
// Users of this mix in should;
// - Implement the OnBeginFrameMixInDelegate function.
// - Recommended (but not required) to call
// BeginFrameObserverMixIn::OnValueInto in their overridden OnValueInto
// function.
class CC_EXPORT BeginFrameObserverMixIn : public BeginFrameObserver {
public:
BeginFrameObserverMixIn();
// BeginFrameObserver
// Traces |args| and DCHECK |args| satisfies pre-conditions then calls
// OnBeginFrameMixInDelegate and updates the last_begin_frame_args_ value on
// true.
virtual void OnBeginFrame(const BeginFrameArgs& args) override;
virtual const BeginFrameArgs LastUsedBeginFrameArgs() const override;
// Outputs last_begin_frame_args_
virtual void AsValueInto(base::debug::TracedValue* dict) const override;
protected:
// Subclasses should override this method!
// Return true if the given argument is (or will be) used.
virtual bool OnBeginFrameMixInDelegate(const BeginFrameArgs& args) = 0;
BeginFrameArgs last_begin_frame_args_;
int64_t dropped_begin_frame_args_;
};
// Interface for a class which produces BeginFrame calls to a
// BeginFrameObserver.
//
// BeginFrame calls *normally* occur just after a vsync interrupt when input
// processing has been finished and provide information about the time values
// of the vsync times. *However*, these values can be heavily modified or even
// plain made up (when no vsync signal is available or vsync throttling is
// turned off). See the BeginFrameObserver for information about the guarantees
// all BeginFrameSources *must* provide.
class CC_EXPORT BeginFrameSource {
public:
virtual ~BeginFrameSource() {}
// SetNeedsBeginFrames is the on/off "switch" for the BeginFrameSource. When
// set to false no more BeginFrame messages should be sent to observer.
virtual bool NeedsBeginFrames() const = 0;
virtual void SetNeedsBeginFrames(bool needs_begin_frames) = 0;
// DidFinishFrame provides back pressure to a frame source about frame
// processing (rather than toggling SetNeedsBeginFrames every frame). It is
// used by systems like the BackToBackFrameSource to make sure only one frame
// is pending at a time.
virtual void DidFinishFrame(size_t remaining_frames) = 0;
// Add/Remove an observer from the source.
// *At the moment* only a single observer can be added to the source, however
// in the future this may be extended to allow multiple observers.
// If making this change, please use base::ObserverList to do so.
virtual void AddObserver(BeginFrameObserver* obs) = 0;
virtual void RemoveObserver(BeginFrameObserver* obs) = 0;
// Tracing support - Recommend (but not required) to call this implementation
// in any override.
virtual void AsValueInto(base::debug::TracedValue* dict) const = 0;
};
// Simple mix in which implements a BeginFrameSource.
// Implementation classes should:
// - Implement the pure virtual (Set)NeedsBeginFrames methods from
// BeginFrameSource.
// - Use the CallOnBeginFrame method to call to the observer(s).
// - Recommended (but not required) to call BeginFrameSourceMixIn::AsValueInto
// in their own AsValueInto implementation.
class CC_EXPORT BeginFrameSourceMixIn : public BeginFrameSource {
public:
virtual ~BeginFrameSourceMixIn() {}
// BeginFrameSource
virtual bool NeedsBeginFrames() const override;
virtual void SetNeedsBeginFrames(bool needs_begin_frames) override;
virtual void DidFinishFrame(size_t remaining_frames) override {}
virtual void AddObserver(BeginFrameObserver* obs) override;
virtual void RemoveObserver(BeginFrameObserver* obs) override;
// Tracing support - Recommend (but not required) to call this implementation
// in any override.
virtual void AsValueInto(base::debug::TracedValue* dict) const override;
protected:
BeginFrameSourceMixIn();
// These methods should be used by subclasses to make the call to the
// observers.
void CallOnBeginFrame(const BeginFrameArgs& args);
// This method should be overridden if you want to change some behaviour on
// needs_begin_frames change.
virtual void OnNeedsBeginFramesChange(bool needs_begin_frames) {}
BeginFrameObserver* observer_;
bool needs_begin_frames_;
private:
bool inside_as_value_into_;
};
// A frame source which calls BeginFrame (at the next possible time) as soon as
// remaining frames reaches zero.
class CC_EXPORT BackToBackBeginFrameSource : public BeginFrameSourceMixIn {
public:
static scoped_ptr<BackToBackBeginFrameSource> Create(
base::SingleThreadTaskRunner* task_runner);
virtual ~BackToBackBeginFrameSource();
// BeginFrameSource
virtual void DidFinishFrame(size_t remaining_frames) override;
// Tracing
virtual void AsValueInto(base::debug::TracedValue* dict) const override;
protected:
explicit BackToBackBeginFrameSource(
base::SingleThreadTaskRunner* task_runner);
virtual base::TimeTicks Now(); // Now overridable for testing
base::WeakPtrFactory<BackToBackBeginFrameSource> weak_factory_;
base::SingleThreadTaskRunner* task_runner_;
bool send_begin_frame_posted_;
// BeginFrameSourceMixIn
virtual void OnNeedsBeginFramesChange(bool needs_begin_frames) override;
void BeginFrame();
};
// A frame source which is locked to an external parameters provides from a
// vsync source and generates BeginFrameArgs for it.
class CC_EXPORT SyntheticBeginFrameSource : public BeginFrameSourceMixIn,
public VSyncParameterObserver,
public TimeSourceClient {
public:
static scoped_ptr<SyntheticBeginFrameSource> Create(
base::SingleThreadTaskRunner* task_runner,
base::TimeTicks initial_vsync_timebase,
base::TimeDelta initial_vsync_interval);
virtual ~SyntheticBeginFrameSource();
// BeginFrameSource
virtual bool NeedsBeginFrames() const override;
// Tracing
virtual void AsValueInto(base::debug::TracedValue* dict) const override;
// VSyncParameterObserver
virtual void OnUpdateVSyncParameters(
base::TimeTicks new_vsync_timebase,
base::TimeDelta new_vsync_interval) override;
// TimeSourceClient
virtual void OnTimerTick() override;
protected:
explicit SyntheticBeginFrameSource(
scoped_refptr<DelayBasedTimeSource> time_source);
BeginFrameArgs CreateBeginFrameArgs(base::TimeTicks frame_time,
BeginFrameArgs::BeginFrameArgsType type);
// BeginFrameSourceMixIn
virtual void OnNeedsBeginFramesChange(bool needs_begin_frames) override;
scoped_refptr<DelayBasedTimeSource> time_source_;
};
// A "virtual" frame source which lets you switch between multiple other frame
// sources while making sure the BeginFrameArgs stays increasing (possibly
// enforcing minimum boundry between BeginFrameArgs messages).
class CC_EXPORT BeginFrameSourceMultiplexer : public BeginFrameSourceMixIn,
public BeginFrameObserver {
public:
static scoped_ptr<BeginFrameSourceMultiplexer> Create();
virtual ~BeginFrameSourceMultiplexer();
void SetMinimumInterval(base::TimeDelta new_minimum_interval);
void AddSource(BeginFrameSource* new_source);
void RemoveSource(BeginFrameSource* existing_source);
void SetActiveSource(BeginFrameSource* new_source);
const BeginFrameSource* ActiveSource();
// BeginFrameObserver
// The mux is an BeginFrameObserver as it needs to proxy the OnBeginFrame
// calls to preserve the monotonicity of the BeginFrameArgs when switching
// sources.
virtual void OnBeginFrame(const BeginFrameArgs& args) override;
virtual const BeginFrameArgs LastUsedBeginFrameArgs() const override;
// BeginFrameSource
virtual bool NeedsBeginFrames() const override;
virtual void SetNeedsBeginFrames(bool needs_begin_frames) override;
virtual void DidFinishFrame(size_t remaining_frames) override;
// Tracing
virtual void AsValueInto(base::debug::TracedValue* dict) const override;
protected:
BeginFrameSourceMultiplexer();
explicit BeginFrameSourceMultiplexer(base::TimeDelta minimum_interval);
bool HasSource(BeginFrameSource* source);
bool IsIncreasing(const BeginFrameArgs& args);
base::TimeDelta minimum_interval_;
BeginFrameSource* active_source_;
std::set<BeginFrameSource*> source_list_;
};
} // namespace cc
#endif // CC_SCHEDULER_BEGIN_FRAME_SOURCE_H_