LCOV - code coverage report
Current view: top level - source3/rpc_server - srv_pipe_hnd.c (source / functions) Hit Total Coverage
Test: coverage report for master 98b443d9 Lines: 134 160 83.8 %
Date: 2024-05-31 13:13:24 Functions: 11 11 100.0 %

          Line data    Source code
       1             : /*
       2             :  *  Unix SMB/CIFS implementation.
       3             :  *  RPC Pipe client / server routines
       4             :  *  Copyright (C) Andrew Tridgell              1992-1998,
       5             :  *  Largely re-written : 2005
       6             :  *  Copyright (C) Jeremy Allison                1998 - 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 "fake_file.h"
      24             : #include "rpc_dce.h"
      25             : #include "ntdomain.h"
      26             : #include "rpc_server/rpc_ncacn_np.h"
      27             : #include "rpc_server/srv_pipe_hnd.h"
      28             : #include "rpc_client/local_np.h"
      29             : #include "rpc_server/rpc_server.h"
      30             : #include "rpc_server/rpc_config.h"
      31             : #include "../lib/tsocket/tsocket.h"
      32             : #include "../lib/util/tevent_ntstatus.h"
      33             : #include "librpc/ndr/ndr_table.h"
      34             : 
      35             : #undef DBGC_CLASS
      36             : #define DBGC_CLASS DBGC_RPC_SRV
      37             : 
      38      336058 : bool fsp_is_np(struct files_struct *fsp)
      39             : {
      40        7242 :         enum FAKE_FILE_TYPE type;
      41             : 
      42      336058 :         if ((fsp == NULL) || (fsp->fake_file_handle == NULL)) {
      43           0 :                 return false;
      44             :         }
      45             : 
      46      336058 :         type = fsp->fake_file_handle->type;
      47             : 
      48      336058 :         return (type == FAKE_FILE_TYPE_NAMED_PIPE_PROXY);
      49             : }
      50             : 
      51       17578 : NTSTATUS np_open(TALLOC_CTX *mem_ctx, const char *name,
      52             :                  const struct tsocket_address *remote_client_address,
      53             :                  const struct tsocket_address *local_server_address,
      54             :                  struct auth_session_info *session_info,
      55             :                  struct tevent_context *ev_ctx,
      56             :                  struct messaging_context *msg_ctx,
      57             :                  struct dcesrv_context *dce_ctx,
      58             :                  struct fake_file_handle **phandle)
      59             : {
      60         620 :         struct fake_file_handle *handle;
      61       17578 :         struct npa_state *npa = NULL;
      62         620 :         int ret;
      63             : 
      64       17578 :         handle = talloc(mem_ctx, struct fake_file_handle);
      65       17578 :         if (handle == NULL) {
      66           0 :                 return NT_STATUS_NO_MEMORY;
      67             :         }
      68             : 
      69       17578 :         npa = npa_state_init(handle);
      70       17578 :         if (npa == NULL) {
      71           0 :                 TALLOC_FREE(handle);
      72           0 :                 return NT_STATUS_NO_MEMORY;
      73             :         }
      74       17578 :         *handle = (struct fake_file_handle) {
      75             :                 .type = FAKE_FILE_TYPE_NAMED_PIPE_PROXY,
      76             :                 .private_data = npa,
      77             :         };
      78             : 
      79       17578 :         ret = local_np_connect(
      80             :                 name,
      81             :                 NCACN_NP,
      82             :                 NULL,
      83             :                 remote_client_address,
      84             :                 NULL,
      85             :                 local_server_address,
      86             :                 session_info,
      87             :                 false,
      88             :                 npa,
      89             :                 &npa->stream);
      90       17578 :         if (ret != 0) {
      91         151 :                 DBG_DEBUG("local_np_connect failed: %s\n",
      92             :                           strerror(ret));
      93         151 :                 TALLOC_FREE(handle);
      94         151 :                 return map_nt_error_from_unix(ret);
      95             :         }
      96             : 
      97       17427 :         *phandle = handle;
      98             : 
      99       17427 :         return NT_STATUS_OK;
     100             : }
     101             : 
     102        1418 : bool np_read_in_progress(struct fake_file_handle *handle)
     103             : {
     104        1418 :         if (handle->type == FAKE_FILE_TYPE_NAMED_PIPE_PROXY) {
     105           0 :                 struct npa_state *p =
     106        1418 :                         talloc_get_type_abort(handle->private_data,
     107             :                                               struct npa_state);
     108           0 :                 size_t read_count;
     109             : 
     110        1418 :                 read_count = tevent_queue_length(p->read_queue);
     111        1418 :                 if (read_count > 0) {
     112           0 :                         return true;
     113             :                 }
     114             : 
     115        1418 :                 return false;
     116             :         }
     117             : 
     118           0 :         return false;
     119             : }
     120             : 
     121             : struct np_write_state {
     122             :         struct tevent_context *ev;
     123             :         struct npa_state *p;
     124             :         struct iovec iov;
     125             :         ssize_t nwritten;
     126             : };
     127             : 
     128             : static void np_write_done(struct tevent_req *subreq);
     129             : 
     130      327118 : struct tevent_req *np_write_send(TALLOC_CTX *mem_ctx, struct tevent_context *ev,
     131             :                                  struct fake_file_handle *handle,
     132             :                                  const uint8_t *data, size_t len)
     133             : {
     134        7194 :         struct tevent_req *req;
     135        7194 :         struct np_write_state *state;
     136      327118 :         struct npa_state *p = NULL;
     137      327118 :         struct tevent_req *subreq = NULL;
     138             : 
     139      327118 :         DBG_INFO("len: %zu\n", len);
     140      327118 :         dump_data(50, data, len);
     141             : 
     142      327118 :         req = tevent_req_create(mem_ctx, &state, struct np_write_state);
     143      327118 :         if (req == NULL) {
     144           0 :                 return NULL;
     145             :         }
     146             : 
     147      327118 :         if (handle->type != FAKE_FILE_TYPE_NAMED_PIPE_PROXY) {
     148           0 :                 tevent_req_nterror(req, NT_STATUS_INVALID_HANDLE);
     149           0 :                 return tevent_req_post(req, ev);
     150             :         }
     151             : 
     152      327118 :         if (len == 0) {
     153           0 :                 state->nwritten = 0;
     154           0 :                 tevent_req_done(req);
     155           0 :                 return tevent_req_post(req, ev);
     156             :         }
     157             : 
     158      327118 :         p = talloc_get_type_abort(handle->private_data, struct npa_state);
     159             : 
     160      327118 :         state->ev = ev;
     161      327118 :         state->p = p;
     162      327118 :         state->iov.iov_base = discard_const_p(void, data);
     163      327118 :         state->iov.iov_len = len;
     164             : 
     165      334312 :         subreq = tstream_writev_queue_send(
     166      327118 :                 state, ev, p->stream, p->write_queue, &state->iov, 1);
     167      327118 :         if (tevent_req_nomem(subreq, req)) {
     168           0 :                 return tevent_req_post(req, ev);
     169             :         }
     170      327118 :         tevent_req_set_callback(subreq, np_write_done, req);
     171      327118 :         return req;
     172             : }
     173             : 
     174      327118 : static void np_write_done(struct tevent_req *subreq)
     175             : {
     176      327118 :         struct tevent_req *req = tevent_req_callback_data(
     177             :                 subreq, struct tevent_req);
     178      327118 :         struct np_write_state *state = tevent_req_data(
     179             :                 req, struct np_write_state);
     180        7194 :         ssize_t received;
     181        7194 :         int err;
     182             : 
     183      327118 :         received = tstream_writev_queue_recv(subreq, &err);
     184      327118 :         if (received < 0) {
     185           0 :                 tevent_req_nterror(req, map_nt_error_from_unix(err));
     186           0 :                 return;
     187             :         }
     188      327118 :         state->nwritten = received;
     189      327118 :         tevent_req_done(req);
     190             : }
     191             : 
     192      327118 : NTSTATUS np_write_recv(struct tevent_req *req, ssize_t *pnwritten)
     193             : {
     194      327118 :         struct np_write_state *state = tevent_req_data(
     195             :                 req, struct np_write_state);
     196        7194 :         NTSTATUS status;
     197             : 
     198      327118 :         if (tevent_req_is_nterror(req, &status)) {
     199           0 :                 return status;
     200             :         }
     201      327118 :         *pnwritten = state->nwritten;
     202      327118 :         return NT_STATUS_OK;
     203             : }
     204             : 
     205             : struct np_ipc_readv_next_vector_state {
     206             :         uint8_t *buf;
     207             :         size_t len;
     208             :         off_t ofs;
     209             :         size_t remaining;
     210             : };
     211             : 
     212      327485 : static void np_ipc_readv_next_vector_init(struct np_ipc_readv_next_vector_state *s,
     213             :                                           uint8_t *buf, size_t len)
     214             : {
     215      334679 :         ZERO_STRUCTP(s);
     216             : 
     217      327485 :         s->buf = buf;
     218      327485 :         s->len = MIN(len, UINT16_MAX);
     219      320291 : }
     220             : 
     221      982425 : static int np_ipc_readv_next_vector(struct tstream_context *stream,
     222             :                                     void *private_data,
     223             :                                     TALLOC_CTX *mem_ctx,
     224             :                                     struct iovec **_vector,
     225             :                                     size_t *count)
     226             : {
     227      982425 :         struct np_ipc_readv_next_vector_state *state =
     228             :                 (struct np_ipc_readv_next_vector_state *)private_data;
     229       21576 :         struct iovec *vector;
     230       21576 :         ssize_t pending;
     231       21576 :         size_t wanted;
     232             : 
     233      982425 :         if (state->ofs == state->len) {
     234        4050 :                 *_vector = NULL;
     235        4050 :                 *count = 0;
     236        4050 :                 return 0;
     237             :         }
     238             : 
     239      978375 :         pending = tstream_pending_bytes(stream);
     240      978375 :         if (pending == -1) {
     241           0 :                 return -1;
     242             :         }
     243             : 
     244      978375 :         if (pending == 0 && state->ofs != 0) {
     245             :                 /* return a short read */
     246      323420 :                 *_vector = NULL;
     247      323420 :                 *count = 0;
     248      323420 :                 return 0;
     249             :         }
     250             : 
     251      654955 :         if (pending == 0) {
     252             :                 /* we want at least one byte and recheck again */
     253      320291 :                 wanted = 1;
     254             :         } else {
     255      327470 :                 size_t missing = state->len - state->ofs;
     256      327470 :                 if (pending > missing) {
     257             :                         /* there's more available */
     258           0 :                         state->remaining = pending - missing;
     259           0 :                         wanted = missing;
     260             :                 } else {
     261             :                         /* read what we can get and recheck in the next cycle */
     262      320279 :                         wanted = pending;
     263             :                 }
     264             :         }
     265             : 
     266      654955 :         vector = talloc_array(mem_ctx, struct iovec, 1);
     267      654955 :         if (!vector) {
     268           0 :                 return -1;
     269             :         }
     270             : 
     271      654955 :         vector[0].iov_base = state->buf + state->ofs;
     272      654955 :         vector[0].iov_len = wanted;
     273             : 
     274      654955 :         state->ofs += wanted;
     275             : 
     276      654955 :         *_vector = vector;
     277      654955 :         *count = 1;
     278      654955 :         return 0;
     279             : }
     280             : 
     281             : struct np_read_state {
     282             :         struct npa_state *p;
     283             :         struct np_ipc_readv_next_vector_state next_vector;
     284             : 
     285             :         ssize_t nread;
     286             :         bool is_data_outstanding;
     287             : };
     288             : 
     289             : static void np_read_done(struct tevent_req *subreq);
     290             : 
     291      327485 : struct tevent_req *np_read_send(TALLOC_CTX *mem_ctx, struct tevent_context *ev,
     292             :                                 struct fake_file_handle *handle,
     293             :                                 uint8_t *data, size_t len)
     294             : {
     295        7194 :         struct tevent_req *req;
     296        7194 :         struct np_read_state *state;
     297      327485 :         struct npa_state *p = NULL;
     298      327485 :         struct tevent_req *subreq = NULL;
     299             : 
     300      327485 :         req = tevent_req_create(mem_ctx, &state, struct np_read_state);
     301      327485 :         if (req == NULL) {
     302           0 :                 return NULL;
     303             :         }
     304             : 
     305      327485 :         if (handle->type != FAKE_FILE_TYPE_NAMED_PIPE_PROXY) {
     306           0 :                 tevent_req_nterror(req, NT_STATUS_INVALID_HANDLE);
     307           0 :                 return tevent_req_post(req, ev);
     308             :         }
     309             : 
     310      327485 :         p = talloc_get_type_abort(handle->private_data, struct npa_state);
     311             : 
     312      327485 :         np_ipc_readv_next_vector_init(&state->next_vector, data, len);
     313             : 
     314      334679 :         subreq = tstream_readv_pdu_queue_send(
     315             :                 state,
     316             :                 ev,
     317             :                 p->stream,
     318             :                 p->read_queue,
     319             :                 np_ipc_readv_next_vector,
     320      327485 :                 &state->next_vector);
     321      327485 :         if (tevent_req_nomem(subreq, req)) {
     322           0 :                 return tevent_req_post(req, ev);
     323             :         }
     324      327485 :         tevent_req_set_callback(subreq, np_read_done, req);
     325      327485 :         return req;
     326             : }
     327             : 
     328      327485 : static void np_read_done(struct tevent_req *subreq)
     329             : {
     330      327485 :         struct tevent_req *req = tevent_req_callback_data(
     331             :                 subreq, struct tevent_req);
     332      327485 :         struct np_read_state *state = tevent_req_data(
     333             :                 req, struct np_read_state);
     334        7194 :         ssize_t ret;
     335        7194 :         int err;
     336             : 
     337      327485 :         ret = tstream_readv_pdu_queue_recv(subreq, &err);
     338      327485 :         TALLOC_FREE(subreq);
     339      327485 :         if (ret == -1) {
     340          15 :                 tevent_req_nterror(req, map_nt_error_from_unix(err));
     341          15 :                 return;
     342             :         }
     343             : 
     344      327470 :         state->nread = ret;
     345      327470 :         state->is_data_outstanding = (state->next_vector.remaining > 0);
     346             : 
     347      327470 :         tevent_req_done(req);
     348      327470 :         return;
     349             : }
     350             : 
     351      327485 : NTSTATUS np_read_recv(struct tevent_req *req, ssize_t *nread,
     352             :                       bool *is_data_outstanding)
     353             : {
     354      327485 :         struct np_read_state *state = tevent_req_data(
     355             :                 req, struct np_read_state);
     356        7194 :         NTSTATUS status;
     357             : 
     358      327485 :         if (tevent_req_is_nterror(req, &status)) {
     359          15 :                 return status;
     360             :         }
     361             : 
     362      327470 :         DEBUG(10, ("Received %d bytes. There is %smore data outstanding\n",
     363             :                    (int)state->nread, state->is_data_outstanding?"":"no "));
     364             : 
     365      327470 :         *nread = state->nread;
     366      327470 :         *is_data_outstanding = state->is_data_outstanding;
     367      327470 :         return NT_STATUS_OK;
     368             : }

Generated by: LCOV version 1.14