summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--media/base/filter_host.h34
-rw-r--r--media/base/filter_host_impl.cc63
-rw-r--r--media/base/filter_host_impl.h64
-rw-r--r--media/base/mock_filter_host.h29
-rw-r--r--media/base/pipeline.h163
-rw-r--r--media/base/pipeline_impl.cc293
-rw-r--r--media/base/pipeline_impl.h71
-rw-r--r--media/filters/video_renderer_base.cc2
-rw-r--r--media/player/movie.cc2
9 files changed, 245 insertions, 476 deletions
diff --git a/media/base/filter_host.h b/media/base/filter_host.h
index b911473..4907df2a 100644
--- a/media/base/filter_host.h
+++ b/media/base/filter_host.h
@@ -1,4 +1,4 @@
-// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
+// Copyright (c) 2008-2009 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.
@@ -24,31 +24,6 @@ namespace media {
class FilterHost {
public:
- // The PipelineStatus class allows read-only access to the pipeline state.
- // This is the same object that is used by the pipeline client to examine
- // the state of the running pipeline. The lifetime of the PipelineStatus
- // interface is the same as the lifetime of the FilterHost interface, so
- // it is acceptable for filters to use the returned pointer until their
- // Stop method has been called.
- virtual const PipelineStatus* GetPipelineStatus() const = 0;
-
- // Registers a callback to receive global clock update notifications. The
- // callback will be called repeatedly and filters do not need to re-register
- // after each invocation of the callback. To remove the callback, filters
- // may call this method passing NULL for the callback argument.
- //
- // Callback arguments:
- // base::TimeDelta - the new pipeline time, in microseconds.
- virtual void SetTimeUpdateCallback(Callback1<base::TimeDelta>::Type* cb) = 0;
-
- // Request that the time callback be called at the specified stream
- // time. This will set a timer specific to the filter that will be fired
- // no sooner than the specified time based on the interpolated time. Note
- // that, becuase the callback will be made with the interpolated time, it is
- // possible for time to move "backward" slightly when the audio device updates
- // the pipeline time though the SetTime method.
- virtual void ScheduleTimeUpdateCallback(base::TimeDelta time) = 0;
-
// Filters must call this method to indicate that their initialization is
// complete. They may call this from within their Initialize() method or may
// choose call it after processing some data.
@@ -58,8 +33,11 @@ class FilterHost {
// method with PIPELINE_OK or PIPELINE_STOPPING (used internally by pipeline).
virtual void Error(PipelineError error) = 0;
- // Sets the current time. Any filters that have registered a callback through
- // the SetTimeUpdateCallback method will be notified of the change.
+ // Gets the current time in microseconds.
+ virtual base::TimeDelta GetTime() const = 0;
+
+ // Updates the current time. Other filters should poll to examine the updated
+ // time.
virtual void SetTime(base::TimeDelta time) = 0;
// Get the duration of the media in microseconds. If the duration has not
diff --git a/media/base/filter_host_impl.cc b/media/base/filter_host_impl.cc
index 0cd348a..96e4e93 100644
--- a/media/base/filter_host_impl.cc
+++ b/media/base/filter_host_impl.cc
@@ -1,4 +1,4 @@
-// Copyright (c) 2009 The Chromium Authors. All rights reserved.
+// Copyright (c) 2008-2009 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.
@@ -6,56 +6,6 @@
namespace media {
-void FilterHostImpl::SetTimeUpdateCallback(
- Callback1<base::TimeDelta>::Type* callback) {
- time_update_callback_.reset(callback);
-}
-
-void FilterHostImpl::RunTimeUpdateCallback(base::TimeDelta time) {
- if (time_update_callback_.get()) {
- time_update_callback_->Run(time);
- }
-}
-
-// To understand why this method takes the |caller| parameter, see the comments
-// of the TimeUpdatedTask in the file filter_host_impl.h.
-void FilterHostImpl::RunScheduledTimeUpdateCallback(TimeUpdateTask* caller) {
- time_update_lock_.Acquire();
- if (caller == scheduled_time_update_task_) {
- scheduled_time_update_task_ = NULL;
- }
- time_update_lock_.Release();
- RunTimeUpdateCallback(pipeline()->GetInterpolatedTime());
-}
-
-void FilterHostImpl::ScheduleTimeUpdateCallback(base::TimeDelta time) {
- time_update_lock_.Acquire();
- if (stopped_) {
- time_update_lock_.Release();
- } else {
- DCHECK(time_update_callback_.get());
- if (scheduled_time_update_task_) {
- scheduled_time_update_task_->Cancel();
- }
- scheduled_time_update_task_ = new TimeUpdateTask(this);
- Task* task = scheduled_time_update_task_;
- time_update_lock_.Release();
-
- // Here, we compute the delta from now & adjust it by the playback rate.
- time -= pipeline()->GetInterpolatedTime();
- int delay = static_cast<int>(time.InMilliseconds());
- float rate = pipeline()->GetPlaybackRate();
- if (rate > 0.0f) {
- delay = static_cast<int>(delay / rate);
- }
- if (delay > 0) {
- pipeline_thread_->message_loop()->PostDelayedTask(FROM_HERE, task, delay);
- } else {
- pipeline_thread_->message_loop()->PostTask(FROM_HERE, task);
- }
- }
-}
-
void FilterHostImpl::InitializationComplete() {
pipeline_thread_->InitializationComplete(this);
}
@@ -64,6 +14,10 @@ void FilterHostImpl::Error(PipelineError error) {
pipeline_thread_->Error(error);
}
+base::TimeDelta FilterHostImpl::GetTime() const {
+ return pipeline()->GetTime();
+}
+
void FilterHostImpl::SetTime(base::TimeDelta time) {
pipeline_thread_->SetTime(time);
}
@@ -88,18 +42,11 @@ void FilterHostImpl::SetVideoSize(size_t width, size_t height) {
pipeline()->SetVideoSize(width, height);
}
-const PipelineStatus* FilterHostImpl::GetPipelineStatus() const {
- return pipeline();
-}
-
void FilterHostImpl::Stop() {
if (!stopped_) {
filter_->Stop();
AutoLock auto_lock(time_update_lock_);
stopped_ = true;
- if (scheduled_time_update_task_) {
- scheduled_time_update_task_->Cancel();
- }
}
}
diff --git a/media/base/filter_host_impl.h b/media/base/filter_host_impl.h
index 4ae9260..1fcb0c4 100644
--- a/media/base/filter_host_impl.h
+++ b/media/base/filter_host_impl.h
@@ -1,4 +1,4 @@
-// Copyright (c) 2006-2009 The Chromium Authors. All rights reserved.
+// Copyright (c) 2008-2009 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.
@@ -16,11 +16,9 @@ namespace media {
class FilterHostImpl : public FilterHost {
public:
// FilterHost interface.
- virtual const PipelineStatus* GetPipelineStatus() const;
- virtual void SetTimeUpdateCallback(Callback1<base::TimeDelta>::Type* cb);
- virtual void ScheduleTimeUpdateCallback(base::TimeDelta time);
virtual void InitializationComplete();
virtual void Error(PipelineError error);
+ virtual base::TimeDelta GetTime() const;
virtual void SetTime(base::TimeDelta time);
virtual void SetDuration(base::TimeDelta duration);
virtual void SetBufferedTime(base::TimeDelta buffered_time);
@@ -31,7 +29,7 @@ class FilterHostImpl : public FilterHost {
// These methods are public, but are intended for use by the
// PipelineThread class only.
- // Creates a FilterHostImpl object and populates the filter_type_ member
+ // Creates a FilterHostImpl object and populates the |filter_type_| member
// by calling the Filter class's static filter_type() method. This ensures
// that the GetFilter method can safely cast the filter interface from the
// MediaFilter base class interface to the specific Filter interface.
@@ -40,12 +38,11 @@ class FilterHostImpl : public FilterHost {
: pipeline_thread_(pipeline_thread),
filter_type_(Filter::filter_type()),
filter_(filter),
- scheduled_time_update_task_(NULL),
stopped_(false) {
}
~FilterHostImpl() {}
- // If this FilterHost contains a filter of the specifed Filter class, then
+ // If this FilterHost contains a filter of the specified Filter class, then
// this method returns a pointer to the interface, otherwise it returns NULL
// in |*filter_out|.
template <class Filter>
@@ -54,10 +51,6 @@ class FilterHostImpl : public FilterHost {
reinterpret_cast<Filter*>(media_filter()) : NULL;
}
- // Call the filter if it has registered a time update callback if the filter
- // has registered one though the FilterHost::SetTimeUpdateCallback method.
- void RunTimeUpdateCallback(base::TimeDelta time);
-
// Stops the filter.
void Stop();
@@ -65,49 +58,6 @@ class FilterHostImpl : public FilterHost {
MediaFilter* media_filter() const { return filter_; }
private:
- // This task class is used to schedule a time update callback for the filter.
- // Because a filter may call the ScheduleTimeUpdateCallback method from any
- // thread, and becuase we only want to honor the last call to that method,
- // we always have only one current task.
- // We are required to keep a pointer to the host and a boolean that tells
- // us if the task was canceled because the cancelation could happen on one
- // thread, just as the pipeline thread is calling the Run method on this task.
- // So, we can't just NULL out the host_ to cancel this because it could
- // fault. Once we have called the host, it needs to enter it's critical
- // section and make sure that the task that has Run is, in fact, the last one
- // that was scheduled.
- // In the case where the filter host is Stopping (or being destroyed), it will
- // be guaranteed to happen on the pipeline thread, thus making the setting
- // of the |canceled_| bool thread safe since the task would only execute on
- // the pipeline thread. This means that it could be possible for a task to
- // hold a pointer to a |host_| that has been deleted, but it will never access
- // that pointer because the task was canceled.
- class TimeUpdateTask : public CancelableTask {
- public:
- explicit TimeUpdateTask(FilterHostImpl* host)
- : host_(host),
- canceled_(false) {}
-
- virtual void Run() {
- if (!canceled_) {
- host_->RunScheduledTimeUpdateCallback(this);
- }
- }
-
- virtual void Cancel() {
- canceled_ = true;
- }
-
- private:
- FilterHostImpl* const host_;
- bool canceled_;
-
- DISALLOW_COPY_AND_ASSIGN(TimeUpdateTask);
- };
-
- // Method used by the TimeUpdateTask to call back to the filter.
- void RunScheduledTimeUpdateCallback(TimeUpdateTask* caller);
-
// Useful method for getting the pipeline.
PipelineImpl* pipeline() const { return pipeline_thread_->pipeline(); }
@@ -120,15 +70,9 @@ class FilterHostImpl : public FilterHost {
// A pointer to the filter's MediaFilter base interface.
scoped_refptr<MediaFilter> filter_;
- // An optional callback that will be called when the time is updated.
- scoped_ptr<Callback1<base::TimeDelta>::Type> time_update_callback_;
-
// Critical section used for scheduled time update callbacks.
Lock time_update_lock_;
- // Pointer to the current time update callback task.
- TimeUpdateTask* scheduled_time_update_task_;
-
// Used to avoid calling Filter's Stop() method multiple times.
bool stopped_;
diff --git a/media/base/mock_filter_host.h b/media/base/mock_filter_host.h
index 0c1ae43..d6fde0b 100644
--- a/media/base/mock_filter_host.h
+++ b/media/base/mock_filter_host.h
@@ -40,18 +40,6 @@ class MockFilterHost : public FilterHost {
virtual ~MockFilterHost() {}
- virtual const PipelineStatus* GetPipelineStatus() const {
- return mock_pipeline_;
- }
-
- virtual void SetTimeUpdateCallback(Callback1<base::TimeDelta>::Type* cb) {
- time_update_callback_.reset(cb);
- }
-
- virtual void ScheduleTimeUpdateCallback(base::TimeDelta time) {
- scheduled_callback_time_ = time;
- }
-
virtual void InitializationComplete() {
EXPECT_FALSE(initialized_);
initialized_ = true;
@@ -64,6 +52,10 @@ class MockFilterHost : public FilterHost {
wait_for_error_.Signal();
}
+ virtual base::TimeDelta GetTime() const {
+ return mock_pipeline_->GetTime();
+ }
+
virtual void SetTime(base::TimeDelta time) {
mock_pipeline_->SetTime(time);
}
@@ -88,15 +80,6 @@ class MockFilterHost : public FilterHost {
mock_pipeline_->SetVideoSize(width, height);
}
- // Used by unit tests to manipulate the filter.
- base::TimeDelta GetScheduledCallbackTime() const {
- return scheduled_callback_time_;
- }
-
- Callback1<base::TimeDelta>::Type* GetTimeUpdateCallback() const {
- return time_update_callback_.get();
- }
-
bool IsInitialized() const {
return initialized_;
}
@@ -124,10 +107,6 @@ class MockFilterHost : public FilterHost {
private:
MockPipeline* mock_pipeline_;
scoped_refptr<Filter> filter_;
- scoped_ptr<Callback1<base::TimeDelta>::Type> time_update_callback_;
-
- // Keeps track of the time passed into ScheduleTimeUpdateCallback().
- base::TimeDelta scheduled_callback_time_;
// Tracks if the filter has executed InitializationComplete().
bool initialized_;
diff --git a/media/base/pipeline.h b/media/base/pipeline.h
index 66dab005..d418e4b 100644
--- a/media/base/pipeline.h
+++ b/media/base/pipeline.h
@@ -1,4 +1,4 @@
-// Copyright (c) 2006-2009 The Chromium Authors. All rights reserved.
+// Copyright (c) 2009 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.
@@ -15,7 +15,7 @@
#include "media/base/factory.h"
namespace base {
- class TimeDelta;
+class TimeDelta;
}
namespace media {
@@ -44,88 +44,15 @@ enum PipelineError {
DEMUXER_ERROR_COULD_NOT_CREATE_THREAD,
};
-// Base class for Pipeline class which allows for read-only access to members.
-// Filters are allowed to access the PipelineStatus interface but are not given
-// access to the client Pipeline methods.
-class PipelineStatus {
- public:
- // Returns the current initialization state of the pipeline. Clients can
- // examine this to determine if it is acceptable to call SetRate/SetVolume/
- // Seek after calling Start on the pipeline. Note that this will be
- // set to true prior to a call to the client's |init_complete_callback| if
- // initialization is successful.
- virtual bool IsInitialized() const = 0;
-
- // Get the duration of the media in microseconds. If the duration has not
- // been determined yet, then returns 0.
- virtual base::TimeDelta GetDuration() const = 0;
-
- // Get the approximate amount of playable data buffered so far in micro-
- // seconds.
- virtual base::TimeDelta GetBufferedTime() const = 0;
-
- // Get the total size of the media file. If the size has not yet been
- // determined or can not be determined, this value is 0.
- virtual int64 GetTotalBytes() const = 0;
-
- // Get the total number of bytes that are buffered on the client and ready to
- // be played.
- virtual int64 GetBufferedBytes() const = 0;
-
- // Gets the size of the video output in pixel units. If there is no video
- // or the video has not been rendered yet, the width and height will be 0.
- virtual void GetVideoSize(size_t* width_out, size_t* height_out) const = 0;
-
- // Gets the current volume setting being used by the audio renderer. When
- // the pipeline is started, this value will be 1.0f. Valid values range
- // from 0.0f to 1.0f.
- virtual float GetVolume() const = 0;
-
- // Gets the current playback rate of the pipeline. When the pipeline is
- // started, the playback rate will be 0.0f. A rate of 1.0f indicates
- // that the pipeline is rendering the media at the standard rate. Valid
- // values for playback rate are >= 0.0f.
- virtual float GetPlaybackRate() const = 0;
-
- // Gets the current pipeline time in microseconds. For a pipeline "time"
- // progresses from 0 to the end of the media. This time base is updated
- // by the audio renderer to allow for synchronization of audio and video.
- // Note that a more accurate time may be obtained by calling the
- // GetInterpolatedTime method which estimates the position of the audio
- // device based on a combination of the last time the audio device reported
- // it's position and the current system time.
- virtual base::TimeDelta GetTime() const = 0;
-
- // Gets the current pipeline time in microseconds. For a pipeline "time"
- // progresses from 0 to the end of the media. Becuase this method provides
- // an estimated time, it is possible that subsequent calls to this method will
- // actually progress backwards slightly, so callers must not assume that this
- // method will always return times larger than the last one.
- virtual base::TimeDelta GetInterpolatedTime() const = 0;
-
- // Gets the current error status for the pipeline. If the pipeline is
- // operating correctly, this will return OK.
- virtual PipelineError GetError() const = 0;
-
- // If the |major_mime_type| exists in the pipeline and is being rendered, this
- // method will return true. Types are defined in media/base/media_foramt.h.
- // For example, to determine if a pipeline contains video:
- // bool has_video = pipeline->IsRendered(mime_type::kMajorTypeVideo);
- virtual bool IsRendered(const std::string& major_mime_type) const = 0;
-
- protected:
- virtual ~PipelineStatus() {}
-};
-
// Client-provided callbacks for various pipeline operations.
//
// TODO(scherkus): consider returning a PipelineError instead of a bool, or
// perhaps a client callback interface.
typedef Callback1<bool>::Type PipelineCallback;
-class Pipeline : public PipelineStatus {
+class Pipeline {
public:
- // Build a pipeline to render the given URI using the given filter factory to
+ // Build a pipeline to render the given URL using the given filter factory to
// construct a filter chain. Returns true if successful, false otherwise
// (i.e., pipeline already started). Note that a return value of true
// only indicates that the initialization process has started successfully.
@@ -149,42 +76,90 @@ class Pipeline : public PipelineStatus {
// the pipeline class, not on the thread that originally called the Start()
// method.
virtual bool Start(FilterFactory* filter_factory,
- const std::string& uri,
+ const std::string& url,
PipelineCallback* start_callback) = 0;
// Stops the pipeline and resets to an uninitialized state. This method
// will block the calling thread until the pipeline has been completely
- // torn down and reset to an uninitialized state. After calling Stop, it
- // is acceptable to call Start again since Stop leaves the pipeline
+ // torn down and reset to an uninitialized state. After calling Stop(), it
+ // is acceptable to call Start() again since Stop() leaves the pipeline
// in a state identical to a newly created pipeline.
- // Calling this method is not strictly required because the pipeline
- // destructor will stop it pipeline if it has not been stopped already.
+ //
+ // Stop() must be called before destroying the pipeline.
+ //
+ // TODO(scherkus): it shouldn't be acceptable to call Start() again after you
+ // Stop() a pipeline -- it should be destroyed and replaced with a new
+ // instance.
virtual void Stop() = 0;
+ // Attempt to seek to the position specified by time. |seek_callback| will be
+ // executed when the all filters in the pipeline have processed the seek.
+ // The callback will return true if the seek was carried out, false otherwise
+ // (i.e., streaming media).
+ virtual void Seek(base::TimeDelta time, PipelineCallback* seek_callback) = 0;
+
+ // Returns the current initialization state of the pipeline. Note that this
+ // will be set to true prior to a executing |init_complete_callback| if
+ // initialization is successful.
+ virtual bool IsInitialized() const = 0;
+
+ // If the |major_mime_type| exists in the pipeline and is being rendered, this
+ // method will return true. Types are defined in media/base/media_foramt.h.
+ // For example, to determine if a pipeline contains video:
+ // bool has_video = pipeline->IsRendered(mime_type::kMajorTypeVideo);
+ virtual bool IsRendered(const std::string& major_mime_type) const = 0;
+
+ // Gets the current playback rate of the pipeline. When the pipeline is
+ // started, the playback rate will be 0.0f. A rate of 1.0f indicates
+ // that the pipeline is rendering the media at the standard rate. Valid
+ // values for playback rate are >= 0.0f.
+ virtual float GetPlaybackRate() const = 0;
+
// Attempt to adjust the playback rate. Setting a playback rate of 0.0f pauses
// all rendering of the media. A rate of 1.0f indicates a normal playback
// rate. Values for the playback rate must be greater than or equal to 0.0f.
- // TODO(ralphl): What about maximum rate? Does HTML5 specify a max?
//
- // This method must be called only after initialization has completed.
+ // TODO(scherkus): What about maximum rate? Does HTML5 specify a max?
virtual void SetPlaybackRate(float playback_rate) = 0;
- // Attempt to seek to the position specified by time. |seek_callback| will be
- // executed when the all filters in the pipeline have processed the seek.
- // The callback will return true if the seek was carried out, false otherwise
- // (i.e., streaming media).
- //
- // This method must be called only after initialization has completed.
- virtual void Seek(base::TimeDelta time,
- PipelineCallback* seek_callback) = 0;
+ // Gets the current volume setting being used by the audio renderer. When
+ // the pipeline is started, this value will be 1.0f. Valid values range
+ // from 0.0f to 1.0f.
+ virtual float GetVolume() const = 0;
// Attempt to set the volume of the audio renderer. Valid values for volume
// range from 0.0f (muted) to 1.0f (full volume). This value affects all
// channels proportionately for multi-channel audio streams.
- //
- // This method must be called only after initialization has completed.
virtual void SetVolume(float volume) = 0;
+ // Gets the current pipeline time. For a pipeline "time" progresses from 0 to
+ // the end of the media.
+ virtual base::TimeDelta GetTime() const = 0;
+
+ // Get the approximate amount of playable data buffered so far in micro-
+ // seconds.
+ virtual base::TimeDelta GetBufferedTime() const = 0;
+
+ // Get the duration of the media in microseconds. If the duration has not
+ // been determined yet, then returns 0.
+ virtual base::TimeDelta GetDuration() const = 0;
+
+ // Get the total number of bytes that are buffered on the client and ready to
+ // be played.
+ virtual int64 GetBufferedBytes() const = 0;
+
+ // Get the total size of the media file. If the size has not yet been
+ // determined or can not be determined, this value is 0.
+ virtual int64 GetTotalBytes() const = 0;
+
+ // Gets the size of the video output in pixel units. If there is no video
+ // or the video has not been rendered yet, the width and height will be 0.
+ virtual void GetVideoSize(size_t* width_out, size_t* height_out) const = 0;
+
+ // Gets the current error status for the pipeline. If the pipeline is
+ // operating correctly, this will return OK.
+ virtual PipelineError GetError() const = 0;
+
protected:
virtual ~Pipeline() {}
};
diff --git a/media/base/pipeline_impl.cc b/media/base/pipeline_impl.cc
index 16148d3..e9edc4f 100644
--- a/media/base/pipeline_impl.cc
+++ b/media/base/pipeline_impl.cc
@@ -54,110 +54,6 @@ PipelineImpl::~PipelineImpl() {
Stop();
}
-bool PipelineImpl::IsInitialized() const {
- AutoLock auto_lock(const_cast<Lock&>(lock_));
- return initialized_;
-}
-
-base::TimeDelta PipelineImpl::GetDuration() const {
- AutoLock auto_lock(const_cast<Lock&>(lock_));
- return duration_;
-}
-
-base::TimeDelta PipelineImpl::GetBufferedTime() const {
- AutoLock auto_lock(const_cast<Lock&>(lock_));
- return buffered_time_;
-}
-
-int64 PipelineImpl::GetTotalBytes() const {
- AutoLock auto_lock(const_cast<Lock&>(lock_));
- return total_bytes_;
-}
-
-int64 PipelineImpl::GetBufferedBytes() const {
- AutoLock auto_lock(const_cast<Lock&>(lock_));
- return buffered_bytes_;
-}
-
-void PipelineImpl::GetVideoSize(size_t* width_out, size_t* height_out) const {
- DCHECK(width_out);
- DCHECK(height_out);
- AutoLock auto_lock(const_cast<Lock&>(lock_));
- *width_out = video_width_;
- *height_out = video_height_;
-}
-
-float PipelineImpl::GetVolume() const {
- AutoLock auto_lock(const_cast<Lock&>(lock_));
- return volume_;
-}
-
-float PipelineImpl::GetPlaybackRate() const {
- AutoLock auto_lock(const_cast<Lock&>(lock_));
- return playback_rate_;
-}
-
-base::TimeDelta PipelineImpl::GetTime() const {
- AutoLock auto_lock(const_cast<Lock&>(lock_));
- return time_;
-}
-
-base::TimeDelta PipelineImpl::GetInterpolatedTime() const {
- AutoLock auto_lock(const_cast<Lock&>(lock_));
- base::TimeDelta time = time_;
- if (playback_rate_ > 0.0f) {
- base::TimeDelta delta = base::TimeTicks::Now() - ticks_at_last_set_time_;
- if (playback_rate_ == 1.0f) {
- time += delta;
- } else {
- int64 adjusted_delta = static_cast<int64>(delta.InMicroseconds() *
- playback_rate_);
- time += base::TimeDelta::FromMicroseconds(adjusted_delta);
- }
- }
- return time;
-}
-
-void PipelineImpl::SetTime(base::TimeDelta time) {
- AutoLock auto_lock(lock_);
- time_ = time;
- ticks_at_last_set_time_ = base::TimeTicks::Now();
-}
-
-void PipelineImpl::InternalSetPlaybackRate(float rate) {
- AutoLock auto_lock(lock_);
- if (playback_rate_ == 0.0f && rate > 0.0f) {
- ticks_at_last_set_time_ = base::TimeTicks::Now();
- }
- playback_rate_ = rate;
-}
-
-PipelineError PipelineImpl::GetError() const {
- AutoLock auto_lock(const_cast<Lock&>(lock_));
- return error_;
-}
-
-bool PipelineImpl::IsRendered(const std::string& major_mime_type) const {
- AutoLock auto_lock(const_cast<Lock&>(lock_));
- bool is_rendered = (rendered_mime_types_.find(major_mime_type) !=
- rendered_mime_types_.end());
- return is_rendered;
-}
-
-
-bool PipelineImpl::InternalSetError(PipelineError error) {
- // Don't want callers to set an error of "OK". STOPPING is a special value
- // that should only be used internally by the StopTask() method.
- DCHECK(PIPELINE_OK != error && PIPELINE_STOPPING != error);
- AutoLock auto_lock(lock_);
- bool changed_error = false;
- if (PIPELINE_OK == error_) {
- error_ = error;
- changed_error = true;
- }
- return changed_error;
-}
-
// Creates the PipelineThread and calls it's start method.
bool PipelineImpl::Start(FilterFactory* factory,
const std::string& url,
@@ -192,6 +88,34 @@ void PipelineImpl::Stop() {
ResetState();
}
+void PipelineImpl::Seek(base::TimeDelta time,
+ PipelineCallback* seek_callback) {
+ DCHECK(!IsPipelineThread());
+
+ if (IsPipelineOk()) {
+ pipeline_thread_->Seek(time, seek_callback);
+ } else {
+ NOTREACHED();
+ }
+}
+
+bool PipelineImpl::IsInitialized() const {
+ AutoLock auto_lock(lock_);
+ return initialized_;
+}
+
+bool PipelineImpl::IsRendered(const std::string& major_mime_type) const {
+ AutoLock auto_lock(lock_);
+ bool is_rendered = (rendered_mime_types_.find(major_mime_type) !=
+ rendered_mime_types_.end());
+ return is_rendered;
+}
+
+float PipelineImpl::GetPlaybackRate() const {
+ AutoLock auto_lock(lock_);
+ return playback_rate_;
+}
+
void PipelineImpl::SetPlaybackRate(float rate) {
DCHECK(!IsPipelineThread());
@@ -203,15 +127,9 @@ void PipelineImpl::SetPlaybackRate(float rate) {
}
}
-void PipelineImpl::Seek(base::TimeDelta time,
- PipelineCallback* seek_callback) {
- DCHECK(!IsPipelineThread());
-
- if (IsPipelineOk()) {
- pipeline_thread_->Seek(time, seek_callback);
- } else {
- NOTREACHED();
- }
+float PipelineImpl::GetVolume() const {
+ AutoLock auto_lock(lock_);
+ return volume_;
}
void PipelineImpl::SetVolume(float volume) {
@@ -224,6 +142,44 @@ void PipelineImpl::SetVolume(float volume) {
}
}
+base::TimeDelta PipelineImpl::GetTime() const {
+ AutoLock auto_lock(lock_);
+ return time_;
+}
+
+base::TimeDelta PipelineImpl::GetBufferedTime() const {
+ AutoLock auto_lock(lock_);
+ return buffered_time_;
+}
+
+base::TimeDelta PipelineImpl::GetDuration() const {
+ AutoLock auto_lock(lock_);
+ return duration_;
+}
+
+int64 PipelineImpl::GetBufferedBytes() const {
+ AutoLock auto_lock(lock_);
+ return buffered_bytes_;
+}
+
+int64 PipelineImpl::GetTotalBytes() const {
+ AutoLock auto_lock(lock_);
+ return total_bytes_;
+}
+
+void PipelineImpl::GetVideoSize(size_t* width_out, size_t* height_out) const {
+ CHECK(width_out);
+ CHECK(height_out);
+ AutoLock auto_lock(lock_);
+ *width_out = video_width_;
+ *height_out = video_height_;
+}
+
+PipelineError PipelineImpl::GetError() const {
+ AutoLock auto_lock(lock_);
+ return error_;
+}
+
void PipelineImpl::ResetState() {
AutoLock auto_lock(lock_);
pipeline_thread_ = NULL;
@@ -238,7 +194,6 @@ void PipelineImpl::ResetState() {
playback_rate_ = 0.0f;
error_ = PIPELINE_OK;
time_ = base::TimeDelta();
- ticks_at_last_set_time_ = base::TimeTicks::Now();
rendered_mime_types_.clear();
}
@@ -277,6 +232,29 @@ void PipelineImpl::SetVideoSize(size_t width, size_t height) {
video_height_ = height;
}
+void PipelineImpl::SetTime(base::TimeDelta time) {
+ AutoLock auto_lock(lock_);
+ time_ = time;
+}
+
+void PipelineImpl::InternalSetPlaybackRate(float rate) {
+ AutoLock auto_lock(lock_);
+ playback_rate_ = rate;
+}
+
+bool PipelineImpl::InternalSetError(PipelineError error) {
+ // Don't want callers to set an error of "OK". STOPPING is a special value
+ // that should only be used internally by the StopTask() method.
+ DCHECK(PIPELINE_OK != error && PIPELINE_STOPPING != error);
+ AutoLock auto_lock(lock_);
+ bool changed_error = false;
+ if (PIPELINE_OK == error_) {
+ error_ = error;
+ changed_error = true;
+ }
+ return changed_error;
+}
+
void PipelineImpl::InsertRenderedMimeType(const std::string& major_mime_type) {
AutoLock auto_lock(lock_);
rendered_mime_types_.insert(major_mime_type);
@@ -288,7 +266,6 @@ void PipelineImpl::InsertRenderedMimeType(const std::string& major_mime_type) {
PipelineThread::PipelineThread(PipelineImpl* pipeline)
: pipeline_(pipeline),
thread_("PipelineThread"),
- time_update_callback_scheduled_(false),
state_(kCreated) {
}
@@ -308,7 +285,8 @@ bool PipelineThread::Start(FilterFactory* filter_factory,
filter_factory_ = filter_factory;
url_ = url;
init_callback_.reset(init_complete_callback);
- PostTask(NewRunnableMethod(this, &PipelineThread::StartTask));
+ message_loop()->PostTask(FROM_HERE,
+ NewRunnableMethod(this, &PipelineThread::StartTask));
return true;
}
return false;
@@ -319,7 +297,8 @@ bool PipelineThread::Start(FilterFactory* filter_factory,
// stopped.
void PipelineThread::Stop() {
if (thread_.IsRunning()) {
- PostTask(NewRunnableMethod(this, &PipelineThread::StopTask));
+ message_loop()->PostTask(FROM_HERE,
+ NewRunnableMethod(this, &PipelineThread::StopTask));
thread_.Stop();
}
DCHECK(filter_hosts_.empty());
@@ -327,37 +306,36 @@ void PipelineThread::Stop() {
}
// Called on client's thread.
-void PipelineThread::SetPlaybackRate(float rate) {
- PostTask(NewRunnableMethod(this, &PipelineThread::SetPlaybackRateTask, rate));
+void PipelineThread::Seek(base::TimeDelta time,
+ PipelineCallback* seek_callback) {
+ message_loop()->PostTask(FROM_HERE,
+ NewRunnableMethod(this, &PipelineThread::SeekTask, time, seek_callback));
}
// Called on client's thread.
-void PipelineThread::Seek(base::TimeDelta time,
- PipelineCallback* seek_callback) {
- PostTask(NewRunnableMethod(this, &PipelineThread::SeekTask, time,
- seek_callback));
+void PipelineThread::SetPlaybackRate(float rate) {
+ message_loop()->PostTask(FROM_HERE,
+ NewRunnableMethod(this, &PipelineThread::SetPlaybackRateTask, rate));
}
// Called on client's thread.
void PipelineThread::SetVolume(float volume) {
- PostTask(NewRunnableMethod(this, &PipelineThread::SetVolumeTask, volume));
+ message_loop()->PostTask(FROM_HERE,
+ NewRunnableMethod(this, &PipelineThread::SetVolumeTask, volume));
}
+// Called from any thread.
void PipelineThread::InitializationComplete(FilterHostImpl* host) {
if (IsPipelineOk()) {
// Continue the start task by proceeding to the next stage.
- PostTask(NewRunnableMethod(this, &PipelineThread::StartTask));
+ message_loop()->PostTask(FROM_HERE,
+ NewRunnableMethod(this, &PipelineThread::StartTask));
}
}
-// Called from any thread. Updates the pipeline time and schedules a task to
-// call back to filters that have registered a callback for time updates.
+// Called from any thread. Updates the pipeline time.
void PipelineThread::SetTime(base::TimeDelta time) {
pipeline()->SetTime(time);
- if (!time_update_callback_scheduled_) {
- time_update_callback_scheduled_ = true;
- PostTask(NewRunnableMethod(this, &PipelineThread::SetTimeTask));
- }
}
// Called from any thread. Sets the pipeline |error_| member and schedules a
@@ -368,17 +346,19 @@ void PipelineThread::Error(PipelineError error) {
// If this method returns false, then an error has already happened, so no
// reason to run the StopTask again. It's going to happen.
if (pipeline()->InternalSetError(error)) {
- PostTask(NewRunnableMethod(this, &PipelineThread::StopTask));
+ message_loop()->PostTask(FROM_HERE,
+ NewRunnableMethod(this, &PipelineThread::StopTask));
}
}
-// This is a helper method to post task on message_loop(). This method is only
-// called from this class or from Pipeline.
-void PipelineThread::PostTask(Task* task) {
- message_loop()->PostTask(FROM_HERE, task);
+// Called as a result of destruction of the thread.
+//
+// TODO(scherkus): this can block the client due to synchronous Stop() API call.
+void PipelineThread::WillDestroyCurrentMessageLoop() {
+ STLDeleteElements(&filter_hosts_);
+ STLDeleteElements(&filter_threads_);
}
-
// Main initialization method called on the pipeline thread. This code attempts
// to use the specified filter factory to build a pipeline.
// Initialization step performed in this method depends on current state of this
@@ -603,29 +583,6 @@ void PipelineThread::SetVolumeTask(float volume) {
}
}
-void PipelineThread::SetTimeTask() {
- DCHECK_EQ(PlatformThread::CurrentId(), thread_.thread_id());
-
- time_update_callback_scheduled_ = false;
- for (FilterHostVector::iterator iter = filter_hosts_.begin();
- iter != filter_hosts_.end();
- ++iter) {
- (*iter)->RunTimeUpdateCallback(pipeline_->time_);
- }
-}
-
-template <class Filter>
-void PipelineThread::GetFilter(scoped_refptr<Filter>* filter_out) const {
- DCHECK_EQ(PlatformThread::CurrentId(), thread_.thread_id());
-
- *filter_out = NULL;
- for (FilterHostVector::const_iterator iter = filter_hosts_.begin();
- iter != filter_hosts_.end() && NULL == *filter_out;
- iter++) {
- (*iter)->GetFilter(filter_out);
- }
-}
-
template <class Filter, class Source>
void PipelineThread::CreateFilter(FilterFactory* filter_factory,
Source source,
@@ -723,12 +680,16 @@ bool PipelineThread::CreateRenderer() {
return false;
}
-// Called as a result of destruction of the thread.
-//
-// TODO(scherkus): this can block the client due to synchronous Stop() API call.
-void PipelineThread::WillDestroyCurrentMessageLoop() {
- STLDeleteElements(&filter_hosts_);
- STLDeleteElements(&filter_threads_);
+template <class Filter>
+void PipelineThread::GetFilter(scoped_refptr<Filter>* filter_out) const {
+ DCHECK_EQ(PlatformThread::CurrentId(), thread_.thread_id());
+
+ *filter_out = NULL;
+ for (FilterHostVector::const_iterator iter = filter_hosts_.begin();
+ iter != filter_hosts_.end() && NULL == *filter_out;
+ iter++) {
+ (*iter)->GetFilter(filter_out);
+ }
}
} // namespace media
diff --git a/media/base/pipeline_impl.h b/media/base/pipeline_impl.h
index 5e348d5..a8e9e09 100644
--- a/media/base/pipeline_impl.h
+++ b/media/base/pipeline_impl.h
@@ -1,4 +1,4 @@
-// Copyright (c) 2006-2009 The Chromium Authors. All rights reserved.
+// Copyright (c) 2008-2009 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.
@@ -32,29 +32,25 @@ class PipelineImpl : public Pipeline {
PipelineImpl();
virtual ~PipelineImpl();
- // Implementation of PipelineStatus methods.
+ // Pipeline implementation.
+ virtual bool Start(FilterFactory* filter_factory,
+ const std::string& uri,
+ PipelineCallback* start_callback);
+ virtual void Stop();
+ virtual void Seek(base::TimeDelta time, PipelineCallback* seek_callback);
virtual bool IsInitialized() const;
- virtual base::TimeDelta GetDuration() const;
+ virtual bool IsRendered(const std::string& major_mime_type) const;
+ virtual float GetPlaybackRate() const;
+ virtual void SetPlaybackRate(float playback_rate);
+ virtual float GetVolume() const;
+ virtual void SetVolume(float volume);
+ virtual base::TimeDelta GetTime() const;
virtual base::TimeDelta GetBufferedTime() const;
+ virtual base::TimeDelta GetDuration() const;
+ virtual int64 GetBufferedBytes() const;
virtual int64 GetTotalBytes() const;
- virtual int64 GetBufferedBytes()const;
virtual void GetVideoSize(size_t* width_out, size_t* height_out) const;
- virtual float GetVolume() const;
- virtual float GetPlaybackRate() const;
- virtual base::TimeDelta GetTime() const;
- virtual base::TimeDelta GetInterpolatedTime() const;
virtual PipelineError GetError() const;
- virtual bool IsRendered(const std::string& major_mime_type) const;
-
- // Implementation of Pipeline methods.
- virtual bool Start(FilterFactory* filter_factory,
- const std::string& url,
- PipelineCallback* start_callback);
- virtual void Stop();
- virtual void SetPlaybackRate(float rate);
- virtual void Seek(base::TimeDelta time,
- PipelineCallback* seek_callback);
- virtual void SetVolume(float volume);
private:
friend class FilterHostImpl;
@@ -117,7 +113,7 @@ class PipelineImpl : public Pipeline {
int64 total_bytes_;
// Lock used to serialize access for getter/setter methods.
- Lock lock_;
+ mutable Lock lock_;
// Video width and height. Set by a FilterHostImpl object on behalf
// of a filter. The video_size_access_lock_ is used to make sure access
@@ -142,10 +138,6 @@ class PipelineImpl : public Pipeline {
// audio renderer filter.
base::TimeDelta time_;
- // Internal system timer at last time the SetTime method was called. Used to
- // compute interpolated time.
- base::TimeTicks ticks_at_last_set_time_;
-
// Status of the pipeline. Initialized to PIPELINE_OK which indicates that
// the pipeline is operating correctly. Any other value indicates that the
// pipeline is stopped or is stopping. Clients can call the Stop method to
@@ -195,8 +187,8 @@ class PipelineThread : public base::RefCountedThreadSafe<PipelineThread>,
const std::string& url_media_source,
PipelineCallback* init_complete_callback);
void Stop();
- void SetPlaybackRate(float rate);
void Seek(base::TimeDelta time, PipelineCallback* seek_callback);
+ void SetPlaybackRate(float rate);
void SetVolume(float volume);
// Methods called by a FilterHostImpl object. These methods may be called
@@ -215,10 +207,6 @@ class PipelineThread : public base::RefCountedThreadSafe<PipelineThread>,
// If the pipeline is running a nested message loop, it will be exited.
void Error(PipelineError error);
- // Called by a FilterHostImpl on behalf of a filter that calls the
- // FilterHost::PostTask method.
- void PostTask(Task* task);
-
// Simple accessor used by the FilterHostImpl class to get access to the
// pipeline object.
PipelineImpl* pipeline() const { return pipeline_; }
@@ -231,6 +219,10 @@ class PipelineThread : public base::RefCountedThreadSafe<PipelineThread>,
PlatformThreadId thread_id() const { return thread_.thread_id(); }
private:
+ // Only allow ourselves to be destroyed via ref-counting.
+ friend class base::RefCountedThreadSafe<PipelineThread>;
+ virtual ~PipelineThread();
+
enum State {
kCreated,
kInitDataSource,
@@ -244,16 +236,6 @@ class PipelineThread : public base::RefCountedThreadSafe<PipelineThread>,
kError,
};
- // Implementation of MessageLoop::DestructionObserver. StartTask registers
- // this class as a destruction observer on the thread's message loop.
- // It is used to destroy the list of FilterHosts
- // (and thus destroy the associated filters) when all tasks have been
- // processed and the message loop has been quit.
- virtual void WillDestroyCurrentMessageLoop();
-
- friend class base::RefCountedThreadSafe<PipelineThread>;
- virtual ~PipelineThread();
-
// Simple method used to make sure the pipeline is running normally.
bool IsPipelineOk() { return PIPELINE_OK == pipeline_->error_; }
@@ -267,6 +249,13 @@ class PipelineThread : public base::RefCountedThreadSafe<PipelineThread>,
state_ == kInitVideoRenderer;
}
+ // Implementation of MessageLoop::DestructionObserver. StartTask registers
+ // this class as a destruction observer on the thread's message loop.
+ // It is used to destroy the list of FilterHosts
+ // (and thus destroy the associated filters) when all tasks have been
+ // processed and the message loop has been quit.
+ virtual void WillDestroyCurrentMessageLoop();
+
// The following "task" methods correspond to the public methods, but these
// methods are run as the result of posting a task to the PipelineThread's
// message loop.
@@ -280,14 +269,10 @@ class PipelineThread : public base::RefCountedThreadSafe<PipelineThread>,
void SetPlaybackRateTask(float rate);
void SeekTask(base::TimeDelta time, PipelineCallback* seek_callback);
void SetVolumeTask(float volume);
- void SetTimeTask();
// Internal methods used in the implementation of the pipeline thread. All
// of these methods are only called on the pipeline thread.
- // Calls the Stop method on every filter in the pipeline
- void StopFilters();
-
// The following template functions make use of the fact that media filter
// derived interfaces are self-describing in the sense that they all contain
// the static method filter_type() which returns a FilterType enum that
diff --git a/media/filters/video_renderer_base.cc b/media/filters/video_renderer_base.cc
index 2a9ba66..094c92a 100644
--- a/media/filters/video_renderer_base.cc
+++ b/media/filters/video_renderer_base.cc
@@ -188,7 +188,7 @@ void VideoRendererBase::ThreadMain() {
OnFrameAvailable();
// Determine the current and next presentation timestamps.
- base::TimeDelta now = host_->GetPipelineStatus()->GetTime();
+ base::TimeDelta now = host_->GetTime();
base::TimeDelta this_pts = current_frame_->GetTimestamp();
base::TimeDelta next_pts;
if (next_frame) {
diff --git a/media/player/movie.cc b/media/player/movie.cc
index 1ef840f..e37a497 100644
--- a/media/player/movie.cc
+++ b/media/player/movie.cc
@@ -119,7 +119,7 @@ float Movie::GetDuration() {
float Movie::GetPosition() {
float position = 0.f;
if (pipeline_.get())
- position = (pipeline_->GetInterpolatedTime()).InMicroseconds() / 1000000.0f;
+ position = (pipeline_->GetTime()).InMicroseconds() / 1000000.0f;
return position;
}