blob: 4e8c89b3bf63d7fadef682c722a23a0a36940650 [file] [log] [blame]
deadbeefe814a0d2017-02-25 18:15:09 -08001/*
2 * Copyright 2017 The WebRTC project authors. All Rights Reserved.
3 *
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
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
11#include "webrtc/ortc/ortcrtpreceiveradapter.h"
12
13#include <utility>
14
deadbeefe814a0d2017-02-25 18:15:09 -080015#include "webrtc/media/base/mediaconstants.h"
16#include "webrtc/ortc/rtptransportadapter.h"
Edward Lemurc20978e2017-07-06 19:44:34 +020017#include "webrtc/rtc_base/checks.h"
18#include "webrtc/rtc_base/helpers.h" // For "CreateRandomX".
deadbeefe814a0d2017-02-25 18:15:09 -080019
20namespace {
21
22void FillAudioReceiverParameters(webrtc::RtpParameters* parameters) {
23 for (webrtc::RtpCodecParameters& codec : parameters->codecs) {
24 if (!codec.num_channels) {
25 codec.num_channels = rtc::Optional<int>(1);
26 }
27 }
28}
29
30void FillVideoReceiverParameters(webrtc::RtpParameters* parameters) {
31 for (webrtc::RtpCodecParameters& codec : parameters->codecs) {
32 if (!codec.clock_rate) {
33 codec.clock_rate = rtc::Optional<int>(cricket::kVideoCodecClockrate);
34 }
35 }
36}
37
38} // namespace
39
40namespace webrtc {
41
42BEGIN_OWNED_PROXY_MAP(OrtcRtpReceiver)
43PROXY_SIGNALING_THREAD_DESTRUCTOR()
44PROXY_CONSTMETHOD0(rtc::scoped_refptr<MediaStreamTrackInterface>, GetTrack)
45PROXY_METHOD1(RTCError, SetTransport, RtpTransportInterface*)
46PROXY_CONSTMETHOD0(RtpTransportInterface*, GetTransport)
47PROXY_METHOD1(RTCError, Receive, const RtpParameters&)
48PROXY_CONSTMETHOD0(RtpParameters, GetParameters)
49PROXY_CONSTMETHOD0(cricket::MediaType, GetKind)
50END_PROXY_MAP()
51
52// static
53std::unique_ptr<OrtcRtpReceiverInterface> OrtcRtpReceiverAdapter::CreateProxy(
54 std::unique_ptr<OrtcRtpReceiverAdapter> wrapped_receiver) {
55 RTC_DCHECK(wrapped_receiver);
56 rtc::Thread* signaling =
57 wrapped_receiver->rtp_transport_controller_->signaling_thread();
58 rtc::Thread* worker =
59 wrapped_receiver->rtp_transport_controller_->worker_thread();
60 return OrtcRtpReceiverProxy::Create(signaling, worker,
61 std::move(wrapped_receiver));
62}
63
64OrtcRtpReceiverAdapter::~OrtcRtpReceiverAdapter() {
65 internal_receiver_ = nullptr;
66 SignalDestroyed();
67}
68
69rtc::scoped_refptr<MediaStreamTrackInterface> OrtcRtpReceiverAdapter::GetTrack()
70 const {
71 return internal_receiver_ ? internal_receiver_->track() : nullptr;
72}
73
74RTCError OrtcRtpReceiverAdapter::SetTransport(
75 RtpTransportInterface* transport) {
76 LOG_AND_RETURN_ERROR(
77 RTCErrorType::UNSUPPORTED_OPERATION,
78 "Changing the transport of an RtpReceiver is not yet supported.");
79}
80
81RtpTransportInterface* OrtcRtpReceiverAdapter::GetTransport() const {
82 return transport_;
83}
84
85RTCError OrtcRtpReceiverAdapter::Receive(const RtpParameters& parameters) {
86 RtpParameters filled_parameters = parameters;
87 RTCError err;
88 switch (kind_) {
89 case cricket::MEDIA_TYPE_AUDIO:
90 FillAudioReceiverParameters(&filled_parameters);
91 err = rtp_transport_controller_->ValidateAndApplyAudioReceiverParameters(
92 filled_parameters);
93 if (!err.ok()) {
94 return err;
95 }
96 break;
97 case cricket::MEDIA_TYPE_VIDEO:
98 FillVideoReceiverParameters(&filled_parameters);
99 err = rtp_transport_controller_->ValidateAndApplyVideoReceiverParameters(
100 filled_parameters);
101 if (!err.ok()) {
102 return err;
103 }
104 break;
105 case cricket::MEDIA_TYPE_DATA:
106 RTC_NOTREACHED();
107 return webrtc::RTCError(webrtc::RTCErrorType::INTERNAL_ERROR);
108 }
109 last_applied_parameters_ = filled_parameters;
110
111 // Now that parameters were applied, can create (or recreate) the internal
112 // receiver.
113 //
114 // This is analogous to a PeerConnection creating a receiver after
115 // SetRemoteDescription is successful.
116 MaybeRecreateInternalReceiver();
117 return RTCError::OK();
118}
119
120RtpParameters OrtcRtpReceiverAdapter::GetParameters() const {
121 return last_applied_parameters_;
122}
123
124cricket::MediaType OrtcRtpReceiverAdapter::GetKind() const {
125 return kind_;
126}
127
128OrtcRtpReceiverAdapter::OrtcRtpReceiverAdapter(
129 cricket::MediaType kind,
130 RtpTransportInterface* transport,
131 RtpTransportControllerAdapter* rtp_transport_controller)
132 : kind_(kind),
133 transport_(transport),
134 rtp_transport_controller_(rtp_transport_controller) {}
135
136void OrtcRtpReceiverAdapter::MaybeRecreateInternalReceiver() {
137 if (last_applied_parameters_.encodings.empty()) {
138 internal_receiver_ = nullptr;
139 return;
140 }
141 // An SSRC of 0 is valid; this is used to identify "the default SSRC" (which
142 // is the first one seen by the underlying media engine).
143 uint32_t ssrc = 0;
144 if (last_applied_parameters_.encodings[0].ssrc) {
145 ssrc = *last_applied_parameters_.encodings[0].ssrc;
146 }
147 if (internal_receiver_ && ssrc == internal_receiver_->ssrc()) {
148 // SSRC not changing; nothing to do.
149 return;
150 }
151 internal_receiver_ = nullptr;
152 switch (kind_) {
153 case cricket::MEDIA_TYPE_AUDIO:
154 internal_receiver_ =
155 new AudioRtpReceiver(rtc::CreateRandomUuid(), ssrc,
156 rtp_transport_controller_->voice_channel());
157 break;
158 case cricket::MEDIA_TYPE_VIDEO:
159 internal_receiver_ = new VideoRtpReceiver(
160 rtc::CreateRandomUuid(), rtp_transport_controller_->worker_thread(),
161 ssrc, rtp_transport_controller_->video_channel());
162 break;
163 case cricket::MEDIA_TYPE_DATA:
164 RTC_NOTREACHED();
165 }
166}
167
168} // namespace webrtc