blob: c804b54867466d9adf3fe8add778a868aa39fb40 [file] [log] [blame]
// Copyright 2012 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#include "cobalt/media/filters/shell_demuxer.h"
#include <inttypes.h>
#include "base/bind.h"
#include "base/callback.h"
#include "base/callback_helpers.h"
#include "base/debug/trace_event.h"
#include "base/memory/scoped_ptr.h"
#include "base/message_loop.h"
#include "base/stringprintf.h"
#include "base/task_runner_util.h"
#include "base/time.h"
#include "cobalt/media/base/bind_to_current_loop.h"
#include "cobalt/media/base/data_source.h"
#include "cobalt/media/base/shell_media_platform.h"
#include "cobalt/media/base/timestamp_constants.h"
#include "starboard/types.h"
namespace cobalt {
namespace media {
ShellDemuxerStream::ShellDemuxerStream(ShellDemuxer* demuxer, Type type)
: demuxer_(demuxer),
type_(type),
last_buffer_timestamp_(kNoTimestamp),
stopped_(false),
total_buffer_size_(0) {
TRACE_EVENT0("media_stack", "ShellDemuxerStream::ShellDemuxerStream()");
DCHECK(demuxer_);
}
void ShellDemuxerStream::Read(const ReadCB& read_cb) {
TRACE_EVENT0("media_stack", "ShellDemuxerStream::Read()");
DCHECK(!read_cb.is_null());
base::AutoLock auto_lock(lock_);
// Don't accept any additional reads if we've been told to stop.
// The demuxer_ may have been destroyed in the pipleine thread.
if (stopped_) {
TRACE_EVENT0("media_stack", "ShellDemuxerStream::Read() EOS sent.");
read_cb.Run(DemuxerStream::kOk,
scoped_refptr<DecoderBuffer>(DecoderBuffer::CreateEOSBuffer()));
return;
}
// Buffers are only queued when there are no pending reads.
DCHECK(buffer_queue_.empty() || read_queue_.empty());
if (!buffer_queue_.empty()) {
// Send the oldest buffer back.
scoped_refptr<DecoderBuffer> buffer = buffer_queue_.front();
if (buffer->end_of_stream()) {
TRACE_EVENT0("media_stack", "ShellDemuxerStream::Read() EOS sent.");
} else {
// Do not pop EOS buffers, so that subsequent read requests also get EOS
total_buffer_size_ -= buffer->data_size();
buffer_queue_.pop_front();
}
read_cb.Run(
DemuxerStream::kOk,
ShellMediaPlatform::Instance()->ProcessBeforeLeavingDemuxer(buffer));
} else {
TRACE_EVENT0("media_stack", "ShellDemuxerStream::Read() request queued.");
read_queue_.push_back(read_cb);
}
}
AudioDecoderConfig ShellDemuxerStream::audio_decoder_config() {
return demuxer_->AudioConfig();
}
VideoDecoderConfig ShellDemuxerStream::video_decoder_config() {
return demuxer_->VideoConfig();
}
Ranges<base::TimeDelta> ShellDemuxerStream::GetBufferedRanges() {
base::AutoLock auto_lock(lock_);
return buffered_ranges_;
}
DemuxerStream::Type ShellDemuxerStream::type() const { return type_; }
void ShellDemuxerStream::EnableBitstreamConverter() { NOTIMPLEMENTED(); }
void ShellDemuxerStream::EnqueueBuffer(scoped_refptr<DecoderBuffer> buffer) {
TRACE_EVENT1("media_stack", "ShellDemuxerStream::EnqueueBuffer()",
"timestamp", buffer->end_of_stream() ?
-1 : buffer->timestamp().InMicroseconds());
base::AutoLock auto_lock(lock_);
if (stopped_) {
// it's possible due to pipelining both downstream and within the
// demuxer that several pipelined reads will be enqueuing packets
// on a stopped stream. Drop them after complaining.
DLOG(WARNING) << "attempted to enqueue packet on stopped stream";
return;
}
if (buffer->end_of_stream()) {
TRACE_EVENT0("media_stack",
"ShellDemuxerStream::EnqueueBuffer() EOS received.");
} else if (buffer->timestamp() != kNoTimestamp) {
if (last_buffer_timestamp_ != kNoTimestamp &&
last_buffer_timestamp_ < buffer->timestamp()) {
buffered_ranges_.Add(last_buffer_timestamp_, buffer->timestamp());
}
last_buffer_timestamp_ = buffer->timestamp();
} else {
DLOG(WARNING) << "bad timestamp info on enqueued buffer.";
}
// Check for any already waiting reads, service oldest read if there
if (read_queue_.size()) {
// assumption here is that buffer queue is empty
DCHECK_EQ(buffer_queue_.size(), 0);
ReadCB read_cb(read_queue_.front());
read_queue_.pop_front();
read_cb.Run(
DemuxerStream::kOk,
ShellMediaPlatform::Instance()->ProcessBeforeLeavingDemuxer(buffer));
} else {
// save the buffer for next read request
buffer_queue_.push_back(buffer);
if (!buffer->end_of_stream()) {
total_buffer_size_ += buffer->data_size();
}
}
}
base::TimeDelta ShellDemuxerStream::GetLastBufferTimestamp() const {
base::AutoLock auto_lock(lock_);
return last_buffer_timestamp_;
}
size_t ShellDemuxerStream::GetTotalBufferSize() const {
base::AutoLock auto_lock(lock_);
return total_buffer_size_;
}
void ShellDemuxerStream::FlushBuffers() {
TRACE_EVENT0("media_stack", "ShellDemuxerStream::FlushBuffers()");
base::AutoLock auto_lock(lock_);
// TODO: Investigate if the following warning is valid.
DLOG_IF(WARNING, !read_queue_.empty()) << "Read requests should be empty";
buffer_queue_.clear();
total_buffer_size_ = 0;
last_buffer_timestamp_ = kNoTimestamp;
}
void ShellDemuxerStream::Stop() {
TRACE_EVENT0("media_stack", "ShellDemuxerStream::Stop()");
DCHECK(demuxer_->MessageLoopBelongsToCurrentThread());
base::AutoLock auto_lock(lock_);
buffer_queue_.clear();
total_buffer_size_ = 0;
last_buffer_timestamp_ = kNoTimestamp;
// fulfill any pending callbacks with EOS buffers set to end timestamp
for (ReadQueue::iterator it = read_queue_.begin(); it != read_queue_.end();
++it) {
TRACE_EVENT0("media_stack", "ShellDemuxerStream::Stop() EOS sent.");
it->Run(DemuxerStream::kOk,
scoped_refptr<DecoderBuffer>(DecoderBuffer::CreateEOSBuffer()));
}
read_queue_.clear();
stopped_ = true;
}
//
// ShellDemuxer
//
ShellDemuxer::ShellDemuxer(
const scoped_refptr<base::MessageLoopProxy>& message_loop,
DecoderBuffer::Allocator* buffer_allocator, DataSource* data_source,
const scoped_refptr<MediaLog>& media_log)
: message_loop_(message_loop),
buffer_allocator_(buffer_allocator),
host_(NULL),
blocking_thread_("ShellDemuxerBlockingThread"),
data_source_(data_source),
media_log_(media_log),
stopped_(false),
flushing_(false),
audio_reached_eos_(false),
video_reached_eos_(false) {
DCHECK(message_loop_);
DCHECK(buffer_allocator_);
DCHECK(data_source_);
DCHECK(media_log_);
reader_ = new ShellDataSourceReader();
reader_->SetDataSource(data_source_);
}
ShellDemuxer::~ShellDemuxer() {
// Explicitly stop |blocking_thread_| to ensure that it stops before the
// destructiing of any other members.
blocking_thread_.Stop();
}
void ShellDemuxer::Initialize(DemuxerHost* host,
const PipelineStatusCB& status_cb,
bool enable_text_tracks) {
TRACE_EVENT0("media_stack", "ShellDemuxer::Initialize()");
DCHECK(!enable_text_tracks);
DCHECK(MessageLoopBelongsToCurrentThread());
DCHECK(reader_);
DCHECK(!parser_);
DLOG(INFO) << "this is a PROGRESSIVE playback.";
host_ = host;
// create audio and video demuxer stream objects
audio_demuxer_stream_.reset(
new ShellDemuxerStream(this, DemuxerStream::AUDIO));
video_demuxer_stream_.reset(
new ShellDemuxerStream(this, DemuxerStream::VIDEO));
// start the blocking thread and have it download and parse the media config
if (!blocking_thread_.Start()) {
status_cb.Run(DEMUXER_ERROR_COULD_NOT_PARSE);
return;
}
blocking_thread_.message_loop_proxy()->PostTask(
FROM_HERE, base::Bind(&ShellDemuxer::ParseConfigBlocking,
base::Unretained(this), status_cb));
}
void ShellDemuxer::ParseConfigBlocking(const PipelineStatusCB& status_cb) {
DCHECK(blocking_thread_.message_loop_proxy()->BelongsToCurrentThread());
DCHECK(!parser_);
// construct stream parser with error callback
PipelineStatus status = ShellParser::Construct(reader_, &parser_, media_log_);
// if we can't construct a parser for this stream it's a fatal error, return
// false so ParseConfigDone will notify the caller to Initialize() via
// status_cb.
if (!parser_ || status != PIPELINE_OK) {
DCHECK(!parser_);
DCHECK_NE(status, PIPELINE_OK);
if (status == PIPELINE_OK) {
status = DEMUXER_ERROR_COULD_NOT_PARSE;
}
ParseConfigDone(status_cb, status);
return;
}
// instruct the parser to extract audio and video config from the file
if (!parser_->ParseConfig()) {
ParseConfigDone(status_cb, DEMUXER_ERROR_COULD_NOT_PARSE);
return;
}
// make sure we got a valid and complete configuration
if (!parser_->IsConfigComplete()) {
ParseConfigDone(status_cb, DEMUXER_ERROR_COULD_NOT_PARSE);
return;
}
// IsConfigComplete() should guarantee we know the duration
DCHECK(parser_->Duration() != kInfiniteDuration);
host_->SetDuration(parser_->Duration());
// Bitrate may not be known, however
uint32 bitrate = parser_->BitsPerSecond();
if (bitrate > 0) {
data_source_->SetBitrate(bitrate);
}
// successful parse of config data, inform the nonblocking demuxer thread
DCHECK_EQ(status, PIPELINE_OK);
ParseConfigDone(status_cb, PIPELINE_OK);
}
void ShellDemuxer::ParseConfigDone(const PipelineStatusCB& status_cb,
PipelineStatus status) {
DCHECK(blocking_thread_.message_loop_proxy()->BelongsToCurrentThread());
if (HasStopCalled()) {
return;
}
// if the blocking parser thread cannot parse config we're done.
if (status != PIPELINE_OK) {
status_cb.Run(status);
return;
}
DCHECK(parser_);
// start downloading data
Request(DemuxerStream::AUDIO);
status_cb.Run(PIPELINE_OK);
}
void ShellDemuxer::Request(DemuxerStream::Type type) {
if (!blocking_thread_.message_loop_proxy()->BelongsToCurrentThread()) {
blocking_thread_.message_loop_proxy()->PostTask(
FROM_HERE,
base::Bind(&ShellDemuxer::Request, base::Unretained(this), type));
return;
}
DCHECK(!requested_au_) << "overlapping requests not supported!";
flushing_ = false;
// Ask parser for next AU
scoped_refptr<ShellAU> au = parser_->GetNextAU(type);
// fatal parsing error returns NULL or malformed AU
if (!au || !au->IsValid()) {
if (!HasStopCalled()) {
DLOG(ERROR) << "got back bad AU from parser";
host_->OnDemuxerError(DEMUXER_ERROR_COULD_NOT_PARSE);
}
return;
}
// make sure we got back an AU of the correct type
DCHECK(au->GetType() == type);
const char* ALLOW_UNUSED event_type =
type == DemuxerStream::AUDIO ? "audio" : "video";
TRACE_EVENT2("media_stack", "ShellDemuxer::RequestTask()", "type", event_type,
"timestamp", au->GetTimestamp().InMicroseconds());
// don't issue allocation requests for EOS AUs
if (au->IsEndOfStream()) {
TRACE_EVENT0("media_stack", "ShellDemuxer::RequestTask() EOS sent");
// enqueue EOS buffer with correct stream
scoped_refptr<DecoderBuffer> eos_buffer = DecoderBuffer::CreateEOSBuffer();
if (type == DemuxerStream::AUDIO) {
audio_reached_eos_ = true;
audio_demuxer_stream_->EnqueueBuffer(eos_buffer);
} else if (type == DemuxerStream::VIDEO) {
video_reached_eos_ = true;
video_demuxer_stream_->EnqueueBuffer(eos_buffer);
}
IssueNextRequest();
return;
}
// enqueue the request
requested_au_ = au;
AllocateBuffer();
}
void ShellDemuxer::AllocateBuffer() {
DCHECK(requested_au_);
if (HasStopCalled()) {
return;
}
if (requested_au_) {
size_t total_buffer_size = audio_demuxer_stream_->GetTotalBufferSize() +
video_demuxer_stream_->GetTotalBufferSize();
if (total_buffer_size >= COBALT_MEDIA_BUFFER_PROGRESSIVE_BUDGET) {
// Retry after 100 milliseconds.
const base::TimeDelta kDelay = base::TimeDelta::FromMilliseconds(100);
blocking_thread_.message_loop()->PostDelayedTask(
FROM_HERE,
base::Bind(&ShellDemuxer::AllocateBuffer, base::Unretained(this)),
kDelay);
return;
}
// Note that "new DecoderBuffer" may return NULL if it is unable to allocate
// any DecoderBuffer.
scoped_refptr<DecoderBuffer> decoder_buffer(
DecoderBuffer::Create(buffer_allocator_, requested_au_->GetType(),
requested_au_->GetMaxSize()));
if (decoder_buffer) {
decoder_buffer->set_is_key_frame(requested_au_->IsKeyframe());
Download(decoder_buffer);
} else {
// As the buffer is full of media data, it is safe to delay 100
// milliseconds.
const base::TimeDelta kDelay = base::TimeDelta::FromMilliseconds(100);
blocking_thread_.message_loop()->PostDelayedTask(
FROM_HERE,
base::Bind(&ShellDemuxer::AllocateBuffer, base::Unretained(this)),
kDelay);
}
}
}
void ShellDemuxer::Download(scoped_refptr<DecoderBuffer> buffer) {
DCHECK(blocking_thread_.message_loop_proxy()->BelongsToCurrentThread());
// We need a requested_au_ or to have canceled this request and
// are buffering to a new location for this to make sense
DCHECK(requested_au_);
const char* ALLOW_UNUSED event_type =
requested_au_->GetType() == DemuxerStream::AUDIO ? "audio" : "video";
TRACE_EVENT2("media_stack", "ShellDemuxer::Download()", "type", event_type,
"timestamp", requested_au_->GetTimestamp().InMicroseconds());
// do nothing if stopped
if (HasStopCalled()) {
DLOG(INFO) << "aborting download task, stopped";
return;
}
// Flushing is a signal to restart the request->download cycle with
// a new request. Drop current request and issue a new one.
// flushing_ will be reset by the next call to RequestTask()
if (flushing_) {
DLOG(INFO) << "skipped AU download due to flush";
requested_au_ = NULL;
IssueNextRequest();
return;
}
if (!requested_au_->Read(reader_, buffer)) {
DLOG(ERROR) << "au read failed";
host_->OnDemuxerError(PIPELINE_ERROR_READ);
return;
}
// copy timestamp and duration values
buffer->set_timestamp(requested_au_->GetTimestamp());
buffer->set_duration(requested_au_->GetDuration());
// enqueue buffer into appropriate stream
if (requested_au_->GetType() == DemuxerStream::AUDIO) {
audio_demuxer_stream_->EnqueueBuffer(buffer);
} else if (requested_au_->GetType() == DemuxerStream::VIDEO) {
video_demuxer_stream_->EnqueueBuffer(buffer);
} else {
NOTREACHED() << "invalid buffer type enqueued";
}
// finished with this au, deref
requested_au_ = NULL;
// Calculate total range of buffered data for both audio and video.
Ranges<base::TimeDelta> buffered(
audio_demuxer_stream_->GetBufferedRanges().IntersectionWith(
video_demuxer_stream_->GetBufferedRanges()));
// Notify host of each disjoint range.
host_->OnBufferedTimeRangesChanged(buffered);
// Post the task with a delay to make the request loop a bit friendly to
// other tasks as otherwise IssueNextRequest(), Request(), AllocateBuffer(),
// and Download() can form a tight loop on the |blocking_thread_|.
const base::TimeDelta kDelay = base::TimeDelta::FromMilliseconds(5);
blocking_thread_.message_loop_proxy()->PostDelayedTask(
FROM_HERE,
base::Bind(&ShellDemuxer::IssueNextRequest, base::Unretained(this)),
kDelay);
}
void ShellDemuxer::IssueNextRequest() {
DCHECK(!requested_au_);
// if we're stopped don't download anymore
if (HasStopCalled()) {
DLOG(INFO) << "stopped so request loop is stopping";
return;
}
DemuxerStream::Type type = DemuxerStream::UNKNOWN;
// if we have eos in one or both buffers the decision is easy
if (audio_reached_eos_ || video_reached_eos_) {
if (audio_reached_eos_) {
if (video_reached_eos_) {
// both are true, issue no more requests!
DLOG(INFO) << "both streams at EOS, request loop stopping";
return;
} else {
// audio is at eos, video isn't, get more video
type = DemuxerStream::VIDEO;
}
} else {
// audio is not at eos, video is, get more audio
type = DemuxerStream::AUDIO;
}
} else {
// priority order for figuring out what to download next
base::TimeDelta audio_stamp =
audio_demuxer_stream_->GetLastBufferTimestamp();
base::TimeDelta video_stamp =
video_demuxer_stream_->GetLastBufferTimestamp();
// if the audio demuxer stream is empty, always fill it first
if (audio_stamp == kNoTimestamp) {
type = DemuxerStream::AUDIO;
} else if (video_stamp == kNoTimestamp) {
// the video demuxer stream is empty, we need data for it
type = DemuxerStream::VIDEO;
} else if (video_stamp < audio_stamp) {
// video is earlier, fill it first
type = DemuxerStream::VIDEO;
} else {
type = DemuxerStream::AUDIO;
}
}
DCHECK_NE(type, DemuxerStream::UNKNOWN);
// We cannot call Request() directly even if this function is also run on
// |blocking_thread_| as otherwise it is possible that this function is
// running in a tight loop and seek or stop request has no chance to kick in.
blocking_thread_.message_loop_proxy()->PostTask(
FROM_HERE,
base::Bind(&ShellDemuxer::Request, base::Unretained(this), type));
}
void ShellDemuxer::Stop() {
DCHECK(MessageLoopBelongsToCurrentThread());
// set our internal stop flag, to not treat read failures as
// errors anymore but as a natural part of stopping
{
base::AutoLock auto_lock(lock_for_stopped_);
stopped_ = true;
}
// stop the reader, which will stop the datasource and call back
reader_->Stop();
}
void ShellDemuxer::DataSourceStopped(const base::Closure& callback) {
TRACE_EVENT0("media_stack", "ShellDemuxer::DataSourceStopped()");
DCHECK(MessageLoopBelongsToCurrentThread());
// stop the download thread
blocking_thread_.Stop();
// tell downstream we've stopped
if (audio_demuxer_stream_) audio_demuxer_stream_->Stop();
if (video_demuxer_stream_) video_demuxer_stream_->Stop();
callback.Run();
}
bool ShellDemuxer::HasStopCalled() {
base::AutoLock auto_lock(lock_for_stopped_);
return stopped_;
}
void ShellDemuxer::Seek(base::TimeDelta time, const PipelineStatusCB& cb) {
blocking_thread_.message_loop()->PostTask(
FROM_HERE, base::Bind(&ShellDemuxer::SeekTask, base::Unretained(this),
time, BindToCurrentLoop(cb)));
}
// runs on blocking thread
void ShellDemuxer::SeekTask(base::TimeDelta time, const PipelineStatusCB& cb) {
TRACE_EVENT1("media_stack", "ShellDemuxer::SeekTask()", "timestamp",
time.InMicroseconds());
DLOG(INFO) << base::StringPrintf("seek to: %" PRId64 " ms",
time.InMilliseconds());
// clear any enqueued buffers on demuxer streams
audio_demuxer_stream_->FlushBuffers();
video_demuxer_stream_->FlushBuffers();
// advance parser to new timestamp
if (!parser_->SeekTo(time)) {
DLOG(ERROR) << "parser seek failed.";
cb.Run(PIPELINE_ERROR_READ);
return;
}
// if both streams had finished downloading, we need to restart the request
bool issue_new_request = audio_reached_eos_ && video_reached_eos_;
audio_reached_eos_ = false;
video_reached_eos_ = false;
flushing_ = true;
cb.Run(PIPELINE_OK);
if (issue_new_request) {
DLOG(INFO) << "restarting stopped request loop";
Request(DemuxerStream::AUDIO);
}
}
DemuxerStream* ShellDemuxer::GetStream(media::DemuxerStream::Type type) {
if (type == DemuxerStream::AUDIO) {
return audio_demuxer_stream_.get();
} else if (type == DemuxerStream::VIDEO) {
return video_demuxer_stream_.get();
} else {
DLOG(WARNING) << "unsupported stream type requested";
}
return NULL;
}
base::TimeDelta ShellDemuxer::GetStartTime() const {
// we always assume a start time of 0
return base::TimeDelta();
}
const AudioDecoderConfig& ShellDemuxer::AudioConfig() {
return parser_->AudioConfig();
}
const VideoDecoderConfig& ShellDemuxer::VideoConfig() {
return parser_->VideoConfig();
}
bool ShellDemuxer::MessageLoopBelongsToCurrentThread() const {
return message_loop_->BelongsToCurrentThread();
}
} // namespace media
} // namespace cobalt