// Copyright 2014 Google Inc. All rights reserved. // // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file or at // https://developers.google.com/open-source/licenses/bsd #include "packager/media/base/demuxer.h" #include "packager/base/bind.h" #include "packager/base/logging.h" #include "packager/base/stl_util.h" #include "packager/media/base/container_names.h" #include "packager/media/base/decryptor_source.h" #include "packager/media/base/key_source.h" #include "packager/media/base/media_sample.h" #include "packager/media/base/media_stream.h" #include "packager/media/base/stream_info.h" #include "packager/media/file/file.h" #include "packager/media/formats/mp2t/mp2t_media_parser.h" #include "packager/media/formats/mp4/mp4_media_parser.h" #include "packager/media/formats/wvm/wvm_media_parser.h" namespace { const size_t kInitBufSize = 0x10000; // 65KB, sufficient to determine the // container and likely all init data. const size_t kBufSize = 0x200000; // 2MB } namespace edash_packager { namespace media { Demuxer::Demuxer(const std::string& file_name) : file_name_(file_name), media_file_(NULL), init_event_received_(false), buffer_(new uint8_t[kBufSize]), cancelled_(false) { } Demuxer::~Demuxer() { if (media_file_) media_file_->Close(); STLDeleteElements(&streams_); } void Demuxer::SetKeySource(scoped_ptr key_source) { key_source_ = key_source.Pass(); } Status Demuxer::Initialize() { DCHECK(!media_file_); DCHECK(!init_event_received_); media_file_ = File::Open(file_name_.c_str(), "r"); if (!media_file_) { return Status(error::FILE_FAILURE, "Cannot open file for reading " + file_name_); } // Determine media container. int64_t bytes_read = media_file_->Read(buffer_.get(), kInitBufSize); if (bytes_read <= 0) return Status(error::FILE_FAILURE, "Cannot read file " + file_name_); MediaContainerName container = DetermineContainer(buffer_.get(), bytes_read); // Initialize media parser. switch (container) { case CONTAINER_MOV: parser_.reset(new mp4::MP4MediaParser()); break; case CONTAINER_MPEG2TS: parser_.reset(new mp2t::Mp2tMediaParser()); break; case CONTAINER_MPEG2PS: parser_.reset(new wvm::WvmMediaParser()); break; default: NOTIMPLEMENTED(); return Status(error::UNIMPLEMENTED, "Container not supported."); } parser_->Init(base::Bind(&Demuxer::ParserInitEvent, base::Unretained(this)), base::Bind(&Demuxer::NewSampleEvent, base::Unretained(this)), key_source_.get()); if (!parser_->Parse(buffer_.get(), bytes_read)) { init_parsing_status_ = Status(error::PARSER_FAILURE, "Cannot parse media file " + file_name_); } // Parse until init event received or on error. while (!init_event_received_ && init_parsing_status_.ok()) init_parsing_status_ = Parse(); // Defer error reporting if init completed successfully. return init_event_received_ ? Status::OK : init_parsing_status_; } void Demuxer::ParserInitEvent( const std::vector >& streams) { init_event_received_ = true; std::vector >::const_iterator it = streams.begin(); for (; it != streams.end(); ++it) { streams_.push_back(new MediaStream(*it, this)); } } bool Demuxer::NewSampleEvent(uint32_t track_id, const scoped_refptr& sample) { std::vector::iterator it = streams_.begin(); for (; it != streams_.end(); ++it) { if (track_id == (*it)->info()->track_id()) { return (*it)->PushSample(sample).ok(); } } return false; } Status Demuxer::Run() { Status status; // Start the streams. for (std::vector::iterator it = streams_.begin(); it != streams_.end(); ++it) { status = (*it)->Start(MediaStream::kPush); if (!status.ok()) return status; } while (!cancelled_ && (status = Parse()).ok()) continue; if (cancelled_ && status.ok()) return Status(error::CANCELLED, "Demuxer run cancelled"); if (status.error_code() == error::END_OF_STREAM) { // Push EOS sample to muxer to indicate end of stream. const scoped_refptr& sample = MediaSample::CreateEOSBuffer(); for (std::vector::iterator it = streams_.begin(); it != streams_.end(); ++it) { status = (*it)->PushSample(sample); if (!status.ok()) return status; } } return status; } Status Demuxer::Parse() { DCHECK(media_file_); DCHECK(parser_); DCHECK(buffer_); // Return early and avoid call Parse(...) again if it has already failed at // the initialization. if (!init_parsing_status_.ok()) return init_parsing_status_; int64_t bytes_read = media_file_->Read(buffer_.get(), kBufSize); if (bytes_read == 0) { parser_->Flush(); return Status(error::END_OF_STREAM, ""); } else if (bytes_read < 0) { return Status(error::FILE_FAILURE, "Cannot read file " + file_name_); } return parser_->Parse(buffer_.get(), bytes_read) ? Status::OK : Status(error::PARSER_FAILURE, "Cannot parse media file " + file_name_); } void Demuxer::Cancel() { cancelled_ = true; } } // namespace media } // namespace edash_packager