Chromium Code Reviews| Index: webrtc/ortc/rtpreceivershim.cc |
| diff --git a/webrtc/ortc/rtpreceivershim.cc b/webrtc/ortc/rtpreceivershim.cc |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..a4f40db37699ae8381c62392fbfd789a75541276 |
| --- /dev/null |
| +++ b/webrtc/ortc/rtpreceivershim.cc |
| @@ -0,0 +1,193 @@ |
| +/* |
| + * Copyright 2017 The WebRTC project authors. All Rights Reserved. |
| + * |
| + * Use of this source code is governed by a BSD-style license |
| + * that can be found in the LICENSE file in the root of the source |
| + * tree. An additional intellectual property rights grant can be found |
| + * in the file PATENTS. All contributing project authors may |
| + * be found in the AUTHORS file in the root of the source tree. |
| + */ |
| + |
| +#include "webrtc/ortc/rtpreceivershim.h" |
| + |
| +#include "webrtc/base/checks.h" |
| +#include "webrtc/base/helpers.h" // For "CreateRandomX". |
| + |
| +namespace { |
| + |
| +static const int kDefaultVideoClockrate = 90000; |
|
pthatcher1
2017/02/10 22:36:52
That's defined in mediaconstants.h as kVideoCodecC
|
| + |
| +void FillAudioReceiverParameters(webrtc::RtpParameters* parameters) { |
| + for (webrtc::RtpCodecParameters& codec : parameters->codecs) { |
| + if (!codec.num_channels) { |
| + codec.num_channels = rtc::Optional<int>(1); |
| + } |
| + } |
| +} |
| + |
| +void FillVideoReceiverParameters(webrtc::RtpParameters* parameters) { |
| + for (webrtc::RtpCodecParameters& codec : parameters->codecs) { |
| + if (!codec.clock_rate) { |
| + codec.clock_rate = rtc::Optional<int>(kDefaultVideoClockrate); |
| + } |
| + } |
| +} |
| + |
| +} // namespace |
| + |
| +namespace webrtc { |
| + |
| +BEGIN_OWNED_PROXY_MAP(OrtcRtpReceiver) |
| +PROXY_SIGNALING_THREAD_DESTRUCTOR() |
| +PROXY_CONSTMETHOD0(rtc::scoped_refptr<MediaStreamTrackInterface>, GetTrack) |
| +PROXY_METHOD1(RTCError, SetTransport, RtpTransportInterface*) |
| +PROXY_CONSTMETHOD0(RtpTransportInterface*, GetTransport) |
| +PROXY_METHOD1(RTCError, Receive, const RtpParameters&) |
| +PROXY_CONSTMETHOD0(RtpParameters, GetParameters) |
| +PROXY_CONSTMETHOD0(cricket::MediaType, GetKind) |
| +END_PROXY_MAP() |
| + |
| +// static |
| +RTCErrorOr<std::unique_ptr<OrtcRtpReceiverInterface>> |
| +RtpReceiverShim::CreateProxied(cricket::MediaType kind, |
| + RtpTransportShim* transport) { |
| + RTC_DCHECK(transport); |
| + RtpTransportControllerShim* rtp_transport_controller = |
| + transport->rtp_transport_controller(); |
| + // Call "attach" method to ensure more than one receiver of the same type |
| + // isn't attached to the same transport. |
| + RTCError err; |
| + switch (kind) { |
| + case cricket::MEDIA_TYPE_AUDIO: |
| + err = rtp_transport_controller->AttachAudioReceiver(transport); |
| + break; |
| + case cricket::MEDIA_TYPE_VIDEO: |
| + err = rtp_transport_controller->AttachVideoReceiver(transport); |
| + break; |
| + case cricket::MEDIA_TYPE_DATA: |
| + RTC_NOTREACHED(); |
|
pthatcher1
2017/02/10 22:36:52
Should this be an RTC_DCHECK()?
Taylor Brandstetter
2017/02/14 06:55:05
What would be in the RTC_DCHECK()?
|
| + } |
| + if (!err.ok()) { |
| + return err; |
| + } |
| + |
| + return OrtcRtpReceiverProxy::Create( |
| + rtp_transport_controller->signaling_thread(), |
| + rtp_transport_controller->worker_thread(), |
| + new RtpReceiverShim(kind, transport, rtp_transport_controller)); |
| +} |
| + |
| +RtpReceiverShim::~RtpReceiverShim() { |
| + internal_receiver_ = nullptr; |
| + // Need to detach from transport (was attached in Create method). |
| + switch (kind_) { |
| + case cricket::MEDIA_TYPE_AUDIO: |
| + rtp_transport_controller_->DetachAudioReceiver(); |
| + break; |
| + case cricket::MEDIA_TYPE_VIDEO: |
| + rtp_transport_controller_->DetachVideoReceiver(); |
| + break; |
| + case cricket::MEDIA_TYPE_DATA: |
| + RTC_NOTREACHED(); |
|
pthatcher1
2017/02/10 22:36:52
Should this be an RTC_DCHECK()?
Taylor Brandstetter
2017/02/14 06:55:05
What would be in the RTC_DCHECK()?
Taylor Brandstetter
2017/02/14 06:55:05
What would be in the RTC_DCHECK()?
|
| + } |
| +} |
| + |
| +rtc::scoped_refptr<MediaStreamTrackInterface> RtpReceiverShim::GetTrack() |
| + const { |
| + return internal_receiver_ ? internal_receiver_->track() : nullptr; |
| +} |
| + |
| +RTCError RtpReceiverShim::SetTransport(RtpTransportInterface* transport) { |
| + LOG(LS_ERROR) << "Changing the transport of an RtpReceiver is not yet " |
| + << "supported."; |
| + return RTCError(RTCErrorType::UNSUPPORTED_PARAMETER); |
| +} |
| + |
| +RtpTransportInterface* RtpReceiverShim::GetTransport() const { |
| + return transport_; |
| +} |
| + |
| +RTCError RtpReceiverShim::Receive(const RtpParameters& parameters) { |
| + RtpParameters filled_parameters = parameters; |
| + RTCError err; |
| + switch (kind_) { |
| + case cricket::MEDIA_TYPE_AUDIO: |
| + FillAudioReceiverParameters(&filled_parameters); |
| + err = rtp_transport_controller_->ValidateAndApplyAudioReceiverParameters( |
| + filled_parameters); |
| + if (!err.ok()) { |
| + return err; |
| + } |
| + break; |
| + case cricket::MEDIA_TYPE_VIDEO: |
| + FillVideoReceiverParameters(&filled_parameters); |
| + err = rtp_transport_controller_->ValidateAndApplyVideoReceiverParameters( |
| + filled_parameters); |
| + if (!err.ok()) { |
| + return err; |
| + } |
| + break; |
| + case cricket::MEDIA_TYPE_DATA: |
| + RTC_NOTREACHED(); |
| + return webrtc::RTCError(webrtc::RTCErrorType::INTERNAL_ERROR); |
| + } |
| + last_applied_parameters_ = filled_parameters; |
| + |
| + // Now that parameters were applied, can create (or recreate) the internal |
| + // receiver. |
| + // |
| + // This is analogous to a PeerConnection creating a receiver after |
| + // SetRemoteDescription is successful. |
| + MaybeRecreateInternalReceiver(); |
| + return RTCError(); |
| +} |
| + |
| +RtpParameters RtpReceiverShim::GetParameters() const { |
| + return last_applied_parameters_; |
| +} |
| + |
| +cricket::MediaType RtpReceiverShim::GetKind() const { |
| + return cricket::MediaTypeFromString(GetTrack()->kind()); |
| +} |
| + |
| +RtpReceiverShim::RtpReceiverShim( |
| + cricket::MediaType kind, |
| + RtpTransportShim* transport, |
| + RtpTransportControllerShim* rtp_transport_controller) |
| + : kind_(kind), |
| + transport_(transport), |
| + rtp_transport_controller_(rtp_transport_controller) {} |
| + |
| +void RtpReceiverShim::MaybeRecreateInternalReceiver() { |
| + if (last_applied_parameters_.encodings.empty()) { |
| + internal_receiver_ = nullptr; |
| + return; |
| + } |
| + // An SSRC of 0 is valid; this is used to identify "the default SSRC" (which |
| + // is the first one seen by the underlying media engine). |
| + uint32_t ssrc = 0; |
| + if (last_applied_parameters_.encodings[0].ssrc) { |
| + ssrc = *last_applied_parameters_.encodings[0].ssrc; |
| + } |
| + if (internal_receiver_ && ssrc == internal_receiver_->ssrc()) { |
| + // SSRC not changing; nothing to do. |
| + return; |
| + } |
| + internal_receiver_ = nullptr; |
| + switch (kind_) { |
| + case cricket::MEDIA_TYPE_AUDIO: |
| + internal_receiver_ = |
| + new AudioRtpReceiver(rtc::CreateRandomUuid(), ssrc, |
| + rtp_transport_controller_->voice_channel()); |
| + break; |
| + case cricket::MEDIA_TYPE_VIDEO: |
| + internal_receiver_ = new VideoRtpReceiver( |
| + rtc::CreateRandomUuid(), rtp_transport_controller_->worker_thread(), |
| + ssrc, rtp_transport_controller_->video_channel()); |
| + break; |
| + case cricket::MEDIA_TYPE_DATA: |
| + RTC_NOTREACHED(); |
| + } |
| +} |
| + |
| +} // namespace webrtc |