LCOV - code coverage report
Current view: top level - libcli/nbt - nbtsocket.c (source / functions) Hit Total Coverage
Test: coverage report for master 98b443d9 Lines: 220 288 76.4 %
Date: 2024-05-31 13:13:24 Functions: 13 14 92.9 %

          Line data    Source code
       1             : /*
       2             :    Unix SMB/CIFS implementation.
       3             : 
       4             :    low level socket handling for nbt requests
       5             : 
       6             :    Copyright (C) Andrew Tridgell 2005
       7             : 
       8             :    This program is free software; you can redistribute it and/or modify
       9             :    it under the terms of the GNU General Public License as published by
      10             :    the Free Software Foundation; either version 3 of the License, or
      11             :    (at your option) any later version.
      12             : 
      13             :    This program is distributed in the hope that it will be useful,
      14             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      15             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
      16             :    GNU General Public License for more details.
      17             : 
      18             :    You should have received a copy of the GNU General Public License
      19             :    along with this program.  If not, see <http://www.gnu.org/licenses/>.
      20             : */
      21             : 
      22             : #include "includes.h"
      23             : #include "lib/events/events.h"
      24             : #include "../lib/util/dlinklist.h"
      25             : #include "../libcli/nbt/libnbt.h"
      26             : #include "../libcli/nbt/nbt_proto.h"
      27             : #include "lib/socket/socket.h"
      28             : #include "librpc/gen_ndr/ndr_nbt.h"
      29             : #include "param/param.h"
      30             : #include "lib/util/idtree_random.h"
      31             : 
      32             : #define NBT_MAX_REPLIES 1000
      33             : 
      34             : /*
      35             :   destroy a pending request
      36             : */
      37       24443 : static int nbt_name_request_destructor(struct nbt_name_request *req)
      38             : {
      39       24443 :         if (req->state == NBT_REQUEST_SEND) {
      40        9205 :                 DLIST_REMOVE(req->nbtsock->send_queue, req);
      41             :         }
      42       24443 :         if (req->state == NBT_REQUEST_WAIT) {
      43        3864 :                 req->nbtsock->num_pending--;
      44             :         }
      45       24443 :         if (req->name_trn_id != 0 && !req->is_reply) {
      46       11420 :                 idr_remove(req->nbtsock->idr, req->name_trn_id);
      47       11420 :                 req->name_trn_id = 0;
      48             :         }
      49       24443 :         TALLOC_FREE(req->te);
      50       24443 :         if (req->nbtsock->send_queue == NULL) {
      51       23343 :                 TEVENT_FD_NOT_WRITEABLE(req->nbtsock->fde);
      52             :         }
      53       24443 :         if (req->nbtsock->num_pending == 0 &&
      54       19842 :             req->nbtsock->incoming.handler == NULL) {
      55       19751 :                 TEVENT_FD_NOT_READABLE(req->nbtsock->fde);
      56             :         }
      57       24443 :         return 0;
      58             : }
      59             : 
      60             : 
      61             : /*
      62             :   handle send events on a nbt name socket
      63             : */
      64       12904 : static void nbt_name_socket_send(struct nbt_name_socket *nbtsock)
      65             : {
      66          91 :         struct nbt_name_request *req;
      67       12904 :         TALLOC_CTX *tmp_ctx = talloc_new(nbtsock);
      68          91 :         NTSTATUS status;
      69             : 
      70       17698 :         while ((req = nbtsock->send_queue)) {
      71         118 :                 size_t len;
      72             : 
      73       13995 :                 len = req->encoded.length;
      74       14113 :                 status = socket_sendto(nbtsock->sock, &req->encoded, &len,
      75       13995 :                                        req->dest);
      76       13995 :                 if (NT_STATUS_IS_ERR(status)) goto failed;
      77             : 
      78        4794 :                 if (!NT_STATUS_IS_OK(status)) {
      79           0 :                         talloc_free(tmp_ctx);
      80           0 :                         return;
      81             :                 }
      82             : 
      83        4794 :                 DLIST_REMOVE(nbtsock->send_queue, req);
      84        4794 :                 req->state = NBT_REQUEST_WAIT;
      85        4794 :                 if (req->is_reply) {
      86        1606 :                         talloc_free(req);
      87             :                 } else {
      88        3188 :                         TEVENT_FD_READABLE(nbtsock->fde);
      89        3188 :                         nbtsock->num_pending++;
      90             :                 }
      91             :         }
      92             : 
      93        3703 :         TEVENT_FD_NOT_WRITEABLE(nbtsock->fde);
      94        3703 :         talloc_free(tmp_ctx);
      95        3703 :         return;
      96             : 
      97        9201 : failed:
      98        9201 :         DLIST_REMOVE(nbtsock->send_queue, req);
      99        9201 :         nbt_name_request_destructor(req);
     100        9201 :         req->status = status;
     101        9201 :         req->state = NBT_REQUEST_ERROR;
     102        9201 :         talloc_free(tmp_ctx);
     103        9201 :         if (req->async.fn) {
     104        9097 :                 req->async.fn(req);
     105         104 :         } else if (req->is_reply) {
     106          39 :                 talloc_free(req);
     107             :         }
     108        9185 :         return;
     109             : }
     110             : 
     111             : 
     112             : /*
     113             :   handle a request timeout
     114             : */
     115        1794 : static void nbt_name_socket_timeout(struct tevent_context *ev, struct tevent_timer *te,
     116             :                                     struct timeval t, void *private_data)
     117             : {
     118        1794 :         struct nbt_name_request *req = talloc_get_type(private_data,
     119             :                                                        struct nbt_name_request);
     120             : 
     121        1794 :         if (req->num_retries != 0) {
     122         962 :                 req->num_retries--;
     123         962 :                 req->te = tevent_add_timer(req->nbtsock->event_ctx, req,
     124             :                                            timeval_add(&t, req->timeout, 0),
     125             :                                            nbt_name_socket_timeout, req);
     126         962 :                 if (req->state != NBT_REQUEST_SEND) {
     127         925 :                         req->state = NBT_REQUEST_SEND;
     128         925 :                         DLIST_ADD_END(req->nbtsock->send_queue, req);
     129             :                 }
     130         962 :                 TEVENT_FD_WRITEABLE(req->nbtsock->fde);
     131         962 :                 return;
     132             :         }
     133             : 
     134         832 :         nbt_name_request_destructor(req);
     135         832 :         if (req->num_replies == 0) {
     136         832 :                 req->state = NBT_REQUEST_TIMEOUT;
     137         832 :                 req->status = NT_STATUS_IO_TIMEOUT;
     138             :         } else {
     139           0 :                 req->state = NBT_REQUEST_DONE;
     140           0 :                 req->status = NT_STATUS_OK;
     141             :         }
     142         832 :         if (req->async.fn) {
     143         767 :                 req->async.fn(req);
     144          65 :         } else if (req->is_reply) {
     145           0 :                 talloc_free(req);
     146             :         }
     147             : }
     148             : 
     149             : 
     150             : 
     151             : /**
     152             :   handle recv events on a nbt name socket
     153             : */
     154       14209 : static void nbt_name_socket_recv(struct nbt_name_socket *nbtsock)
     155             : {
     156       14209 :         TALLOC_CTX *tmp_ctx = talloc_new(nbtsock);
     157         131 :         NTSTATUS status;
     158         131 :         enum ndr_err_code ndr_err;
     159         131 :         struct socket_address *src;
     160         131 :         DATA_BLOB blob;
     161         131 :         size_t nread, dsize;
     162         131 :         struct nbt_name_packet *packet;
     163         131 :         struct nbt_name_request *req;
     164             : 
     165       14209 :         status = socket_pending(nbtsock->sock, &dsize);
     166       14209 :         if (!NT_STATUS_IS_OK(status)) {
     167           0 :                 talloc_free(tmp_ctx);
     168       12850 :                 return;
     169             :         }
     170             : 
     171             :         /*
     172             :          * Given a zero length, data_blob_talloc() returns the
     173             :          * NULL blob {NULL, 0}.
     174             :          *
     175             :          * We only want to error return here on a real out of memory condition
     176             :          * (i.e. dsize != 0, so the UDP packet has data, but the return of the
     177             :          * allocation failed, so blob.data==NULL).
     178             :          *
     179             :          * Given an actual zero length UDP packet having blob.data == NULL
     180             :          * isn't an out of memory error condition, that's the defined semantics
     181             :          * of data_blob_talloc() when asked for zero bytes.
     182             :          *
     183             :          * We still need to continue to do the zero-length socket_recvfrom()
     184             :          * read in order to clear the "read pending" condition on the socket.
     185             :          */
     186       14209 :         blob = data_blob_talloc(tmp_ctx, NULL, dsize);
     187       14209 :         if (blob.data == NULL && dsize != 0) {
     188           0 :                 talloc_free(tmp_ctx);
     189           0 :                 return;
     190             :         }
     191             : 
     192       14209 :         status = socket_recvfrom(nbtsock->sock, blob.data, blob.length, &nread,
     193             :                                  tmp_ctx, &src);
     194       14209 :         if (!NT_STATUS_IS_OK(status)) {
     195           2 :                 talloc_free(tmp_ctx);
     196           2 :                 return;
     197             :         }
     198             : 
     199       14207 :         packet = talloc(tmp_ctx, struct nbt_name_packet);
     200       14207 :         if (packet == NULL) {
     201           0 :                 talloc_free(tmp_ctx);
     202           0 :                 return;
     203             :         }
     204             : 
     205             :         /* parse the request */
     206       14207 :         ndr_err = ndr_pull_struct_blob(&blob, packet, packet,
     207             :                                        (ndr_pull_flags_fn_t)ndr_pull_nbt_name_packet);
     208       14207 :         if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
     209           0 :                 status = ndr_map_error2ntstatus(ndr_err);
     210           0 :                 DEBUG(2,("Failed to parse incoming NBT name packet - %s\n",
     211             :                          nt_errstr(status)));
     212           0 :                 talloc_free(tmp_ctx);
     213           0 :                 return;
     214             :         }
     215             : 
     216       14207 :         if (DEBUGLVL(10)) {
     217           0 :                 DEBUG(10,("Received nbt packet of length %d from %s:%d\n",
     218             :                           (int)blob.length, src->addr, src->port));
     219           0 :                 NDR_PRINT_DEBUG(nbt_name_packet, packet);
     220             :         }
     221             : 
     222             :         /* if its not a reply then pass it off to the incoming request
     223             :            handler, if any */
     224       14207 :         if (!(packet->operation & NBT_FLAG_REPLY)) {
     225       12842 :                 if (nbtsock->incoming.handler) {
     226       12842 :                         nbtsock->incoming.handler(nbtsock, packet, src);
     227             :                 }
     228       12842 :                 talloc_free(tmp_ctx);
     229       12842 :                 return;
     230             :         }
     231             : 
     232             :         /* find the matching request */
     233        1390 :         req = (struct nbt_name_request *)idr_find(nbtsock->idr,
     234        1365 :                                                   packet->name_trn_id);
     235        1365 :         if (req == NULL) {
     236           6 :                 if (nbtsock->unexpected.handler) {
     237           1 :                         nbtsock->unexpected.handler(nbtsock, packet, src);
     238             :                 } else {
     239           5 :                         DEBUG(10,("Failed to match request for incoming name packet id 0x%04x on %p\n",
     240             :                                  packet->name_trn_id, nbtsock));
     241             :                 }
     242           6 :                 talloc_free(tmp_ctx);
     243           6 :                 return;
     244             :         }
     245             : 
     246        1359 :         talloc_steal(req, packet);
     247        1359 :         talloc_steal(req, src);
     248        1359 :         talloc_free(tmp_ctx);
     249        1359 :         nbt_name_socket_handle_response_packet(req, packet, src);
     250             : }
     251             : 
     252        1359 : void nbt_name_socket_handle_response_packet(struct nbt_name_request *req,
     253             :                                             struct nbt_name_packet *packet,
     254             :                                             struct socket_address *src)
     255             : {
     256             :         /* if this is a WACK response, this we need to go back to waiting,
     257             :            but perhaps increase the timeout */
     258        1359 :         if ((packet->operation & NBT_OPCODE) == NBT_OPCODE_WACK) {
     259           0 :                 uint32_t ttl;
     260          18 :                 if (req->received_wack || packet->ancount < 1) {
     261           0 :                         nbt_name_request_destructor(req);
     262           0 :                         req->status = NT_STATUS_INVALID_NETWORK_RESPONSE;
     263           0 :                         req->state  = NBT_REQUEST_ERROR;
     264           0 :                         goto done;
     265             :                 }
     266          18 :                 talloc_free(req->te);
     267             :                 /* we know we won't need any more retries - the server
     268             :                    has received our request */
     269          18 :                 req->num_retries   = 0;
     270          18 :                 req->received_wack = true;
     271             :                 /*
     272             :                  * there is a timeout in the packet,
     273             :                  * it is 5 + 4 * num_old_addresses
     274             :                  *
     275             :                  * although w2k3 screws it up
     276             :                  * and uses num_old_addresses = 0
     277             :                  *
     278             :                  * so we better fallback to the maximum
     279             :                  * of num_old_addresses = 25 if we got
     280             :                  * a timeout of less than 9s (5 + 4*1)
     281             :                  * or more than 105s (5 + 4*25).
     282             :                  */
     283          18 :                 ttl = packet->answers[0].ttl;
     284          18 :                 if ((ttl < (5 + 4*1)) || (ttl > (5 + 4*25))) {
     285           0 :                         ttl = 5 + 4*25;
     286             :                 }
     287          18 :                 req->timeout = ttl;
     288          18 :                 req->te = tevent_add_timer(req->nbtsock->event_ctx, req,
     289             :                                            timeval_current_ofs(req->timeout, 0),
     290             :                                            nbt_name_socket_timeout, req);
     291          18 :                 return;
     292             :         }
     293             : 
     294             : 
     295        1341 :         req->replies = talloc_realloc(req, req->replies, struct nbt_name_reply, req->num_replies+1);
     296        1341 :         if (req->replies == NULL) {
     297           0 :                 nbt_name_request_destructor(req);
     298           0 :                 req->state  = NBT_REQUEST_ERROR;
     299           0 :                 req->status = NT_STATUS_NO_MEMORY;
     300           0 :                 goto done;
     301             :         }
     302             : 
     303        1341 :         talloc_steal(req, src);
     304        1341 :         req->replies[req->num_replies].dest   = src;
     305        1341 :         talloc_steal(req, packet);
     306        1341 :         req->replies[req->num_replies].packet = packet;
     307        1341 :         req->num_replies++;
     308             : 
     309             :         /* if we don't want multiple replies then we are done */
     310        1341 :         if (req->allow_multiple_replies &&
     311           0 :             req->num_replies < NBT_MAX_REPLIES) {
     312           0 :                 return;
     313             :         }
     314             : 
     315        1341 :         nbt_name_request_destructor(req);
     316        1341 :         req->state  = NBT_REQUEST_DONE;
     317        1341 :         req->status = NT_STATUS_OK;
     318             : 
     319        1341 : done:
     320        1341 :         if (req->async.fn) {
     321         120 :                 req->async.fn(req);
     322             :         }
     323             : }
     324             : 
     325             : /*
     326             :   handle fd events on a nbt_name_socket
     327             : */
     328       26736 : static void nbt_name_socket_handler(struct tevent_context *ev, struct tevent_fd *fde,
     329             :                                     uint16_t flags, void *private_data)
     330             : {
     331       26736 :         struct nbt_name_socket *nbtsock = talloc_get_type(private_data,
     332             :                                                           struct nbt_name_socket);
     333       26736 :         if (flags & TEVENT_FD_WRITE) {
     334       12904 :                 nbt_name_socket_send(nbtsock);
     335             :         }
     336       26736 :         if (flags & TEVENT_FD_READ) {
     337       14209 :                 nbt_name_socket_recv(nbtsock);
     338             :         }
     339       26736 : }
     340             : 
     341             : 
     342             : /*
     343             :   initialise a nbt_name_socket. The event_ctx is optional, if provided
     344             :   then operations will use that event context
     345             : */
     346        9797 : _PUBLIC_ struct nbt_name_socket *nbt_name_socket_init(TALLOC_CTX *mem_ctx,
     347             :                                              struct tevent_context *event_ctx)
     348             : {
     349          34 :         struct nbt_name_socket *nbtsock;
     350          34 :         NTSTATUS status;
     351             : 
     352        9797 :         nbtsock = talloc(mem_ctx, struct nbt_name_socket);
     353        9797 :         if (nbtsock == NULL) goto failed;
     354             : 
     355        9797 :         nbtsock->event_ctx = event_ctx;
     356        9797 :         if (nbtsock->event_ctx == NULL) goto failed;
     357             : 
     358        9797 :         status = socket_create(nbtsock, "ip", SOCKET_TYPE_DGRAM,
     359             :                                &nbtsock->sock, 0);
     360        9797 :         if (!NT_STATUS_IS_OK(status)) goto failed;
     361             : 
     362        9797 :         socket_set_option(nbtsock->sock, "SO_BROADCAST", "1");
     363             : 
     364        9797 :         nbtsock->idr = idr_init(nbtsock);
     365        9797 :         if (nbtsock->idr == NULL) goto failed;
     366             : 
     367        9797 :         nbtsock->send_queue = NULL;
     368        9797 :         nbtsock->num_pending = 0;
     369        9797 :         nbtsock->incoming.handler = NULL;
     370        9797 :         nbtsock->unexpected.handler = NULL;
     371             : 
     372        9797 :         nbtsock->fde = tevent_add_fd(nbtsock->event_ctx, nbtsock,
     373             :                                      socket_get_fd(nbtsock->sock), 0,
     374             :                                      nbt_name_socket_handler, nbtsock);
     375             : 
     376        9797 :         return nbtsock;
     377             : 
     378           0 : failed:
     379           0 :         talloc_free(nbtsock);
     380           0 :         return NULL;
     381             : }
     382             : 
     383             : /*
     384             :   send off a nbt name request
     385             : */
     386       11420 : struct nbt_name_request *nbt_name_request_send(TALLOC_CTX *mem_ctx,
     387             :                                                struct nbt_name_socket *nbtsock,
     388             :                                                struct socket_address *dest,
     389             :                                                struct nbt_name_packet *request,
     390             :                                                int timeout, int retries,
     391             :                                                bool allow_multiple_replies)
     392             : {
     393          65 :         struct nbt_name_request *req;
     394          65 :         int id;
     395          65 :         enum ndr_err_code ndr_err;
     396             : 
     397       11420 :         req = talloc_zero(mem_ctx, struct nbt_name_request);
     398       11420 :         if (req == NULL) goto failed;
     399             : 
     400       11420 :         req->nbtsock                = nbtsock;
     401       11420 :         req->allow_multiple_replies = allow_multiple_replies;
     402       11420 :         req->state                  = NBT_REQUEST_SEND;
     403       11420 :         req->is_reply               = false;
     404       11420 :         req->timeout                = timeout;
     405       11420 :         req->num_retries            = retries;
     406       11420 :         req->dest                   = socket_address_copy(req, dest);
     407       11420 :         if (req->dest == NULL) goto failed;
     408             : 
     409             :         /* we select a random transaction id unless the user supplied one */
     410       11420 :         if (request->name_trn_id == 0) {
     411       11420 :                 id = idr_get_new_random(
     412       11420 :                         req->nbtsock->idr, req, 1, UINT16_MAX);
     413             :         } else {
     414           0 :                 if (idr_find(req->nbtsock->idr, request->name_trn_id)) goto failed;
     415           0 :                 id = idr_get_new_above(req->nbtsock->idr, req, request->name_trn_id,
     416             :                                        UINT16_MAX);
     417             :         }
     418       11420 :         if (id == -1) goto failed;
     419             : 
     420       11420 :         request->name_trn_id = id;
     421       11420 :         req->name_trn_id     = id;
     422             : 
     423       11420 :         req->te = tevent_add_timer(nbtsock->event_ctx, req,
     424             :                                    timeval_current_ofs(req->timeout, 0),
     425             :                                    nbt_name_socket_timeout, req);
     426             : 
     427       11420 :         talloc_set_destructor(req, nbt_name_request_destructor);
     428             : 
     429       11420 :         ndr_err = ndr_push_struct_blob(&req->encoded, req,
     430             :                                        request,
     431             :                                        (ndr_push_flags_fn_t)ndr_push_nbt_name_packet);
     432       11420 :         if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) goto failed;
     433             : 
     434       11420 :         DLIST_ADD_END(nbtsock->send_queue, req);
     435             : 
     436       11420 :         if (DEBUGLVL(10)) {
     437           0 :                 DEBUG(10,("Queueing nbt packet to %s:%d\n",
     438             :                           req->dest->addr, req->dest->port));
     439           0 :                 NDR_PRINT_DEBUG(nbt_name_packet, request);
     440             :         }
     441             : 
     442       11420 :         TEVENT_FD_WRITEABLE(nbtsock->fde);
     443             : 
     444       11420 :         return req;
     445             : 
     446           0 : failed:
     447           0 :         talloc_free(req);
     448           0 :         return NULL;
     449             : }
     450             : 
     451             : /*
     452             :   send off a nbt name packet
     453             : */
     454           0 : _PUBLIC_ NTSTATUS nbt_name_send_raw(struct nbt_name_socket *nbtsock,
     455             :                                     struct socket_address *dest,
     456             :                                     const DATA_BLOB pkt_blob)
     457             : {
     458           0 :         struct nbt_name_request *req;
     459             : 
     460           0 :         req = talloc_zero(nbtsock, struct nbt_name_request);
     461           0 :         NT_STATUS_HAVE_NO_MEMORY(req);
     462             : 
     463           0 :         req->nbtsock = nbtsock;
     464           0 :         req->dest = socket_address_copy(req, dest);
     465           0 :         if (req->dest == NULL) {
     466           0 :                 goto failed;
     467             :         }
     468           0 :         req->state = NBT_REQUEST_SEND;
     469             :         /*
     470             :          * We don't expect a response so
     471             :          * just pretent it is a request,
     472             :          * but we really don't care about the
     473             :          * content.
     474             :          */
     475           0 :         req->is_reply = true;
     476             : 
     477           0 :         req->encoded = data_blob_dup_talloc(req, pkt_blob);
     478           0 :         if (req->encoded.length != pkt_blob.length) {
     479           0 :                 goto failed;
     480             :         }
     481             : 
     482           0 :         talloc_set_destructor(req, nbt_name_request_destructor);
     483             : 
     484           0 :         DLIST_ADD_END(nbtsock->send_queue, req);
     485             : 
     486           0 :         TEVENT_FD_WRITEABLE(nbtsock->fde);
     487             : 
     488           0 :         return NT_STATUS_OK;
     489             : 
     490           0 : failed:
     491           0 :         talloc_free(req);
     492           0 :         return NT_STATUS_NO_MEMORY;
     493             : }
     494             : 
     495             : 
     496             : /*
     497             :   send off a nbt name reply
     498             : */
     499        1649 : _PUBLIC_ NTSTATUS nbt_name_reply_send(struct nbt_name_socket *nbtsock,
     500             :                              struct socket_address *dest,
     501             :                              struct nbt_name_packet *request)
     502             : {
     503          25 :         struct nbt_name_request *req;
     504          25 :         enum ndr_err_code ndr_err;
     505             : 
     506        1649 :         req = talloc_zero(nbtsock, struct nbt_name_request);
     507        1649 :         NT_STATUS_HAVE_NO_MEMORY(req);
     508             : 
     509        1649 :         req->nbtsock   = nbtsock;
     510        1649 :         req->dest = socket_address_copy(req, dest);
     511        1649 :         if (req->dest == NULL) goto failed;
     512        1649 :         req->state     = NBT_REQUEST_SEND;
     513        1649 :         req->is_reply = true;
     514             : 
     515        1649 :         talloc_set_destructor(req, nbt_name_request_destructor);
     516             : 
     517        1649 :         if (DEBUGLVL(10)) {
     518           0 :                 NDR_PRINT_DEBUG(nbt_name_packet, request);
     519             :         }
     520             : 
     521        1649 :         ndr_err = ndr_push_struct_blob(&req->encoded, req,
     522             :                                        request,
     523             :                                        (ndr_push_flags_fn_t)ndr_push_nbt_name_packet);
     524        1649 :         if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
     525           4 :                 talloc_free(req);
     526           4 :                 return ndr_map_error2ntstatus(ndr_err);
     527             :         }
     528             : 
     529        1645 :         DLIST_ADD_END(nbtsock->send_queue, req);
     530             : 
     531        1645 :         TEVENT_FD_WRITEABLE(nbtsock->fde);
     532             : 
     533        1645 :         return NT_STATUS_OK;
     534             : 
     535           0 : failed:
     536           0 :         talloc_free(req);
     537           0 :         return NT_STATUS_NO_MEMORY;
     538             : }
     539             : 
     540             : /*
     541             :   wait for a nbt request to complete
     542             : */
     543       11335 : NTSTATUS nbt_name_request_recv(struct nbt_name_request *req)
     544             : {
     545       11335 :         if (!req) return NT_STATUS_NO_MEMORY;
     546             : 
     547       14253 :         while (req->state < NBT_REQUEST_DONE) {
     548        2918 :                 if (tevent_loop_once(req->nbtsock->event_ctx) != 0) {
     549           0 :                         req->state = NBT_REQUEST_ERROR;
     550           0 :                         req->status = NT_STATUS_UNEXPECTED_NETWORK_ERROR;
     551           0 :                         break;
     552             :                 }
     553             :         }
     554       11335 :         return req->status;
     555             : }
     556             : 
     557             : 
     558             : /*
     559             :   setup a handler for incoming requests
     560             : */
     561         551 : _PUBLIC_ NTSTATUS nbt_set_incoming_handler(struct nbt_name_socket *nbtsock,
     562             :                                   void (*handler)(struct nbt_name_socket *, struct nbt_name_packet *,
     563             :                                                   struct socket_address *),
     564             :                                   void *private_data)
     565             : {
     566         551 :         nbtsock->incoming.handler = handler;
     567         551 :         nbtsock->incoming.private_data = private_data;
     568         551 :         TEVENT_FD_READABLE(nbtsock->fde);
     569         551 :         return NT_STATUS_OK;
     570             : }
     571             : 
     572             : /*
     573             :   setup a handler for unexpected requests
     574             : */
     575         130 : NTSTATUS nbt_set_unexpected_handler(struct nbt_name_socket *nbtsock,
     576             :                                     void (*handler)(struct nbt_name_socket *, struct nbt_name_packet *,
     577             :                                                     struct socket_address *),
     578             :                                     void *private_data)
     579             : {
     580         130 :         nbtsock->unexpected.handler = handler;
     581         130 :         nbtsock->unexpected.private_data = private_data;
     582         130 :         TEVENT_FD_READABLE(nbtsock->fde);
     583         130 :         return NT_STATUS_OK;
     584             : }
     585             : 
     586             : /*
     587             :   turn a NBT rcode into a NTSTATUS
     588             : */
     589          30 : _PUBLIC_ NTSTATUS nbt_rcode_to_ntstatus(uint8_t rcode)
     590             : {
     591           0 :         size_t i;
     592           0 :         struct {
     593             :                 enum nbt_rcode rcode;
     594             :                 NTSTATUS status;
     595          30 :         } map[] = {
     596             :                 { NBT_RCODE_FMT, NT_STATUS_INVALID_PARAMETER },
     597             :                 { NBT_RCODE_SVR, NT_STATUS_SERVER_DISABLED },
     598             :                 { NBT_RCODE_NAM, NT_STATUS_OBJECT_NAME_NOT_FOUND },
     599             :                 { NBT_RCODE_IMP, NT_STATUS_NOT_SUPPORTED },
     600             :                 { NBT_RCODE_RFS, NT_STATUS_ACCESS_DENIED },
     601             :                 { NBT_RCODE_ACT, NT_STATUS_ADDRESS_ALREADY_EXISTS },
     602             :                 { NBT_RCODE_CFT, NT_STATUS_CONFLICTING_ADDRESSES }
     603             :         };
     604          90 :         for (i=0;i<ARRAY_SIZE(map);i++) {
     605          90 :                 if (map[i].rcode == rcode) {
     606          30 :                         return map[i].status;
     607             :                 }
     608             :         }
     609           0 :         return NT_STATUS_UNSUCCESSFUL;
     610             : }

Generated by: LCOV version 1.14