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

Side by Side Diff: webrtc/p2p/base/p2ptransportchannel.cc

Issue 1498993002: Add ufrag to the ICE candidate signaling. (Closed) Base URL: https://chromium.googlesource.com/external/webrtc@master
Patch Set: Created 5 years 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 2004 The WebRTC Project Authors. All rights reserved. 2 * Copyright 2004 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 200 matching lines...) Expand 10 before | Expand all | Expand 10 after
211 allocator_(allocator), 211 allocator_(allocator),
212 worker_thread_(rtc::Thread::Current()), 212 worker_thread_(rtc::Thread::Current()),
213 incoming_only_(false), 213 incoming_only_(false),
214 error_(0), 214 error_(0),
215 best_connection_(NULL), 215 best_connection_(NULL),
216 pending_best_connection_(NULL), 216 pending_best_connection_(NULL),
217 sort_dirty_(false), 217 sort_dirty_(false),
218 remote_ice_mode_(ICEMODE_FULL), 218 remote_ice_mode_(ICEMODE_FULL),
219 ice_role_(ICEROLE_UNKNOWN), 219 ice_role_(ICEROLE_UNKNOWN),
220 tiebreaker_(0), 220 tiebreaker_(0),
221 remote_candidate_generation_(0),
222 gathering_state_(kIceGatheringNew), 221 gathering_state_(kIceGatheringNew),
223 check_receiving_delay_(MIN_CHECK_RECEIVING_DELAY * 5), 222 check_receiving_delay_(MIN_CHECK_RECEIVING_DELAY * 5),
224 receiving_timeout_(MIN_CHECK_RECEIVING_DELAY * 50), 223 receiving_timeout_(MIN_CHECK_RECEIVING_DELAY * 50),
225 backup_connection_ping_interval_(0) { 224 backup_connection_ping_interval_(0) {
226 uint32_t weak_ping_delay = ::strtoul( 225 uint32_t weak_ping_delay = ::strtoul(
227 webrtc::field_trial::FindFullName("WebRTC-StunInterPacketDelay").c_str(), 226 webrtc::field_trial::FindFullName("WebRTC-StunInterPacketDelay").c_str(),
228 nullptr, 10); 227 nullptr, 10);
229 if (weak_ping_delay) { 228 if (weak_ping_delay) {
230 weak_ping_delay_ = weak_ping_delay; 229 weak_ping_delay_ = weak_ping_delay;
231 } 230 }
(...skipping 108 matching lines...) Expand 10 before | Expand all | Expand 10 after
340 ASSERT(worker_thread_ == rtc::Thread::Current()); 339 ASSERT(worker_thread_ == rtc::Thread::Current());
341 ice_ufrag_ = ice_ufrag; 340 ice_ufrag_ = ice_ufrag;
342 ice_pwd_ = ice_pwd; 341 ice_pwd_ = ice_pwd;
343 // Note: Candidate gathering will restart when MaybeStartGathering is next 342 // Note: Candidate gathering will restart when MaybeStartGathering is next
344 // called. 343 // called.
345 } 344 }
346 345
347 void P2PTransportChannel::SetRemoteIceCredentials(const std::string& ice_ufrag, 346 void P2PTransportChannel::SetRemoteIceCredentials(const std::string& ice_ufrag,
348 const std::string& ice_pwd) { 347 const std::string& ice_pwd) {
349 ASSERT(worker_thread_ == rtc::Thread::Current()); 348 ASSERT(worker_thread_ == rtc::Thread::Current());
350 bool ice_restart = false; 349 if (remote_ice_generations_.empty() || (remote_ice_ufrag() != ice_ufrag) ||
351 if (!remote_ice_ufrag_.empty() && !remote_ice_pwd_.empty()) { 350 (remote_ice_pwd() != ice_pwd)) {
352 ice_restart = (remote_ice_ufrag_ != ice_ufrag) || 351 // Keep the old-generation ICE credentials so that newer connections
353 (remote_ice_pwd_!= ice_pwd); 352 // are prioritized over the older ones.
353 remote_ice_generations_.push_back(IceGeneration(ice_ufrag, ice_pwd));
354 } 354 }
pthatcher1 2015/12/10 22:08:06 I think this might be a little better using an opt
honghaiz3 2015/12/11 04:47:29 Done.
355 355
356 remote_ice_ufrag_ = ice_ufrag;
357 remote_ice_pwd_ = ice_pwd;
358
359 // We need to update the credentials for any peer reflexive candidates. 356 // We need to update the credentials for any peer reflexive candidates.
360 std::vector<Connection*>::iterator it = connections_.begin(); 357 std::vector<Connection*>::iterator it = connections_.begin();
361 for (; it != connections_.end(); ++it) { 358 for (; it != connections_.end(); ++it) {
362 (*it)->MaybeSetRemoteIceCredentials(ice_ufrag, ice_pwd); 359 (*it)->MaybeSetRemoteIceCredentials(ice_ufrag, ice_pwd);
363 } 360 }
364
365 if (ice_restart) {
366 // We need to keep track of the remote ice restart so newer
367 // connections are prioritized over the older.
368 ++remote_candidate_generation_;
369 }
370 } 361 }
371 362
372 void P2PTransportChannel::SetRemoteIceMode(IceMode mode) { 363 void P2PTransportChannel::SetRemoteIceMode(IceMode mode) {
373 remote_ice_mode_ = mode; 364 remote_ice_mode_ = mode;
374 } 365 }
375 366
376 void P2PTransportChannel::SetIceConfig(const IceConfig& config) { 367 void P2PTransportChannel::SetIceConfig(const IceConfig& config) {
377 gather_continually_ = config.gather_continually; 368 gather_continually_ = config.gather_continually;
378 LOG(LS_INFO) << "Set gather_continually to " << gather_continually_; 369 LOG(LS_INFO) << "Set gather_continually to " << gather_continually_;
379 370
(...skipping 149 matching lines...) Expand 10 before | Expand all | Expand 10 after
529 break; 520 break;
530 } 521 }
531 // We don't want to break here because we may find a match of the address 522 // We don't want to break here because we may find a match of the address
532 // later. 523 // later.
533 } 524 }
534 } 525 }
535 526
536 // The STUN binding request may arrive after setRemoteDescription and before 527 // The STUN binding request may arrive after setRemoteDescription and before
537 // adding remote candidate, so we need to set the password to the shared 528 // adding remote candidate, so we need to set the password to the shared
538 // password if the user name matches. 529 // password if the user name matches.
539 if (remote_password.empty() && remote_username == remote_ice_ufrag_) { 530 if (remote_password.empty() && !remote_ice_generations_.empty() &&
540 remote_password = remote_ice_pwd_; 531 remote_username == remote_ice_ufrag()) {
532 remote_password = remote_ice_pwd();
541 } 533 }
pthatcher1 2015/12/10 22:08:06 Same here, with optional: if (remote_password.emp
honghaiz3 2015/12/11 04:47:29 Done.
542 534
543 Candidate remote_candidate; 535 Candidate remote_candidate;
544 bool remote_candidate_is_new = (candidate == nullptr); 536 bool remote_candidate_is_new = (candidate == nullptr);
545 if (!remote_candidate_is_new) { 537 if (!remote_candidate_is_new) {
546 remote_candidate = *candidate; 538 remote_candidate = *candidate;
547 if (ufrag_per_port) { 539 if (ufrag_per_port) {
548 remote_candidate.set_address(address); 540 remote_candidate.set_address(address);
549 } 541 }
550 } else { 542 } else {
551 // Create a new candidate with this address. 543 // Create a new candidate with this address.
(...skipping 96 matching lines...) Expand 10 before | Expand all | Expand 10 after
648 } else { 640 } else {
649 LOG(LS_INFO) << "Not switching the best connection on controlled side yet," 641 LOG(LS_INFO) << "Not switching the best connection on controlled side yet,"
650 << " because it's not writable: " << conn->ToString(); 642 << " because it's not writable: " << conn->ToString();
651 pending_best_connection_ = conn; 643 pending_best_connection_ = conn;
652 } 644 }
653 } 645 }
654 646
655 void P2PTransportChannel::AddRemoteCandidate(const Candidate& candidate) { 647 void P2PTransportChannel::AddRemoteCandidate(const Candidate& candidate) {
656 ASSERT(worker_thread_ == rtc::Thread::Current()); 648 ASSERT(worker_thread_ == rtc::Thread::Current());
657 649
658 uint32_t generation = candidate.generation(); 650 int generation = GetRemoteCandidateGeneration(candidate);
659 // Network may not guarantee the order of the candidate delivery. If a 651 // If a remote candidate with a previous generation arrives, drop it.
660 // remote candidate with an older generation arrives, drop it. 652 if (generation < remote_ice_generation()) {
661 if (generation != 0 && generation < remote_candidate_generation_) { 653 LOG(LS_WARNING) << "Dropping a remote candidate because its ufrag "
662 LOG(LS_WARNING) << "Dropping a remote candidate because its generation " 654 << candidate.username()
663 << generation 655 << " indicates it was for an previous generation.";
664 << " is lower than the current remote generation "
665 << remote_candidate_generation_;
666 return; 656 return;
667 } 657 }
668 658
659 Candidate new_remote_candidate(candidate);
660 new_remote_candidate.set_generation(generation);
661 // ICE candidates don't need to have username and password set, but
662 // the code below this (specifically, ConnectionRequest::Prepare in
663 // port.cc) uses the remote candidates's username. So, we set it
664 // here.
665 if (!remote_ice_generations_.empty()) {
666 if (candidate.username().empty()) {
667 new_remote_candidate.set_username(remote_ice_ufrag());
668 }
669 if (new_remote_candidate.username() == remote_ice_ufrag()) {
670 if (candidate.password().empty()) {
671 new_remote_candidate.set_password(remote_ice_pwd());
672 }
673 } else {
674 // The candidate belongs to the next generation. Its pwd will be set
675 // when the new remote ICE credentials arrive.
676 LOG(LS_WARNING) << "A remote candidate arrives with an unknown ufrag: "
677 << candidate.username();
678 }
679 }
680
669 // Create connections to this remote candidate. 681 // Create connections to this remote candidate.
670 CreateConnections(candidate, NULL); 682 CreateConnections(new_remote_candidate, NULL);
671 683
672 // Resort the connections list, which may have new elements. 684 // Resort the connections list, which may have new elements.
673 SortConnections(); 685 SortConnections();
674 } 686 }
675 687
676 // Creates connections from all of the ports that we care about to the given 688 // Creates connections from all of the ports that we care about to the given
677 // remote candidate. The return value is true if we created a connection from 689 // remote candidate. The return value is true if we created a connection from
678 // the origin port. 690 // the origin port.
679 bool P2PTransportChannel::CreateConnections(const Candidate& remote_candidate, 691 bool P2PTransportChannel::CreateConnections(const Candidate& remote_candidate,
680 PortInterface* origin_port) { 692 PortInterface* origin_port) {
681 ASSERT(worker_thread_ == rtc::Thread::Current()); 693 ASSERT(worker_thread_ == rtc::Thread::Current());
682 694
683 Candidate new_remote_candidate(remote_candidate);
684 new_remote_candidate.set_generation(
685 GetRemoteCandidateGeneration(remote_candidate));
686 // ICE candidates don't need to have username and password set, but
687 // the code below this (specifically, ConnectionRequest::Prepare in
688 // port.cc) uses the remote candidates's username. So, we set it
689 // here.
690 if (remote_candidate.username().empty()) {
691 new_remote_candidate.set_username(remote_ice_ufrag_);
692 }
693 if (remote_candidate.password().empty()) {
694 new_remote_candidate.set_password(remote_ice_pwd_);
695 }
696
697 // If we've already seen the new remote candidate (in the current candidate 695 // If we've already seen the new remote candidate (in the current candidate
698 // generation), then we shouldn't try creating connections for it. 696 // generation), then we shouldn't try creating connections for it.
699 // We either already have a connection for it, or we previously created one 697 // We either already have a connection for it, or we previously created one
700 // and then later pruned it. If we don't return, the channel will again 698 // and then later pruned it. If we don't return, the channel will again
701 // re-create any connections that were previously pruned, which will then 699 // re-create any connections that were previously pruned, which will then
702 // immediately be re-pruned, churning the network for no purpose. 700 // immediately be re-pruned, churning the network for no purpose.
703 // This only applies to candidates received over signaling (i.e. origin_port 701 // This only applies to candidates received over signaling (i.e. origin_port
704 // is NULL). 702 // is NULL).
705 if (!origin_port && IsDuplicateRemoteCandidate(new_remote_candidate)) { 703 if (!origin_port && IsDuplicateRemoteCandidate(remote_candidate)) {
706 // return true to indicate success, without creating any new connections. 704 // return true to indicate success, without creating any new connections.
707 return true; 705 return true;
708 } 706 }
709 707
710 // Add a new connection for this candidate to every port that allows such a 708 // Add a new connection for this candidate to every port that allows such a
711 // connection (i.e., if they have compatible protocols) and that does not 709 // connection (i.e., if they have compatible protocols) and that does not
712 // already have a connection to an equivalent candidate. We must be careful 710 // already have a connection to an equivalent candidate. We must be careful
713 // to make sure that the origin port is included, even if it was pruned, 711 // to make sure that the origin port is included, even if it was pruned,
714 // since that may be the only port that can create this connection. 712 // since that may be the only port that can create this connection.
715 bool created = false; 713 bool created = false;
716 std::vector<PortInterface *>::reverse_iterator it; 714 std::vector<PortInterface *>::reverse_iterator it;
717 for (it = ports_.rbegin(); it != ports_.rend(); ++it) { 715 for (it = ports_.rbegin(); it != ports_.rend(); ++it) {
718 if (CreateConnection(*it, new_remote_candidate, origin_port)) { 716 if (CreateConnection(*it, remote_candidate, origin_port)) {
719 if (*it == origin_port) 717 if (*it == origin_port)
720 created = true; 718 created = true;
721 } 719 }
722 } 720 }
723 721
724 if ((origin_port != NULL) && 722 if ((origin_port != NULL) &&
725 std::find(ports_.begin(), ports_.end(), origin_port) == ports_.end()) { 723 std::find(ports_.begin(), ports_.end(), origin_port) == ports_.end()) {
726 if (CreateConnection(origin_port, new_remote_candidate, origin_port)) 724 if (CreateConnection(origin_port, remote_candidate, origin_port))
727 created = true; 725 created = true;
728 } 726 }
729 727
730 // Remember this remote candidate so that we can add it to future ports. 728 // Remember this remote candidate so that we can add it to future ports.
731 RememberRemoteCandidate(new_remote_candidate, origin_port); 729 RememberRemoteCandidate(remote_candidate, origin_port);
732 730
733 return created; 731 return created;
734 } 732 }
735 733
736 // Setup a connection object for the local and remote candidate combination. 734 // Setup a connection object for the local and remote candidate combination.
737 // And then listen to connection object for changes. 735 // And then listen to connection object for changes.
738 bool P2PTransportChannel::CreateConnection(PortInterface* port, 736 bool P2PTransportChannel::CreateConnection(PortInterface* port,
739 const Candidate& remote_candidate, 737 const Candidate& remote_candidate,
740 PortInterface* origin_port) { 738 PortInterface* origin_port) {
741 // Look for an existing connection with this remote address. If one is not 739 // Look for an existing connection with this remote address. If one is not
(...skipping 37 matching lines...) Expand 10 before | Expand all | Expand 10 after
779 cricket::Connection* connection) const { 777 cricket::Connection* connection) const {
780 std::vector<Connection*>::const_iterator citer = 778 std::vector<Connection*>::const_iterator citer =
781 std::find(connections_.begin(), connections_.end(), connection); 779 std::find(connections_.begin(), connections_.end(), connection);
782 return citer != connections_.end(); 780 return citer != connections_.end();
783 } 781 }
784 782
785 uint32_t P2PTransportChannel::GetRemoteCandidateGeneration( 783 uint32_t P2PTransportChannel::GetRemoteCandidateGeneration(
786 const Candidate& candidate) { 784 const Candidate& candidate) {
787 // We need to keep track of the remote ice restart so newer 785 // We need to keep track of the remote ice restart so newer
788 // connections are prioritized over the older. 786 // connections are prioritized over the older.
789 ASSERT(candidate.generation() == 0 || 787 const std::string& ufrag = candidate.username();
790 candidate.generation() == remote_candidate_generation_); 788 int generation = 0;
791 return remote_candidate_generation_; 789 if (!ufrag.empty()) {
790 // If remote side sets the ufrag, we use that to determine the candidate
791 // generation.
792 generation = remote_ice_generations_.size() - 1;
793 for (; generation >= 0; --generation) {
794 if (ufrag == remote_ice_generations_[generation].ufrag) {
795 return generation;
796 }
797 }
798 // If not found, this is the next generation candidate.
799 return remote_ice_generations_.size();
800 }
pthatcher1 2015/12/10 22:08:06 I think this could be more readable by using std::
honghaiz3 2015/12/11 04:47:29 Done.
801 // Use the generation set in the candidate.
802 generation = candidate.generation();
803 int cur_remote_generation = remote_ice_generation();
804 if (generation == 0 && cur_remote_generation > 0) {
805 // 0 indicates it may not be set by the peer.
806 generation = cur_remote_generation;
807 }
808 return generation;
pthatcher1 2015/12/10 22:08:06 I think this would be readable as: // If candidat
honghaiz3 2015/12/11 04:47:29 Done.
792 } 809 }
793 810
794 // Check if remote candidate is already cached. 811 // Check if remote candidate is already cached.
795 bool P2PTransportChannel::IsDuplicateRemoteCandidate( 812 bool P2PTransportChannel::IsDuplicateRemoteCandidate(
796 const Candidate& candidate) { 813 const Candidate& candidate) {
797 for (size_t i = 0; i < remote_candidates_.size(); ++i) { 814 for (size_t i = 0; i < remote_candidates_.size(); ++i) {
798 if (remote_candidates_[i].IsEquivalent(candidate)) { 815 if (remote_candidates_[i].IsEquivalent(candidate)) {
799 return true; 816 return true;
800 } 817 }
801 } 818 }
(...skipping 592 matching lines...) Expand 10 before | Expand all | Expand 10 after
1394 SignalSentPacket(this, sent_packet); 1411 SignalSentPacket(this, sent_packet);
1395 } 1412 }
1396 1413
1397 void P2PTransportChannel::OnReadyToSend(Connection* connection) { 1414 void P2PTransportChannel::OnReadyToSend(Connection* connection) {
1398 if (connection == best_connection_ && writable()) { 1415 if (connection == best_connection_ && writable()) {
1399 SignalReadyToSend(this); 1416 SignalReadyToSend(this);
1400 } 1417 }
1401 } 1418 }
1402 1419
1403 } // namespace cricket 1420 } // namespace cricket
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698