// Copyright 2013 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. #include "media/cast/cast_sender_impl.h" #include "base/bind.h" #include "base/callback.h" #include "base/logging.h" #include "base/macros.h" #include "base/message_loop/message_loop.h" #include "media/base/video_frame.h" #include "media/cast/sender/video_frame_factory.h" namespace media { namespace cast { // The LocalVideoFrameInput class posts all incoming video frames to the main // cast thread for processing. class LocalVideoFrameInput : public VideoFrameInput { public: LocalVideoFrameInput(scoped_refptr cast_environment, base::WeakPtr video_sender) : cast_environment_(cast_environment), video_sender_(video_sender), video_frame_factory_( video_sender.get() ? video_sender->CreateVideoFrameFactory().release() : nullptr) {} void InsertRawVideoFrame(const scoped_refptr& video_frame, const base::TimeTicks& capture_time) final { cast_environment_->PostTask(CastEnvironment::MAIN, FROM_HERE, base::Bind(&VideoSender::InsertRawVideoFrame, video_sender_, video_frame, capture_time)); } scoped_refptr MaybeCreateOptimizedFrame( const gfx::Size& frame_size, base::TimeDelta timestamp) final { return video_frame_factory_ ? video_frame_factory_->MaybeCreateFrame(frame_size, timestamp) : nullptr; } bool CanCreateOptimizedFrames() const final { return video_frame_factory_.get() != nullptr; } protected: ~LocalVideoFrameInput() final {} private: friend class base::RefCountedThreadSafe; const scoped_refptr cast_environment_; const base::WeakPtr video_sender_; const scoped_ptr video_frame_factory_; DISALLOW_COPY_AND_ASSIGN(LocalVideoFrameInput); }; // The LocalAudioFrameInput class posts all incoming audio frames to the main // cast thread for processing. Therefore frames can be inserted from any thread. class LocalAudioFrameInput : public AudioFrameInput { public: LocalAudioFrameInput(scoped_refptr cast_environment, base::WeakPtr audio_sender) : cast_environment_(cast_environment), audio_sender_(audio_sender) {} void InsertAudio(scoped_ptr audio_bus, const base::TimeTicks& recorded_time) final { cast_environment_->PostTask(CastEnvironment::MAIN, FROM_HERE, base::Bind(&AudioSender::InsertAudio, audio_sender_, base::Passed(&audio_bus), recorded_time)); } protected: ~LocalAudioFrameInput() final {} private: friend class base::RefCountedThreadSafe; scoped_refptr cast_environment_; base::WeakPtr audio_sender_; DISALLOW_COPY_AND_ASSIGN(LocalAudioFrameInput); }; scoped_ptr CastSender::Create( scoped_refptr cast_environment, CastTransportSender* const transport_sender) { CHECK(cast_environment.get()); return scoped_ptr( new CastSenderImpl(cast_environment, transport_sender)); } CastSenderImpl::CastSenderImpl( scoped_refptr cast_environment, CastTransportSender* const transport_sender) : cast_environment_(cast_environment), transport_sender_(transport_sender), weak_factory_(this) { CHECK(cast_environment.get()); } void CastSenderImpl::InitializeAudio( const AudioSenderConfig& audio_config, const StatusChangeCallback& status_change_cb) { DCHECK(cast_environment_->CurrentlyOn(CastEnvironment::MAIN)); CHECK(audio_config.use_external_encoder || cast_environment_->HasAudioThread()); VLOG(1) << "CastSenderImpl@" << this << "::InitializeAudio()"; audio_sender_.reset( new AudioSender(cast_environment_, audio_config, base::Bind(&CastSenderImpl::OnAudioStatusChange, weak_factory_.GetWeakPtr(), status_change_cb), transport_sender_)); if (video_sender_) { DCHECK(audio_sender_->GetTargetPlayoutDelay() == video_sender_->GetTargetPlayoutDelay()); } } void CastSenderImpl::InitializeVideo( const VideoSenderConfig& video_config, const StatusChangeCallback& status_change_cb, const CreateVideoEncodeAcceleratorCallback& create_vea_cb, const CreateVideoEncodeMemoryCallback& create_video_encode_mem_cb) { DCHECK(cast_environment_->CurrentlyOn(CastEnvironment::MAIN)); VLOG(1) << "CastSenderImpl@" << this << "::InitializeVideo()"; video_sender_.reset(new VideoSender( cast_environment_, video_config, base::Bind(&CastSenderImpl::OnVideoStatusChange, weak_factory_.GetWeakPtr(), status_change_cb), create_vea_cb, create_video_encode_mem_cb, transport_sender_, base::Bind(&CastSenderImpl::SetTargetPlayoutDelay, weak_factory_.GetWeakPtr()))); if (audio_sender_) { DCHECK(audio_sender_->GetTargetPlayoutDelay() == video_sender_->GetTargetPlayoutDelay()); } } CastSenderImpl::~CastSenderImpl() { VLOG(1) << "CastSenderImpl@" << this << "::~CastSenderImpl()"; } scoped_refptr CastSenderImpl::audio_frame_input() { return audio_frame_input_; } scoped_refptr CastSenderImpl::video_frame_input() { return video_frame_input_; } void CastSenderImpl::SetTargetPlayoutDelay( base::TimeDelta new_target_playout_delay) { VLOG(1) << "CastSenderImpl@" << this << "::SetTargetPlayoutDelay(" << new_target_playout_delay.InMilliseconds() << " ms)"; if (audio_sender_) { audio_sender_->SetTargetPlayoutDelay(new_target_playout_delay); } if (video_sender_) { video_sender_->SetTargetPlayoutDelay(new_target_playout_delay); } } void CastSenderImpl::OnAudioStatusChange( const StatusChangeCallback& status_change_cb, OperationalStatus status) { DCHECK(cast_environment_->CurrentlyOn(CastEnvironment::MAIN)); if (status == STATUS_INITIALIZED && !audio_frame_input_) { audio_frame_input_ = new LocalAudioFrameInput(cast_environment_, audio_sender_->AsWeakPtr()); } status_change_cb.Run(status); } void CastSenderImpl::OnVideoStatusChange( const StatusChangeCallback& status_change_cb, OperationalStatus status) { DCHECK(cast_environment_->CurrentlyOn(CastEnvironment::MAIN)); if (status == STATUS_INITIALIZED && !video_frame_input_) { video_frame_input_ = new LocalVideoFrameInput(cast_environment_, video_sender_->AsWeakPtr()); } status_change_cb.Run(status); } } // namespace cast } // namespace media