blob: c35a687fff54cf8241cd8eea117bdc235cc22056 [file] [log] [blame]
henrike@webrtc.org47be73b2014-05-13 18:00:261/*
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/base/firewallsocketserver.h"
12
13#include <assert.h>
14
15#include <algorithm>
16
17#include "webrtc/base/asyncsocket.h"
18#include "webrtc/base/logging.h"
19
20namespace rtc {
21
22class FirewallSocket : public AsyncSocketAdapter {
23 public:
24 FirewallSocket(FirewallSocketServer* server, AsyncSocket* socket, int type)
25 : AsyncSocketAdapter(socket), server_(server), type_(type) {
26 }
27
kwiberg@webrtc.org786b6342015-03-09 22:21:5328 int Connect(const SocketAddress& addr) override {
henrike@webrtc.org47be73b2014-05-13 18:00:2629 if (type_ == SOCK_STREAM) {
30 if (!server_->Check(FP_TCP, GetLocalAddress(), addr)) {
31 LOG(LS_VERBOSE) << "FirewallSocket outbound TCP connection from "
32 << GetLocalAddress().ToSensitiveString() << " to "
33 << addr.ToSensitiveString() << " denied";
34 // TODO: Handle this asynchronously.
35 SetError(EHOSTUNREACH);
36 return SOCKET_ERROR;
37 }
38 }
39 return AsyncSocketAdapter::Connect(addr);
40 }
kwiberg@webrtc.org786b6342015-03-09 22:21:5341 int Send(const void* pv, size_t cb) override {
henrike@webrtc.org47be73b2014-05-13 18:00:2642 return SendTo(pv, cb, GetRemoteAddress());
43 }
kwiberg@webrtc.org786b6342015-03-09 22:21:5344 int SendTo(const void* pv, size_t cb, const SocketAddress& addr) override {
henrike@webrtc.org47be73b2014-05-13 18:00:2645 if (type_ == SOCK_DGRAM) {
46 if (!server_->Check(FP_UDP, GetLocalAddress(), addr)) {
47 LOG(LS_VERBOSE) << "FirewallSocket outbound UDP packet from "
48 << GetLocalAddress().ToSensitiveString() << " to "
49 << addr.ToSensitiveString() << " dropped";
50 return static_cast<int>(cb);
51 }
52 }
53 return AsyncSocketAdapter::SendTo(pv, cb, addr);
54 }
kwiberg@webrtc.org786b6342015-03-09 22:21:5355 int Recv(void* pv, size_t cb) override {
henrike@webrtc.org47be73b2014-05-13 18:00:2656 SocketAddress addr;
57 return RecvFrom(pv, cb, &addr);
58 }
kwiberg@webrtc.org786b6342015-03-09 22:21:5359 int RecvFrom(void* pv, size_t cb, SocketAddress* paddr) override {
henrike@webrtc.org47be73b2014-05-13 18:00:2660 if (type_ == SOCK_DGRAM) {
61 while (true) {
62 int res = AsyncSocketAdapter::RecvFrom(pv, cb, paddr);
63 if (res <= 0)
64 return res;
65 if (server_->Check(FP_UDP, *paddr, GetLocalAddress()))
66 return res;
67 LOG(LS_VERBOSE) << "FirewallSocket inbound UDP packet from "
68 << paddr->ToSensitiveString() << " to "
69 << GetLocalAddress().ToSensitiveString() << " dropped";
70 }
71 }
72 return AsyncSocketAdapter::RecvFrom(pv, cb, paddr);
73 }
74
kwiberg@webrtc.org786b6342015-03-09 22:21:5375 int Listen(int backlog) override {
henrike@webrtc.org47be73b2014-05-13 18:00:2676 if (!server_->tcp_listen_enabled()) {
77 LOG(LS_VERBOSE) << "FirewallSocket listen attempt denied";
78 return -1;
79 }
80
81 return AsyncSocketAdapter::Listen(backlog);
82 }
kwiberg@webrtc.org786b6342015-03-09 22:21:5383 AsyncSocket* Accept(SocketAddress* paddr) override {
henrike@webrtc.org47be73b2014-05-13 18:00:2684 SocketAddress addr;
85 while (AsyncSocket* sock = AsyncSocketAdapter::Accept(&addr)) {
86 if (server_->Check(FP_TCP, addr, GetLocalAddress())) {
87 if (paddr)
88 *paddr = addr;
89 return sock;
90 }
91 sock->Close();
92 delete sock;
93 LOG(LS_VERBOSE) << "FirewallSocket inbound TCP connection from "
94 << addr.ToSensitiveString() << " to "
95 << GetLocalAddress().ToSensitiveString() << " denied";
96 }
97 return 0;
98 }
99
100 private:
101 FirewallSocketServer* server_;
102 int type_;
103};
104
105FirewallSocketServer::FirewallSocketServer(SocketServer* server,
106 FirewallManager* manager,
107 bool should_delete_server)
108 : server_(server), manager_(manager),
109 should_delete_server_(should_delete_server),
110 udp_sockets_enabled_(true), tcp_sockets_enabled_(true),
111 tcp_listen_enabled_(true) {
112 if (manager_)
113 manager_->AddServer(this);
114}
115
116FirewallSocketServer::~FirewallSocketServer() {
117 if (manager_)
118 manager_->RemoveServer(this);
119
120 if (server_ && should_delete_server_) {
121 delete server_;
122 server_ = NULL;
123 }
124}
125
126void FirewallSocketServer::AddRule(bool allow, FirewallProtocol p,
127 FirewallDirection d,
128 const SocketAddress& addr) {
129 SocketAddress src, dst;
130 if (d == FD_IN) {
131 dst = addr;
132 } else {
133 src = addr;
134 }
135 AddRule(allow, p, src, dst);
136}
137
138
139void FirewallSocketServer::AddRule(bool allow, FirewallProtocol p,
140 const SocketAddress& src,
141 const SocketAddress& dst) {
142 Rule r;
143 r.allow = allow;
144 r.p = p;
145 r.src = src;
146 r.dst = dst;
147 CritScope scope(&crit_);
148 rules_.push_back(r);
149}
150
151void FirewallSocketServer::ClearRules() {
152 CritScope scope(&crit_);
153 rules_.clear();
154}
155
156bool FirewallSocketServer::Check(FirewallProtocol p,
157 const SocketAddress& src,
158 const SocketAddress& dst) {
159 CritScope scope(&crit_);
160 for (size_t i = 0; i < rules_.size(); ++i) {
161 const Rule& r = rules_[i];
162 if ((r.p != p) && (r.p != FP_ANY))
163 continue;
164 if ((r.src.ipaddr() != src.ipaddr()) && !r.src.IsNil())
165 continue;
166 if ((r.src.port() != src.port()) && (r.src.port() != 0))
167 continue;
168 if ((r.dst.ipaddr() != dst.ipaddr()) && !r.dst.IsNil())
169 continue;
170 if ((r.dst.port() != dst.port()) && (r.dst.port() != 0))
171 continue;
172 return r.allow;
173 }
174 return true;
175}
176
177Socket* FirewallSocketServer::CreateSocket(int type) {
178 return CreateSocket(AF_INET, type);
179}
180
181Socket* FirewallSocketServer::CreateSocket(int family, int type) {
182 return WrapSocket(server_->CreateAsyncSocket(family, type), type);
183}
184
185AsyncSocket* FirewallSocketServer::CreateAsyncSocket(int type) {
186 return CreateAsyncSocket(AF_INET, type);
187}
188
189AsyncSocket* FirewallSocketServer::CreateAsyncSocket(int family, int type) {
190 return WrapSocket(server_->CreateAsyncSocket(family, type), type);
191}
192
kwiberg@webrtc.org786b6342015-03-09 22:21:53193void FirewallSocketServer::SetMessageQueue(MessageQueue* queue) {
194 server_->SetMessageQueue(queue);
195}
196
197bool FirewallSocketServer::Wait(int cms, bool process_io) {
198 return server_->Wait(cms, process_io);
199}
200
201void FirewallSocketServer::WakeUp() {
202 return server_->WakeUp();
203}
204
henrike@webrtc.org47be73b2014-05-13 18:00:26205AsyncSocket* FirewallSocketServer::WrapSocket(AsyncSocket* sock, int type) {
206 if (!sock ||
207 (type == SOCK_STREAM && !tcp_sockets_enabled_) ||
208 (type == SOCK_DGRAM && !udp_sockets_enabled_)) {
209 LOG(LS_VERBOSE) << "FirewallSocketServer socket creation denied";
210 delete sock;
211 return NULL;
212 }
213 return new FirewallSocket(this, sock, type);
214}
215
216FirewallManager::FirewallManager() {
217}
218
219FirewallManager::~FirewallManager() {
220 assert(servers_.empty());
221}
222
223void FirewallManager::AddServer(FirewallSocketServer* server) {
224 CritScope scope(&crit_);
225 servers_.push_back(server);
226}
227
228void FirewallManager::RemoveServer(FirewallSocketServer* server) {
229 CritScope scope(&crit_);
230 servers_.erase(std::remove(servers_.begin(), servers_.end(), server),
231 servers_.end());
232}
233
234void FirewallManager::AddRule(bool allow, FirewallProtocol p,
235 FirewallDirection d, const SocketAddress& addr) {
236 CritScope scope(&crit_);
237 for (std::vector<FirewallSocketServer*>::const_iterator it =
238 servers_.begin(); it != servers_.end(); ++it) {
239 (*it)->AddRule(allow, p, d, addr);
240 }
241}
242
243void FirewallManager::ClearRules() {
244 CritScope scope(&crit_);
245 for (std::vector<FirewallSocketServer*>::const_iterator it =
246 servers_.begin(); it != servers_.end(); ++it) {
247 (*it)->ClearRules();
248 }
249}
250
251} // namespace rtc