OLD | NEW |
---|---|
(Empty) | |
1 /* | |
2 * Copyright 2016 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/api/trackmediainfomap.h" | |
12 | |
13 #include <utility> | |
14 | |
15 namespace webrtc { | |
16 | |
17 namespace { | |
18 | |
19 template<typename T> | |
20 T* CheckedTrackCast(MediaStreamTrackInterface* track); | |
21 | |
22 template<> | |
23 AudioTrackInterface* CheckedTrackCast(MediaStreamTrackInterface* track) { | |
24 RTC_DCHECK_EQ(track->kind(), MediaStreamTrackInterface::kAudioKind); | |
25 return static_cast<AudioTrackInterface*>(track); | |
26 } | |
27 | |
28 template<> | |
29 VideoTrackInterface* CheckedTrackCast(MediaStreamTrackInterface* track) { | |
30 RTC_DCHECK_EQ(track->kind(), MediaStreamTrackInterface::kVideoKind); | |
31 return static_cast<VideoTrackInterface*>(track); | |
32 } | |
33 | |
34 // |T|: |AudioTrackInterface| or |VideoTrackInterface|, must match with | |
35 // |media_type|. | |
36 template<typename T> | |
37 std::map<uint32_t, T*> GetSsrcToTrack( | |
38 cricket::MediaType media_type, | |
39 const std::vector<rtc::scoped_refptr<RtpSenderInterface>>& rtp_senders, | |
40 const std::vector<rtc::scoped_refptr<RtpReceiverInterface>>& | |
41 rtp_receivers) { | |
42 std::map<uint32_t, T*> ssrc_to_track; | |
43 for (const rtc::scoped_refptr<RtpSenderInterface>& rtp_sender : rtp_senders) { | |
44 if (rtp_sender->media_type() != media_type || !rtp_sender->track()) { | |
45 continue; | |
46 } | |
47 // TODO(deadbeef): |ssrc| should be removed in favor of |GetParameters|. | |
48 if (rtp_sender->ssrc() != 0) { | |
49 RTC_DCHECK(ssrc_to_track.find(rtp_sender->ssrc()) == ssrc_to_track.end()); | |
50 ssrc_to_track[rtp_sender->ssrc()] = CheckedTrackCast<T>( | |
51 rtp_sender->track()); | |
52 } | |
53 // TODO(hbos): This uses a proxy to the signaling thread, and our | |
54 // implementation invokes on the worker thread. (This means one thread jump | |
55 // if on signaling thread and two thread jumps if on any other threads). Is | |
56 // there a way to avoid thread jumps per senders (and same for receivers)? | |
57 RtpParameters params = rtp_sender->GetParameters(); | |
Taylor Brandstetter
2017/01/11 22:12:05
Given that currently, GetParameters will just retu
hbos
2017/01/12 11:03:14
Done.
| |
58 for (const RtpEncodingParameters& encoding : params.encodings) { | |
59 if (!encoding.ssrc || *encoding.ssrc == rtp_sender->ssrc()) { | |
60 continue; | |
61 } | |
62 RTC_DCHECK(ssrc_to_track.find(*encoding.ssrc) == ssrc_to_track.end()); | |
63 ssrc_to_track[*encoding.ssrc] = CheckedTrackCast<T>(rtp_sender->track()); | |
64 } | |
65 } | |
66 for (const rtc::scoped_refptr<RtpReceiverInterface>& rtp_receiver : | |
67 rtp_receivers) { | |
68 if (rtp_receiver->media_type() != media_type) { | |
69 continue; | |
70 } | |
71 RTC_DCHECK(rtp_receiver->track()); | |
72 RtpParameters params = rtp_receiver->GetParameters(); | |
73 for (const RtpEncodingParameters& encoding : params.encodings) { | |
74 if (!encoding.ssrc) { | |
75 continue; | |
76 } | |
77 RTC_DCHECK(ssrc_to_track.find(*encoding.ssrc) == ssrc_to_track.end()); | |
78 ssrc_to_track[*encoding.ssrc] = CheckedTrackCast<T>( | |
79 rtp_receiver->track()); | |
80 } | |
81 } | |
82 return ssrc_to_track; | |
83 } | |
84 | |
85 // Template in order to support both the audio and video case. | |
86 template<typename TrackType, typename InfoType> | |
87 TrackType* GetAssociatedTrack( | |
88 const std::map<uint32_t, TrackType*>& ssrc_to_track, | |
89 const InfoType& info) { | |
90 typename std::map<uint32_t, TrackType*>::const_iterator it = | |
91 ssrc_to_track.find(info.ssrc()); | |
92 if (it == ssrc_to_track.end()) { | |
93 return nullptr; | |
94 } | |
95 return it->second; | |
96 } | |
97 | |
98 // Template in order to support both the audio and video case. | |
99 template<typename TrackType, | |
100 typename MediaInfoType, | |
101 typename SenderInfoToTrackType, | |
102 typename LocalTrackToInfosType, | |
103 typename ReceiverInfoToTrackType, | |
104 typename RemoteTrackToInfosType> | |
105 void MapTracksAndInfos( | |
106 const std::map<uint32_t, TrackType*>& ssrc_to_track, | |
107 MediaInfoType* media_info, | |
108 SenderInfoToTrackType* sender_info_to_track, | |
109 LocalTrackToInfosType* local_track_to_infos, | |
110 ReceiverInfoToTrackType* receiver_info_to_track, | |
111 RemoteTrackToInfosType* remote_track_to_info) { | |
112 for (auto& sender_info : media_info->senders) { | |
113 TrackType* associated_track = GetAssociatedTrack<TrackType>( | |
114 ssrc_to_track, sender_info); | |
115 if (associated_track) { | |
116 // One sender is associated with at most one track. | |
117 // One track may be associated with multiple senders. | |
118 (*sender_info_to_track)[&sender_info] = associated_track; | |
119 (*local_track_to_infos)[associated_track].push_back(&sender_info); | |
120 } | |
121 } | |
122 for (auto& receiver_info : media_info->receivers) { | |
123 TrackType* associated_track = GetAssociatedTrack<TrackType>( | |
124 ssrc_to_track, receiver_info); | |
125 if (associated_track) { | |
126 // One receiver is associated with at most one track, which is uniquely | |
127 // associated with that receiver. | |
128 (*receiver_info_to_track)[&receiver_info] = associated_track; | |
129 RTC_DCHECK(remote_track_to_info->find(associated_track) == | |
130 remote_track_to_info->end()); | |
131 (*remote_track_to_info)[associated_track] = &receiver_info; | |
132 } | |
133 } | |
134 } | |
135 | |
136 } // namespace | |
137 | |
138 TrackMediaInfoMap::TrackMediaInfoMap( | |
139 std::unique_ptr<cricket::VoiceMediaInfo> voice_media_info, | |
140 std::unique_ptr<cricket::VideoMediaInfo> video_media_info, | |
141 const std::vector<rtc::scoped_refptr<RtpSenderInterface>>& rtp_senders, | |
142 const std::vector<rtc::scoped_refptr<RtpReceiverInterface>>& rtp_receivers) | |
143 : voice_media_info_(std::move(voice_media_info)), | |
144 video_media_info_(std::move(video_media_info)) { | |
145 if (voice_media_info_) { | |
146 std::map<uint32_t, AudioTrackInterface*> ssrc_to_audio_track = | |
147 GetSsrcToTrack<AudioTrackInterface>( | |
148 cricket::MEDIA_TYPE_AUDIO, rtp_senders, rtp_receivers); | |
149 MapTracksAndInfos<AudioTrackInterface>(ssrc_to_audio_track, | |
150 voice_media_info_.get(), | |
151 &voice_sender_info_to_track_, | |
152 &local_audio_track_to_infos_, | |
153 &voice_receiver_info_to_track_, | |
154 &remote_audio_track_to_info_); | |
155 } | |
156 if (video_media_info_) { | |
157 std::map<uint32_t, VideoTrackInterface*> ssrc_to_video_track = | |
158 GetSsrcToTrack<VideoTrackInterface>( | |
159 cricket::MEDIA_TYPE_VIDEO, rtp_senders, rtp_receivers); | |
160 MapTracksAndInfos<VideoTrackInterface>(ssrc_to_video_track, | |
161 video_media_info_.get(), | |
162 &video_sender_info_to_track_, | |
163 &local_video_track_to_infos_, | |
164 &video_receiver_info_to_track_, | |
165 &remote_video_track_to_info_); | |
166 } | |
167 } | |
168 | |
169 const std::vector<cricket::VoiceSenderInfo*>* | |
170 TrackMediaInfoMap::GetVoiceSenderInfos( | |
171 const AudioTrackInterface& local_audio_track) const { | |
172 auto it = local_audio_track_to_infos_.find(&local_audio_track); | |
173 if (it == local_audio_track_to_infos_.end()) { | |
174 return nullptr; | |
175 } | |
176 return &it->second; | |
177 } | |
178 | |
179 const cricket::VoiceReceiverInfo* TrackMediaInfoMap::GetVoiceReceiverInfo( | |
180 const AudioTrackInterface& remote_audio_track) const { | |
181 auto it = remote_audio_track_to_info_.find(&remote_audio_track); | |
182 if (it == remote_audio_track_to_info_.end()) { | |
183 return nullptr; | |
184 } | |
185 return it->second; | |
186 } | |
187 | |
188 const std::vector<cricket::VideoSenderInfo*>* | |
189 TrackMediaInfoMap::GetVideoSenderInfos( | |
190 const VideoTrackInterface& local_video_track) const { | |
191 auto it = local_video_track_to_infos_.find(&local_video_track); | |
192 if (it == local_video_track_to_infos_.end()) { | |
193 return nullptr; | |
194 } | |
195 return &it->second; | |
196 } | |
197 | |
198 const cricket::VideoReceiverInfo* TrackMediaInfoMap::GetVideoReceiverInfo( | |
199 const VideoTrackInterface& remote_video_track) const { | |
200 auto it =remote_video_track_to_info_.find(&remote_video_track); | |
201 if (it == remote_video_track_to_info_.end()) { | |
202 return nullptr; | |
203 } | |
204 return it->second; | |
205 } | |
206 | |
207 rtc::scoped_refptr<AudioTrackInterface> TrackMediaInfoMap::GetAudioTrack( | |
208 const cricket::VoiceSenderInfo& voice_sender_info) const { | |
209 auto it = voice_sender_info_to_track_.find(&voice_sender_info); | |
210 if (it == voice_sender_info_to_track_.end()) { | |
211 return nullptr; | |
212 } | |
213 return it->second; | |
214 } | |
215 | |
216 rtc::scoped_refptr<AudioTrackInterface> TrackMediaInfoMap::GetAudioTrack( | |
217 const cricket::VoiceReceiverInfo& voice_receiver_info) const { | |
218 auto it = voice_receiver_info_to_track_.find(&voice_receiver_info); | |
219 if (it == voice_receiver_info_to_track_.end()) { | |
220 return nullptr; | |
221 } | |
222 return it->second; | |
223 } | |
224 | |
225 rtc::scoped_refptr<VideoTrackInterface> TrackMediaInfoMap::GetVideoTrack( | |
226 const cricket::VideoSenderInfo& video_sender_info) const { | |
227 auto it = video_sender_info_to_track_.find(&video_sender_info); | |
228 if (it == video_sender_info_to_track_.end()) { | |
229 return nullptr; | |
230 } | |
231 return it->second; | |
232 } | |
233 | |
234 rtc::scoped_refptr<VideoTrackInterface> TrackMediaInfoMap::GetVideoTrack( | |
235 const cricket::VideoReceiverInfo& video_receiver_info) const { | |
236 auto it = video_receiver_info_to_track_.find(&video_receiver_info); | |
237 if (it == video_receiver_info_to_track_.end()) { | |
238 return nullptr; | |
239 } | |
240 return it->second; | |
241 } | |
242 | |
243 } // namespace webrtc | |
OLD | NEW |