Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(498)

Side by Side Diff: webrtc/pc/webrtcsdp.cc

Issue 2742903002: Parse the connection data in SDP (c= line). (Closed)
Patch Set: Move the GetDefaultDestination logic to JsepSessionDescription. Created 3 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 /* 1 /*
2 * Copyright 2011 The WebRTC project authors. All Rights Reserved. 2 * Copyright 2011 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
(...skipping 231 matching lines...) Expand 10 before | Expand all | Expand 10 after
242 static void BuildRtpMap(const MediaContentDescription* media_desc, 242 static void BuildRtpMap(const MediaContentDescription* media_desc,
243 const MediaType media_type, 243 const MediaType media_type,
244 std::string* message); 244 std::string* message);
245 static void BuildCandidate(const std::vector<Candidate>& candidates, 245 static void BuildCandidate(const std::vector<Candidate>& candidates,
246 bool include_ufrag, 246 bool include_ufrag,
247 std::string* message); 247 std::string* message);
248 static void BuildIceOptions(const std::vector<std::string>& transport_options, 248 static void BuildIceOptions(const std::vector<std::string>& transport_options,
249 std::string* message); 249 std::string* message);
250 static bool IsRtp(const std::string& protocol); 250 static bool IsRtp(const std::string& protocol);
251 static bool IsDtlsSctp(const std::string& protocol); 251 static bool IsDtlsSctp(const std::string& protocol);
252 static bool ParseSessionDescription(const std::string& message, size_t* pos, 252 static bool ParseSessionDescription(const std::string& message,
253 size_t* pos,
253 std::string* session_id, 254 std::string* session_id,
254 std::string* session_version, 255 std::string* session_version,
255 TransportDescription* session_td, 256 TransportDescription* session_td,
256 RtpHeaderExtensions* session_extmaps, 257 RtpHeaderExtensions* session_extmaps,
258 rtc::SocketAddress* connection_addr,
257 cricket::SessionDescription* desc, 259 cricket::SessionDescription* desc,
258 SdpParseError* error); 260 SdpParseError* error);
259 static bool ParseGroupAttribute(const std::string& line, 261 static bool ParseGroupAttribute(const std::string& line,
260 cricket::SessionDescription* desc, 262 cricket::SessionDescription* desc,
261 SdpParseError* error); 263 SdpParseError* error);
262 static bool ParseMediaDescription( 264 static bool ParseMediaDescription(
263 const std::string& message, 265 const std::string& message,
264 const TransportDescription& session_td, 266 const TransportDescription& session_td,
265 const RtpHeaderExtensions& session_extmaps, 267 const RtpHeaderExtensions& session_extmaps,
266 size_t* pos, cricket::SessionDescription* desc, 268 size_t* pos,
269 const rtc::SocketAddress& session_connection_addr,
270 cricket::SessionDescription* desc,
267 std::vector<JsepIceCandidate*>* candidates, 271 std::vector<JsepIceCandidate*>* candidates,
268 SdpParseError* error); 272 SdpParseError* error);
269 static bool ParseContent(const std::string& message, 273 static bool ParseContent(const std::string& message,
270 const MediaType media_type, 274 const MediaType media_type,
271 int mline_index, 275 int mline_index,
276 int port,
272 const std::string& protocol, 277 const std::string& protocol,
273 const std::vector<int>& payload_types, 278 const std::vector<int>& payload_types,
274 size_t* pos, 279 size_t* pos,
275 std::string* content_name, 280 std::string* content_name,
276 bool* bundle_only, 281 bool* bundle_only,
277 MediaContentDescription* media_desc, 282 MediaContentDescription* media_desc,
278 TransportDescription* transport, 283 TransportDescription* transport,
279 std::vector<JsepIceCandidate*>* candidates, 284 std::vector<JsepIceCandidate*>* candidates,
280 SdpParseError* error); 285 SdpParseError* error);
281 static bool ParseSsrcAttribute(const std::string& line, 286 static bool ParseSsrcAttribute(const std::string& line,
(...skipping 424 matching lines...) Expand 10 before | Expand all | Expand 10 after
706 } else if (family == AF_INET6) { 711 } else if (family == AF_INET6) {
707 addr_type->assign(kConnectionIpv6Addrtype); 712 addr_type->assign(kConnectionIpv6Addrtype);
708 } 713 }
709 current_preference = preference; 714 current_preference = preference;
710 current_family = family; 715 current_family = family;
711 *port = it->address().PortAsString(); 716 *port = it->address().PortAsString();
712 *ip = it->address().ipaddr().ToString(); 717 *ip = it->address().ipaddr().ToString();
713 } 718 }
714 } 719 }
715 720
716 // Update |mline|'s default destination and append a c line after it. 721 // Update |mline|'s default destination.
717 static void UpdateMediaDefaultDestination( 722 static void UpdateMediaDefaultDestination(
Taylor Brandstetter 2017/03/17 00:20:42 Since this is now done in JsepSessionDescription,
Zhi Huang 2017/03/17 23:42:06 You are right. I should remove this.
718 const std::vector<Candidate>& candidates, 723 const std::vector<Candidate>& candidates,
719 const std::string& mline, 724 const std::string& mline,
720 std::string* message) { 725 std::string* message) {
721 std::string new_lines; 726 std::string new_lines;
722 AddLine(mline, &new_lines); 727 AddLine(mline, &new_lines);
723 // RFC 4566 728 // RFC 4566
724 // m=<media> <port> <proto> <fmt> ... 729 // m=<media> <port> <proto> <fmt> ...
725 std::vector<std::string> fields; 730 std::vector<std::string> fields;
726 rtc::split(mline, kSdpDelimiterSpace, &fields); 731 rtc::split(mline, kSdpDelimiterSpace, &fields);
727 if (fields.size() < 3) { 732 if (fields.size() < 3) {
(...skipping 10 matching lines...) Expand all
738 // destination for media. For streams based on RTP, this is done by 743 // destination for media. For streams based on RTP, this is done by
739 // placing the IP address and port of the RTP candidate into the c and m 744 // placing the IP address and port of the RTP candidate into the c and m
740 // lines, respectively. 745 // lines, respectively.
741 // Update the port in the m line. 746 // Update the port in the m line.
742 // If this is a m-line with port equal to 0, we don't change it. 747 // If this is a m-line with port equal to 0, we don't change it.
743 if (fields[1] != kMediaPortRejected) { 748 if (fields[1] != kMediaPortRejected) {
744 new_lines.replace(fields[0].size() + 1, 749 new_lines.replace(fields[0].size() + 1,
745 fields[1].size(), 750 fields[1].size(),
746 rtp_port); 751 rtp_port);
747 } 752 }
748 // Add the c line.
749 // RFC 4566
750 // c=<nettype> <addrtype> <connection-address>
751 InitLine(kLineTypeConnection, kConnectionNettype, &os);
752 os << " " << addr_type << " " << rtp_ip;
753 AddLine(os.str(), &new_lines);
754 message->append(new_lines); 753 message->append(new_lines);
755 } 754 }
756 755
757 // Gets "a=rtcp" line if found default RTCP candidate from |candidates|. 756 // Gets "a=rtcp" line if found default RTCP candidate from |candidates|.
758 static std::string GetRtcpLine(const std::vector<Candidate>& candidates) { 757 static std::string GetRtcpLine(const std::vector<Candidate>& candidates) {
759 std::string rtcp_line, rtcp_port, rtcp_ip, addr_type; 758 std::string rtcp_line, rtcp_port, rtcp_ip, addr_type;
760 GetDefaultDestination(candidates, ICE_CANDIDATE_COMPONENT_RTCP, 759 GetDefaultDestination(candidates, ICE_CANDIDATE_COMPONENT_RTCP,
761 &rtcp_port, &rtcp_ip, &addr_type); 760 &rtcp_port, &rtcp_ip, &addr_type);
762 // Found default RTCP candidate. 761 // Found default RTCP candidate.
763 // RFC 5245 762 // RFC 5245
(...skipping 131 matching lines...) Expand 10 before | Expand all | Expand 10 after
895 return message; 894 return message;
896 } 895 }
897 896
898 bool SdpDeserialize(const std::string& message, 897 bool SdpDeserialize(const std::string& message,
899 JsepSessionDescription* jdesc, 898 JsepSessionDescription* jdesc,
900 SdpParseError* error) { 899 SdpParseError* error) {
901 std::string session_id; 900 std::string session_id;
902 std::string session_version; 901 std::string session_version;
903 TransportDescription session_td("", ""); 902 TransportDescription session_td("", "");
904 RtpHeaderExtensions session_extmaps; 903 RtpHeaderExtensions session_extmaps;
904 rtc::SocketAddress session_connection_addr;
905 cricket::SessionDescription* desc = new cricket::SessionDescription(); 905 cricket::SessionDescription* desc = new cricket::SessionDescription();
906 std::vector<JsepIceCandidate*> candidates; 906 std::vector<JsepIceCandidate*> candidates;
907 size_t current_pos = 0; 907 size_t current_pos = 0;
908 908
909 // Session Description 909 // Session Description
910 if (!ParseSessionDescription(message, &current_pos, &session_id, 910 if (!ParseSessionDescription(message, &current_pos, &session_id,
911 &session_version, &session_td, &session_extmaps, 911 &session_version, &session_td, &session_extmaps,
912 desc, error)) { 912 &session_connection_addr, desc, error)) {
913 delete desc; 913 delete desc;
914 return false; 914 return false;
915 } 915 }
916 916
917 // Media Description 917 // Media Description
918 if (!ParseMediaDescription(message, session_td, session_extmaps, &current_pos, 918 if (!ParseMediaDescription(message, session_td, session_extmaps, &current_pos,
919 desc, &candidates, error)) { 919 session_connection_addr, desc, &candidates,
920 error)) {
920 delete desc; 921 delete desc;
921 for (std::vector<JsepIceCandidate*>::const_iterator 922 for (std::vector<JsepIceCandidate*>::const_iterator
922 it = candidates.begin(); it != candidates.end(); ++it) { 923 it = candidates.begin(); it != candidates.end(); ++it) {
923 delete *it; 924 delete *it;
924 } 925 }
925 return false; 926 return false;
926 } 927 }
927 928
928 jdesc->Initialize(desc, session_id, session_version); 929 jdesc->Initialize(desc, session_id, session_version);
929 930
(...skipping 391 matching lines...) Expand 10 before | Expand all | Expand 10 after
1321 1322
1322 rtc::SSLFingerprint* fp = (transport_info) ? 1323 rtc::SSLFingerprint* fp = (transport_info) ?
1323 transport_info->description.identity_fingerprint.get() : NULL; 1324 transport_info->description.identity_fingerprint.get() : NULL;
1324 1325
1325 // Add the m and c lines. 1326 // Add the m and c lines.
1326 InitLine(kLineTypeMedia, type, &os); 1327 InitLine(kLineTypeMedia, type, &os);
1327 os << " " << port << " " << media_desc->protocol() << fmt; 1328 os << " " << port << " " << media_desc->protocol() << fmt;
1328 std::string mline = os.str(); 1329 std::string mline = os.str();
1329 UpdateMediaDefaultDestination(candidates, mline, message); 1330 UpdateMediaDefaultDestination(candidates, mline, message);
1330 1331
1332 InitLine(kLineTypeConnection, kConnectionNettype, &os);
1333 if (media_desc->connection_address().family() == AF_INET) {
1334 os << " " << kConnectionIpv4Addrtype << " "
1335 << media_desc->connection_address().ipaddr().ToString();
1336 } else {
1337 os << " " << kConnectionIpv6Addrtype << " "
1338 << media_desc->connection_address().ipaddr().ToString();
1339 }
1340 AddLine(os.str(), message);
1341
1331 // RFC 4566 1342 // RFC 4566
1332 // b=AS:<bandwidth> 1343 // b=AS:<bandwidth>
1333 if (media_desc->bandwidth() >= 1000) { 1344 if (media_desc->bandwidth() >= 1000) {
1334 InitLine(kLineTypeSessionBandwidth, kApplicationSpecificMaximum, &os); 1345 InitLine(kLineTypeSessionBandwidth, kApplicationSpecificMaximum, &os);
1335 os << kSdpDelimiterColon << (media_desc->bandwidth() / 1000); 1346 os << kSdpDelimiterColon << (media_desc->bandwidth() / 1000);
1336 AddLine(os.str(), message); 1347 AddLine(os.str(), message);
1337 } 1348 }
1338 1349
1339 // Add the a=bundle-only line. 1350 // Add the a=bundle-only line.
1340 if (content_info->bundle_only) { 1351 if (content_info->bundle_only) {
(...skipping 552 matching lines...) Expand 10 before | Expand all | Expand 10 after
1893 bool IsRtp(const std::string& protocol) { 1904 bool IsRtp(const std::string& protocol) {
1894 return protocol.empty() || 1905 return protocol.empty() ||
1895 (protocol.find(cricket::kMediaProtocolRtpPrefix) != std::string::npos); 1906 (protocol.find(cricket::kMediaProtocolRtpPrefix) != std::string::npos);
1896 } 1907 }
1897 1908
1898 bool IsDtlsSctp(const std::string& protocol) { 1909 bool IsDtlsSctp(const std::string& protocol) {
1899 // This intentionally excludes "SCTP" and "SCTP/DTLS". 1910 // This intentionally excludes "SCTP" and "SCTP/DTLS".
1900 return protocol.find(cricket::kMediaProtocolDtlsSctp) != std::string::npos; 1911 return protocol.find(cricket::kMediaProtocolDtlsSctp) != std::string::npos;
1901 } 1912 }
1902 1913
1903 bool ParseSessionDescription(const std::string& message, size_t* pos, 1914 bool ParseConnectionData(const std::string& line,
1915 rtc::SocketAddress* addr,
1916 SdpParseError* error) {
1917 // Parse the line from left to right.
1918 std::string token;
1919 std::string rightpart;
1920 // RFC 4566
1921 // c=<nettype> <addrtype> <connection-address>
1922 // Skip the "c="
1923 if (!rtc::tokenize_first(line, kSdpDelimiterEqual, &token, &rightpart)) {
1924 return ParseFailed(line, "Failed to parse the network type.", error);
1925 }
1926
1927 // Extract and verify the <nettype>
1928 if (!rtc::tokenize_first(rightpart, kSdpDelimiterSpace, &token, &rightpart) ||
1929 token != kConnectionNettype) {
1930 return ParseFailed(line,
1931 "Failed to parse the connection data. The network type "
1932 "is not currently supported.",
1933 error);
1934 }
1935
1936 // Extract the "<addrtype>" and "<connection-address>".
1937 if (!rtc::tokenize_first(rightpart, kSdpDelimiterSpace, &token, &rightpart)) {
1938 return ParseFailed(line, "Failed to parse the address type.", error);
1939 }
1940
1941 // The rightpart part should be the IP address without the slash which is used
1942 // for multicast.
1943 if (rightpart.find('/') != std::string::npos) {
1944 return ParseFailed(line,
1945 "Failed to parse the connection data. Multicast is not "
1946 "currently supported.",
1947 error);
1948 }
1949 addr->SetIP(rightpart);
1950
1951 // Verify that the addrtype matches the type of the parsed address.
1952 if ((addr->family() == AF_INET && token != "IP4") ||
1953 (addr->family() == AF_INET6 && token != "IP6")) {
1954 addr->Clear();
1955 return ParseFailed(
1956 line,
1957 "Failed to parse the connection data. The address type is mismatching.",
1958 error);
1959 }
1960 return true;
1961 }
1962
1963 bool ParseSessionDescription(const std::string& message,
1964 size_t* pos,
1904 std::string* session_id, 1965 std::string* session_id,
1905 std::string* session_version, 1966 std::string* session_version,
1906 TransportDescription* session_td, 1967 TransportDescription* session_td,
1907 RtpHeaderExtensions* session_extmaps, 1968 RtpHeaderExtensions* session_extmaps,
1969 rtc::SocketAddress* connection_addr,
1908 cricket::SessionDescription* desc, 1970 cricket::SessionDescription* desc,
1909 SdpParseError* error) { 1971 SdpParseError* error) {
1910 std::string line; 1972 std::string line;
1911 1973
1912 desc->set_msid_supported(false); 1974 desc->set_msid_supported(false);
1913 1975
1914 // RFC 4566 1976 // RFC 4566
1915 // v= (protocol version) 1977 // v= (protocol version)
1916 if (!GetLineWithType(message, pos, &line, kLineTypeVersion)) { 1978 if (!GetLineWithType(message, pos, &line, kLineTypeVersion)) {
1917 return ParseFailedExpectLine(message, *pos, kLineTypeVersion, 1979 return ParseFailedExpectLine(message, *pos, kLineTypeVersion,
(...skipping 37 matching lines...) Expand 10 before | Expand all | Expand 10 after
1955 // e=* (email address) 2017 // e=* (email address)
1956 GetLineWithType(message, pos, &line, kLineTypeSessionEmail); 2018 GetLineWithType(message, pos, &line, kLineTypeSessionEmail);
1957 2019
1958 // RFC 4566 2020 // RFC 4566
1959 // p=* (phone number) 2021 // p=* (phone number)
1960 GetLineWithType(message, pos, &line, kLineTypeSessionPhone); 2022 GetLineWithType(message, pos, &line, kLineTypeSessionPhone);
1961 2023
1962 // RFC 4566 2024 // RFC 4566
1963 // c=* (connection information -- not required if included in 2025 // c=* (connection information -- not required if included in
1964 // all media) 2026 // all media)
1965 GetLineWithType(message, pos, &line, kLineTypeConnection); 2027 if (GetLineWithType(message, pos, &line, kLineTypeConnection)) {
2028 if (!ParseConnectionData(line, connection_addr, error)) {
2029 return false;
2030 }
2031 }
1966 2032
1967 // RFC 4566 2033 // RFC 4566
1968 // b=* (zero or more bandwidth information lines) 2034 // b=* (zero or more bandwidth information lines)
1969 while (GetLineWithType(message, pos, &line, kLineTypeSessionBandwidth)) { 2035 while (GetLineWithType(message, pos, &line, kLineTypeSessionBandwidth)) {
1970 // By pass zero or more b lines. 2036 // By pass zero or more b lines.
1971 } 2037 }
1972 2038
1973 // RFC 4566 2039 // RFC 4566
1974 // One or more time descriptions ("t=" and "r=" lines; see below) 2040 // One or more time descriptions ("t=" and "r=" lines; see below)
1975 // t= (time the session is active) 2041 // t= (time the session is active)
(...skipping 256 matching lines...) Expand 10 before | Expand all | Expand 10 after
2232 media_desc->AddCodec(cricket::AudioCodec(payload_type, encoding_name, 2298 media_desc->AddCodec(cricket::AudioCodec(payload_type, encoding_name,
2233 clock_rate, 0, channels)); 2299 clock_rate, 0, channels));
2234 } 2300 }
2235 } 2301 }
2236 } 2302 }
2237 2303
2238 template <class C> 2304 template <class C>
2239 static C* ParseContentDescription(const std::string& message, 2305 static C* ParseContentDescription(const std::string& message,
2240 const MediaType media_type, 2306 const MediaType media_type,
2241 int mline_index, 2307 int mline_index,
2308 int port,
2242 const std::string& protocol, 2309 const std::string& protocol,
2243 const std::vector<int>& payload_types, 2310 const std::vector<int>& payload_types,
2244 size_t* pos, 2311 size_t* pos,
2245 std::string* content_name, 2312 std::string* content_name,
2246 bool* bundle_only, 2313 bool* bundle_only,
2247 TransportDescription* transport, 2314 TransportDescription* transport,
2248 std::vector<JsepIceCandidate*>* candidates, 2315 std::vector<JsepIceCandidate*>* candidates,
2249 webrtc::SdpParseError* error) { 2316 webrtc::SdpParseError* error) {
2250 C* media_desc = new C(); 2317 C* media_desc = new C();
2251 switch (media_type) { 2318 switch (media_type) {
2252 case cricket::MEDIA_TYPE_AUDIO: 2319 case cricket::MEDIA_TYPE_AUDIO:
2253 *content_name = cricket::CN_AUDIO; 2320 *content_name = cricket::CN_AUDIO;
2254 break; 2321 break;
2255 case cricket::MEDIA_TYPE_VIDEO: 2322 case cricket::MEDIA_TYPE_VIDEO:
2256 *content_name = cricket::CN_VIDEO; 2323 *content_name = cricket::CN_VIDEO;
2257 break; 2324 break;
2258 case cricket::MEDIA_TYPE_DATA: 2325 case cricket::MEDIA_TYPE_DATA:
2259 *content_name = cricket::CN_DATA; 2326 *content_name = cricket::CN_DATA;
2260 break; 2327 break;
2261 default: 2328 default:
2262 RTC_NOTREACHED(); 2329 RTC_NOTREACHED();
2263 break; 2330 break;
2264 } 2331 }
2265 if (!ParseContent(message, media_type, mline_index, protocol, payload_types, 2332 if (!ParseContent(message, media_type, mline_index, port, protocol,
2266 pos, content_name, bundle_only, media_desc, transport, 2333 payload_types, pos, content_name, bundle_only, media_desc,
2267 candidates, error)) { 2334 transport, candidates, error)) {
2268 delete media_desc; 2335 delete media_desc;
2269 return NULL; 2336 return nullptr;
2270 } 2337 }
2271 // Sort the codecs according to the m-line fmt list. 2338 // Sort the codecs according to the m-line fmt list.
2272 std::unordered_map<int, int> payload_type_preferences; 2339 std::unordered_map<int, int> payload_type_preferences;
2273 // "size + 1" so that the lowest preference payload type has a preference of 2340 // "size + 1" so that the lowest preference payload type has a preference of
2274 // 1, which is greater than the default (0) for payload types not in the fmt 2341 // 1, which is greater than the default (0) for payload types not in the fmt
2275 // list. 2342 // list.
2276 int preference = static_cast<int>(payload_types.size() + 1); 2343 int preference = static_cast<int>(payload_types.size() + 1);
2277 for (int pt : payload_types) { 2344 for (int pt : payload_types) {
2278 payload_type_preferences[pt] = preference--; 2345 payload_type_preferences[pt] = preference--;
2279 } 2346 }
2280 std::vector<typename C::CodecType> codecs = media_desc->codecs(); 2347 std::vector<typename C::CodecType> codecs = media_desc->codecs();
2281 std::sort(codecs.begin(), codecs.end(), [&payload_type_preferences]( 2348 std::sort(codecs.begin(), codecs.end(), [&payload_type_preferences](
2282 const typename C::CodecType& a, 2349 const typename C::CodecType& a,
2283 const typename C::CodecType& b) { 2350 const typename C::CodecType& b) {
2284 return payload_type_preferences[a.id] > payload_type_preferences[b.id]; 2351 return payload_type_preferences[a.id] > payload_type_preferences[b.id];
2285 }); 2352 });
2286 media_desc->set_codecs(codecs); 2353 media_desc->set_codecs(codecs);
2287 return media_desc; 2354 return media_desc;
2288 } 2355 }
2289 2356
2290 bool ParseMediaDescription(const std::string& message, 2357 bool ParseMediaDescription(const std::string& message,
2291 const TransportDescription& session_td, 2358 const TransportDescription& session_td,
2292 const RtpHeaderExtensions& session_extmaps, 2359 const RtpHeaderExtensions& session_extmaps,
2293 size_t* pos, 2360 size_t* pos,
2361 const rtc::SocketAddress& session_connection_addr,
2294 cricket::SessionDescription* desc, 2362 cricket::SessionDescription* desc,
2295 std::vector<JsepIceCandidate*>* candidates, 2363 std::vector<JsepIceCandidate*>* candidates,
2296 SdpParseError* error) { 2364 SdpParseError* error) {
2297 RTC_DCHECK(desc != NULL); 2365 RTC_DCHECK(desc != NULL);
2298 std::string line; 2366 std::string line;
2299 int mline_index = -1; 2367 int mline_index = -1;
2300 2368
2301 // Zero or more media descriptions 2369 // Zero or more media descriptions
2302 // RFC 4566 2370 // RFC 4566
2303 // m=<media> <port> <proto> <fmt> 2371 // m=<media> <port> <proto> <fmt>
2304 while (GetLineWithType(message, pos, &line, kLineTypeMedia)) { 2372 while (GetLineWithType(message, pos, &line, kLineTypeMedia)) {
2305 ++mline_index; 2373 ++mline_index;
2306 2374
2307 std::vector<std::string> fields; 2375 std::vector<std::string> fields;
2308 rtc::split(line.substr(kLinePrefixLength), 2376 rtc::split(line.substr(kLinePrefixLength),
2309 kSdpDelimiterSpace, &fields); 2377 kSdpDelimiterSpace, &fields);
2310 2378
2311 const size_t expected_min_fields = 4; 2379 const size_t expected_min_fields = 4;
2312 if (fields.size() < expected_min_fields) { 2380 if (fields.size() < expected_min_fields) {
2313 return ParseFailedExpectMinFieldNum(line, expected_min_fields, error); 2381 return ParseFailedExpectMinFieldNum(line, expected_min_fields, error);
2314 } 2382 }
2315 bool port_rejected = false; 2383 bool port_rejected = false;
2316 // RFC 3264 2384 // RFC 3264
2317 // To reject an offered stream, the port number in the corresponding stream 2385 // To reject an offered stream, the port number in the corresponding stream
2318 // in the answer MUST be set to zero. 2386 // in the answer MUST be set to zero.
2319 if (fields[1] == kMediaPortRejected) { 2387 if (fields[1] == kMediaPortRejected) {
2320 port_rejected = true; 2388 port_rejected = true;
2321 } 2389 }
2322 2390
2391 int port = atoi(fields[1].c_str());
Taylor Brandstetter 2017/03/17 00:20:42 Can use "rtc::FromString<int>" and check for parse
Zhi Huang 2017/03/17 23:42:06 Done.
2323 std::string protocol = fields[2]; 2392 std::string protocol = fields[2];
2324 2393
2325 // <fmt> 2394 // <fmt>
2326 std::vector<int> payload_types; 2395 std::vector<int> payload_types;
2327 if (IsRtp(protocol)) { 2396 if (IsRtp(protocol)) {
2328 for (size_t j = 3 ; j < fields.size(); ++j) { 2397 for (size_t j = 3 ; j < fields.size(); ++j) {
2329 // TODO(wu): Remove when below bug is fixed. 2398 // TODO(wu): Remove when below bug is fixed.
2330 // https://bugzilla.mozilla.org/show_bug.cgi?id=996329 2399 // https://bugzilla.mozilla.org/show_bug.cgi?id=996329
2331 if (fields[j].empty() && j == fields.size() - 1) { 2400 if (fields[j].empty() && j == fields.size() - 1) {
2332 continue; 2401 continue;
(...skipping 12 matching lines...) Expand all
2345 TransportDescription transport( 2414 TransportDescription transport(
2346 session_td.transport_options, session_td.ice_ufrag, session_td.ice_pwd, 2415 session_td.transport_options, session_td.ice_ufrag, session_td.ice_pwd,
2347 session_td.ice_mode, session_td.connection_role, 2416 session_td.ice_mode, session_td.connection_role,
2348 session_td.identity_fingerprint.get()); 2417 session_td.identity_fingerprint.get());
2349 2418
2350 std::unique_ptr<MediaContentDescription> content; 2419 std::unique_ptr<MediaContentDescription> content;
2351 std::string content_name; 2420 std::string content_name;
2352 bool bundle_only = false; 2421 bool bundle_only = false;
2353 if (HasAttribute(line, kMediaTypeVideo)) { 2422 if (HasAttribute(line, kMediaTypeVideo)) {
2354 content.reset(ParseContentDescription<VideoContentDescription>( 2423 content.reset(ParseContentDescription<VideoContentDescription>(
2355 message, cricket::MEDIA_TYPE_VIDEO, mline_index, protocol, 2424 message, cricket::MEDIA_TYPE_VIDEO, mline_index, port, protocol,
Taylor Brandstetter 2017/03/17 00:20:42 To avoid having to add the port as an argument eve
Zhi Huang 2017/03/17 23:42:06 Yes, but I need to create a copy of those since th
2356 payload_types, pos, &content_name, &bundle_only, &transport, 2425 payload_types, pos, &content_name, &bundle_only, &transport,
2357 candidates, error)); 2426 candidates, error));
2358 } else if (HasAttribute(line, kMediaTypeAudio)) { 2427 } else if (HasAttribute(line, kMediaTypeAudio)) {
2359 content.reset(ParseContentDescription<AudioContentDescription>( 2428 content.reset(ParseContentDescription<AudioContentDescription>(
2360 message, cricket::MEDIA_TYPE_AUDIO, mline_index, protocol, 2429 message, cricket::MEDIA_TYPE_AUDIO, mline_index, port, protocol,
2361 payload_types, pos, &content_name, &bundle_only, &transport, 2430 payload_types, pos, &content_name, &bundle_only, &transport,
2362 candidates, error)); 2431 candidates, error));
2363 } else if (HasAttribute(line, kMediaTypeData)) { 2432 } else if (HasAttribute(line, kMediaTypeData)) {
2364 DataContentDescription* data_desc = 2433 DataContentDescription* data_desc =
2365 ParseContentDescription<DataContentDescription>( 2434 ParseContentDescription<DataContentDescription>(
2366 message, cricket::MEDIA_TYPE_DATA, mline_index, protocol, 2435 message, cricket::MEDIA_TYPE_DATA, mline_index, port, protocol,
2367 payload_types, pos, &content_name, &bundle_only, &transport, 2436 payload_types, pos, &content_name, &bundle_only, &transport,
2368 candidates, error); 2437 candidates, error);
2369 content.reset(data_desc); 2438 content.reset(data_desc);
2370 2439
2371 if (data_desc && IsDtlsSctp(protocol)) { 2440 if (data_desc && IsDtlsSctp(protocol)) {
2372 int p; 2441 int p;
2373 if (rtc::FromString(fields[3], &p)) { 2442 if (rtc::FromString(fields[3], &p)) {
2374 if (!AddSctpDataCodec(data_desc, p)) { 2443 if (!AddSctpDataCodec(data_desc, p)) {
2375 return false; 2444 return false;
2376 } 2445 }
(...skipping 36 matching lines...) Expand 10 before | Expand all | Expand 10 after
2413 return ParseFailed("", 2482 return ParseFailed("",
2414 "The a=extmap MUST be either all session level or " 2483 "The a=extmap MUST be either all session level or "
2415 "all media level.", 2484 "all media level.",
2416 error); 2485 error);
2417 } 2486 }
2418 for (size_t i = 0; i < session_extmaps.size(); ++i) { 2487 for (size_t i = 0; i < session_extmaps.size(); ++i) {
2419 content->AddRtpHeaderExtension(session_extmaps[i]); 2488 content->AddRtpHeaderExtension(session_extmaps[i]);
2420 } 2489 }
2421 } 2490 }
2422 content->set_protocol(protocol); 2491 content->set_protocol(protocol);
2492
2493 // Use the session level connection address if the media level addresses are
2494 // not specified.
2495 if (content->connection_address().IsNil()) {
2496 content->set_connection_address(session_connection_addr);
Taylor Brandstetter 2017/03/17 00:20:42 If this condition is hit, the port from the m= lin
Zhi Huang 2017/03/17 23:42:06 Fixed.
2497 }
2498
2423 desc->AddContent(content_name, 2499 desc->AddContent(content_name,
2424 IsDtlsSctp(protocol) ? cricket::NS_JINGLE_DRAFT_SCTP 2500 IsDtlsSctp(protocol) ? cricket::NS_JINGLE_DRAFT_SCTP
2425 : cricket::NS_JINGLE_RTP, 2501 : cricket::NS_JINGLE_RTP,
2426 content_rejected, bundle_only, content.release()); 2502 content_rejected, bundle_only, content.release());
2427 // Create TransportInfo with the media level "ice-pwd" and "ice-ufrag". 2503 // Create TransportInfo with the media level "ice-pwd" and "ice-ufrag".
2428 TransportInfo transport_info(content_name, transport); 2504 TransportInfo transport_info(content_name, transport);
2429 2505
2430 if (!desc->AddTransportInfo(transport_info)) { 2506 if (!desc->AddTransportInfo(transport_info)) {
2431 std::ostringstream description; 2507 std::ostringstream description;
2432 description << "Failed to AddTransportInfo with content name: " 2508 description << "Failed to AddTransportInfo with content name: "
(...skipping 155 matching lines...) Expand 10 before | Expand all | Expand 10 after
2588 for (std::vector<cricket::AudioCodec>::iterator iter = codecs.begin(); 2664 for (std::vector<cricket::AudioCodec>::iterator iter = codecs.begin();
2589 iter != codecs.end(); ++iter) { 2665 iter != codecs.end(); ++iter) {
2590 iter->params[name] = value; 2666 iter->params[name] = value;
2591 } 2667 }
2592 audio_desc->set_codecs(codecs); 2668 audio_desc->set_codecs(codecs);
2593 } 2669 }
2594 2670
2595 bool ParseContent(const std::string& message, 2671 bool ParseContent(const std::string& message,
2596 const MediaType media_type, 2672 const MediaType media_type,
2597 int mline_index, 2673 int mline_index,
2674 int port,
2598 const std::string& protocol, 2675 const std::string& protocol,
2599 const std::vector<int>& payload_types, 2676 const std::vector<int>& payload_types,
2600 size_t* pos, 2677 size_t* pos,
2601 std::string* content_name, 2678 std::string* content_name,
2602 bool* bundle_only, 2679 bool* bundle_only,
2603 MediaContentDescription* media_desc, 2680 MediaContentDescription* media_desc,
2604 TransportDescription* transport, 2681 TransportDescription* transport,
2605 std::vector<JsepIceCandidate*>* candidates, 2682 std::vector<JsepIceCandidate*>* candidates,
2606 SdpParseError* error) { 2683 SdpParseError* error) {
2607 RTC_DCHECK(media_desc != NULL); 2684 RTC_DCHECK(media_desc != NULL);
(...skipping 53 matching lines...) Expand 10 before | Expand all | Expand 10 after
2661 return ParseFailed(line, description.str(), error); 2738 return ParseFailed(line, description.str(), error);
2662 } 2739 }
2663 // Prevent integer overflow. 2740 // Prevent integer overflow.
2664 b = std::min(b, INT_MAX / 1000); 2741 b = std::min(b, INT_MAX / 1000);
2665 media_desc->set_bandwidth(b * 1000); 2742 media_desc->set_bandwidth(b * 1000);
2666 } 2743 }
2667 } 2744 }
2668 continue; 2745 continue;
2669 } 2746 }
2670 2747
2748 // Parse the media level connection data.
2749 if (IsLineType(line, kLineTypeConnection)) {
2750 rtc::SocketAddress addr;
2751 if (!ParseConnectionData(line, &addr, error)) {
2752 return false;
2753 }
2754 addr.SetPort(port);
2755 media_desc->set_connection_address(addr);
2756 continue;
2757 }
2758
2671 if (!IsLineType(line, kLineTypeAttributes)) { 2759 if (!IsLineType(line, kLineTypeAttributes)) {
2672 // TODO: Handle other lines if needed. 2760 // TODO: Handle other lines if needed.
2673 LOG(LS_INFO) << "Ignored line: " << line; 2761 LOG(LS_INFO) << "Ignored line: " << line;
2674 continue; 2762 continue;
2675 } 2763 }
2676 2764
2677 // Handle attributes common to SCTP and RTP. 2765 // Handle attributes common to SCTP and RTP.
2678 if (HasAttribute(line, kAttributeMid)) { 2766 if (HasAttribute(line, kAttributeMid)) {
2679 // RFC 3388 2767 // RFC 3388
2680 // mid-attribute = "a=mid:" identification-tag 2768 // mid-attribute = "a=mid:" identification-tag
(...skipping 553 matching lines...) Expand 10 before | Expand all | Expand 10 after
3234 UpdateCodec<AudioContentDescription, cricket::AudioCodec>( 3322 UpdateCodec<AudioContentDescription, cricket::AudioCodec>(
3235 media_desc, payload_type, feedback_param); 3323 media_desc, payload_type, feedback_param);
3236 } else if (media_type == cricket::MEDIA_TYPE_VIDEO) { 3324 } else if (media_type == cricket::MEDIA_TYPE_VIDEO) {
3237 UpdateCodec<VideoContentDescription, cricket::VideoCodec>( 3325 UpdateCodec<VideoContentDescription, cricket::VideoCodec>(
3238 media_desc, payload_type, feedback_param); 3326 media_desc, payload_type, feedback_param);
3239 } 3327 }
3240 return true; 3328 return true;
3241 } 3329 }
3242 3330
3243 } // namespace webrtc 3331 } // namespace webrtc
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698