OLD | NEW |
---|---|
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 |
OLD | NEW |