Line data Source code
1 : /*
2 : Unix SMB/CIFS implementation.
3 : handle unexpected packets
4 : Copyright (C) Andrew Tridgell 2000
5 :
6 : This program is free software; you can redistribute it and/or modify
7 : it under the terms of the GNU General Public License as published by
8 : the Free Software Foundation; either version 3 of the License, or
9 : (at your option) any later version.
10 :
11 : This program is distributed in the hope that it will be useful,
12 : but WITHOUT ANY WARRANTY; without even the implied warranty of
13 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 : GNU General Public License for more details.
15 :
16 : You should have received a copy of the GNU General Public License
17 : along with this program. If not, see <http://www.gnu.org/licenses/>.
18 :
19 : */
20 :
21 : #include "includes.h"
22 : #include "libsmb/unexpected.h"
23 : #include "../lib/util/tevent_ntstatus.h"
24 : #include "lib/util_tsock.h"
25 : #include "libsmb/nmblib.h"
26 : #include "lib/tsocket/tsocket.h"
27 : #include "lib/util/sys_rw.h"
28 :
29 903 : static const char *nmbd_socket_dir(void)
30 : {
31 903 : return lp_parm_const_string(-1, "nmbd", "socket dir",
32 : get_dyn_NMBDSOCKETDIR());
33 : }
34 :
35 : struct nb_packet_query {
36 : enum packet_type type;
37 : size_t mailslot_namelen;
38 : int trn_id;
39 : };
40 :
41 : struct nb_packet_client;
42 :
43 : struct nb_packet_server {
44 : struct tevent_context *ev;
45 : int listen_sock;
46 : struct tevent_fd *listen_fde;
47 : int max_clients;
48 : int num_clients;
49 : struct nb_packet_client *clients;
50 : };
51 :
52 : struct nb_packet_client {
53 : struct nb_packet_client *prev, *next;
54 : struct nb_packet_server *server;
55 :
56 : enum packet_type type;
57 : int trn_id;
58 : char *mailslot_name;
59 :
60 : struct {
61 : uint8_t byte;
62 : struct iovec iov[1];
63 : } ack;
64 :
65 : struct tstream_context *sock;
66 : struct tevent_queue *out_queue;
67 : };
68 :
69 : static int nb_packet_server_destructor(struct nb_packet_server *s);
70 : static void nb_packet_server_listener(struct tevent_context *ev,
71 : struct tevent_fd *fde,
72 : uint16_t flags,
73 : void *private_data);
74 :
75 43 : NTSTATUS nb_packet_server_create(TALLOC_CTX *mem_ctx,
76 : struct tevent_context *ev,
77 : int max_clients,
78 : struct nb_packet_server **presult)
79 : {
80 0 : struct nb_packet_server *result;
81 0 : NTSTATUS status;
82 0 : int rc;
83 :
84 43 : result = talloc_zero(mem_ctx, struct nb_packet_server);
85 43 : if (result == NULL) {
86 0 : status = NT_STATUS_NO_MEMORY;
87 0 : goto fail;
88 : }
89 43 : result->ev = ev;
90 43 : result->max_clients = max_clients;
91 :
92 43 : result->listen_sock = create_pipe_sock(
93 : nmbd_socket_dir(), "unexpected", 0755);
94 43 : if (result->listen_sock == -1) {
95 0 : status = map_nt_error_from_unix(errno);
96 0 : goto fail;
97 : }
98 43 : rc = listen(result->listen_sock, 5);
99 43 : if (rc < 0) {
100 0 : status = map_nt_error_from_unix(errno);
101 0 : goto fail;
102 : }
103 43 : talloc_set_destructor(result, nb_packet_server_destructor);
104 :
105 43 : result->listen_fde = tevent_add_fd(ev, result,
106 : result->listen_sock,
107 : TEVENT_FD_READ,
108 : nb_packet_server_listener,
109 : result);
110 43 : if (result->listen_fde == NULL) {
111 0 : status = NT_STATUS_NO_MEMORY;
112 0 : goto fail;
113 : }
114 :
115 43 : *presult = result;
116 43 : return NT_STATUS_OK;
117 0 : fail:
118 0 : TALLOC_FREE(result);
119 0 : return status;
120 : }
121 :
122 0 : static int nb_packet_server_destructor(struct nb_packet_server *s)
123 : {
124 0 : TALLOC_FREE(s->listen_fde);
125 :
126 0 : if (s->listen_sock != -1) {
127 0 : close(s->listen_sock);
128 0 : s->listen_sock = -1;
129 : }
130 0 : return 0;
131 : }
132 :
133 : static int nb_packet_client_destructor(struct nb_packet_client *c);
134 : static ssize_t nb_packet_client_more(uint8_t *buf, size_t buflen,
135 : void *private_data);
136 : static void nb_packet_got_query(struct tevent_req *req);
137 : static void nb_packet_client_ack_done(struct tevent_req *req);
138 : static void nb_packet_client_read_done(struct tevent_req *req);
139 :
140 657 : static void nb_packet_server_listener(struct tevent_context *ev,
141 : struct tevent_fd *fde,
142 : uint16_t flags,
143 : void *private_data)
144 : {
145 657 : struct nb_packet_server *server = talloc_get_type_abort(
146 : private_data, struct nb_packet_server);
147 0 : struct nb_packet_client *client;
148 0 : struct tevent_req *req;
149 0 : struct sockaddr_un sunaddr;
150 0 : socklen_t len;
151 0 : int sock;
152 0 : int ret;
153 :
154 657 : len = sizeof(sunaddr);
155 :
156 657 : sock = accept(server->listen_sock, (struct sockaddr *)(void *)&sunaddr,
157 : &len);
158 657 : if (sock == -1) {
159 0 : return;
160 : }
161 657 : smb_set_close_on_exec(sock);
162 657 : DEBUG(6,("accepted socket %d\n", sock));
163 :
164 657 : client = talloc_zero(server, struct nb_packet_client);
165 657 : if (client == NULL) {
166 0 : DEBUG(10, ("talloc failed\n"));
167 0 : close(sock);
168 0 : return;
169 : }
170 657 : ret = tstream_bsd_existing_socket(client, sock, &client->sock);
171 657 : if (ret != 0) {
172 0 : DEBUG(10, ("tstream_bsd_existing_socket failed\n"));
173 0 : TALLOC_FREE(client);
174 0 : close(sock);
175 0 : return;
176 : }
177 : /* as server we want to fail early */
178 657 : tstream_bsd_fail_readv_first_error(client->sock, true);
179 :
180 657 : client->server = server;
181 :
182 657 : client->out_queue = tevent_queue_create(
183 : client, "unexpected packet output");
184 657 : if (client->out_queue == NULL) {
185 0 : DEBUG(10, ("tevent_queue_create failed\n"));
186 0 : TALLOC_FREE(client);
187 0 : return;
188 : }
189 :
190 657 : req = tstream_read_packet_send(client, ev, client->sock,
191 : sizeof(struct nb_packet_query),
192 : nb_packet_client_more, NULL);
193 657 : if (req == NULL) {
194 0 : DEBUG(10, ("tstream_read_packet_send failed\n"));
195 0 : TALLOC_FREE(client);
196 0 : return;
197 : }
198 657 : tevent_req_set_callback(req, nb_packet_got_query, client);
199 :
200 657 : DLIST_ADD(server->clients, client);
201 657 : server->num_clients += 1;
202 :
203 657 : talloc_set_destructor(client, nb_packet_client_destructor);
204 :
205 657 : if (server->num_clients > server->max_clients) {
206 0 : DEBUG(10, ("Too many clients, dropping oldest\n"));
207 :
208 : /*
209 : * no TALLOC_FREE here, don't mess with the list structs
210 : */
211 0 : talloc_free(server->clients->prev);
212 : }
213 : }
214 :
215 675 : static ssize_t nb_packet_client_more(uint8_t *buf, size_t buflen,
216 : void *private_data)
217 : {
218 0 : struct nb_packet_query q;
219 675 : if (buflen > sizeof(struct nb_packet_query)) {
220 18 : return 0;
221 : }
222 : /* Take care of alignment */
223 657 : memcpy(&q, buf, sizeof(q));
224 657 : if (q.mailslot_namelen > 1024) {
225 0 : DEBUG(10, ("Got invalid mailslot namelen %d\n",
226 : (int)q.mailslot_namelen));
227 0 : return -1;
228 : }
229 657 : return q.mailslot_namelen;
230 : }
231 :
232 657 : static int nb_packet_client_destructor(struct nb_packet_client *c)
233 : {
234 657 : tevent_queue_stop(c->out_queue);
235 657 : TALLOC_FREE(c->sock);
236 :
237 657 : DLIST_REMOVE(c->server->clients, c);
238 657 : c->server->num_clients -= 1;
239 657 : return 0;
240 : }
241 :
242 657 : static void nb_packet_got_query(struct tevent_req *req)
243 : {
244 657 : struct nb_packet_client *client = tevent_req_callback_data(
245 : req, struct nb_packet_client);
246 0 : struct nb_packet_query q;
247 0 : uint8_t *buf;
248 0 : ssize_t nread;
249 0 : int err;
250 :
251 657 : nread = tstream_read_packet_recv(req, talloc_tos(), &buf, &err);
252 657 : TALLOC_FREE(req);
253 657 : if (nread < (ssize_t)sizeof(struct nb_packet_query)) {
254 0 : DEBUG(10, ("read_packet_recv returned %d (%s)\n",
255 : (int)nread,
256 : (nread == -1) ? strerror(err) : "wrong length"));
257 0 : TALLOC_FREE(client);
258 0 : return;
259 : }
260 :
261 : /* Take care of alignment */
262 657 : memcpy(&q, buf, sizeof(q));
263 :
264 657 : if ((size_t)nread !=
265 657 : sizeof(struct nb_packet_query) + q.mailslot_namelen) {
266 0 : DEBUG(10, ("nb_packet_got_query: Invalid mailslot namelength\n"));
267 0 : TALLOC_FREE(client);
268 0 : return;
269 : }
270 :
271 657 : client->trn_id = q.trn_id;
272 657 : client->type = q.type;
273 657 : if (q.mailslot_namelen > 0) {
274 36 : client->mailslot_name = talloc_strndup(
275 18 : client, (char *)buf + sizeof(q),
276 : q.mailslot_namelen);
277 18 : if (client->mailslot_name == NULL) {
278 0 : TALLOC_FREE(client);
279 0 : return;
280 : }
281 : }
282 :
283 657 : client->ack.byte = 0;
284 657 : client->ack.iov[0].iov_base = &client->ack.byte;
285 657 : client->ack.iov[0].iov_len = 1;
286 657 : req = tstream_writev_queue_send(client, client->server->ev,
287 : client->sock,
288 : client->out_queue,
289 657 : client->ack.iov, 1);
290 657 : if (req == NULL) {
291 0 : DEBUG(10, ("tstream_writev_queue_send failed\n"));
292 0 : TALLOC_FREE(client);
293 0 : return;
294 : }
295 657 : tevent_req_set_callback(req, nb_packet_client_ack_done, client);
296 :
297 657 : req = tstream_read_packet_send(client, client->server->ev,
298 : client->sock, 1, NULL, NULL);
299 657 : if (req == NULL) {
300 0 : DEBUG(10, ("Could not activate reader for client exit "
301 : "detection\n"));
302 0 : TALLOC_FREE(client);
303 0 : return;
304 : }
305 657 : tevent_req_set_callback(req, nb_packet_client_read_done,
306 : client);
307 : }
308 :
309 657 : static void nb_packet_client_ack_done(struct tevent_req *req)
310 : {
311 657 : struct nb_packet_client *client = tevent_req_callback_data(
312 : req, struct nb_packet_client);
313 0 : ssize_t nwritten;
314 0 : int err;
315 :
316 657 : nwritten = tstream_writev_queue_recv(req, &err);
317 :
318 657 : TALLOC_FREE(req);
319 :
320 657 : if (nwritten == -1) {
321 0 : DEBUG(10, ("tstream_writev_queue_recv failed: %s\n",
322 : strerror(err)));
323 0 : TALLOC_FREE(client);
324 0 : return;
325 : }
326 : }
327 :
328 657 : static void nb_packet_client_read_done(struct tevent_req *req)
329 : {
330 657 : struct nb_packet_client *client = tevent_req_callback_data(
331 : req, struct nb_packet_client);
332 0 : ssize_t nread;
333 0 : uint8_t *buf;
334 0 : int err;
335 :
336 657 : nread = tstream_read_packet_recv(req, talloc_tos(), &buf, &err);
337 657 : TALLOC_FREE(req);
338 657 : if (nread == 1) {
339 0 : DEBUG(10, ("Protocol error, received data on write-only "
340 : "unexpected socket: 0x%2.2x\n", (*buf)));
341 : }
342 657 : TALLOC_FREE(client);
343 657 : }
344 :
345 : static void nb_packet_client_send(struct nb_packet_client *client,
346 : struct packet_struct *p);
347 :
348 1059 : void nb_packet_dispatch(struct nb_packet_server *server,
349 : struct packet_struct *p)
350 : {
351 0 : struct nb_packet_client *c;
352 0 : uint16_t trn_id;
353 :
354 1059 : switch (p->packet_type) {
355 12 : case NMB_PACKET:
356 12 : trn_id = p->packet.nmb.header.name_trn_id;
357 12 : break;
358 1047 : case DGRAM_PACKET:
359 1047 : trn_id = p->packet.dgram.header.dgm_id;
360 1047 : break;
361 0 : default:
362 0 : DEBUG(10, ("Got invalid packet type %d\n",
363 : (int)p->packet_type));
364 0 : return;
365 : }
366 1109 : for (c = server->clients; c != NULL; c = c->next) {
367 :
368 50 : if (c->type != p->packet_type) {
369 34 : DEBUG(10, ("client expects packet %d, got %d\n",
370 : c->type, p->packet_type));
371 34 : continue;
372 : }
373 :
374 16 : if (p->packet_type == NMB_PACKET) {
375 : /*
376 : * See if the client specified transaction
377 : * ID. Filter if it did.
378 : */
379 0 : if ((c->trn_id != -1) &&
380 0 : (c->trn_id != trn_id)) {
381 0 : DEBUG(10, ("client expects trn %d, got %d\n",
382 : c->trn_id, trn_id));
383 0 : continue;
384 : }
385 : } else {
386 : /*
387 : * See if the client specified a mailslot
388 : * name. Filter if it did.
389 : */
390 16 : if ((c->mailslot_name != NULL) &&
391 16 : !match_mailslot_name(p, c->mailslot_name)) {
392 0 : continue;
393 : }
394 : }
395 16 : nb_packet_client_send(c, p);
396 : }
397 : }
398 :
399 : struct nb_packet_client_header {
400 : size_t len;
401 : enum packet_type type;
402 : time_t timestamp;
403 : struct in_addr ip;
404 : int port;
405 : };
406 :
407 : struct nb_packet_client_state {
408 : struct nb_packet_client *client;
409 : struct iovec iov[2];
410 : struct nb_packet_client_header hdr;
411 : char buf[1024];
412 : };
413 :
414 : static void nb_packet_client_send_done(struct tevent_req *req);
415 :
416 16 : static void nb_packet_client_send(struct nb_packet_client *client,
417 : struct packet_struct *p)
418 : {
419 0 : struct nb_packet_client_state *state;
420 0 : struct tevent_req *req;
421 :
422 16 : if (tevent_queue_length(client->out_queue) > 10) {
423 : /*
424 : * Skip clients that don't listen anyway, some form of DoS
425 : * protection
426 : */
427 0 : return;
428 : }
429 :
430 16 : state = talloc_zero(client, struct nb_packet_client_state);
431 16 : if (state == NULL) {
432 0 : DEBUG(10, ("talloc failed\n"));
433 0 : return;
434 : }
435 :
436 16 : state->client = client;
437 :
438 16 : state->hdr.ip = p->ip;
439 16 : state->hdr.port = p->port;
440 16 : state->hdr.timestamp = p->timestamp;
441 16 : state->hdr.type = p->packet_type;
442 16 : state->hdr.len = build_packet(state->buf, sizeof(state->buf), p);
443 :
444 16 : state->iov[0].iov_base = (char *)&state->hdr;
445 16 : state->iov[0].iov_len = sizeof(state->hdr);
446 16 : state->iov[1].iov_base = state->buf;
447 16 : state->iov[1].iov_len = state->hdr.len;
448 :
449 16 : req = tstream_writev_queue_send(state, client->server->ev,
450 : client->sock,
451 : client->out_queue,
452 16 : state->iov, 2);
453 16 : if (req == NULL) {
454 0 : DEBUG(10, ("tstream_writev_queue_send failed\n"));
455 0 : return;
456 : }
457 16 : tevent_req_set_callback(req, nb_packet_client_send_done, state);
458 : }
459 :
460 16 : static void nb_packet_client_send_done(struct tevent_req *req)
461 : {
462 16 : struct nb_packet_client_state *state = tevent_req_callback_data(
463 : req, struct nb_packet_client_state);
464 16 : struct nb_packet_client *client = state->client;
465 0 : ssize_t nwritten;
466 0 : int err;
467 :
468 16 : nwritten = tstream_writev_queue_recv(req, &err);
469 :
470 16 : TALLOC_FREE(req);
471 16 : TALLOC_FREE(state);
472 :
473 16 : if (nwritten == -1) {
474 0 : DEBUG(10, ("tstream_writev_queue failed: %s\n", strerror(err)));
475 0 : TALLOC_FREE(client);
476 0 : return;
477 : }
478 : }
479 :
480 : struct nb_packet_reader {
481 : struct tstream_context *sock;
482 : };
483 :
484 : struct nb_packet_reader_state {
485 : struct tevent_context *ev;
486 : struct nb_packet_query query;
487 : const char *mailslot_name;
488 : struct iovec iov[2];
489 : struct nb_packet_reader *reader;
490 : };
491 :
492 : static void nb_packet_reader_connected(struct tevent_req *subreq);
493 : static void nb_packet_reader_sent_query(struct tevent_req *subreq);
494 : static void nb_packet_reader_got_ack(struct tevent_req *subreq);
495 :
496 860 : struct tevent_req *nb_packet_reader_send(TALLOC_CTX *mem_ctx,
497 : struct tevent_context *ev,
498 : enum packet_type type,
499 : int trn_id,
500 : const char *mailslot_name)
501 : {
502 0 : struct tevent_req *req, *subreq;
503 0 : struct nb_packet_reader_state *state;
504 0 : struct tsocket_address *laddr;
505 0 : char *rpath;
506 0 : struct tsocket_address *raddr;
507 0 : int ret;
508 :
509 860 : req = tevent_req_create(mem_ctx, &state,
510 : struct nb_packet_reader_state);
511 860 : if (req == NULL) {
512 0 : return NULL;
513 : }
514 860 : state->ev = ev;
515 860 : state->query.trn_id = trn_id;
516 860 : state->query.type = type;
517 860 : state->mailslot_name = mailslot_name;
518 :
519 860 : if (mailslot_name != NULL) {
520 5 : state->query.mailslot_namelen = strlen(mailslot_name);
521 : }
522 :
523 860 : state->reader = talloc_zero(state, struct nb_packet_reader);
524 860 : if (tevent_req_nomem(state->reader, req)) {
525 0 : return tevent_req_post(req, ev);
526 : }
527 :
528 860 : ret = tsocket_address_unix_from_path(state, NULL, &laddr);
529 860 : if (ret != 0) {
530 0 : tevent_req_nterror(req, map_nt_error_from_unix(errno));
531 0 : return tevent_req_post(req, ev);
532 : }
533 860 : rpath = talloc_asprintf(state, "%s/%s", nmbd_socket_dir(),
534 : "unexpected");
535 860 : if (tevent_req_nomem(rpath, req)) {
536 0 : return tevent_req_post(req, ev);
537 : }
538 860 : ret = tsocket_address_unix_from_path(state, rpath, &raddr);
539 860 : if (ret != 0) {
540 0 : tevent_req_nterror(req, map_nt_error_from_unix(errno));
541 0 : return tevent_req_post(req, ev);
542 : }
543 :
544 860 : subreq = tstream_unix_connect_send(state, ev, laddr, raddr);
545 860 : if (tevent_req_nomem(subreq, req)) {
546 0 : return tevent_req_post(req, ev);
547 : }
548 860 : tevent_req_set_callback(subreq, nb_packet_reader_connected, req);
549 860 : return req;
550 : }
551 :
552 860 : static void nb_packet_reader_connected(struct tevent_req *subreq)
553 : {
554 860 : struct tevent_req *req = tevent_req_callback_data(
555 : subreq, struct tevent_req);
556 860 : struct nb_packet_reader_state *state = tevent_req_data(
557 : req, struct nb_packet_reader_state);
558 0 : int res, err;
559 860 : int num_iovecs = 1;
560 :
561 860 : res = tstream_unix_connect_recv(subreq, &err, state->reader,
562 : &state->reader->sock);
563 860 : TALLOC_FREE(subreq);
564 860 : if (res == -1) {
565 525 : DEBUG(10, ("tstream_unix_connect failed: %s\n", strerror(err)));
566 525 : tevent_req_nterror(req, map_nt_error_from_unix(err));
567 525 : return;
568 : }
569 :
570 335 : state->iov[0].iov_base = (char *)&state->query;
571 335 : state->iov[0].iov_len = sizeof(state->query);
572 :
573 335 : if (state->mailslot_name != NULL) {
574 5 : num_iovecs = 2;
575 5 : state->iov[1].iov_base = discard_const_p(
576 : char, state->mailslot_name);
577 5 : state->iov[1].iov_len = state->query.mailslot_namelen;
578 : }
579 :
580 335 : subreq = tstream_writev_send(state, state->ev, state->reader->sock,
581 335 : state->iov, num_iovecs);
582 335 : if (tevent_req_nomem(subreq, req)) {
583 0 : return;
584 : }
585 335 : tevent_req_set_callback(subreq, nb_packet_reader_sent_query, req);
586 : }
587 :
588 335 : static void nb_packet_reader_sent_query(struct tevent_req *subreq)
589 : {
590 335 : struct tevent_req *req = tevent_req_callback_data(
591 : subreq, struct tevent_req);
592 335 : struct nb_packet_reader_state *state = tevent_req_data(
593 : req, struct nb_packet_reader_state);
594 0 : ssize_t written;
595 0 : int err;
596 :
597 335 : written = tstream_writev_recv(subreq, &err);
598 335 : TALLOC_FREE(subreq);
599 335 : if (written == -1) {
600 0 : tevent_req_nterror(req, map_nt_error_from_unix(err));
601 0 : return;
602 : }
603 335 : if ((size_t)written !=
604 335 : sizeof(state->query) + state->query.mailslot_namelen) {
605 0 : tevent_req_nterror(req, NT_STATUS_UNEXPECTED_IO_ERROR);
606 0 : return;
607 : }
608 335 : subreq = tstream_read_packet_send(state, state->ev,
609 335 : state->reader->sock,
610 : 1, NULL, NULL);
611 335 : if (tevent_req_nomem(subreq, req)) {
612 0 : return;
613 : }
614 335 : tevent_req_set_callback(subreq, nb_packet_reader_got_ack, req);
615 : }
616 :
617 335 : static void nb_packet_reader_got_ack(struct tevent_req *subreq)
618 : {
619 335 : struct tevent_req *req = tevent_req_callback_data(
620 : subreq, struct tevent_req);
621 335 : struct nb_packet_reader_state *state = tevent_req_data(
622 : req, struct nb_packet_reader_state);
623 0 : ssize_t nread;
624 0 : int err;
625 0 : uint8_t *buf;
626 :
627 335 : nread = tstream_read_packet_recv(subreq, state, &buf, &err);
628 335 : TALLOC_FREE(subreq);
629 335 : if (nread == -1) {
630 0 : DEBUG(10, ("read_packet_recv returned %s\n",
631 : strerror(err)));
632 0 : tevent_req_nterror(req, map_nt_error_from_unix(err));
633 0 : return;
634 : }
635 335 : if (nread != 1) {
636 0 : DBG_DEBUG("read = %zd, expected 1\n", nread);
637 0 : tevent_req_nterror(req, NT_STATUS_UNEXPECTED_IO_ERROR);
638 0 : return;
639 : }
640 335 : tevent_req_done(req);
641 : }
642 :
643 860 : NTSTATUS nb_packet_reader_recv(struct tevent_req *req, TALLOC_CTX *mem_ctx,
644 : struct nb_packet_reader **preader)
645 : {
646 860 : struct nb_packet_reader_state *state = tevent_req_data(
647 : req, struct nb_packet_reader_state);
648 0 : NTSTATUS status;
649 :
650 860 : if (tevent_req_is_nterror(req, &status)) {
651 525 : tevent_req_received(req);
652 525 : return status;
653 : }
654 335 : *preader = talloc_move(mem_ctx, &state->reader);
655 335 : tevent_req_received(req);
656 335 : return NT_STATUS_OK;
657 : }
658 :
659 : struct nb_packet_read_state {
660 : struct nb_packet_client_header hdr;
661 : uint8_t *buf;
662 : size_t buflen;
663 : };
664 :
665 : static ssize_t nb_packet_read_more(uint8_t *buf, size_t buflen, void *p);
666 : static void nb_packet_read_done(struct tevent_req *subreq);
667 :
668 333 : struct tevent_req *nb_packet_read_send(TALLOC_CTX *mem_ctx,
669 : struct tevent_context *ev,
670 : struct nb_packet_reader *reader)
671 : {
672 0 : struct tevent_req *req, *subreq;
673 0 : struct nb_packet_read_state *state;
674 :
675 333 : req = tevent_req_create(mem_ctx, &state, struct nb_packet_read_state);
676 333 : if (req == NULL) {
677 0 : return NULL;
678 : }
679 333 : subreq = tstream_read_packet_send(state, ev, reader->sock,
680 : sizeof(struct nb_packet_client_header),
681 : nb_packet_read_more, state);
682 333 : if (tevent_req_nomem(subreq, req)) {
683 0 : return tevent_req_post(req, ev);
684 : }
685 333 : tevent_req_set_callback(subreq, nb_packet_read_done, req);
686 333 : return req;
687 : }
688 :
689 6 : static ssize_t nb_packet_read_more(uint8_t *buf, size_t buflen, void *p)
690 : {
691 6 : struct nb_packet_read_state *state = talloc_get_type_abort(
692 : p, struct nb_packet_read_state);
693 :
694 6 : if (buflen > sizeof(struct nb_packet_client_header)) {
695 : /*
696 : * Been here, done
697 : */
698 3 : return 0;
699 : }
700 3 : memcpy(&state->hdr, buf, sizeof(struct nb_packet_client_header));
701 3 : return state->hdr.len;
702 : }
703 :
704 3 : static void nb_packet_read_done(struct tevent_req *subreq)
705 : {
706 3 : struct tevent_req *req = tevent_req_callback_data(
707 : subreq, struct tevent_req);
708 3 : struct nb_packet_read_state *state = tevent_req_data(
709 : req, struct nb_packet_read_state);
710 0 : ssize_t nread;
711 0 : int err;
712 :
713 3 : nread = tstream_read_packet_recv(subreq, state, &state->buf, &err);
714 3 : if (nread == -1) {
715 0 : tevent_req_nterror(req, map_nt_error_from_unix(err));
716 0 : return;
717 : }
718 3 : state->buflen = nread;
719 3 : tevent_req_done(req);
720 : }
721 :
722 3 : NTSTATUS nb_packet_read_recv(struct tevent_req *req, TALLOC_CTX *mem_ctx,
723 : struct packet_struct **ppacket)
724 : {
725 3 : struct nb_packet_read_state *state = tevent_req_data(
726 : req, struct nb_packet_read_state);
727 0 : struct nb_packet_client_header hdr;
728 0 : struct packet_struct *packet;
729 0 : NTSTATUS status;
730 :
731 3 : if (tevent_req_is_nterror(req, &status)) {
732 0 : tevent_req_received(req);
733 0 : return status;
734 : }
735 :
736 3 : memcpy(&hdr, state->buf, sizeof(hdr));
737 :
738 3 : packet = parse_packet_talloc(
739 : mem_ctx,
740 3 : (char *)state->buf + sizeof(struct nb_packet_client_header),
741 3 : state->buflen - sizeof(struct nb_packet_client_header),
742 : state->hdr.type, state->hdr.ip, state->hdr.port);
743 3 : if (packet == NULL) {
744 0 : tevent_req_received(req);
745 0 : return NT_STATUS_INVALID_NETWORK_RESPONSE;
746 : }
747 :
748 3 : *ppacket = packet;
749 3 : tevent_req_received(req);
750 3 : return NT_STATUS_OK;
751 : }
|