blob: 5f6889e6dc23618d6bf9dbbb033793c67a276a2d [file] [log] [blame]
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +00001/*
2 * Copyright 2004 The WebRTC Project Authors. All rights reserved.
3 *
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
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
11#include "webrtc/p2p/base/stunport.h"
12
13#include "webrtc/p2p/base/common.h"
14#include "webrtc/p2p/base/portallocator.h"
15#include "webrtc/p2p/base/stun.h"
Edward Lemurc20978e2017-07-06 19:44:34 +020016#include "webrtc/rtc_base/checks.h"
17#include "webrtc/rtc_base/helpers.h"
18#include "webrtc/rtc_base/ipaddress.h"
19#include "webrtc/rtc_base/logging.h"
20#include "webrtc/rtc_base/nethelpers.h"
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000021
22namespace cricket {
23
24// TODO: Move these to a common place (used in relayport too)
25const int KEEPALIVE_DELAY = 10 * 1000; // 10 seconds - sort timeouts
honghaize2af9ef2016-03-03 08:27:47 -080026const int RETRY_TIMEOUT = 50 * 1000; // 50 seconds
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000027
28// Handles a binding request sent to the STUN server.
29class StunBindingRequest : public StunRequest {
30 public:
honghaiz45b0efd2015-12-04 08:57:26 -080031 StunBindingRequest(UDPPort* port,
32 const rtc::SocketAddress& addr,
Honghai Zhang351d77b2016-05-20 15:08:29 -070033 int64_t start_time)
34 : port_(port), server_addr_(addr), start_time_(start_time) {}
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000035
36 virtual ~StunBindingRequest() {
37 }
38
39 const rtc::SocketAddress& server_addr() const { return server_addr_; }
40
Peter Thatcher1cf6f812015-05-15 10:40:45 -070041 virtual void Prepare(StunMessage* request) override {
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000042 request->SetType(STUN_BINDING_REQUEST);
43 }
44
Peter Thatcher1cf6f812015-05-15 10:40:45 -070045 virtual void OnResponse(StunMessage* response) override {
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000046 const StunAddressAttribute* addr_attr =
47 response->GetAddress(STUN_ATTR_MAPPED_ADDRESS);
48 if (!addr_attr) {
49 LOG(LS_ERROR) << "Binding response missing mapped address.";
50 } else if (addr_attr->family() != STUN_ADDRESS_IPV4 &&
51 addr_attr->family() != STUN_ADDRESS_IPV6) {
52 LOG(LS_ERROR) << "Binding address has bad family";
53 } else {
54 rtc::SocketAddress addr(addr_attr->ipaddr(), addr_attr->port());
55 port_->OnStunBindingRequestSucceeded(server_addr_, addr);
56 }
57
honghaize2af9ef2016-03-03 08:27:47 -080058 // The keep-alive requests will be stopped after its lifetime has passed.
nisse1bffc1d2016-05-02 08:18:55 -070059 if (WithinLifetime(rtc::TimeMillis())) {
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000060 port_->requests_.SendDelayed(
Honghai Zhang351d77b2016-05-20 15:08:29 -070061 new StunBindingRequest(port_, server_addr_, start_time_),
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000062 port_->stun_keepalive_delay());
63 }
64 }
65
Peter Thatcher1cf6f812015-05-15 10:40:45 -070066 virtual void OnErrorResponse(StunMessage* response) override {
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000067 const StunErrorCodeAttribute* attr = response->GetErrorCode();
68 if (!attr) {
deadbeef996fc6b2017-04-26 09:21:22 -070069 LOG(LS_ERROR) << "Missing binding response error code.";
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000070 } else {
71 LOG(LS_ERROR) << "Binding error response:"
honghaize2af9ef2016-03-03 08:27:47 -080072 << " class=" << attr->eclass()
73 << " number=" << attr->number() << " reason='"
74 << attr->reason() << "'";
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000075 }
76
77 port_->OnStunBindingOrResolveRequestFailed(server_addr_);
78
nisse1bffc1d2016-05-02 08:18:55 -070079 int64_t now = rtc::TimeMillis();
honghaize2af9ef2016-03-03 08:27:47 -080080 if (WithinLifetime(now) &&
Honghai Zhang82d78622016-05-06 11:29:15 -070081 rtc::TimeDiff(now, start_time_) < RETRY_TIMEOUT) {
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000082 port_->requests_.SendDelayed(
Honghai Zhang351d77b2016-05-20 15:08:29 -070083 new StunBindingRequest(port_, server_addr_, start_time_),
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000084 port_->stun_keepalive_delay());
85 }
86 }
Peter Thatcher1cf6f812015-05-15 10:40:45 -070087 virtual void OnTimeout() override {
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000088 LOG(LS_ERROR) << "Binding request timed out from "
honghaize2af9ef2016-03-03 08:27:47 -080089 << port_->GetLocalAddress().ToSensitiveString() << " ("
90 << port_->Network()->name() << ")";
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000091
92 port_->OnStunBindingOrResolveRequestFailed(server_addr_);
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +000093 }
94
95 private:
honghaize2af9ef2016-03-03 08:27:47 -080096 // Returns true if |now| is within the lifetime of the request (a negative
97 // lifetime means infinite).
honghaiz34b11eb2016-03-16 08:55:44 -070098 bool WithinLifetime(int64_t now) const {
Honghai Zhang351d77b2016-05-20 15:08:29 -070099 int lifetime = port_->stun_keepalive_lifetime();
100 return lifetime < 0 || rtc::TimeDiff(now, start_time_) <= lifetime;
honghaize2af9ef2016-03-03 08:27:47 -0800101 }
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000102 UDPPort* port_;
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000103 const rtc::SocketAddress server_addr_;
honghaize2af9ef2016-03-03 08:27:47 -0800104
honghaiz34b11eb2016-03-16 08:55:44 -0700105 int64_t start_time_;
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000106};
107
108UDPPort::AddressResolver::AddressResolver(
109 rtc::PacketSocketFactory* factory)
110 : socket_factory_(factory) {}
111
112UDPPort::AddressResolver::~AddressResolver() {
113 for (ResolverMap::iterator it = resolvers_.begin();
114 it != resolvers_.end(); ++it) {
Guo-wei Shieh8a4f5472015-10-30 09:12:34 -0700115 // TODO(guoweis): Change to asynchronous DNS resolution to prevent the hang
116 // when passing true to the Destroy() which is a safer way to avoid the code
117 // unloaded before the thread exits. Please see webrtc bug 5139.
118 it->second->Destroy(false);
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000119 }
120}
121
122void UDPPort::AddressResolver::Resolve(
123 const rtc::SocketAddress& address) {
124 if (resolvers_.find(address) != resolvers_.end())
125 return;
126
127 rtc::AsyncResolverInterface* resolver =
128 socket_factory_->CreateAsyncResolver();
129 resolvers_.insert(
130 std::pair<rtc::SocketAddress, rtc::AsyncResolverInterface*>(
131 address, resolver));
132
133 resolver->SignalDone.connect(this,
134 &UDPPort::AddressResolver::OnResolveResult);
135
136 resolver->Start(address);
137}
138
139bool UDPPort::AddressResolver::GetResolvedAddress(
140 const rtc::SocketAddress& input,
141 int family,
142 rtc::SocketAddress* output) const {
143 ResolverMap::const_iterator it = resolvers_.find(input);
144 if (it == resolvers_.end())
145 return false;
146
147 return it->second->GetResolvedAddress(family, output);
148}
149
150void UDPPort::AddressResolver::OnResolveResult(
151 rtc::AsyncResolverInterface* resolver) {
152 for (ResolverMap::iterator it = resolvers_.begin();
153 it != resolvers_.end(); ++it) {
154 if (it->second == resolver) {
155 SignalDone(it->first, resolver->GetError());
156 return;
157 }
158 }
159}
160
161UDPPort::UDPPort(rtc::Thread* thread,
162 rtc::PacketSocketFactory* factory,
163 rtc::Network* network,
164 rtc::AsyncPacketSocket* socket,
pkasting@chromium.org332331f2014-11-06 20:19:22 +0000165 const std::string& username,
pthatcher@webrtc.org0ba15332015-01-10 00:47:02 +0000166 const std::string& password,
Guo-wei Shiehfe3bc9d2015-08-20 08:48:20 -0700167 const std::string& origin,
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800168 bool emit_local_for_anyaddress)
169 : Port(thread,
Honghai Zhangd00c0572016-06-28 09:44:47 -0700170 LOCAL_PORT_TYPE,
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800171 factory,
172 network,
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800173 username,
174 password),
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000175 requests_(thread),
176 socket_(socket),
177 error_(0),
178 ready_(false),
Guo-wei Shiehfe3bc9d2015-08-20 08:48:20 -0700179 stun_keepalive_delay_(KEEPALIVE_DELAY),
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800180 emit_local_for_anyaddress_(emit_local_for_anyaddress) {
pthatcher@webrtc.org0ba15332015-01-10 00:47:02 +0000181 requests_.set_origin(origin);
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000182}
183
184UDPPort::UDPPort(rtc::Thread* thread,
185 rtc::PacketSocketFactory* factory,
186 rtc::Network* network,
Peter Boström0c4e06b2015-10-07 12:23:21 +0200187 uint16_t min_port,
188 uint16_t max_port,
pkasting@chromium.org332331f2014-11-06 20:19:22 +0000189 const std::string& username,
pthatcher@webrtc.org0ba15332015-01-10 00:47:02 +0000190 const std::string& password,
Guo-wei Shiehfe3bc9d2015-08-20 08:48:20 -0700191 const std::string& origin,
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800192 bool emit_local_for_anyaddress)
Peter Boström0c4e06b2015-10-07 12:23:21 +0200193 : Port(thread,
194 LOCAL_PORT_TYPE,
195 factory,
196 network,
Peter Boström0c4e06b2015-10-07 12:23:21 +0200197 min_port,
198 max_port,
199 username,
200 password),
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000201 requests_(thread),
202 socket_(NULL),
203 error_(0),
204 ready_(false),
Guo-wei Shiehfe3bc9d2015-08-20 08:48:20 -0700205 stun_keepalive_delay_(KEEPALIVE_DELAY),
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800206 emit_local_for_anyaddress_(emit_local_for_anyaddress) {
pthatcher@webrtc.org0ba15332015-01-10 00:47:02 +0000207 requests_.set_origin(origin);
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000208}
209
210bool UDPPort::Init() {
Honghai Zhang351d77b2016-05-20 15:08:29 -0700211 stun_keepalive_lifetime_ = GetStunKeepaliveLifetime();
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000212 if (!SharedSocket()) {
nisseede5da42017-01-12 05:15:36 -0800213 RTC_DCHECK(socket_ == NULL);
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000214 socket_ = socket_factory()->CreateUdpSocket(
deadbeef5c3c1042017-08-04 15:01:57 -0700215 rtc::SocketAddress(Network()->GetBestIP(), 0), min_port(), max_port());
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000216 if (!socket_) {
217 LOG_J(LS_WARNING, this) << "UDP socket creation failed";
218 return false;
219 }
220 socket_->SignalReadPacket.connect(this, &UDPPort::OnReadPacket);
221 }
stefanc1aeaf02015-10-15 07:26:07 -0700222 socket_->SignalSentPacket.connect(this, &UDPPort::OnSentPacket);
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000223 socket_->SignalReadyToSend.connect(this, &UDPPort::OnReadyToSend);
224 socket_->SignalAddressReady.connect(this, &UDPPort::OnLocalAddressReady);
225 requests_.SignalSendPacket.connect(this, &UDPPort::OnSendPacket);
226 return true;
227}
228
229UDPPort::~UDPPort() {
230 if (!SharedSocket())
231 delete socket_;
232}
233
234void UDPPort::PrepareAddress() {
nisseede5da42017-01-12 05:15:36 -0800235 RTC_DCHECK(requests_.empty());
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000236 if (socket_->GetState() == rtc::AsyncPacketSocket::STATE_BOUND) {
237 OnLocalAddressReady(socket_, socket_->GetLocalAddress());
238 }
239}
240
241void UDPPort::MaybePrepareStunCandidate() {
242 // Sending binding request to the STUN server if address is available to
243 // prepare STUN candidate.
244 if (!server_addresses_.empty()) {
245 SendStunBindingRequests();
246 } else {
247 // Port is done allocating candidates.
248 MaybeSetPortCompleteOrError();
249 }
250}
251
252Connection* UDPPort::CreateConnection(const Candidate& address,
Honghai Zhangf9945b22015-12-15 12:20:13 -0800253 CandidateOrigin origin) {
254 if (!SupportsProtocol(address.protocol())) {
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000255 return NULL;
Honghai Zhangf9945b22015-12-15 12:20:13 -0800256 }
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000257
258 if (!IsCompatibleAddress(address.address())) {
259 return NULL;
260 }
261
262 if (SharedSocket() && Candidates()[0].type() != LOCAL_PORT_TYPE) {
nissec80e7412017-01-11 05:56:46 -0800263 RTC_NOTREACHED();
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000264 return NULL;
265 }
266
267 Connection* conn = new ProxyConnection(this, 0, address);
honghaiz36f50e82016-06-01 15:57:03 -0700268 AddOrReplaceConnection(conn);
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000269 return conn;
270}
271
272int UDPPort::SendTo(const void* data, size_t size,
273 const rtc::SocketAddress& addr,
274 const rtc::PacketOptions& options,
275 bool payload) {
276 int sent = socket_->SendTo(data, size, addr, options);
277 if (sent < 0) {
278 error_ = socket_->GetError();
279 LOG_J(LS_ERROR, this) << "UDP send of " << size
280 << " bytes failed with error " << error_;
281 }
282 return sent;
283}
284
Honghai Zhang351d77b2016-05-20 15:08:29 -0700285void UDPPort::UpdateNetworkCost() {
286 Port::UpdateNetworkCost();
287 stun_keepalive_lifetime_ = GetStunKeepaliveLifetime();
288}
289
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000290int UDPPort::SetOption(rtc::Socket::Option opt, int value) {
291 return socket_->SetOption(opt, value);
292}
293
294int UDPPort::GetOption(rtc::Socket::Option opt, int* value) {
295 return socket_->GetOption(opt, value);
296}
297
298int UDPPort::GetError() {
299 return error_;
300}
301
302void UDPPort::OnLocalAddressReady(rtc::AsyncPacketSocket* socket,
303 const rtc::SocketAddress& address) {
Guo-wei Shiehfe3bc9d2015-08-20 08:48:20 -0700304 // When adapter enumeration is disabled and binding to the any address, the
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800305 // default local address will be issued as a candidate instead if
306 // |emit_local_for_anyaddress| is true. This is to allow connectivity for
307 // applications which absolutely requires a HOST candidate.
Guo-wei Shiehfe3bc9d2015-08-20 08:48:20 -0700308 rtc::SocketAddress addr = address;
Guo-wei Shiehe03cab92015-11-11 11:11:19 -0800309
310 // If MaybeSetDefaultLocalAddress fails, we keep the "any" IP so that at
311 // least the port is listening.
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800312 MaybeSetDefaultLocalAddress(&addr);
Guo-wei Shiehfe3bc9d2015-08-20 08:48:20 -0700313
314 AddAddress(addr, addr, rtc::SocketAddress(), UDP_PROTOCOL_NAME, "", "",
zhihuang26d99c22017-02-13 12:47:27 -0800315 LOCAL_PORT_TYPE, ICE_TYPE_PREFERENCE_HOST, 0, "", false);
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000316 MaybePrepareStunCandidate();
317}
318
tfarina20a34612015-11-02 16:20:22 -0800319void UDPPort::OnReadPacket(rtc::AsyncPacketSocket* socket,
320 const char* data,
321 size_t size,
322 const rtc::SocketAddress& remote_addr,
323 const rtc::PacketTime& packet_time) {
nisseede5da42017-01-12 05:15:36 -0800324 RTC_DCHECK(socket == socket_);
325 RTC_DCHECK(!remote_addr.IsUnresolvedIP());
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000326
327 // Look for a response from the STUN server.
328 // Even if the response doesn't match one of our outstanding requests, we
329 // will eat it because it might be a response to a retransmitted packet, and
330 // we already cleared the request when we got the first response.
331 if (server_addresses_.find(remote_addr) != server_addresses_.end()) {
332 requests_.CheckResponse(data, size);
333 return;
334 }
335
336 if (Connection* conn = GetConnection(remote_addr)) {
337 conn->OnReadPacket(data, size, packet_time);
338 } else {
339 Port::OnReadPacket(data, size, remote_addr, PROTO_UDP);
340 }
341}
342
stefanc1aeaf02015-10-15 07:26:07 -0700343void UDPPort::OnSentPacket(rtc::AsyncPacketSocket* socket,
344 const rtc::SentPacket& sent_packet) {
Stefan Holmer55674ff2016-01-14 15:49:16 +0100345 PortInterface::SignalSentPacket(sent_packet);
stefanc1aeaf02015-10-15 07:26:07 -0700346}
347
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000348void UDPPort::OnReadyToSend(rtc::AsyncPacketSocket* socket) {
349 Port::OnReadyToSend();
350}
351
352void UDPPort::SendStunBindingRequests() {
353 // We will keep pinging the stun server to make sure our NAT pin-hole stays
honghaiz45b0efd2015-12-04 08:57:26 -0800354 // open until the deadline (specified in SendStunBindingRequest).
nisseede5da42017-01-12 05:15:36 -0800355 RTC_DCHECK(requests_.empty());
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000356
357 for (ServerAddresses::const_iterator it = server_addresses_.begin();
358 it != server_addresses_.end(); ++it) {
359 SendStunBindingRequest(*it);
360 }
361}
362
363void UDPPort::ResolveStunAddress(const rtc::SocketAddress& stun_addr) {
364 if (!resolver_) {
365 resolver_.reset(new AddressResolver(socket_factory()));
366 resolver_->SignalDone.connect(this, &UDPPort::OnResolveResult);
367 }
368
Honghai Zhang0f490a52015-12-07 12:06:20 -0800369 LOG_J(LS_INFO, this) << "Starting STUN host lookup for "
370 << stun_addr.ToSensitiveString();
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000371 resolver_->Resolve(stun_addr);
372}
373
374void UDPPort::OnResolveResult(const rtc::SocketAddress& input,
375 int error) {
nisseede5da42017-01-12 05:15:36 -0800376 RTC_DCHECK(resolver_.get() != NULL);
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000377
378 rtc::SocketAddress resolved;
deadbeef5c3c1042017-08-04 15:01:57 -0700379 if (error != 0 || !resolver_->GetResolvedAddress(
380 input, Network()->GetBestIP().family(), &resolved)) {
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000381 LOG_J(LS_WARNING, this) << "StunPort: stun host lookup received error "
382 << error;
383 OnStunBindingOrResolveRequestFailed(input);
384 return;
385 }
386
387 server_addresses_.erase(input);
388
389 if (server_addresses_.find(resolved) == server_addresses_.end()) {
390 server_addresses_.insert(resolved);
391 SendStunBindingRequest(resolved);
392 }
393}
394
tfarina20a34612015-11-02 16:20:22 -0800395void UDPPort::SendStunBindingRequest(const rtc::SocketAddress& stun_addr) {
396 if (stun_addr.IsUnresolvedIP()) {
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000397 ResolveStunAddress(stun_addr);
398
399 } else if (socket_->GetState() == rtc::AsyncPacketSocket::STATE_BOUND) {
400 // Check if |server_addr_| is compatible with the port's ip.
401 if (IsCompatibleAddress(stun_addr)) {
Honghai Zhang351d77b2016-05-20 15:08:29 -0700402 requests_.Send(
403 new StunBindingRequest(this, stun_addr, rtc::TimeMillis()));
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000404 } else {
405 // Since we can't send stun messages to the server, we should mark this
406 // port ready.
407 LOG(LS_WARNING) << "STUN server address is incompatible.";
408 OnStunBindingOrResolveRequestFailed(stun_addr);
409 }
410 }
411}
412
Guo-wei Shiehe03cab92015-11-11 11:11:19 -0800413bool UDPPort::MaybeSetDefaultLocalAddress(rtc::SocketAddress* addr) const {
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800414 if (!addr->IsAnyIP() || !emit_local_for_anyaddress_ ||
415 !Network()->default_local_address_provider()) {
Guo-wei Shiehe03cab92015-11-11 11:11:19 -0800416 return true;
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800417 }
418 rtc::IPAddress default_address;
419 bool result =
420 Network()->default_local_address_provider()->GetDefaultLocalAddress(
421 addr->family(), &default_address);
Guo-wei Shieh953eabc2015-11-24 12:00:33 -0800422 if (!result || default_address.IsNil()) {
Guo-wei Shiehe03cab92015-11-11 11:11:19 -0800423 return false;
424 }
425
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800426 addr->SetIP(default_address);
Guo-wei Shiehe03cab92015-11-11 11:11:19 -0800427 return true;
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800428}
429
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000430void UDPPort::OnStunBindingRequestSucceeded(
431 const rtc::SocketAddress& stun_server_addr,
432 const rtc::SocketAddress& stun_reflected_addr) {
433 if (bind_request_succeeded_servers_.find(stun_server_addr) !=
434 bind_request_succeeded_servers_.end()) {
435 return;
436 }
437 bind_request_succeeded_servers_.insert(stun_server_addr);
438
439 // If socket is shared and |stun_reflected_addr| is equal to local socket
440 // address, or if the same address has been added by another STUN server,
441 // then discarding the stun address.
442 // For STUN, related address is the local socket address.
443 if ((!SharedSocket() || stun_reflected_addr != socket_->GetLocalAddress()) &&
444 !HasCandidateWithAddress(stun_reflected_addr)) {
445
446 rtc::SocketAddress related_address = socket_->GetLocalAddress();
Guo-wei Shiehe03cab92015-11-11 11:11:19 -0800447 // If we can't stamp the related address correctly, empty it to avoid leak.
Taylor Brandstetter417eebe2016-05-23 16:02:19 -0700448 if (!MaybeSetDefaultLocalAddress(&related_address)) {
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000449 related_address = rtc::EmptySocketAddressWithFamily(
450 related_address.family());
451 }
452
zhihuang26d99c22017-02-13 12:47:27 -0800453 std::ostringstream url;
454 url << "stun:" << stun_server_addr.ipaddr().ToString() << ":"
455 << stun_server_addr.port();
Guo-wei Shieh3d564c12015-08-19 16:51:15 -0700456 AddAddress(stun_reflected_addr, socket_->GetLocalAddress(), related_address,
457 UDP_PROTOCOL_NAME, "", "", STUN_PORT_TYPE,
zhihuang26d99c22017-02-13 12:47:27 -0800458 ICE_TYPE_PREFERENCE_SRFLX, 0, url.str(), false);
henrike@webrtc.org269fb4b2014-10-28 22:20:11 +0000459 }
460 MaybeSetPortCompleteOrError();
461}
462
463void UDPPort::OnStunBindingOrResolveRequestFailed(
464 const rtc::SocketAddress& stun_server_addr) {
465 if (bind_request_failed_servers_.find(stun_server_addr) !=
466 bind_request_failed_servers_.end()) {
467 return;
468 }
469 bind_request_failed_servers_.insert(stun_server_addr);
470 MaybeSetPortCompleteOrError();
471}
472
473void UDPPort::MaybeSetPortCompleteOrError() {
474 if (ready_)
475 return;
476
477 // Do not set port ready if we are still waiting for bind responses.
478 const size_t servers_done_bind_request = bind_request_failed_servers_.size() +
479 bind_request_succeeded_servers_.size();
480 if (server_addresses_.size() != servers_done_bind_request) {
481 return;
482 }
483
484 // Setting ready status.
485 ready_ = true;
486
487 // The port is "completed" if there is no stun server provided, or the bind
488 // request succeeded for any stun server, or the socket is shared.
489 if (server_addresses_.empty() ||
490 bind_request_succeeded_servers_.size() > 0 ||
491 SharedSocket()) {
492 SignalPortComplete(this);
493 } else {
494 SignalPortError(this);
495 }
496}
497
498// TODO: merge this with SendTo above.
499void UDPPort::OnSendPacket(const void* data, size_t size, StunRequest* req) {
500 StunBindingRequest* sreq = static_cast<StunBindingRequest*>(req);
501 rtc::PacketOptions options(DefaultDscpValue());
502 if (socket_->SendTo(data, size, sreq->server_addr(), options) < 0)
503 PLOG(LERROR, socket_->GetError()) << "sendto";
504}
505
506bool UDPPort::HasCandidateWithAddress(const rtc::SocketAddress& addr) const {
507 const std::vector<Candidate>& existing_candidates = Candidates();
508 std::vector<Candidate>::const_iterator it = existing_candidates.begin();
509 for (; it != existing_candidates.end(); ++it) {
510 if (it->address() == addr)
511 return true;
512 }
513 return false;
514}
515
516} // namespace cricket