| OLD | NEW |
| 1 /* | 1 /* |
| 2 * Copyright (c) 2013 The WebRTC project authors. All Rights Reserved. | 2 * Copyright (c) 2013 The WebRTC project authors. All Rights Reserved. |
| 3 * | 3 * |
| 4 * Use of this source code is governed by a BSD-style license | 4 * Use of this source code is governed by a BSD-style license |
| 5 * that can be found in the LICENSE file in the root of the source | 5 * that can be found in the LICENSE file in the root of the source |
| 6 * tree. An additional intellectual property rights grant can be found | 6 * tree. An additional intellectual property rights grant can be found |
| 7 * in the file PATENTS. All contributing project authors may | 7 * in the file PATENTS. All contributing project authors may |
| 8 * be found in the AUTHORS file in the root of the source tree. | 8 * be found in the AUTHORS file in the root of the source tree. |
| 9 */ | 9 */ |
| 10 | 10 |
| 11 #include "webrtc/video/video_receive_stream.h" | 11 #include "webrtc/video/video_receive_stream.h" |
| 12 | 12 |
| 13 #include <stdlib.h> | 13 #include <stdlib.h> |
| 14 | 14 |
| 15 #include <set> | 15 #include <set> |
| 16 #include <string> | 16 #include <string> |
| 17 | 17 |
| 18 #include "webrtc/base/checks.h" | 18 #include "webrtc/base/checks.h" |
| 19 #include "webrtc/base/logging.h" | 19 #include "webrtc/base/logging.h" |
| 20 #include "webrtc/common_video/libyuv/include/webrtc_libyuv.h" | 20 #include "webrtc/common_video/libyuv/include/webrtc_libyuv.h" |
| 21 #include "webrtc/modules/congestion_controller/include/congestion_controller.h" | 21 #include "webrtc/modules/congestion_controller/include/congestion_controller.h" |
| 22 #include "webrtc/modules/utility/include/process_thread.h" | 22 #include "webrtc/modules/utility/include/process_thread.h" |
| 23 #include "webrtc/modules/video_coding/include/video_coding.h" | 23 #include "webrtc/modules/video_coding/include/video_coding.h" |
| 24 #include "webrtc/modules/video_coding/utility/ivf_file_writer.h" |
| 24 #include "webrtc/system_wrappers/include/clock.h" | 25 #include "webrtc/system_wrappers/include/clock.h" |
| 25 #include "webrtc/video/call_stats.h" | 26 #include "webrtc/video/call_stats.h" |
| 26 #include "webrtc/video/receive_statistics_proxy.h" | 27 #include "webrtc/video/receive_statistics_proxy.h" |
| 27 #include "webrtc/video/vie_remb.h" | 28 #include "webrtc/video/vie_remb.h" |
| 28 #include "webrtc/video_receive_stream.h" | 29 #include "webrtc/video_receive_stream.h" |
| 29 | 30 |
| 30 namespace webrtc { | 31 namespace webrtc { |
| 31 | 32 |
| 32 static bool UseSendSideBwe(const VideoReceiveStream::Config& config) { | 33 static bool UseSendSideBwe(const VideoReceiveStream::Config& config) { |
| 33 if (!config.rtp.transport_cc) | 34 if (!config.rtp.transport_cc) |
| (...skipping 122 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 156 process_thread_(process_thread), | 157 process_thread_(process_thread), |
| 157 clock_(Clock::GetRealTimeClock()), | 158 clock_(Clock::GetRealTimeClock()), |
| 158 decode_thread_(DecodeThreadFunction, this, "DecodingThread"), | 159 decode_thread_(DecodeThreadFunction, this, "DecodingThread"), |
| 159 congestion_controller_(congestion_controller), | 160 congestion_controller_(congestion_controller), |
| 160 call_stats_(call_stats), | 161 call_stats_(call_stats), |
| 161 remb_(remb), | 162 remb_(remb), |
| 162 vcm_(VideoCodingModule::Create(clock_, | 163 vcm_(VideoCodingModule::Create(clock_, |
| 163 nullptr, | 164 nullptr, |
| 164 nullptr, | 165 nullptr, |
| 165 this, | 166 this, |
| 167 this, |
| 166 this)), | 168 this)), |
| 167 incoming_video_stream_(0, config.disable_prerenderer_smoothing), | 169 incoming_video_stream_(0, config.disable_prerenderer_smoothing), |
| 168 stats_proxy_(config_, clock_), | 170 stats_proxy_(config_, clock_), |
| 169 vie_channel_(&transport_adapter_, | 171 vie_channel_(&transport_adapter_, |
| 170 process_thread, | 172 process_thread, |
| 171 nullptr, | 173 nullptr, |
| 172 vcm_.get(), | 174 vcm_.get(), |
| 173 nullptr, | 175 nullptr, |
| 174 nullptr, | 176 nullptr, |
| 175 nullptr, | 177 nullptr, |
| (...skipping 107 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 283 | 285 |
| 284 VideoCodec codec = CreateDecoderVideoCodec(decoder); | 286 VideoCodec codec = CreateDecoderVideoCodec(decoder); |
| 285 | 287 |
| 286 RTC_CHECK(vie_receiver_->SetReceiveCodec(codec)); | 288 RTC_CHECK(vie_receiver_->SetReceiveCodec(codec)); |
| 287 RTC_CHECK_EQ(VCM_OK, | 289 RTC_CHECK_EQ(VCM_OK, |
| 288 vcm_->RegisterReceiveCodec(&codec, num_cpu_cores, false)); | 290 vcm_->RegisterReceiveCodec(&codec, num_cpu_cores, false)); |
| 289 } | 291 } |
| 290 | 292 |
| 291 vcm_->SetRenderDelay(config.render_delay_ms); | 293 vcm_->SetRenderDelay(config.render_delay_ms); |
| 292 incoming_video_stream_.SetExpectedRenderDelay(config.render_delay_ms); | 294 incoming_video_stream_.SetExpectedRenderDelay(config.render_delay_ms); |
| 293 vcm_->RegisterPreDecodeImageCallback(this); | |
| 294 incoming_video_stream_.SetExternalCallback(this); | 295 incoming_video_stream_.SetExternalCallback(this); |
| 295 vie_channel_.SetIncomingVideoStream(&incoming_video_stream_); | 296 vie_channel_.SetIncomingVideoStream(&incoming_video_stream_); |
| 296 vie_channel_.RegisterPreRenderCallback(this); | 297 vie_channel_.RegisterPreRenderCallback(this); |
| 297 | 298 |
| 298 process_thread_->RegisterModule(vcm_.get()); | 299 process_thread_->RegisterModule(vcm_.get()); |
| 299 process_thread_->RegisterModule(&vie_sync_); | 300 process_thread_->RegisterModule(&vie_sync_); |
| 300 } | 301 } |
| 301 | 302 |
| 302 VideoReceiveStream::~VideoReceiveStream() { | 303 VideoReceiveStream::~VideoReceiveStream() { |
| 303 LOG(LS_INFO) << "~VideoReceiveStream: " << config_.ToString(); | 304 LOG(LS_INFO) << "~VideoReceiveStream: " << config_.ToString(); |
| 304 Stop(); | 305 Stop(); |
| 305 | 306 |
| 306 process_thread_->DeRegisterModule(&vie_sync_); | 307 process_thread_->DeRegisterModule(&vie_sync_); |
| 307 process_thread_->DeRegisterModule(vcm_.get()); | 308 process_thread_->DeRegisterModule(vcm_.get()); |
| 308 | 309 |
| 309 // Deregister external decoders so that they are no longer running during | 310 // Deregister external decoders so that they are no longer running during |
| 310 // destruction. This effectively stops the VCM since the decoder thread is | 311 // destruction. This effectively stops the VCM since the decoder thread is |
| 311 // stopped, the VCM is deregistered and no asynchronous decoder threads are | 312 // stopped, the VCM is deregistered and no asynchronous decoder threads are |
| 312 // running. | 313 // running. |
| 313 for (const Decoder& decoder : config_.decoders) | 314 for (const Decoder& decoder : config_.decoders) |
| 314 vcm_->RegisterExternalDecoder(nullptr, decoder.payload_type); | 315 vcm_->RegisterExternalDecoder(nullptr, decoder.payload_type); |
| 315 | 316 |
| 316 vie_channel_.RegisterPreRenderCallback(nullptr); | 317 vie_channel_.RegisterPreRenderCallback(nullptr); |
| 317 vcm_->RegisterPreDecodeImageCallback(nullptr); | |
| 318 | 318 |
| 319 call_stats_->DeregisterStatsObserver(vie_channel_.GetStatsObserver()); | 319 call_stats_->DeregisterStatsObserver(vie_channel_.GetStatsObserver()); |
| 320 rtp_rtcp_->SetREMBStatus(false); | 320 rtp_rtcp_->SetREMBStatus(false); |
| 321 remb_->RemoveReceiveChannel(rtp_rtcp_); | 321 remb_->RemoveReceiveChannel(rtp_rtcp_); |
| 322 | 322 |
| 323 congestion_controller_->GetRemoteBitrateEstimator(UseSendSideBwe(config_)) | 323 congestion_controller_->GetRemoteBitrateEstimator(UseSendSideBwe(config_)) |
| 324 ->RemoveStream(vie_receiver_->GetRemoteSsrc()); | 324 ->RemoveStream(vie_receiver_->GetRemoteSsrc()); |
| 325 } | 325 } |
| 326 | 326 |
| 327 void VideoReceiveStream::Start() { | 327 void VideoReceiveStream::Start() { |
| (...skipping 71 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 399 int32_t VideoReceiveStream::Encoded( | 399 int32_t VideoReceiveStream::Encoded( |
| 400 const EncodedImage& encoded_image, | 400 const EncodedImage& encoded_image, |
| 401 const CodecSpecificInfo* codec_specific_info, | 401 const CodecSpecificInfo* codec_specific_info, |
| 402 const RTPFragmentationHeader* fragmentation) { | 402 const RTPFragmentationHeader* fragmentation) { |
| 403 stats_proxy_.OnPreDecode(encoded_image, codec_specific_info); | 403 stats_proxy_.OnPreDecode(encoded_image, codec_specific_info); |
| 404 if (config_.pre_decode_callback) { | 404 if (config_.pre_decode_callback) { |
| 405 // TODO(asapersson): Remove EncodedFrameCallbackAdapter. | 405 // TODO(asapersson): Remove EncodedFrameCallbackAdapter. |
| 406 encoded_frame_proxy_.Encoded( | 406 encoded_frame_proxy_.Encoded( |
| 407 encoded_image, codec_specific_info, fragmentation); | 407 encoded_image, codec_specific_info, fragmentation); |
| 408 } | 408 } |
| 409 if (kEnableFrameRecording) { |
| 410 if (!ivf_writer_.get()) { |
| 411 RTC_DCHECK(codec_specific_info); |
| 412 RtpVideoCodecTypes rtp_codec_type; |
| 413 switch (codec_specific_info->codecType) { |
| 414 case kVideoCodecVP8: |
| 415 rtp_codec_type = kRtpVideoVp8; |
| 416 break; |
| 417 case kVideoCodecVP9: |
| 418 rtp_codec_type = kRtpVideoVp9; |
| 419 break; |
| 420 case kVideoCodecH264: |
| 421 rtp_codec_type = kRtpVideoH264; |
| 422 break; |
| 423 default: |
| 424 rtp_codec_type = kRtpVideoNone; |
| 425 RTC_NOTREACHED() << "Unsupported codec " |
| 426 << codec_specific_info->codecType; |
| 427 } |
| 428 std::ostringstream oss; |
| 429 oss << "receive_bitstream_ssrc_" << config_.rtp.remote_ssrc << ".ivf"; |
| 430 ivf_writer_ = IvfFileWriter::Open(oss.str(), rtp_codec_type); |
| 431 } |
| 432 if (ivf_writer_.get()) { |
| 433 bool ok = ivf_writer_->WriteFrame(encoded_image); |
| 434 RTC_DCHECK(ok); |
| 435 } |
| 436 } |
| 437 |
| 409 return 0; | 438 return 0; |
| 410 } | 439 } |
| 411 | 440 |
| 412 void VideoReceiveStream::SignalNetworkState(NetworkState state) { | 441 void VideoReceiveStream::SignalNetworkState(NetworkState state) { |
| 413 rtp_rtcp_->SetRTCPStatus(state == kNetworkUp ? config_.rtp.rtcp_mode | 442 rtp_rtcp_->SetRTCPStatus(state == kNetworkUp ? config_.rtp.rtcp_mode |
| 414 : RtcpMode::kOff); | 443 : RtcpMode::kOff); |
| 415 } | 444 } |
| 416 | 445 |
| 417 bool VideoReceiveStream::DecodeThreadFunction(void* ptr) { | 446 bool VideoReceiveStream::DecodeThreadFunction(void* ptr) { |
| 418 static_cast<VideoReceiveStream*>(ptr)->Decode(); | 447 static_cast<VideoReceiveStream*>(ptr)->Decode(); |
| 419 return true; | 448 return true; |
| 420 } | 449 } |
| 421 | 450 |
| 422 void VideoReceiveStream::Decode() { | 451 void VideoReceiveStream::Decode() { |
| 423 static const int kMaxDecodeWaitTimeMs = 50; | 452 static const int kMaxDecodeWaitTimeMs = 50; |
| 424 vcm_->Decode(kMaxDecodeWaitTimeMs); | 453 vcm_->Decode(kMaxDecodeWaitTimeMs); |
| 425 } | 454 } |
| 426 | 455 |
| 427 void VideoReceiveStream::SendNack( | 456 void VideoReceiveStream::SendNack( |
| 428 const std::vector<uint16_t>& sequence_numbers) { | 457 const std::vector<uint16_t>& sequence_numbers) { |
| 429 rtp_rtcp_->SendNack(sequence_numbers); | 458 rtp_rtcp_->SendNack(sequence_numbers); |
| 430 } | 459 } |
| 431 | 460 |
| 432 void VideoReceiveStream::RequestKeyFrame() { | 461 void VideoReceiveStream::RequestKeyFrame() { |
| 433 rtp_rtcp_->RequestKeyFrame(); | 462 rtp_rtcp_->RequestKeyFrame(); |
| 434 } | 463 } |
| 435 | 464 |
| 436 } // namespace internal | 465 } // namespace internal |
| 437 } // namespace webrtc | 466 } // namespace webrtc |
| OLD | NEW |