OLD | NEW |
1 /* | 1 // TODO(pthatcher): Remove this file once Chrome's build files no |
2 * Copyright 2004 The WebRTC Project Authors. All rights reserved. | 2 // longer refer to it. |
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/rawtransportchannel.h" | |
12 | |
13 #include <string> | |
14 #include <vector> | |
15 #include "webrtc/p2p/base/constants.h" | |
16 #include "webrtc/p2p/base/portallocator.h" | |
17 #include "webrtc/p2p/base/portinterface.h" | |
18 #include "webrtc/p2p/base/rawtransport.h" | |
19 #include "webrtc/p2p/base/relayport.h" | |
20 #include "webrtc/p2p/base/stunport.h" | |
21 #include "webrtc/base/common.h" | |
22 | |
23 #if defined(FEATURE_ENABLE_PSTN) | |
24 | |
25 namespace { | |
26 | |
27 const uint32 MSG_DESTROY_RTC_UNUSED_PORTS = 1; | |
28 | |
29 } // namespace | |
30 | |
31 namespace cricket { | |
32 | |
33 RawTransportChannel::RawTransportChannel(const std::string& content_name, | |
34 int component, | |
35 RawTransport* transport, | |
36 rtc::Thread *worker_thread, | |
37 PortAllocator *allocator) | |
38 : TransportChannelImpl(content_name, component), | |
39 raw_transport_(transport), | |
40 allocator_(allocator), | |
41 allocator_session_(NULL), | |
42 stun_port_(NULL), | |
43 relay_port_(NULL), | |
44 port_(NULL), | |
45 use_relay_(false) { | |
46 if (worker_thread == NULL) | |
47 worker_thread_ = raw_transport_->worker_thread(); | |
48 else | |
49 worker_thread_ = worker_thread; | |
50 } | |
51 | |
52 RawTransportChannel::~RawTransportChannel() { | |
53 delete allocator_session_; | |
54 } | |
55 | |
56 int RawTransportChannel::SendPacket(const char *data, size_t size, | |
57 const rtc::PacketOptions& options, | |
58 int flags) { | |
59 if (port_ == NULL) | |
60 return -1; | |
61 if (remote_address_.IsNil()) | |
62 return -1; | |
63 if (flags != 0) | |
64 return -1; | |
65 return port_->SendTo(data, size, remote_address_, options, true); | |
66 } | |
67 | |
68 int RawTransportChannel::SetOption(rtc::Socket::Option opt, int value) { | |
69 // TODO: allow these to be set before we have a port | |
70 if (port_ == NULL) | |
71 return -1; | |
72 return port_->SetOption(opt, value); | |
73 } | |
74 | |
75 bool RawTransportChannel::GetOption(rtc::Socket::Option opt, int* value) { | |
76 return false; | |
77 } | |
78 | |
79 int RawTransportChannel::GetError() { | |
80 return (port_ != NULL) ? port_->GetError() : 0; | |
81 } | |
82 | |
83 void RawTransportChannel::Connect() { | |
84 // Create an allocator that only returns stun and relay ports. | |
85 // Use empty string for ufrag and pwd here. There won't be any STUN or relay | |
86 // interactions when using RawTC. | |
87 // TODO: Change raw to only use local udp ports. | |
88 allocator_session_ = allocator_->CreateSession( | |
89 SessionId(), content_name(), component(), "", ""); | |
90 | |
91 uint32 flags = PORTALLOCATOR_DISABLE_UDP | PORTALLOCATOR_DISABLE_TCP; | |
92 | |
93 #if !defined(FEATURE_ENABLE_STUN_CLASSIFICATION) | |
94 flags |= PORTALLOCATOR_DISABLE_RELAY; | |
95 #endif | |
96 allocator_session_->set_flags(flags); | |
97 allocator_session_->SignalPortReady.connect( | |
98 this, &RawTransportChannel::OnPortReady); | |
99 allocator_session_->SignalCandidatesReady.connect( | |
100 this, &RawTransportChannel::OnCandidatesReady); | |
101 | |
102 // The initial ports will include stun. | |
103 allocator_session_->StartGettingPorts(); | |
104 } | |
105 | |
106 void RawTransportChannel::Reset() { | |
107 set_readable(false); | |
108 set_writable(false); | |
109 | |
110 delete allocator_session_; | |
111 | |
112 allocator_session_ = NULL; | |
113 stun_port_ = NULL; | |
114 relay_port_ = NULL; | |
115 port_ = NULL; | |
116 remote_address_ = rtc::SocketAddress(); | |
117 } | |
118 | |
119 void RawTransportChannel::OnCandidate(const Candidate& candidate) { | |
120 remote_address_ = candidate.address(); | |
121 ASSERT(!remote_address_.IsNil()); | |
122 set_readable(true); | |
123 | |
124 // We can write once we have a port and a remote address. | |
125 if (port_ != NULL) | |
126 SetWritable(); | |
127 } | |
128 | |
129 void RawTransportChannel::OnRemoteAddress( | |
130 const rtc::SocketAddress& remote_address) { | |
131 remote_address_ = remote_address; | |
132 set_readable(true); | |
133 | |
134 if (port_ != NULL) | |
135 SetWritable(); | |
136 } | |
137 | |
138 // Note about stun classification | |
139 // Code to classify our NAT type and use the relay port if we are behind an | |
140 // asymmetric NAT is under a FEATURE_ENABLE_STUN_CLASSIFICATION #define. | |
141 // To turn this one we will have to enable a second stun address and make sure | |
142 // that the relay server works for raw UDP. | |
143 // | |
144 // Another option is to classify the NAT type early and not offer the raw | |
145 // transport type at all if we can't support it. | |
146 | |
147 void RawTransportChannel::OnPortReady( | |
148 PortAllocatorSession* session, PortInterface* port) { | |
149 ASSERT(session == allocator_session_); | |
150 | |
151 if (port->Type() == STUN_PORT_TYPE) { | |
152 stun_port_ = static_cast<StunPort*>(port); | |
153 } else if (port->Type() == RELAY_PORT_TYPE) { | |
154 relay_port_ = static_cast<RelayPort*>(port); | |
155 } else { | |
156 ASSERT(false); | |
157 } | |
158 } | |
159 | |
160 void RawTransportChannel::OnCandidatesReady( | |
161 PortAllocatorSession *session, const std::vector<Candidate>& candidates) { | |
162 ASSERT(session == allocator_session_); | |
163 ASSERT(candidates.size() >= 1); | |
164 | |
165 // The most recent candidate is the one we haven't seen yet. | |
166 Candidate c = candidates[candidates.size() - 1]; | |
167 | |
168 if (c.type() == STUN_PORT_TYPE) { | |
169 ASSERT(stun_port_ != NULL); | |
170 | |
171 #if defined(FEATURE_ENABLE_STUN_CLASSIFICATION) | |
172 // We need to wait until we have two addresses. | |
173 if (stun_port_->candidates().size() < 2) | |
174 return; | |
175 | |
176 // This is the second address. If these addresses are the same, then we | |
177 // are not behind a symmetric NAT. Hence, a stun port should be sufficient. | |
178 if (stun_port_->candidates()[0].address() == | |
179 stun_port_->candidates()[1].address()) { | |
180 SetPort(stun_port_); | |
181 return; | |
182 } | |
183 | |
184 // We will need to use relay. | |
185 use_relay_ = true; | |
186 | |
187 // If we already have a relay address, we're good. Otherwise, we will need | |
188 // to wait until one arrives. | |
189 if (relay_port_->candidates().size() > 0) | |
190 SetPort(relay_port_); | |
191 #else // defined(FEATURE_ENABLE_STUN_CLASSIFICATION) | |
192 // Always use the stun port. We don't classify right now so just assume it | |
193 // will work fine. | |
194 SetPort(stun_port_); | |
195 #endif | |
196 } else if (c.type() == RELAY_PORT_TYPE) { | |
197 if (use_relay_) | |
198 SetPort(relay_port_); | |
199 } else { | |
200 ASSERT(false); | |
201 } | |
202 } | |
203 | |
204 void RawTransportChannel::SetPort(PortInterface* port) { | |
205 ASSERT(port_ == NULL); | |
206 port_ = port; | |
207 | |
208 // We don't need any ports other than the one we picked. | |
209 allocator_session_->StopGettingPorts(); | |
210 worker_thread_->Post( | |
211 this, MSG_DESTROY_RTC_UNUSED_PORTS, NULL); | |
212 | |
213 // Send a message to the other client containing our address. | |
214 | |
215 ASSERT(port_->Candidates().size() >= 1); | |
216 ASSERT(port_->Candidates()[0].protocol() == "udp"); | |
217 SignalCandidateReady(this, port_->Candidates()[0]); | |
218 | |
219 // Read all packets from this port. | |
220 port_->EnablePortPackets(); | |
221 port_->SignalReadPacket.connect(this, &RawTransportChannel::OnReadPacket); | |
222 | |
223 // We can write once we have a port and a remote address. | |
224 if (!remote_address_.IsAny()) | |
225 SetWritable(); | |
226 } | |
227 | |
228 void RawTransportChannel::SetWritable() { | |
229 ASSERT(port_ != NULL); | |
230 ASSERT(!remote_address_.IsAny()); | |
231 | |
232 set_writable(true); | |
233 | |
234 Candidate remote_candidate; | |
235 remote_candidate.set_address(remote_address_); | |
236 SignalRouteChange(this, remote_candidate); | |
237 } | |
238 | |
239 void RawTransportChannel::OnReadPacket( | |
240 PortInterface* port, const char* data, size_t size, | |
241 const rtc::SocketAddress& addr) { | |
242 ASSERT(port_ == port); | |
243 SignalReadPacket(this, data, size, rtc::CreatePacketTime(0), 0); | |
244 } | |
245 | |
246 void RawTransportChannel::OnMessage(rtc::Message* msg) { | |
247 ASSERT(msg->message_id == MSG_DESTROY_RTC_UNUSED_PORTS); | |
248 ASSERT(port_ != NULL); | |
249 if (port_ != stun_port_) { | |
250 stun_port_->Destroy(); | |
251 stun_port_ = NULL; | |
252 } | |
253 if (port_ != relay_port_ && relay_port_ != NULL) { | |
254 relay_port_->Destroy(); | |
255 relay_port_ = NULL; | |
256 } | |
257 } | |
258 | |
259 } // namespace cricket | |
260 #endif // defined(FEATURE_ENABLE_PSTN) | |
OLD | NEW |