Line data Source code
1 : /*
2 : Unix SMB/CIFS implementation.
3 :
4 : Fire connect requests to a host and a number of ports, with a timeout
5 : between the connect request. Return if the first connect comes back
6 : successfully or return the last error.
7 :
8 : Copyright (C) Volker Lendecke 2005
9 :
10 : This program is free software; you can redistribute it and/or modify
11 : it under the terms of the GNU General Public License as published by
12 : the Free Software Foundation; either version 3 of the License, or
13 : (at your option) any later version.
14 :
15 : This program is distributed in the hope that it will be useful,
16 : but WITHOUT ANY WARRANTY; without even the implied warranty of
17 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 : GNU General Public License for more details.
19 :
20 : You should have received a copy of the GNU General Public License
21 : along with this program. If not, see <http://www.gnu.org/licenses/>.
22 : */
23 :
24 : #include "includes.h"
25 : #include "lib/socket/socket.h"
26 : #include "lib/events/events.h"
27 : #include "libcli/composite/composite.h"
28 : #include "libcli/resolve/resolve.h"
29 :
30 : #define MULTI_PORT_DELAY 2000 /* microseconds */
31 :
32 : /*
33 : overall state
34 : */
35 : struct connect_multi_state {
36 : struct socket_address **server_address;
37 : unsigned num_address, current_address, current_port;
38 : int num_ports;
39 : uint16_t *ports;
40 :
41 : struct socket_context *sock;
42 : uint16_t result_port;
43 :
44 : int num_connects_sent, num_connects_recv;
45 :
46 : struct socket_connect_multi_ex *ex;
47 : };
48 :
49 : /*
50 : state of an individual socket_connect_send() call
51 : */
52 : struct connect_one_state {
53 : struct composite_context *result;
54 : struct socket_context *sock;
55 : struct socket_address *addr;
56 : };
57 :
58 : static void continue_resolve_name(struct composite_context *creq);
59 : static void connect_multi_timer(struct tevent_context *ev,
60 : struct tevent_timer *te,
61 : struct timeval tv, void *p);
62 : static void connect_multi_next_socket(struct composite_context *result);
63 : static void continue_one(struct composite_context *creq);
64 : static void continue_one_ex(struct tevent_req *subreq);
65 :
66 : /*
67 : setup an async socket_connect, with multiple ports
68 : */
69 41897 : _PUBLIC_ struct composite_context *socket_connect_multi_ex_send(
70 : TALLOC_CTX *mem_ctx,
71 : const char *server_name,
72 : int num_server_ports,
73 : uint16_t *server_ports,
74 : struct resolve_context *resolve_ctx,
75 : struct tevent_context *event_ctx,
76 : struct socket_connect_multi_ex *ex)
77 : {
78 964 : struct composite_context *result;
79 964 : struct connect_multi_state *multi;
80 964 : int i;
81 :
82 964 : struct nbt_name name;
83 964 : struct composite_context *creq;
84 :
85 41897 : result = talloc_zero(mem_ctx, struct composite_context);
86 41897 : if (result == NULL) return NULL;
87 41897 : result->state = COMPOSITE_STATE_IN_PROGRESS;
88 41897 : result->event_ctx = event_ctx;
89 :
90 41897 : multi = talloc_zero(result, struct connect_multi_state);
91 41897 : if (composite_nomem(multi, result)) goto failed;
92 41897 : result->private_data = multi;
93 :
94 41897 : multi->num_ports = num_server_ports;
95 41897 : multi->ports = talloc_array(multi, uint16_t, multi->num_ports);
96 41897 : if (composite_nomem(multi->ports, result)) goto failed;
97 :
98 99335 : for (i=0; i<multi->num_ports; i++) {
99 57438 : multi->ports[i] = server_ports[i];
100 : }
101 :
102 41897 : multi->ex = ex;
103 :
104 : /*
105 : we don't want to do the name resolution separately
106 : for each port, so start it now, then only start on
107 : the real sockets once we have an IP
108 : */
109 41897 : make_nbt_name_server(&name, server_name);
110 :
111 41897 : creq = resolve_name_all_send(resolve_ctx, multi, 0, multi->ports[0], &name, result->event_ctx);
112 41897 : if (composite_nomem(creq, result)) goto failed;
113 :
114 41897 : composite_continue(result, creq, continue_resolve_name, result);
115 :
116 41897 : return result;
117 :
118 :
119 0 : failed:
120 0 : composite_error(result, result->status);
121 0 : return result;
122 : }
123 :
124 : /*
125 : start connecting to the next socket/port in the list
126 : */
127 41951 : static void connect_multi_next_socket(struct composite_context *result)
128 : {
129 41951 : struct connect_multi_state *multi = talloc_get_type(result->private_data,
130 : struct connect_multi_state);
131 964 : struct connect_one_state *state;
132 964 : struct composite_context *creq;
133 41951 : int next = multi->num_connects_sent;
134 :
135 41951 : if (next == multi->num_address * multi->num_ports) {
136 : /* don't do anything, just wait for the existing ones to finish */
137 0 : return;
138 : }
139 :
140 41951 : if (multi->current_address == multi->num_address) {
141 10 : multi->current_address = 0;
142 10 : multi->current_port += 1;
143 : }
144 41951 : multi->num_connects_sent += 1;
145 :
146 41951 : if (multi->server_address == NULL || multi->server_address[multi->current_address] == NULL) {
147 0 : composite_error(result, NT_STATUS_OBJECT_NAME_NOT_FOUND);
148 0 : return;
149 : }
150 :
151 41951 : state = talloc(multi, struct connect_one_state);
152 41951 : if (composite_nomem(state, result)) return;
153 :
154 41951 : state->result = result;
155 41951 : result->status = socket_create(
156 41951 : state, multi->server_address[multi->current_address]->family,
157 : SOCKET_TYPE_STREAM, &state->sock, 0);
158 41951 : if (!composite_is_ok(result)) return;
159 :
160 41951 : state->addr = socket_address_copy(state, multi->server_address[multi->current_address]);
161 41951 : if (composite_nomem(state->addr, result)) return;
162 :
163 41951 : socket_address_set_port(state->addr, multi->ports[multi->current_port]);
164 :
165 41951 : creq = socket_connect_send(state->sock, NULL,
166 : state->addr, 0,
167 : result->event_ctx);
168 41951 : if (composite_nomem(creq, result)) return;
169 41951 : talloc_steal(state, creq);
170 :
171 41951 : multi->current_address++;
172 41951 : composite_continue(result, creq, continue_one, state);
173 :
174 : /* if there are more ports / addresses to go then setup a timer to fire when we have waited
175 : for a couple of milli-seconds, when that goes off we try the next port regardless
176 : of whether this port has completed */
177 41951 : if (multi->num_ports * multi->num_address > multi->num_connects_sent) {
178 : /* note that this timer is a child of the single
179 : connect attempt state, so it will go away when this
180 : request completes */
181 41246 : tevent_add_timer(result->event_ctx, state,
182 : timeval_current_ofs_usec(MULTI_PORT_DELAY),
183 : connect_multi_timer, result);
184 : }
185 : }
186 :
187 : /*
188 : a timer has gone off telling us that we should try the next port
189 : */
190 55 : static void connect_multi_timer(struct tevent_context *ev,
191 : struct tevent_timer *te,
192 : struct timeval tv, void *p)
193 : {
194 55 : struct composite_context *result = talloc_get_type(p, struct composite_context);
195 55 : connect_multi_next_socket(result);
196 55 : }
197 :
198 :
199 : /*
200 : recv name resolution reply then send the next connect
201 : */
202 41897 : static void continue_resolve_name(struct composite_context *creq)
203 : {
204 41897 : struct composite_context *result = talloc_get_type(creq->async.private_data,
205 : struct composite_context);
206 41897 : struct connect_multi_state *multi = talloc_get_type(result->private_data,
207 : struct connect_multi_state);
208 964 : struct socket_address **addr;
209 964 : unsigned i;
210 :
211 41897 : result->status = resolve_name_all_recv(creq, multi, &addr, NULL);
212 41897 : if (!composite_is_ok(result)) return;
213 :
214 119562 : for(i=0; addr[i]; i++);
215 41896 : multi->num_address = i;
216 41896 : multi->server_address = talloc_steal(multi, addr);
217 :
218 41896 : connect_multi_next_socket(result);
219 : }
220 :
221 : /*
222 : one of our socket_connect_send() calls hash finished. If it got a
223 : connection or there are none left then we are done
224 : */
225 41896 : static void continue_one(struct composite_context *creq)
226 : {
227 41896 : struct connect_one_state *state = talloc_get_type(creq->async.private_data,
228 : struct connect_one_state);
229 41896 : struct composite_context *result = state->result;
230 41896 : struct connect_multi_state *multi = talloc_get_type(result->private_data,
231 : struct connect_multi_state);
232 964 : NTSTATUS status;
233 :
234 41896 : status = socket_connect_recv(creq);
235 :
236 41896 : if (multi->ex) {
237 842 : struct tevent_req *subreq;
238 :
239 15540 : subreq = multi->ex->establish_send(state,
240 : result->event_ctx,
241 : state->sock,
242 : state->addr,
243 14698 : multi->ex->private_data);
244 56472 : if (composite_nomem(subreq, result)) return;
245 15540 : tevent_req_set_callback(subreq, continue_one_ex, state);
246 15540 : return;
247 : }
248 :
249 26356 : multi->num_connects_recv++;
250 :
251 26356 : if (NT_STATUS_IS_OK(status)) {
252 26356 : multi->sock = talloc_steal(multi, state->sock);
253 26356 : multi->result_port = state->addr->port;
254 : }
255 :
256 26356 : talloc_free(state);
257 :
258 26356 : if (NT_STATUS_IS_OK(status) ||
259 0 : multi->num_connects_recv == (multi->num_address * multi->num_ports)) {
260 26356 : result->status = status;
261 26356 : composite_done(result);
262 26356 : return;
263 : }
264 :
265 : /* try the next port */
266 0 : connect_multi_next_socket(result);
267 : }
268 :
269 : /*
270 : one of our multi->ex->establish_send() calls hash finished. If it got a
271 : connection or there are none left then we are done
272 : */
273 15540 : static void continue_one_ex(struct tevent_req *subreq)
274 : {
275 842 : struct connect_one_state *state =
276 15540 : tevent_req_callback_data(subreq,
277 : struct connect_one_state);
278 15540 : struct composite_context *result = state->result;
279 842 : struct connect_multi_state *multi =
280 15540 : talloc_get_type_abort(result->private_data,
281 : struct connect_multi_state);
282 842 : NTSTATUS status;
283 15540 : multi->num_connects_recv++;
284 :
285 15540 : status = multi->ex->establish_recv(subreq);
286 15540 : TALLOC_FREE(subreq);
287 :
288 15540 : if (NT_STATUS_IS_OK(status)) {
289 15540 : multi->sock = talloc_steal(multi, state->sock);
290 15540 : multi->result_port = state->addr->port;
291 : }
292 :
293 15540 : talloc_free(state);
294 :
295 15540 : if (NT_STATUS_IS_OK(status) ||
296 0 : multi->num_connects_recv == (multi->num_address * multi->num_ports)) {
297 15540 : result->status = status;
298 15540 : composite_done(result);
299 15540 : return;
300 : }
301 :
302 : /* try the next port */
303 0 : connect_multi_next_socket(result);
304 : }
305 :
306 : /*
307 : async recv routine for socket_connect_multi()
308 : */
309 41897 : _PUBLIC_ NTSTATUS socket_connect_multi_ex_recv(struct composite_context *ctx,
310 : TALLOC_CTX *mem_ctx,
311 : struct socket_context **sock,
312 : uint16_t *port)
313 : {
314 41897 : NTSTATUS status = composite_wait(ctx);
315 41897 : if (NT_STATUS_IS_OK(status)) {
316 964 : struct connect_multi_state *multi =
317 41896 : talloc_get_type(ctx->private_data,
318 : struct connect_multi_state);
319 41896 : *sock = talloc_steal(mem_ctx, multi->sock);
320 41896 : *port = multi->result_port;
321 : }
322 41897 : talloc_free(ctx);
323 41897 : return status;
324 : }
325 :
326 124 : NTSTATUS socket_connect_multi_ex(TALLOC_CTX *mem_ctx,
327 : const char *server_address,
328 : int num_server_ports, uint16_t *server_ports,
329 : struct resolve_context *resolve_ctx,
330 : struct tevent_context *event_ctx,
331 : struct socket_connect_multi_ex *ex,
332 : struct socket_context **result,
333 : uint16_t *result_port)
334 : {
335 0 : struct composite_context *ctx =
336 124 : socket_connect_multi_ex_send(mem_ctx, server_address,
337 : num_server_ports, server_ports,
338 : resolve_ctx,
339 : event_ctx,
340 : ex);
341 124 : return socket_connect_multi_ex_recv(ctx, mem_ctx, result, result_port);
342 : }
343 :
344 : /*
345 : setup an async socket_connect, with multiple ports
346 : */
347 26232 : _PUBLIC_ struct composite_context *socket_connect_multi_send(
348 : TALLOC_CTX *mem_ctx,
349 : const char *server_name,
350 : int num_server_ports,
351 : uint16_t *server_ports,
352 : struct resolve_context *resolve_ctx,
353 : struct tevent_context *event_ctx)
354 : {
355 26232 : return socket_connect_multi_ex_send(mem_ctx,
356 : server_name,
357 : num_server_ports,
358 : server_ports,
359 : resolve_ctx,
360 : event_ctx,
361 : NULL); /* ex */
362 : }
363 :
364 : /*
365 : async recv routine for socket_connect_multi()
366 : */
367 26232 : _PUBLIC_ NTSTATUS socket_connect_multi_recv(struct composite_context *ctx,
368 : TALLOC_CTX *mem_ctx,
369 : struct socket_context **sock,
370 : uint16_t *port)
371 : {
372 26232 : return socket_connect_multi_ex_recv(ctx, mem_ctx, sock, port);
373 : }
374 :
375 124 : NTSTATUS socket_connect_multi(TALLOC_CTX *mem_ctx,
376 : const char *server_address,
377 : int num_server_ports, uint16_t *server_ports,
378 : struct resolve_context *resolve_ctx,
379 : struct tevent_context *event_ctx,
380 : struct socket_context **result,
381 : uint16_t *result_port)
382 : {
383 124 : return socket_connect_multi_ex(mem_ctx,
384 : server_address,
385 : num_server_ports,
386 : server_ports,
387 : resolve_ctx,
388 : event_ctx,
389 : NULL, /* ex */
390 : result,
391 : result_port);
392 : }
|