diff options
author | scherkus@chromium.org <scherkus@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2012-11-17 02:01:55 +0000 |
---|---|---|
committer | scherkus@chromium.org <scherkus@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2012-11-17 02:01:55 +0000 |
commit | 0ae5fc527cc3975a3218ca4e2affbc8963343f0b (patch) | |
tree | 9b329d13e7a9f8ff75b576a6c542b9144613f6b3 | |
parent | 4a74b43b846e193603fb8be995fa82cbd86d922f (diff) | |
download | chromium_src-0ae5fc527cc3975a3218ca4e2affbc8963343f0b.zip chromium_src-0ae5fc527cc3975a3218ca4e2affbc8963343f0b.tar.gz chromium_src-0ae5fc527cc3975a3218ca4e2affbc8963343f0b.tar.bz2 |
Check in demuxer_bench, a tool for benchmarking FFmpegDemuxer reading.
Seeing as I'm making substantial changes to how we demux, it would be prudent to benchmark any differences in performance.
BUG=160640
Review URL: https://codereview.chromium.org/11412016
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@168353 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r-- | media/media.gyp | 11 | ||||
-rw-r--r-- | media/tools/demuxer_bench/demuxer_bench.cc | 219 |
2 files changed, 230 insertions, 0 deletions
diff --git a/media/media.gyp b/media/media.gyp index 310d1be..eca9c7f 100644 --- a/media/media.gyp +++ b/media/media.gyp @@ -1019,6 +1019,17 @@ 'tools/seek_tester/seek_tester.cc', ], }, + { + 'target_name': 'demuxer_bench', + 'type': 'executable', + 'dependencies': [ + 'media', + '../base/base.gyp:base', + ], + 'sources': [ + 'tools/demuxer_bench/demuxer_bench.cc', + ], + }, ], }], ['OS=="win"', { diff --git a/media/tools/demuxer_bench/demuxer_bench.cc b/media/tools/demuxer_bench/demuxer_bench.cc new file mode 100644 index 0000000..9de8d7c --- /dev/null +++ b/media/tools/demuxer_bench/demuxer_bench.cc @@ -0,0 +1,219 @@ +// Copyright (c) 2012 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. +// +// demuxer_bench is a standalone benchmarking tool for FFmpegDemuxer. It +// simulates the reading requirements for playback by reading from the stream +// that has the earliest timestamp. + +#include <iostream> + +#include "base/at_exit.h" +#include "base/bind.h" +#include "base/command_line.h" +#include "base/logging.h" +#include "base/message_loop.h" +#include "base/string_number_conversions.h" +#include "media/base/media.h" +#include "media/filters/ffmpeg_demuxer.h" +#include "media/filters/file_data_source.h" + +class DemuxerHostImpl : public media::DemuxerHost { + public: + DemuxerHostImpl() {} + virtual ~DemuxerHostImpl() {} + + // DataSourceHost implementation. + virtual void SetTotalBytes(int64 total_bytes) OVERRIDE {} + virtual void AddBufferedByteRange(int64 start, int64 end) OVERRIDE {} + virtual void AddBufferedTimeRange(base::TimeDelta start, + base::TimeDelta end) OVERRIDE {} + + // DemuxerHost implementation. + virtual void SetDuration(base::TimeDelta duration) OVERRIDE {} + virtual void OnDemuxerError(media::PipelineStatus error) OVERRIDE {} + + private: + DISALLOW_COPY_AND_ASSIGN(DemuxerHostImpl); +}; + +void QuitLoopWithStatus(MessageLoop* message_loop, + media::PipelineStatus status) { + CHECK_EQ(status, media::PIPELINE_OK); + message_loop->PostTask(FROM_HERE, MessageLoop::QuitWhenIdleClosure()); +} + +typedef std::vector<scoped_refptr<media::DemuxerStream> > Streams; + +// Simulates playback reading requirements by reading from each stream +// present in |demuxer| in as-close-to-monotonically-increasing timestamp order. +class StreamReader { + public: + explicit StreamReader(const scoped_refptr<media::Demuxer>& demuxer); + ~StreamReader(); + + // Performs a single step read. + void Read(); + + // Returns true when all streams have reached end of stream. + bool IsDone(); + + int number_of_streams() { return streams_.size(); } + const Streams& streams() { return streams_; } + const std::vector<int>& counts() { return counts_; } + + private: + void OnReadDone(MessageLoop* message_loop, + bool* end_of_stream, + base::TimeDelta* timestamp, + media::DemuxerStream::Status status, + const scoped_refptr<media::DecoderBuffer>& buffer); + int GetNextStreamIndexToRead(); + + Streams streams_; + std::vector<bool> end_of_stream_; + std::vector<base::TimeDelta> last_read_timestamp_; + std::vector<int> counts_; + + DISALLOW_COPY_AND_ASSIGN(StreamReader); +}; + +StreamReader::StreamReader(const scoped_refptr<media::Demuxer> &demuxer) { + scoped_refptr<media::DemuxerStream> stream; + stream = demuxer->GetStream(media::DemuxerStream::AUDIO); + if (stream) { + streams_.push_back(stream); + end_of_stream_.push_back(false); + last_read_timestamp_.push_back(media::kNoTimestamp()); + counts_.push_back(0); + } + + stream = demuxer->GetStream(media::DemuxerStream::VIDEO); + if (stream) { + streams_.push_back(stream); + end_of_stream_.push_back(false); + last_read_timestamp_.push_back(media::kNoTimestamp()); + counts_.push_back(0); + } +} + +StreamReader::~StreamReader() {} + +void StreamReader::Read() { + int index = GetNextStreamIndexToRead(); + bool end_of_stream = false; + base::TimeDelta timestamp; + + streams_[index]->Read(base::Bind( + &StreamReader::OnReadDone, base::Unretained(this), MessageLoop::current(), + &end_of_stream, ×tamp)); + MessageLoop::current()->Run(); + + CHECK(timestamp != media::kNoTimestamp()); + end_of_stream_[index] = end_of_stream; + last_read_timestamp_[index] = timestamp; + counts_[index]++; +} + +bool StreamReader::IsDone() { + for (size_t i = 0; i < end_of_stream_.size(); ++i) { + if (!end_of_stream_[i]) + return false; + } + return true; +} + +void StreamReader::OnReadDone( + MessageLoop* message_loop, + bool* end_of_stream, + base::TimeDelta* timestamp, + media::DemuxerStream::Status status, + const scoped_refptr<media::DecoderBuffer>& buffer) { + CHECK_EQ(status, media::DemuxerStream::kOk); + CHECK(buffer); + *end_of_stream = buffer->IsEndOfStream(); + *timestamp = buffer->GetTimestamp(); + message_loop->PostTask(FROM_HERE, MessageLoop::QuitWhenIdleClosure()); +} + +int StreamReader::GetNextStreamIndexToRead() { + int index = -1; + for (int i = 0; i < number_of_streams(); ++i) { + // Ignore streams at EOS. + if (end_of_stream_[i]) + continue; + + // Use a stream if it hasn't been read from yet. + if (last_read_timestamp_[i] == media::kNoTimestamp()) + return i; + + if (index < 0 || + last_read_timestamp_[i] < last_read_timestamp_[index]) { + index = i; + } + } + CHECK_GE(index, 0) << "Couldn't find a stream to read"; + return index; +} + +int main(int argc, char** argv) { + base::AtExitManager at_exit; + media::InitializeMediaLibraryForTesting(); + + CommandLine::Init(argc, argv); + CommandLine* cmd_line = CommandLine::ForCurrentProcess(); + + if (cmd_line->GetArgs().empty()) { + std::cerr << "Usage: " << argv[0] << " [file]" << std::endl; + return 1; + } + + MessageLoop message_loop; + DemuxerHostImpl demuxer_host; + FilePath file_path(cmd_line->GetArgs()[0]); + + // Setup. + scoped_refptr<media::FileDataSource> data_source = + new media::FileDataSource(); + CHECK(data_source->Initialize(file_path)); + + scoped_refptr<media::FFmpegDemuxer> demuxer = + new media::FFmpegDemuxer(message_loop.message_loop_proxy(), data_source); + + demuxer->Initialize(&demuxer_host, base::Bind( + &QuitLoopWithStatus, &message_loop)); + message_loop.Run(); + + StreamReader stream_reader(demuxer); + + // Benchmark. + base::TimeTicks start = base::TimeTicks::HighResNow(); + while (!stream_reader.IsDone()) { + stream_reader.Read(); + } + base::TimeTicks end = base::TimeTicks::HighResNow(); + + // Results. + std::cout << "Time: " << (end - start).InMillisecondsF() << " ms\n"; + for (int i = 0; i < stream_reader.number_of_streams(); ++i) { + scoped_refptr<media::DemuxerStream> stream = stream_reader.streams()[i]; + std::cout << "Stream #" << i << ": "; + + if (stream->type() == media::DemuxerStream::AUDIO) { + std::cout << "audio"; + } else if (stream->type() == media::DemuxerStream::VIDEO) { + std::cout << "video"; + } else { + std::cout << "unknown"; + } + + std::cout << ", " << stream_reader.counts()[i] << " packets" << std::endl; + } + + // Teardown. + demuxer->Stop(base::Bind( + &QuitLoopWithStatus, &message_loop, media::PIPELINE_OK)); + message_loop.Run(); + + return 0; +} |