2014-02-14 23:21:05 +00:00
|
|
|
// 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
|
2013-10-11 21:44:55 +00:00
|
|
|
|
|
|
|
#include "media/base/muxer.h"
|
|
|
|
|
|
|
|
#include "media/base/encryptor_source.h"
|
|
|
|
#include "media/base/media_sample.h"
|
|
|
|
#include "media/base/media_stream.h"
|
|
|
|
|
|
|
|
namespace media {
|
|
|
|
|
2014-01-14 01:38:34 +00:00
|
|
|
Muxer::Muxer(const MuxerOptions& options)
|
2013-12-12 23:49:31 +00:00
|
|
|
: options_(options),
|
|
|
|
encryptor_source_(NULL),
|
2014-04-09 17:34:55 +00:00
|
|
|
initialized_(false),
|
2013-12-12 23:49:31 +00:00
|
|
|
clear_lead_in_seconds_(0),
|
2014-02-28 02:39:52 +00:00
|
|
|
muxer_listener_(NULL),
|
|
|
|
clock_(NULL) {}
|
2013-10-11 21:44:55 +00:00
|
|
|
|
|
|
|
Muxer::~Muxer() {}
|
|
|
|
|
2014-01-14 01:38:34 +00:00
|
|
|
void Muxer::SetEncryptorSource(EncryptorSource* encryptor_source,
|
|
|
|
double clear_lead_in_seconds) {
|
|
|
|
encryptor_source_ = encryptor_source;
|
|
|
|
clear_lead_in_seconds_ = clear_lead_in_seconds;
|
|
|
|
}
|
|
|
|
|
2014-04-09 17:34:55 +00:00
|
|
|
void Muxer::AddStream(MediaStream* stream) {
|
|
|
|
DCHECK(stream);
|
2013-10-11 21:44:55 +00:00
|
|
|
stream->Connect(this);
|
|
|
|
streams_.push_back(stream);
|
|
|
|
}
|
|
|
|
|
|
|
|
Status Muxer::Run() {
|
|
|
|
DCHECK(!streams_.empty());
|
|
|
|
|
|
|
|
Status status;
|
|
|
|
// Start the streams.
|
|
|
|
for (std::vector<MediaStream*>::iterator it = streams_.begin();
|
|
|
|
it != streams_.end();
|
|
|
|
++it) {
|
|
|
|
status = (*it)->Start(MediaStream::kPull);
|
|
|
|
if (!status.ok())
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2013-11-12 20:37:58 +00:00
|
|
|
uint32 current_stream_id = 0;
|
2013-10-11 21:44:55 +00:00
|
|
|
while (status.ok()) {
|
|
|
|
scoped_refptr<MediaSample> sample;
|
2013-11-12 20:37:58 +00:00
|
|
|
status = streams_[current_stream_id]->PullSample(&sample);
|
2013-10-11 21:44:55 +00:00
|
|
|
if (!status.ok())
|
|
|
|
break;
|
2013-11-12 20:37:58 +00:00
|
|
|
status = AddSample(streams_[current_stream_id], sample);
|
|
|
|
|
|
|
|
// Switch to next stream if the current stream is ready for fragmentation.
|
2014-04-09 17:34:55 +00:00
|
|
|
if (status.error_code() == error::FRAGMENT_FINALIZED) {
|
2013-11-12 20:37:58 +00:00
|
|
|
current_stream_id = (current_stream_id + 1) % streams_.size();
|
|
|
|
status.Clear();
|
|
|
|
}
|
2013-10-11 21:44:55 +00:00
|
|
|
}
|
2014-04-09 17:34:55 +00:00
|
|
|
// Finalize the muxer after reaching end of stream.
|
|
|
|
return status.error_code() == error::END_OF_STREAM ? Finalize() : status;
|
2013-10-11 21:44:55 +00:00
|
|
|
}
|
|
|
|
|
2013-12-12 23:49:31 +00:00
|
|
|
void Muxer::SetMuxerListener(media::event::MuxerListener* muxer_listener) {
|
|
|
|
muxer_listener_ = muxer_listener;
|
|
|
|
}
|
|
|
|
|
2014-04-09 17:34:55 +00:00
|
|
|
Status Muxer::AddSample(const MediaStream* stream,
|
|
|
|
scoped_refptr<MediaSample> sample) {
|
|
|
|
DCHECK(std::find(streams_.begin(), streams_.end(), stream) != streams_.end());
|
|
|
|
|
|
|
|
if (!initialized_) {
|
|
|
|
Status status = Initialize();
|
|
|
|
if (!status.ok())
|
|
|
|
return status;
|
|
|
|
initialized_ = true;
|
|
|
|
}
|
|
|
|
if (sample->end_of_stream()) {
|
|
|
|
// EOS sample should be sent only when the sample was pushed from Demuxer
|
|
|
|
// to Muxer. In this case, there should be only one stream in Muxer.
|
|
|
|
DCHECK_EQ(1u, streams_.size());
|
|
|
|
return Finalize();
|
|
|
|
}
|
|
|
|
return DoAddSample(stream, sample);
|
|
|
|
}
|
|
|
|
|
2013-10-11 21:44:55 +00:00
|
|
|
} // namespace media
|