LCOV - code coverage report
Current view: top level - source3/smbd - smb2_aio.c (source / functions) Hit Total Coverage
Test: coverage report for fix-15632 9995c5c2 Lines: 211 260 81.2 %
Date: 2024-04-13 12:30:31 Functions: 12 13 92.3 %

          Line data    Source code
       1             : /*
       2             :    Unix SMB/Netbios implementation.
       3             :    Version 3.0
       4             :    async_io read handling using POSIX async io.
       5             :    Copyright (C) Jeremy Allison 2005.
       6             : 
       7             :    This program is free software; you can redistribute it and/or modify
       8             :    it under the terms of the GNU General Public License as published by
       9             :    the Free Software Foundation; either version 3 of the License, or
      10             :    (at your option) any later version.
      11             : 
      12             :    This program is distributed in the hope that it will be useful,
      13             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      14             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
      15             :    GNU General Public License for more details.
      16             : 
      17             :    You should have received a copy of the GNU General Public License
      18             :    along with this program.  If not, see <http://www.gnu.org/licenses/>.
      19             : */
      20             : 
      21             : #include "includes.h"
      22             : #include "smbd/smbd.h"
      23             : #include "smbd/globals.h"
      24             : #include "../lib/util/tevent_ntstatus.h"
      25             : #include "../lib/util/tevent_unix.h"
      26             : 
      27             : /****************************************************************************
      28             :  Accessor function to return write_through state.
      29             : *****************************************************************************/
      30             : 
      31           0 : bool aio_write_through_requested(struct aio_extra *aio_ex)
      32             : {
      33           0 :         return aio_ex->write_through;
      34             : }
      35             : 
      36             : /****************************************************************************
      37             :  Create the extended aio struct we must keep around for the lifetime
      38             :  of the aio call.
      39             : *****************************************************************************/
      40             : 
      41      199460 : struct aio_extra *create_aio_extra(TALLOC_CTX *mem_ctx,
      42             :                                    files_struct *fsp,
      43             :                                    size_t buflen)
      44             : {
      45      199460 :         struct aio_extra *aio_ex = talloc_zero(mem_ctx, struct aio_extra);
      46             : 
      47      199460 :         if (!aio_ex) {
      48           0 :                 return NULL;
      49             :         }
      50             : 
      51             :         /* The output buffer stored in the aio_ex is the start of
      52             :            the smb return buffer. The buffer used in the acb
      53             :            is the start of the reply data portion of that buffer. */
      54             : 
      55      199460 :         if (buflen) {
      56      142066 :                 aio_ex->outbuf = data_blob_talloc(aio_ex, NULL, buflen);
      57      142066 :                 if (!aio_ex->outbuf.data) {
      58           0 :                         TALLOC_FREE(aio_ex);
      59           0 :                         return NULL;
      60             :                 }
      61             :         }
      62      199460 :         aio_ex->fsp = fsp;
      63      199460 :         return aio_ex;
      64             : }
      65             : 
      66             : struct aio_req_fsp_link {
      67             : #ifdef DEVELOPER
      68             :         struct smbd_server_connection *sconn;
      69             : #endif
      70             :         files_struct *fsp;
      71             :         struct tevent_req *req;
      72             : };
      73             : 
      74      528936 : static int aio_del_req_from_fsp(struct aio_req_fsp_link *lnk)
      75             : {
      76        7054 :         unsigned i;
      77      528936 :         files_struct *fsp = lnk->fsp;
      78      528936 :         struct tevent_req *req = lnk->req;
      79             : 
      80             : #ifdef DEVELOPER
      81      528936 :         struct files_struct *ifsp = NULL;
      82      528936 :         bool found = false;
      83             : 
      84             :         /*
      85             :          * When this is called, lnk->fsp must still exist
      86             :          * on the files list for this connection. Panic if not.
      87             :          */
      88     1131771 :         for (ifsp = lnk->sconn->files; ifsp; ifsp = ifsp->next) {
      89      602835 :                 if (ifsp == fsp) {
      90      528936 :                         found = true;
      91             :                 }
      92             :         }
      93      528936 :         if (!found) {
      94           0 :                 smb_panic("orphaned lnk on fsp aio list.\n");
      95             :         }
      96             : #endif
      97             : 
      98      529446 :         for (i=0; i<fsp->num_aio_requests; i++) {
      99      529446 :                 if (fsp->aio_requests[i] == req) {
     100      521882 :                         break;
     101             :                 }
     102             :         }
     103      528936 :         if (i == fsp->num_aio_requests) {
     104           0 :                 DEBUG(1, ("req %p not found in fsp %p\n", req, fsp));
     105           0 :                 return 0;
     106             :         }
     107      528936 :         fsp->num_aio_requests -= 1;
     108      528936 :         fsp->aio_requests[i] = fsp->aio_requests[fsp->num_aio_requests];
     109             : 
     110      528936 :         if (fsp->num_aio_requests == 0) {
     111      524360 :                 TALLOC_FREE(fsp->aio_requests);
     112             :         }
     113      521882 :         return 0;
     114             : }
     115             : 
     116      528936 : bool aio_add_req_to_fsp(files_struct *fsp, struct tevent_req *req)
     117             : {
     118        7054 :         size_t array_len;
     119        7054 :         struct aio_req_fsp_link *lnk;
     120             : 
     121      528936 :         lnk = talloc(req, struct aio_req_fsp_link);
     122      528936 :         if (lnk == NULL) {
     123           0 :                 return false;
     124             :         }
     125             : 
     126      528936 :         array_len = talloc_array_length(fsp->aio_requests);
     127      528936 :         if (array_len <= fsp->num_aio_requests) {
     128        7054 :                 struct tevent_req **tmp;
     129             : 
     130      524360 :                 if (fsp->num_aio_requests + 10 < 10) {
     131             :                         /* Integer wrap. */
     132           0 :                         TALLOC_FREE(lnk);
     133           0 :                         return false;
     134             :                 }
     135             : 
     136             :                 /*
     137             :                  * Allocate in blocks of 10 so we don't allocate
     138             :                  * on every aio request.
     139             :                  */
     140      524360 :                 tmp = talloc_realloc(
     141             :                         fsp, fsp->aio_requests, struct tevent_req *,
     142             :                         fsp->num_aio_requests+10);
     143      524360 :                 if (tmp == NULL) {
     144           0 :                         TALLOC_FREE(lnk);
     145           0 :                         return false;
     146             :                 }
     147      524360 :                 fsp->aio_requests = tmp;
     148             :         }
     149      528936 :         fsp->aio_requests[fsp->num_aio_requests] = req;
     150      528936 :         fsp->num_aio_requests += 1;
     151             : 
     152      528936 :         lnk->fsp = fsp;
     153      528936 :         lnk->req = req;
     154             : #ifdef DEVELOPER
     155      528936 :         lnk->sconn = fsp->conn->sconn;
     156             : #endif
     157      528936 :         talloc_set_destructor(lnk, aio_del_req_from_fsp);
     158             : 
     159      528936 :         return true;
     160             : }
     161             : 
     162             : struct pwrite_fsync_state {
     163             :         struct tevent_context *ev;
     164             :         files_struct *fsp;
     165             :         bool write_through;
     166             :         ssize_t nwritten;
     167             : };
     168             : 
     169             : static void pwrite_fsync_write_done(struct tevent_req *subreq);
     170             : static void pwrite_fsync_sync_done(struct tevent_req *subreq);
     171             : 
     172      185738 : struct tevent_req *pwrite_fsync_send(TALLOC_CTX *mem_ctx,
     173             :                                      struct tevent_context *ev,
     174             :                                      struct files_struct *fsp,
     175             :                                      const void *data,
     176             :                                      size_t n, off_t offset,
     177             :                                      bool write_through)
     178             : {
     179          52 :         struct tevent_req *req, *subreq;
     180          52 :         struct pwrite_fsync_state *state;
     181          52 :         bool ok;
     182             : 
     183      185738 :         req = tevent_req_create(mem_ctx, &state, struct pwrite_fsync_state);
     184      185738 :         if (req == NULL) {
     185           0 :                 return NULL;
     186             :         }
     187      185738 :         state->ev = ev;
     188      185738 :         state->fsp = fsp;
     189      185738 :         state->write_through = write_through;
     190             : 
     191      185738 :         ok = vfs_valid_pwrite_range(offset, n);
     192      185738 :         if (!ok) {
     193          30 :                 tevent_req_error(req, EINVAL);
     194          30 :                 return tevent_req_post(req, ev);
     195             :         }
     196             : 
     197      185708 :         if (n == 0) {
     198           0 :                 tevent_req_done(req);
     199           0 :                 return tevent_req_post(req, ev);
     200             :         }
     201             : 
     202      185708 :         subreq = SMB_VFS_PWRITE_SEND(state, ev, fsp, data, n, offset);
     203      185708 :         if (tevent_req_nomem(subreq, req)) {
     204           0 :                 return tevent_req_post(req, ev);
     205             :         }
     206      185708 :         tevent_req_set_callback(subreq, pwrite_fsync_write_done, req);
     207      185708 :         return req;
     208             : }
     209             : 
     210      185704 : static void pwrite_fsync_write_done(struct tevent_req *subreq)
     211             : {
     212      185704 :         struct tevent_req *req = tevent_req_callback_data(
     213             :                 subreq, struct tevent_req);
     214      185704 :         struct pwrite_fsync_state *state = tevent_req_data(
     215             :                 req, struct pwrite_fsync_state);
     216      185704 :         connection_struct *conn = state->fsp->conn;
     217          52 :         bool do_sync;
     218          52 :         struct vfs_aio_state vfs_aio_state;
     219             : 
     220      185704 :         state->nwritten = SMB_VFS_PWRITE_RECV(subreq, &vfs_aio_state);
     221      185704 :         TALLOC_FREE(subreq);
     222      185704 :         if (state->nwritten == -1) {
     223           0 :                 tevent_req_error(req, vfs_aio_state.error);
     224      185631 :                 return;
     225             :         }
     226             : 
     227      557060 :         do_sync = (lp_strict_sync(SNUM(conn)) &&
     228      371408 :                    (lp_sync_always(SNUM(conn)) || state->write_through));
     229      185704 :         if (!do_sync) {
     230      185631 :                 tevent_req_done(req);
     231      185631 :                 return;
     232             :         }
     233             : 
     234          73 :         subreq = SMB_VFS_FSYNC_SEND(state, state->ev, state->fsp);
     235          73 :         if (tevent_req_nomem(subreq, req)) {
     236           0 :                 return;
     237             :         }
     238          73 :         tevent_req_set_callback(subreq, pwrite_fsync_sync_done, req);
     239             : }
     240             : 
     241          73 : static void pwrite_fsync_sync_done(struct tevent_req *subreq)
     242             : {
     243          73 :         struct tevent_req *req = tevent_req_callback_data(
     244             :                 subreq, struct tevent_req);
     245           1 :         int ret;
     246           1 :         struct vfs_aio_state vfs_aio_state;
     247             : 
     248          73 :         ret = SMB_VFS_FSYNC_RECV(subreq, &vfs_aio_state);
     249          73 :         TALLOC_FREE(subreq);
     250          73 :         if (ret == -1) {
     251           0 :                 tevent_req_error(req, vfs_aio_state.error);
     252           0 :                 return;
     253             :         }
     254          73 :         tevent_req_done(req);
     255             : }
     256             : 
     257      185734 : ssize_t pwrite_fsync_recv(struct tevent_req *req, int *perr)
     258             : {
     259      185734 :         struct pwrite_fsync_state *state = tevent_req_data(
     260             :                 req, struct pwrite_fsync_state);
     261             : 
     262      185734 :         if (tevent_req_is_unix_error(req, perr)) {
     263          30 :                 return -1;
     264             :         }
     265      185704 :         return state->nwritten;
     266             : }
     267             : 
     268           2 : bool cancel_smb2_aio(struct smb_request *smbreq)
     269             : {
     270           2 :         struct smbd_smb2_request *smb2req = smbreq->smb2req;
     271           2 :         struct aio_extra *aio_ex = NULL;
     272             : 
     273           2 :         if (smb2req) {
     274           2 :                 aio_ex = talloc_get_type(smbreq->async_priv,
     275             :                                          struct aio_extra);
     276             :         }
     277             : 
     278           2 :         if (aio_ex == NULL) {
     279           0 :                 return false;
     280             :         }
     281             : 
     282           2 :         if (aio_ex->fsp == NULL) {
     283           0 :                 return false;
     284             :         }
     285             : 
     286             :         /*
     287             :          * We let the aio request run and don't try to cancel it which means
     288             :          * processing of the SMB2 request must continue as normal, cf MS-SMB2
     289             :          * 3.3.5.16:
     290             :          *
     291             :          *   If the target request is not successfully canceled, processing of
     292             :          *   the target request MUST continue and no response is sent to the
     293             :          *   cancel request.
     294             :          */
     295             : 
     296           2 :         return false;
     297             : }
     298             : 
     299             : static void aio_pread_smb2_done(struct tevent_req *req);
     300             : 
     301             : /****************************************************************************
     302             :  Set up an aio request from a SMB2 read call.
     303             : *****************************************************************************/
     304             : 
     305        4929 : NTSTATUS schedule_smb2_aio_read(connection_struct *conn,
     306             :                                 struct smb_request *smbreq,
     307             :                                 files_struct *fsp,
     308             :                                 TALLOC_CTX *ctx,
     309             :                                 DATA_BLOB *preadbuf,
     310             :                                 off_t startpos,
     311             :                                 size_t smb_maxcnt)
     312             : {
     313           0 :         struct aio_extra *aio_ex;
     314        4929 :         size_t min_aio_read_size = lp_aio_read_size(SNUM(conn));
     315           0 :         struct tevent_req *req;
     316        4929 :         bool is_compound = false;
     317        4929 :         bool is_last_in_compound = false;
     318           0 :         bool ok;
     319             : 
     320        4929 :         ok = vfs_valid_pread_range(startpos, smb_maxcnt);
     321        4929 :         if (!ok) {
     322          36 :                 return NT_STATUS_INVALID_PARAMETER;
     323             :         }
     324             : 
     325        4893 :         if (fsp_is_alternate_stream(fsp)) {
     326         704 :                 DEBUG(10, ("AIO on streams not yet supported\n"));
     327         704 :                 return NT_STATUS_RETRY;
     328             :         }
     329             : 
     330        4189 :         if (fsp->op == NULL) {
     331             :                 /* No AIO on internal opens. */
     332           0 :                 return NT_STATUS_RETRY;
     333             :         }
     334             : 
     335        4189 :         if ((!min_aio_read_size || (smb_maxcnt < min_aio_read_size))
     336          38 :             && !SMB_VFS_AIO_FORCE(fsp)) {
     337             :                 /* Too small a read for aio request. */
     338          38 :                 DEBUG(10,("smb2: read size (%u) too small "
     339             :                         "for minimum aio_read of %u\n",
     340             :                         (unsigned int)smb_maxcnt,
     341             :                         (unsigned int)min_aio_read_size ));
     342          38 :                 return NT_STATUS_RETRY;
     343             :         }
     344             : 
     345        4151 :         is_compound = smbd_smb2_is_compound(smbreq->smb2req);
     346        4151 :         is_last_in_compound = smbd_smb2_is_last_in_compound(smbreq->smb2req);
     347             : 
     348        4151 :         if (is_compound && !is_last_in_compound) {
     349             :                 /*
     350             :                  * Only allow going async if this is the last
     351             :                  * request in a compound.
     352             :                  */
     353          26 :                 return NT_STATUS_RETRY;
     354             :         }
     355             : 
     356             :         /* Create the out buffer. */
     357        4125 :         *preadbuf = data_blob_talloc(ctx, NULL, smb_maxcnt);
     358        4125 :         if (preadbuf->data == NULL) {
     359           0 :                 return NT_STATUS_NO_MEMORY;
     360             :         }
     361             : 
     362        4125 :         if (!(aio_ex = create_aio_extra(smbreq->smb2req, fsp, 0))) {
     363           0 :                 return NT_STATUS_NO_MEMORY;
     364             :         }
     365             : 
     366        4125 :         init_strict_lock_struct(fsp,
     367        4125 :                         fsp->op->global->open_persistent_id,
     368             :                         (uint64_t)startpos,
     369             :                         (uint64_t)smb_maxcnt,
     370             :                         READ_LOCK,
     371             :                         lp_posix_cifsu_locktype(fsp),
     372             :                         &aio_ex->lock);
     373             : 
     374             :         /* Take the lock until the AIO completes. */
     375        4125 :         if (!SMB_VFS_STRICT_LOCK_CHECK(conn, fsp, &aio_ex->lock)) {
     376           4 :                 TALLOC_FREE(aio_ex);
     377           4 :                 return NT_STATUS_FILE_LOCK_CONFLICT;
     378             :         }
     379             : 
     380        4121 :         aio_ex->nbyte = smb_maxcnt;
     381        4121 :         aio_ex->offset = startpos;
     382             : 
     383        4121 :         req = SMB_VFS_PREAD_SEND(aio_ex, fsp->conn->sconn->ev_ctx, fsp,
     384             :                                  preadbuf->data, smb_maxcnt, startpos);
     385        4121 :         if (req == NULL) {
     386           0 :                 DEBUG(0, ("smb2: SMB_VFS_PREAD_SEND failed. "
     387             :                           "Error %s\n", strerror(errno)));
     388           0 :                 TALLOC_FREE(aio_ex);
     389           0 :                 return NT_STATUS_RETRY;
     390             :         }
     391        4121 :         tevent_req_set_callback(req, aio_pread_smb2_done, aio_ex);
     392             : 
     393        4121 :         if (!aio_add_req_to_fsp(fsp, req)) {
     394           0 :                 DEBUG(1, ("Could not add req to fsp\n"));
     395           0 :                 TALLOC_FREE(aio_ex);
     396           0 :                 return NT_STATUS_RETRY;
     397             :         }
     398             : 
     399             :         /* We don't need talloc_move here as both aio_ex and
     400             :          * smbreq are children of smbreq->smb2req. */
     401        4121 :         aio_ex->smbreq = smbreq;
     402        4121 :         smbreq->async_priv = aio_ex;
     403             : 
     404        4121 :         DEBUG(10,("smb2: scheduled aio_read for file %s, "
     405             :                 "offset %.0f, len = %u (mid = %u)\n",
     406             :                 fsp_str_dbg(fsp), (double)startpos, (unsigned int)smb_maxcnt,
     407             :                 (unsigned int)aio_ex->smbreq->mid ));
     408             : 
     409        4121 :         return NT_STATUS_OK;
     410             : }
     411             : 
     412        4121 : static void aio_pread_smb2_done(struct tevent_req *req)
     413             : {
     414        4121 :         struct aio_extra *aio_ex = tevent_req_callback_data(
     415             :                 req, struct aio_extra);
     416        4121 :         struct tevent_req *subreq = aio_ex->smbreq->smb2req->subreq;
     417        4121 :         files_struct *fsp = aio_ex->fsp;
     418           0 :         NTSTATUS status;
     419           0 :         ssize_t nread;
     420        4121 :         struct vfs_aio_state vfs_aio_state = { 0 };
     421             : 
     422        4121 :         nread = SMB_VFS_PREAD_RECV(req, &vfs_aio_state);
     423        4121 :         TALLOC_FREE(req);
     424             : 
     425        4121 :         DEBUG(10, ("pread_recv returned %d, err = %s\n", (int)nread,
     426             :                    (nread == -1) ? strerror(vfs_aio_state.error) : "no error"));
     427             : 
     428             :         /* Common error or success code processing for async or sync
     429             :            read returns. */
     430             : 
     431        4121 :         status = smb2_read_complete(subreq, nread, vfs_aio_state.error);
     432             : 
     433        4121 :         if (nread > 0) {
     434        3775 :                 fh_set_pos(fsp->fh, aio_ex->offset + nread);
     435        3775 :                 fh_set_position_information(fsp->fh,
     436        3775 :                                                 fh_get_pos(fsp->fh));
     437             :         }
     438             : 
     439        4121 :         DEBUG(10, ("smb2: scheduled aio_read completed "
     440             :                    "for file %s, offset %.0f, len = %u "
     441             :                    "(errcode = %d, NTSTATUS = %s)\n",
     442             :                    fsp_str_dbg(aio_ex->fsp),
     443             :                    (double)aio_ex->offset,
     444             :                    (unsigned int)nread,
     445             :                    vfs_aio_state.error, nt_errstr(status)));
     446             : 
     447        4121 :         if (tevent_req_nterror(subreq, status)) {
     448         362 :                 return;
     449             :         }
     450        3759 :         tevent_req_done(subreq);
     451             : }
     452             : 
     453             : static void aio_pwrite_smb2_done(struct tevent_req *req);
     454             : 
     455             : /****************************************************************************
     456             :  Set up an aio request from a SMB2write call.
     457             : *****************************************************************************/
     458             : 
     459       56409 : NTSTATUS schedule_aio_smb2_write(connection_struct *conn,
     460             :                                 struct smb_request *smbreq,
     461             :                                 files_struct *fsp,
     462             :                                 uint64_t in_offset,
     463             :                                 DATA_BLOB in_data,
     464             :                                 bool write_through)
     465             : {
     466       56409 :         struct aio_extra *aio_ex = NULL;
     467       56409 :         size_t min_aio_write_size = lp_aio_write_size(SNUM(conn));
     468           0 :         struct tevent_req *req;
     469       56409 :         bool is_compound = false;
     470       56409 :         bool is_last_in_compound = false;
     471             : 
     472       56409 :         if (fsp_is_alternate_stream(fsp)) {
     473             :                 /* No AIO on streams yet */
     474        3108 :                 DEBUG(10, ("AIO on streams not yet supported\n"));
     475        3108 :                 return NT_STATUS_RETRY;
     476             :         }
     477             : 
     478       53301 :         if (fsp->op == NULL) {
     479             :                 /* No AIO on internal opens. */
     480           0 :                 return NT_STATUS_RETRY;
     481             :         }
     482             : 
     483       53301 :         if ((!min_aio_write_size || (in_data.length < min_aio_write_size))
     484          24 :             && !SMB_VFS_AIO_FORCE(fsp)) {
     485             :                 /* Too small a write for aio request. */
     486          24 :                 DEBUG(10,("smb2: write size (%u) too "
     487             :                         "small for minimum aio_write of %u\n",
     488             :                         (unsigned int)in_data.length,
     489             :                         (unsigned int)min_aio_write_size ));
     490          24 :                 return NT_STATUS_RETRY;
     491             :         }
     492             : 
     493       53277 :         is_compound = smbd_smb2_is_compound(smbreq->smb2req);
     494       53277 :         is_last_in_compound = smbd_smb2_is_last_in_compound(smbreq->smb2req);
     495             : 
     496       53277 :         if (is_compound && !is_last_in_compound) {
     497             :                 /*
     498             :                  * Only allow going async if this is the last
     499             :                  * request in a compound.
     500             :                  */
     501           8 :                 return NT_STATUS_RETRY;
     502             :         }
     503             : 
     504       53269 :         if (smbreq->unread_bytes) {
     505             :                 /* Can't do async with recvfile. */
     506           0 :                 return NT_STATUS_RETRY;
     507             :         }
     508             : 
     509       53269 :         if (!(aio_ex = create_aio_extra(smbreq->smb2req, fsp, 0))) {
     510           0 :                 return NT_STATUS_NO_MEMORY;
     511             :         }
     512             : 
     513       53269 :         aio_ex->write_through = write_through;
     514             : 
     515       53269 :         init_strict_lock_struct(fsp,
     516       53269 :                         fsp->op->global->open_persistent_id,
     517             :                         in_offset,
     518       53269 :                         (uint64_t)in_data.length,
     519             :                         WRITE_LOCK,
     520             :                         lp_posix_cifsu_locktype(fsp),
     521             :                         &aio_ex->lock);
     522             : 
     523             :         /* Take the lock until the AIO completes. */
     524       53269 :         if (!SMB_VFS_STRICT_LOCK_CHECK(conn, fsp, &aio_ex->lock)) {
     525          12 :                 TALLOC_FREE(aio_ex);
     526          12 :                 return NT_STATUS_FILE_LOCK_CONFLICT;
     527             :         }
     528             : 
     529       53257 :         aio_ex->nbyte = in_data.length;
     530       53257 :         aio_ex->offset = in_offset;
     531             : 
     532       53257 :         req = pwrite_fsync_send(aio_ex, fsp->conn->sconn->ev_ctx, fsp,
     533       53257 :                                 in_data.data, in_data.length, in_offset,
     534             :                                 write_through);
     535       53257 :         if (req == NULL) {
     536           0 :                 DEBUG(3, ("smb2: SMB_VFS_PWRITE_SEND failed. "
     537             :                           "Error %s\n", strerror(errno)));
     538           0 :                 TALLOC_FREE(aio_ex);
     539           0 :                 return NT_STATUS_RETRY;
     540             :         }
     541       53257 :         tevent_req_set_callback(req, aio_pwrite_smb2_done, aio_ex);
     542             : 
     543       53257 :         if (!aio_add_req_to_fsp(fsp, req)) {
     544           0 :                 DEBUG(1, ("Could not add req to fsp\n"));
     545           0 :                 TALLOC_FREE(aio_ex);
     546           0 :                 return NT_STATUS_RETRY;
     547             :         }
     548             : 
     549             :         /* We don't need talloc_move here as both aio_ex and
     550             :         * smbreq are children of smbreq->smb2req. */
     551       53257 :         aio_ex->smbreq = smbreq;
     552       53257 :         smbreq->async_priv = aio_ex;
     553             : 
     554             :         /* This should actually be improved to span the write. */
     555       53257 :         contend_level2_oplocks_begin(fsp, LEVEL2_CONTEND_WRITE);
     556       53257 :         contend_level2_oplocks_end(fsp, LEVEL2_CONTEND_WRITE);
     557             : 
     558             :         /*
     559             :          * We don't want to do write behind due to ownership
     560             :          * issues of the request structs. Maybe add it if I
     561             :          * figure those out. JRA.
     562             :          */
     563             : 
     564       53257 :         DEBUG(10,("smb2: scheduled aio_write for file "
     565             :                 "%s, offset %.0f, len = %u (mid = %u)\n",
     566             :                 fsp_str_dbg(fsp),
     567             :                 (double)in_offset,
     568             :                 (unsigned int)in_data.length,
     569             :                 (unsigned int)aio_ex->smbreq->mid));
     570             : 
     571       53257 :         return NT_STATUS_OK;
     572             : }
     573             : 
     574       53253 : static void aio_pwrite_smb2_done(struct tevent_req *req)
     575             : {
     576       53253 :         struct aio_extra *aio_ex = tevent_req_callback_data(
     577             :                 req, struct aio_extra);
     578       53253 :         ssize_t numtowrite = aio_ex->nbyte;
     579       53253 :         struct tevent_req *subreq = aio_ex->smbreq->smb2req->subreq;
     580       53253 :         files_struct *fsp = aio_ex->fsp;
     581           0 :         NTSTATUS status;
     582           0 :         ssize_t nwritten;
     583       53253 :         int err = 0;
     584             : 
     585       53253 :         nwritten = pwrite_fsync_recv(req, &err);
     586       53253 :         TALLOC_FREE(req);
     587             : 
     588       53253 :         DEBUG(10, ("pwrite_recv returned %d, err = %s\n", (int)nwritten,
     589             :                    (nwritten == -1) ? strerror(err) : "no error"));
     590             : 
     591       53253 :         mark_file_modified(fsp);
     592             : 
     593       53253 :         status = smb2_write_complete_nosync(subreq, nwritten, err);
     594             : 
     595       53253 :         DEBUG(10, ("smb2: scheduled aio_write completed "
     596             :                    "for file %s, offset %.0f, requested %u, "
     597             :                    "written = %u (errcode = %d, NTSTATUS = %s)\n",
     598             :                    fsp_str_dbg(fsp),
     599             :                    (double)aio_ex->offset,
     600             :                    (unsigned int)numtowrite,
     601             :                    (unsigned int)nwritten,
     602             :                    err, nt_errstr(status)));
     603             : 
     604       53253 :         if (tevent_req_nterror(subreq, status)) {
     605          30 :                 return;
     606             :         }
     607       53223 :         tevent_req_done(subreq);
     608             : }

Generated by: LCOV version 1.14