blob: b555b1628ccfc1fd1f2def9244b03c109a571e87 [file] [log] [blame]
/*
* Copyright 2018 The WebRTC project authors. All Rights Reserved.
*
* Use of this source code is governed by a BSD-style license
* that can be found in the LICENSE file in the root of the source
* tree. An additional intellectual property rights grant can be found
* in the file PATENTS. All contributing project authors may
* be found in the AUTHORS file in the root of the source tree.
*/
#include "api/jsep_ice_candidate.h"
#include <cstddef>
#include <cstdint>
#include <iterator>
#include <limits>
#include <memory>
#include <string>
#include <utility>
#include "absl/algorithm/container.h"
#include "absl/base/nullability.h"
#include "absl/memory/memory.h"
#include "absl/strings/string_view.h"
#include "api/candidate.h"
#include "api/jsep.h"
#include "api/rtc_error.h"
#include "api/sequence_checker.h"
#include "rtc_base/checks.h"
#include "rtc_base/logging.h"
namespace webrtc {
namespace {
// The sdpMLineIndex property is an unsigned short, a zero based index of the
// m-line associated with the candidate. This function ensures we consistently
// set the property to -1 for out-of-bounds values, to make candidate
// comparisons more robust.
int EnsureValidMLineIndex(int sdp_mline_index) {
if (sdp_mline_index < 0 ||
sdp_mline_index > std::numeric_limits<uint16_t>::max())
return -1;
return sdp_mline_index;
}
} // namespace
// static
std::unique_ptr<IceCandidate> IceCandidate::Create(absl::string_view mid,
int sdp_mline_index,
absl::string_view sdp,
SdpParseError* absl_nullable
error /*= nullptr*/) {
RTCErrorOr<Candidate> c = Candidate::ParseCandidateString(sdp);
if (!c.ok()) {
if (error) {
error->line = sdp;
error->description = c.error().message();
}
return nullptr;
}
return std::make_unique<IceCandidate>(mid, sdp_mline_index, c.value());
}
IceCandidate::IceCandidate(absl::string_view sdp_mid,
int sdp_mline_index,
const Candidate& candidate)
: sdp_mid_(sdp_mid),
sdp_mline_index_(EnsureValidMLineIndex(sdp_mline_index)),
candidate_(candidate) {
if (sdp_mid_.empty() && sdp_mline_index_ < 0) {
RTC_LOG(LS_ERROR) << "Neither mid nor index supplied for IceCandidate.";
}
}
void IceCandidateCollection::add(std::unique_ptr<IceCandidate> candidate) {
RTC_DCHECK_RUN_ON(&sequence_checker_);
candidates_.push_back(std::move(candidate));
}
void IceCandidateCollection::add(IceCandidate* candidate) {
RTC_DCHECK_RUN_ON(&sequence_checker_);
candidates_.push_back(absl::WrapUnique(candidate));
}
void IceCandidateCollection::Append(IceCandidateCollection collection) {
RTC_DCHECK_RUN_ON(&sequence_checker_);
RTC_DCHECK_RUN_ON(&collection.sequence_checker_);
candidates_.insert(candidates_.end(),
std::make_move_iterator(collection.candidates_.begin()),
std::make_move_iterator(collection.candidates_.end()));
}
const IceCandidate* IceCandidateCollection::at(size_t index) const {
RTC_DCHECK_RUN_ON(&sequence_checker_);
return candidates_[index].get();
}
bool IceCandidateCollection::HasCandidate(const IceCandidate* candidate) const {
RTC_DCHECK_RUN_ON(&sequence_checker_);
const auto sdp_mid = candidate->sdp_mid(); // avoid string copy per entry.
return absl::c_any_of(
candidates_, [&](const std::unique_ptr<IceCandidate>& entry) {
if (!entry->candidate().IsEquivalent(candidate->candidate())) {
return false;
}
if (!sdp_mid.empty()) {
// In this case, we ignore the `sdp_mline_index()` property.
return sdp_mid == entry->sdp_mid();
}
RTC_DCHECK_NE(candidate->sdp_mline_index(), -1);
return candidate->sdp_mline_index() == entry->sdp_mline_index();
});
}
size_t IceCandidateCollection::remove(const IceCandidate* candidate) {
RTC_DCHECK_RUN_ON(&sequence_checker_);
RTC_DCHECK(candidate);
auto iter =
absl::c_find_if(candidates_, [&](const std::unique_ptr<IceCandidate>& c) {
return c->candidate().MatchesForRemoval(candidate->candidate());
});
if (iter != candidates_.end()) {
candidates_.erase(iter);
return 1u;
}
return 0u;
}
void IceCandidateCollection::RelinquishThreadOwnership() {
sequence_checker_.Detach();
}
std::string IceCandidate::ToString() const {
return candidate_.ToCandidateAttribute(true);
}
} // namespace webrtc