• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * libjingle
3  * Copyright 2010 Google Inc.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are met:
7  *
8  *  1. Redistributions of source code must retain the above copyright notice,
9  *     this list of conditions and the following disclaimer.
10  *  2. Redistributions in binary form must reproduce the above copyright notice,
11  *     this list of conditions and the following disclaimer in the documentation
12  *     and/or other materials provided with the distribution.
13  *  3. The name of the author may not be used to endorse or promote products
14  *     derived from this software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
17  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
19  * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
22  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
23  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
24  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
25  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26  */
27 
28 /*
29  * Documentation is in mediamessages.h.
30  */
31 
32 #include "talk/session/media/mediamessages.h"
33 
34 #include "talk/p2p/base/constants.h"
35 #include "talk/p2p/base/parsing.h"
36 #include "talk/session/media/mediasessionclient.h"
37 #include "webrtc/libjingle/xmllite/xmlelement.h"
38 #include "webrtc/base/logging.h"
39 #include "webrtc/base/stringencode.h"
40 
41 namespace cricket {
42 
43 namespace {
44 
45 // NOTE: There is no check here for duplicate streams, so check before
46 // adding.
AddStream(std::vector<StreamParams> * streams,const StreamParams & stream)47 void AddStream(std::vector<StreamParams>* streams, const StreamParams& stream) {
48   streams->push_back(stream);
49 }
50 
ParseSsrc(const std::string & string,uint32 * ssrc)51 bool ParseSsrc(const std::string& string, uint32* ssrc) {
52   return rtc::FromString(string, ssrc);
53 }
54 
55 // Builds a <view> element according to the following spec:
56 // goto/jinglemuc
CreateViewElem(const std::string & name,const std::string & type)57 buzz::XmlElement* CreateViewElem(const std::string& name,
58                                  const std::string& type) {
59   buzz::XmlElement* view_elem =
60       new buzz::XmlElement(QN_JINGLE_DRAFT_VIEW, true);
61   view_elem->AddAttr(QN_NAME, name);
62   view_elem->SetAttr(QN_TYPE, type);
63   return view_elem;
64 }
65 
CreateVideoViewElem(const std::string & content_name,const std::string & type)66 buzz::XmlElement* CreateVideoViewElem(const std::string& content_name,
67                                       const std::string& type) {
68   return CreateViewElem(content_name, type);
69 }
70 
CreateNoneVideoViewElem(const std::string & content_name)71 buzz::XmlElement* CreateNoneVideoViewElem(const std::string& content_name) {
72   return CreateVideoViewElem(content_name, STR_JINGLE_DRAFT_VIEW_TYPE_NONE);
73 }
74 
CreateStaticVideoViewElem(const std::string & content_name,const StaticVideoView & view)75 buzz::XmlElement* CreateStaticVideoViewElem(const std::string& content_name,
76                                             const StaticVideoView& view) {
77   buzz::XmlElement* view_elem =
78       CreateVideoViewElem(content_name, STR_JINGLE_DRAFT_VIEW_TYPE_STATIC);
79   AddXmlAttr(view_elem, QN_SSRC, view.selector.ssrc);
80 
81   buzz::XmlElement* params_elem = new buzz::XmlElement(QN_JINGLE_DRAFT_PARAMS);
82   AddXmlAttr(params_elem, QN_WIDTH, view.width);
83   AddXmlAttr(params_elem, QN_HEIGHT, view.height);
84   AddXmlAttr(params_elem, QN_FRAMERATE, view.framerate);
85   AddXmlAttr(params_elem, QN_PREFERENCE, view.preference);
86   view_elem->AddElement(params_elem);
87 
88   return view_elem;
89 }
90 
91 }  //  namespace
92 
GetAudioStream(const StreamSelector & selector,StreamParams * stream)93 bool MediaStreams::GetAudioStream(
94     const StreamSelector& selector, StreamParams* stream) {
95   return GetStream(audio_, selector, stream);
96 }
97 
GetVideoStream(const StreamSelector & selector,StreamParams * stream)98 bool MediaStreams::GetVideoStream(
99     const StreamSelector& selector, StreamParams* stream) {
100   return GetStream(video_, selector, stream);
101 }
102 
GetDataStream(const StreamSelector & selector,StreamParams * stream)103 bool MediaStreams::GetDataStream(
104     const StreamSelector& selector, StreamParams* stream) {
105   return GetStream(data_, selector, stream);
106 }
107 
CopyFrom(const MediaStreams & streams)108 void MediaStreams::CopyFrom(const MediaStreams& streams) {
109   audio_ = streams.audio_;
110   video_ = streams.video_;
111   data_ = streams.data_;
112 }
113 
AddAudioStream(const StreamParams & stream)114 void MediaStreams::AddAudioStream(const StreamParams& stream) {
115   AddStream(&audio_, stream);
116 }
117 
AddVideoStream(const StreamParams & stream)118 void MediaStreams::AddVideoStream(const StreamParams& stream) {
119   AddStream(&video_, stream);
120 }
121 
AddDataStream(const StreamParams & stream)122 void MediaStreams::AddDataStream(const StreamParams& stream) {
123   AddStream(&data_, stream);
124 }
125 
RemoveAudioStream(const StreamSelector & selector)126 bool MediaStreams::RemoveAudioStream(
127     const StreamSelector& selector) {
128   return RemoveStream(&audio_, selector);
129 }
130 
RemoveVideoStream(const StreamSelector & selector)131 bool MediaStreams::RemoveVideoStream(
132     const StreamSelector& selector) {
133   return RemoveStream(&video_, selector);
134 }
135 
RemoveDataStream(const StreamSelector & selector)136 bool MediaStreams::RemoveDataStream(
137     const StreamSelector& selector) {
138   return RemoveStream(&data_, selector);
139 }
140 
IsJingleViewRequest(const buzz::XmlElement * action_elem)141 bool IsJingleViewRequest(const buzz::XmlElement* action_elem) {
142   return action_elem->FirstNamed(QN_JINGLE_DRAFT_VIEW) != NULL;
143 }
144 
ParseStaticVideoView(const buzz::XmlElement * view_elem,StaticVideoView * view,ParseError * error)145 bool ParseStaticVideoView(const buzz::XmlElement* view_elem,
146                           StaticVideoView* view,
147                           ParseError* error) {
148   uint32 ssrc;
149   if (!ParseSsrc(view_elem->Attr(QN_SSRC), &ssrc)) {
150     return BadParse("Invalid or missing view ssrc.", error);
151   }
152   view->selector = StreamSelector(ssrc);
153 
154   const buzz::XmlElement* params_elem =
155       view_elem->FirstNamed(QN_JINGLE_DRAFT_PARAMS);
156   if (params_elem) {
157     view->width = GetXmlAttr(params_elem, QN_WIDTH, 0);
158     view->height = GetXmlAttr(params_elem, QN_HEIGHT, 0);
159     view->framerate = GetXmlAttr(params_elem, QN_FRAMERATE, 0);
160     view->preference = GetXmlAttr(params_elem, QN_PREFERENCE, 0);
161   } else {
162     return BadParse("Missing view params.", error);
163   }
164 
165   return true;
166 }
167 
ParseJingleViewRequest(const buzz::XmlElement * action_elem,ViewRequest * view_request,ParseError * error)168 bool ParseJingleViewRequest(const buzz::XmlElement* action_elem,
169                             ViewRequest* view_request,
170                             ParseError* error) {
171   for (const buzz::XmlElement* view_elem =
172            action_elem->FirstNamed(QN_JINGLE_DRAFT_VIEW);
173        view_elem != NULL;
174        view_elem = view_elem->NextNamed(QN_JINGLE_DRAFT_VIEW)) {
175     std::string type = view_elem->Attr(QN_TYPE);
176     if (STR_JINGLE_DRAFT_VIEW_TYPE_NONE == type) {
177       view_request->static_video_views.clear();
178       return true;
179     } else if (STR_JINGLE_DRAFT_VIEW_TYPE_STATIC == type) {
180       StaticVideoView static_video_view(StreamSelector(0), 0, 0, 0);
181       if (!ParseStaticVideoView(view_elem, &static_video_view, error)) {
182         return false;
183       }
184       view_request->static_video_views.push_back(static_video_view);
185     } else {
186       LOG(LS_INFO) << "Ingnoring unknown view type: " << type;
187     }
188   }
189   return true;
190 }
191 
WriteJingleViewRequest(const std::string & content_name,const ViewRequest & request,XmlElements * elems,WriteError * error)192 bool WriteJingleViewRequest(const std::string& content_name,
193                             const ViewRequest& request,
194                             XmlElements* elems,
195                             WriteError* error) {
196   if (request.static_video_views.empty()) {
197     elems->push_back(CreateNoneVideoViewElem(content_name));
198   } else {
199     for (StaticVideoViews::const_iterator view =
200              request.static_video_views.begin();
201          view != request.static_video_views.end(); ++view) {
202       elems->push_back(CreateStaticVideoViewElem(content_name, *view));
203     }
204   }
205   return true;
206 }
207 
ParseSsrcAsLegacyStream(const buzz::XmlElement * desc_elem,std::vector<StreamParams> * streams,ParseError * error)208 bool ParseSsrcAsLegacyStream(const buzz::XmlElement* desc_elem,
209                              std::vector<StreamParams>* streams,
210                              ParseError* error) {
211   const std::string ssrc_str = desc_elem->Attr(QN_SSRC);
212   if (!ssrc_str.empty()) {
213     uint32 ssrc;
214     if (!ParseSsrc(ssrc_str, &ssrc)) {
215       return BadParse("Missing or invalid ssrc.", error);
216     }
217 
218     streams->push_back(StreamParams::CreateLegacy(ssrc));
219   }
220   return true;
221 }
222 
ParseSsrcs(const buzz::XmlElement * parent_elem,std::vector<uint32> * ssrcs,ParseError * error)223 bool ParseSsrcs(const buzz::XmlElement* parent_elem,
224                 std::vector<uint32>* ssrcs,
225                 ParseError* error) {
226   for (const buzz::XmlElement* ssrc_elem =
227            parent_elem->FirstNamed(QN_JINGLE_DRAFT_SSRC);
228        ssrc_elem != NULL;
229        ssrc_elem = ssrc_elem->NextNamed(QN_JINGLE_DRAFT_SSRC)) {
230     uint32 ssrc;
231     if (!ParseSsrc(ssrc_elem->BodyText(), &ssrc)) {
232       return BadParse("Missing or invalid ssrc.", error);
233     }
234 
235     ssrcs->push_back(ssrc);
236   }
237   return true;
238 }
239 
ParseSsrcGroups(const buzz::XmlElement * parent_elem,std::vector<SsrcGroup> * ssrc_groups,ParseError * error)240 bool ParseSsrcGroups(const buzz::XmlElement* parent_elem,
241                      std::vector<SsrcGroup>* ssrc_groups,
242                      ParseError* error) {
243   for (const buzz::XmlElement* group_elem =
244            parent_elem->FirstNamed(QN_JINGLE_DRAFT_SSRC_GROUP);
245        group_elem != NULL;
246        group_elem = group_elem->NextNamed(QN_JINGLE_DRAFT_SSRC_GROUP)) {
247     std::string semantics = group_elem->Attr(QN_SEMANTICS);
248     std::vector<uint32> ssrcs;
249     if (!ParseSsrcs(group_elem, &ssrcs, error)) {
250       return false;
251     }
252     ssrc_groups->push_back(SsrcGroup(semantics, ssrcs));
253   }
254   return true;
255 }
256 
ParseJingleStream(const buzz::XmlElement * stream_elem,std::vector<StreamParams> * streams,ParseError * error)257 bool ParseJingleStream(const buzz::XmlElement* stream_elem,
258                        std::vector<StreamParams>* streams,
259                        ParseError* error) {
260   StreamParams stream;
261   // We treat the nick as a stream groupid.
262   stream.groupid = stream_elem->Attr(QN_NICK);
263   stream.id = stream_elem->Attr(QN_NAME);
264   stream.type = stream_elem->Attr(QN_TYPE);
265   stream.display = stream_elem->Attr(QN_DISPLAY);
266   stream.cname = stream_elem->Attr(QN_CNAME);
267   if (!ParseSsrcs(stream_elem, &(stream.ssrcs), error)) {
268     return false;
269   }
270   std::vector<SsrcGroup> ssrc_groups;
271   if (!ParseSsrcGroups(stream_elem, &(stream.ssrc_groups), error)) {
272     return false;
273   }
274   streams->push_back(stream);
275   return true;
276 }
277 
ParseJingleRtpHeaderExtensions(const buzz::XmlElement * parent_elem,std::vector<RtpHeaderExtension> * hdrexts,ParseError * error)278 bool ParseJingleRtpHeaderExtensions(const buzz::XmlElement* parent_elem,
279                                     std::vector<RtpHeaderExtension>* hdrexts,
280                                     ParseError* error) {
281   for (const buzz::XmlElement* hdrext_elem =
282            parent_elem->FirstNamed(QN_JINGLE_RTP_HDREXT);
283        hdrext_elem != NULL;
284        hdrext_elem = hdrext_elem->NextNamed(QN_JINGLE_RTP_HDREXT)) {
285     std::string uri = hdrext_elem->Attr(QN_URI);
286     int id = GetXmlAttr(hdrext_elem, QN_ID, 0);
287     if (id <= 0) {
288       return BadParse("Invalid RTP header extension id.", error);
289     }
290     hdrexts->push_back(RtpHeaderExtension(uri, id));
291   }
292   return true;
293 }
294 
HasJingleStreams(const buzz::XmlElement * desc_elem)295 bool HasJingleStreams(const buzz::XmlElement* desc_elem) {
296   const buzz::XmlElement* streams_elem =
297       desc_elem->FirstNamed(QN_JINGLE_DRAFT_STREAMS);
298   return (streams_elem != NULL);
299 }
300 
ParseJingleStreams(const buzz::XmlElement * desc_elem,std::vector<StreamParams> * streams,ParseError * error)301 bool ParseJingleStreams(const buzz::XmlElement* desc_elem,
302                         std::vector<StreamParams>* streams,
303                         ParseError* error) {
304   const buzz::XmlElement* streams_elem =
305       desc_elem->FirstNamed(QN_JINGLE_DRAFT_STREAMS);
306   if (streams_elem == NULL) {
307     return BadParse("Missing streams element.", error);
308   }
309   for (const buzz::XmlElement* stream_elem =
310            streams_elem->FirstNamed(QN_JINGLE_DRAFT_STREAM);
311        stream_elem != NULL;
312        stream_elem = stream_elem->NextNamed(QN_JINGLE_DRAFT_STREAM)) {
313     if (!ParseJingleStream(stream_elem, streams, error)) {
314       return false;
315     }
316   }
317   return true;
318 }
319 
WriteSsrcs(const std::vector<uint32> & ssrcs,buzz::XmlElement * parent_elem)320 void WriteSsrcs(const std::vector<uint32>& ssrcs,
321                 buzz::XmlElement* parent_elem) {
322   for (std::vector<uint32>::const_iterator ssrc = ssrcs.begin();
323        ssrc != ssrcs.end(); ++ssrc) {
324     buzz::XmlElement* ssrc_elem =
325         new buzz::XmlElement(QN_JINGLE_DRAFT_SSRC, false);
326     SetXmlBody(ssrc_elem, *ssrc);
327 
328     parent_elem->AddElement(ssrc_elem);
329   }
330 }
331 
WriteSsrcGroups(const std::vector<SsrcGroup> & groups,buzz::XmlElement * parent_elem)332 void WriteSsrcGroups(const std::vector<SsrcGroup>& groups,
333                      buzz::XmlElement* parent_elem) {
334   for (std::vector<SsrcGroup>::const_iterator group = groups.begin();
335        group != groups.end(); ++group) {
336     buzz::XmlElement* group_elem =
337         new buzz::XmlElement(QN_JINGLE_DRAFT_SSRC_GROUP, false);
338     AddXmlAttrIfNonEmpty(group_elem, QN_SEMANTICS, group->semantics);
339     WriteSsrcs(group->ssrcs, group_elem);
340 
341     parent_elem->AddElement(group_elem);
342   }
343 }
344 
WriteJingleStream(const StreamParams & stream,buzz::XmlElement * parent_elem)345 void WriteJingleStream(const StreamParams& stream,
346                        buzz::XmlElement* parent_elem) {
347   buzz::XmlElement* stream_elem =
348       new buzz::XmlElement(QN_JINGLE_DRAFT_STREAM, false);
349   // We treat the nick as a stream groupid.
350   AddXmlAttrIfNonEmpty(stream_elem, QN_NICK, stream.groupid);
351   AddXmlAttrIfNonEmpty(stream_elem, QN_NAME, stream.id);
352   AddXmlAttrIfNonEmpty(stream_elem, QN_TYPE, stream.type);
353   AddXmlAttrIfNonEmpty(stream_elem, QN_DISPLAY, stream.display);
354   AddXmlAttrIfNonEmpty(stream_elem, QN_CNAME, stream.cname);
355   WriteSsrcs(stream.ssrcs, stream_elem);
356   WriteSsrcGroups(stream.ssrc_groups, stream_elem);
357 
358   parent_elem->AddElement(stream_elem);
359 }
360 
WriteJingleStreams(const std::vector<StreamParams> & streams,buzz::XmlElement * parent_elem)361 void WriteJingleStreams(const std::vector<StreamParams>& streams,
362                         buzz::XmlElement* parent_elem) {
363   buzz::XmlElement* streams_elem =
364       new buzz::XmlElement(QN_JINGLE_DRAFT_STREAMS, true);
365   for (std::vector<StreamParams>::const_iterator stream = streams.begin();
366        stream != streams.end(); ++stream) {
367     WriteJingleStream(*stream, streams_elem);
368   }
369 
370   parent_elem->AddElement(streams_elem);
371 }
372 
WriteJingleRtpHeaderExtensions(const std::vector<RtpHeaderExtension> & hdrexts,buzz::XmlElement * parent_elem)373 void WriteJingleRtpHeaderExtensions(
374     const std::vector<RtpHeaderExtension>& hdrexts,
375     buzz::XmlElement* parent_elem) {
376   for (std::vector<RtpHeaderExtension>::const_iterator hdrext = hdrexts.begin();
377        hdrext != hdrexts.end(); ++hdrext) {
378     buzz::XmlElement* hdrext_elem =
379       new buzz::XmlElement(QN_JINGLE_RTP_HDREXT, false);
380     AddXmlAttr(hdrext_elem, QN_URI, hdrext->uri);
381     AddXmlAttr(hdrext_elem, QN_ID, hdrext->id);
382     parent_elem->AddElement(hdrext_elem);
383   }
384 }
385 
386 
387 }  // namespace cricket
388