LCOV - code coverage report
Current view: top level - source3/libsmb - unexpected.c (source / functions) Hit Total Coverage
Test: coverage report for vadcx-master-patch-75612 fe003de8 Lines: 227 369 61.5 %
Date: 2024-02-29 22:57:05 Functions: 20 21 95.2 %

          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             : }

Generated by: LCOV version 1.14