LCOV - code coverage report
Current view: top level - source3/libsmb - clifile.c (source / functions) Hit Total Coverage
Test: coverage report for master 70ed9daf Lines: 2309 3496 66.0 %
Date: 2024-01-11 09:59:51 Functions: 210 241 87.1 %

          Line data    Source code
       1             : /* 
       2             :    Unix SMB/CIFS implementation.
       3             :    client file operations
       4             :    Copyright (C) Andrew Tridgell 1994-1998
       5             :    Copyright (C) Jeremy Allison 2001-2009
       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 "system/filesys.h"
      23             : #include "libsmb/libsmb.h"
      24             : #include "../lib/util/tevent_ntstatus.h"
      25             : #include "async_smb.h"
      26             : #include "libsmb/clirap.h"
      27             : #include "trans2.h"
      28             : #include "ntioctl.h"
      29             : #include "libcli/security/security.h"
      30             : #include "../libcli/smb/smbXcli_base.h"
      31             : #include "libcli/smb/reparse.h"
      32             : 
      33             : struct cli_setpathinfo_state {
      34             :         uint16_t setup;
      35             :         uint8_t *param;
      36             : };
      37             : 
      38             : static void cli_setpathinfo_done(struct tevent_req *subreq);
      39             : 
      40         900 : struct tevent_req *cli_setpathinfo_send(TALLOC_CTX *mem_ctx,
      41             :                                         struct tevent_context *ev,
      42             :                                         struct cli_state *cli,
      43             :                                         uint16_t level,
      44             :                                         const char *path,
      45             :                                         uint8_t *data,
      46             :                                         size_t data_len)
      47             : {
      48           0 :         struct tevent_req *req, *subreq;
      49           0 :         struct cli_setpathinfo_state *state;
      50         900 :         uint16_t additional_flags2 = 0;
      51         900 :         char *path_cp = NULL;
      52             : 
      53         900 :         req = tevent_req_create(mem_ctx, &state,
      54             :                                 struct cli_setpathinfo_state);
      55         900 :         if (req == NULL) {
      56           0 :                 return NULL;
      57             :         }
      58             : 
      59             :         /* Setup setup word. */
      60         900 :         SSVAL(&state->setup, 0, TRANSACT2_SETPATHINFO);
      61             : 
      62             :         /* Setup param array. */
      63         900 :         state->param = talloc_zero_array(state, uint8_t, 6);
      64         900 :         if (tevent_req_nomem(state->param, req)) {
      65           0 :                 return tevent_req_post(req, ev);
      66             :         }
      67         900 :         SSVAL(state->param, 0, level);
      68             : 
      69             :         /* Check for DFS. */
      70         900 :         path_cp = smb1_dfs_share_path(state, cli, path);
      71         900 :         if (tevent_req_nomem(path_cp, req)) {
      72           0 :                 return tevent_req_post(req, ev);
      73             :         }
      74         900 :         state->param = trans2_bytes_push_str(state->param,
      75         900 :                                         smbXcli_conn_use_unicode(cli->conn),
      76             :                                         path_cp,
      77         900 :                                         strlen(path_cp)+1,
      78             :                                         NULL);
      79         900 :         if (tevent_req_nomem(state->param, req)) {
      80           0 :                 return tevent_req_post(req, ev);
      81             :         }
      82             : 
      83         900 :         if (clistr_is_previous_version_path(path) &&
      84           0 :                         !INFO_LEVEL_IS_UNIX(level)) {
      85           0 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
      86             :         }
      87             : 
      88        1800 :         subreq = cli_trans_send(
      89             :                 state,                  /* mem ctx. */
      90             :                 ev,                     /* event ctx. */
      91             :                 cli,                    /* cli_state. */
      92             :                 additional_flags2,      /* additional_flags2 */
      93             :                 SMBtrans2,              /* cmd. */
      94             :                 NULL,                   /* pipe name. */
      95             :                 -1,                     /* fid. */
      96             :                 0,                      /* function. */
      97             :                 0,                      /* flags. */
      98         900 :                 &state->setup,           /* setup. */
      99             :                 1,                      /* num setup uint16_t words. */
     100             :                 0,                      /* max returned setup. */
     101         900 :                 state->param,                /* param. */
     102         900 :                 talloc_get_size(state->param),       /* num param. */
     103             :                 2,                      /* max returned param. */
     104             :                 data,                   /* data. */
     105             :                 data_len,               /* num data. */
     106             :                 0);                     /* max returned data. */
     107             : 
     108         900 :         if (tevent_req_nomem(subreq, req)) {
     109           0 :                 return tevent_req_post(req, ev);
     110             :         }
     111         900 :         tevent_req_set_callback(subreq, cli_setpathinfo_done, req);
     112         900 :         return req;
     113             : }
     114             : 
     115         900 : static void cli_setpathinfo_done(struct tevent_req *subreq)
     116             : {
     117         900 :         NTSTATUS status = cli_trans_recv(subreq, NULL, NULL, NULL, 0, NULL,
     118             :                                          NULL, 0, NULL, NULL, 0, NULL);
     119         900 :         tevent_req_simple_finish_ntstatus(subreq, status);
     120         900 : }
     121             : 
     122         900 : NTSTATUS cli_setpathinfo_recv(struct tevent_req *req)
     123             : {
     124         900 :         return tevent_req_simple_recv_ntstatus(req);
     125             : }
     126             : 
     127         172 : NTSTATUS cli_setpathinfo(struct cli_state *cli,
     128             :                          uint16_t level,
     129             :                          const char *path,
     130             :                          uint8_t *data,
     131             :                          size_t data_len)
     132             : {
     133         172 :         TALLOC_CTX *frame = talloc_stackframe();
     134           0 :         struct tevent_context *ev;
     135           0 :         struct tevent_req *req;
     136         172 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
     137             : 
     138         172 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
     139             :                 /*
     140             :                  * Can't use sync call while an async call is in flight
     141             :                  */
     142           0 :                 status = NT_STATUS_INVALID_PARAMETER;
     143           0 :                 goto fail;
     144             :         }
     145         172 :         ev = samba_tevent_context_init(frame);
     146         172 :         if (ev == NULL) {
     147           0 :                 goto fail;
     148             :         }
     149         172 :         req = cli_setpathinfo_send(ev, ev, cli, level, path, data, data_len);
     150         172 :         if (req == NULL) {
     151           0 :                 goto fail;
     152             :         }
     153         172 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
     154           0 :                 goto fail;
     155             :         }
     156         172 :         status = cli_setpathinfo_recv(req);
     157         172 :  fail:
     158         172 :         TALLOC_FREE(frame);
     159         172 :         return status;
     160             : }
     161             : 
     162             : struct cli_setfileinfo_state {
     163             :         uint16_t setup;
     164             :         uint8_t param[6];
     165             : };
     166             : 
     167             : static void cli_setfileinfo_done(struct tevent_req *subreq);
     168             : 
     169         144 : struct tevent_req *cli_setfileinfo_send(
     170             :         TALLOC_CTX *mem_ctx,
     171             :         struct tevent_context *ev,
     172             :         struct cli_state *cli,
     173             :         uint16_t fnum,
     174             :         uint16_t level,
     175             :         uint8_t *data,
     176             :         size_t data_len)
     177             : {
     178         144 :         struct tevent_req *req = NULL, *subreq = NULL;
     179         144 :         struct cli_setfileinfo_state *state = NULL;
     180             : 
     181         144 :         req = tevent_req_create(mem_ctx, &state, struct cli_setfileinfo_state);
     182         144 :         if (req == NULL) {
     183           0 :                 return NULL;
     184             :         }
     185         144 :         PUSH_LE_U16(&state->setup, 0, TRANSACT2_SETFILEINFO);
     186             : 
     187         144 :         PUSH_LE_U16(state->param, 0, fnum);
     188         144 :         PUSH_LE_U16(state->param, 2, level);
     189             : 
     190         144 :         subreq = cli_trans_send(state,          /* mem ctx. */
     191             :                                 ev,             /* event ctx. */
     192             :                                 cli,            /* cli_state. */
     193             :                                 0,              /* additional_flags2 */
     194             :                                 SMBtrans2,      /* cmd. */
     195             :                                 NULL,           /* pipe name. */
     196             :                                 -1,             /* fid. */
     197             :                                 0,              /* function. */
     198             :                                 0,              /* flags. */
     199         144 :                                 &state->setup,   /* setup. */
     200             :                                 1,              /* num setup uint16_t words. */
     201             :                                 0,              /* max returned setup. */
     202         144 :                                 state->param,        /* param. */
     203             :                                 6,              /* num param. */
     204             :                                 2,              /* max returned param. */
     205             :                                 data,           /* data. */
     206             :                                 data_len,       /* num data. */
     207             :                                 0);             /* max returned data. */
     208             : 
     209         144 :         if (tevent_req_nomem(subreq, req)) {
     210           0 :                 return tevent_req_post(req, ev);
     211             :         }
     212         144 :         tevent_req_set_callback(subreq, cli_setfileinfo_done, req);
     213         144 :         return req;
     214             : }
     215             : 
     216         144 : static void cli_setfileinfo_done(struct tevent_req *subreq)
     217             : {
     218         144 :         NTSTATUS status = cli_trans_recv(
     219             :                 subreq,         /* req */
     220             :                 NULL,           /* mem_ctx */
     221             :                 NULL,           /* recv_flags2 */
     222             :                 NULL,           /* setup */
     223             :                 0,              /* min_setup */
     224             :                 NULL,           /* num_setup */
     225             :                 NULL,           /* param */
     226             :                 0,              /* min_param */
     227             :                 NULL,           /* num_param */
     228             :                 NULL,           /* data */
     229             :                 0,              /* min_data */
     230             :                 NULL);          /* num_data */
     231         144 :         tevent_req_simple_finish_ntstatus(subreq, status);
     232         144 : }
     233             : 
     234         144 : NTSTATUS cli_setfileinfo_recv(struct tevent_req *req)
     235             : {
     236         144 :         return tevent_req_simple_recv_ntstatus(req);
     237             : }
     238             : 
     239             : /****************************************************************************
     240             :  Hard/Symlink a file (UNIX extensions).
     241             :  Creates new name (sym)linked to link_target.
     242             : ****************************************************************************/
     243             : 
     244             : struct cli_posix_link_internal_state {
     245             :         uint8_t *data;
     246             : };
     247             : 
     248             : static void cli_posix_link_internal_done(struct tevent_req *subreq);
     249             : 
     250         128 : static struct tevent_req *cli_posix_link_internal_send(TALLOC_CTX *mem_ctx,
     251             :                                         struct tevent_context *ev,
     252             :                                         struct cli_state *cli,
     253             :                                         uint16_t level,
     254             :                                         const char *link_target,
     255             :                                         const char *newname)
     256             : {
     257         128 :         struct tevent_req *req = NULL, *subreq = NULL;
     258         128 :         struct cli_posix_link_internal_state *state = NULL;
     259             : 
     260         128 :         req = tevent_req_create(mem_ctx, &state,
     261             :                                 struct cli_posix_link_internal_state);
     262         128 :         if (req == NULL) {
     263           0 :                 return NULL;
     264             :         }
     265             : 
     266             :         /* Setup data array. */
     267         128 :         state->data = talloc_array(state, uint8_t, 0);
     268         128 :         if (tevent_req_nomem(state->data, req)) {
     269           0 :                 return tevent_req_post(req, ev);
     270             :         }
     271         384 :         state->data = trans2_bytes_push_str(
     272         128 :                 state->data, smbXcli_conn_use_unicode(cli->conn),
     273         128 :                 link_target, strlen(link_target)+1, NULL);
     274             : 
     275         256 :         subreq = cli_setpathinfo_send(
     276             :                 state, ev, cli, level, newname,
     277         128 :                 state->data, talloc_get_size(state->data));
     278         128 :         if (tevent_req_nomem(subreq, req)) {
     279           0 :                 return tevent_req_post(req, ev);
     280             :         }
     281         128 :         tevent_req_set_callback(subreq, cli_posix_link_internal_done, req);
     282         128 :         return req;
     283             : }
     284             : 
     285         128 : static void cli_posix_link_internal_done(struct tevent_req *subreq)
     286             : {
     287         128 :         NTSTATUS status = cli_setpathinfo_recv(subreq);
     288         128 :         tevent_req_simple_finish_ntstatus(subreq, status);
     289         128 : }
     290             : 
     291         128 : static NTSTATUS cli_posix_link_internal_recv(struct tevent_req *req)
     292             : {
     293         128 :         return tevent_req_simple_recv_ntstatus(req);
     294             : }
     295             : 
     296             : /****************************************************************************
     297             :  Symlink a file (UNIX extensions).
     298             : ****************************************************************************/
     299             : 
     300             : struct cli_posix_symlink_state {
     301             :         uint8_t dummy;
     302             : };
     303             : 
     304             : static void cli_posix_symlink_done(struct tevent_req *subreq);
     305             : 
     306         120 : struct tevent_req *cli_posix_symlink_send(TALLOC_CTX *mem_ctx,
     307             :                                         struct tevent_context *ev,
     308             :                                         struct cli_state *cli,
     309             :                                         const char *link_target,
     310             :                                         const char *newname)
     311             : {
     312         120 :         struct tevent_req *req = NULL, *subreq = NULL;
     313         120 :         struct cli_posix_symlink_state *state = NULL;
     314             : 
     315         120 :         req = tevent_req_create(
     316             :                 mem_ctx, &state, struct cli_posix_symlink_state);
     317         120 :         if (req == NULL) {
     318           0 :                 return NULL;
     319             :         }
     320             : 
     321         120 :         subreq = cli_posix_link_internal_send(
     322             :                 mem_ctx, ev, cli, SMB_SET_FILE_UNIX_LINK, link_target, newname);
     323         120 :         if (tevent_req_nomem(subreq, req)) {
     324           0 :                 return tevent_req_post(req, ev);
     325             :         }
     326         120 :         tevent_req_set_callback(subreq, cli_posix_symlink_done, req);
     327         120 :         return req;
     328             : }
     329             : 
     330         120 : static void cli_posix_symlink_done(struct tevent_req *subreq)
     331             : {
     332         120 :         NTSTATUS status = cli_posix_link_internal_recv(subreq);
     333         120 :         tevent_req_simple_finish_ntstatus(subreq, status);
     334         120 : }
     335             : 
     336         120 : NTSTATUS cli_posix_symlink_recv(struct tevent_req *req)
     337             : {
     338         120 :         return tevent_req_simple_recv_ntstatus(req);
     339             : }
     340             : 
     341         110 : NTSTATUS cli_posix_symlink(struct cli_state *cli,
     342             :                         const char *link_target,
     343             :                         const char *newname)
     344             : {
     345         110 :         TALLOC_CTX *frame = talloc_stackframe();
     346         110 :         struct tevent_context *ev = NULL;
     347         110 :         struct tevent_req *req = NULL;
     348         110 :         NTSTATUS status = NT_STATUS_OK;
     349             : 
     350         110 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
     351             :                 /*
     352             :                  * Can't use sync call while an async call is in flight
     353             :                  */
     354           0 :                 status = NT_STATUS_INVALID_PARAMETER;
     355           0 :                 goto fail;
     356             :         }
     357             : 
     358         110 :         ev = samba_tevent_context_init(frame);
     359         110 :         if (ev == NULL) {
     360           0 :                 status = NT_STATUS_NO_MEMORY;
     361           0 :                 goto fail;
     362             :         }
     363             : 
     364         110 :         req = cli_posix_symlink_send(frame,
     365             :                                 ev,
     366             :                                 cli,
     367             :                                 link_target,
     368             :                                 newname);
     369         110 :         if (req == NULL) {
     370           0 :                 status = NT_STATUS_NO_MEMORY;
     371           0 :                 goto fail;
     372             :         }
     373             : 
     374         110 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
     375           0 :                 goto fail;
     376             :         }
     377             : 
     378         110 :         status = cli_posix_symlink_recv(req);
     379             : 
     380         110 :  fail:
     381         110 :         TALLOC_FREE(frame);
     382         110 :         return status;
     383             : }
     384             : 
     385             : /****************************************************************************
     386             :  Read a POSIX symlink.
     387             : ****************************************************************************/
     388             : 
     389             : struct cli_posix_readlink_state {
     390             :         struct cli_state *cli;
     391             :         char *converted;
     392             : };
     393             : 
     394             : static void cli_posix_readlink_done(struct tevent_req *subreq);
     395             : 
     396          16 : struct tevent_req *cli_posix_readlink_send(TALLOC_CTX *mem_ctx,
     397             :                                         struct tevent_context *ev,
     398             :                                         struct cli_state *cli,
     399             :                                         const char *fname)
     400             : {
     401          16 :         struct tevent_req *req = NULL, *subreq = NULL;
     402          16 :         struct cli_posix_readlink_state *state = NULL;
     403             : 
     404          16 :         req = tevent_req_create(
     405             :                 mem_ctx, &state, struct cli_posix_readlink_state);
     406          16 :         if (req == NULL) {
     407           0 :                 return NULL;
     408             :         }
     409          16 :         state->cli = cli;
     410             : 
     411          16 :         subreq = cli_qpathinfo_send(
     412             :                 state,
     413             :                 ev,
     414             :                 cli,
     415             :                 fname,
     416             :                 SMB_QUERY_FILE_UNIX_LINK,
     417             :                 1,
     418             :                 UINT16_MAX);
     419          16 :         if (tevent_req_nomem(subreq, req)) {
     420           0 :                 return tevent_req_post(req, ev);
     421             :         }
     422          16 :         tevent_req_set_callback(subreq, cli_posix_readlink_done, req);
     423          16 :         return req;
     424             : }
     425             : 
     426          16 : static void cli_posix_readlink_done(struct tevent_req *subreq)
     427             : {
     428          16 :         struct tevent_req *req = tevent_req_callback_data(
     429             :                 subreq, struct tevent_req);
     430          16 :         struct cli_posix_readlink_state *state = tevent_req_data(
     431             :                 req, struct cli_posix_readlink_state);
     432           0 :         NTSTATUS status;
     433          16 :         uint8_t *data = NULL;
     434          16 :         uint32_t num_data = 0;
     435           0 :         charset_t charset;
     436           0 :         size_t converted_size;
     437           0 :         bool ok;
     438             : 
     439          16 :         status = cli_qpathinfo_recv(subreq, state, &data, &num_data);
     440          16 :         TALLOC_FREE(subreq);
     441          16 :         if (tevent_req_nterror(req, status)) {
     442           0 :                 return;
     443             :         }
     444             :         /*
     445             :          * num_data is > 1, we've given 1 as minimum to cli_qpathinfo_send
     446             :          */
     447          16 :         if (data == NULL || data[num_data-1] != '\0') {
     448           0 :                 tevent_req_nterror(req, NT_STATUS_DATA_ERROR);
     449           0 :                 return;
     450             :         }
     451             : 
     452          16 :         charset = smbXcli_conn_use_unicode(state->cli->conn) ?
     453          16 :                 CH_UTF16LE : CH_DOS;
     454             : 
     455             :         /* The returned data is a pushed string, not raw data. */
     456          16 :         ok = convert_string_talloc(
     457             :                 state,
     458             :                 charset,
     459             :                 CH_UNIX,
     460             :                 data,
     461             :                 num_data,
     462          16 :                 &state->converted,
     463             :                 &converted_size);
     464          16 :         if (!ok) {
     465           0 :                 tevent_req_oom(req);
     466           0 :                 return;
     467             :         }
     468          16 :         tevent_req_done(req);
     469             : }
     470             : 
     471          16 : NTSTATUS cli_posix_readlink_recv(
     472             :         struct tevent_req *req, TALLOC_CTX *mem_ctx, char **target)
     473             : {
     474          16 :         struct cli_posix_readlink_state *state = tevent_req_data(
     475             :                 req, struct cli_posix_readlink_state);
     476           0 :         NTSTATUS status;
     477             : 
     478          16 :         if (tevent_req_is_nterror(req, &status)) {
     479           0 :                 return status;
     480             :         }
     481          16 :         *target = talloc_move(mem_ctx, &state->converted);
     482          16 :         tevent_req_received(req);
     483          16 :         return NT_STATUS_OK;
     484             : }
     485             : 
     486             : /****************************************************************************
     487             :  Hard link a file (UNIX extensions).
     488             : ****************************************************************************/
     489             : 
     490             : struct cli_posix_hardlink_state {
     491             :         uint8_t dummy;
     492             : };
     493             : 
     494             : static void cli_posix_hardlink_done(struct tevent_req *subreq);
     495             : 
     496           8 : struct tevent_req *cli_posix_hardlink_send(TALLOC_CTX *mem_ctx,
     497             :                                         struct tevent_context *ev,
     498             :                                         struct cli_state *cli,
     499             :                                         const char *oldname,
     500             :                                         const char *newname)
     501             : {
     502           8 :         struct tevent_req *req = NULL, *subreq = NULL;
     503           8 :         struct cli_posix_hardlink_state *state = NULL;
     504             : 
     505           8 :         req = tevent_req_create(
     506             :                 mem_ctx, &state, struct cli_posix_hardlink_state);
     507           8 :         if (req == NULL) {
     508           0 :                 return NULL;
     509             :         }
     510             : 
     511           8 :         subreq = cli_posix_link_internal_send(
     512             :                 state, ev, cli, SMB_SET_FILE_UNIX_HLINK, oldname, newname);
     513           8 :         if (tevent_req_nomem(subreq, req)) {
     514           0 :                 return tevent_req_post(req, ev);
     515             :         }
     516           8 :         tevent_req_set_callback(subreq, cli_posix_hardlink_done, req);
     517           8 :         return req;
     518             : }
     519             : 
     520           8 : static void cli_posix_hardlink_done(struct tevent_req *subreq)
     521             : {
     522           8 :         NTSTATUS status = cli_posix_link_internal_recv(subreq);
     523           8 :         tevent_req_simple_finish_ntstatus(subreq, status);
     524           8 : }
     525             : 
     526           8 : NTSTATUS cli_posix_hardlink_recv(struct tevent_req *req)
     527             : {
     528           8 :         return tevent_req_simple_recv_ntstatus(req);
     529             : }
     530             : 
     531           8 : NTSTATUS cli_posix_hardlink(struct cli_state *cli,
     532             :                         const char *oldname,
     533             :                         const char *newname)
     534             : {
     535           8 :         TALLOC_CTX *frame = talloc_stackframe();
     536           8 :         struct tevent_context *ev = NULL;
     537           8 :         struct tevent_req *req = NULL;
     538           8 :         NTSTATUS status = NT_STATUS_OK;
     539             : 
     540           8 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
     541             :                 /*
     542             :                  * Can't use sync call while an async call is in flight
     543             :                  */
     544           0 :                 status = NT_STATUS_INVALID_PARAMETER;
     545           0 :                 goto fail;
     546             :         }
     547             : 
     548           8 :         ev = samba_tevent_context_init(frame);
     549           8 :         if (ev == NULL) {
     550           0 :                 status = NT_STATUS_NO_MEMORY;
     551           0 :                 goto fail;
     552             :         }
     553             : 
     554           8 :         req = cli_posix_hardlink_send(frame,
     555             :                                 ev,
     556             :                                 cli,
     557             :                                 oldname,
     558             :                                 newname);
     559           8 :         if (req == NULL) {
     560           0 :                 status = NT_STATUS_NO_MEMORY;
     561           0 :                 goto fail;
     562             :         }
     563             : 
     564           8 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
     565           0 :                 goto fail;
     566             :         }
     567             : 
     568           8 :         status = cli_posix_hardlink_recv(req);
     569             : 
     570           8 :  fail:
     571           8 :         TALLOC_FREE(frame);
     572           8 :         return status;
     573             : }
     574             : 
     575             : /****************************************************************************
     576             :  Do a POSIX getacl - pathname based ACL get (UNIX extensions).
     577             : ****************************************************************************/
     578             : 
     579             : struct getacl_state {
     580             :         uint32_t num_data;
     581             :         uint8_t *data;
     582             : };
     583             : 
     584             : static void cli_posix_getacl_done(struct tevent_req *subreq);
     585             : 
     586          68 : struct tevent_req *cli_posix_getacl_send(TALLOC_CTX *mem_ctx,
     587             :                                         struct tevent_context *ev,
     588             :                                         struct cli_state *cli,
     589             :                                         const char *fname)
     590             : {
     591          68 :         struct tevent_req *req = NULL, *subreq = NULL;
     592          68 :         struct getacl_state *state = NULL;
     593             : 
     594          68 :         req = tevent_req_create(mem_ctx, &state, struct getacl_state);
     595          68 :         if (req == NULL) {
     596           0 :                 return NULL;
     597             :         }
     598          68 :         subreq = cli_qpathinfo_send(state, ev, cli, fname, SMB_QUERY_POSIX_ACL,
     599             :                                     0, CLI_BUFFER_SIZE);
     600          68 :         if (tevent_req_nomem(subreq, req)) {
     601           0 :                 return tevent_req_post(req, ev);
     602             :         }
     603          68 :         tevent_req_set_callback(subreq, cli_posix_getacl_done, req);
     604          68 :         return req;
     605             : }
     606             : 
     607          68 : static void cli_posix_getacl_done(struct tevent_req *subreq)
     608             : {
     609          68 :         struct tevent_req *req = tevent_req_callback_data(
     610             :                 subreq, struct tevent_req);
     611          68 :         struct getacl_state *state = tevent_req_data(
     612             :                 req, struct getacl_state);
     613           0 :         NTSTATUS status;
     614             : 
     615          68 :         status = cli_qpathinfo_recv(subreq, state, &state->data,
     616             :                                     &state->num_data);
     617          68 :         TALLOC_FREE(subreq);
     618          68 :         if (tevent_req_nterror(req, status)) {
     619           4 :                 return;
     620             :         }
     621          64 :         tevent_req_done(req);
     622             : }
     623             : 
     624          68 : NTSTATUS cli_posix_getacl_recv(struct tevent_req *req,
     625             :                                 TALLOC_CTX *mem_ctx,
     626             :                                 size_t *prb_size,
     627             :                                 char **retbuf)
     628             : {
     629          68 :         struct getacl_state *state = tevent_req_data(req, struct getacl_state);
     630           0 :         NTSTATUS status;
     631             : 
     632          68 :         if (tevent_req_is_nterror(req, &status)) {
     633           4 :                 return status;
     634             :         }
     635          64 :         *prb_size = (size_t)state->num_data;
     636          64 :         *retbuf = (char *)talloc_move(mem_ctx, &state->data);
     637          64 :         return NT_STATUS_OK;
     638             : }
     639             : 
     640          64 : NTSTATUS cli_posix_getacl(struct cli_state *cli,
     641             :                         const char *fname,
     642             :                         TALLOC_CTX *mem_ctx,
     643             :                         size_t *prb_size,
     644             :                         char **retbuf)
     645             : {
     646          64 :         TALLOC_CTX *frame = talloc_stackframe();
     647          64 :         struct tevent_context *ev = NULL;
     648          64 :         struct tevent_req *req = NULL;
     649          64 :         NTSTATUS status = NT_STATUS_OK;
     650             : 
     651          64 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
     652             :                 /*
     653             :                  * Can't use sync call while an async call is in flight
     654             :                  */
     655           0 :                 status = NT_STATUS_INVALID_PARAMETER;
     656           0 :                 goto fail;
     657             :         }
     658             : 
     659          64 :         ev = samba_tevent_context_init(frame);
     660          64 :         if (ev == NULL) {
     661           0 :                 status = NT_STATUS_NO_MEMORY;
     662           0 :                 goto fail;
     663             :         }
     664             : 
     665          64 :         req = cli_posix_getacl_send(frame,
     666             :                                 ev,
     667             :                                 cli,
     668             :                                 fname);
     669          64 :         if (req == NULL) {
     670           0 :                 status = NT_STATUS_NO_MEMORY;
     671           0 :                 goto fail;
     672             :         }
     673             : 
     674          64 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
     675           0 :                 goto fail;
     676             :         }
     677             : 
     678          64 :         status = cli_posix_getacl_recv(req, mem_ctx, prb_size, retbuf);
     679             : 
     680          64 :  fail:
     681          64 :         TALLOC_FREE(frame);
     682          64 :         return status;
     683             : }
     684             : 
     685             : /****************************************************************************
     686             :  Do a POSIX setacl - pathname based ACL set (UNIX extensions).
     687             : ****************************************************************************/
     688             : 
     689             : struct setacl_state {
     690             :         uint8_t *data;
     691             : };
     692             : 
     693             : static void cli_posix_setacl_done(struct tevent_req *subreq);
     694             : 
     695           8 : struct tevent_req *cli_posix_setacl_send(TALLOC_CTX *mem_ctx,
     696             :                                         struct tevent_context *ev,
     697             :                                         struct cli_state *cli,
     698             :                                         const char *fname,
     699             :                                         const void *data,
     700             :                                         size_t num_data)
     701             : {
     702           8 :         struct tevent_req *req = NULL, *subreq = NULL;
     703           8 :         struct setacl_state *state = NULL;
     704             : 
     705           8 :         req = tevent_req_create(mem_ctx, &state, struct setacl_state);
     706           8 :         if (req == NULL) {
     707           0 :                 return NULL;
     708             :         }
     709           8 :         state->data = talloc_memdup(state, data, num_data);
     710           8 :         if (tevent_req_nomem(state->data, req)) {
     711           0 :                 return tevent_req_post(req, ev);
     712             :         }
     713             : 
     714           8 :         subreq = cli_setpathinfo_send(state,
     715             :                                 ev,
     716             :                                 cli,
     717             :                                 SMB_SET_POSIX_ACL,
     718             :                                 fname,
     719           8 :                                 state->data,
     720             :                                 num_data);
     721           8 :         if (tevent_req_nomem(subreq, req)) {
     722           0 :                 return tevent_req_post(req, ev);
     723             :         }
     724           8 :         tevent_req_set_callback(subreq, cli_posix_setacl_done, req);
     725           8 :         return req;
     726             : }
     727             : 
     728           8 : static void cli_posix_setacl_done(struct tevent_req *subreq)
     729             : {
     730           8 :         NTSTATUS status = cli_setpathinfo_recv(subreq);
     731           8 :         tevent_req_simple_finish_ntstatus(subreq, status);
     732           8 : }
     733             : 
     734           8 : NTSTATUS cli_posix_setacl_recv(struct tevent_req *req)
     735             : {
     736           8 :         return tevent_req_simple_recv_ntstatus(req);
     737             : }
     738             : 
     739           8 : NTSTATUS cli_posix_setacl(struct cli_state *cli,
     740             :                         const char *fname,
     741             :                         const void *acl_buf,
     742             :                         size_t acl_buf_size)
     743             : {
     744           8 :         TALLOC_CTX *frame = talloc_stackframe();
     745           8 :         struct tevent_context *ev = NULL;
     746           8 :         struct tevent_req *req = NULL;
     747           8 :         NTSTATUS status = NT_STATUS_OK;
     748             : 
     749           8 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
     750             :                 /*
     751             :                  * Can't use sync call while an async call is in flight
     752             :                  */
     753           0 :                 status = NT_STATUS_INVALID_PARAMETER;
     754           0 :                 goto fail;
     755             :         }
     756             : 
     757           8 :         ev = samba_tevent_context_init(frame);
     758           8 :         if (ev == NULL) {
     759           0 :                 status = NT_STATUS_NO_MEMORY;
     760           0 :                 goto fail;
     761             :         }
     762             : 
     763           8 :         req = cli_posix_setacl_send(frame,
     764             :                                 ev,
     765             :                                 cli,
     766             :                                 fname,
     767             :                                 acl_buf,
     768             :                                 acl_buf_size);
     769           8 :         if (req == NULL) {
     770           0 :                 status = NT_STATUS_NO_MEMORY;
     771           0 :                 goto fail;
     772             :         }
     773             : 
     774           8 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
     775           0 :                 goto fail;
     776             :         }
     777             : 
     778           8 :         status = cli_posix_setacl_recv(req);
     779             : 
     780           8 :  fail:
     781           8 :         TALLOC_FREE(frame);
     782           8 :         return status;
     783             : }
     784             : 
     785             : /****************************************************************************
     786             :  Stat a file (UNIX extensions).
     787             : ****************************************************************************/
     788             : 
     789             : struct cli_posix_stat_state {
     790             :         struct stat_ex sbuf;
     791             : };
     792             : 
     793             : static void cli_posix_stat_done(struct tevent_req *subreq);
     794             : 
     795         106 : struct tevent_req *cli_posix_stat_send(TALLOC_CTX *mem_ctx,
     796             :                                        struct tevent_context *ev,
     797             :                                        struct cli_state *cli,
     798             :                                        const char *fname)
     799             : {
     800         106 :         struct tevent_req *req = NULL, *subreq = NULL;
     801         106 :         struct stat_state *state = NULL;
     802             : 
     803         106 :         req = tevent_req_create(mem_ctx, &state, struct cli_posix_stat_state);
     804         106 :         if (req == NULL) {
     805           0 :                 return NULL;
     806             :         }
     807             : 
     808         106 :         subreq = cli_qpathinfo_send(state, ev, cli, fname,
     809             :                                     SMB_QUERY_FILE_UNIX_BASIC, 100, 100);
     810         106 :         if (tevent_req_nomem(subreq, req)) {
     811           0 :                 return tevent_req_post(req, ev);
     812             :         }
     813         106 :         tevent_req_set_callback(subreq, cli_posix_stat_done, req);
     814         106 :         return req;
     815             : }
     816             : 
     817         106 : static void cli_posix_stat_done(struct tevent_req *subreq)
     818             : {
     819         106 :         struct tevent_req *req = tevent_req_callback_data(
     820             :                                 subreq, struct tevent_req);
     821         106 :         struct cli_posix_stat_state *state = tevent_req_data(
     822             :                 req, struct cli_posix_stat_state);
     823         106 :         SMB_STRUCT_STAT *sbuf = &state->sbuf;
     824           0 :         uint8_t *data;
     825         106 :         uint32_t num_data = 0;
     826           0 :         NTSTATUS status;
     827             : 
     828         106 :         status = cli_qpathinfo_recv(subreq, state, &data, &num_data);
     829         106 :         TALLOC_FREE(subreq);
     830         106 :         if (tevent_req_nterror(req, status)) {
     831           0 :                 return;
     832             :         }
     833             : 
     834         106 :         if (num_data != 100) {
     835             :                 /*
     836             :                  * Paranoia, cli_qpathinfo should have guaranteed
     837             :                  * this, but you never know...
     838             :                  */
     839           0 :                 tevent_req_nterror(req, NT_STATUS_INVALID_NETWORK_RESPONSE);
     840           0 :                 return;
     841             :         }
     842             : 
     843             :         /* total size, in bytes */
     844         106 :         sbuf->st_ex_size = IVAL2_TO_SMB_BIG_UINT(data, 0);
     845             : 
     846             :         /* number of blocks allocated */
     847         106 :         sbuf->st_ex_blocks = IVAL2_TO_SMB_BIG_UINT(data,8);
     848             : #if defined (HAVE_STAT_ST_BLOCKS) && defined(STAT_ST_BLOCKSIZE)
     849         106 :         sbuf->st_ex_blocks /= STAT_ST_BLOCKSIZE;
     850             : #else
     851             :         /* assume 512 byte blocks */
     852             :         sbuf->st_ex_blocks /= 512;
     853             : #endif
     854             :         /* time of last change */
     855         106 :         sbuf->st_ex_ctime = interpret_long_date(BVAL(data, 16));
     856             : 
     857             :         /* time of last access */
     858         106 :         sbuf->st_ex_atime = interpret_long_date(BVAL(data, 24));
     859             : 
     860             :         /* time of last modification */
     861         106 :         sbuf->st_ex_mtime = interpret_long_date(BVAL(data, 32));
     862             : 
     863         106 :         sbuf->st_ex_uid = (uid_t) IVAL(data, 40); /* user ID of owner */
     864         106 :         sbuf->st_ex_gid = (gid_t) IVAL(data, 48); /* group ID of owner */
     865         106 :         sbuf->st_ex_mode = unix_filetype_from_wire(IVAL(data, 56));
     866             : 
     867             : #if defined(HAVE_MAKEDEV)
     868             :         {
     869         106 :                 uint32_t dev_major = IVAL(data,60);
     870         106 :                 uint32_t dev_minor = IVAL(data,68);
     871         106 :                 sbuf->st_ex_rdev = makedev(dev_major, dev_minor);
     872             :         }
     873             : #endif
     874             :         /* inode */
     875         106 :         sbuf->st_ex_ino = (SMB_INO_T)IVAL2_TO_SMB_BIG_UINT(data, 76);
     876             : 
     877             :         /* protection */
     878         106 :         sbuf->st_ex_mode |= wire_perms_to_unix(IVAL(data, 84));
     879             : 
     880             :         /* number of hard links */
     881         106 :         sbuf->st_ex_nlink = BIG_UINT(data, 92);
     882             : 
     883         106 :         tevent_req_done(req);
     884             : }
     885             : 
     886         106 : NTSTATUS cli_posix_stat_recv(struct tevent_req *req, struct stat_ex *sbuf)
     887             : {
     888         106 :         struct cli_posix_stat_state *state = tevent_req_data(
     889             :                 req, struct cli_posix_stat_state);
     890           0 :         NTSTATUS status;
     891             : 
     892         106 :         if (tevent_req_is_nterror(req, &status)) {
     893           0 :                 return status;
     894             :         }
     895         106 :         *sbuf = state->sbuf;
     896         106 :         return NT_STATUS_OK;
     897             : }
     898             : 
     899         104 : NTSTATUS cli_posix_stat(struct cli_state *cli,
     900             :                         const char *fname,
     901             :                         struct stat_ex *sbuf)
     902             : {
     903         104 :         TALLOC_CTX *frame = talloc_stackframe();
     904         104 :         struct tevent_context *ev = NULL;
     905         104 :         struct tevent_req *req = NULL;
     906         104 :         NTSTATUS status = NT_STATUS_OK;
     907             : 
     908         104 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
     909             :                 /*
     910             :                  * Can't use sync call while an async call is in flight
     911             :                  */
     912           0 :                 status = NT_STATUS_INVALID_PARAMETER;
     913           0 :                 goto fail;
     914             :         }
     915             : 
     916         104 :         ev = samba_tevent_context_init(frame);
     917         104 :         if (ev == NULL) {
     918           0 :                 status = NT_STATUS_NO_MEMORY;
     919           0 :                 goto fail;
     920             :         }
     921             : 
     922         104 :         req = cli_posix_stat_send(frame, ev, cli, fname);
     923         104 :         if (req == NULL) {
     924           0 :                 status = NT_STATUS_NO_MEMORY;
     925           0 :                 goto fail;
     926             :         }
     927             : 
     928         104 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
     929           0 :                 goto fail;
     930             :         }
     931             : 
     932         104 :         status = cli_posix_stat_recv(req, sbuf);
     933             : 
     934         104 :  fail:
     935         104 :         TALLOC_FREE(frame);
     936         104 :         return status;
     937             : }
     938             : 
     939             : /****************************************************************************
     940             :  Chmod or chown a file internal (UNIX extensions).
     941             : ****************************************************************************/
     942             : 
     943             : struct cli_posix_chown_chmod_internal_state {
     944             :         uint8_t data[100];
     945             : };
     946             : 
     947             : static void cli_posix_chown_chmod_internal_done(struct tevent_req *subreq);
     948             : 
     949          12 : static struct tevent_req *cli_posix_chown_chmod_internal_send(TALLOC_CTX *mem_ctx,
     950             :                                         struct tevent_context *ev,
     951             :                                         struct cli_state *cli,
     952             :                                         const char *fname,
     953             :                                         uint32_t mode,
     954             :                                         uint32_t uid,
     955             :                                         uint32_t gid)
     956             : {
     957          12 :         struct tevent_req *req = NULL, *subreq = NULL;
     958          12 :         struct cli_posix_chown_chmod_internal_state *state = NULL;
     959             : 
     960          12 :         req = tevent_req_create(mem_ctx, &state,
     961             :                                 struct cli_posix_chown_chmod_internal_state);
     962          12 :         if (req == NULL) {
     963           0 :                 return NULL;
     964             :         }
     965             : 
     966          12 :         memset(state->data, 0xff, 40); /* Set all sizes/times to no change. */
     967          12 :         SIVAL(state->data,40,uid);
     968          12 :         SIVAL(state->data,48,gid);
     969          12 :         SIVAL(state->data,84,mode);
     970             : 
     971          12 :         subreq = cli_setpathinfo_send(state, ev, cli, SMB_SET_FILE_UNIX_BASIC,
     972          12 :                                       fname, state->data, sizeof(state->data));
     973          12 :         if (tevent_req_nomem(subreq, req)) {
     974           0 :                 return tevent_req_post(req, ev);
     975             :         }
     976          12 :         tevent_req_set_callback(subreq, cli_posix_chown_chmod_internal_done,
     977             :                                 req);
     978          12 :         return req;
     979             : }
     980             : 
     981          12 : static void cli_posix_chown_chmod_internal_done(struct tevent_req *subreq)
     982             : {
     983          12 :         NTSTATUS status = cli_setpathinfo_recv(subreq);
     984          12 :         tevent_req_simple_finish_ntstatus(subreq, status);
     985          12 : }
     986             : 
     987          12 : static NTSTATUS cli_posix_chown_chmod_internal_recv(struct tevent_req *req)
     988             : {
     989          12 :         return tevent_req_simple_recv_ntstatus(req);
     990             : }
     991             : 
     992             : /****************************************************************************
     993             :  chmod a file (UNIX extensions).
     994             : ****************************************************************************/
     995             : 
     996             : struct cli_posix_chmod_state {
     997             :         uint8_t dummy;
     998             : };
     999             : 
    1000             : static void cli_posix_chmod_done(struct tevent_req *subreq);
    1001             : 
    1002          12 : struct tevent_req *cli_posix_chmod_send(TALLOC_CTX *mem_ctx,
    1003             :                                         struct tevent_context *ev,
    1004             :                                         struct cli_state *cli,
    1005             :                                         const char *fname,
    1006             :                                         mode_t mode)
    1007             : {
    1008          12 :         struct tevent_req *req = NULL, *subreq = NULL;
    1009          12 :         struct cli_posix_chmod_state *state = NULL;
    1010             : 
    1011          12 :         req = tevent_req_create(mem_ctx, &state, struct cli_posix_chmod_state);
    1012          12 :         if (req == NULL) {
    1013           0 :                 return NULL;
    1014             :         }
    1015             : 
    1016          12 :         subreq = cli_posix_chown_chmod_internal_send(
    1017             :                 state,
    1018             :                 ev,
    1019             :                 cli,
    1020             :                 fname,
    1021             :                 unix_perms_to_wire(mode),
    1022             :                 SMB_UID_NO_CHANGE,
    1023             :                 SMB_GID_NO_CHANGE);
    1024          12 :         if (tevent_req_nomem(subreq, req)) {
    1025           0 :                 return tevent_req_post(req, ev);
    1026             :         }
    1027          12 :         tevent_req_set_callback(subreq, cli_posix_chmod_done, req);
    1028          12 :         return req;
    1029             : }
    1030             : 
    1031          12 : static void cli_posix_chmod_done(struct tevent_req *subreq)
    1032             : {
    1033          12 :         NTSTATUS status = cli_posix_chown_chmod_internal_recv(subreq);
    1034          12 :         tevent_req_simple_finish_ntstatus(subreq, status);
    1035          12 : }
    1036             : 
    1037          12 : NTSTATUS cli_posix_chmod_recv(struct tevent_req *req)
    1038             : {
    1039          12 :         return tevent_req_simple_recv_ntstatus(req);
    1040             : }
    1041             : 
    1042          12 : NTSTATUS cli_posix_chmod(struct cli_state *cli, const char *fname, mode_t mode)
    1043             : {
    1044          12 :         TALLOC_CTX *frame = talloc_stackframe();
    1045          12 :         struct tevent_context *ev = NULL;
    1046          12 :         struct tevent_req *req = NULL;
    1047          12 :         NTSTATUS status = NT_STATUS_OK;
    1048             : 
    1049          12 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    1050             :                 /*
    1051             :                  * Can't use sync call while an async call is in flight
    1052             :                  */
    1053           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    1054           0 :                 goto fail;
    1055             :         }
    1056             : 
    1057          12 :         ev = samba_tevent_context_init(frame);
    1058          12 :         if (ev == NULL) {
    1059           0 :                 status = NT_STATUS_NO_MEMORY;
    1060           0 :                 goto fail;
    1061             :         }
    1062             : 
    1063          12 :         req = cli_posix_chmod_send(frame,
    1064             :                                 ev,
    1065             :                                 cli,
    1066             :                                 fname,
    1067             :                                 mode);
    1068          12 :         if (req == NULL) {
    1069           0 :                 status = NT_STATUS_NO_MEMORY;
    1070           0 :                 goto fail;
    1071             :         }
    1072             : 
    1073          12 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    1074           0 :                 goto fail;
    1075             :         }
    1076             : 
    1077          12 :         status = cli_posix_chmod_recv(req);
    1078             : 
    1079          12 :  fail:
    1080          12 :         TALLOC_FREE(frame);
    1081          12 :         return status;
    1082             : }
    1083             : 
    1084             : /****************************************************************************
    1085             :  chown a file (UNIX extensions).
    1086             : ****************************************************************************/
    1087             : 
    1088             : struct cli_posix_chown_state {
    1089             :         uint8_t dummy;
    1090             : };
    1091             : 
    1092             : static void cli_posix_chown_done(struct tevent_req *subreq);
    1093             : 
    1094           0 : struct tevent_req *cli_posix_chown_send(TALLOC_CTX *mem_ctx,
    1095             :                                         struct tevent_context *ev,
    1096             :                                         struct cli_state *cli,
    1097             :                                         const char *fname,
    1098             :                                         uid_t uid,
    1099             :                                         gid_t gid)
    1100             : {
    1101           0 :         struct tevent_req *req = NULL, *subreq = NULL;
    1102           0 :         struct cli_posix_chown_state *state = NULL;
    1103             : 
    1104           0 :         req = tevent_req_create(
    1105             :                 mem_ctx, &state, struct cli_posix_chown_state);
    1106           0 :         if (req == NULL) {
    1107           0 :                 return NULL;
    1108             :         }
    1109             : 
    1110           0 :         subreq = cli_posix_chown_chmod_internal_send(
    1111             :                 state,
    1112             :                 ev,
    1113             :                 cli,
    1114             :                 fname,
    1115             :                 SMB_MODE_NO_CHANGE,
    1116             :                 (uint32_t)uid,
    1117             :                 (uint32_t)gid);
    1118           0 :         if (tevent_req_nomem(subreq, req)) {
    1119           0 :                 return tevent_req_post(req, ev);
    1120             :         }
    1121           0 :         tevent_req_set_callback(subreq, cli_posix_chown_done, req);
    1122           0 :         return req;
    1123             : }
    1124             : 
    1125           0 : static void cli_posix_chown_done(struct tevent_req *subreq)
    1126             : {
    1127           0 :         NTSTATUS status = cli_posix_chown_chmod_internal_recv(subreq);
    1128           0 :         tevent_req_simple_finish_ntstatus(subreq, status);
    1129           0 : }
    1130             : 
    1131           0 : NTSTATUS cli_posix_chown_recv(struct tevent_req *req)
    1132             : {
    1133           0 :         return tevent_req_simple_recv_ntstatus(req);
    1134             : }
    1135             : 
    1136           0 : NTSTATUS cli_posix_chown(struct cli_state *cli,
    1137             :                         const char *fname,
    1138             :                         uid_t uid,
    1139             :                         gid_t gid)
    1140             : {
    1141           0 :         TALLOC_CTX *frame = talloc_stackframe();
    1142           0 :         struct tevent_context *ev = NULL;
    1143           0 :         struct tevent_req *req = NULL;
    1144           0 :         NTSTATUS status = NT_STATUS_OK;
    1145             : 
    1146           0 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    1147             :                 /*
    1148             :                  * Can't use sync call while an async call is in flight
    1149             :                  */
    1150           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    1151           0 :                 goto fail;
    1152             :         }
    1153             : 
    1154           0 :         ev = samba_tevent_context_init(frame);
    1155           0 :         if (ev == NULL) {
    1156           0 :                 status = NT_STATUS_NO_MEMORY;
    1157           0 :                 goto fail;
    1158             :         }
    1159             : 
    1160           0 :         req = cli_posix_chown_send(frame,
    1161             :                                 ev,
    1162             :                                 cli,
    1163             :                                 fname,
    1164             :                                 uid,
    1165             :                                 gid);
    1166           0 :         if (req == NULL) {
    1167           0 :                 status = NT_STATUS_NO_MEMORY;
    1168           0 :                 goto fail;
    1169             :         }
    1170             : 
    1171           0 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    1172           0 :                 goto fail;
    1173             :         }
    1174             : 
    1175           0 :         status = cli_posix_chown_recv(req);
    1176             : 
    1177           0 :  fail:
    1178           0 :         TALLOC_FREE(frame);
    1179           0 :         return status;
    1180             : }
    1181             : 
    1182             : struct cli_smb1_posix_mknod_state {
    1183             :         uint8_t data[100];
    1184             : };
    1185             : 
    1186             : static void cli_smb1_posix_mknod_done(struct tevent_req *subreq);
    1187             : 
    1188           2 : static struct tevent_req *cli_smb1_posix_mknod_send(
    1189             :         TALLOC_CTX *mem_ctx,
    1190             :         struct tevent_context *ev,
    1191             :         struct cli_state *cli,
    1192             :         const char *fname,
    1193             :         mode_t mode,
    1194             :         dev_t dev)
    1195             : {
    1196           2 :         struct tevent_req *req = NULL, *subreq = NULL;
    1197           2 :         struct cli_smb1_posix_mknod_state *state = NULL;
    1198           2 :         mode_t type = mode & S_IFMT;
    1199           2 :         uint32_t smb_unix_type = 0xFFFFFFFF;
    1200             : 
    1201           2 :         req = tevent_req_create(
    1202             :                 mem_ctx, &state, struct cli_smb1_posix_mknod_state);
    1203           2 :         if (req == NULL) {
    1204           0 :                 return NULL;
    1205             :         }
    1206             :         /*
    1207             :          * Set all sizes/times/ids to no change.
    1208             :          */
    1209           2 :         memset(state->data, 0xff, 56);
    1210             : 
    1211           2 :         switch (type) {
    1212           0 :         case S_IFREG:
    1213           0 :                 smb_unix_type = UNIX_TYPE_FILE;
    1214           0 :                 break;
    1215           0 :         case S_IFDIR:
    1216           0 :                 smb_unix_type = UNIX_TYPE_DIR;
    1217           0 :                 break;
    1218           0 :         case S_IFLNK:
    1219           0 :                 smb_unix_type = UNIX_TYPE_SYMLINK;
    1220           0 :                 break;
    1221           0 :         case S_IFCHR:
    1222           0 :                 smb_unix_type = UNIX_TYPE_CHARDEV;
    1223           0 :                 break;
    1224           0 :         case S_IFBLK:
    1225           0 :                 smb_unix_type = UNIX_TYPE_BLKDEV;
    1226           0 :                 break;
    1227           1 :         case S_IFIFO:
    1228           1 :                 smb_unix_type = UNIX_TYPE_FIFO;
    1229           1 :                 break;
    1230           1 :         case S_IFSOCK:
    1231           1 :                 smb_unix_type = UNIX_TYPE_SOCKET;
    1232           1 :                 break;
    1233             :         }
    1234           2 :         PUSH_LE_U32(state->data, 56, smb_unix_type);
    1235             : 
    1236           2 :         if ((type == S_IFCHR) || (type == S_IFBLK)) {
    1237           0 :                 PUSH_LE_U64(state->data, 60, unix_dev_major(dev));
    1238           0 :                 PUSH_LE_U64(state->data, 68, unix_dev_minor(dev));
    1239             :         }
    1240             : 
    1241           2 :         PUSH_LE_U32(state->data, 84, unix_perms_to_wire(mode));
    1242             : 
    1243           2 :         subreq = cli_setpathinfo_send(
    1244             :                 state,
    1245             :                 ev,
    1246             :                 cli,
    1247             :                 SMB_SET_FILE_UNIX_BASIC,
    1248             :                 fname,
    1249           2 :                 state->data,
    1250             :                 sizeof(state->data));
    1251           2 :         if (tevent_req_nomem(subreq, req)) {
    1252           0 :                 return tevent_req_post(req, ev);
    1253             :         }
    1254           2 :         tevent_req_set_callback(subreq, cli_smb1_posix_mknod_done, req);
    1255           2 :         return req;
    1256             : }
    1257             : 
    1258           2 : static void cli_smb1_posix_mknod_done(struct tevent_req *subreq)
    1259             : {
    1260           2 :         NTSTATUS status = cli_setpathinfo_recv(subreq);
    1261           2 :         tevent_req_simple_finish_ntstatus(subreq, status);
    1262           2 : }
    1263             : 
    1264           2 : static NTSTATUS cli_smb1_posix_mknod_recv(struct tevent_req *req)
    1265             : {
    1266           2 :         return tevent_req_simple_recv_ntstatus(req);
    1267             : }
    1268             : 
    1269             : struct cli_mknod_state {
    1270             :         uint8_t buf[24];
    1271             : };
    1272             : 
    1273             : static void cli_mknod_done1(struct tevent_req *subreq);
    1274             : static void cli_mknod_reparse_done(struct tevent_req *subreq);
    1275             : 
    1276           2 : struct tevent_req *cli_mknod_send(
    1277             :         TALLOC_CTX *mem_ctx,
    1278             :         struct tevent_context *ev,
    1279             :         struct cli_state *cli,
    1280             :         const char *fname,
    1281             :         mode_t mode,
    1282             :         dev_t dev)
    1283             : {
    1284           2 :         struct tevent_req *req = NULL, *subreq = NULL;
    1285           2 :         struct cli_mknod_state *state = NULL;
    1286           2 :         struct reparse_data_buffer reparse_buf = {
    1287             :                 .tag = IO_REPARSE_TAG_NFS,
    1288             :         };
    1289           2 :         struct nfs_reparse_data_buffer *nfs = &reparse_buf.parsed.nfs;
    1290           0 :         ssize_t buflen;
    1291             : 
    1292           2 :         req = tevent_req_create(mem_ctx, &state, struct cli_mknod_state);
    1293           2 :         if (req == NULL) {
    1294           0 :                 return NULL;
    1295             :         }
    1296             : 
    1297           2 :         if (cli->requested_posix_capabilities != 0) {
    1298           2 :                 subreq = cli_smb1_posix_mknod_send(
    1299             :                         state, ev, cli, fname, mode, dev);
    1300           2 :                 if (tevent_req_nomem(subreq, req)) {
    1301           0 :                         return tevent_req_post(req, ev);
    1302             :                 }
    1303           2 :                 tevent_req_set_callback(subreq, cli_mknod_done1, req);
    1304           2 :                 return req;
    1305             :         }
    1306             : 
    1307             :         /*
    1308             :          * Ignored for all but BLK and CHR
    1309             :          */
    1310           0 :         nfs->data.dev.major = major(dev);
    1311           0 :         nfs->data.dev.minor = minor(dev);
    1312             : 
    1313           0 :         switch (mode & S_IFMT) {
    1314           0 :         case S_IFIFO:
    1315           0 :                 nfs->type = NFS_SPECFILE_FIFO;
    1316           0 :                 break;
    1317           0 :         case S_IFSOCK:
    1318           0 :                 nfs->type = NFS_SPECFILE_SOCK;
    1319           0 :                 break;
    1320           0 :         case S_IFCHR:
    1321           0 :                 nfs->type = NFS_SPECFILE_CHR;
    1322           0 :                 break;
    1323           0 :         case S_IFBLK:
    1324           0 :                 nfs->type = NFS_SPECFILE_BLK;
    1325           0 :                 break;
    1326             :         }
    1327             : 
    1328           0 :         buflen = reparse_data_buffer_marshall(&reparse_buf,
    1329           0 :                                               state->buf,
    1330             :                                               sizeof(state->buf));
    1331           0 :         if ((buflen == -1) || (buflen > sizeof(state->buf))) {
    1332           0 :                 tevent_req_nterror(req, NT_STATUS_INTERNAL_ERROR);
    1333           0 :                 return tevent_req_post(req, ev);
    1334             :         }
    1335             : 
    1336           0 :         subreq = cli_create_reparse_point_send(state,
    1337             :                                                ev,
    1338             :                                                cli,
    1339             :                                                fname,
    1340           0 :                                                (DATA_BLOB){
    1341           0 :                                                        .data = state->buf,
    1342             :                                                        .length = buflen,
    1343             :                                                });
    1344           0 :         if (tevent_req_nomem(subreq, req)) {
    1345           0 :                 return tevent_req_post(req, ev);
    1346             :         }
    1347           0 :         tevent_req_set_callback(subreq, cli_mknod_reparse_done, req);
    1348           0 :         return req;
    1349             : }
    1350             : 
    1351           2 : static void cli_mknod_done1(struct tevent_req *subreq)
    1352             : {
    1353           2 :         NTSTATUS status = cli_smb1_posix_mknod_recv(subreq);
    1354           2 :         tevent_req_simple_finish_ntstatus(subreq, status);
    1355           2 : }
    1356             : 
    1357           0 : static void cli_mknod_reparse_done(struct tevent_req *subreq)
    1358             : {
    1359           0 :         NTSTATUS status = cli_create_reparse_point_recv(subreq);
    1360           0 :         tevent_req_simple_finish_ntstatus(subreq, status);
    1361           0 : }
    1362             : 
    1363           2 : NTSTATUS cli_mknod_recv(struct tevent_req *req)
    1364             : {
    1365           2 :         return tevent_req_simple_recv_ntstatus(req);
    1366             : }
    1367             : 
    1368             : NTSTATUS
    1369           0 : cli_mknod(struct cli_state *cli, const char *fname, mode_t mode, dev_t dev)
    1370             : {
    1371           0 :         TALLOC_CTX *frame = talloc_stackframe();
    1372           0 :         struct tevent_context *ev;
    1373           0 :         struct tevent_req *req;
    1374           0 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    1375             : 
    1376           0 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    1377             :                 /*
    1378             :                  * Can't use sync call while an async call is in flight
    1379             :                  */
    1380           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    1381           0 :                 goto fail;
    1382             :         }
    1383           0 :         ev = samba_tevent_context_init(frame);
    1384           0 :         if (ev == NULL) {
    1385           0 :                 goto fail;
    1386             :         }
    1387           0 :         req = cli_mknod_send(ev, ev, cli, fname, mode, dev);
    1388           0 :         if (req == NULL) {
    1389           0 :                 goto fail;
    1390             :         }
    1391           0 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    1392           0 :                 goto fail;
    1393             :         }
    1394           0 :         status = cli_mknod_recv(req);
    1395           0 : fail:
    1396           0 :         TALLOC_FREE(frame);
    1397           0 :         return status;
    1398             : }
    1399             : 
    1400             : /****************************************************************************
    1401             :  Rename a file.
    1402             : ****************************************************************************/
    1403             : 
    1404             : struct cli_smb1_rename_state {
    1405             :         uint8_t *data;
    1406             : };
    1407             : 
    1408             : static void cli_smb1_rename_done(struct tevent_req *subreq);
    1409             : 
    1410           6 : static struct tevent_req *cli_smb1_rename_send(TALLOC_CTX *mem_ctx,
    1411             :                                                struct tevent_context *ev,
    1412             :                                                struct cli_state *cli,
    1413             :                                                const char *fname_src,
    1414             :                                                const char *fname_dst,
    1415             :                                                bool replace)
    1416             : {
    1417           0 :         NTSTATUS status;
    1418           6 :         struct tevent_req *req = NULL, *subreq = NULL;
    1419           6 :         struct cli_smb1_rename_state *state = NULL;
    1420           6 :         smb_ucs2_t *converted_str = NULL;
    1421           6 :         size_t converted_size_bytes = 0;
    1422             : 
    1423           6 :         req = tevent_req_create(mem_ctx, &state, struct cli_smb1_rename_state);
    1424           6 :         if (req == NULL) {
    1425           0 :                 return NULL;
    1426             :         }
    1427             : 
    1428             :         /*
    1429             :          * Strip a MSDFS path from fname_dst if we were given one.
    1430             :          */
    1431           6 :         status = cli_dfs_target_check(state,
    1432             :                                 cli,
    1433             :                                 fname_dst,
    1434             :                                 &fname_dst);
    1435           6 :         if (!NT_STATUS_IS_OK(status)) {
    1436           0 :                 goto fail;
    1437             :         }
    1438             : 
    1439           6 :         if (!push_ucs2_talloc(talloc_tos(), &converted_str, fname_dst,
    1440             :                               &converted_size_bytes)) {
    1441           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    1442           0 :                 goto fail;
    1443             :         }
    1444             : 
    1445             :         /* W2K8 insists the dest name is not null
    1446             :            terminated. Remove the last 2 zero bytes
    1447             :            and reduce the name length. */
    1448             : 
    1449           6 :         if (converted_size_bytes < 2) {
    1450           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    1451           0 :                 goto fail;
    1452             :         }
    1453           6 :         converted_size_bytes -= 2;
    1454             : 
    1455          12 :         state->data =
    1456           6 :             talloc_zero_array(state, uint8_t, 12 + converted_size_bytes);
    1457           6 :         if (state->data == NULL) {
    1458           0 :                 status = NT_STATUS_NO_MEMORY;
    1459           0 :                 goto fail;
    1460             :         }
    1461             : 
    1462           6 :         if (replace) {
    1463           6 :                 SCVAL(state->data, 0, 1);
    1464             :         }
    1465             : 
    1466           6 :         SIVAL(state->data, 8, converted_size_bytes);
    1467           6 :         memcpy(state->data + 12, converted_str, converted_size_bytes);
    1468             : 
    1469           6 :         TALLOC_FREE(converted_str);
    1470             : 
    1471          12 :         subreq = cli_setpathinfo_send(
    1472           6 :             state, ev, cli, SMB_FILE_RENAME_INFORMATION, fname_src, state->data,
    1473           6 :             talloc_get_size(state->data));
    1474           6 :         if (tevent_req_nomem(subreq, req)) {
    1475           0 :                 status = NT_STATUS_NO_MEMORY;
    1476           0 :                 goto fail;
    1477             :         }
    1478           6 :         tevent_req_set_callback(subreq, cli_smb1_rename_done, req);
    1479           6 :         return req;
    1480             : 
    1481           0 : fail:
    1482           0 :         TALLOC_FREE(converted_str);
    1483           0 :         tevent_req_nterror(req, status);
    1484           0 :         return tevent_req_post(req, ev);
    1485             : }
    1486             : 
    1487           6 : static void cli_smb1_rename_done(struct tevent_req *subreq)
    1488             : {
    1489           6 :         NTSTATUS status = cli_setpathinfo_recv(subreq);
    1490           6 :         tevent_req_simple_finish_ntstatus(subreq, status);
    1491           6 : }
    1492             : 
    1493           6 : static NTSTATUS cli_smb1_rename_recv(struct tevent_req *req)
    1494             : {
    1495           6 :         return tevent_req_simple_recv_ntstatus(req);
    1496             : }
    1497             : 
    1498             : static void cli_cifs_rename_done(struct tevent_req *subreq);
    1499             : 
    1500             : struct cli_cifs_rename_state {
    1501             :         uint16_t vwv[1];
    1502             : };
    1503             : 
    1504         139 : static struct tevent_req *cli_cifs_rename_send(TALLOC_CTX *mem_ctx,
    1505             :                                                struct tevent_context *ev,
    1506             :                                                struct cli_state *cli,
    1507             :                                                const char *fname_src,
    1508             :                                                const char *fname_dst,
    1509             :                                                bool replace)
    1510             : {
    1511         139 :         struct tevent_req *req = NULL, *subreq = NULL;
    1512         139 :         struct cli_cifs_rename_state *state = NULL;
    1513         139 :         uint8_t additional_flags = 0;
    1514         139 :         uint16_t additional_flags2 = 0;
    1515         139 :         uint8_t *bytes = NULL;
    1516         139 :         char *fname_src_cp = NULL;
    1517         139 :         char *fname_dst_cp = NULL;
    1518             : 
    1519         139 :         req = tevent_req_create(mem_ctx, &state, struct cli_cifs_rename_state);
    1520         139 :         if (req == NULL) {
    1521           0 :                 return NULL;
    1522             :         }
    1523             : 
    1524         139 :         if (replace) {
    1525             :                 /*
    1526             :                  * CIFS doesn't support replace
    1527             :                  */
    1528           0 :                 tevent_req_nterror(req, NT_STATUS_INVALID_PARAMETER);
    1529           0 :                 return tevent_req_post(req, ev);
    1530             :         }
    1531             : 
    1532         139 :         SSVAL(state->vwv+0, 0, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_DIRECTORY);
    1533             : 
    1534         139 :         bytes = talloc_array(state, uint8_t, 1);
    1535         139 :         if (tevent_req_nomem(bytes, req)) {
    1536           0 :                 return tevent_req_post(req, ev);
    1537             :         }
    1538             : 
    1539             :         /*
    1540             :          * SMBmv on a DFS share uses DFS names for src and dst.
    1541             :          * See smbtorture3: SMB1-DFS-PATHS: test_smb1_mv().
    1542             :          */
    1543             : 
    1544         139 :         fname_src_cp = smb1_dfs_share_path(state, cli, fname_src);
    1545         139 :         if (tevent_req_nomem(fname_src_cp, req)) {
    1546           0 :                 return tevent_req_post(req, ev);
    1547             :         }
    1548         139 :         bytes[0] = 4;
    1549         139 :         bytes = smb_bytes_push_str(bytes,
    1550         139 :                                    smbXcli_conn_use_unicode(cli->conn),
    1551             :                                    fname_src_cp,
    1552         139 :                                    strlen(fname_src_cp)+1,
    1553             :                                    NULL);
    1554         139 :         if (tevent_req_nomem(bytes, req)) {
    1555           0 :                 return tevent_req_post(req, ev);
    1556             :         }
    1557             : 
    1558         139 :         if (clistr_is_previous_version_path(fname_src)) {
    1559           0 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
    1560             :         }
    1561             : 
    1562         139 :         bytes = talloc_realloc(state, bytes, uint8_t,
    1563             :                         talloc_get_size(bytes)+1);
    1564         139 :         if (tevent_req_nomem(bytes, req)) {
    1565           0 :                 return tevent_req_post(req, ev);
    1566             :         }
    1567             : 
    1568             :         /*
    1569             :          * SMBmv on a DFS share uses DFS names for src and dst.
    1570             :          * See smbtorture3: SMB1-DFS-PATHS: test_smb1_mv().
    1571             :          */
    1572             : 
    1573         139 :         fname_dst_cp = smb1_dfs_share_path(state, cli, fname_dst);
    1574         139 :         if (tevent_req_nomem(fname_dst_cp, req)) {
    1575           0 :                 return tevent_req_post(req, ev);
    1576             :         }
    1577         139 :         bytes[talloc_get_size(bytes)-1] = 4;
    1578         139 :         bytes = smb_bytes_push_str(bytes,
    1579         139 :                                    smbXcli_conn_use_unicode(cli->conn),
    1580             :                                    fname_dst_cp,
    1581         139 :                                    strlen(fname_dst_cp)+1,
    1582             :                                    NULL);
    1583         139 :         if (tevent_req_nomem(bytes, req)) {
    1584           0 :                 return tevent_req_post(req, ev);
    1585             :         }
    1586             : 
    1587         139 :         subreq = cli_smb_send(state, ev, cli, SMBmv, additional_flags,
    1588             :                         additional_flags2,
    1589         139 :                         1, state->vwv, talloc_get_size(bytes), bytes);
    1590         139 :         if (tevent_req_nomem(subreq, req)) {
    1591           0 :                 return tevent_req_post(req, ev);
    1592             :         }
    1593         139 :         tevent_req_set_callback(subreq, cli_cifs_rename_done, req);
    1594         139 :         return req;
    1595             : }
    1596             : 
    1597         139 : static void cli_cifs_rename_done(struct tevent_req *subreq)
    1598             : {
    1599         139 :         NTSTATUS status = cli_smb_recv(
    1600             :                 subreq, NULL, NULL, 0, NULL, NULL, NULL, NULL);
    1601         139 :         tevent_req_simple_finish_ntstatus(subreq, status);
    1602         139 : }
    1603             : 
    1604         139 : static NTSTATUS cli_cifs_rename_recv(struct tevent_req *req)
    1605             : {
    1606         139 :         return tevent_req_simple_recv_ntstatus(req);
    1607             : }
    1608             : 
    1609             : struct cli_rename_state {
    1610             :         uint8_t dummy;
    1611             : };
    1612             : 
    1613             : static void cli_rename_done1(struct tevent_req *subreq);
    1614             : static void cli_rename_done_cifs(struct tevent_req *subreq);
    1615             : static void cli_rename_done2(struct tevent_req *subreq);
    1616             : 
    1617         370 : struct tevent_req *cli_rename_send(TALLOC_CTX *mem_ctx,
    1618             :                                    struct tevent_context *ev,
    1619             :                                    struct cli_state *cli,
    1620             :                                    const char *fname_src,
    1621             :                                    const char *fname_dst,
    1622             :                                    bool replace)
    1623             : {
    1624         370 :         struct tevent_req *req = NULL, *subreq = NULL;
    1625         370 :         struct cli_rename_state *state = NULL;
    1626             : 
    1627         370 :         req = tevent_req_create(mem_ctx, &state, struct cli_rename_state);
    1628         370 :         if (req == NULL) {
    1629           0 :                 return NULL;
    1630             :         }
    1631             : 
    1632         370 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    1633         225 :                 subreq = cli_smb2_rename_send(
    1634             :                         state, ev, cli, fname_src, fname_dst, replace);
    1635         225 :                 if (tevent_req_nomem(subreq, req)) {
    1636           0 :                         return tevent_req_post(req, ev);
    1637             :                 }
    1638         225 :                 tevent_req_set_callback(subreq, cli_rename_done2, req);
    1639         225 :                 return req;
    1640             :         }
    1641             : 
    1642         145 :         if (replace && smbXcli_conn_support_passthrough(cli->conn)) {
    1643           6 :                 subreq = cli_smb1_rename_send(
    1644             :                         state, ev, cli, fname_src, fname_dst, replace);
    1645           6 :                 if (tevent_req_nomem(subreq, req)) {
    1646           0 :                         return tevent_req_post(req, ev);
    1647             :                 }
    1648           6 :                 tevent_req_set_callback(subreq, cli_rename_done1, req);
    1649           6 :                 return req;
    1650             :         }
    1651             : 
    1652         139 :         subreq = cli_cifs_rename_send(
    1653             :                 state, ev, cli, fname_src,fname_dst, replace);
    1654         139 :         if (tevent_req_nomem(subreq, req)) {
    1655           0 :                 return tevent_req_post(req, ev);
    1656             :         }
    1657         139 :         tevent_req_set_callback(subreq, cli_rename_done_cifs, req);
    1658         139 :         return req;
    1659             : }
    1660             : 
    1661           6 : static void cli_rename_done1(struct tevent_req *subreq)
    1662             : {
    1663           6 :         NTSTATUS status = cli_smb1_rename_recv(subreq);
    1664           6 :         tevent_req_simple_finish_ntstatus(subreq, status);
    1665           6 : }
    1666             : 
    1667         139 : static void cli_rename_done_cifs(struct tevent_req *subreq)
    1668             : {
    1669         139 :         NTSTATUS status = cli_cifs_rename_recv(subreq);
    1670         139 :         tevent_req_simple_finish_ntstatus(subreq, status);
    1671         139 : }
    1672             : 
    1673         225 : static void cli_rename_done2(struct tevent_req *subreq)
    1674             : {
    1675         225 :         NTSTATUS status = cli_smb2_rename_recv(subreq);
    1676         225 :         tevent_req_simple_finish_ntstatus(subreq, status);
    1677         225 : }
    1678             : 
    1679         370 : NTSTATUS cli_rename_recv(struct tevent_req *req)
    1680             : {
    1681         370 :         return tevent_req_simple_recv_ntstatus(req);
    1682             : }
    1683             : 
    1684         352 : NTSTATUS cli_rename(struct cli_state *cli,
    1685             :                     const char *fname_src,
    1686             :                     const char *fname_dst,
    1687             :                     bool replace)
    1688             : {
    1689         352 :         TALLOC_CTX *frame = NULL;
    1690           0 :         struct tevent_context *ev;
    1691           0 :         struct tevent_req *req;
    1692         352 :         NTSTATUS status = NT_STATUS_OK;
    1693             : 
    1694         352 :         frame = talloc_stackframe();
    1695             : 
    1696         352 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    1697             :                 /*
    1698             :                  * Can't use sync call while an async call is in flight
    1699             :                  */
    1700           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    1701           0 :                 goto fail;
    1702             :         }
    1703             : 
    1704         352 :         ev = samba_tevent_context_init(frame);
    1705         352 :         if (ev == NULL) {
    1706           0 :                 status = NT_STATUS_NO_MEMORY;
    1707           0 :                 goto fail;
    1708             :         }
    1709             : 
    1710         352 :         req = cli_rename_send(frame, ev, cli, fname_src, fname_dst, replace);
    1711         352 :         if (req == NULL) {
    1712           0 :                 status = NT_STATUS_NO_MEMORY;
    1713           0 :                 goto fail;
    1714             :         }
    1715             : 
    1716         352 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    1717           0 :                 goto fail;
    1718             :         }
    1719             : 
    1720         352 :         status = cli_rename_recv(req);
    1721         352 :         cli->raw_status = status; /* cli_smb2_rename_recv doesn't set this */
    1722             : 
    1723         352 :  fail:
    1724         352 :         TALLOC_FREE(frame);
    1725         352 :         return status;
    1726             : }
    1727             : 
    1728             : /****************************************************************************
    1729             :  NT Rename a file.
    1730             : ****************************************************************************/
    1731             : 
    1732             : static void cli_ntrename_internal_done(struct tevent_req *subreq);
    1733             : 
    1734             : struct cli_ntrename_internal_state {
    1735             :         uint16_t vwv[4];
    1736             : };
    1737             : 
    1738          15 : static struct tevent_req *cli_ntrename_internal_send(TALLOC_CTX *mem_ctx,
    1739             :                                 struct tevent_context *ev,
    1740             :                                 struct cli_state *cli,
    1741             :                                 const char *fname_src,
    1742             :                                 const char *fname_dst,
    1743             :                                 uint16_t rename_flag)
    1744             : {
    1745          15 :         struct tevent_req *req = NULL, *subreq = NULL;
    1746          15 :         struct cli_ntrename_internal_state *state = NULL;
    1747          15 :         uint8_t additional_flags = 0;
    1748          15 :         uint16_t additional_flags2 = 0;
    1749          15 :         uint8_t *bytes = NULL;
    1750          15 :         char *fname_src_cp = NULL;
    1751          15 :         char *fname_dst_cp = NULL;
    1752             : 
    1753          15 :         req = tevent_req_create(mem_ctx, &state,
    1754             :                                 struct cli_ntrename_internal_state);
    1755          15 :         if (req == NULL) {
    1756           0 :                 return NULL;
    1757             :         }
    1758             : 
    1759          15 :         SSVAL(state->vwv+0, 0 ,FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_DIRECTORY);
    1760          15 :         SSVAL(state->vwv+1, 0, rename_flag);
    1761             : 
    1762          15 :         bytes = talloc_array(state, uint8_t, 1);
    1763          15 :         if (tevent_req_nomem(bytes, req)) {
    1764           0 :                 return tevent_req_post(req, ev);
    1765             :         }
    1766             :         /*
    1767             :          * SMBntrename on a DFS share uses DFS names for src and dst.
    1768             :          * See smbtorture3: SMB1-DFS-PATHS: test_smb1_ntrename_rename().
    1769             :          */
    1770          15 :         fname_src_cp = smb1_dfs_share_path(state, cli, fname_src);
    1771          15 :         if (tevent_req_nomem(fname_src_cp, req)) {
    1772           0 :                 return tevent_req_post(req, ev);
    1773             :         }
    1774          15 :         bytes[0] = 4;
    1775          15 :         bytes = smb_bytes_push_str(bytes,
    1776          15 :                                    smbXcli_conn_use_unicode(cli->conn),
    1777             :                                    fname_src_cp,
    1778          15 :                                    strlen(fname_src_cp)+1,
    1779             :                                    NULL);
    1780          15 :         if (tevent_req_nomem(bytes, req)) {
    1781           0 :                 return tevent_req_post(req, ev);
    1782             :         }
    1783             : 
    1784          15 :         if (clistr_is_previous_version_path(fname_src)) {
    1785           0 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
    1786             :         }
    1787             : 
    1788          15 :         bytes = talloc_realloc(state, bytes, uint8_t,
    1789             :                         talloc_get_size(bytes)+1);
    1790          15 :         if (tevent_req_nomem(bytes, req)) {
    1791           0 :                 return tevent_req_post(req, ev);
    1792             :         }
    1793             : 
    1794             :         /*
    1795             :          * SMBntrename on a DFS share uses DFS names for src and dst.
    1796             :          * See smbtorture3: SMB1-DFS-PATHS: test_smb1_ntrename_rename().
    1797             :          * and smbtorture3: SMB1-DFS-PATHS: test_smb1_ntrename_hardlink()
    1798             :          */
    1799          15 :         fname_dst_cp = smb1_dfs_share_path(state, cli, fname_dst);
    1800          15 :         if (tevent_req_nomem(fname_dst_cp, req)) {
    1801           0 :                 return tevent_req_post(req, ev);
    1802             :         }
    1803          15 :         bytes[talloc_get_size(bytes)-1] = 4;
    1804          15 :         bytes = smb_bytes_push_str(bytes,
    1805          15 :                                    smbXcli_conn_use_unicode(cli->conn),
    1806             :                                    fname_dst_cp,
    1807          15 :                                    strlen(fname_dst_cp)+1,
    1808             :                                    NULL);
    1809          15 :         if (tevent_req_nomem(bytes, req)) {
    1810           0 :                 return tevent_req_post(req, ev);
    1811             :         }
    1812             : 
    1813          15 :         subreq = cli_smb_send(state, ev, cli, SMBntrename, additional_flags,
    1814             :                         additional_flags2,
    1815          15 :                         4, state->vwv, talloc_get_size(bytes), bytes);
    1816          15 :         if (tevent_req_nomem(subreq, req)) {
    1817           0 :                 return tevent_req_post(req, ev);
    1818             :         }
    1819          15 :         tevent_req_set_callback(subreq, cli_ntrename_internal_done, req);
    1820          15 :         return req;
    1821             : }
    1822             : 
    1823          15 : static void cli_ntrename_internal_done(struct tevent_req *subreq)
    1824             : {
    1825          15 :         NTSTATUS status = cli_smb_recv(
    1826             :                 subreq, NULL, NULL, 0, NULL, NULL, NULL, NULL);
    1827          15 :         tevent_req_simple_finish_ntstatus(subreq, status);
    1828          15 : }
    1829             : 
    1830          15 : static NTSTATUS cli_ntrename_internal_recv(struct tevent_req *req)
    1831             : {
    1832          15 :         return tevent_req_simple_recv_ntstatus(req);
    1833             : }
    1834             : 
    1835           0 : struct tevent_req *cli_ntrename_send(TALLOC_CTX *mem_ctx,
    1836             :                                 struct tevent_context *ev,
    1837             :                                 struct cli_state *cli,
    1838             :                                 const char *fname_src,
    1839             :                                 const char *fname_dst)
    1840             : {
    1841           0 :         return cli_ntrename_internal_send(mem_ctx,
    1842             :                                           ev,
    1843             :                                           cli,
    1844             :                                           fname_src,
    1845             :                                           fname_dst,
    1846             :                                           RENAME_FLAG_RENAME);
    1847             : }
    1848             : 
    1849           0 : NTSTATUS cli_ntrename_recv(struct tevent_req *req)
    1850             : {
    1851           0 :         return cli_ntrename_internal_recv(req);
    1852             : }
    1853             : 
    1854           0 : NTSTATUS cli_ntrename(struct cli_state *cli, const char *fname_src, const char *fname_dst)
    1855             : {
    1856           0 :         TALLOC_CTX *frame = talloc_stackframe();
    1857           0 :         struct tevent_context *ev;
    1858           0 :         struct tevent_req *req;
    1859           0 :         NTSTATUS status = NT_STATUS_OK;
    1860             : 
    1861           0 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    1862             :                 /*
    1863             :                  * Can't use sync call while an async call is in flight
    1864             :                  */
    1865           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    1866           0 :                 goto fail;
    1867             :         }
    1868             : 
    1869           0 :         ev = samba_tevent_context_init(frame);
    1870           0 :         if (ev == NULL) {
    1871           0 :                 status = NT_STATUS_NO_MEMORY;
    1872           0 :                 goto fail;
    1873             :         }
    1874             : 
    1875           0 :         req = cli_ntrename_send(frame, ev, cli, fname_src, fname_dst);
    1876           0 :         if (req == NULL) {
    1877           0 :                 status = NT_STATUS_NO_MEMORY;
    1878           0 :                 goto fail;
    1879             :         }
    1880             : 
    1881           0 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    1882           0 :                 goto fail;
    1883             :         }
    1884             : 
    1885           0 :         status = cli_ntrename_recv(req);
    1886             : 
    1887           0 :  fail:
    1888           0 :         TALLOC_FREE(frame);
    1889           0 :         return status;
    1890             : }
    1891             : 
    1892             : /****************************************************************************
    1893             :  NT hardlink a file.
    1894             : ****************************************************************************/
    1895             : 
    1896          15 : static struct tevent_req *cli_nt_hardlink_send(TALLOC_CTX *mem_ctx,
    1897             :                                 struct tevent_context *ev,
    1898             :                                 struct cli_state *cli,
    1899             :                                 const char *fname_src,
    1900             :                                 const char *fname_dst)
    1901             : {
    1902          15 :         return cli_ntrename_internal_send(mem_ctx,
    1903             :                                           ev,
    1904             :                                           cli,
    1905             :                                           fname_src,
    1906             :                                           fname_dst,
    1907             :                                           RENAME_FLAG_HARD_LINK);
    1908             : }
    1909             : 
    1910          15 : static NTSTATUS cli_nt_hardlink_recv(struct tevent_req *req)
    1911             : {
    1912          15 :         return cli_ntrename_internal_recv(req);
    1913             : }
    1914             : 
    1915             : struct cli_smb2_hardlink_state {
    1916             :         struct tevent_context *ev;
    1917             :         struct cli_state *cli;
    1918             :         uint16_t fnum_src;
    1919             :         const char *fname_dst;
    1920             :         bool overwrite;
    1921             :         NTSTATUS status;
    1922             : };
    1923             : 
    1924             : static void cli_smb2_hardlink_opened(struct tevent_req *subreq);
    1925             : static void cli_smb2_hardlink_info_set(struct tevent_req *subreq);
    1926             : static void cli_smb2_hardlink_closed(struct tevent_req *subreq);
    1927             : 
    1928          14 : static struct tevent_req *cli_smb2_hardlink_send(
    1929             :         TALLOC_CTX *mem_ctx,
    1930             :         struct tevent_context *ev,
    1931             :         struct cli_state *cli,
    1932             :         const char *fname_src,
    1933             :         const char *fname_dst,
    1934             :         bool overwrite,
    1935             :         struct smb2_create_blobs *in_cblobs)
    1936             : {
    1937          14 :         struct tevent_req *req = NULL, *subreq = NULL;
    1938          14 :         struct cli_smb2_hardlink_state *state = NULL;
    1939           0 :         NTSTATUS status;
    1940             : 
    1941          14 :         req = tevent_req_create(
    1942             :                 mem_ctx, &state, struct cli_smb2_hardlink_state);
    1943          14 :         if (req == NULL) {
    1944           0 :                 return NULL;
    1945             :         }
    1946             : 
    1947             :         /*
    1948             :          * Strip a MSDFS path from fname_dst if we were given one.
    1949             :          */
    1950          14 :         status = cli_dfs_target_check(state,
    1951             :                                 cli,
    1952             :                                 fname_dst,
    1953             :                                 &fname_dst);
    1954          14 :         if (tevent_req_nterror(req, status)) {
    1955           0 :                 return tevent_req_post(req, ev);
    1956             :         }
    1957             : 
    1958          14 :         state->ev = ev;
    1959          14 :         state->cli = cli;
    1960          14 :         state->fname_dst = fname_dst;
    1961          14 :         state->overwrite = overwrite;
    1962             : 
    1963          14 :         subreq = cli_smb2_create_fnum_send(
    1964             :                 state,
    1965             :                 ev,
    1966             :                 cli,
    1967             :                 fname_src,
    1968          14 :                 (struct cli_smb2_create_flags){0},
    1969             :                 SMB2_IMPERSONATION_IMPERSONATION,
    1970             :                 FILE_WRITE_ATTRIBUTES,
    1971             :                 0,                      /* file attributes */
    1972             :                 FILE_SHARE_READ|
    1973             :                 FILE_SHARE_WRITE|
    1974             :                 FILE_SHARE_DELETE,       /* share_access */
    1975             :                 FILE_OPEN,               /* create_disposition */
    1976             :                 FILE_NON_DIRECTORY_FILE, /* no hardlinks on directories */
    1977             :                 in_cblobs);
    1978          14 :         if (tevent_req_nomem(subreq, req)) {
    1979           0 :                 return tevent_req_post(req, ev);
    1980             :         }
    1981          14 :         tevent_req_set_callback(subreq, cli_smb2_hardlink_opened, req);
    1982          14 :         return req;
    1983             : }
    1984             : 
    1985          14 : static void cli_smb2_hardlink_opened(struct tevent_req *subreq)
    1986             : {
    1987          14 :         struct tevent_req *req = tevent_req_callback_data(
    1988             :                 subreq, struct tevent_req);
    1989          14 :         struct cli_smb2_hardlink_state *state = tevent_req_data(
    1990             :                 req, struct cli_smb2_hardlink_state);
    1991           0 :         NTSTATUS status;
    1992           0 :         smb_ucs2_t *ucs2_dst;
    1993           0 :         size_t ucs2_len;
    1994           0 :         DATA_BLOB inbuf;
    1995           0 :         bool ok;
    1996             : 
    1997          14 :         status = cli_smb2_create_fnum_recv(
    1998             :                 subreq, &state->fnum_src, NULL, NULL, NULL, NULL);
    1999          14 :         TALLOC_FREE(subreq);
    2000          14 :         if (tevent_req_nterror(req, status)) {
    2001           0 :                 return;
    2002             :         }
    2003             : 
    2004          14 :         ok = push_ucs2_talloc(state, &ucs2_dst, state->fname_dst, &ucs2_len);
    2005          14 :         if (!ok || (ucs2_len < 2)) {
    2006           0 :                 tevent_req_nterror(req, NT_STATUS_INVALID_PARAMETER);
    2007           0 :                 return;
    2008             :         }
    2009             :         /* Don't 0-terminate the name */
    2010          14 :         ucs2_len -= 2;
    2011             : 
    2012          14 :         inbuf = data_blob_talloc_zero(state, ucs2_len + 20);
    2013          14 :         if (tevent_req_nomem(inbuf.data, req)) {
    2014           0 :                 return;
    2015             :         }
    2016             : 
    2017          14 :         if (state->overwrite) {
    2018           0 :                 SCVAL(inbuf.data, 0, 1);
    2019             :         }
    2020          14 :         SIVAL(inbuf.data, 16, ucs2_len);
    2021          14 :         memcpy(inbuf.data + 20, ucs2_dst, ucs2_len);
    2022          14 :         TALLOC_FREE(ucs2_dst);
    2023             : 
    2024          14 :         subreq = cli_smb2_set_info_fnum_send(
    2025             :                 state,
    2026             :                 state->ev,
    2027             :                 state->cli,
    2028          14 :                 state->fnum_src,
    2029             :                 1,              /* in_info_type */
    2030             :                 SMB_FILE_LINK_INFORMATION - 1000, /* in_file_info_class */
    2031             :                 &inbuf,
    2032             :                 0);             /* in_additional_info */
    2033          14 :         if (tevent_req_nomem(subreq, req)) {
    2034           0 :                 return;
    2035             :         }
    2036          14 :         tevent_req_set_callback(subreq, cli_smb2_hardlink_info_set, req);
    2037             : }
    2038             : 
    2039          14 : static void cli_smb2_hardlink_info_set(struct tevent_req *subreq)
    2040             : {
    2041          14 :         struct tevent_req *req = tevent_req_callback_data(
    2042             :                 subreq, struct tevent_req);
    2043          14 :         struct cli_smb2_hardlink_state *state = tevent_req_data(
    2044             :                 req, struct cli_smb2_hardlink_state);
    2045             : 
    2046          14 :         state->status = cli_smb2_set_info_fnum_recv(subreq);
    2047          14 :         TALLOC_FREE(subreq);
    2048             : 
    2049             :         /* ignore error here, we need to close the file */
    2050             : 
    2051          14 :         subreq = cli_smb2_close_fnum_send(state,
    2052             :                                           state->ev,
    2053             :                                           state->cli,
    2054          14 :                                           state->fnum_src,
    2055             :                                           0);
    2056          14 :         if (tevent_req_nomem(subreq, req)) {
    2057           0 :                 return;
    2058             :         }
    2059          14 :         tevent_req_set_callback(subreq, cli_smb2_hardlink_closed, req);
    2060             : }
    2061             : 
    2062          14 : static void cli_smb2_hardlink_closed(struct tevent_req *subreq)
    2063             : {
    2064          14 :         NTSTATUS status = cli_smb2_close_fnum_recv(subreq);
    2065          14 :         tevent_req_simple_finish_ntstatus(subreq, status);
    2066          14 : }
    2067             : 
    2068          14 : static NTSTATUS cli_smb2_hardlink_recv(struct tevent_req *req)
    2069             : {
    2070          14 :         struct cli_smb2_hardlink_state *state = tevent_req_data(
    2071             :                 req, struct cli_smb2_hardlink_state);
    2072           0 :         NTSTATUS status;
    2073             : 
    2074          14 :         if (tevent_req_is_nterror(req, &status)) {
    2075           0 :                 return status;
    2076             :         }
    2077          14 :         return state->status;
    2078             : }
    2079             : 
    2080             : struct cli_hardlink_state {
    2081             :         uint8_t dummy;
    2082             : };
    2083             : 
    2084             : static void cli_hardlink_done(struct tevent_req *subreq);
    2085             : static void cli_hardlink_done2(struct tevent_req *subreq);
    2086             : 
    2087          29 : struct tevent_req *cli_hardlink_send(
    2088             :         TALLOC_CTX *mem_ctx,
    2089             :         struct tevent_context *ev,
    2090             :         struct cli_state *cli,
    2091             :         const char *fname_src,
    2092             :         const char *fname_dst)
    2093             : {
    2094          29 :         struct tevent_req *req = NULL, *subreq = NULL;
    2095           0 :         struct cli_hardlink_state *state;
    2096             : 
    2097          29 :         req = tevent_req_create(mem_ctx, &state, struct cli_hardlink_state);
    2098          29 :         if (req == NULL) {
    2099           0 :                 return NULL;
    2100             :         }
    2101             : 
    2102          29 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    2103          14 :                 subreq = cli_smb2_hardlink_send(
    2104             :                         state, ev, cli, fname_src, fname_dst, false, NULL);
    2105          14 :                 if (tevent_req_nomem(subreq, req)) {
    2106           0 :                         return tevent_req_post(req, ev);
    2107             :                 }
    2108          14 :                 tevent_req_set_callback(subreq, cli_hardlink_done2, req);
    2109          14 :                 return req;
    2110             :         }
    2111             : 
    2112          15 :         subreq = cli_nt_hardlink_send(state, ev, cli, fname_src, fname_dst);
    2113          15 :         if (tevent_req_nomem(subreq, req)) {
    2114           0 :                 return tevent_req_post(req, ev);
    2115             :         }
    2116          15 :         tevent_req_set_callback(subreq, cli_hardlink_done, req);
    2117          15 :         return req;
    2118             : }
    2119             : 
    2120          15 : static void cli_hardlink_done(struct tevent_req *subreq)
    2121             : {
    2122          15 :         NTSTATUS status = cli_nt_hardlink_recv(subreq);
    2123          15 :         tevent_req_simple_finish_ntstatus(subreq, status);
    2124          15 : }
    2125             : 
    2126          14 : static void cli_hardlink_done2(struct tevent_req *subreq)
    2127             : {
    2128          14 :         NTSTATUS status = cli_smb2_hardlink_recv(subreq);
    2129          14 :         tevent_req_simple_finish_ntstatus(subreq, status);
    2130          14 : }
    2131             : 
    2132          29 : NTSTATUS cli_hardlink_recv(struct tevent_req *req)
    2133             : {
    2134          29 :         return tevent_req_simple_recv_ntstatus(req);
    2135             : }
    2136             : 
    2137          29 : NTSTATUS cli_hardlink(
    2138             :         struct cli_state *cli, const char *fname_src, const char *fname_dst)
    2139             : {
    2140          29 :         TALLOC_CTX *frame = talloc_stackframe();
    2141          29 :         struct tevent_context *ev = NULL;
    2142          29 :         struct tevent_req *req = NULL;
    2143          29 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    2144             : 
    2145          29 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    2146           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    2147           0 :                 goto fail;
    2148             :         }
    2149          29 :         ev = samba_tevent_context_init(frame);
    2150          29 :         if (ev == NULL) {
    2151           0 :                 goto fail;
    2152             :         }
    2153          29 :         req = cli_hardlink_send(frame, ev, cli, fname_src, fname_dst);
    2154          29 :         if (req == NULL) {
    2155           0 :                 goto fail;
    2156             :         }
    2157          29 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    2158           0 :                 goto fail;
    2159             :         }
    2160          29 :         status = cli_hardlink_recv(req);
    2161          29 :  fail:
    2162          29 :         TALLOC_FREE(frame);
    2163          29 :         return status;
    2164             : }
    2165             : 
    2166             : /****************************************************************************
    2167             :  Delete a file.
    2168             : ****************************************************************************/
    2169             : 
    2170             : static void cli_unlink_done(struct tevent_req *subreq);
    2171             : static void cli_unlink_done2(struct tevent_req *subreq);
    2172             : 
    2173             : struct cli_unlink_state {
    2174             :         uint16_t vwv[1];
    2175             : };
    2176             : 
    2177        4646 : struct tevent_req *cli_unlink_send(TALLOC_CTX *mem_ctx,
    2178             :                                 struct tevent_context *ev,
    2179             :                                 struct cli_state *cli,
    2180             :                                 const char *fname,
    2181             :                                 uint32_t mayhave_attrs)
    2182             : {
    2183        4646 :         struct tevent_req *req = NULL, *subreq = NULL;
    2184        4646 :         struct cli_unlink_state *state = NULL;
    2185        4646 :         uint8_t additional_flags = 0;
    2186        4646 :         uint16_t additional_flags2 = 0;
    2187        4646 :         uint8_t *bytes = NULL;
    2188        4646 :         char *fname_cp = NULL;
    2189             : 
    2190        4646 :         req = tevent_req_create(mem_ctx, &state, struct cli_unlink_state);
    2191        4646 :         if (req == NULL) {
    2192           0 :                 return NULL;
    2193             :         }
    2194             : 
    2195        4646 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    2196        2321 :                 subreq = cli_smb2_unlink_send(state, ev, cli, fname, NULL);
    2197        2321 :                 if (tevent_req_nomem(subreq, req)) {
    2198           0 :                         return tevent_req_post(req, ev);
    2199             :                 }
    2200        2321 :                 tevent_req_set_callback(subreq, cli_unlink_done2, req);
    2201        2321 :                 return req;
    2202             :         }
    2203             : 
    2204        2325 :         if (mayhave_attrs & 0xFFFF0000) {
    2205             :                 /*
    2206             :                  * Don't allow attributes greater than
    2207             :                  * 16-bits for a 16-bit protocol value.
    2208             :                  */
    2209           5 :                 if (tevent_req_nterror(req, NT_STATUS_INVALID_PARAMETER)) {
    2210           5 :                         return tevent_req_post(req, ev);
    2211             :                 }
    2212             :         }
    2213             : 
    2214        2320 :         SSVAL(state->vwv+0, 0, mayhave_attrs);
    2215             : 
    2216        2320 :         bytes = talloc_array(state, uint8_t, 1);
    2217        2320 :         if (tevent_req_nomem(bytes, req)) {
    2218           0 :                 return tevent_req_post(req, ev);
    2219             :         }
    2220             :         /*
    2221             :          * SMBunlink on a DFS share must use DFS names.
    2222             :          */
    2223        2320 :         fname_cp = smb1_dfs_share_path(state, cli, fname);
    2224        2320 :         if (tevent_req_nomem(fname_cp, req)) {
    2225           0 :                 return tevent_req_post(req, ev);
    2226             :         }
    2227        2320 :         bytes[0] = 4;
    2228        2320 :         bytes = smb_bytes_push_str(bytes,
    2229        2320 :                                    smbXcli_conn_use_unicode(cli->conn),
    2230             :                                    fname_cp,
    2231        2320 :                                    strlen(fname_cp)+1,
    2232             :                                    NULL);
    2233             : 
    2234        2320 :         if (tevent_req_nomem(bytes, req)) {
    2235           0 :                 return tevent_req_post(req, ev);
    2236             :         }
    2237             : 
    2238        2320 :         if (clistr_is_previous_version_path(fname)) {
    2239           0 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
    2240             :         }
    2241             : 
    2242        2320 :         subreq = cli_smb_send(state, ev, cli, SMBunlink, additional_flags,
    2243             :                                 additional_flags2,
    2244        2320 :                                 1, state->vwv, talloc_get_size(bytes), bytes);
    2245        2320 :         if (tevent_req_nomem(subreq, req)) {
    2246           0 :                 return tevent_req_post(req, ev);
    2247             :         }
    2248        2320 :         tevent_req_set_callback(subreq, cli_unlink_done, req);
    2249        2320 :         return req;
    2250             : }
    2251             : 
    2252        2320 : static void cli_unlink_done(struct tevent_req *subreq)
    2253             : {
    2254        2320 :         NTSTATUS status = cli_smb_recv(
    2255             :                 subreq, NULL, NULL, 0, NULL, NULL, NULL, NULL);
    2256        2320 :         tevent_req_simple_finish_ntstatus(subreq, status);
    2257        2320 : }
    2258             : 
    2259        2321 : static void cli_unlink_done2(struct tevent_req *subreq)
    2260             : {
    2261        2321 :         NTSTATUS status = cli_smb2_unlink_recv(subreq);
    2262        2321 :         tevent_req_simple_finish_ntstatus(subreq, status);
    2263        2321 : }
    2264             : 
    2265        4646 : NTSTATUS cli_unlink_recv(struct tevent_req *req)
    2266             : {
    2267        4646 :         return tevent_req_simple_recv_ntstatus(req);
    2268             : }
    2269             : 
    2270        3829 : NTSTATUS cli_unlink(struct cli_state *cli, const char *fname, uint32_t mayhave_attrs)
    2271             : {
    2272        3829 :         TALLOC_CTX *frame = NULL;
    2273           0 :         struct tevent_context *ev;
    2274           0 :         struct tevent_req *req;
    2275        3829 :         NTSTATUS status = NT_STATUS_OK;
    2276             : 
    2277        3829 :         frame = talloc_stackframe();
    2278             : 
    2279        3829 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    2280             :                 /*
    2281             :                  * Can't use sync call while an async call is in flight
    2282             :                  */
    2283           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    2284           0 :                 goto fail;
    2285             :         }
    2286             : 
    2287        3829 :         ev = samba_tevent_context_init(frame);
    2288        3829 :         if (ev == NULL) {
    2289           0 :                 status = NT_STATUS_NO_MEMORY;
    2290           0 :                 goto fail;
    2291             :         }
    2292             : 
    2293        3829 :         req = cli_unlink_send(frame, ev, cli, fname, mayhave_attrs);
    2294        3829 :         if (req == NULL) {
    2295           0 :                 status = NT_STATUS_NO_MEMORY;
    2296           0 :                 goto fail;
    2297             :         }
    2298             : 
    2299        3829 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    2300           0 :                 goto fail;
    2301             :         }
    2302             : 
    2303        3829 :         status = cli_unlink_recv(req);
    2304        3829 :         cli->raw_status = status; /* cli_smb2_unlink_recv doesn't set this */
    2305             : 
    2306        3829 :  fail:
    2307        3829 :         TALLOC_FREE(frame);
    2308        3829 :         return status;
    2309             : }
    2310             : 
    2311             : /****************************************************************************
    2312             :  Create a directory.
    2313             : ****************************************************************************/
    2314             : 
    2315             : static void cli_mkdir_done(struct tevent_req *subreq);
    2316             : static void cli_mkdir_done2(struct tevent_req *subreq);
    2317             : 
    2318             : struct cli_mkdir_state {
    2319             :         int dummy;
    2320             : };
    2321             : 
    2322        3382 : struct tevent_req *cli_mkdir_send(TALLOC_CTX *mem_ctx,
    2323             :                                   struct tevent_context *ev,
    2324             :                                   struct cli_state *cli,
    2325             :                                   const char *dname)
    2326             : {
    2327        3382 :         struct tevent_req *req = NULL, *subreq = NULL;
    2328        3382 :         struct cli_mkdir_state *state = NULL;
    2329        3382 :         uint8_t additional_flags = 0;
    2330        3382 :         uint16_t additional_flags2 = 0;
    2331        3382 :         uint8_t *bytes = NULL;
    2332        3382 :         char *dname_cp = NULL;
    2333             : 
    2334        3382 :         req = tevent_req_create(mem_ctx, &state, struct cli_mkdir_state);
    2335        3382 :         if (req == NULL) {
    2336           0 :                 return NULL;
    2337             :         }
    2338             : 
    2339        3382 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    2340        3204 :                 subreq = cli_smb2_mkdir_send(state, ev, cli, dname);
    2341        3204 :                 if (tevent_req_nomem(subreq, req)) {
    2342           0 :                         return tevent_req_post(req, ev);
    2343             :                 }
    2344        3204 :                 tevent_req_set_callback(subreq, cli_mkdir_done2, req);
    2345        3204 :                 return req;
    2346             :         }
    2347             : 
    2348         178 :         bytes = talloc_array(state, uint8_t, 1);
    2349         178 :         if (tevent_req_nomem(bytes, req)) {
    2350           0 :                 return tevent_req_post(req, ev);
    2351             :         }
    2352             :         /*
    2353             :          * SMBmkdir on a DFS share must use DFS names.
    2354             :          */
    2355         178 :         dname_cp = smb1_dfs_share_path(state, cli, dname);
    2356         178 :         if (tevent_req_nomem(dname_cp, req)) {
    2357           0 :                 return tevent_req_post(req, ev);
    2358             :         }
    2359         178 :         bytes[0] = 4;
    2360         178 :         bytes = smb_bytes_push_str(bytes,
    2361         178 :                                    smbXcli_conn_use_unicode(cli->conn),
    2362             :                                    dname_cp,
    2363         178 :                                    strlen(dname_cp)+1,
    2364             :                                    NULL);
    2365             : 
    2366         178 :         if (tevent_req_nomem(bytes, req)) {
    2367           0 :                 return tevent_req_post(req, ev);
    2368             :         }
    2369             : 
    2370         178 :         if (clistr_is_previous_version_path(dname)) {
    2371           0 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
    2372             :         }
    2373             : 
    2374         178 :         subreq = cli_smb_send(state, ev, cli, SMBmkdir, additional_flags,
    2375             :                         additional_flags2,
    2376         178 :                         0, NULL, talloc_get_size(bytes), bytes);
    2377         178 :         if (tevent_req_nomem(subreq, req)) {
    2378           0 :                 return tevent_req_post(req, ev);
    2379             :         }
    2380         178 :         tevent_req_set_callback(subreq, cli_mkdir_done, req);
    2381         178 :         return req;
    2382             : }
    2383             : 
    2384         178 : static void cli_mkdir_done(struct tevent_req *subreq)
    2385             : {
    2386         178 :         struct tevent_req *req = tevent_req_callback_data(
    2387             :                 subreq, struct tevent_req);
    2388           0 :         NTSTATUS status;
    2389             : 
    2390         178 :         status = cli_smb_recv(subreq, NULL, NULL, 0, NULL, NULL, NULL, NULL);
    2391         178 :         TALLOC_FREE(subreq);
    2392         178 :         if (tevent_req_nterror(req, status)) {
    2393           4 :                 return;
    2394             :         }
    2395         174 :         tevent_req_done(req);
    2396             : }
    2397             : 
    2398        3204 : static void cli_mkdir_done2(struct tevent_req *subreq)
    2399             : {
    2400        3204 :         NTSTATUS status = cli_smb2_mkdir_recv(subreq);
    2401        3204 :         tevent_req_simple_finish_ntstatus(subreq, status);
    2402        3204 : }
    2403             : 
    2404        3382 : NTSTATUS cli_mkdir_recv(struct tevent_req *req)
    2405             : {
    2406        3382 :         return tevent_req_simple_recv_ntstatus(req);
    2407             : }
    2408             : 
    2409        1912 : NTSTATUS cli_mkdir(struct cli_state *cli, const char *dname)
    2410             : {
    2411        1912 :         TALLOC_CTX *frame = NULL;
    2412           0 :         struct tevent_context *ev;
    2413           0 :         struct tevent_req *req;
    2414        1912 :         NTSTATUS status = NT_STATUS_OK;
    2415             : 
    2416        1912 :         frame = talloc_stackframe();
    2417             : 
    2418        1912 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    2419             :                 /*
    2420             :                  * Can't use sync call while an async call is in flight
    2421             :                  */
    2422           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    2423           0 :                 goto fail;
    2424             :         }
    2425             : 
    2426        1912 :         ev = samba_tevent_context_init(frame);
    2427        1912 :         if (ev == NULL) {
    2428           0 :                 status = NT_STATUS_NO_MEMORY;
    2429           0 :                 goto fail;
    2430             :         }
    2431             : 
    2432        1912 :         req = cli_mkdir_send(frame, ev, cli, dname);
    2433        1912 :         if (req == NULL) {
    2434           0 :                 status = NT_STATUS_NO_MEMORY;
    2435           0 :                 goto fail;
    2436             :         }
    2437             : 
    2438        1912 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    2439           0 :                 goto fail;
    2440             :         }
    2441             : 
    2442        1912 :         status = cli_mkdir_recv(req);
    2443        1912 :         cli->raw_status = status; /* cli_smb2_mkdir_recv doesn't set this */
    2444             : 
    2445        1912 :  fail:
    2446        1912 :         TALLOC_FREE(frame);
    2447        1912 :         return status;
    2448             : }
    2449             : 
    2450             : /****************************************************************************
    2451             :  Remove a directory.
    2452             : ****************************************************************************/
    2453             : 
    2454             : static void cli_rmdir_done(struct tevent_req *subreq);
    2455             : static void cli_rmdir_done2(struct tevent_req *subreq);
    2456             : 
    2457             : struct cli_rmdir_state {
    2458             :         int dummy;
    2459             : };
    2460             : 
    2461        2795 : struct tevent_req *cli_rmdir_send(TALLOC_CTX *mem_ctx,
    2462             :                                   struct tevent_context *ev,
    2463             :                                   struct cli_state *cli,
    2464             :                                   const char *dname)
    2465             : {
    2466        2795 :         struct tevent_req *req = NULL, *subreq = NULL;
    2467        2795 :         struct cli_rmdir_state *state = NULL;
    2468        2795 :         uint8_t additional_flags = 0;
    2469        2795 :         uint16_t additional_flags2 = 0;
    2470        2795 :         uint8_t *bytes = NULL;
    2471        2795 :         char *dname_cp = NULL;
    2472             : 
    2473        2795 :         req = tevent_req_create(mem_ctx, &state, struct cli_rmdir_state);
    2474        2795 :         if (req == NULL) {
    2475           0 :                 return NULL;
    2476             :         }
    2477             : 
    2478        2795 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    2479        2460 :                 subreq = cli_smb2_rmdir_send(state, ev, cli, dname, NULL);
    2480        2460 :                 if (tevent_req_nomem(subreq, req)) {
    2481           0 :                         return tevent_req_post(req, ev);
    2482             :                 }
    2483        2460 :                 tevent_req_set_callback(subreq, cli_rmdir_done2, req);
    2484        2460 :                 return req;
    2485             :         }
    2486             : 
    2487         335 :         bytes = talloc_array(state, uint8_t, 1);
    2488         335 :         if (tevent_req_nomem(bytes, req)) {
    2489           0 :                 return tevent_req_post(req, ev);
    2490             :         }
    2491             :         /*
    2492             :          * SMBrmdir on a DFS share must use DFS names.
    2493             :          */
    2494         335 :         dname_cp = smb1_dfs_share_path(state, cli, dname);
    2495         335 :         if (tevent_req_nomem(dname_cp, req)) {
    2496           0 :                 return tevent_req_post(req, ev);
    2497             :         }
    2498         335 :         bytes[0] = 4;
    2499         335 :         bytes = smb_bytes_push_str(bytes,
    2500         335 :                                    smbXcli_conn_use_unicode(cli->conn),
    2501             :                                    dname_cp,
    2502         335 :                                    strlen(dname_cp)+1,
    2503             :                                    NULL);
    2504             : 
    2505         335 :         if (tevent_req_nomem(bytes, req)) {
    2506           0 :                 return tevent_req_post(req, ev);
    2507             :         }
    2508             : 
    2509         335 :         if (clistr_is_previous_version_path(dname)) {
    2510           0 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
    2511             :         }
    2512             : 
    2513         335 :         subreq = cli_smb_send(state, ev, cli, SMBrmdir, additional_flags,
    2514             :                         additional_flags2,
    2515         335 :                         0, NULL, talloc_get_size(bytes), bytes);
    2516         335 :         if (tevent_req_nomem(subreq, req)) {
    2517           0 :                 return tevent_req_post(req, ev);
    2518             :         }
    2519         335 :         tevent_req_set_callback(subreq, cli_rmdir_done, req);
    2520         335 :         return req;
    2521             : }
    2522             : 
    2523         335 : static void cli_rmdir_done(struct tevent_req *subreq)
    2524             : {
    2525         335 :         NTSTATUS status = cli_smb_recv(
    2526             :                 subreq, NULL, NULL, 0, NULL, NULL, NULL, NULL);
    2527         335 :         tevent_req_simple_finish_ntstatus(subreq, status);
    2528         335 : }
    2529             : 
    2530        2460 : static void cli_rmdir_done2(struct tevent_req *subreq)
    2531             : {
    2532        2460 :         NTSTATUS status = cli_smb2_rmdir_recv(subreq);
    2533        2460 :         tevent_req_simple_finish_ntstatus(subreq, status);
    2534        2460 : }
    2535             : 
    2536        2795 : NTSTATUS cli_rmdir_recv(struct tevent_req *req)
    2537             : {
    2538        2795 :         return tevent_req_simple_recv_ntstatus(req);
    2539             : }
    2540             : 
    2541        1224 : NTSTATUS cli_rmdir(struct cli_state *cli, const char *dname)
    2542             : {
    2543        1224 :         TALLOC_CTX *frame = NULL;
    2544           0 :         struct tevent_context *ev;
    2545           0 :         struct tevent_req *req;
    2546        1224 :         NTSTATUS status = NT_STATUS_OK;
    2547             : 
    2548        1224 :         frame = talloc_stackframe();
    2549             : 
    2550        1224 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    2551             :                 /*
    2552             :                  * Can't use sync call while an async call is in flight
    2553             :                  */
    2554           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    2555           0 :                 goto fail;
    2556             :         }
    2557             : 
    2558        1224 :         ev = samba_tevent_context_init(frame);
    2559        1224 :         if (ev == NULL) {
    2560           0 :                 status = NT_STATUS_NO_MEMORY;
    2561           0 :                 goto fail;
    2562             :         }
    2563             : 
    2564        1224 :         req = cli_rmdir_send(frame, ev, cli, dname);
    2565        1224 :         if (req == NULL) {
    2566           0 :                 status = NT_STATUS_NO_MEMORY;
    2567           0 :                 goto fail;
    2568             :         }
    2569             : 
    2570        1224 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    2571           0 :                 goto fail;
    2572             :         }
    2573             : 
    2574        1224 :         status = cli_rmdir_recv(req);
    2575        1224 :         cli->raw_status = status; /* cli_smb2_rmdir_recv doesn't set this */
    2576             : 
    2577        1224 :  fail:
    2578        1224 :         TALLOC_FREE(frame);
    2579        1224 :         return status;
    2580             : }
    2581             : 
    2582             : /****************************************************************************
    2583             :  Set or clear the delete on close flag.
    2584             : ****************************************************************************/
    2585             : 
    2586             : struct doc_state {
    2587             :         uint8_t data[1];
    2588             : };
    2589             : 
    2590             : static void cli_nt_delete_on_close_smb1_done(struct tevent_req *subreq);
    2591             : static void cli_nt_delete_on_close_smb2_done(struct tevent_req *subreq);
    2592             : 
    2593        1246 : struct tevent_req *cli_nt_delete_on_close_send(TALLOC_CTX *mem_ctx,
    2594             :                                         struct tevent_context *ev,
    2595             :                                         struct cli_state *cli,
    2596             :                                         uint16_t fnum,
    2597             :                                         bool flag)
    2598             : {
    2599        1246 :         struct tevent_req *req = NULL, *subreq = NULL;
    2600        1246 :         struct doc_state *state = NULL;
    2601             : 
    2602        1246 :         req = tevent_req_create(mem_ctx, &state, struct doc_state);
    2603        1246 :         if (req == NULL) {
    2604           0 :                 return NULL;
    2605             :         }
    2606             : 
    2607        1246 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    2608        1114 :                 subreq = cli_smb2_delete_on_close_send(state, ev, cli,
    2609             :                                                        fnum, flag);
    2610        1114 :                 if (tevent_req_nomem(subreq, req)) {
    2611           0 :                         return tevent_req_post(req, ev);
    2612             :                 }
    2613        1114 :                 tevent_req_set_callback(subreq,
    2614             :                                         cli_nt_delete_on_close_smb2_done,
    2615             :                                         req);
    2616        1114 :                 return req;
    2617             :         }
    2618             : 
    2619             :         /* Setup data array. */
    2620         132 :         SCVAL(&state->data[0], 0, flag ? 1 : 0);
    2621             : 
    2622         132 :         subreq = cli_setfileinfo_send(
    2623             :                 state,
    2624             :                 ev,
    2625             :                 cli,
    2626             :                 fnum,
    2627             :                 SMB_SET_FILE_DISPOSITION_INFO,
    2628         132 :                 state->data,
    2629             :                 sizeof(state->data));
    2630             : 
    2631         132 :         if (tevent_req_nomem(subreq, req)) {
    2632           0 :                 return tevent_req_post(req, ev);
    2633             :         }
    2634         132 :         tevent_req_set_callback(subreq,
    2635             :                                 cli_nt_delete_on_close_smb1_done,
    2636             :                                 req);
    2637         132 :         return req;
    2638             : }
    2639             : 
    2640         132 : static void cli_nt_delete_on_close_smb1_done(struct tevent_req *subreq)
    2641             : {
    2642         132 :         NTSTATUS status = cli_setfileinfo_recv(subreq);
    2643         132 :         tevent_req_simple_finish_ntstatus(subreq, status);
    2644         132 : }
    2645             : 
    2646        1114 : static void cli_nt_delete_on_close_smb2_done(struct tevent_req *subreq)
    2647             : {
    2648        1114 :         NTSTATUS status = cli_smb2_delete_on_close_recv(subreq);
    2649        1114 :         tevent_req_simple_finish_ntstatus(subreq, status);
    2650        1114 : }
    2651             : 
    2652        1246 : NTSTATUS cli_nt_delete_on_close_recv(struct tevent_req *req)
    2653             : {
    2654        1246 :         return tevent_req_simple_recv_ntstatus(req);
    2655             : }
    2656             : 
    2657          88 : NTSTATUS cli_nt_delete_on_close(struct cli_state *cli, uint16_t fnum, bool flag)
    2658             : {
    2659          88 :         TALLOC_CTX *frame = talloc_stackframe();
    2660          88 :         struct tevent_context *ev = NULL;
    2661          88 :         struct tevent_req *req = NULL;
    2662          88 :         NTSTATUS status = NT_STATUS_OK;
    2663             : 
    2664          88 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    2665             :                 /*
    2666             :                  * Can't use sync call while an async call is in flight
    2667             :                  */
    2668           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    2669           0 :                 goto fail;
    2670             :         }
    2671             : 
    2672          88 :         ev = samba_tevent_context_init(frame);
    2673          88 :         if (ev == NULL) {
    2674           0 :                 status = NT_STATUS_NO_MEMORY;
    2675           0 :                 goto fail;
    2676             :         }
    2677             : 
    2678          88 :         req = cli_nt_delete_on_close_send(frame,
    2679             :                                 ev,
    2680             :                                 cli,
    2681             :                                 fnum,
    2682             :                                 flag);
    2683          88 :         if (req == NULL) {
    2684           0 :                 status = NT_STATUS_NO_MEMORY;
    2685           0 :                 goto fail;
    2686             :         }
    2687             : 
    2688          88 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    2689           0 :                 goto fail;
    2690             :         }
    2691             : 
    2692          88 :         status = cli_nt_delete_on_close_recv(req);
    2693             : 
    2694          88 :  fail:
    2695          88 :         TALLOC_FREE(frame);
    2696          88 :         return status;
    2697             : }
    2698             : 
    2699             : struct cli_ntcreate1_state {
    2700             :         uint16_t vwv[24];
    2701             :         uint16_t fnum;
    2702             :         struct smb_create_returns cr;
    2703             :         struct tevent_req *subreq;
    2704             : };
    2705             : 
    2706             : static void cli_ntcreate1_done(struct tevent_req *subreq);
    2707             : static bool cli_ntcreate1_cancel(struct tevent_req *req);
    2708             : 
    2709        3703 : static struct tevent_req *cli_ntcreate1_send(TALLOC_CTX *mem_ctx,
    2710             :                                              struct tevent_context *ev,
    2711             :                                              struct cli_state *cli,
    2712             :                                              const char *fname,
    2713             :                                              uint32_t CreatFlags,
    2714             :                                              uint32_t DesiredAccess,
    2715             :                                              uint32_t FileAttributes,
    2716             :                                              uint32_t ShareAccess,
    2717             :                                              uint32_t CreateDisposition,
    2718             :                                              uint32_t CreateOptions,
    2719             :                                              uint32_t ImpersonationLevel,
    2720             :                                              uint8_t SecurityFlags)
    2721             : {
    2722           0 :         struct tevent_req *req, *subreq;
    2723           0 :         struct cli_ntcreate1_state *state;
    2724           0 :         uint16_t *vwv;
    2725           0 :         uint8_t *bytes;
    2726           0 :         size_t converted_len;
    2727        3703 :         uint16_t additional_flags2 = 0;
    2728        3703 :         char *fname_cp = NULL;
    2729             : 
    2730        3703 :         req = tevent_req_create(mem_ctx, &state, struct cli_ntcreate1_state);
    2731        3703 :         if (req == NULL) {
    2732           0 :                 return NULL;
    2733             :         }
    2734             : 
    2735        3703 :         vwv = state->vwv;
    2736             : 
    2737        3703 :         SCVAL(vwv+0, 0, 0xFF);
    2738        3703 :         SCVAL(vwv+0, 1, 0);
    2739        3703 :         SSVAL(vwv+1, 0, 0);
    2740        3703 :         SCVAL(vwv+2, 0, 0);
    2741             : 
    2742        3703 :         if (cli->use_oplocks) {
    2743           4 :                 CreatFlags |= (REQUEST_OPLOCK|REQUEST_BATCH_OPLOCK);
    2744             :         }
    2745        3703 :         SIVAL(vwv+3, 1, CreatFlags);
    2746        3703 :         SIVAL(vwv+5, 1, 0x0);   /* RootDirectoryFid */
    2747        3703 :         SIVAL(vwv+7, 1, DesiredAccess);
    2748        3703 :         SIVAL(vwv+9, 1, 0x0);   /* AllocationSize */
    2749        3703 :         SIVAL(vwv+11, 1, 0x0);  /* AllocationSize */
    2750        3703 :         SIVAL(vwv+13, 1, FileAttributes);
    2751        3703 :         SIVAL(vwv+15, 1, ShareAccess);
    2752        3703 :         SIVAL(vwv+17, 1, CreateDisposition);
    2753        3703 :         SIVAL(vwv+19, 1, CreateOptions |
    2754             :                 (cli->backup_intent ? FILE_OPEN_FOR_BACKUP_INTENT : 0));
    2755        3703 :         SIVAL(vwv+21, 1, ImpersonationLevel);
    2756        3703 :         SCVAL(vwv+23, 1, SecurityFlags);
    2757             : 
    2758        3703 :         bytes = talloc_array(state, uint8_t, 0);
    2759        3703 :         if (tevent_req_nomem(bytes, req)) {
    2760           0 :                 return tevent_req_post(req, ev);
    2761             :         }
    2762             :         /*
    2763             :          * SMBntcreateX on a DFS share must use DFS names.
    2764             :          */
    2765        3703 :         fname_cp = smb1_dfs_share_path(state, cli, fname);
    2766        3703 :         if (tevent_req_nomem(fname_cp, req)) {
    2767           0 :                 return tevent_req_post(req, ev);
    2768             :         }
    2769        3703 :         bytes = smb_bytes_push_str(bytes,
    2770        3703 :                                    smbXcli_conn_use_unicode(cli->conn),
    2771             :                                    fname_cp,
    2772        3703 :                                    strlen(fname_cp)+1,
    2773             :                                    &converted_len);
    2774        3703 :         if (tevent_req_nomem(bytes, req)) {
    2775           0 :                 return tevent_req_post(req, ev);
    2776             :         }
    2777             : 
    2778        3703 :         if (clistr_is_previous_version_path(fname)) {
    2779         410 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
    2780             :         }
    2781             : 
    2782             :         /* sigh. this copes with broken netapp filer behaviour */
    2783        3703 :         bytes = smb_bytes_push_str(bytes, smbXcli_conn_use_unicode(cli->conn), "", 1, NULL);
    2784             : 
    2785        3703 :         if (tevent_req_nomem(bytes, req)) {
    2786           0 :                 return tevent_req_post(req, ev);
    2787             :         }
    2788             : 
    2789        3703 :         SSVAL(vwv+2, 1, converted_len);
    2790             : 
    2791        3703 :         subreq = cli_smb_send(state, ev, cli, SMBntcreateX, 0,
    2792             :                         additional_flags2, 24, vwv,
    2793        3703 :                         talloc_get_size(bytes), bytes);
    2794        3703 :         if (tevent_req_nomem(subreq, req)) {
    2795           0 :                 return tevent_req_post(req, ev);
    2796             :         }
    2797        3703 :         tevent_req_set_callback(subreq, cli_ntcreate1_done, req);
    2798             : 
    2799        3703 :         state->subreq = subreq;
    2800        3703 :         tevent_req_set_cancel_fn(req, cli_ntcreate1_cancel);
    2801             : 
    2802        3703 :         return req;
    2803             : }
    2804             : 
    2805        3703 : static void cli_ntcreate1_done(struct tevent_req *subreq)
    2806             : {
    2807        3703 :         struct tevent_req *req = tevent_req_callback_data(
    2808             :                 subreq, struct tevent_req);
    2809        3703 :         struct cli_ntcreate1_state *state = tevent_req_data(
    2810             :                 req, struct cli_ntcreate1_state);
    2811           0 :         uint8_t wct;
    2812           0 :         uint16_t *vwv;
    2813           0 :         uint32_t num_bytes;
    2814           0 :         uint8_t *bytes;
    2815           0 :         NTSTATUS status;
    2816             : 
    2817        3703 :         status = cli_smb_recv(subreq, state, NULL, 34, &wct, &vwv,
    2818             :                               &num_bytes, &bytes);
    2819        3703 :         TALLOC_FREE(subreq);
    2820        3703 :         if (tevent_req_nterror(req, status)) {
    2821         490 :                 return;
    2822             :         }
    2823        3213 :         state->cr.oplock_level = CVAL(vwv+2, 0);
    2824        3213 :         state->fnum = SVAL(vwv+2, 1);
    2825        3213 :         state->cr.create_action = IVAL(vwv+3, 1);
    2826        3213 :         state->cr.creation_time = BVAL(vwv+5, 1);
    2827        3213 :         state->cr.last_access_time = BVAL(vwv+9, 1);
    2828        3213 :         state->cr.last_write_time = BVAL(vwv+13, 1);
    2829        3213 :         state->cr.change_time   = BVAL(vwv+17, 1);
    2830        3213 :         state->cr.file_attributes = IVAL(vwv+21, 1);
    2831        3213 :         state->cr.allocation_size = BVAL(vwv+23, 1);
    2832        3213 :         state->cr.end_of_file   = BVAL(vwv+27, 1);
    2833             : 
    2834        3213 :         tevent_req_done(req);
    2835             : }
    2836             : 
    2837           0 : static bool cli_ntcreate1_cancel(struct tevent_req *req)
    2838             : {
    2839           0 :         struct cli_ntcreate1_state *state = tevent_req_data(
    2840             :                 req, struct cli_ntcreate1_state);
    2841           0 :         return tevent_req_cancel(state->subreq);
    2842             : }
    2843             : 
    2844        3703 : static NTSTATUS cli_ntcreate1_recv(struct tevent_req *req,
    2845             :                                    uint16_t *pfnum,
    2846             :                                    struct smb_create_returns *cr)
    2847             : {
    2848        3703 :         struct cli_ntcreate1_state *state = tevent_req_data(
    2849             :                 req, struct cli_ntcreate1_state);
    2850           0 :         NTSTATUS status;
    2851             : 
    2852        3703 :         if (tevent_req_is_nterror(req, &status)) {
    2853         490 :                 return status;
    2854             :         }
    2855        3213 :         *pfnum = state->fnum;
    2856        3213 :         if (cr != NULL) {
    2857        3213 :                 *cr = state->cr;
    2858             :         }
    2859        3213 :         return NT_STATUS_OK;
    2860             : }
    2861             : 
    2862             : struct cli_ntcreate_state {
    2863             :         struct smb_create_returns cr;
    2864             :         uint16_t fnum;
    2865             :         struct tevent_req *subreq;
    2866             : };
    2867             : 
    2868             : static void cli_ntcreate_done_nt1(struct tevent_req *subreq);
    2869             : static void cli_ntcreate_done_smb2(struct tevent_req *subreq);
    2870             : static bool cli_ntcreate_cancel(struct tevent_req *req);
    2871             : 
    2872       23844 : struct tevent_req *cli_ntcreate_send(TALLOC_CTX *mem_ctx,
    2873             :                                      struct tevent_context *ev,
    2874             :                                      struct cli_state *cli,
    2875             :                                      const char *fname,
    2876             :                                      uint32_t create_flags,
    2877             :                                      uint32_t desired_access,
    2878             :                                      uint32_t file_attributes,
    2879             :                                      uint32_t share_access,
    2880             :                                      uint32_t create_disposition,
    2881             :                                      uint32_t create_options,
    2882             :                                      uint32_t impersonation_level,
    2883             :                                      uint8_t security_flags)
    2884             : {
    2885           0 :         struct tevent_req *req, *subreq;
    2886           0 :         struct cli_ntcreate_state *state;
    2887             : 
    2888       23844 :         req = tevent_req_create(mem_ctx, &state, struct cli_ntcreate_state);
    2889       23844 :         if (req == NULL) {
    2890           0 :                 return NULL;
    2891             :         }
    2892             : 
    2893       23844 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    2894       20141 :                 struct cli_smb2_create_flags cflags = {0};
    2895             : 
    2896       20141 :                 if (cli->use_oplocks) {
    2897           4 :                         create_flags |= REQUEST_OPLOCK|REQUEST_BATCH_OPLOCK;
    2898             :                 }
    2899             : 
    2900       20141 :                 cflags = (struct cli_smb2_create_flags) {
    2901       20141 :                         .batch_oplock = (create_flags & REQUEST_BATCH_OPLOCK),
    2902       20141 :                         .exclusive_oplock = (create_flags & REQUEST_OPLOCK),
    2903             :                 };
    2904             : 
    2905       20141 :                 subreq = cli_smb2_create_fnum_send(
    2906             :                         state,
    2907             :                         ev,
    2908             :                         cli,
    2909             :                         fname,
    2910             :                         cflags,
    2911             :                         impersonation_level,
    2912             :                         desired_access,
    2913             :                         file_attributes,
    2914             :                         share_access,
    2915             :                         create_disposition,
    2916             :                         create_options,
    2917             :                         NULL);
    2918       20141 :                 if (tevent_req_nomem(subreq, req)) {
    2919           0 :                         return tevent_req_post(req, ev);
    2920             :                 }
    2921       20141 :                 tevent_req_set_callback(subreq, cli_ntcreate_done_smb2, req);
    2922             :         } else {
    2923        3703 :                 subreq = cli_ntcreate1_send(
    2924             :                         state, ev, cli, fname, create_flags, desired_access,
    2925             :                         file_attributes, share_access, create_disposition,
    2926             :                         create_options, impersonation_level, security_flags);
    2927        3703 :                 if (tevent_req_nomem(subreq, req)) {
    2928           0 :                         return tevent_req_post(req, ev);
    2929             :                 }
    2930        3703 :                 tevent_req_set_callback(subreq, cli_ntcreate_done_nt1, req);
    2931             :         }
    2932             : 
    2933       23844 :         state->subreq = subreq;
    2934       23844 :         tevent_req_set_cancel_fn(req, cli_ntcreate_cancel);
    2935             : 
    2936       23844 :         return req;
    2937             : }
    2938             : 
    2939        3703 : static void cli_ntcreate_done_nt1(struct tevent_req *subreq)
    2940             : {
    2941        3703 :         struct tevent_req *req = tevent_req_callback_data(
    2942             :                 subreq, struct tevent_req);
    2943        3703 :         struct cli_ntcreate_state *state = tevent_req_data(
    2944             :                 req, struct cli_ntcreate_state);
    2945           0 :         NTSTATUS status;
    2946             : 
    2947        3703 :         status = cli_ntcreate1_recv(subreq, &state->fnum, &state->cr);
    2948        3703 :         TALLOC_FREE(subreq);
    2949        3703 :         if (tevent_req_nterror(req, status)) {
    2950         490 :                 return;
    2951             :         }
    2952        3213 :         tevent_req_done(req);
    2953             : }
    2954             : 
    2955       20141 : static void cli_ntcreate_done_smb2(struct tevent_req *subreq)
    2956             : {
    2957       20141 :         struct tevent_req *req = tevent_req_callback_data(
    2958             :                 subreq, struct tevent_req);
    2959       20141 :         struct cli_ntcreate_state *state = tevent_req_data(
    2960             :                 req, struct cli_ntcreate_state);
    2961           0 :         NTSTATUS status;
    2962             : 
    2963       20141 :         status = cli_smb2_create_fnum_recv(
    2964             :                 subreq,
    2965             :                 &state->fnum,
    2966             :                 &state->cr,
    2967             :                 NULL,
    2968             :                 NULL,
    2969             :                 NULL);
    2970       20141 :         TALLOC_FREE(subreq);
    2971       20141 :         if (tevent_req_nterror(req, status)) {
    2972        1988 :                 return;
    2973             :         }
    2974       18153 :         tevent_req_done(req);
    2975             : }
    2976             : 
    2977           2 : static bool cli_ntcreate_cancel(struct tevent_req *req)
    2978             : {
    2979           2 :         struct cli_ntcreate_state *state = tevent_req_data(
    2980             :                 req, struct cli_ntcreate_state);
    2981           2 :         return tevent_req_cancel(state->subreq);
    2982             : }
    2983             : 
    2984       23844 : NTSTATUS cli_ntcreate_recv(struct tevent_req *req, uint16_t *fnum,
    2985             :                            struct smb_create_returns *cr)
    2986             : {
    2987       23844 :         struct cli_ntcreate_state *state = tevent_req_data(
    2988             :                 req, struct cli_ntcreate_state);
    2989           0 :         NTSTATUS status;
    2990             : 
    2991       23844 :         if (tevent_req_is_nterror(req, &status)) {
    2992        2478 :                 return status;
    2993             :         }
    2994       21366 :         if (fnum != NULL) {
    2995       21366 :                 *fnum = state->fnum;
    2996             :         }
    2997       21366 :         if (cr != NULL) {
    2998        3606 :                 *cr = state->cr;
    2999             :         }
    3000       21366 :         return NT_STATUS_OK;
    3001             : }
    3002             : 
    3003       13067 : NTSTATUS cli_ntcreate(struct cli_state *cli,
    3004             :                       const char *fname,
    3005             :                       uint32_t CreatFlags,
    3006             :                       uint32_t DesiredAccess,
    3007             :                       uint32_t FileAttributes,
    3008             :                       uint32_t ShareAccess,
    3009             :                       uint32_t CreateDisposition,
    3010             :                       uint32_t CreateOptions,
    3011             :                       uint8_t SecurityFlags,
    3012             :                       uint16_t *pfid,
    3013             :                       struct smb_create_returns *cr)
    3014             : {
    3015       13067 :         TALLOC_CTX *frame = talloc_stackframe();
    3016           0 :         struct tevent_context *ev;
    3017           0 :         struct tevent_req *req;
    3018       13067 :         uint32_t ImpersonationLevel = SMB2_IMPERSONATION_IMPERSONATION;
    3019       13067 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    3020             : 
    3021       13067 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    3022             :                 /*
    3023             :                  * Can't use sync call while an async call is in flight
    3024             :                  */
    3025           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    3026           0 :                 goto fail;
    3027             :         }
    3028             : 
    3029       13067 :         ev = samba_tevent_context_init(frame);
    3030       13067 :         if (ev == NULL) {
    3031           0 :                 goto fail;
    3032             :         }
    3033             : 
    3034       13067 :         req = cli_ntcreate_send(frame, ev, cli, fname, CreatFlags,
    3035             :                                 DesiredAccess, FileAttributes, ShareAccess,
    3036             :                                 CreateDisposition, CreateOptions,
    3037             :                                 ImpersonationLevel, SecurityFlags);
    3038       13067 :         if (req == NULL) {
    3039           0 :                 goto fail;
    3040             :         }
    3041             : 
    3042       13067 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    3043           0 :                 goto fail;
    3044             :         }
    3045             : 
    3046       13067 :         status = cli_ntcreate_recv(req, pfid, cr);
    3047       13067 :  fail:
    3048       13067 :         TALLOC_FREE(frame);
    3049       13067 :         return status;
    3050             : }
    3051             : 
    3052             : struct cli_nttrans_create_state {
    3053             :         uint16_t fnum;
    3054             :         struct smb_create_returns cr;
    3055             : };
    3056             : 
    3057             : static void cli_nttrans_create_done(struct tevent_req *subreq);
    3058             : 
    3059           4 : struct tevent_req *cli_nttrans_create_send(TALLOC_CTX *mem_ctx,
    3060             :                                            struct tevent_context *ev,
    3061             :                                            struct cli_state *cli,
    3062             :                                            const char *fname,
    3063             :                                            uint32_t CreatFlags,
    3064             :                                            uint32_t DesiredAccess,
    3065             :                                            uint32_t FileAttributes,
    3066             :                                            uint32_t ShareAccess,
    3067             :                                            uint32_t CreateDisposition,
    3068             :                                            uint32_t CreateOptions,
    3069             :                                            uint8_t SecurityFlags,
    3070             :                                            struct security_descriptor *secdesc,
    3071             :                                            struct ea_struct *eas,
    3072             :                                            int num_eas)
    3073             : {
    3074           0 :         struct tevent_req *req, *subreq;
    3075           0 :         struct cli_nttrans_create_state *state;
    3076           0 :         uint8_t *param;
    3077           0 :         uint8_t *secdesc_buf;
    3078           0 :         size_t secdesc_len;
    3079           0 :         NTSTATUS status;
    3080           0 :         size_t converted_len;
    3081           4 :         uint16_t additional_flags2 = 0;
    3082           4 :         char *fname_cp = NULL;
    3083             : 
    3084           4 :         req = tevent_req_create(mem_ctx,
    3085             :                                 &state, struct cli_nttrans_create_state);
    3086           4 :         if (req == NULL) {
    3087           0 :                 return NULL;
    3088             :         }
    3089             : 
    3090           4 :         if (secdesc != NULL) {
    3091           0 :                 status = marshall_sec_desc(talloc_tos(), secdesc,
    3092             :                                            &secdesc_buf, &secdesc_len);
    3093           0 :                 if (tevent_req_nterror(req, status)) {
    3094           0 :                         DEBUG(10, ("marshall_sec_desc failed: %s\n",
    3095             :                                    nt_errstr(status)));
    3096           0 :                         return tevent_req_post(req, ev);
    3097             :                 }
    3098             :         } else {
    3099           4 :                 secdesc_buf = NULL;
    3100           4 :                 secdesc_len = 0;
    3101             :         }
    3102             : 
    3103           4 :         if (num_eas != 0) {
    3104             :                 /*
    3105             :                  * TODO ;-)
    3106             :                  */
    3107           0 :                 tevent_req_nterror(req, NT_STATUS_NOT_IMPLEMENTED);
    3108           0 :                 return tevent_req_post(req, ev);
    3109             :         }
    3110             : 
    3111           4 :         param = talloc_array(state, uint8_t, 53);
    3112           4 :         if (tevent_req_nomem(param, req)) {
    3113           0 :                 return tevent_req_post(req, ev);
    3114             :         }
    3115             : 
    3116             :         /*
    3117             :          * SMBntcreateX on a DFS share must use DFS names.
    3118             :          */
    3119           4 :         fname_cp = smb1_dfs_share_path(state, cli, fname);
    3120           4 :         if (tevent_req_nomem(fname_cp, req)) {
    3121           0 :                 return tevent_req_post(req, ev);
    3122             :         }
    3123           4 :         param = trans2_bytes_push_str(param,
    3124           4 :                                       smbXcli_conn_use_unicode(cli->conn),
    3125             :                                       fname_cp,
    3126             :                                       strlen(fname_cp),
    3127             :                                       &converted_len);
    3128           4 :         if (tevent_req_nomem(param, req)) {
    3129           0 :                 return tevent_req_post(req, ev);
    3130             :         }
    3131             : 
    3132           4 :         if (clistr_is_previous_version_path(fname)) {
    3133           0 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
    3134             :         }
    3135             : 
    3136           4 :         SIVAL(param, 0, CreatFlags);
    3137           4 :         SIVAL(param, 4, 0x0);   /* RootDirectoryFid */
    3138           4 :         SIVAL(param, 8, DesiredAccess);
    3139           4 :         SIVAL(param, 12, 0x0);  /* AllocationSize */
    3140           4 :         SIVAL(param, 16, 0x0);  /* AllocationSize */
    3141           4 :         SIVAL(param, 20, FileAttributes);
    3142           4 :         SIVAL(param, 24, ShareAccess);
    3143           4 :         SIVAL(param, 28, CreateDisposition);
    3144           4 :         SIVAL(param, 32, CreateOptions |
    3145             :                 (cli->backup_intent ? FILE_OPEN_FOR_BACKUP_INTENT : 0));
    3146           4 :         SIVAL(param, 36, secdesc_len);
    3147           4 :         SIVAL(param, 40, 0);     /* EA length*/
    3148           4 :         SIVAL(param, 44, converted_len);
    3149           4 :         SIVAL(param, 48, 0x02); /* ImpersonationLevel */
    3150           4 :         SCVAL(param, 52, SecurityFlags);
    3151             : 
    3152           4 :         subreq = cli_trans_send(state, ev, cli,
    3153             :                                 additional_flags2, /* additional_flags2 */
    3154             :                                 SMBnttrans,
    3155             :                                 NULL, -1, /* name, fid */
    3156             :                                 NT_TRANSACT_CREATE, 0,
    3157             :                                 NULL, 0, 0, /* setup */
    3158           4 :                                 param, talloc_get_size(param), 128, /* param */
    3159             :                                 secdesc_buf, secdesc_len, 0); /* data */
    3160           4 :         if (tevent_req_nomem(subreq, req)) {
    3161           0 :                 return tevent_req_post(req, ev);
    3162             :         }
    3163           4 :         tevent_req_set_callback(subreq, cli_nttrans_create_done, req);
    3164           4 :         return req;
    3165             : }
    3166             : 
    3167           4 : static void cli_nttrans_create_done(struct tevent_req *subreq)
    3168             : {
    3169           4 :         struct tevent_req *req = tevent_req_callback_data(
    3170             :                 subreq, struct tevent_req);
    3171           4 :         struct cli_nttrans_create_state *state = tevent_req_data(
    3172             :                 req, struct cli_nttrans_create_state);
    3173           0 :         uint8_t *param;
    3174           0 :         uint32_t num_param;
    3175           0 :         NTSTATUS status;
    3176             : 
    3177           4 :         status = cli_trans_recv(subreq, talloc_tos(), NULL,
    3178             :                                 NULL, 0, NULL, /* rsetup */
    3179             :                                 &param, 69, &num_param,
    3180             :                                 NULL, 0, NULL);
    3181           4 :         if (tevent_req_nterror(req, status)) {
    3182           0 :                 return;
    3183             :         }
    3184           4 :         state->cr.oplock_level = CVAL(param, 0);
    3185           4 :         state->fnum = SVAL(param, 2);
    3186           4 :         state->cr.create_action = IVAL(param, 4);
    3187           4 :         state->cr.creation_time = BVAL(param, 12);
    3188           4 :         state->cr.last_access_time = BVAL(param, 20);
    3189           4 :         state->cr.last_write_time = BVAL(param, 28);
    3190           4 :         state->cr.change_time   = BVAL(param, 36);
    3191           4 :         state->cr.file_attributes = IVAL(param, 44);
    3192           4 :         state->cr.allocation_size = BVAL(param, 48);
    3193           4 :         state->cr.end_of_file   = BVAL(param, 56);
    3194             : 
    3195           4 :         TALLOC_FREE(param);
    3196           4 :         tevent_req_done(req);
    3197             : }
    3198             : 
    3199           4 : NTSTATUS cli_nttrans_create_recv(struct tevent_req *req,
    3200             :                         uint16_t *fnum,
    3201             :                         struct smb_create_returns *cr)
    3202             : {
    3203           4 :         struct cli_nttrans_create_state *state = tevent_req_data(
    3204             :                 req, struct cli_nttrans_create_state);
    3205           0 :         NTSTATUS status;
    3206             : 
    3207           4 :         if (tevent_req_is_nterror(req, &status)) {
    3208           0 :                 return status;
    3209             :         }
    3210           4 :         *fnum = state->fnum;
    3211           4 :         if (cr != NULL) {
    3212           0 :                 *cr = state->cr;
    3213             :         }
    3214           4 :         return NT_STATUS_OK;
    3215             : }
    3216             : 
    3217           4 : NTSTATUS cli_nttrans_create(struct cli_state *cli,
    3218             :                             const char *fname,
    3219             :                             uint32_t CreatFlags,
    3220             :                             uint32_t DesiredAccess,
    3221             :                             uint32_t FileAttributes,
    3222             :                             uint32_t ShareAccess,
    3223             :                             uint32_t CreateDisposition,
    3224             :                             uint32_t CreateOptions,
    3225             :                             uint8_t SecurityFlags,
    3226             :                             struct security_descriptor *secdesc,
    3227             :                             struct ea_struct *eas,
    3228             :                             int num_eas,
    3229             :                             uint16_t *pfid,
    3230             :                             struct smb_create_returns *cr)
    3231             : {
    3232           4 :         TALLOC_CTX *frame = talloc_stackframe();
    3233           0 :         struct tevent_context *ev;
    3234           0 :         struct tevent_req *req;
    3235           4 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    3236             : 
    3237           4 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    3238             :                 /*
    3239             :                  * Can't use sync call while an async call is in flight
    3240             :                  */
    3241           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    3242           0 :                 goto fail;
    3243             :         }
    3244           4 :         ev = samba_tevent_context_init(frame);
    3245           4 :         if (ev == NULL) {
    3246           0 :                 goto fail;
    3247             :         }
    3248           4 :         req = cli_nttrans_create_send(frame, ev, cli, fname, CreatFlags,
    3249             :                                       DesiredAccess, FileAttributes,
    3250             :                                       ShareAccess, CreateDisposition,
    3251             :                                       CreateOptions, SecurityFlags,
    3252             :                                       secdesc, eas, num_eas);
    3253           4 :         if (req == NULL) {
    3254           0 :                 goto fail;
    3255             :         }
    3256           4 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    3257           0 :                 goto fail;
    3258             :         }
    3259           4 :         status = cli_nttrans_create_recv(req, pfid, cr);
    3260           4 :  fail:
    3261           4 :         TALLOC_FREE(frame);
    3262           4 :         return status;
    3263             : }
    3264             : 
    3265             : /****************************************************************************
    3266             :  Open a file
    3267             :  WARNING: if you open with O_WRONLY then getattrE won't work!
    3268             : ****************************************************************************/
    3269             : 
    3270             : struct cli_openx_state {
    3271             :         const char *fname;
    3272             :         uint16_t vwv[15];
    3273             :         uint16_t fnum;
    3274             :         struct iovec bytes;
    3275             : };
    3276             : 
    3277             : static void cli_openx_done(struct tevent_req *subreq);
    3278             : 
    3279         932 : struct tevent_req *cli_openx_create(TALLOC_CTX *mem_ctx,
    3280             :                                    struct tevent_context *ev,
    3281             :                                    struct cli_state *cli, const char *fname,
    3282             :                                    int flags, int share_mode,
    3283             :                                    struct tevent_req **psmbreq)
    3284             : {
    3285           0 :         struct tevent_req *req, *subreq;
    3286           0 :         struct cli_openx_state *state;
    3287           0 :         unsigned openfn;
    3288           0 :         unsigned accessmode;
    3289           0 :         uint8_t additional_flags;
    3290         932 :         uint16_t additional_flags2 = 0;
    3291           0 :         uint8_t *bytes;
    3292         932 :         char *fname_cp = NULL;
    3293             : 
    3294         932 :         req = tevent_req_create(mem_ctx, &state, struct cli_openx_state);
    3295         932 :         if (req == NULL) {
    3296           0 :                 return NULL;
    3297             :         }
    3298             : 
    3299         932 :         openfn = 0;
    3300         932 :         if (flags & O_CREAT) {
    3301         755 :                 openfn |= (1<<4);
    3302             :         }
    3303         932 :         if (!(flags & O_EXCL)) {
    3304         772 :                 if (flags & O_TRUNC)
    3305          35 :                         openfn |= (1<<1);
    3306             :                 else
    3307         737 :                         openfn |= (1<<0);
    3308             :         }
    3309             : 
    3310         932 :         accessmode = (share_mode<<4);
    3311             : 
    3312         932 :         if ((flags & O_ACCMODE) == O_RDWR) {
    3313         872 :                 accessmode |= 2;
    3314          60 :         } else if ((flags & O_ACCMODE) == O_WRONLY) {
    3315           0 :                 accessmode |= 1;
    3316             :         }
    3317             : 
    3318             : #if defined(O_SYNC)
    3319         932 :         if ((flags & O_SYNC) == O_SYNC) {
    3320           0 :                 accessmode |= (1<<14);
    3321             :         }
    3322             : #endif /* O_SYNC */
    3323             : 
    3324         932 :         if (share_mode == DENY_FCB) {
    3325           0 :                 accessmode = 0xFF;
    3326             :         }
    3327             : 
    3328         932 :         SCVAL(state->vwv + 0, 0, 0xFF);
    3329         932 :         SCVAL(state->vwv + 0, 1, 0);
    3330         932 :         SSVAL(state->vwv + 1, 0, 0);
    3331         932 :         SSVAL(state->vwv + 2, 0, 0);  /* no additional info */
    3332         932 :         SSVAL(state->vwv + 3, 0, accessmode);
    3333         932 :         SSVAL(state->vwv + 4, 0, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    3334         932 :         SSVAL(state->vwv + 5, 0, 0);
    3335         932 :         SIVAL(state->vwv + 6, 0, 0);
    3336         932 :         SSVAL(state->vwv + 8, 0, openfn);
    3337         932 :         SIVAL(state->vwv + 9, 0, 0);
    3338         932 :         SIVAL(state->vwv + 11, 0, 0);
    3339         932 :         SIVAL(state->vwv + 13, 0, 0);
    3340             : 
    3341         932 :         additional_flags = 0;
    3342             : 
    3343         932 :         if (cli->use_oplocks) {
    3344             :                 /* if using oplocks then ask for a batch oplock via
    3345             :                    core and extended methods */
    3346          23 :                 additional_flags =
    3347             :                         FLAG_REQUEST_OPLOCK|FLAG_REQUEST_BATCH_OPLOCK;
    3348          23 :                 SSVAL(state->vwv+2, 0, SVAL(state->vwv+2, 0) | 6);
    3349             :         }
    3350             : 
    3351         932 :         bytes = talloc_array(state, uint8_t, 0);
    3352         932 :         if (tevent_req_nomem(bytes, req)) {
    3353           0 :                 return tevent_req_post(req, ev);
    3354             :         }
    3355             :         /*
    3356             :          * SMBopenX on a DFS share must use DFS names.
    3357             :          */
    3358         932 :         fname_cp = smb1_dfs_share_path(state, cli, fname);
    3359         932 :         if (tevent_req_nomem(fname_cp, req)) {
    3360           0 :                 return tevent_req_post(req, ev);
    3361             :         }
    3362         932 :         bytes = smb_bytes_push_str(bytes,
    3363         932 :                                    smbXcli_conn_use_unicode(cli->conn),
    3364             :                                    fname_cp,
    3365         932 :                                    strlen(fname_cp)+1,
    3366             :                                    NULL);
    3367             : 
    3368         932 :         if (tevent_req_nomem(bytes, req)) {
    3369           0 :                 return tevent_req_post(req, ev);
    3370             :         }
    3371             : 
    3372         932 :         if (clistr_is_previous_version_path(fname)) {
    3373           0 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
    3374             :         }
    3375             : 
    3376         932 :         state->bytes.iov_base = (void *)bytes;
    3377         932 :         state->bytes.iov_len = talloc_get_size(bytes);
    3378             : 
    3379         932 :         subreq = cli_smb_req_create(state, ev, cli, SMBopenX, additional_flags,
    3380         932 :                         additional_flags2, 15, state->vwv, 1, &state->bytes);
    3381         932 :         if (subreq == NULL) {
    3382           0 :                 TALLOC_FREE(req);
    3383           0 :                 return NULL;
    3384             :         }
    3385         932 :         tevent_req_set_callback(subreq, cli_openx_done, req);
    3386         932 :         *psmbreq = subreq;
    3387         932 :         return req;
    3388             : }
    3389             : 
    3390         922 : struct tevent_req *cli_openx_send(TALLOC_CTX *mem_ctx, struct tevent_context *ev,
    3391             :                                  struct cli_state *cli, const char *fname,
    3392             :                                  int flags, int share_mode)
    3393             : {
    3394           0 :         struct tevent_req *req, *subreq;
    3395           0 :         NTSTATUS status;
    3396             : 
    3397         922 :         req = cli_openx_create(mem_ctx, ev, cli, fname, flags, share_mode,
    3398             :                               &subreq);
    3399         922 :         if (req == NULL) {
    3400           0 :                 return NULL;
    3401             :         }
    3402             : 
    3403         922 :         status = smb1cli_req_chain_submit(&subreq, 1);
    3404         922 :         if (tevent_req_nterror(req, status)) {
    3405           0 :                 return tevent_req_post(req, ev);
    3406             :         }
    3407         922 :         return req;
    3408             : }
    3409             : 
    3410         932 : static void cli_openx_done(struct tevent_req *subreq)
    3411             : {
    3412         932 :         struct tevent_req *req = tevent_req_callback_data(
    3413             :                 subreq, struct tevent_req);
    3414         932 :         struct cli_openx_state *state = tevent_req_data(
    3415             :                 req, struct cli_openx_state);
    3416           0 :         uint8_t wct;
    3417           0 :         uint16_t *vwv;
    3418           0 :         NTSTATUS status;
    3419             : 
    3420         932 :         status = cli_smb_recv(subreq, state, NULL, 3, &wct, &vwv, NULL,
    3421             :                               NULL);
    3422         932 :         TALLOC_FREE(subreq);
    3423         932 :         if (tevent_req_nterror(req, status)) {
    3424          49 :                 return;
    3425             :         }
    3426         883 :         state->fnum = SVAL(vwv+2, 0);
    3427         883 :         tevent_req_done(req);
    3428             : }
    3429             : 
    3430         932 : NTSTATUS cli_openx_recv(struct tevent_req *req, uint16_t *pfnum)
    3431             : {
    3432         932 :         struct cli_openx_state *state = tevent_req_data(
    3433             :                 req, struct cli_openx_state);
    3434           0 :         NTSTATUS status;
    3435             : 
    3436         932 :         if (tevent_req_is_nterror(req, &status)) {
    3437          49 :                 return status;
    3438             :         }
    3439         883 :         *pfnum = state->fnum;
    3440         883 :         return NT_STATUS_OK;
    3441             : }
    3442             : 
    3443         917 : NTSTATUS cli_openx(struct cli_state *cli, const char *fname, int flags,
    3444             :              int share_mode, uint16_t *pfnum)
    3445             : {
    3446         917 :         TALLOC_CTX *frame = talloc_stackframe();
    3447           0 :         struct tevent_context *ev;
    3448           0 :         struct tevent_req *req;
    3449         917 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    3450             : 
    3451         917 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    3452             :                 /*
    3453             :                  * Can't use sync call while an async call is in flight
    3454             :                  */
    3455           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    3456           0 :                 goto fail;
    3457             :         }
    3458             : 
    3459         917 :         ev = samba_tevent_context_init(frame);
    3460         917 :         if (ev == NULL) {
    3461           0 :                 goto fail;
    3462             :         }
    3463             : 
    3464         917 :         req = cli_openx_send(frame, ev, cli, fname, flags, share_mode);
    3465         917 :         if (req == NULL) {
    3466           0 :                 goto fail;
    3467             :         }
    3468             : 
    3469         917 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    3470           0 :                 goto fail;
    3471             :         }
    3472             : 
    3473         917 :         status = cli_openx_recv(req, pfnum);
    3474         917 :  fail:
    3475         917 :         TALLOC_FREE(frame);
    3476         917 :         return status;
    3477             : }
    3478             : /****************************************************************************
    3479             :  Synchronous wrapper function that does an NtCreateX open by preference
    3480             :  and falls back to openX if this fails.
    3481             : ****************************************************************************/
    3482             : 
    3483        3692 : NTSTATUS cli_open(struct cli_state *cli, const char *fname, int flags,
    3484             :                         int share_mode_in, uint16_t *pfnum)
    3485             : {
    3486           0 :         NTSTATUS status;
    3487        3692 :         unsigned int openfn = 0;
    3488        3692 :         unsigned int dos_deny = 0;
    3489           0 :         uint32_t access_mask, share_mode, create_disposition, create_options;
    3490        3692 :         struct smb_create_returns cr = {0};
    3491             : 
    3492             :         /* Do the initial mapping into OpenX parameters. */
    3493        3692 :         if (flags & O_CREAT) {
    3494        1637 :                 openfn |= (1<<4);
    3495             :         }
    3496        3692 :         if (!(flags & O_EXCL)) {
    3497        3692 :                 if (flags & O_TRUNC)
    3498        1633 :                         openfn |= (1<<1);
    3499             :                 else
    3500        2059 :                         openfn |= (1<<0);
    3501             :         }
    3502             : 
    3503        3692 :         dos_deny = (share_mode_in<<4);
    3504             : 
    3505        3692 :         if ((flags & O_ACCMODE) == O_RDWR) {
    3506        1217 :                 dos_deny |= 2;
    3507        2475 :         } else if ((flags & O_ACCMODE) == O_WRONLY) {
    3508         420 :                 dos_deny |= 1;
    3509             :         }
    3510             : 
    3511             : #if defined(O_SYNC)
    3512        3692 :         if ((flags & O_SYNC) == O_SYNC) {
    3513           0 :                 dos_deny |= (1<<14);
    3514             :         }
    3515             : #endif /* O_SYNC */
    3516             : 
    3517        3692 :         if (share_mode_in == DENY_FCB) {
    3518           0 :                 dos_deny = 0xFF;
    3519             :         }
    3520             : 
    3521        3692 :         if (!map_open_params_to_ntcreate(fname, dos_deny,
    3522             :                                         openfn, &access_mask,
    3523             :                                         &share_mode, &create_disposition,
    3524             :                                         &create_options, NULL)) {
    3525           0 :                 goto try_openx;
    3526             :         }
    3527             : 
    3528        3692 :         status = cli_ntcreate(cli,
    3529             :                                 fname,
    3530             :                                 0,
    3531             :                                 access_mask,
    3532             :                                 0,
    3533             :                                 share_mode,
    3534             :                                 create_disposition,
    3535             :                                 create_options,
    3536             :                                 0,
    3537             :                                 pfnum,
    3538             :                                 &cr);
    3539             : 
    3540             :         /* Try and cope will all variants of "we don't do this call"
    3541             :            and fall back to openX. */
    3542             : 
    3543        3692 :         if (NT_STATUS_EQUAL(status,NT_STATUS_NOT_IMPLEMENTED) ||
    3544        3692 :                         NT_STATUS_EQUAL(status,NT_STATUS_INVALID_INFO_CLASS) ||
    3545        3692 :                         NT_STATUS_EQUAL(status,NT_STATUS_PROCEDURE_NOT_FOUND) ||
    3546        3692 :                         NT_STATUS_EQUAL(status,NT_STATUS_INVALID_LEVEL) ||
    3547        3692 :                         NT_STATUS_EQUAL(status,NT_STATUS_INVALID_PARAMETER) ||
    3548        3692 :                         NT_STATUS_EQUAL(status,NT_STATUS_INVALID_DEVICE_REQUEST) ||
    3549        3692 :                         NT_STATUS_EQUAL(status,NT_STATUS_INVALID_DEVICE_STATE) ||
    3550        3692 :                         NT_STATUS_EQUAL(status,NT_STATUS_CTL_FILE_NOT_SUPPORTED) ||
    3551        3692 :                         NT_STATUS_EQUAL(status,NT_STATUS_UNSUCCESSFUL)) {
    3552           0 :                 goto try_openx;
    3553             :         }
    3554             : 
    3555        3692 :         if (NT_STATUS_IS_OK(status) &&
    3556        2844 :             (create_options & FILE_NON_DIRECTORY_FILE) &&
    3557        2844 :             (cr.file_attributes & FILE_ATTRIBUTE_DIRECTORY))
    3558             :         {
    3559             :                 /*
    3560             :                  * Some (broken) servers return a valid handle
    3561             :                  * for directories even if FILE_NON_DIRECTORY_FILE
    3562             :                  * is set. Just close the handle and set the
    3563             :                  * error explicitly to NT_STATUS_FILE_IS_A_DIRECTORY.
    3564             :                  */
    3565           0 :                 status = cli_close(cli, *pfnum);
    3566           0 :                 if (!NT_STATUS_IS_OK(status)) {
    3567           0 :                         return status;
    3568             :                 }
    3569           0 :                 status = NT_STATUS_FILE_IS_A_DIRECTORY;
    3570             :                 /* Set this so libsmbclient can retrieve it. */
    3571           0 :                 cli->raw_status = status;
    3572             :         }
    3573             : 
    3574        3692 :         return status;
    3575             : 
    3576           0 :   try_openx:
    3577             : 
    3578           0 :         return cli_openx(cli, fname, flags, share_mode_in, pfnum);
    3579             : }
    3580             : 
    3581             : /****************************************************************************
    3582             :  Close a file.
    3583             : ****************************************************************************/
    3584             : 
    3585             : struct cli_smb1_close_state {
    3586             :         uint16_t vwv[3];
    3587             : };
    3588             : 
    3589             : static void cli_smb1_close_done(struct tevent_req *subreq);
    3590             : 
    3591        4174 : struct tevent_req *cli_smb1_close_create(TALLOC_CTX *mem_ctx,
    3592             :                                 struct tevent_context *ev,
    3593             :                                 struct cli_state *cli,
    3594             :                                 uint16_t fnum,
    3595             :                                 struct tevent_req **psubreq)
    3596             : {
    3597           0 :         struct tevent_req *req, *subreq;
    3598           0 :         struct cli_smb1_close_state *state;
    3599             : 
    3600        4174 :         req = tevent_req_create(mem_ctx, &state, struct cli_smb1_close_state);
    3601        4174 :         if (req == NULL) {
    3602           0 :                 return NULL;
    3603             :         }
    3604             : 
    3605        4174 :         SSVAL(state->vwv+0, 0, fnum);
    3606        4174 :         SIVALS(state->vwv+1, 0, -1);
    3607             : 
    3608        4174 :         subreq = cli_smb_req_create(state, ev, cli, SMBclose, 0, 0,
    3609        4174 :                                 3, state->vwv, 0, NULL);
    3610        4174 :         if (subreq == NULL) {
    3611           0 :                 TALLOC_FREE(req);
    3612           0 :                 return NULL;
    3613             :         }
    3614        4174 :         tevent_req_set_callback(subreq, cli_smb1_close_done, req);
    3615        4174 :         *psubreq = subreq;
    3616        4174 :         return req;
    3617             : }
    3618             : 
    3619        4174 : static void cli_smb1_close_done(struct tevent_req *subreq)
    3620             : {
    3621        4174 :         struct tevent_req *req = tevent_req_callback_data(
    3622             :                 subreq, struct tevent_req);
    3623           0 :         NTSTATUS status;
    3624             : 
    3625        4174 :         status = cli_smb_recv(subreq, NULL, NULL, 0, NULL, NULL, NULL, NULL);
    3626        4174 :         TALLOC_FREE(subreq);
    3627        4174 :         if (tevent_req_nterror(req, status)) {
    3628          43 :                 return;
    3629             :         }
    3630        4131 :         tevent_req_done(req);
    3631             : }
    3632             : 
    3633             : struct cli_close_state {
    3634             :         int dummy;
    3635             : };
    3636             : 
    3637             : static void cli_close_done(struct tevent_req *subreq);
    3638             : 
    3639       24437 : struct tevent_req *cli_close_send(TALLOC_CTX *mem_ctx,
    3640             :                                   struct tevent_context *ev,
    3641             :                                   struct cli_state *cli,
    3642             :                                   uint16_t fnum,
    3643             :                                   uint16_t flags)
    3644             : {
    3645           0 :         struct tevent_req *req, *subreq;
    3646           0 :         struct cli_close_state *state;
    3647           0 :         NTSTATUS status;
    3648             : 
    3649       24437 :         req = tevent_req_create(mem_ctx, &state, struct cli_close_state);
    3650       24437 :         if (req == NULL) {
    3651           0 :                 return NULL;
    3652             :         }
    3653             : 
    3654       24437 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    3655       20273 :                 subreq = cli_smb2_close_fnum_send(state, ev, cli, fnum, flags);
    3656       20273 :                 if (tevent_req_nomem(subreq, req)) {
    3657           0 :                         return tevent_req_post(req, ev);
    3658             :                 }
    3659             :         } else {
    3660        4164 :                 struct tevent_req *ch_req = NULL;
    3661        4164 :                 subreq = cli_smb1_close_create(state, ev, cli, fnum, &ch_req);
    3662        4164 :                 if (tevent_req_nomem(subreq, req)) {
    3663           0 :                         return tevent_req_post(req, ev);
    3664             :                 }
    3665        4164 :                 status = smb1cli_req_chain_submit(&ch_req, 1);
    3666        4164 :                 if (tevent_req_nterror(req, status)) {
    3667           0 :                         return tevent_req_post(req, ev);
    3668             :                 }
    3669             :         }
    3670             : 
    3671       24437 :         tevent_req_set_callback(subreq, cli_close_done, req);
    3672       24437 :         return req;
    3673             : }
    3674             : 
    3675       24437 : static void cli_close_done(struct tevent_req *subreq)
    3676             : {
    3677       24437 :         struct tevent_req *req = tevent_req_callback_data(
    3678             :                 subreq, struct tevent_req);
    3679       24437 :         NTSTATUS status = NT_STATUS_OK;
    3680       24437 :         bool err = tevent_req_is_nterror(subreq, &status);
    3681             : 
    3682       24437 :         TALLOC_FREE(subreq);
    3683       24437 :         if (err) {
    3684          55 :                 tevent_req_nterror(req, status);
    3685          55 :                 return;
    3686             :         }
    3687       24382 :         tevent_req_done(req);
    3688             : }
    3689             : 
    3690       24447 : NTSTATUS cli_close_recv(struct tevent_req *req)
    3691             : {
    3692       24447 :         return tevent_req_simple_recv_ntstatus(req);
    3693             : }
    3694             : 
    3695       13067 : NTSTATUS cli_close(struct cli_state *cli, uint16_t fnum)
    3696             : {
    3697       13067 :         TALLOC_CTX *frame = NULL;
    3698           0 :         struct tevent_context *ev;
    3699           0 :         struct tevent_req *req;
    3700       13067 :         NTSTATUS status = NT_STATUS_OK;
    3701             : 
    3702       13067 :         frame = talloc_stackframe();
    3703             : 
    3704       13067 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    3705             :                 /*
    3706             :                  * Can't use sync call while an async call is in flight
    3707             :                  */
    3708           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    3709           0 :                 goto fail;
    3710             :         }
    3711             : 
    3712       13067 :         ev = samba_tevent_context_init(frame);
    3713       13067 :         if (ev == NULL) {
    3714           0 :                 status = NT_STATUS_NO_MEMORY;
    3715           0 :                 goto fail;
    3716             :         }
    3717             : 
    3718       13067 :         req = cli_close_send(frame, ev, cli, fnum, 0);
    3719       13067 :         if (req == NULL) {
    3720           0 :                 status = NT_STATUS_NO_MEMORY;
    3721           0 :                 goto fail;
    3722             :         }
    3723             : 
    3724       13067 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    3725           0 :                 goto fail;
    3726             :         }
    3727             : 
    3728       13067 :         status = cli_close_recv(req);
    3729       13067 :  fail:
    3730       13067 :         TALLOC_FREE(frame);
    3731       13067 :         return status;
    3732             : }
    3733             : 
    3734             : /****************************************************************************
    3735             :  Truncate a file to a specified size
    3736             : ****************************************************************************/
    3737             : 
    3738             : struct ftrunc_state {
    3739             :         uint8_t data[8];
    3740             : };
    3741             : 
    3742          12 : static void cli_ftruncate_done(struct tevent_req *subreq)
    3743             : {
    3744          12 :         NTSTATUS status = cli_setfileinfo_recv(subreq);
    3745          12 :         tevent_req_simple_finish_ntstatus(subreq, status);
    3746          12 : }
    3747             : 
    3748          12 : struct tevent_req *cli_ftruncate_send(TALLOC_CTX *mem_ctx,
    3749             :                                         struct tevent_context *ev,
    3750             :                                         struct cli_state *cli,
    3751             :                                         uint16_t fnum,
    3752             :                                         uint64_t size)
    3753             : {
    3754          12 :         struct tevent_req *req = NULL, *subreq = NULL;
    3755          12 :         struct ftrunc_state *state = NULL;
    3756             : 
    3757          12 :         req = tevent_req_create(mem_ctx, &state, struct ftrunc_state);
    3758          12 :         if (req == NULL) {
    3759           0 :                 return NULL;
    3760             :         }
    3761             : 
    3762             :         /* Setup data array. */
    3763          12 :         SBVAL(state->data, 0, size);
    3764             : 
    3765          12 :         subreq = cli_setfileinfo_send(
    3766             :                 state,
    3767             :                 ev,
    3768             :                 cli,
    3769             :                 fnum,
    3770             :                 SMB_SET_FILE_END_OF_FILE_INFO,
    3771          12 :                 state->data,
    3772             :                 sizeof(state->data));
    3773             : 
    3774          12 :         if (tevent_req_nomem(subreq, req)) {
    3775           0 :                 return tevent_req_post(req, ev);
    3776             :         }
    3777          12 :         tevent_req_set_callback(subreq, cli_ftruncate_done, req);
    3778          12 :         return req;
    3779             : }
    3780             : 
    3781          12 : NTSTATUS cli_ftruncate_recv(struct tevent_req *req)
    3782             : {
    3783          12 :         return tevent_req_simple_recv_ntstatus(req);
    3784             : }
    3785             : 
    3786          62 : NTSTATUS cli_ftruncate(struct cli_state *cli, uint16_t fnum, uint64_t size)
    3787             : {
    3788          62 :         TALLOC_CTX *frame = NULL;
    3789          62 :         struct tevent_context *ev = NULL;
    3790          62 :         struct tevent_req *req = NULL;
    3791          62 :         NTSTATUS status = NT_STATUS_OK;
    3792             : 
    3793          62 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    3794          50 :                 return cli_smb2_ftruncate(cli, fnum, size);
    3795             :         }
    3796             : 
    3797          12 :         frame = talloc_stackframe();
    3798             : 
    3799          12 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    3800             :                 /*
    3801             :                  * Can't use sync call while an async call is in flight
    3802             :                  */
    3803           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    3804           0 :                 goto fail;
    3805             :         }
    3806             : 
    3807          12 :         ev = samba_tevent_context_init(frame);
    3808          12 :         if (ev == NULL) {
    3809           0 :                 status = NT_STATUS_NO_MEMORY;
    3810           0 :                 goto fail;
    3811             :         }
    3812             : 
    3813          12 :         req = cli_ftruncate_send(frame,
    3814             :                                 ev,
    3815             :                                 cli,
    3816             :                                 fnum,
    3817             :                                 size);
    3818          12 :         if (req == NULL) {
    3819           0 :                 status = NT_STATUS_NO_MEMORY;
    3820           0 :                 goto fail;
    3821             :         }
    3822             : 
    3823          12 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    3824           0 :                 goto fail;
    3825             :         }
    3826             : 
    3827          12 :         status = cli_ftruncate_recv(req);
    3828             : 
    3829          12 :  fail:
    3830          12 :         TALLOC_FREE(frame);
    3831          12 :         return status;
    3832             : }
    3833             : 
    3834        9022 : static uint8_t *cli_lockingx_put_locks(
    3835             :         uint8_t *buf,
    3836             :         bool large,
    3837             :         uint16_t num_locks,
    3838             :         const struct smb1_lock_element *locks)
    3839             : {
    3840           0 :         uint16_t i;
    3841             : 
    3842       13519 :         for (i=0; i<num_locks; i++) {
    3843        4497 :                 const struct smb1_lock_element *e = &locks[i];
    3844        4497 :                 if (large) {
    3845           0 :                         SSVAL(buf, 0, e->pid);
    3846           0 :                         SSVAL(buf, 2, 0);
    3847           0 :                         SOFF_T_R(buf, 4, e->offset);
    3848           0 :                         SOFF_T_R(buf, 12, e->length);
    3849           0 :                         buf += 20;
    3850             :                 } else {
    3851        4497 :                         SSVAL(buf, 0, e->pid);
    3852        4497 :                         SIVAL(buf, 2, e->offset);
    3853        4497 :                         SIVAL(buf, 6, e->length);
    3854        4497 :                         buf += 10;
    3855             :                 }
    3856             :         }
    3857        9022 :         return buf;
    3858             : }
    3859             : 
    3860             : struct cli_lockingx_state {
    3861             :         uint16_t vwv[8];
    3862             :         struct iovec bytes;
    3863             :         struct tevent_req *subreq;
    3864             : };
    3865             : 
    3866             : static void cli_lockingx_done(struct tevent_req *subreq);
    3867             : static bool cli_lockingx_cancel(struct tevent_req *req);
    3868             : 
    3869        4511 : struct tevent_req *cli_lockingx_create(
    3870             :         TALLOC_CTX *mem_ctx,
    3871             :         struct tevent_context *ev,
    3872             :         struct cli_state *cli,
    3873             :         uint16_t fnum,
    3874             :         uint8_t typeoflock,
    3875             :         uint8_t newoplocklevel,
    3876             :         int32_t timeout,
    3877             :         uint16_t num_unlocks,
    3878             :         const struct smb1_lock_element *unlocks,
    3879             :         uint16_t num_locks,
    3880             :         const struct smb1_lock_element *locks,
    3881             :         struct tevent_req **psmbreq)
    3882             : {
    3883        4511 :         struct tevent_req *req = NULL, *subreq = NULL;
    3884        4511 :         struct cli_lockingx_state *state = NULL;
    3885           0 :         uint16_t *vwv;
    3886           0 :         uint8_t *p;
    3887        4511 :         const bool large = (typeoflock & LOCKING_ANDX_LARGE_FILES);
    3888        4511 :         const size_t element_len = large ? 20 : 10;
    3889             : 
    3890             :         /* uint16->size_t, no overflow */
    3891        4511 :         const size_t num_elements = (size_t)num_locks + (size_t)num_unlocks;
    3892             : 
    3893             :         /* at most 20*2*65535 = 2621400, no overflow */
    3894        4511 :         const size_t num_bytes = num_elements * element_len;
    3895             : 
    3896        4511 :         req = tevent_req_create(mem_ctx, &state, struct cli_lockingx_state);
    3897        4511 :         if (req == NULL) {
    3898           0 :                 return NULL;
    3899             :         }
    3900        4511 :         vwv = state->vwv;
    3901             : 
    3902        4511 :         SCVAL(vwv + 0, 0, 0xFF);
    3903        4511 :         SCVAL(vwv + 0, 1, 0);
    3904        4511 :         SSVAL(vwv + 1, 0, 0);
    3905        4511 :         SSVAL(vwv + 2, 0, fnum);
    3906        4511 :         SCVAL(vwv + 3, 0, typeoflock);
    3907        4511 :         SCVAL(vwv + 3, 1, newoplocklevel);
    3908        4511 :         SIVALS(vwv + 4, 0, timeout);
    3909        4511 :         SSVAL(vwv + 6, 0, num_unlocks);
    3910        4511 :         SSVAL(vwv + 7, 0, num_locks);
    3911             : 
    3912        4511 :         state->bytes.iov_len = num_bytes;
    3913        4511 :         state->bytes.iov_base = talloc_array(state, uint8_t, num_bytes);
    3914        4511 :         if (tevent_req_nomem(state->bytes.iov_base, req)) {
    3915           0 :                 return tevent_req_post(req, ev);
    3916             :         }
    3917             : 
    3918        4511 :         p = cli_lockingx_put_locks(
    3919        4511 :                 state->bytes.iov_base, large, num_unlocks, unlocks);
    3920        4511 :         cli_lockingx_put_locks(p, large, num_locks, locks);
    3921             : 
    3922        4511 :         subreq = cli_smb_req_create(
    3923        4511 :                 state, ev, cli, SMBlockingX, 0, 0, 8, vwv, 1, &state->bytes);
    3924        4511 :         if (tevent_req_nomem(subreq, req)) {
    3925           0 :                 return tevent_req_post(req, ev);
    3926             :         }
    3927        4511 :         tevent_req_set_callback(subreq, cli_lockingx_done, req);
    3928        4511 :         *psmbreq = subreq;
    3929        4511 :         return req;
    3930             : }
    3931             : 
    3932        4501 : struct tevent_req *cli_lockingx_send(
    3933             :         TALLOC_CTX *mem_ctx,
    3934             :         struct tevent_context *ev,
    3935             :         struct cli_state *cli,
    3936             :         uint16_t fnum,
    3937             :         uint8_t typeoflock,
    3938             :         uint8_t newoplocklevel,
    3939             :         int32_t timeout,
    3940             :         uint16_t num_unlocks,
    3941             :         const struct smb1_lock_element *unlocks,
    3942             :         uint16_t num_locks,
    3943             :         const struct smb1_lock_element *locks)
    3944             : {
    3945        4501 :         struct tevent_req *req = NULL, *subreq = NULL;
    3946        4501 :         struct cli_lockingx_state *state = NULL;
    3947           0 :         NTSTATUS status;
    3948             : 
    3949        4501 :         req = cli_lockingx_create(
    3950             :                 mem_ctx,
    3951             :                 ev,
    3952             :                 cli,
    3953             :                 fnum,
    3954             :                 typeoflock,
    3955             :                 newoplocklevel,
    3956             :                 timeout,
    3957             :                 num_unlocks,
    3958             :                 unlocks,
    3959             :                 num_locks,
    3960             :                 locks,
    3961             :                 &subreq);
    3962        4501 :         if (req == NULL) {
    3963           0 :                 return NULL;
    3964             :         }
    3965        4501 :         state = tevent_req_data(req, struct cli_lockingx_state);
    3966        4501 :         state->subreq = subreq;
    3967             : 
    3968        4501 :         status = smb1cli_req_chain_submit(&subreq, 1);
    3969        4501 :         if (tevent_req_nterror(req, status)) {
    3970           0 :                 return tevent_req_post(req, ev);
    3971             :         }
    3972        4501 :         tevent_req_set_cancel_fn(req, cli_lockingx_cancel);
    3973        4501 :         return req;
    3974             : }
    3975             : 
    3976        4511 : static void cli_lockingx_done(struct tevent_req *subreq)
    3977             : {
    3978        4511 :         NTSTATUS status = cli_smb_recv(
    3979             :                 subreq, NULL, NULL, 0, NULL, NULL, NULL, NULL);
    3980        4511 :         tevent_req_simple_finish_ntstatus(subreq, status);
    3981        4511 : }
    3982             : 
    3983           5 : static bool cli_lockingx_cancel(struct tevent_req *req)
    3984             : {
    3985           5 :         struct cli_lockingx_state *state = tevent_req_data(
    3986             :                 req, struct cli_lockingx_state);
    3987           5 :         if (state->subreq == NULL) {
    3988           0 :                 return false;
    3989             :         }
    3990           5 :         return tevent_req_cancel(state->subreq);
    3991             : }
    3992             : 
    3993        4511 : NTSTATUS cli_lockingx_recv(struct tevent_req *req)
    3994             : {
    3995        4511 :         return tevent_req_simple_recv_ntstatus(req);
    3996             : }
    3997             : 
    3998        3369 : NTSTATUS cli_lockingx(
    3999             :         struct cli_state *cli,
    4000             :         uint16_t fnum,
    4001             :         uint8_t typeoflock,
    4002             :         uint8_t newoplocklevel,
    4003             :         int32_t timeout,
    4004             :         uint16_t num_unlocks,
    4005             :         const struct smb1_lock_element *unlocks,
    4006             :         uint16_t num_locks,
    4007             :         const struct smb1_lock_element *locks)
    4008             : {
    4009        3369 :         TALLOC_CTX *frame = talloc_stackframe();
    4010        3369 :         struct tevent_context *ev = NULL;
    4011        3369 :         struct tevent_req *req = NULL;
    4012        3369 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    4013        3369 :         unsigned int set_timeout = 0;
    4014        3369 :         unsigned int saved_timeout = 0;
    4015             : 
    4016        3369 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    4017           0 :                 return NT_STATUS_INVALID_PARAMETER;
    4018             :         }
    4019        3369 :         ev = samba_tevent_context_init(frame);
    4020        3369 :         if (ev == NULL) {
    4021           0 :                 goto fail;
    4022             :         }
    4023             : 
    4024        3369 :         if (timeout != 0) {
    4025          13 :                 if (timeout == -1) {
    4026           4 :                         set_timeout = 0x7FFFFFFF;
    4027             :                 } else {
    4028           9 :                         set_timeout = timeout + 2*1000;
    4029             :                 }
    4030          13 :                 saved_timeout = cli_set_timeout(cli, set_timeout);
    4031             :         }
    4032             : 
    4033        3369 :         req = cli_lockingx_send(
    4034             :                 frame,
    4035             :                 ev,
    4036             :                 cli,
    4037             :                 fnum,
    4038             :                 typeoflock,
    4039             :                 newoplocklevel,
    4040             :                 timeout,
    4041             :                 num_unlocks,
    4042             :                 unlocks,
    4043             :                 num_locks,
    4044             :                 locks);
    4045        3369 :         if (req == NULL) {
    4046           0 :                 goto fail;
    4047             :         }
    4048        3369 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    4049           0 :                 goto fail;
    4050             :         }
    4051        3369 :         status = cli_lockingx_recv(req);
    4052             : 
    4053        3369 :         if (saved_timeout != 0) {
    4054          13 :                 cli_set_timeout(cli, saved_timeout);
    4055             :         }
    4056        3356 : fail:
    4057        3369 :         TALLOC_FREE(frame);
    4058        3369 :         return status;
    4059             : }
    4060             : 
    4061             : /****************************************************************************
    4062             :  send a lock with a specified locktype
    4063             :  this is used for testing LOCKING_ANDX_CANCEL_LOCK
    4064             : ****************************************************************************/
    4065             : 
    4066        3364 : NTSTATUS cli_locktype(struct cli_state *cli, uint16_t fnum,
    4067             :                       uint32_t offset, uint32_t len,
    4068             :                       int timeout, unsigned char locktype)
    4069             : {
    4070        3364 :         struct smb1_lock_element lck = {
    4071        3364 :                 .pid = cli_getpid(cli),
    4072             :                 .offset = offset,
    4073             :                 .length = len,
    4074             :         };
    4075           0 :         NTSTATUS status;
    4076             : 
    4077        3364 :         status = cli_lockingx(
    4078             :                 cli,                            /* cli */
    4079             :                 fnum,                           /* fnum */
    4080             :                 locktype,                       /* typeoflock */
    4081             :                 0,                              /* newoplocklevel */
    4082             :                 timeout,                        /* timeout */
    4083             :                 0,                              /* num_unlocks */
    4084             :                 NULL,                           /* unlocks */
    4085             :                 1,                              /* num_locks */
    4086             :                 &lck);                              /* locks */
    4087        3364 :         return status;
    4088             : }
    4089             : 
    4090             : /****************************************************************************
    4091             :  Lock a file.
    4092             :  note that timeout is in units of 2 milliseconds
    4093             : ****************************************************************************/
    4094             : 
    4095        3339 : NTSTATUS cli_lock32(struct cli_state *cli, uint16_t fnum,
    4096             :                   uint32_t offset, uint32_t len, int timeout,
    4097             :                   enum brl_type lock_type)
    4098             : {
    4099           0 :         NTSTATUS status;
    4100             : 
    4101        3339 :         status = cli_locktype(cli, fnum, offset, len, timeout,
    4102             :                               (lock_type == READ_LOCK? 1 : 0));
    4103        3339 :         return status;
    4104             : }
    4105             : 
    4106             : /****************************************************************************
    4107             :  Unlock a file.
    4108             : ****************************************************************************/
    4109             : 
    4110             : struct cli_unlock_state {
    4111             :         struct smb1_lock_element lck;
    4112             : };
    4113             : 
    4114             : static void cli_unlock_done(struct tevent_req *subreq);
    4115             : 
    4116        1118 : struct tevent_req *cli_unlock_send(TALLOC_CTX *mem_ctx,
    4117             :                                 struct tevent_context *ev,
    4118             :                                 struct cli_state *cli,
    4119             :                                 uint16_t fnum,
    4120             :                                 uint64_t offset,
    4121             :                                 uint64_t len)
    4122             : 
    4123             : {
    4124        1118 :         struct tevent_req *req = NULL, *subreq = NULL;
    4125        1118 :         struct cli_unlock_state *state = NULL;
    4126             : 
    4127        1118 :         req = tevent_req_create(mem_ctx, &state, struct cli_unlock_state);
    4128        1118 :         if (req == NULL) {
    4129           0 :                 return NULL;
    4130             :         }
    4131        2236 :         state->lck = (struct smb1_lock_element) {
    4132        1118 :                 .pid = cli_getpid(cli),
    4133             :                 .offset = offset,
    4134             :                 .length = len,
    4135             :         };
    4136             : 
    4137        1118 :         subreq = cli_lockingx_send(
    4138             :                 state,                          /* mem_ctx */
    4139             :                 ev,                             /* tevent_context */
    4140             :                 cli,                            /* cli */
    4141             :                 fnum,                           /* fnum */
    4142             :                 0,                              /* typeoflock */
    4143             :                 0,                              /* newoplocklevel */
    4144             :                 0,                              /* timeout */
    4145             :                 1,                              /* num_unlocks */
    4146        1118 :                 &state->lck,                     /* unlocks */
    4147             :                 0,                              /* num_locks */
    4148             :                 NULL);                          /* locks */
    4149        1118 :         if (tevent_req_nomem(subreq, req)) {
    4150           0 :                 return tevent_req_post(req, ev);
    4151             :         }
    4152        1118 :         tevent_req_set_callback(subreq, cli_unlock_done, req);
    4153        1118 :         return req;
    4154             : }
    4155             : 
    4156        1118 : static void cli_unlock_done(struct tevent_req *subreq)
    4157             : {
    4158        1118 :         NTSTATUS status = cli_lockingx_recv(subreq);
    4159        1118 :         tevent_req_simple_finish_ntstatus(subreq, status);
    4160        1118 : }
    4161             : 
    4162        1118 : NTSTATUS cli_unlock_recv(struct tevent_req *req)
    4163             : {
    4164        1118 :         return tevent_req_simple_recv_ntstatus(req);
    4165             : }
    4166             : 
    4167        1118 : NTSTATUS cli_unlock(struct cli_state *cli,
    4168             :                         uint16_t fnum,
    4169             :                         uint32_t offset,
    4170             :                         uint32_t len)
    4171             : {
    4172        1118 :         TALLOC_CTX *frame = talloc_stackframe();
    4173           0 :         struct tevent_context *ev;
    4174           0 :         struct tevent_req *req;
    4175        1118 :         NTSTATUS status = NT_STATUS_OK;
    4176             : 
    4177        1118 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    4178             :                 /*
    4179             :                  * Can't use sync call while an async call is in flight
    4180             :                  */
    4181           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    4182           0 :                 goto fail;
    4183             :         }
    4184             : 
    4185        1118 :         ev = samba_tevent_context_init(frame);
    4186        1118 :         if (ev == NULL) {
    4187           0 :                 status = NT_STATUS_NO_MEMORY;
    4188           0 :                 goto fail;
    4189             :         }
    4190             : 
    4191        1118 :         req = cli_unlock_send(frame, ev, cli,
    4192             :                         fnum, offset, len);
    4193        1118 :         if (req == NULL) {
    4194           0 :                 status = NT_STATUS_NO_MEMORY;
    4195           0 :                 goto fail;
    4196             :         }
    4197             : 
    4198        1118 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    4199           0 :                 goto fail;
    4200             :         }
    4201             : 
    4202        1118 :         status = cli_unlock_recv(req);
    4203             : 
    4204        1118 :  fail:
    4205        1118 :         TALLOC_FREE(frame);
    4206        1118 :         return status;
    4207             : }
    4208             : 
    4209             : /****************************************************************************
    4210             :  Get/unlock a POSIX lock on a file - internal function.
    4211             : ****************************************************************************/
    4212             : 
    4213             : struct posix_lock_state {
    4214             :         uint16_t setup;
    4215             :         uint8_t param[4];
    4216             :         uint8_t data[POSIX_LOCK_DATA_SIZE];
    4217             : };
    4218             : 
    4219          36 : static void cli_posix_unlock_internal_done(struct tevent_req *subreq)
    4220             : {
    4221          36 :         NTSTATUS status = cli_trans_recv(subreq, NULL, NULL, NULL, 0, NULL,
    4222             :                                          NULL, 0, NULL, NULL, 0, NULL);
    4223          36 :         tevent_req_simple_finish_ntstatus(subreq, status);
    4224          36 : }
    4225             : 
    4226          36 : static struct tevent_req *cli_posix_lock_internal_send(TALLOC_CTX *mem_ctx,
    4227             :                                         struct tevent_context *ev,
    4228             :                                         struct cli_state *cli,
    4229             :                                         uint16_t fnum,
    4230             :                                         uint64_t offset,
    4231             :                                         uint64_t len,
    4232             :                                         bool wait_lock,
    4233             :                                         enum brl_type lock_type)
    4234             : {
    4235          36 :         struct tevent_req *req = NULL, *subreq = NULL;
    4236          36 :         struct posix_lock_state *state = NULL;
    4237             : 
    4238          36 :         req = tevent_req_create(mem_ctx, &state, struct posix_lock_state);
    4239          36 :         if (req == NULL) {
    4240           0 :                 return NULL;
    4241             :         }
    4242             : 
    4243             :         /* Setup setup word. */
    4244          36 :         SSVAL(&state->setup, 0, TRANSACT2_SETFILEINFO);
    4245             : 
    4246             :         /* Setup param array. */
    4247          36 :         SSVAL(&state->param, 0, fnum);
    4248          36 :         SSVAL(&state->param, 2, SMB_SET_POSIX_LOCK);
    4249             : 
    4250             :         /* Setup data array. */
    4251          36 :         switch (lock_type) {
    4252           4 :                 case READ_LOCK:
    4253           4 :                         SSVAL(&state->data, POSIX_LOCK_TYPE_OFFSET,
    4254             :                                 POSIX_LOCK_TYPE_READ);
    4255           4 :                         break;
    4256          24 :                 case WRITE_LOCK:
    4257          24 :                         SSVAL(&state->data, POSIX_LOCK_TYPE_OFFSET,
    4258             :                                 POSIX_LOCK_TYPE_WRITE);
    4259          24 :                         break;
    4260           8 :                 case UNLOCK_LOCK:
    4261           8 :                         SSVAL(&state->data, POSIX_LOCK_TYPE_OFFSET,
    4262             :                                 POSIX_LOCK_TYPE_UNLOCK);
    4263           8 :                         break;
    4264           0 :                 default:
    4265           0 :                         return NULL;
    4266             :         }
    4267             : 
    4268          36 :         if (wait_lock) {
    4269           8 :                 SSVAL(&state->data, POSIX_LOCK_FLAGS_OFFSET,
    4270             :                                 POSIX_LOCK_FLAG_WAIT);
    4271             :         } else {
    4272          28 :                 SSVAL(state->data, POSIX_LOCK_FLAGS_OFFSET,
    4273             :                                 POSIX_LOCK_FLAG_NOWAIT);
    4274             :         }
    4275             : 
    4276          36 :         SIVAL(&state->data, POSIX_LOCK_PID_OFFSET, cli_getpid(cli));
    4277          36 :         SOFF_T(&state->data, POSIX_LOCK_START_OFFSET, offset);
    4278          36 :         SOFF_T(&state->data, POSIX_LOCK_LEN_OFFSET, len);
    4279             : 
    4280          36 :         subreq = cli_trans_send(state,                  /* mem ctx. */
    4281             :                                 ev,                     /* event ctx. */
    4282             :                                 cli,                    /* cli_state. */
    4283             :                                 0,                      /* additional_flags2 */
    4284             :                                 SMBtrans2,              /* cmd. */
    4285             :                                 NULL,                   /* pipe name. */
    4286             :                                 -1,                     /* fid. */
    4287             :                                 0,                      /* function. */
    4288             :                                 0,                      /* flags. */
    4289          36 :                                 &state->setup,          /* setup. */
    4290             :                                 1,                      /* num setup uint16_t words. */
    4291             :                                 0,                      /* max returned setup. */
    4292          36 :                                 state->param,           /* param. */
    4293             :                                 4,                      /* num param. */
    4294             :                                 2,                      /* max returned param. */
    4295          36 :                                 state->data,            /* data. */
    4296             :                                 POSIX_LOCK_DATA_SIZE,   /* num data. */
    4297             :                                 0);                     /* max returned data. */
    4298             : 
    4299          36 :         if (tevent_req_nomem(subreq, req)) {
    4300           0 :                 return tevent_req_post(req, ev);
    4301             :         }
    4302          36 :         tevent_req_set_callback(subreq, cli_posix_unlock_internal_done, req);
    4303          36 :         return req;
    4304             : }
    4305             : 
    4306             : /****************************************************************************
    4307             :  POSIX Lock a file.
    4308             : ****************************************************************************/
    4309             : 
    4310          32 : struct tevent_req *cli_posix_lock_send(TALLOC_CTX *mem_ctx,
    4311             :                                         struct tevent_context *ev,
    4312             :                                         struct cli_state *cli,
    4313             :                                         uint16_t fnum,
    4314             :                                         uint64_t offset,
    4315             :                                         uint64_t len,
    4316             :                                         bool wait_lock,
    4317             :                                         enum brl_type lock_type)
    4318             : {
    4319          32 :         return cli_posix_lock_internal_send(mem_ctx, ev, cli, fnum, offset, len,
    4320             :                                         wait_lock, lock_type);
    4321             : }
    4322             : 
    4323          32 : NTSTATUS cli_posix_lock_recv(struct tevent_req *req)
    4324             : {
    4325          32 :         return tevent_req_simple_recv_ntstatus(req);
    4326             : }
    4327             : 
    4328          20 : NTSTATUS cli_posix_lock(struct cli_state *cli, uint16_t fnum,
    4329             :                         uint64_t offset, uint64_t len,
    4330             :                         bool wait_lock, enum brl_type lock_type)
    4331             : {
    4332          20 :         TALLOC_CTX *frame = talloc_stackframe();
    4333          20 :         struct tevent_context *ev = NULL;
    4334          20 :         struct tevent_req *req = NULL;
    4335          20 :         NTSTATUS status = NT_STATUS_OK;
    4336             : 
    4337          20 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    4338             :                 /*
    4339             :                  * Can't use sync call while an async call is in flight
    4340             :                  */
    4341           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    4342           0 :                 goto fail;
    4343             :         }
    4344             : 
    4345          20 :         if (lock_type != READ_LOCK && lock_type != WRITE_LOCK) {
    4346           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    4347           0 :                 goto fail;
    4348             :         }
    4349             : 
    4350          20 :         ev = samba_tevent_context_init(frame);
    4351          20 :         if (ev == NULL) {
    4352           0 :                 status = NT_STATUS_NO_MEMORY;
    4353           0 :                 goto fail;
    4354             :         }
    4355             : 
    4356          20 :         req = cli_posix_lock_send(frame,
    4357             :                                 ev,
    4358             :                                 cli,
    4359             :                                 fnum,
    4360             :                                 offset,
    4361             :                                 len,
    4362             :                                 wait_lock,
    4363             :                                 lock_type);
    4364          20 :         if (req == NULL) {
    4365           0 :                 status = NT_STATUS_NO_MEMORY;
    4366           0 :                 goto fail;
    4367             :         }
    4368             : 
    4369          20 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    4370           0 :                 goto fail;
    4371             :         }
    4372             : 
    4373          20 :         status = cli_posix_lock_recv(req);
    4374             : 
    4375          20 :  fail:
    4376          20 :         TALLOC_FREE(frame);
    4377          20 :         return status;
    4378             : }
    4379             : 
    4380             : /****************************************************************************
    4381             :  POSIX Unlock a file.
    4382             : ****************************************************************************/
    4383             : 
    4384           4 : struct tevent_req *cli_posix_unlock_send(TALLOC_CTX *mem_ctx,
    4385             :                                         struct tevent_context *ev,
    4386             :                                         struct cli_state *cli,
    4387             :                                         uint16_t fnum,
    4388             :                                         uint64_t offset,
    4389             :                                         uint64_t len)
    4390             : {
    4391           4 :         return cli_posix_lock_internal_send(mem_ctx, ev, cli, fnum, offset, len,
    4392             :                                         false, UNLOCK_LOCK);
    4393             : }
    4394             : 
    4395           4 : NTSTATUS cli_posix_unlock_recv(struct tevent_req *req)
    4396             : {
    4397           4 :         return tevent_req_simple_recv_ntstatus(req);
    4398             : }
    4399             : 
    4400           4 : NTSTATUS cli_posix_unlock(struct cli_state *cli, uint16_t fnum, uint64_t offset, uint64_t len)
    4401             : {
    4402           4 :         TALLOC_CTX *frame = talloc_stackframe();
    4403           4 :         struct tevent_context *ev = NULL;
    4404           4 :         struct tevent_req *req = NULL;
    4405           4 :         NTSTATUS status = NT_STATUS_OK;
    4406             : 
    4407           4 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    4408             :                 /*
    4409             :                  * Can't use sync call while an async call is in flight
    4410             :                  */
    4411           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    4412           0 :                 goto fail;
    4413             :         }
    4414             : 
    4415           4 :         ev = samba_tevent_context_init(frame);
    4416           4 :         if (ev == NULL) {
    4417           0 :                 status = NT_STATUS_NO_MEMORY;
    4418           0 :                 goto fail;
    4419             :         }
    4420             : 
    4421           4 :         req = cli_posix_unlock_send(frame,
    4422             :                                 ev,
    4423             :                                 cli,
    4424             :                                 fnum,
    4425             :                                 offset,
    4426             :                                 len);
    4427           4 :         if (req == NULL) {
    4428           0 :                 status = NT_STATUS_NO_MEMORY;
    4429           0 :                 goto fail;
    4430             :         }
    4431             : 
    4432           4 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    4433           0 :                 goto fail;
    4434             :         }
    4435             : 
    4436           4 :         status = cli_posix_unlock_recv(req);
    4437             : 
    4438           4 :  fail:
    4439           4 :         TALLOC_FREE(frame);
    4440           4 :         return status;
    4441             : }
    4442             : 
    4443             : /****************************************************************************
    4444             :  Do a SMBgetattrE call.
    4445             : ****************************************************************************/
    4446             : 
    4447             : static void cli_getattrE_done(struct tevent_req *subreq);
    4448             : 
    4449             : struct cli_getattrE_state {
    4450             :         uint16_t vwv[1];
    4451             :         int zone_offset;
    4452             :         uint32_t attr;
    4453             :         off_t size;
    4454             :         time_t change_time;
    4455             :         time_t access_time;
    4456             :         time_t write_time;
    4457             : };
    4458             : 
    4459           0 : struct tevent_req *cli_getattrE_send(TALLOC_CTX *mem_ctx,
    4460             :                                 struct tevent_context *ev,
    4461             :                                 struct cli_state *cli,
    4462             :                                 uint16_t fnum)
    4463             : {
    4464           0 :         struct tevent_req *req = NULL, *subreq = NULL;
    4465           0 :         struct cli_getattrE_state *state = NULL;
    4466           0 :         uint8_t additional_flags = 0;
    4467             : 
    4468           0 :         req = tevent_req_create(mem_ctx, &state, struct cli_getattrE_state);
    4469           0 :         if (req == NULL) {
    4470           0 :                 return NULL;
    4471             :         }
    4472             : 
    4473           0 :         state->zone_offset = smb1cli_conn_server_time_zone(cli->conn);
    4474           0 :         SSVAL(state->vwv+0,0,fnum);
    4475             : 
    4476           0 :         subreq = cli_smb_send(state, ev, cli, SMBgetattrE, additional_flags, 0,
    4477           0 :                               1, state->vwv, 0, NULL);
    4478           0 :         if (tevent_req_nomem(subreq, req)) {
    4479           0 :                 return tevent_req_post(req, ev);
    4480             :         }
    4481           0 :         tevent_req_set_callback(subreq, cli_getattrE_done, req);
    4482           0 :         return req;
    4483             : }
    4484             : 
    4485           0 : static void cli_getattrE_done(struct tevent_req *subreq)
    4486             : {
    4487           0 :         struct tevent_req *req = tevent_req_callback_data(
    4488             :                 subreq, struct tevent_req);
    4489           0 :         struct cli_getattrE_state *state = tevent_req_data(
    4490             :                 req, struct cli_getattrE_state);
    4491           0 :         uint8_t wct;
    4492           0 :         uint16_t *vwv = NULL;
    4493           0 :         NTSTATUS status;
    4494             : 
    4495           0 :         status = cli_smb_recv(subreq, state, NULL, 11, &wct, &vwv,
    4496             :                               NULL, NULL);
    4497           0 :         TALLOC_FREE(subreq);
    4498           0 :         if (tevent_req_nterror(req, status)) {
    4499           0 :                 return;
    4500             :         }
    4501             : 
    4502           0 :         state->size = (off_t)IVAL(vwv+6,0);
    4503           0 :         state->attr = SVAL(vwv+10,0);
    4504           0 :         state->change_time = make_unix_date2(vwv+0, state->zone_offset);
    4505           0 :         state->access_time = make_unix_date2(vwv+2, state->zone_offset);
    4506           0 :         state->write_time = make_unix_date2(vwv+4, state->zone_offset);
    4507             : 
    4508           0 :         tevent_req_done(req);
    4509             : }
    4510             : 
    4511           0 : NTSTATUS cli_getattrE_recv(struct tevent_req *req,
    4512             :                         uint32_t *pattr,
    4513             :                         off_t *size,
    4514             :                         time_t *change_time,
    4515             :                         time_t *access_time,
    4516             :                         time_t *write_time)
    4517             : {
    4518           0 :         struct cli_getattrE_state *state = tevent_req_data(
    4519             :                                 req, struct cli_getattrE_state);
    4520           0 :         NTSTATUS status;
    4521             : 
    4522           0 :         if (tevent_req_is_nterror(req, &status)) {
    4523           0 :                 return status;
    4524             :         }
    4525           0 :         if (pattr) {
    4526           0 :                 *pattr = state->attr;
    4527             :         }
    4528           0 :         if (size) {
    4529           0 :                 *size = state->size;
    4530             :         }
    4531           0 :         if (change_time) {
    4532           0 :                 *change_time = state->change_time;
    4533             :         }
    4534           0 :         if (access_time) {
    4535           0 :                 *access_time = state->access_time;
    4536             :         }
    4537           0 :         if (write_time) {
    4538           0 :                 *write_time = state->write_time;
    4539             :         }
    4540           0 :         return NT_STATUS_OK;
    4541             : }
    4542             : 
    4543             : /****************************************************************************
    4544             :  Do a SMBgetatr call
    4545             : ****************************************************************************/
    4546             : 
    4547             : static void cli_getatr_done(struct tevent_req *subreq);
    4548             : 
    4549             : struct cli_getatr_state {
    4550             :         int zone_offset;
    4551             :         uint32_t attr;
    4552             :         off_t size;
    4553             :         time_t write_time;
    4554             : };
    4555             : 
    4556        1093 : struct tevent_req *cli_getatr_send(TALLOC_CTX *mem_ctx,
    4557             :                                 struct tevent_context *ev,
    4558             :                                 struct cli_state *cli,
    4559             :                                 const char *fname)
    4560             : {
    4561        1093 :         struct tevent_req *req = NULL, *subreq = NULL;
    4562        1093 :         struct cli_getatr_state *state = NULL;
    4563        1093 :         uint8_t additional_flags = 0;
    4564        1093 :         uint16_t additional_flags2 = 0;
    4565        1093 :         uint8_t *bytes = NULL;
    4566        1093 :         char *fname_cp = NULL;
    4567             : 
    4568        1093 :         req = tevent_req_create(mem_ctx, &state, struct cli_getatr_state);
    4569        1093 :         if (req == NULL) {
    4570           0 :                 return NULL;
    4571             :         }
    4572             : 
    4573        1093 :         state->zone_offset = smb1cli_conn_server_time_zone(cli->conn);
    4574             : 
    4575        1093 :         bytes = talloc_array(state, uint8_t, 1);
    4576        1093 :         if (tevent_req_nomem(bytes, req)) {
    4577           0 :                 return tevent_req_post(req, ev);
    4578             :         }
    4579             :         /*
    4580             :          * SMBgetatr on a DFS share must use DFS names.
    4581             :          */
    4582        1093 :         fname_cp = smb1_dfs_share_path(state, cli, fname);
    4583        1093 :         if (tevent_req_nomem(fname_cp, req)) {
    4584           0 :                 return tevent_req_post(req, ev);
    4585             :         }
    4586        1093 :         bytes[0] = 4;
    4587        1093 :         bytes = smb_bytes_push_str(bytes,
    4588        1093 :                                    smbXcli_conn_use_unicode(cli->conn),
    4589             :                                    fname_cp,
    4590        1093 :                                    strlen(fname_cp)+1,
    4591             :                                    NULL);
    4592             : 
    4593        1093 :         if (tevent_req_nomem(bytes, req)) {
    4594           0 :                 return tevent_req_post(req, ev);
    4595             :         }
    4596             : 
    4597        1093 :         if (clistr_is_previous_version_path(fname)) {
    4598           0 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
    4599             :         }
    4600             : 
    4601        1093 :         subreq = cli_smb_send(state, ev, cli, SMBgetatr, additional_flags,
    4602             :                         additional_flags2,
    4603        1093 :                         0, NULL, talloc_get_size(bytes), bytes);
    4604        1093 :         if (tevent_req_nomem(subreq, req)) {
    4605           0 :                 return tevent_req_post(req, ev);
    4606             :         }
    4607        1093 :         tevent_req_set_callback(subreq, cli_getatr_done, req);
    4608        1093 :         return req;
    4609             : }
    4610             : 
    4611        1093 : static void cli_getatr_done(struct tevent_req *subreq)
    4612             : {
    4613        1093 :         struct tevent_req *req = tevent_req_callback_data(
    4614             :                 subreq, struct tevent_req);
    4615        1093 :         struct cli_getatr_state *state = tevent_req_data(
    4616             :                 req, struct cli_getatr_state);
    4617           0 :         uint8_t wct;
    4618        1093 :         uint16_t *vwv = NULL;
    4619           0 :         NTSTATUS status;
    4620             : 
    4621        1093 :         status = cli_smb_recv(subreq, state, NULL, 4, &wct, &vwv, NULL,
    4622             :                               NULL);
    4623        1093 :         TALLOC_FREE(subreq);
    4624        1093 :         if (tevent_req_nterror(req, status)) {
    4625           4 :                 return;
    4626             :         }
    4627             : 
    4628        1089 :         state->attr = SVAL(vwv+0,0);
    4629        1089 :         state->size = (off_t)IVAL(vwv+3,0);
    4630        1089 :         state->write_time = make_unix_date3(vwv+1, state->zone_offset);
    4631             : 
    4632        1089 :         tevent_req_done(req);
    4633             : }
    4634             : 
    4635        1093 : NTSTATUS cli_getatr_recv(struct tevent_req *req,
    4636             :                         uint32_t *pattr,
    4637             :                         off_t *size,
    4638             :                         time_t *write_time)
    4639             : {
    4640        1093 :         struct cli_getatr_state *state = tevent_req_data(
    4641             :                                 req, struct cli_getatr_state);
    4642           0 :         NTSTATUS status;
    4643             : 
    4644        1093 :         if (tevent_req_is_nterror(req, &status)) {
    4645           4 :                 return status;
    4646             :         }
    4647        1089 :         if (pattr) {
    4648        1065 :                 *pattr = state->attr;
    4649             :         }
    4650        1089 :         if (size) {
    4651          10 :                 *size = state->size;
    4652             :         }
    4653        1089 :         if (write_time) {
    4654          14 :                 *write_time = state->write_time;
    4655             :         }
    4656        1089 :         return NT_STATUS_OK;
    4657             : }
    4658             : 
    4659        2273 : NTSTATUS cli_getatr(struct cli_state *cli,
    4660             :                         const char *fname,
    4661             :                         uint32_t *pattr,
    4662             :                         off_t *size,
    4663             :                         time_t *write_time)
    4664             : {
    4665        2273 :         TALLOC_CTX *frame = NULL;
    4666        2273 :         struct tevent_context *ev = NULL;
    4667        2273 :         struct tevent_req *req = NULL;
    4668        2273 :         NTSTATUS status = NT_STATUS_OK;
    4669             : 
    4670        2273 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    4671        1180 :                 struct stat_ex sbuf = {
    4672             :                         .st_ex_nlink = 0,
    4673             :                 };
    4674           0 :                 uint32_t attr;
    4675             : 
    4676        1180 :                 status = cli_smb2_qpathinfo_basic(cli, fname, &sbuf, &attr);
    4677        1180 :                 if (!NT_STATUS_IS_OK(status)) {
    4678           0 :                         return status;
    4679             :                 }
    4680             : 
    4681        1180 :                 if (pattr != NULL) {
    4682        1180 :                         *pattr = attr;
    4683             :                 }
    4684        1180 :                 if (size != NULL) {
    4685           0 :                         *size = sbuf.st_ex_size;
    4686             :                 }
    4687        1180 :                 if (write_time != NULL) {
    4688           0 :                         *write_time = sbuf.st_ex_mtime.tv_sec;
    4689             :                 }
    4690        1180 :                 return NT_STATUS_OK;
    4691             :         }
    4692             : 
    4693        1093 :         frame = talloc_stackframe();
    4694             : 
    4695        1093 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    4696             :                 /*
    4697             :                  * Can't use sync call while an async call is in flight
    4698             :                  */
    4699           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    4700           0 :                 goto fail;
    4701             :         }
    4702             : 
    4703        1093 :         ev = samba_tevent_context_init(frame);
    4704        1093 :         if (ev == NULL) {
    4705           0 :                 status = NT_STATUS_NO_MEMORY;
    4706           0 :                 goto fail;
    4707             :         }
    4708             : 
    4709        1093 :         req = cli_getatr_send(frame, ev, cli, fname);
    4710        1093 :         if (req == NULL) {
    4711           0 :                 status = NT_STATUS_NO_MEMORY;
    4712           0 :                 goto fail;
    4713             :         }
    4714             : 
    4715        1093 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    4716           0 :                 goto fail;
    4717             :         }
    4718             : 
    4719        1093 :         status = cli_getatr_recv(req,
    4720             :                                 pattr,
    4721             :                                 size,
    4722             :                                 write_time);
    4723             : 
    4724        1093 :  fail:
    4725        1093 :         TALLOC_FREE(frame);
    4726        1093 :         return status;
    4727             : }
    4728             : 
    4729             : /****************************************************************************
    4730             :  Do a SMBsetattrE call.
    4731             : ****************************************************************************/
    4732             : 
    4733             : static void cli_setattrE_done(struct tevent_req *subreq);
    4734             : 
    4735             : struct cli_setattrE_state {
    4736             :         uint16_t vwv[7];
    4737             : };
    4738             : 
    4739           0 : struct tevent_req *cli_setattrE_send(TALLOC_CTX *mem_ctx,
    4740             :                                 struct tevent_context *ev,
    4741             :                                 struct cli_state *cli,
    4742             :                                 uint16_t fnum,
    4743             :                                 time_t change_time,
    4744             :                                 time_t access_time,
    4745             :                                 time_t write_time)
    4746             : {
    4747           0 :         struct tevent_req *req = NULL, *subreq = NULL;
    4748           0 :         struct cli_setattrE_state *state = NULL;
    4749           0 :         uint8_t additional_flags = 0;
    4750             : 
    4751           0 :         req = tevent_req_create(mem_ctx, &state, struct cli_setattrE_state);
    4752           0 :         if (req == NULL) {
    4753           0 :                 return NULL;
    4754             :         }
    4755             : 
    4756           0 :         SSVAL(state->vwv+0, 0, fnum);
    4757           0 :         push_dos_date2((uint8_t *)&state->vwv[1], 0, change_time,
    4758             :                        smb1cli_conn_server_time_zone(cli->conn));
    4759           0 :         push_dos_date2((uint8_t *)&state->vwv[3], 0, access_time,
    4760             :                        smb1cli_conn_server_time_zone(cli->conn));
    4761           0 :         push_dos_date2((uint8_t *)&state->vwv[5], 0, write_time,
    4762             :                        smb1cli_conn_server_time_zone(cli->conn));
    4763             : 
    4764           0 :         subreq = cli_smb_send(state, ev, cli, SMBsetattrE, additional_flags, 0,
    4765           0 :                               7, state->vwv, 0, NULL);
    4766           0 :         if (tevent_req_nomem(subreq, req)) {
    4767           0 :                 return tevent_req_post(req, ev);
    4768             :         }
    4769           0 :         tevent_req_set_callback(subreq, cli_setattrE_done, req);
    4770           0 :         return req;
    4771             : }
    4772             : 
    4773           0 : static void cli_setattrE_done(struct tevent_req *subreq)
    4774             : {
    4775           0 :         struct tevent_req *req = tevent_req_callback_data(
    4776             :                 subreq, struct tevent_req);
    4777           0 :         NTSTATUS status;
    4778             : 
    4779           0 :         status = cli_smb_recv(subreq, NULL, NULL, 0, NULL, NULL, NULL, NULL);
    4780           0 :         TALLOC_FREE(subreq);
    4781           0 :         if (tevent_req_nterror(req, status)) {
    4782           0 :                 return;
    4783             :         }
    4784           0 :         tevent_req_done(req);
    4785             : }
    4786             : 
    4787           0 : NTSTATUS cli_setattrE_recv(struct tevent_req *req)
    4788             : {
    4789           0 :         return tevent_req_simple_recv_ntstatus(req);
    4790             : }
    4791             : 
    4792           0 : NTSTATUS cli_setattrE(struct cli_state *cli,
    4793             :                         uint16_t fnum,
    4794             :                         time_t change_time,
    4795             :                         time_t access_time,
    4796             :                         time_t write_time)
    4797             : {
    4798           0 :         TALLOC_CTX *frame = NULL;
    4799           0 :         struct tevent_context *ev = NULL;
    4800           0 :         struct tevent_req *req = NULL;
    4801           0 :         NTSTATUS status = NT_STATUS_OK;
    4802             : 
    4803           0 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    4804           0 :                 return cli_smb2_setattrE(cli,
    4805             :                                         fnum,
    4806             :                                         change_time,
    4807             :                                         access_time,
    4808             :                                         write_time);
    4809             :         }
    4810             : 
    4811           0 :         frame = talloc_stackframe();
    4812             : 
    4813           0 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    4814             :                 /*
    4815             :                  * Can't use sync call while an async call is in flight
    4816             :                  */
    4817           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    4818           0 :                 goto fail;
    4819             :         }
    4820             : 
    4821           0 :         ev = samba_tevent_context_init(frame);
    4822           0 :         if (ev == NULL) {
    4823           0 :                 status = NT_STATUS_NO_MEMORY;
    4824           0 :                 goto fail;
    4825             :         }
    4826             : 
    4827           0 :         req = cli_setattrE_send(frame, ev,
    4828             :                         cli,
    4829             :                         fnum,
    4830             :                         change_time,
    4831             :                         access_time,
    4832             :                         write_time);
    4833             : 
    4834           0 :         if (req == NULL) {
    4835           0 :                 status = NT_STATUS_NO_MEMORY;
    4836           0 :                 goto fail;
    4837             :         }
    4838             : 
    4839           0 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    4840           0 :                 goto fail;
    4841             :         }
    4842             : 
    4843           0 :         status = cli_setattrE_recv(req);
    4844             : 
    4845           0 :  fail:
    4846           0 :         TALLOC_FREE(frame);
    4847           0 :         return status;
    4848             : }
    4849             : 
    4850             : /****************************************************************************
    4851             :  Do a SMBsetatr call.
    4852             : ****************************************************************************/
    4853             : 
    4854             : static void cli_setatr_done(struct tevent_req *subreq);
    4855             : 
    4856             : struct cli_setatr_state {
    4857             :         uint16_t vwv[8];
    4858             : };
    4859             : 
    4860        1338 : struct tevent_req *cli_setatr_send(TALLOC_CTX *mem_ctx,
    4861             :                                 struct tevent_context *ev,
    4862             :                                 struct cli_state *cli,
    4863             :                                 const char *fname,
    4864             :                                 uint32_t attr,
    4865             :                                 time_t mtime)
    4866             : {
    4867        1338 :         struct tevent_req *req = NULL, *subreq = NULL;
    4868        1338 :         struct cli_setatr_state *state = NULL;
    4869        1338 :         uint8_t additional_flags = 0;
    4870        1338 :         uint16_t additional_flags2 = 0;
    4871        1338 :         uint8_t *bytes = NULL;
    4872        1338 :         char *fname_cp = NULL;
    4873             : 
    4874        1338 :         req = tevent_req_create(mem_ctx, &state, struct cli_setatr_state);
    4875        1338 :         if (req == NULL) {
    4876           0 :                 return NULL;
    4877             :         }
    4878             : 
    4879        1338 :         if (attr & 0xFFFF0000) {
    4880             :                 /*
    4881             :                  * Don't allow attributes greater than
    4882             :                  * 16-bits for a 16-bit protocol value.
    4883             :                  */
    4884           5 :                 if (tevent_req_nterror(req, NT_STATUS_INVALID_PARAMETER)) {
    4885           5 :                         return tevent_req_post(req, ev);
    4886             :                 }
    4887             :         }
    4888             : 
    4889        1333 :         SSVAL(state->vwv+0, 0, attr);
    4890        1333 :         push_dos_date3((uint8_t *)&state->vwv[1], 0, mtime, smb1cli_conn_server_time_zone(cli->conn));
    4891             : 
    4892        1333 :         bytes = talloc_array(state, uint8_t, 1);
    4893        1333 :         if (tevent_req_nomem(bytes, req)) {
    4894           0 :                 return tevent_req_post(req, ev);
    4895             :         }
    4896             :         /*
    4897             :          * SMBsetatr on a DFS share must use DFS names.
    4898             :          */
    4899        1333 :         fname_cp = smb1_dfs_share_path(state, cli, fname);
    4900        1333 :         if (tevent_req_nomem(fname_cp, req)) {
    4901           0 :                 return tevent_req_post(req, ev);
    4902             :         }
    4903        1333 :         bytes[0] = 4;
    4904        1333 :         bytes = smb_bytes_push_str(bytes,
    4905        1333 :                                    smbXcli_conn_use_unicode(cli->conn),
    4906             :                                    fname_cp,
    4907        1333 :                                    strlen(fname_cp)+1,
    4908             :                                    NULL);
    4909        1333 :         if (tevent_req_nomem(bytes, req)) {
    4910           0 :                 return tevent_req_post(req, ev);
    4911             :         }
    4912        1333 :         bytes = talloc_realloc(state, bytes, uint8_t,
    4913             :                         talloc_get_size(bytes)+1);
    4914        1333 :         if (tevent_req_nomem(bytes, req)) {
    4915           0 :                 return tevent_req_post(req, ev);
    4916             :         }
    4917             : 
    4918        1333 :         bytes[talloc_get_size(bytes)-1] = 4;
    4919        1333 :         bytes = smb_bytes_push_str(bytes, smbXcli_conn_use_unicode(cli->conn), "",
    4920             :                                    1, NULL);
    4921        1333 :         if (tevent_req_nomem(bytes, req)) {
    4922           0 :                 return tevent_req_post(req, ev);
    4923             :         }
    4924             : 
    4925        1333 :         if (clistr_is_previous_version_path(fname)) {
    4926           0 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
    4927             :         }
    4928             : 
    4929        1333 :         subreq = cli_smb_send(state, ev, cli, SMBsetatr, additional_flags,
    4930             :                         additional_flags2,
    4931        1333 :                         8, state->vwv, talloc_get_size(bytes), bytes);
    4932        1333 :         if (tevent_req_nomem(subreq, req)) {
    4933           0 :                 return tevent_req_post(req, ev);
    4934             :         }
    4935        1333 :         tevent_req_set_callback(subreq, cli_setatr_done, req);
    4936        1333 :         return req;
    4937             : }
    4938             : 
    4939        1333 : static void cli_setatr_done(struct tevent_req *subreq)
    4940             : {
    4941        1333 :         struct tevent_req *req = tevent_req_callback_data(
    4942             :                 subreq, struct tevent_req);
    4943           0 :         NTSTATUS status;
    4944             : 
    4945        1333 :         status = cli_smb_recv(subreq, NULL, NULL, 0, NULL, NULL, NULL, NULL);
    4946        1333 :         TALLOC_FREE(subreq);
    4947        1333 :         if (tevent_req_nterror(req, status)) {
    4948         124 :                 return;
    4949             :         }
    4950        1209 :         tevent_req_done(req);
    4951             : }
    4952             : 
    4953        1338 : NTSTATUS cli_setatr_recv(struct tevent_req *req)
    4954             : {
    4955        1338 :         return tevent_req_simple_recv_ntstatus(req);
    4956             : }
    4957             : 
    4958        2620 : NTSTATUS cli_setatr(struct cli_state *cli,
    4959             :                 const char *fname,
    4960             :                 uint32_t attr,
    4961             :                 time_t mtime)
    4962             : {
    4963        2620 :         TALLOC_CTX *frame = NULL;
    4964        2620 :         struct tevent_context *ev = NULL;
    4965        2620 :         struct tevent_req *req = NULL;
    4966        2620 :         NTSTATUS status = NT_STATUS_OK;
    4967             : 
    4968        2620 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    4969        1282 :                 return cli_smb2_setatr(cli,
    4970             :                                         fname,
    4971             :                                         attr,
    4972             :                                         mtime);
    4973             :         }
    4974             : 
    4975        1338 :         frame = talloc_stackframe();
    4976             : 
    4977        1338 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    4978             :                 /*
    4979             :                  * Can't use sync call while an async call is in flight
    4980             :                  */
    4981           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    4982           0 :                 goto fail;
    4983             :         }
    4984             : 
    4985        1338 :         ev = samba_tevent_context_init(frame);
    4986        1338 :         if (ev == NULL) {
    4987           0 :                 status = NT_STATUS_NO_MEMORY;
    4988           0 :                 goto fail;
    4989             :         }
    4990             : 
    4991        1338 :         req = cli_setatr_send(frame, ev, cli, fname, attr, mtime);
    4992        1338 :         if (req == NULL) {
    4993           0 :                 status = NT_STATUS_NO_MEMORY;
    4994           0 :                 goto fail;
    4995             :         }
    4996             : 
    4997        1338 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    4998           0 :                 goto fail;
    4999             :         }
    5000             : 
    5001        1338 :         status = cli_setatr_recv(req);
    5002             : 
    5003        1338 :  fail:
    5004        1338 :         TALLOC_FREE(frame);
    5005        1338 :         return status;
    5006             : }
    5007             : 
    5008             : /****************************************************************************
    5009             :  Check for existence of a dir.
    5010             : ****************************************************************************/
    5011             : 
    5012             : static void cli_chkpath_done(struct tevent_req *subreq);
    5013             : static void cli_chkpath_opened(struct tevent_req *subreq);
    5014             : static void cli_chkpath_closed(struct tevent_req *subreq);
    5015             : 
    5016             : struct cli_chkpath_state {
    5017             :         struct tevent_context *ev;
    5018             :         struct cli_state *cli;
    5019             : };
    5020             : 
    5021        7238 : struct tevent_req *cli_chkpath_send(TALLOC_CTX *mem_ctx,
    5022             :                                   struct tevent_context *ev,
    5023             :                                   struct cli_state *cli,
    5024             :                                   const char *fname)
    5025             : {
    5026        7238 :         struct tevent_req *req = NULL, *subreq = NULL;
    5027        7238 :         struct cli_chkpath_state *state = NULL;
    5028        7238 :         uint8_t additional_flags = 0;
    5029        7238 :         uint16_t additional_flags2 = 0;
    5030        7238 :         uint8_t *bytes = NULL;
    5031        7238 :         char *fname_cp = NULL;
    5032             : 
    5033        7238 :         req = tevent_req_create(mem_ctx, &state, struct cli_chkpath_state);
    5034        7238 :         if (req == NULL) {
    5035           0 :                 return NULL;
    5036             :         }
    5037        7238 :         state->ev = ev;
    5038        7238 :         state->cli = cli;
    5039             : 
    5040        7238 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_NT1) {
    5041        7238 :                 subreq = cli_ntcreate_send(
    5042             :                         state,                    /* mem_ctx */
    5043        7238 :                         state->ev,             /* ev */
    5044        7238 :                         state->cli,            /* cli */
    5045             :                         fname,                    /* fname */
    5046             :                         0,                        /* create_flags */
    5047             :                         FILE_READ_ATTRIBUTES,     /* desired_access */
    5048             :                         FILE_ATTRIBUTE_DIRECTORY, /* FileAttributes */
    5049             :                         FILE_SHARE_READ|
    5050             :                         FILE_SHARE_WRITE|
    5051             :                         FILE_SHARE_DELETE, /* share_access */
    5052             :                         FILE_OPEN,      /* CreateDisposition */
    5053             :                         FILE_DIRECTORY_FILE, /* CreateOptions */
    5054             :                         SMB2_IMPERSONATION_IMPERSONATION,
    5055             :                         0);             /* SecurityFlags */
    5056        7238 :                 if (tevent_req_nomem(subreq, req)) {
    5057           0 :                         return tevent_req_post(req, ev);
    5058             :                 }
    5059        7238 :                 tevent_req_set_callback(subreq, cli_chkpath_opened, req);
    5060        7238 :                 return req;
    5061             :         }
    5062             : 
    5063           0 :         bytes = talloc_array(state, uint8_t, 1);
    5064           0 :         if (tevent_req_nomem(bytes, req)) {
    5065           0 :                 return tevent_req_post(req, ev);
    5066             :         }
    5067             :         /*
    5068             :          * SMBcheckpath on a DFS share must use DFS names.
    5069             :          */
    5070           0 :         fname_cp = smb1_dfs_share_path(state, cli, fname);
    5071           0 :         if (tevent_req_nomem(fname_cp, req)) {
    5072           0 :                 return tevent_req_post(req, ev);
    5073             :         }
    5074           0 :         bytes[0] = 4;
    5075           0 :         bytes = smb_bytes_push_str(bytes,
    5076           0 :                                    smbXcli_conn_use_unicode(cli->conn),
    5077             :                                    fname_cp,
    5078           0 :                                    strlen(fname_cp)+1,
    5079             :                                    NULL);
    5080             : 
    5081           0 :         if (tevent_req_nomem(bytes, req)) {
    5082           0 :                 return tevent_req_post(req, ev);
    5083             :         }
    5084             : 
    5085           0 :         if (clistr_is_previous_version_path(fname)) {
    5086           0 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
    5087             :         }
    5088             : 
    5089           0 :         subreq = cli_smb_send(state, ev, cli, SMBcheckpath, additional_flags,
    5090             :                         additional_flags2,
    5091           0 :                         0, NULL, talloc_get_size(bytes), bytes);
    5092           0 :         if (tevent_req_nomem(subreq, req)) {
    5093           0 :                 return tevent_req_post(req, ev);
    5094             :         }
    5095           0 :         tevent_req_set_callback(subreq, cli_chkpath_done, req);
    5096           0 :         return req;
    5097             : }
    5098             : 
    5099           0 : static void cli_chkpath_done(struct tevent_req *subreq)
    5100             : {
    5101           0 :         NTSTATUS status = cli_smb_recv(
    5102             :                 subreq, NULL, NULL, 0, NULL, NULL, NULL, NULL);
    5103           0 :         tevent_req_simple_finish_ntstatus(subreq, status);
    5104           0 : }
    5105             : 
    5106        7238 : static void cli_chkpath_opened(struct tevent_req *subreq)
    5107             : {
    5108        7238 :         struct tevent_req *req = tevent_req_callback_data(
    5109             :                 subreq, struct tevent_req);
    5110        7238 :         struct cli_chkpath_state *state = tevent_req_data(
    5111             :                 req, struct cli_chkpath_state);
    5112           0 :         NTSTATUS status;
    5113           0 :         uint16_t fnum;
    5114             : 
    5115        7238 :         status = cli_ntcreate_recv(subreq, &fnum, NULL);
    5116        7238 :         TALLOC_FREE(subreq);
    5117        7238 :         if (tevent_req_nterror(req, status)) {
    5118        1382 :                 return;
    5119             :         }
    5120             : 
    5121        5856 :         subreq = cli_close_send(state, state->ev, state->cli, fnum, 0);
    5122        5856 :         if (tevent_req_nomem(subreq, req)) {
    5123           0 :                 return;
    5124             :         }
    5125        5856 :         tevent_req_set_callback(subreq, cli_chkpath_closed, req);
    5126             : }
    5127             : 
    5128        5856 : static void cli_chkpath_closed(struct tevent_req *subreq)
    5129             : {
    5130        5856 :         NTSTATUS status = cli_close_recv(subreq);
    5131        5856 :         tevent_req_simple_finish_ntstatus(subreq, status);
    5132        5856 : }
    5133             : 
    5134        7238 : NTSTATUS cli_chkpath_recv(struct tevent_req *req)
    5135             : {
    5136        7238 :         return tevent_req_simple_recv_ntstatus(req);
    5137             : }
    5138             : 
    5139        1752 : NTSTATUS cli_chkpath(struct cli_state *cli, const char *path)
    5140             : {
    5141        1752 :         TALLOC_CTX *frame = NULL;
    5142        1752 :         struct tevent_context *ev = NULL;
    5143        1752 :         struct tevent_req *req = NULL;
    5144        1752 :         char *path2 = NULL;
    5145        1752 :         NTSTATUS status = NT_STATUS_OK;
    5146             : 
    5147        1752 :         frame = talloc_stackframe();
    5148             : 
    5149        1752 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    5150             :                 /*
    5151             :                  * Can't use sync call while an async call is in flight
    5152             :                  */
    5153           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    5154           0 :                 goto fail;
    5155             :         }
    5156             : 
    5157        1752 :         path2 = talloc_strdup(frame, path);
    5158        1752 :         if (!path2) {
    5159           0 :                 status = NT_STATUS_NO_MEMORY;
    5160           0 :                 goto fail;
    5161             :         }
    5162        1752 :         trim_char(path2,'\0','\\');
    5163        1752 :         if (!*path2) {
    5164           0 :                 path2 = talloc_strdup(frame, "\\");
    5165           0 :                 if (!path2) {
    5166           0 :                         status = NT_STATUS_NO_MEMORY;
    5167           0 :                         goto fail;
    5168             :                 }
    5169             :         }
    5170             : 
    5171        1752 :         ev = samba_tevent_context_init(frame);
    5172        1752 :         if (ev == NULL) {
    5173           0 :                 status = NT_STATUS_NO_MEMORY;
    5174           0 :                 goto fail;
    5175             :         }
    5176             : 
    5177        1752 :         req = cli_chkpath_send(frame, ev, cli, path2);
    5178        1752 :         if (req == NULL) {
    5179           0 :                 status = NT_STATUS_NO_MEMORY;
    5180           0 :                 goto fail;
    5181             :         }
    5182             : 
    5183        1752 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    5184           0 :                 goto fail;
    5185             :         }
    5186             : 
    5187        1752 :         status = cli_chkpath_recv(req);
    5188        1752 :         cli->raw_status = status; /* cli_smb2_chkpath_recv doesn't set this */
    5189             : 
    5190        1752 :  fail:
    5191        1752 :         TALLOC_FREE(frame);
    5192        1752 :         return status;
    5193             : }
    5194             : 
    5195             : /****************************************************************************
    5196             :  Query disk space.
    5197             : ****************************************************************************/
    5198             : 
    5199             : static void cli_dskattr_done(struct tevent_req *subreq);
    5200             : 
    5201             : struct cli_dskattr_state {
    5202             :         int bsize;
    5203             :         int total;
    5204             :         int avail;
    5205             : };
    5206             : 
    5207           0 : struct tevent_req *cli_dskattr_send(TALLOC_CTX *mem_ctx,
    5208             :                                   struct tevent_context *ev,
    5209             :                                   struct cli_state *cli)
    5210             : {
    5211           0 :         struct tevent_req *req = NULL, *subreq = NULL;
    5212           0 :         struct cli_dskattr_state *state = NULL;
    5213           0 :         uint8_t additional_flags = 0;
    5214             : 
    5215           0 :         req = tevent_req_create(mem_ctx, &state, struct cli_dskattr_state);
    5216           0 :         if (req == NULL) {
    5217           0 :                 return NULL;
    5218             :         }
    5219             : 
    5220           0 :         subreq = cli_smb_send(state, ev, cli, SMBdskattr, additional_flags, 0,
    5221             :                               0, NULL, 0, NULL);
    5222           0 :         if (tevent_req_nomem(subreq, req)) {
    5223           0 :                 return tevent_req_post(req, ev);
    5224             :         }
    5225           0 :         tevent_req_set_callback(subreq, cli_dskattr_done, req);
    5226           0 :         return req;
    5227             : }
    5228             : 
    5229           0 : static void cli_dskattr_done(struct tevent_req *subreq)
    5230             : {
    5231           0 :         struct tevent_req *req = tevent_req_callback_data(
    5232             :                 subreq, struct tevent_req);
    5233           0 :         struct cli_dskattr_state *state = tevent_req_data(
    5234             :                 req, struct cli_dskattr_state);
    5235           0 :         uint8_t wct;
    5236           0 :         uint16_t *vwv = NULL;
    5237           0 :         NTSTATUS status;
    5238             : 
    5239           0 :         status = cli_smb_recv(subreq, state, NULL, 4, &wct, &vwv, NULL,
    5240             :                               NULL);
    5241           0 :         TALLOC_FREE(subreq);
    5242           0 :         if (tevent_req_nterror(req, status)) {
    5243           0 :                 return;
    5244             :         }
    5245           0 :         state->bsize = SVAL(vwv+1, 0)*SVAL(vwv+2,0);
    5246           0 :         state->total = SVAL(vwv+0, 0);
    5247           0 :         state->avail = SVAL(vwv+3, 0);
    5248           0 :         tevent_req_done(req);
    5249             : }
    5250             : 
    5251           0 : NTSTATUS cli_dskattr_recv(struct tevent_req *req, int *bsize, int *total, int *avail)
    5252             : {
    5253           0 :         struct cli_dskattr_state *state = tevent_req_data(
    5254             :                                 req, struct cli_dskattr_state);
    5255           0 :         NTSTATUS status;
    5256             : 
    5257           0 :         if (tevent_req_is_nterror(req, &status)) {
    5258           0 :                 return status;
    5259             :         }
    5260           0 :         *bsize = state->bsize;
    5261           0 :         *total = state->total;
    5262           0 :         *avail = state->avail;
    5263           0 :         return NT_STATUS_OK;
    5264             : }
    5265             : 
    5266           0 : NTSTATUS cli_dskattr(struct cli_state *cli, int *bsize, int *total, int *avail)
    5267             : {
    5268           0 :         TALLOC_CTX *frame = NULL;
    5269           0 :         struct tevent_context *ev = NULL;
    5270           0 :         struct tevent_req *req = NULL;
    5271           0 :         NTSTATUS status = NT_STATUS_OK;
    5272             : 
    5273           0 :         frame = talloc_stackframe();
    5274             : 
    5275           0 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    5276             :                 /*
    5277             :                  * Can't use sync call while an async call is in flight
    5278             :                  */
    5279           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    5280           0 :                 goto fail;
    5281             :         }
    5282             : 
    5283           0 :         ev = samba_tevent_context_init(frame);
    5284           0 :         if (ev == NULL) {
    5285           0 :                 status = NT_STATUS_NO_MEMORY;
    5286           0 :                 goto fail;
    5287             :         }
    5288             : 
    5289           0 :         req = cli_dskattr_send(frame, ev, cli);
    5290           0 :         if (req == NULL) {
    5291           0 :                 status = NT_STATUS_NO_MEMORY;
    5292           0 :                 goto fail;
    5293             :         }
    5294             : 
    5295           0 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    5296           0 :                 goto fail;
    5297             :         }
    5298             : 
    5299           0 :         status = cli_dskattr_recv(req, bsize, total, avail);
    5300             : 
    5301           0 :  fail:
    5302           0 :         TALLOC_FREE(frame);
    5303           0 :         return status;
    5304             : }
    5305             : 
    5306        1396 : NTSTATUS cli_disk_size(struct cli_state *cli, const char *path, uint64_t *bsize,
    5307             :                        uint64_t *total, uint64_t *avail)
    5308             : {
    5309           0 :         uint64_t sectors_per_block;
    5310           0 :         uint64_t bytes_per_sector;
    5311        1396 :         int old_bsize = 0, old_total = 0, old_avail = 0;
    5312           0 :         NTSTATUS status;
    5313             : 
    5314        1396 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    5315        1100 :                 return cli_smb2_dskattr(cli, path, bsize, total, avail);
    5316             :         }
    5317             : 
    5318             :         /*
    5319             :          * Try the trans2 disk full size info call first.
    5320             :          * We already use this in SMBC_fstatvfs_ctx().
    5321             :          * Ignore 'actual_available_units' as we only
    5322             :          * care about the quota for the caller.
    5323             :          */
    5324             : 
    5325         296 :         status = cli_get_fs_full_size_info(cli,
    5326             :                         total,
    5327             :                         avail,
    5328             :                         NULL,
    5329             :                         &sectors_per_block,
    5330             :                         &bytes_per_sector);
    5331             : 
    5332             :         /* Try and cope will all variants of "we don't do this call"
    5333             :            and fall back to cli_dskattr. */
    5334             : 
    5335         296 :         if (NT_STATUS_EQUAL(status,NT_STATUS_NOT_IMPLEMENTED) ||
    5336         296 :                         NT_STATUS_EQUAL(status,NT_STATUS_NOT_SUPPORTED) ||
    5337         296 :                         NT_STATUS_EQUAL(status,NT_STATUS_INVALID_INFO_CLASS) ||
    5338         296 :                         NT_STATUS_EQUAL(status,NT_STATUS_PROCEDURE_NOT_FOUND) ||
    5339         296 :                         NT_STATUS_EQUAL(status,NT_STATUS_INVALID_LEVEL) ||
    5340         296 :                         NT_STATUS_EQUAL(status,NT_STATUS_INVALID_PARAMETER) ||
    5341         296 :                         NT_STATUS_EQUAL(status,NT_STATUS_INVALID_DEVICE_REQUEST) ||
    5342         296 :                         NT_STATUS_EQUAL(status,NT_STATUS_INVALID_DEVICE_STATE) ||
    5343         296 :                         NT_STATUS_EQUAL(status,NT_STATUS_CTL_FILE_NOT_SUPPORTED) ||
    5344         296 :                         NT_STATUS_EQUAL(status,NT_STATUS_UNSUCCESSFUL)) {
    5345           0 :                 goto try_dskattr;
    5346             :         }
    5347             : 
    5348         296 :         if (!NT_STATUS_IS_OK(status)) {
    5349           0 :                 return status;
    5350             :         }
    5351             : 
    5352         296 :         if (bsize) {
    5353         296 :                 *bsize = sectors_per_block *
    5354             :                          bytes_per_sector;
    5355             :         }
    5356             : 
    5357         296 :         return NT_STATUS_OK;
    5358             : 
    5359           0 :   try_dskattr:
    5360             : 
    5361             :         /* Old SMB1 core protocol fallback. */
    5362           0 :         status = cli_dskattr(cli, &old_bsize, &old_total, &old_avail);
    5363           0 :         if (!NT_STATUS_IS_OK(status)) {
    5364           0 :                 return status;
    5365             :         }
    5366           0 :         if (bsize) {
    5367           0 :                 *bsize = (uint64_t)old_bsize;
    5368             :         }
    5369           0 :         if (total) {
    5370           0 :                 *total = (uint64_t)old_total;
    5371             :         }
    5372           0 :         if (avail) {
    5373           0 :                 *avail = (uint64_t)old_avail;
    5374             :         }
    5375           0 :         return NT_STATUS_OK;
    5376             : }
    5377             : 
    5378             : /****************************************************************************
    5379             :  Create and open a temporary file.
    5380             : ****************************************************************************/
    5381             : 
    5382             : static void cli_ctemp_done(struct tevent_req *subreq);
    5383             : 
    5384             : struct ctemp_state {
    5385             :         uint16_t vwv[3];
    5386             :         char *ret_path;
    5387             :         uint16_t fnum;
    5388             : };
    5389             : 
    5390           5 : struct tevent_req *cli_ctemp_send(TALLOC_CTX *mem_ctx,
    5391             :                                 struct tevent_context *ev,
    5392             :                                 struct cli_state *cli,
    5393             :                                 const char *path)
    5394             : {
    5395           5 :         struct tevent_req *req = NULL, *subreq = NULL;
    5396           5 :         struct ctemp_state *state = NULL;
    5397           5 :         uint8_t additional_flags = 0;
    5398           5 :         uint16_t additional_flags2 = 0;
    5399           5 :         uint8_t *bytes = NULL;
    5400           5 :         char *path_cp = NULL;
    5401             : 
    5402           5 :         req = tevent_req_create(mem_ctx, &state, struct ctemp_state);
    5403           5 :         if (req == NULL) {
    5404           0 :                 return NULL;
    5405             :         }
    5406             : 
    5407           5 :         SSVAL(state->vwv,0,0);
    5408           5 :         SIVALS(state->vwv+1,0,-1);
    5409             : 
    5410           5 :         bytes = talloc_array(state, uint8_t, 1);
    5411           5 :         if (tevent_req_nomem(bytes, req)) {
    5412           0 :                 return tevent_req_post(req, ev);
    5413             :         }
    5414             :         /*
    5415             :          * SMBctemp on a DFS share must use DFS names.
    5416             :          */
    5417           5 :         path_cp = smb1_dfs_share_path(state, cli, path);
    5418           5 :         if (tevent_req_nomem(path_cp, req)) {
    5419           0 :                 return tevent_req_post(req, ev);
    5420             :         }
    5421           5 :         bytes[0] = 4;
    5422           5 :         bytes = smb_bytes_push_str(bytes,
    5423           5 :                                    smbXcli_conn_use_unicode(cli->conn),
    5424             :                                    path_cp,
    5425           5 :                                    strlen(path_cp)+1,
    5426             :                                    NULL);
    5427           5 :         if (tevent_req_nomem(bytes, req)) {
    5428           0 :                 return tevent_req_post(req, ev);
    5429             :         }
    5430             : 
    5431           5 :         if (clistr_is_previous_version_path(path)) {
    5432           0 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
    5433             :         }
    5434             : 
    5435           5 :         subreq = cli_smb_send(state, ev, cli, SMBctemp, additional_flags,
    5436             :                         additional_flags2,
    5437           5 :                         3, state->vwv, talloc_get_size(bytes), bytes);
    5438           5 :         if (tevent_req_nomem(subreq, req)) {
    5439           0 :                 return tevent_req_post(req, ev);
    5440             :         }
    5441           5 :         tevent_req_set_callback(subreq, cli_ctemp_done, req);
    5442           5 :         return req;
    5443             : }
    5444             : 
    5445           5 : static void cli_ctemp_done(struct tevent_req *subreq)
    5446             : {
    5447           5 :         struct tevent_req *req = tevent_req_callback_data(
    5448             :                                 subreq, struct tevent_req);
    5449           5 :         struct ctemp_state *state = tevent_req_data(
    5450             :                                 req, struct ctemp_state);
    5451           0 :         NTSTATUS status;
    5452           0 :         uint8_t wcnt;
    5453           0 :         uint16_t *vwv;
    5454           5 :         uint32_t num_bytes = 0;
    5455           5 :         uint8_t *bytes = NULL;
    5456             : 
    5457           5 :         status = cli_smb_recv(subreq, state, NULL, 1, &wcnt, &vwv,
    5458             :                               &num_bytes, &bytes);
    5459           5 :         TALLOC_FREE(subreq);
    5460           5 :         if (tevent_req_nterror(req, status)) {
    5461           0 :                 return;
    5462             :         }
    5463             : 
    5464           5 :         state->fnum = SVAL(vwv+0, 0);
    5465             : 
    5466             :         /* From W2K3, the result is just the ASCII name */
    5467           5 :         if (num_bytes < 2) {
    5468           0 :                 tevent_req_nterror(req, NT_STATUS_DATA_ERROR);
    5469           0 :                 return;
    5470             :         }
    5471             : 
    5472           5 :         if (pull_string_talloc(state,
    5473             :                         NULL,
    5474             :                         0,
    5475             :                         &state->ret_path,
    5476             :                         bytes,
    5477             :                         num_bytes,
    5478             :                         STR_ASCII) == 0) {
    5479           0 :                 tevent_req_nterror(req, NT_STATUS_NO_MEMORY);
    5480           0 :                 return;
    5481             :         }
    5482           5 :         tevent_req_done(req);
    5483             : }
    5484             : 
    5485           5 : NTSTATUS cli_ctemp_recv(struct tevent_req *req,
    5486             :                         TALLOC_CTX *ctx,
    5487             :                         uint16_t *pfnum,
    5488             :                         char **outfile)
    5489             : {
    5490           5 :         struct ctemp_state *state = tevent_req_data(req,
    5491             :                         struct ctemp_state);
    5492           0 :         NTSTATUS status;
    5493             : 
    5494           5 :         if (tevent_req_is_nterror(req, &status)) {
    5495           0 :                 return status;
    5496             :         }
    5497           5 :         *pfnum = state->fnum;
    5498           5 :         *outfile = talloc_strdup(ctx, state->ret_path);
    5499           5 :         if (!*outfile) {
    5500           0 :                 return NT_STATUS_NO_MEMORY;
    5501             :         }
    5502           5 :         return NT_STATUS_OK;
    5503             : }
    5504             : 
    5505           5 : NTSTATUS cli_ctemp(struct cli_state *cli,
    5506             :                         TALLOC_CTX *ctx,
    5507             :                         const char *path,
    5508             :                         uint16_t *pfnum,
    5509             :                         char **out_path)
    5510             : {
    5511           5 :         TALLOC_CTX *frame = talloc_stackframe();
    5512           0 :         struct tevent_context *ev;
    5513           0 :         struct tevent_req *req;
    5514           5 :         NTSTATUS status = NT_STATUS_OK;
    5515             : 
    5516           5 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    5517             :                 /*
    5518             :                  * Can't use sync call while an async call is in flight
    5519             :                  */
    5520           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    5521           0 :                 goto fail;
    5522             :         }
    5523             : 
    5524           5 :         ev = samba_tevent_context_init(frame);
    5525           5 :         if (ev == NULL) {
    5526           0 :                 status = NT_STATUS_NO_MEMORY;
    5527           0 :                 goto fail;
    5528             :         }
    5529             : 
    5530           5 :         req = cli_ctemp_send(frame, ev, cli, path);
    5531           5 :         if (req == NULL) {
    5532           0 :                 status = NT_STATUS_NO_MEMORY;
    5533           0 :                 goto fail;
    5534             :         }
    5535             : 
    5536           5 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    5537           0 :                 goto fail;
    5538             :         }
    5539             : 
    5540           5 :         status = cli_ctemp_recv(req, ctx, pfnum, out_path);
    5541             : 
    5542           5 :  fail:
    5543           5 :         TALLOC_FREE(frame);
    5544           5 :         return status;
    5545             : }
    5546             : 
    5547             : /*********************************************************
    5548             :  Set an extended attribute utility fn.
    5549             : *********************************************************/
    5550             : 
    5551           8 : static NTSTATUS cli_set_ea(struct cli_state *cli, uint16_t setup_val,
    5552             :                            uint8_t *param, unsigned int param_len,
    5553             :                            const char *ea_name,
    5554             :                            const char *ea_val, size_t ea_len)
    5555             : {
    5556           0 :         uint16_t setup[1];
    5557           8 :         unsigned int data_len = 0;
    5558           8 :         uint8_t *data = NULL;
    5559           0 :         char *p;
    5560           8 :         size_t ea_namelen = strlen(ea_name);
    5561           0 :         NTSTATUS status;
    5562             : 
    5563           8 :         SSVAL(setup, 0, setup_val);
    5564             : 
    5565           8 :         if (ea_namelen == 0 && ea_len == 0) {
    5566           0 :                 data_len = 4;
    5567           0 :                 data = talloc_array(talloc_tos(),
    5568             :                                 uint8_t,
    5569             :                                 data_len);
    5570           0 :                 if (!data) {
    5571           0 :                         return NT_STATUS_NO_MEMORY;
    5572             :                 }
    5573           0 :                 p = (char *)data;
    5574           0 :                 SIVAL(p,0,data_len);
    5575             :         } else {
    5576           8 :                 data_len = 4 + 4 + ea_namelen + 1 + ea_len;
    5577           8 :                 data = talloc_array(talloc_tos(),
    5578             :                                 uint8_t,
    5579             :                                 data_len);
    5580           8 :                 if (!data) {
    5581           0 :                         return NT_STATUS_NO_MEMORY;
    5582             :                 }
    5583           8 :                 p = (char *)data;
    5584           8 :                 SIVAL(p,0,data_len);
    5585           8 :                 p += 4;
    5586           8 :                 SCVAL(p, 0, 0); /* EA flags. */
    5587           8 :                 SCVAL(p, 1, ea_namelen);
    5588           8 :                 SSVAL(p, 2, ea_len);
    5589           8 :                 memcpy(p+4, ea_name, ea_namelen+1); /* Copy in the name. */
    5590           8 :                 memcpy(p+4+ea_namelen+1, ea_val, ea_len);
    5591             :         }
    5592             : 
    5593             :         /*
    5594             :          * FIXME - if we want to do previous version path
    5595             :          * processing on an EA set call we need to turn this
    5596             :          * into calls to cli_trans_send()/cli_trans_recv()
    5597             :          * with a temporary event context, as cli_trans_send()
    5598             :          * have access to the additional_flags2 needed to
    5599             :          * send @GMT- paths. JRA.
    5600             :          */
    5601             : 
    5602           8 :         status = cli_trans(talloc_tos(), cli, SMBtrans2, NULL, -1, 0, 0,
    5603             :                            setup, 1, 0,
    5604             :                            param, param_len, 2,
    5605             :                            data,  data_len, 0,
    5606             :                            NULL,
    5607             :                            NULL, 0, NULL, /* rsetup */
    5608             :                            NULL, 0, NULL, /* rparam */
    5609             :                            NULL, 0, NULL); /* rdata */
    5610           8 :         talloc_free(data);
    5611           8 :         return status;
    5612             : }
    5613             : 
    5614             : /*********************************************************
    5615             :  Set an extended attribute on a pathname.
    5616             : *********************************************************/
    5617             : 
    5618           8 : NTSTATUS cli_set_ea_path(struct cli_state *cli, const char *path,
    5619             :                          const char *ea_name, const char *ea_val,
    5620             :                          size_t ea_len)
    5621             : {
    5622           8 :         unsigned int param_len = 0;
    5623           0 :         uint8_t *param;
    5624           0 :         NTSTATUS status;
    5625           8 :         TALLOC_CTX *frame = NULL;
    5626           8 :         char *path_cp = NULL;
    5627             : 
    5628           8 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    5629           0 :                 return cli_smb2_set_ea_path(cli,
    5630             :                                         path,
    5631             :                                         ea_name,
    5632             :                                         ea_val,
    5633             :                                         ea_len);
    5634             :         }
    5635             : 
    5636           8 :         frame = talloc_stackframe();
    5637             : 
    5638           8 :         param = talloc_array(frame, uint8_t, 6);
    5639           8 :         if (!param) {
    5640           0 :                 status = NT_STATUS_NO_MEMORY;
    5641           0 :                 goto fail;
    5642             :         }
    5643           8 :         SSVAL(param,0,SMB_INFO_SET_EA);
    5644           8 :         SSVAL(param,2,0);
    5645           8 :         SSVAL(param,4,0);
    5646             : 
    5647             :         /*
    5648             :          * TRANSACT2_SETPATHINFO on a DFS share must use DFS names.
    5649             :          */
    5650           8 :         path_cp = smb1_dfs_share_path(frame, cli, path);
    5651           8 :         if (path_cp == NULL) {
    5652           0 :                 status = NT_STATUS_NO_MEMORY;
    5653           0 :                 goto fail;
    5654             :         }
    5655           8 :         param = trans2_bytes_push_str(param,
    5656           8 :                                       smbXcli_conn_use_unicode(cli->conn),
    5657             :                                       path_cp,
    5658           8 :                                       strlen(path_cp)+1,
    5659             :                                       NULL);
    5660           8 :         param_len = talloc_get_size(param);
    5661             : 
    5662           8 :         status = cli_set_ea(cli, TRANSACT2_SETPATHINFO, param, param_len,
    5663             :                             ea_name, ea_val, ea_len);
    5664             : 
    5665           8 :   fail:
    5666             : 
    5667           8 :         TALLOC_FREE(frame);
    5668           8 :         return status;
    5669             : }
    5670             : 
    5671             : /*********************************************************
    5672             :  Set an extended attribute on an fnum.
    5673             : *********************************************************/
    5674             : 
    5675           0 : NTSTATUS cli_set_ea_fnum(struct cli_state *cli, uint16_t fnum,
    5676             :                          const char *ea_name, const char *ea_val,
    5677             :                          size_t ea_len)
    5678             : {
    5679           0 :         uint8_t param[6] = { 0, };
    5680             : 
    5681           0 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    5682           0 :                 return cli_smb2_set_ea_fnum(cli,
    5683             :                                         fnum,
    5684             :                                         ea_name,
    5685             :                                         ea_val,
    5686             :                                         ea_len);
    5687             :         }
    5688             : 
    5689           0 :         SSVAL(param,0,fnum);
    5690           0 :         SSVAL(param,2,SMB_INFO_SET_EA);
    5691             : 
    5692           0 :         return cli_set_ea(cli, TRANSACT2_SETFILEINFO, param, 6,
    5693             :                           ea_name, ea_val, ea_len);
    5694             : }
    5695             : 
    5696             : /*********************************************************
    5697             :  Get an extended attribute list utility fn.
    5698             : *********************************************************/
    5699             : 
    5700           8 : static bool parse_ea_blob(TALLOC_CTX *ctx, const uint8_t *rdata,
    5701             :                           size_t rdata_len,
    5702             :                           size_t *pnum_eas, struct ea_struct **pea_list)
    5703             : {
    5704           8 :         struct ea_struct *ea_list = NULL;
    5705           0 :         size_t num_eas;
    5706           0 :         size_t ea_size;
    5707           0 :         const uint8_t *p;
    5708             : 
    5709           8 :         if (rdata_len < 4) {
    5710           0 :                 return false;
    5711             :         }
    5712             : 
    5713           8 :         ea_size = (size_t)IVAL(rdata,0);
    5714           8 :         if (ea_size > rdata_len) {
    5715           0 :                 return false;
    5716             :         }
    5717             : 
    5718           8 :         if (ea_size == 0) {
    5719             :                 /* No EA's present. */
    5720           0 :                 *pnum_eas = 0;
    5721           0 :                 *pea_list = NULL;
    5722           0 :                 return true;
    5723             :         }
    5724             : 
    5725           8 :         p = rdata + 4;
    5726           8 :         ea_size -= 4;
    5727             : 
    5728             :         /* Validate the EA list and count it. */
    5729          12 :         for (num_eas = 0; ea_size >= 4; num_eas++) {
    5730           4 :                 unsigned int ea_namelen = CVAL(p,1);
    5731           4 :                 unsigned int ea_valuelen = SVAL(p,2);
    5732           4 :                 if (ea_namelen == 0) {
    5733           0 :                         return false;
    5734             :                 }
    5735           4 :                 if (4 + ea_namelen + 1 + ea_valuelen > ea_size) {
    5736           0 :                         return false;
    5737             :                 }
    5738           4 :                 ea_size -= 4 + ea_namelen + 1 + ea_valuelen;
    5739           4 :                 p += 4 + ea_namelen + 1 + ea_valuelen;
    5740             :         }
    5741             : 
    5742           8 :         if (num_eas == 0) {
    5743           4 :                 *pnum_eas = 0;
    5744           4 :                 *pea_list = NULL;
    5745           4 :                 return true;
    5746             :         }
    5747             : 
    5748           4 :         *pnum_eas = num_eas;
    5749           4 :         if (!pea_list) {
    5750             :                 /* Caller only wants number of EA's. */
    5751           0 :                 return true;
    5752             :         }
    5753             : 
    5754           4 :         ea_list = talloc_array(ctx, struct ea_struct, num_eas);
    5755           4 :         if (!ea_list) {
    5756           0 :                 return false;
    5757             :         }
    5758             : 
    5759           4 :         p = rdata + 4;
    5760             : 
    5761           8 :         for (num_eas = 0; num_eas < *pnum_eas; num_eas++ ) {
    5762           4 :                 struct ea_struct *ea = &ea_list[num_eas];
    5763           0 :                 fstring unix_ea_name;
    5764           4 :                 unsigned int ea_namelen = CVAL(p,1);
    5765           4 :                 unsigned int ea_valuelen = SVAL(p,2);
    5766             : 
    5767           4 :                 ea->flags = CVAL(p,0);
    5768           4 :                 unix_ea_name[0] = '\0';
    5769           4 :                 pull_ascii(unix_ea_name, p + 4, sizeof(unix_ea_name), rdata_len - PTR_DIFF(p+4, rdata), STR_TERMINATE);
    5770           4 :                 ea->name = talloc_strdup(ea_list, unix_ea_name);
    5771           4 :                 if (!ea->name) {
    5772           0 :                         goto fail;
    5773             :                 }
    5774             :                 /* Ensure the value is null terminated (in case it's a string). */
    5775           4 :                 ea->value = data_blob_talloc(ea_list, NULL, ea_valuelen + 1);
    5776           4 :                 if (!ea->value.data) {
    5777           0 :                         goto fail;
    5778             :                 }
    5779           4 :                 if (ea_valuelen) {
    5780           4 :                         memcpy(ea->value.data, p+4+ea_namelen+1, ea_valuelen);
    5781             :                 }
    5782           4 :                 ea->value.data[ea_valuelen] = 0;
    5783           4 :                 ea->value.length--;
    5784           4 :                 p += 4 + ea_namelen + 1 + ea_valuelen;
    5785             :         }
    5786             : 
    5787           4 :         *pea_list = ea_list;
    5788           4 :         return true;
    5789             : 
    5790           0 : fail:
    5791           0 :         TALLOC_FREE(ea_list);
    5792           0 :         return false;
    5793             : }
    5794             : 
    5795             : /*********************************************************
    5796             :  Get an extended attribute list from a pathname.
    5797             : *********************************************************/
    5798             : 
    5799             : struct cli_get_ea_list_path_state {
    5800             :         uint32_t num_data;
    5801             :         uint8_t *data;
    5802             : };
    5803             : 
    5804             : static void cli_get_ea_list_path_done(struct tevent_req *subreq);
    5805             : 
    5806           8 : struct tevent_req *cli_get_ea_list_path_send(TALLOC_CTX *mem_ctx,
    5807             :                                              struct tevent_context *ev,
    5808             :                                              struct cli_state *cli,
    5809             :                                              const char *fname)
    5810             : {
    5811           0 :         struct tevent_req *req, *subreq;
    5812           0 :         struct cli_get_ea_list_path_state *state;
    5813             : 
    5814           8 :         req = tevent_req_create(mem_ctx, &state,
    5815             :                                 struct cli_get_ea_list_path_state);
    5816           8 :         if (req == NULL) {
    5817           0 :                 return NULL;
    5818             :         }
    5819           8 :         subreq = cli_qpathinfo_send(state, ev, cli, fname,
    5820             :                                     SMB_INFO_QUERY_ALL_EAS, 4,
    5821             :                                     CLI_BUFFER_SIZE);
    5822           8 :         if (tevent_req_nomem(subreq, req)) {
    5823           0 :                 return tevent_req_post(req, ev);
    5824             :         }
    5825           8 :         tevent_req_set_callback(subreq, cli_get_ea_list_path_done, req);
    5826           8 :         return req;
    5827             : }
    5828             : 
    5829           8 : static void cli_get_ea_list_path_done(struct tevent_req *subreq)
    5830             : {
    5831           8 :         struct tevent_req *req = tevent_req_callback_data(
    5832             :                                 subreq, struct tevent_req);
    5833           8 :         struct cli_get_ea_list_path_state *state = tevent_req_data(
    5834             :                 req, struct cli_get_ea_list_path_state);
    5835           0 :         NTSTATUS status;
    5836             : 
    5837           8 :         status = cli_qpathinfo_recv(subreq, state, &state->data,
    5838             :                                     &state->num_data);
    5839           8 :         TALLOC_FREE(subreq);
    5840           8 :         if (tevent_req_nterror(req, status)) {
    5841           0 :                 return;
    5842             :         }
    5843           8 :         tevent_req_done(req);
    5844             : }
    5845             : 
    5846           8 : NTSTATUS cli_get_ea_list_path_recv(struct tevent_req *req, TALLOC_CTX *mem_ctx,
    5847             :                                    size_t *pnum_eas, struct ea_struct **peas)
    5848             : {
    5849           8 :         struct cli_get_ea_list_path_state *state = tevent_req_data(
    5850             :                 req, struct cli_get_ea_list_path_state);
    5851           0 :         NTSTATUS status;
    5852             : 
    5853           8 :         if (tevent_req_is_nterror(req, &status)) {
    5854           0 :                 return status;
    5855             :         }
    5856           8 :         if (!parse_ea_blob(mem_ctx, state->data, state->num_data,
    5857             :                            pnum_eas, peas)) {
    5858           0 :                 return NT_STATUS_INVALID_NETWORK_RESPONSE;
    5859             :         }
    5860           8 :         return NT_STATUS_OK;
    5861             : }
    5862             : 
    5863           8 : NTSTATUS cli_get_ea_list_path(struct cli_state *cli, const char *path,
    5864             :                 TALLOC_CTX *ctx,
    5865             :                 size_t *pnum_eas,
    5866             :                 struct ea_struct **pea_list)
    5867             : {
    5868           8 :         TALLOC_CTX *frame = NULL;
    5869           8 :         struct tevent_context *ev = NULL;
    5870           8 :         struct tevent_req *req = NULL;
    5871           8 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    5872             : 
    5873           8 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    5874           0 :                 return cli_smb2_get_ea_list_path(cli,
    5875             :                                         path,
    5876             :                                         ctx,
    5877             :                                         pnum_eas,
    5878             :                                         pea_list);
    5879             :         }
    5880             : 
    5881           8 :         frame = talloc_stackframe();
    5882             : 
    5883           8 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    5884             :                 /*
    5885             :                  * Can't use sync call while an async call is in flight
    5886             :                  */
    5887           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    5888           0 :                 goto fail;
    5889             :         }
    5890           8 :         ev = samba_tevent_context_init(frame);
    5891           8 :         if (ev == NULL) {
    5892           0 :                 goto fail;
    5893             :         }
    5894           8 :         req = cli_get_ea_list_path_send(frame, ev, cli, path);
    5895           8 :         if (req == NULL) {
    5896           0 :                 goto fail;
    5897             :         }
    5898           8 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    5899           0 :                 goto fail;
    5900             :         }
    5901           8 :         status = cli_get_ea_list_path_recv(req, ctx, pnum_eas, pea_list);
    5902           8 :  fail:
    5903           8 :         TALLOC_FREE(frame);
    5904           8 :         return status;
    5905             : }
    5906             : 
    5907             : /****************************************************************************
    5908             :  Convert open "flags" arg to uint32_t on wire.
    5909             : ****************************************************************************/
    5910             : 
    5911         130 : static uint32_t open_flags_to_wire(int flags)
    5912             : {
    5913         130 :         int open_mode = flags & O_ACCMODE;
    5914         130 :         uint32_t ret = 0;
    5915             : 
    5916         130 :         switch (open_mode) {
    5917           4 :                 case O_WRONLY:
    5918           4 :                         ret |= SMB_O_WRONLY;
    5919           4 :                         break;
    5920          98 :                 case O_RDWR:
    5921          98 :                         ret |= SMB_O_RDWR;
    5922          98 :                         break;
    5923          28 :                 default:
    5924             :                 case O_RDONLY:
    5925          28 :                         ret |= SMB_O_RDONLY;
    5926          28 :                         break;
    5927             :         }
    5928             : 
    5929         130 :         if (flags & O_CREAT) {
    5930          82 :                 ret |= SMB_O_CREAT;
    5931             :         }
    5932         130 :         if (flags & O_EXCL) {
    5933          38 :                 ret |= SMB_O_EXCL;
    5934             :         }
    5935         130 :         if (flags & O_TRUNC) {
    5936           4 :                 ret |= SMB_O_TRUNC;
    5937             :         }
    5938             : #if defined(O_SYNC)
    5939         130 :         if (flags & O_SYNC) {
    5940           0 :                 ret |= SMB_O_SYNC;
    5941             :         }
    5942             : #endif /* O_SYNC */
    5943         130 :         if (flags & O_APPEND) {
    5944           0 :                 ret |= SMB_O_APPEND;
    5945             :         }
    5946             : #if defined(O_DIRECT)
    5947         130 :         if (flags & O_DIRECT) {
    5948           0 :                 ret |= SMB_O_DIRECT;
    5949             :         }
    5950             : #endif
    5951             : #if defined(O_DIRECTORY)
    5952         130 :         if (flags & O_DIRECTORY) {
    5953           0 :                 ret |= SMB_O_DIRECTORY;
    5954             :         }
    5955             : #endif
    5956         130 :         return ret;
    5957             : }
    5958             : 
    5959             : /****************************************************************************
    5960             :  Open a file - POSIX semantics. Returns fnum. Doesn't request oplock.
    5961             : ****************************************************************************/
    5962             : 
    5963             : struct cli_posix_open_internal_state {
    5964             :         uint16_t setup;
    5965             :         uint8_t *param;
    5966             :         uint8_t data[18];
    5967             :         uint16_t fnum; /* Out */
    5968             : };
    5969             : 
    5970             : static void cli_posix_open_internal_done(struct tevent_req *subreq);
    5971             : 
    5972         186 : static struct tevent_req *cli_posix_open_internal_send(TALLOC_CTX *mem_ctx,
    5973             :                                         struct tevent_context *ev,
    5974             :                                         struct cli_state *cli,
    5975             :                                         const char *fname,
    5976             :                                         uint32_t wire_flags,
    5977             :                                         mode_t mode)
    5978             : {
    5979         186 :         struct tevent_req *req = NULL, *subreq = NULL;
    5980         186 :         struct cli_posix_open_internal_state *state = NULL;
    5981         186 :         char *fname_cp = NULL;
    5982             : 
    5983         186 :         req = tevent_req_create(
    5984             :                 mem_ctx, &state, struct cli_posix_open_internal_state);
    5985         186 :         if (req == NULL) {
    5986           0 :                 return NULL;
    5987             :         }
    5988             : 
    5989             :         /* Setup setup word. */
    5990         186 :         SSVAL(&state->setup, 0, TRANSACT2_SETPATHINFO);
    5991             : 
    5992             :         /* Setup param array. */
    5993         186 :         state->param = talloc_zero_array(state, uint8_t, 6);
    5994         186 :         if (tevent_req_nomem(state->param, req)) {
    5995           0 :                 return tevent_req_post(req, ev);
    5996             :         }
    5997         186 :         SSVAL(state->param, 0, SMB_POSIX_PATH_OPEN);
    5998             : 
    5999             :         /*
    6000             :          * TRANSACT2_SETPATHINFO on a DFS share must use DFS names.
    6001             :          */
    6002         186 :         fname_cp = smb1_dfs_share_path(state, cli, fname);
    6003         186 :         if (tevent_req_nomem(fname_cp, req)) {
    6004           0 :                 return tevent_req_post(req, ev);
    6005             :         }
    6006         558 :         state->param = trans2_bytes_push_str(
    6007         186 :                 state->param,
    6008         186 :                 smbXcli_conn_use_unicode(cli->conn),
    6009             :                 fname_cp,
    6010         186 :                 strlen(fname_cp)+1,
    6011             :                 NULL);
    6012             : 
    6013         186 :         if (tevent_req_nomem(state->param, req)) {
    6014           0 :                 return tevent_req_post(req, ev);
    6015             :         }
    6016             : 
    6017         186 :         SIVAL(state->data,0,0); /* No oplock. */
    6018         186 :         SIVAL(state->data,4,wire_flags);
    6019         186 :         SIVAL(state->data,8,unix_perms_to_wire(mode));
    6020         186 :         SIVAL(state->data,12,0); /* Top bits of perms currently undefined. */
    6021         186 :         SSVAL(state->data,16,SMB_NO_INFO_LEVEL_RETURNED); /* No info level returned. */
    6022             : 
    6023         186 :         subreq = cli_trans_send(state,                  /* mem ctx. */
    6024             :                                 ev,                     /* event ctx. */
    6025             :                                 cli,                    /* cli_state. */
    6026             :                                 0,                      /* additional_flags2 */
    6027             :                                 SMBtrans2,              /* cmd. */
    6028             :                                 NULL,                   /* pipe name. */
    6029             :                                 -1,                     /* fid. */
    6030             :                                 0,                      /* function. */
    6031             :                                 0,                      /* flags. */
    6032         186 :                                 &state->setup,           /* setup. */
    6033             :                                 1,                      /* num setup uint16_t words. */
    6034             :                                 0,                      /* max returned setup. */
    6035         186 :                                 state->param,                /* param. */
    6036         186 :                                 talloc_get_size(state->param),/* num param. */
    6037             :                                 2,                      /* max returned param. */
    6038         186 :                                 state->data,         /* data. */
    6039             :                                 18,                     /* num data. */
    6040             :                                 12);                    /* max returned data. */
    6041             : 
    6042         186 :         if (tevent_req_nomem(subreq, req)) {
    6043           0 :                 return tevent_req_post(req, ev);
    6044             :         }
    6045         186 :         tevent_req_set_callback(subreq, cli_posix_open_internal_done, req);
    6046         186 :         return req;
    6047             : }
    6048             : 
    6049         186 : static void cli_posix_open_internal_done(struct tevent_req *subreq)
    6050             : {
    6051         186 :         struct tevent_req *req = tevent_req_callback_data(
    6052             :                 subreq, struct tevent_req);
    6053         186 :         struct cli_posix_open_internal_state *state = tevent_req_data(
    6054             :                 req, struct cli_posix_open_internal_state);
    6055           0 :         NTSTATUS status;
    6056           0 :         uint8_t *data;
    6057           0 :         uint32_t num_data;
    6058             : 
    6059         186 :         status = cli_trans_recv(
    6060             :                 subreq,
    6061             :                 state,
    6062             :                 NULL,
    6063             :                 NULL,
    6064             :                 0,
    6065             :                 NULL,
    6066             :                 NULL,
    6067             :                 0,
    6068             :                 NULL,
    6069             :                 &data,
    6070             :                 12,
    6071             :                 &num_data);
    6072         186 :         TALLOC_FREE(subreq);
    6073         186 :         if (tevent_req_nterror(req, status)) {
    6074          24 :                 return;
    6075             :         }
    6076         162 :         state->fnum = SVAL(data,2);
    6077         162 :         tevent_req_done(req);
    6078             : }
    6079             : 
    6080         186 : static NTSTATUS cli_posix_open_internal_recv(struct tevent_req *req,
    6081             :                                              uint16_t *pfnum)
    6082             : {
    6083         186 :         struct cli_posix_open_internal_state *state = tevent_req_data(
    6084             :                 req, struct cli_posix_open_internal_state);
    6085           0 :         NTSTATUS status;
    6086             : 
    6087         186 :         if (tevent_req_is_nterror(req, &status)) {
    6088          24 :                 return status;
    6089             :         }
    6090         162 :         *pfnum = state->fnum;
    6091         162 :         return NT_STATUS_OK;
    6092             : }
    6093             : 
    6094             : struct cli_posix_open_state {
    6095             :         uint16_t fnum;
    6096             : };
    6097             : 
    6098             : static void cli_posix_open_done(struct tevent_req *subreq);
    6099             : 
    6100         130 : struct tevent_req *cli_posix_open_send(TALLOC_CTX *mem_ctx,
    6101             :                                         struct tevent_context *ev,
    6102             :                                         struct cli_state *cli,
    6103             :                                         const char *fname,
    6104             :                                         int flags,
    6105             :                                         mode_t mode)
    6106             : {
    6107         130 :         struct tevent_req *req = NULL, *subreq = NULL;
    6108         130 :         struct cli_posix_open_state *state = NULL;
    6109           0 :         uint32_t wire_flags;
    6110             : 
    6111         130 :         req = tevent_req_create(mem_ctx, &state,
    6112             :                                 struct cli_posix_open_state);
    6113         130 :         if (req == NULL) {
    6114           0 :                 return NULL;
    6115             :         }
    6116             : 
    6117         130 :         wire_flags = open_flags_to_wire(flags);
    6118             : 
    6119         130 :         subreq = cli_posix_open_internal_send(
    6120             :                 mem_ctx, ev, cli, fname, wire_flags, mode);
    6121         130 :         if (tevent_req_nomem(subreq, req)) {
    6122           0 :                 return tevent_req_post(req, ev);
    6123             :         }
    6124         130 :         tevent_req_set_callback(subreq, cli_posix_open_done, req);
    6125         130 :         return req;
    6126             : }
    6127             : 
    6128         130 : static void cli_posix_open_done(struct tevent_req *subreq)
    6129             : {
    6130         130 :         struct tevent_req *req = tevent_req_callback_data(
    6131             :                 subreq, struct tevent_req);
    6132         130 :         struct cli_posix_open_state *state = tevent_req_data(
    6133             :                 req, struct cli_posix_open_state);
    6134           0 :         NTSTATUS status;
    6135             : 
    6136         130 :         status = cli_posix_open_internal_recv(subreq, &state->fnum);
    6137         130 :         tevent_req_simple_finish_ntstatus(subreq, status);
    6138         130 : }
    6139             : 
    6140         130 : NTSTATUS cli_posix_open_recv(struct tevent_req *req, uint16_t *pfnum)
    6141             : {
    6142         130 :         struct cli_posix_open_state *state = tevent_req_data(
    6143             :                 req, struct cli_posix_open_state);
    6144           0 :         NTSTATUS status;
    6145             : 
    6146         130 :         if (tevent_req_is_nterror(req, &status)) {
    6147          16 :                 return status;
    6148             :         }
    6149         114 :         *pfnum = state->fnum;
    6150         114 :         return NT_STATUS_OK;
    6151             : }
    6152             : 
    6153             : /****************************************************************************
    6154             :  Open - POSIX semantics. Doesn't request oplock.
    6155             : ****************************************************************************/
    6156             : 
    6157         130 : NTSTATUS cli_posix_open(struct cli_state *cli, const char *fname,
    6158             :                         int flags, mode_t mode, uint16_t *pfnum)
    6159             : {
    6160             : 
    6161         130 :         TALLOC_CTX *frame = talloc_stackframe();
    6162         130 :         struct tevent_context *ev = NULL;
    6163         130 :         struct tevent_req *req = NULL;
    6164         130 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    6165             : 
    6166         130 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    6167             :                 /*
    6168             :                  * Can't use sync call while an async call is in flight
    6169             :                  */
    6170           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    6171           0 :                 goto fail;
    6172             :         }
    6173         130 :         ev = samba_tevent_context_init(frame);
    6174         130 :         if (ev == NULL) {
    6175           0 :                 goto fail;
    6176             :         }
    6177         130 :         req = cli_posix_open_send(
    6178             :                 frame, ev, cli, fname, flags, mode);
    6179         130 :         if (req == NULL) {
    6180           0 :                 goto fail;
    6181             :         }
    6182         130 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    6183           0 :                 goto fail;
    6184             :         }
    6185         130 :         status = cli_posix_open_recv(req, pfnum);
    6186         130 :  fail:
    6187         130 :         TALLOC_FREE(frame);
    6188         130 :         return status;
    6189             : }
    6190             : 
    6191             : struct cli_posix_mkdir_state {
    6192             :         struct tevent_context *ev;
    6193             :         struct cli_state *cli;
    6194             : };
    6195             : 
    6196             : static void cli_posix_mkdir_done(struct tevent_req *subreq);
    6197             : 
    6198          56 : struct tevent_req *cli_posix_mkdir_send(TALLOC_CTX *mem_ctx,
    6199             :                                         struct tevent_context *ev,
    6200             :                                         struct cli_state *cli,
    6201             :                                         const char *fname,
    6202             :                                         mode_t mode)
    6203             : {
    6204          56 :         struct tevent_req *req = NULL, *subreq = NULL;
    6205          56 :         struct cli_posix_mkdir_state *state = NULL;
    6206           0 :         uint32_t wire_flags;
    6207             : 
    6208          56 :         req = tevent_req_create(
    6209             :                 mem_ctx, &state, struct cli_posix_mkdir_state);
    6210          56 :         if (req == NULL) {
    6211           0 :                 return NULL;
    6212             :         }
    6213          56 :         state->ev = ev;
    6214          56 :         state->cli = cli;
    6215             : 
    6216          56 :         wire_flags = SMB_O_CREAT | SMB_O_DIRECTORY;
    6217             : 
    6218          56 :         subreq = cli_posix_open_internal_send(
    6219             :                 mem_ctx, ev, cli, fname, wire_flags, mode);
    6220          56 :         if (tevent_req_nomem(subreq, req)) {
    6221           0 :                 return tevent_req_post(req, ev);
    6222             :         }
    6223          56 :         tevent_req_set_callback(subreq, cli_posix_mkdir_done, req);
    6224          56 :         return req;
    6225             : }
    6226             : 
    6227          56 : static void cli_posix_mkdir_done(struct tevent_req *subreq)
    6228             : {
    6229          56 :         struct tevent_req *req = tevent_req_callback_data(
    6230             :                 subreq, struct tevent_req);
    6231           0 :         NTSTATUS status;
    6232           0 :         uint16_t fnum;
    6233             : 
    6234          56 :         status = cli_posix_open_internal_recv(subreq, &fnum);
    6235          56 :         TALLOC_FREE(subreq);
    6236          56 :         if (tevent_req_nterror(req, status)) {
    6237           8 :                 return;
    6238             :         }
    6239          48 :         tevent_req_done(req);
    6240             : }
    6241             : 
    6242          56 : NTSTATUS cli_posix_mkdir_recv(struct tevent_req *req)
    6243             : {
    6244          56 :         return tevent_req_simple_recv_ntstatus(req);
    6245             : }
    6246             : 
    6247          56 : NTSTATUS cli_posix_mkdir(struct cli_state *cli, const char *fname, mode_t mode)
    6248             : {
    6249          56 :         TALLOC_CTX *frame = talloc_stackframe();
    6250          56 :         struct tevent_context *ev = NULL;
    6251          56 :         struct tevent_req *req = NULL;
    6252          56 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    6253             : 
    6254          56 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    6255             :                 /*
    6256             :                  * Can't use sync call while an async call is in flight
    6257             :                  */
    6258           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    6259           0 :                 goto fail;
    6260             :         }
    6261             : 
    6262          56 :         ev = samba_tevent_context_init(frame);
    6263          56 :         if (ev == NULL) {
    6264           0 :                 goto fail;
    6265             :         }
    6266          56 :         req = cli_posix_mkdir_send(
    6267             :                 frame, ev, cli, fname, mode);
    6268          56 :         if (req == NULL) {
    6269           0 :                 goto fail;
    6270             :         }
    6271          56 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    6272           0 :                 goto fail;
    6273             :         }
    6274          56 :         status = cli_posix_mkdir_recv(req);
    6275          56 :  fail:
    6276          56 :         TALLOC_FREE(frame);
    6277          56 :         return status;
    6278             : }
    6279             : 
    6280             : /****************************************************************************
    6281             :  unlink or rmdir - POSIX semantics.
    6282             : ****************************************************************************/
    6283             : 
    6284             : struct cli_posix_unlink_internal_state {
    6285             :         uint8_t data[2];
    6286             : };
    6287             : 
    6288             : static void cli_posix_unlink_internal_done(struct tevent_req *subreq);
    6289             : 
    6290         572 : static struct tevent_req *cli_posix_unlink_internal_send(TALLOC_CTX *mem_ctx,
    6291             :                                         struct tevent_context *ev,
    6292             :                                         struct cli_state *cli,
    6293             :                                         const char *fname,
    6294             :                                         uint16_t level)
    6295             : {
    6296         572 :         struct tevent_req *req = NULL, *subreq = NULL;
    6297         572 :         struct cli_posix_unlink_internal_state *state = NULL;
    6298             : 
    6299         572 :         req = tevent_req_create(mem_ctx, &state,
    6300             :                                 struct cli_posix_unlink_internal_state);
    6301         572 :         if (req == NULL) {
    6302           0 :                 return NULL;
    6303             :         }
    6304             : 
    6305             :         /* Setup data word. */
    6306         572 :         SSVAL(state->data, 0, level);
    6307             : 
    6308         572 :         subreq = cli_setpathinfo_send(state, ev, cli,
    6309             :                                       SMB_POSIX_PATH_UNLINK,
    6310             :                                       fname,
    6311         572 :                                       state->data, sizeof(state->data));
    6312         572 :         if (tevent_req_nomem(subreq, req)) {
    6313           0 :                 return tevent_req_post(req, ev);
    6314             :         }
    6315         572 :         tevent_req_set_callback(subreq, cli_posix_unlink_internal_done, req);
    6316         572 :         return req;
    6317             : }
    6318             : 
    6319         572 : static void cli_posix_unlink_internal_done(struct tevent_req *subreq)
    6320             : {
    6321         572 :         NTSTATUS status = cli_setpathinfo_recv(subreq);
    6322         572 :         tevent_req_simple_finish_ntstatus(subreq, status);
    6323         572 : }
    6324             : 
    6325         572 : static NTSTATUS cli_posix_unlink_internal_recv(struct tevent_req *req)
    6326             : {
    6327         572 :         return tevent_req_simple_recv_ntstatus(req);
    6328             : }
    6329             : 
    6330             : struct cli_posix_unlink_state {
    6331             :         uint8_t dummy;
    6332             : };
    6333             : 
    6334             : static void cli_posix_unlink_done(struct tevent_req *subreq);
    6335             : 
    6336         472 : struct tevent_req *cli_posix_unlink_send(TALLOC_CTX *mem_ctx,
    6337             :                                         struct tevent_context *ev,
    6338             :                                         struct cli_state *cli,
    6339             :                                         const char *fname)
    6340             : {
    6341         472 :         struct tevent_req *req = NULL, *subreq = NULL;
    6342           0 :         struct cli_posix_unlink_state *state;
    6343             : 
    6344         472 :         req = tevent_req_create(
    6345             :                 mem_ctx, &state, struct cli_posix_unlink_state);
    6346         472 :         if (req == NULL) {
    6347           0 :                 return NULL;
    6348             :         }
    6349         472 :         subreq = cli_posix_unlink_internal_send(
    6350             :                 mem_ctx, ev, cli, fname, SMB_POSIX_UNLINK_FILE_TARGET);
    6351         472 :         if (tevent_req_nomem(subreq, req)) {
    6352           0 :                 return tevent_req_post(req, ev);
    6353             :         }
    6354         472 :         tevent_req_set_callback(subreq, cli_posix_unlink_done, req);
    6355         472 :         return req;
    6356             : }
    6357             : 
    6358         472 : static void cli_posix_unlink_done(struct tevent_req *subreq)
    6359             : {
    6360         472 :         NTSTATUS status = cli_posix_unlink_internal_recv(subreq);
    6361         472 :         tevent_req_simple_finish_ntstatus(subreq, status);
    6362         472 : }
    6363             : 
    6364         472 : NTSTATUS cli_posix_unlink_recv(struct tevent_req *req)
    6365             : {
    6366         472 :         return tevent_req_simple_recv_ntstatus(req);
    6367             : }
    6368             : 
    6369             : /****************************************************************************
    6370             :  unlink - POSIX semantics.
    6371             : ****************************************************************************/
    6372             : 
    6373         472 : NTSTATUS cli_posix_unlink(struct cli_state *cli, const char *fname)
    6374             : {
    6375         472 :         TALLOC_CTX *frame = talloc_stackframe();
    6376         472 :         struct tevent_context *ev = NULL;
    6377         472 :         struct tevent_req *req = NULL;
    6378         472 :         NTSTATUS status = NT_STATUS_OK;
    6379             : 
    6380         472 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    6381             :                 /*
    6382             :                  * Can't use sync call while an async call is in flight
    6383             :                  */
    6384           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    6385           0 :                 goto fail;
    6386             :         }
    6387             : 
    6388         472 :         ev = samba_tevent_context_init(frame);
    6389         472 :         if (ev == NULL) {
    6390           0 :                 status = NT_STATUS_NO_MEMORY;
    6391           0 :                 goto fail;
    6392             :         }
    6393             : 
    6394         472 :         req = cli_posix_unlink_send(frame,
    6395             :                                 ev,
    6396             :                                 cli,
    6397             :                                 fname);
    6398         472 :         if (req == NULL) {
    6399           0 :                 status = NT_STATUS_NO_MEMORY;
    6400           0 :                 goto fail;
    6401             :         }
    6402             : 
    6403         472 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    6404           0 :                 goto fail;
    6405             :         }
    6406             : 
    6407         472 :         status = cli_posix_unlink_recv(req);
    6408             : 
    6409         472 :  fail:
    6410         472 :         TALLOC_FREE(frame);
    6411         472 :         return status;
    6412             : }
    6413             : 
    6414             : /****************************************************************************
    6415             :  rmdir - POSIX semantics.
    6416             : ****************************************************************************/
    6417             : 
    6418             : struct cli_posix_rmdir_state {
    6419             :         uint8_t dummy;
    6420             : };
    6421             : 
    6422             : static void cli_posix_rmdir_done(struct tevent_req *subreq);
    6423             : 
    6424         100 : struct tevent_req *cli_posix_rmdir_send(TALLOC_CTX *mem_ctx,
    6425             :                                         struct tevent_context *ev,
    6426             :                                         struct cli_state *cli,
    6427             :                                         const char *fname)
    6428             : {
    6429         100 :         struct tevent_req *req = NULL, *subreq = NULL;
    6430           0 :         struct cli_posix_rmdir_state *state;
    6431             : 
    6432         100 :         req = tevent_req_create(mem_ctx, &state, struct cli_posix_rmdir_state);
    6433         100 :         if (req == NULL) {
    6434           0 :                 return NULL;
    6435             :         }
    6436         100 :         subreq = cli_posix_unlink_internal_send(
    6437             :                 mem_ctx, ev, cli, fname, SMB_POSIX_UNLINK_DIRECTORY_TARGET);
    6438         100 :         if (tevent_req_nomem(subreq, req)) {
    6439           0 :                 return tevent_req_post(req, ev);
    6440             :         }
    6441         100 :         tevent_req_set_callback(subreq, cli_posix_rmdir_done, req);
    6442         100 :         return req;
    6443             : }
    6444             : 
    6445         100 : static void cli_posix_rmdir_done(struct tevent_req *subreq)
    6446             : {
    6447         100 :         NTSTATUS status = cli_posix_unlink_internal_recv(subreq);
    6448         100 :         tevent_req_simple_finish_ntstatus(subreq, status);
    6449         100 : }
    6450             : 
    6451         100 : NTSTATUS cli_posix_rmdir_recv(struct tevent_req *req, TALLOC_CTX *mem_ctx)
    6452             : {
    6453         100 :         return tevent_req_simple_recv_ntstatus(req);
    6454             : }
    6455             : 
    6456         100 : NTSTATUS cli_posix_rmdir(struct cli_state *cli, const char *fname)
    6457             : {
    6458         100 :         TALLOC_CTX *frame = talloc_stackframe();
    6459         100 :         struct tevent_context *ev = NULL;
    6460         100 :         struct tevent_req *req = NULL;
    6461         100 :         NTSTATUS status = NT_STATUS_OK;
    6462             : 
    6463         100 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    6464             :                 /*
    6465             :                  * Can't use sync call while an async call is in flight
    6466             :                  */
    6467           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    6468           0 :                 goto fail;
    6469             :         }
    6470             : 
    6471         100 :         ev = samba_tevent_context_init(frame);
    6472         100 :         if (ev == NULL) {
    6473           0 :                 status = NT_STATUS_NO_MEMORY;
    6474           0 :                 goto fail;
    6475             :         }
    6476             : 
    6477         100 :         req = cli_posix_rmdir_send(frame,
    6478             :                                 ev,
    6479             :                                 cli,
    6480             :                                 fname);
    6481         100 :         if (req == NULL) {
    6482           0 :                 status = NT_STATUS_NO_MEMORY;
    6483           0 :                 goto fail;
    6484             :         }
    6485             : 
    6486         100 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    6487           0 :                 goto fail;
    6488             :         }
    6489             : 
    6490         100 :         status = cli_posix_rmdir_recv(req, frame);
    6491             : 
    6492         100 :  fail:
    6493         100 :         TALLOC_FREE(frame);
    6494         100 :         return status;
    6495             : }
    6496             : 
    6497             : /****************************************************************************
    6498             :  filechangenotify
    6499             : ****************************************************************************/
    6500             : 
    6501             : struct cli_notify_state {
    6502             :         struct tevent_req *subreq;
    6503             :         uint8_t setup[8];
    6504             :         uint32_t num_changes;
    6505             :         struct notify_change *changes;
    6506             : };
    6507             : 
    6508             : static void cli_notify_done(struct tevent_req *subreq);
    6509             : static void cli_notify_done_smb2(struct tevent_req *subreq);
    6510             : static bool cli_notify_cancel(struct tevent_req *req);
    6511             : 
    6512          42 : struct tevent_req *cli_notify_send(TALLOC_CTX *mem_ctx,
    6513             :                                    struct tevent_context *ev,
    6514             :                                    struct cli_state *cli, uint16_t fnum,
    6515             :                                    uint32_t buffer_size,
    6516             :                                    uint32_t completion_filter, bool recursive)
    6517             : {
    6518           0 :         struct tevent_req *req;
    6519           0 :         struct cli_notify_state *state;
    6520           0 :         unsigned old_timeout;
    6521             : 
    6522          42 :         req = tevent_req_create(mem_ctx, &state, struct cli_notify_state);
    6523          42 :         if (req == NULL) {
    6524           0 :                 return NULL;
    6525             :         }
    6526             : 
    6527          42 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    6528             :                 /*
    6529             :                  * Notifies should not time out
    6530             :                  */
    6531          42 :                 old_timeout = cli_set_timeout(cli, 0);
    6532             : 
    6533          42 :                 state->subreq = cli_smb2_notify_send(
    6534             :                         state,
    6535             :                         ev,
    6536             :                         cli,
    6537             :                         fnum,
    6538             :                         buffer_size,
    6539             :                         completion_filter,
    6540             :                         recursive);
    6541             : 
    6542          42 :                 cli_set_timeout(cli, old_timeout);
    6543             : 
    6544          42 :                 if (tevent_req_nomem(state->subreq, req)) {
    6545           0 :                         return tevent_req_post(req, ev);
    6546             :                 }
    6547          42 :                 tevent_req_set_callback(
    6548             :                         state->subreq, cli_notify_done_smb2, req);
    6549          42 :                 goto done;
    6550             :         }
    6551             : 
    6552           0 :         SIVAL(state->setup, 0, completion_filter);
    6553           0 :         SSVAL(state->setup, 4, fnum);
    6554           0 :         SSVAL(state->setup, 6, recursive);
    6555             : 
    6556             :         /*
    6557             :          * Notifies should not time out
    6558             :          */
    6559           0 :         old_timeout = cli_set_timeout(cli, 0);
    6560             : 
    6561           0 :         state->subreq = cli_trans_send(
    6562             :                 state,                  /* mem ctx. */
    6563             :                 ev,                     /* event ctx. */
    6564             :                 cli,                    /* cli_state. */
    6565             :                 0,                      /* additional_flags2 */
    6566             :                 SMBnttrans,             /* cmd. */
    6567             :                 NULL,                   /* pipe name. */
    6568             :                 -1,                     /* fid. */
    6569             :                 NT_TRANSACT_NOTIFY_CHANGE, /* function. */
    6570             :                 0,                      /* flags. */
    6571           0 :                 (uint16_t *)state->setup, /* setup. */
    6572             :                 4,                      /* num setup uint16_t words. */
    6573             :                 0,                      /* max returned setup. */
    6574             :                 NULL,                   /* param. */
    6575             :                 0,                      /* num param. */
    6576             :                 buffer_size,            /* max returned param. */
    6577             :                 NULL,                   /* data. */
    6578             :                 0,                      /* num data. */
    6579             :                 0);                     /* max returned data. */
    6580             : 
    6581           0 :         cli_set_timeout(cli, old_timeout);
    6582             : 
    6583           0 :         if (tevent_req_nomem(state->subreq, req)) {
    6584           0 :                 return tevent_req_post(req, ev);
    6585             :         }
    6586           0 :         tevent_req_set_callback(state->subreq, cli_notify_done, req);
    6587          42 : done:
    6588          42 :         tevent_req_set_cancel_fn(req, cli_notify_cancel);
    6589          42 :         return req;
    6590             : }
    6591             : 
    6592           0 : static bool cli_notify_cancel(struct tevent_req *req)
    6593             : {
    6594           0 :         struct cli_notify_state *state = tevent_req_data(
    6595             :                 req, struct cli_notify_state);
    6596           0 :         bool ok;
    6597             : 
    6598           0 :         ok = tevent_req_cancel(state->subreq);
    6599           0 :         return ok;
    6600             : }
    6601             : 
    6602           0 : static void cli_notify_done(struct tevent_req *subreq)
    6603             : {
    6604           0 :         struct tevent_req *req = tevent_req_callback_data(
    6605             :                 subreq, struct tevent_req);
    6606           0 :         struct cli_notify_state *state = tevent_req_data(
    6607             :                 req, struct cli_notify_state);
    6608           0 :         NTSTATUS status;
    6609           0 :         uint8_t *params;
    6610           0 :         uint32_t i, ofs, num_params;
    6611           0 :         uint16_t flags2;
    6612             : 
    6613           0 :         status = cli_trans_recv(subreq, talloc_tos(), &flags2, NULL, 0, NULL,
    6614             :                                 &params, 0, &num_params, NULL, 0, NULL);
    6615           0 :         TALLOC_FREE(subreq);
    6616           0 :         state->subreq = NULL;
    6617           0 :         if (tevent_req_nterror(req, status)) {
    6618           0 :                 DEBUG(10, ("cli_trans_recv returned %s\n", nt_errstr(status)));
    6619           0 :                 return;
    6620             :         }
    6621             : 
    6622           0 :         state->num_changes = 0;
    6623           0 :         ofs = 0;
    6624             : 
    6625           0 :         while (num_params - ofs > 12) {
    6626           0 :                 uint32_t next = IVAL(params, ofs);
    6627           0 :                 state->num_changes += 1;
    6628             : 
    6629           0 :                 if ((next == 0) || (ofs+next >= num_params)) {
    6630             :                         break;
    6631             :                 }
    6632           0 :                 ofs += next;
    6633             :         }
    6634             : 
    6635           0 :         state->changes = talloc_array(state, struct notify_change,
    6636             :                                       state->num_changes);
    6637           0 :         if (tevent_req_nomem(state->changes, req)) {
    6638           0 :                 TALLOC_FREE(params);
    6639           0 :                 return;
    6640             :         }
    6641             : 
    6642           0 :         ofs = 0;
    6643             : 
    6644           0 :         for (i=0; i<state->num_changes; i++) {
    6645           0 :                 uint32_t next = IVAL(params, ofs);
    6646           0 :                 uint32_t len = IVAL(params, ofs+8);
    6647           0 :                 ssize_t ret;
    6648           0 :                 char *name;
    6649             : 
    6650           0 :                 if (smb_buffer_oob(num_params, ofs + 12, len)) {
    6651           0 :                         TALLOC_FREE(params);
    6652           0 :                         tevent_req_nterror(
    6653             :                                 req, NT_STATUS_INVALID_NETWORK_RESPONSE);
    6654           0 :                         return;
    6655             :                 }
    6656             : 
    6657           0 :                 state->changes[i].action = IVAL(params, ofs+4);
    6658           0 :                 ret = pull_string_talloc(state->changes,
    6659             :                                          (char *)params,
    6660             :                                          flags2,
    6661             :                                          &name,
    6662           0 :                                          params+ofs+12,
    6663             :                                          len,
    6664             :                                          STR_TERMINATE|STR_UNICODE);
    6665           0 :                 if (ret == -1) {
    6666           0 :                         TALLOC_FREE(params);
    6667           0 :                         tevent_req_nterror(req, NT_STATUS_INTERNAL_ERROR);
    6668           0 :                         return;
    6669             :                 }
    6670           0 :                 state->changes[i].name = name;
    6671           0 :                 ofs += next;
    6672             :         }
    6673             : 
    6674           0 :         TALLOC_FREE(params);
    6675           0 :         tevent_req_done(req);
    6676             : }
    6677             : 
    6678          42 : static void cli_notify_done_smb2(struct tevent_req *subreq)
    6679             : {
    6680          42 :         struct tevent_req *req = tevent_req_callback_data(
    6681             :                 subreq, struct tevent_req);
    6682          42 :         struct cli_notify_state *state = tevent_req_data(
    6683             :                 req, struct cli_notify_state);
    6684           0 :         NTSTATUS status;
    6685             : 
    6686          42 :         status = cli_smb2_notify_recv(
    6687             :                 subreq,
    6688             :                 state,
    6689             :                 &state->changes,
    6690             :                 &state->num_changes);
    6691          42 :         TALLOC_FREE(subreq);
    6692          42 :         if (tevent_req_nterror(req, status)) {
    6693          12 :                 return;
    6694             :         }
    6695          30 :         tevent_req_done(req);
    6696             : }
    6697             : 
    6698          42 : NTSTATUS cli_notify_recv(struct tevent_req *req, TALLOC_CTX *mem_ctx,
    6699             :                          uint32_t *pnum_changes,
    6700             :                          struct notify_change **pchanges)
    6701             : {
    6702          42 :         struct cli_notify_state *state = tevent_req_data(
    6703             :                 req, struct cli_notify_state);
    6704           0 :         NTSTATUS status;
    6705             : 
    6706          42 :         if (tevent_req_is_nterror(req, &status)) {
    6707          12 :                 return status;
    6708             :         }
    6709             : 
    6710          30 :         *pnum_changes = state->num_changes;
    6711          30 :         *pchanges = talloc_move(mem_ctx, &state->changes);
    6712          30 :         return NT_STATUS_OK;
    6713             : }
    6714             : 
    6715           0 : NTSTATUS cli_notify(struct cli_state *cli, uint16_t fnum, uint32_t buffer_size,
    6716             :                     uint32_t completion_filter, bool recursive,
    6717             :                     TALLOC_CTX *mem_ctx, uint32_t *pnum_changes,
    6718             :                     struct notify_change **pchanges)
    6719             : {
    6720           0 :         TALLOC_CTX *frame;
    6721           0 :         struct tevent_context *ev;
    6722           0 :         struct tevent_req *req;
    6723           0 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    6724             : 
    6725           0 :         frame = talloc_stackframe();
    6726             : 
    6727           0 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    6728             :                 /*
    6729             :                  * Can't use sync call while an async call is in flight
    6730             :                  */
    6731           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    6732           0 :                 goto fail;
    6733             :         }
    6734           0 :         ev = samba_tevent_context_init(frame);
    6735           0 :         if (ev == NULL) {
    6736           0 :                 goto fail;
    6737             :         }
    6738           0 :         req = cli_notify_send(ev, ev, cli, fnum, buffer_size,
    6739             :                               completion_filter, recursive);
    6740           0 :         if (req == NULL) {
    6741           0 :                 goto fail;
    6742             :         }
    6743           0 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    6744           0 :                 goto fail;
    6745             :         }
    6746           0 :         status = cli_notify_recv(req, mem_ctx, pnum_changes, pchanges);
    6747           0 :  fail:
    6748           0 :         TALLOC_FREE(frame);
    6749           0 :         return status;
    6750             : }
    6751             : 
    6752             : struct cli_qpathinfo_state {
    6753             :         uint8_t *param;
    6754             :         uint8_t *data;
    6755             :         uint16_t setup[1];
    6756             :         uint32_t min_rdata;
    6757             :         uint8_t *rdata;
    6758             :         uint32_t num_rdata;
    6759             : };
    6760             : 
    6761             : static void cli_qpathinfo_done(struct tevent_req *subreq);
    6762             : static void cli_qpathinfo_done2(struct tevent_req *subreq);
    6763             : 
    6764        8492 : struct tevent_req *cli_qpathinfo_send(TALLOC_CTX *mem_ctx,
    6765             :                                       struct tevent_context *ev,
    6766             :                                       struct cli_state *cli, const char *fname,
    6767             :                                       uint16_t level, uint32_t min_rdata,
    6768             :                                       uint32_t max_rdata)
    6769             : {
    6770           0 :         struct tevent_req *req, *subreq;
    6771           0 :         struct cli_qpathinfo_state *state;
    6772        8492 :         uint16_t additional_flags2 = 0;
    6773        8492 :         char *fname_cp = NULL;
    6774             : 
    6775        8492 :         req = tevent_req_create(mem_ctx, &state, struct cli_qpathinfo_state);
    6776        8492 :         if (req == NULL) {
    6777           0 :                 return NULL;
    6778             :         }
    6779             : 
    6780        8492 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    6781         852 :                 uint16_t smb2_level = 0;
    6782             : 
    6783         852 :                 switch (level) {
    6784         852 :                 case SMB_QUERY_FILE_ALT_NAME_INFO:
    6785         852 :                         smb2_level = FSCC_FILE_ALTERNATE_NAME_INFORMATION;
    6786         852 :                         break;
    6787           0 :                 default:
    6788           0 :                         tevent_req_nterror(req, NT_STATUS_INVALID_LEVEL);
    6789           0 :                         return tevent_req_post(req, ev);
    6790             :                 }
    6791             : 
    6792         852 :                 subreq = cli_smb2_qpathinfo_send(state,
    6793             :                                                  ev,
    6794             :                                                  cli,
    6795             :                                                  fname,
    6796             :                                                  smb2_level,
    6797             :                                                  min_rdata,
    6798             :                                                  max_rdata);
    6799         852 :                 if (tevent_req_nomem(subreq, req)) {
    6800           0 :                         return tevent_req_post(req, ev);
    6801             :                 }
    6802         852 :                 tevent_req_set_callback(subreq, cli_qpathinfo_done2, req);
    6803         852 :                 return req;
    6804             :         }
    6805             : 
    6806        7640 :         state->min_rdata = min_rdata;
    6807        7640 :         SSVAL(state->setup, 0, TRANSACT2_QPATHINFO);
    6808             : 
    6809        7640 :         state->param = talloc_zero_array(state, uint8_t, 6);
    6810        7640 :         if (tevent_req_nomem(state->param, req)) {
    6811           0 :                 return tevent_req_post(req, ev);
    6812             :         }
    6813        7640 :         SSVAL(state->param, 0, level);
    6814             :         /*
    6815             :          * qpathinfo on a DFS share must use DFS names.
    6816             :          */
    6817        7640 :         fname_cp = smb1_dfs_share_path(state, cli, fname);
    6818        7640 :         if (tevent_req_nomem(fname_cp, req)) {
    6819           0 :                 return tevent_req_post(req, ev);
    6820             :         }
    6821        7640 :         state->param = trans2_bytes_push_str(state->param,
    6822        7640 :                                              smbXcli_conn_use_unicode(cli->conn),
    6823             :                                              fname_cp,
    6824        7640 :                                              strlen(fname_cp)+1,
    6825             :                                              NULL);
    6826        7640 :         if (tevent_req_nomem(state->param, req)) {
    6827           0 :                 return tevent_req_post(req, ev);
    6828             :         }
    6829             : 
    6830        7640 :         if (clistr_is_previous_version_path(fname) &&
    6831        3104 :                         !INFO_LEVEL_IS_UNIX(level)) {
    6832        3104 :                 additional_flags2 = FLAGS2_REPARSE_PATH;
    6833             :         }
    6834             : 
    6835        7640 :         subreq = cli_trans_send(
    6836             :                 state,                  /* mem ctx. */
    6837             :                 ev,                     /* event ctx. */
    6838             :                 cli,                    /* cli_state. */
    6839             :                 additional_flags2,      /* additional_flags2 */
    6840             :                 SMBtrans2,              /* cmd. */
    6841             :                 NULL,                   /* pipe name. */
    6842             :                 -1,                     /* fid. */
    6843             :                 0,                      /* function. */
    6844             :                 0,                      /* flags. */
    6845        7640 :                 state->setup,                /* setup. */
    6846             :                 1,                      /* num setup uint16_t words. */
    6847             :                 0,                      /* max returned setup. */
    6848        7640 :                 state->param,                /* param. */
    6849        7640 :                 talloc_get_size(state->param),       /* num param. */
    6850             :                 2,                      /* max returned param. */
    6851             :                 NULL,                   /* data. */
    6852             :                 0,                      /* num data. */
    6853             :                 max_rdata);             /* max returned data. */
    6854             : 
    6855        7640 :         if (tevent_req_nomem(subreq, req)) {
    6856           0 :                 return tevent_req_post(req, ev);
    6857             :         }
    6858        7640 :         tevent_req_set_callback(subreq, cli_qpathinfo_done, req);
    6859        7640 :         return req;
    6860             : }
    6861             : 
    6862        7640 : static void cli_qpathinfo_done(struct tevent_req *subreq)
    6863             : {
    6864        7640 :         struct tevent_req *req = tevent_req_callback_data(
    6865             :                 subreq, struct tevent_req);
    6866        7640 :         struct cli_qpathinfo_state *state = tevent_req_data(
    6867             :                 req, struct cli_qpathinfo_state);
    6868           0 :         NTSTATUS status;
    6869             : 
    6870        7640 :         status = cli_trans_recv(subreq, state, NULL, NULL, 0, NULL,
    6871             :                                 NULL, 0, NULL,
    6872             :                                 &state->rdata, state->min_rdata,
    6873             :                                 &state->num_rdata);
    6874        7640 :         if (tevent_req_nterror(req, status)) {
    6875        1850 :                 return;
    6876             :         }
    6877        5790 :         tevent_req_done(req);
    6878             : }
    6879             : 
    6880         852 : static void cli_qpathinfo_done2(struct tevent_req *subreq)
    6881             : {
    6882           0 :         struct tevent_req *req =
    6883         852 :                 tevent_req_callback_data(subreq, struct tevent_req);
    6884           0 :         struct cli_qpathinfo_state *state =
    6885         852 :                 tevent_req_data(req, struct cli_qpathinfo_state);
    6886           0 :         NTSTATUS status;
    6887             : 
    6888         852 :         status = cli_smb2_qpathinfo_recv(subreq,
    6889             :                                          state,
    6890             :                                          &state->rdata,
    6891             :                                          &state->num_rdata);
    6892         852 :         if (tevent_req_nterror(req, status)) {
    6893          32 :                 return;
    6894             :         }
    6895         820 :         tevent_req_done(req);
    6896             : }
    6897             : 
    6898        8492 : NTSTATUS cli_qpathinfo_recv(struct tevent_req *req, TALLOC_CTX *mem_ctx,
    6899             :                             uint8_t **rdata, uint32_t *num_rdata)
    6900             : {
    6901        8492 :         struct cli_qpathinfo_state *state = tevent_req_data(
    6902             :                 req, struct cli_qpathinfo_state);
    6903           0 :         NTSTATUS status;
    6904             : 
    6905        8492 :         if (tevent_req_is_nterror(req, &status)) {
    6906        1882 :                 return status;
    6907             :         }
    6908        6610 :         if (rdata != NULL) {
    6909        6610 :                 *rdata = talloc_move(mem_ctx, &state->rdata);
    6910             :         } else {
    6911           0 :                 TALLOC_FREE(state->rdata);
    6912             :         }
    6913        6610 :         if (num_rdata != NULL) {
    6914        6610 :                 *num_rdata = state->num_rdata;
    6915             :         }
    6916        6610 :         return NT_STATUS_OK;
    6917             : }
    6918             : 
    6919        3749 : NTSTATUS cli_qpathinfo(TALLOC_CTX *mem_ctx, struct cli_state *cli,
    6920             :                        const char *fname, uint16_t level, uint32_t min_rdata,
    6921             :                        uint32_t max_rdata,
    6922             :                        uint8_t **rdata, uint32_t *num_rdata)
    6923             : {
    6924        3749 :         TALLOC_CTX *frame = talloc_stackframe();
    6925           0 :         struct tevent_context *ev;
    6926           0 :         struct tevent_req *req;
    6927        3749 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    6928             : 
    6929        3749 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    6930             :                 /*
    6931             :                  * Can't use sync call while an async call is in flight
    6932             :                  */
    6933           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    6934           0 :                 goto fail;
    6935             :         }
    6936        3749 :         ev = samba_tevent_context_init(frame);
    6937        3749 :         if (ev == NULL) {
    6938           0 :                 goto fail;
    6939             :         }
    6940        3749 :         req = cli_qpathinfo_send(frame, ev, cli, fname, level, min_rdata,
    6941             :                                  max_rdata);
    6942        3749 :         if (req == NULL) {
    6943           0 :                 goto fail;
    6944             :         }
    6945        3749 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    6946           0 :                 goto fail;
    6947             :         }
    6948        3749 :         status = cli_qpathinfo_recv(req, mem_ctx, rdata, num_rdata);
    6949        3749 :  fail:
    6950        3749 :         TALLOC_FREE(frame);
    6951        3749 :         return status;
    6952             : }
    6953             : 
    6954             : struct cli_qfileinfo_state {
    6955             :         uint16_t setup[1];
    6956             :         uint8_t param[4];
    6957             :         uint8_t *data;
    6958             :         uint16_t recv_flags2;
    6959             :         uint32_t min_rdata;
    6960             :         uint8_t *rdata;
    6961             :         uint32_t num_rdata;
    6962             : };
    6963             : 
    6964             : static void cli_qfileinfo_done(struct tevent_req *subreq);
    6965             : 
    6966         552 : struct tevent_req *cli_qfileinfo_send(TALLOC_CTX *mem_ctx,
    6967             :                                       struct tevent_context *ev,
    6968             :                                       struct cli_state *cli, uint16_t fnum,
    6969             :                                       uint16_t level, uint32_t min_rdata,
    6970             :                                       uint32_t max_rdata)
    6971             : {
    6972           0 :         struct tevent_req *req, *subreq;
    6973           0 :         struct cli_qfileinfo_state *state;
    6974             : 
    6975         552 :         req = tevent_req_create(mem_ctx, &state, struct cli_qfileinfo_state);
    6976         552 :         if (req == NULL) {
    6977           0 :                 return NULL;
    6978             :         }
    6979         552 :         state->min_rdata = min_rdata;
    6980         552 :         SSVAL(state->param, 0, fnum);
    6981         552 :         SSVAL(state->param, 2, level);
    6982         552 :         SSVAL(state->setup, 0, TRANSACT2_QFILEINFO);
    6983             : 
    6984         552 :         subreq = cli_trans_send(
    6985             :                 state,                  /* mem ctx. */
    6986             :                 ev,                     /* event ctx. */
    6987             :                 cli,                    /* cli_state. */
    6988             :                 0,                      /* additional_flags2 */
    6989             :                 SMBtrans2,              /* cmd. */
    6990             :                 NULL,                   /* pipe name. */
    6991             :                 -1,                     /* fid. */
    6992             :                 0,                      /* function. */
    6993             :                 0,                      /* flags. */
    6994         552 :                 state->setup,                /* setup. */
    6995             :                 1,                      /* num setup uint16_t words. */
    6996             :                 0,                      /* max returned setup. */
    6997         552 :                 state->param,                /* param. */
    6998             :                 sizeof(state->param),        /* num param. */
    6999             :                 2,                      /* max returned param. */
    7000             :                 NULL,                   /* data. */
    7001             :                 0,                      /* num data. */
    7002             :                 max_rdata);             /* max returned data. */
    7003             : 
    7004         552 :         if (tevent_req_nomem(subreq, req)) {
    7005           0 :                 return tevent_req_post(req, ev);
    7006             :         }
    7007         552 :         tevent_req_set_callback(subreq, cli_qfileinfo_done, req);
    7008         552 :         return req;
    7009             : }
    7010             : 
    7011         552 : static void cli_qfileinfo_done(struct tevent_req *subreq)
    7012             : {
    7013         552 :         struct tevent_req *req = tevent_req_callback_data(
    7014             :                 subreq, struct tevent_req);
    7015         552 :         struct cli_qfileinfo_state *state = tevent_req_data(
    7016             :                 req, struct cli_qfileinfo_state);
    7017           0 :         NTSTATUS status;
    7018             : 
    7019         552 :         status = cli_trans_recv(subreq, state,
    7020             :                                 &state->recv_flags2,
    7021             :                                 NULL, 0, NULL,
    7022             :                                 NULL, 0, NULL,
    7023             :                                 &state->rdata, state->min_rdata,
    7024             :                                 &state->num_rdata);
    7025         552 :         if (tevent_req_nterror(req, status)) {
    7026         180 :                 return;
    7027             :         }
    7028         372 :         tevent_req_done(req);
    7029             : }
    7030             : 
    7031         552 : NTSTATUS cli_qfileinfo_recv(struct tevent_req *req, TALLOC_CTX *mem_ctx,
    7032             :                             uint16_t *recv_flags2,
    7033             :                             uint8_t **rdata, uint32_t *num_rdata)
    7034             : {
    7035         552 :         struct cli_qfileinfo_state *state = tevent_req_data(
    7036             :                 req, struct cli_qfileinfo_state);
    7037           0 :         NTSTATUS status;
    7038             : 
    7039         552 :         if (tevent_req_is_nterror(req, &status)) {
    7040         180 :                 return status;
    7041             :         }
    7042             : 
    7043         372 :         if (recv_flags2 != NULL) {
    7044           5 :                 *recv_flags2 = state->recv_flags2;
    7045             :         }
    7046         372 :         if (rdata != NULL) {
    7047         372 :                 *rdata = talloc_move(mem_ctx, &state->rdata);
    7048             :         }
    7049         372 :         if (num_rdata != NULL) {
    7050         372 :                 *num_rdata = state->num_rdata;
    7051             :         }
    7052             : 
    7053         372 :         tevent_req_received(req);
    7054         372 :         return NT_STATUS_OK;
    7055             : }
    7056             : 
    7057         185 : NTSTATUS cli_qfileinfo(TALLOC_CTX *mem_ctx, struct cli_state *cli,
    7058             :                        uint16_t fnum, uint16_t level, uint32_t min_rdata,
    7059             :                        uint32_t max_rdata, uint16_t *recv_flags2,
    7060             :                        uint8_t **rdata, uint32_t *num_rdata)
    7061             : {
    7062         185 :         TALLOC_CTX *frame = talloc_stackframe();
    7063           0 :         struct tevent_context *ev;
    7064           0 :         struct tevent_req *req;
    7065         185 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    7066             : 
    7067         185 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    7068             :                 /*
    7069             :                  * Can't use sync call while an async call is in flight
    7070             :                  */
    7071           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    7072           0 :                 goto fail;
    7073             :         }
    7074         185 :         ev = samba_tevent_context_init(frame);
    7075         185 :         if (ev == NULL) {
    7076           0 :                 goto fail;
    7077             :         }
    7078         185 :         req = cli_qfileinfo_send(frame, ev, cli, fnum, level, min_rdata,
    7079             :                                  max_rdata);
    7080         185 :         if (req == NULL) {
    7081           0 :                 goto fail;
    7082             :         }
    7083         185 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    7084           0 :                 goto fail;
    7085             :         }
    7086         185 :         status = cli_qfileinfo_recv(req, mem_ctx, recv_flags2, rdata, num_rdata);
    7087         185 :  fail:
    7088         185 :         TALLOC_FREE(frame);
    7089         185 :         return status;
    7090             : }
    7091             : 
    7092             : struct cli_flush_state {
    7093             :         uint16_t vwv[1];
    7094             : };
    7095             : 
    7096             : static void cli_flush_done(struct tevent_req *subreq);
    7097             : 
    7098           0 : struct tevent_req *cli_flush_send(TALLOC_CTX *mem_ctx,
    7099             :                                   struct tevent_context *ev,
    7100             :                                   struct cli_state *cli,
    7101             :                                   uint16_t fnum)
    7102             : {
    7103           0 :         struct tevent_req *req, *subreq;
    7104           0 :         struct cli_flush_state *state;
    7105             : 
    7106           0 :         req = tevent_req_create(mem_ctx, &state, struct cli_flush_state);
    7107           0 :         if (req == NULL) {
    7108           0 :                 return NULL;
    7109             :         }
    7110           0 :         SSVAL(state->vwv + 0, 0, fnum);
    7111             : 
    7112           0 :         subreq = cli_smb_send(state, ev, cli, SMBflush, 0, 0, 1, state->vwv,
    7113             :                               0, NULL);
    7114           0 :         if (tevent_req_nomem(subreq, req)) {
    7115           0 :                 return tevent_req_post(req, ev);
    7116             :         }
    7117           0 :         tevent_req_set_callback(subreq, cli_flush_done, req);
    7118           0 :         return req;
    7119             : }
    7120             : 
    7121           0 : static void cli_flush_done(struct tevent_req *subreq)
    7122             : {
    7123           0 :         struct tevent_req *req = tevent_req_callback_data(
    7124             :                 subreq, struct tevent_req);
    7125           0 :         NTSTATUS status;
    7126             : 
    7127           0 :         status = cli_smb_recv(subreq, NULL, NULL, 0, NULL, NULL, NULL, NULL);
    7128           0 :         TALLOC_FREE(subreq);
    7129           0 :         if (tevent_req_nterror(req, status)) {
    7130           0 :                 return;
    7131             :         }
    7132           0 :         tevent_req_done(req);
    7133             : }
    7134             : 
    7135           0 : NTSTATUS cli_flush_recv(struct tevent_req *req)
    7136             : {
    7137           0 :         return tevent_req_simple_recv_ntstatus(req);
    7138             : }
    7139             : 
    7140           0 : NTSTATUS cli_flush(TALLOC_CTX *mem_ctx, struct cli_state *cli, uint16_t fnum)
    7141             : {
    7142           0 :         TALLOC_CTX *frame = talloc_stackframe();
    7143           0 :         struct tevent_context *ev;
    7144           0 :         struct tevent_req *req;
    7145           0 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    7146             : 
    7147           0 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    7148             :                 /*
    7149             :                  * Can't use sync call while an async call is in flight
    7150             :                  */
    7151           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    7152           0 :                 goto fail;
    7153             :         }
    7154           0 :         ev = samba_tevent_context_init(frame);
    7155           0 :         if (ev == NULL) {
    7156           0 :                 goto fail;
    7157             :         }
    7158           0 :         req = cli_flush_send(frame, ev, cli, fnum);
    7159           0 :         if (req == NULL) {
    7160           0 :                 goto fail;
    7161             :         }
    7162           0 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    7163           0 :                 goto fail;
    7164             :         }
    7165           0 :         status = cli_flush_recv(req);
    7166           0 :  fail:
    7167           0 :         TALLOC_FREE(frame);
    7168           0 :         return status;
    7169             : }
    7170             : 
    7171             : struct cli_shadow_copy_data_state {
    7172             :         uint16_t setup[4];
    7173             :         uint8_t *data;
    7174             :         uint32_t num_data;
    7175             :         bool get_names;
    7176             : };
    7177             : 
    7178             : static void cli_shadow_copy_data_done(struct tevent_req *subreq);
    7179             : 
    7180        1302 : struct tevent_req *cli_shadow_copy_data_send(TALLOC_CTX *mem_ctx,
    7181             :                                              struct tevent_context *ev,
    7182             :                                              struct cli_state *cli,
    7183             :                                              uint16_t fnum,
    7184             :                                              bool get_names)
    7185             : {
    7186           0 :         struct tevent_req *req, *subreq;
    7187           0 :         struct cli_shadow_copy_data_state *state;
    7188           0 :         uint32_t ret_size;
    7189             : 
    7190        1302 :         req = tevent_req_create(mem_ctx, &state,
    7191             :                                 struct cli_shadow_copy_data_state);
    7192        1302 :         if (req == NULL) {
    7193           0 :                 return NULL;
    7194             :         }
    7195        1302 :         state->get_names = get_names;
    7196        1302 :         ret_size = get_names ? CLI_BUFFER_SIZE : 16;
    7197             : 
    7198        1302 :         SIVAL(state->setup + 0, 0, FSCTL_GET_SHADOW_COPY_DATA);
    7199        1302 :         SSVAL(state->setup + 2, 0, fnum);
    7200        1302 :         SCVAL(state->setup + 3, 0, 1); /* isFsctl */
    7201        1302 :         SCVAL(state->setup + 3, 1, 0); /* compfilter, isFlags (WSSP) */
    7202             : 
    7203        1302 :         subreq = cli_trans_send(
    7204             :                 state, ev, cli, 0, SMBnttrans, NULL, 0, NT_TRANSACT_IOCTL, 0,
    7205        1302 :                 state->setup, ARRAY_SIZE(state->setup),
    7206             :                 ARRAY_SIZE(state->setup),
    7207             :                 NULL, 0, 0,
    7208             :                 NULL, 0, ret_size);
    7209        1302 :         if (tevent_req_nomem(subreq, req)) {
    7210           0 :                 return tevent_req_post(req, ev);
    7211             :         }
    7212        1302 :         tevent_req_set_callback(subreq, cli_shadow_copy_data_done, req);
    7213        1302 :         return req;
    7214             : }
    7215             : 
    7216        1302 : static void cli_shadow_copy_data_done(struct tevent_req *subreq)
    7217             : {
    7218        1302 :         struct tevent_req *req = tevent_req_callback_data(
    7219             :                 subreq, struct tevent_req);
    7220        1302 :         struct cli_shadow_copy_data_state *state = tevent_req_data(
    7221             :                 req, struct cli_shadow_copy_data_state);
    7222           0 :         NTSTATUS status;
    7223             : 
    7224        1302 :         status = cli_trans_recv(subreq, state, NULL,
    7225             :                                 NULL, 0, NULL, /* setup */
    7226             :                                 NULL, 0, NULL, /* param */
    7227             :                                 &state->data, 12, &state->num_data);
    7228        1302 :         TALLOC_FREE(subreq);
    7229        1302 :         if (tevent_req_nterror(req, status)) {
    7230          66 :                 return;
    7231             :         }
    7232        1236 :         tevent_req_done(req);
    7233             : }
    7234             : 
    7235        1302 : NTSTATUS cli_shadow_copy_data_recv(struct tevent_req *req, TALLOC_CTX *mem_ctx,
    7236             :                                    char ***pnames, int *pnum_names)
    7237             : {
    7238        1302 :         struct cli_shadow_copy_data_state *state = tevent_req_data(
    7239             :                 req, struct cli_shadow_copy_data_state);
    7240        1302 :         char **names = NULL;
    7241           0 :         uint32_t i, num_names;
    7242           0 :         uint32_t dlength;
    7243        1302 :         uint8_t *endp = NULL;
    7244           0 :         NTSTATUS status;
    7245             : 
    7246        1302 :         if (tevent_req_is_nterror(req, &status)) {
    7247          66 :                 return status;
    7248             :         }
    7249             : 
    7250        1236 :         if (state->num_data < 16) {
    7251           0 :                 return NT_STATUS_INVALID_NETWORK_RESPONSE;
    7252             :         }
    7253             : 
    7254        1236 :         num_names = IVAL(state->data, 4);
    7255        1236 :         dlength = IVAL(state->data, 8);
    7256             : 
    7257        1236 :         if (num_names > 0x7FFFFFFF) {
    7258           0 :                 return NT_STATUS_INVALID_NETWORK_RESPONSE;
    7259             :         }
    7260             : 
    7261        1236 :         if (!state->get_names) {
    7262         618 :                 *pnum_names = (int)num_names;
    7263         618 :                 return NT_STATUS_OK;
    7264             :         }
    7265             : 
    7266         618 :         if (dlength + 12 < 12) {
    7267           0 :                 return NT_STATUS_INVALID_NETWORK_RESPONSE;
    7268             :         }
    7269         618 :         if (dlength + 12 > state->num_data) {
    7270           0 :                 return NT_STATUS_INVALID_NETWORK_RESPONSE;
    7271             :         }
    7272           0 :         if (state->num_data + (2 * sizeof(SHADOW_COPY_LABEL)) <
    7273             :                         state->num_data) {
    7274             :                 return NT_STATUS_INVALID_NETWORK_RESPONSE;
    7275             :         }
    7276             : 
    7277         618 :         names = talloc_array(mem_ctx, char *, num_names);
    7278         618 :         if (names == NULL) {
    7279           0 :                 return NT_STATUS_NO_MEMORY;
    7280             :         }
    7281             : 
    7282         618 :         endp = state->data + state->num_data;
    7283             : 
    7284        2458 :         for (i=0; i<num_names; i++) {
    7285           0 :                 bool ret;
    7286           0 :                 uint8_t *src;
    7287           0 :                 size_t converted_size;
    7288             : 
    7289        1840 :                 src = state->data + 12 + i * 2 * sizeof(SHADOW_COPY_LABEL);
    7290             : 
    7291        1840 :                 if (src + (2 * sizeof(SHADOW_COPY_LABEL)) > endp) {
    7292           0 :                         return NT_STATUS_INVALID_NETWORK_RESPONSE;
    7293             :                 }
    7294             : 
    7295        1840 :                 ret = convert_string_talloc(
    7296             :                         names, CH_UTF16LE, CH_UNIX,
    7297             :                         src, 2 * sizeof(SHADOW_COPY_LABEL),
    7298        1840 :                         &names[i], &converted_size);
    7299        1840 :                 if (!ret) {
    7300           0 :                         TALLOC_FREE(names);
    7301           0 :                         return NT_STATUS_INVALID_NETWORK_RESPONSE;
    7302             :                 }
    7303             :         }
    7304         618 :         *pnum_names = (int)num_names;
    7305         618 :         *pnames = names;
    7306         618 :         return NT_STATUS_OK;
    7307             : }
    7308             : 
    7309        2742 : NTSTATUS cli_shadow_copy_data(TALLOC_CTX *mem_ctx, struct cli_state *cli,
    7310             :                               uint16_t fnum, bool get_names,
    7311             :                               char ***pnames, int *pnum_names)
    7312             : {
    7313        2742 :         TALLOC_CTX *frame = NULL;
    7314           0 :         struct tevent_context *ev;
    7315           0 :         struct tevent_req *req;
    7316        2742 :         NTSTATUS status = NT_STATUS_NO_MEMORY;
    7317             : 
    7318        2742 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    7319        1440 :                 return cli_smb2_shadow_copy_data(mem_ctx,
    7320             :                                         cli,
    7321             :                                         fnum,
    7322             :                                         get_names,
    7323             :                                         pnames,
    7324             :                                         pnum_names);
    7325             :         }
    7326             : 
    7327        1302 :         frame = talloc_stackframe();
    7328             : 
    7329        1302 :         if (smbXcli_conn_has_async_calls(cli->conn)) {
    7330             :                 /*
    7331             :                  * Can't use sync call while an async call is in flight
    7332             :                  */
    7333           0 :                 status = NT_STATUS_INVALID_PARAMETER;
    7334           0 :                 goto fail;
    7335             :         }
    7336        1302 :         ev = samba_tevent_context_init(frame);
    7337        1302 :         if (ev == NULL) {
    7338           0 :                 goto fail;
    7339             :         }
    7340        1302 :         req = cli_shadow_copy_data_send(frame, ev, cli, fnum, get_names);
    7341        1302 :         if (req == NULL) {
    7342           0 :                 goto fail;
    7343             :         }
    7344        1302 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
    7345           0 :                 goto fail;
    7346             :         }
    7347        1302 :         status = cli_shadow_copy_data_recv(req, mem_ctx, pnames, pnum_names);
    7348        1302 :  fail:
    7349        1302 :         TALLOC_FREE(frame);
    7350        1302 :         return status;
    7351             : }
    7352             : 
    7353             : struct cli_fsctl_state {
    7354             :         DATA_BLOB out;
    7355             : };
    7356             : 
    7357             : static void cli_fsctl_smb1_done(struct tevent_req *subreq);
    7358             : static void cli_fsctl_smb2_done(struct tevent_req *subreq);
    7359             : 
    7360          14 : struct tevent_req *cli_fsctl_send(
    7361             :         TALLOC_CTX *mem_ctx,
    7362             :         struct tevent_context *ev,
    7363             :         struct cli_state *cli,
    7364             :         uint16_t fnum,
    7365             :         uint32_t ctl_code,
    7366             :         const DATA_BLOB *in,
    7367             :         uint32_t max_out)
    7368             : {
    7369          14 :         struct tevent_req *req = NULL, *subreq = NULL;
    7370          14 :         struct cli_fsctl_state *state = NULL;
    7371          14 :         uint16_t *setup = NULL;
    7372          14 :         uint8_t *data = NULL;
    7373          14 :         uint32_t num_data = 0;
    7374             : 
    7375          14 :         req = tevent_req_create(mem_ctx, &state, struct cli_fsctl_state);
    7376          14 :         if (req == NULL) {
    7377           0 :                 return NULL;
    7378             :         }
    7379             : 
    7380          14 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    7381          14 :                 subreq = cli_smb2_fsctl_send(
    7382             :                         state, ev, cli, fnum, ctl_code, in, max_out);
    7383          14 :                 if (tevent_req_nomem(subreq, req)) {
    7384           0 :                         return tevent_req_post(req, ev);
    7385             :                 }
    7386          14 :                 tevent_req_set_callback(subreq, cli_fsctl_smb2_done, req);
    7387          14 :                 return req;
    7388             :         }
    7389             : 
    7390           0 :         setup = talloc_array(state, uint16_t, 4);
    7391           0 :         if (tevent_req_nomem(setup, req)) {
    7392           0 :                 return tevent_req_post(req, ev);
    7393             :         }
    7394           0 :         SIVAL(setup, 0, ctl_code);
    7395           0 :         SSVAL(setup, 4, fnum);
    7396           0 :         SCVAL(setup, 6, 1);     /* IsFcntl */
    7397           0 :         SCVAL(setup, 7, 0);     /* IsFlags */
    7398             : 
    7399           0 :         if (in) {
    7400           0 :                 data = in->data;
    7401           0 :                 num_data = in->length;
    7402             :         }
    7403             : 
    7404           0 :         subreq = cli_trans_send(state,
    7405             :                                 ev,
    7406             :                                 cli,
    7407             :                                 0,                 /* additional_flags2 */
    7408             :                                 SMBnttrans,        /* cmd */
    7409             :                                 NULL,              /* name */
    7410             :                                 -1,                /* fid */
    7411             :                                 NT_TRANSACT_IOCTL, /* function */
    7412             :                                 0,                 /* flags */
    7413             :                                 setup,
    7414             :                                 4,
    7415             :                                 0, /* setup */
    7416             :                                 NULL,
    7417             :                                 0,
    7418             :                                 0, /* param */
    7419             :                                 data,
    7420             :                                 num_data,
    7421             :                                 max_out); /* data */
    7422             : 
    7423           0 :         if (tevent_req_nomem(subreq, req)) {
    7424           0 :                 return tevent_req_post(req, ev);
    7425             :         }
    7426           0 :         tevent_req_set_callback(subreq, cli_fsctl_smb1_done, req);
    7427           0 :         return req;
    7428             : }
    7429             : 
    7430          14 : static void cli_fsctl_smb2_done(struct tevent_req *subreq)
    7431             : {
    7432          14 :         struct tevent_req *req = tevent_req_callback_data(
    7433             :                 subreq, struct tevent_req);
    7434          14 :         struct cli_fsctl_state *state = tevent_req_data(
    7435             :                 req, struct cli_fsctl_state);
    7436           0 :         NTSTATUS status;
    7437             : 
    7438          14 :         status = cli_smb2_fsctl_recv(subreq, state, &state->out);
    7439          14 :         tevent_req_simple_finish_ntstatus(subreq, status);
    7440          14 : }
    7441             : 
    7442           0 : static void cli_fsctl_smb1_done(struct tevent_req *subreq)
    7443             : {
    7444           0 :         struct tevent_req *req = tevent_req_callback_data(
    7445             :                 subreq, struct tevent_req);
    7446           0 :         struct cli_fsctl_state *state = tevent_req_data(
    7447             :                 req, struct cli_fsctl_state);
    7448           0 :         uint8_t *out = NULL;
    7449           0 :         uint32_t out_len;
    7450           0 :         NTSTATUS status;
    7451             : 
    7452           0 :         status = cli_trans_recv(
    7453             :                 subreq, state, NULL,
    7454             :                 NULL, 0, NULL,  /* rsetup */
    7455             :                 NULL, 0, NULL,  /* rparam */
    7456             :                 &out, 0, &out_len);
    7457           0 :         TALLOC_FREE(subreq);
    7458           0 :         if (tevent_req_nterror(req, status)) {
    7459           0 :                 return;
    7460             :         }
    7461           0 :         state->out = (DATA_BLOB) {
    7462             :                 .data = out, .length = out_len,
    7463             :         };
    7464           0 :         tevent_req_done(req);
    7465             : }
    7466             : 
    7467          14 : NTSTATUS cli_fsctl_recv(
    7468             :         struct tevent_req *req, TALLOC_CTX *mem_ctx, DATA_BLOB *out)
    7469             : {
    7470          14 :         struct cli_fsctl_state *state = tevent_req_data(
    7471             :                 req, struct cli_fsctl_state);
    7472           0 :         NTSTATUS status;
    7473             : 
    7474          14 :         if (tevent_req_is_nterror(req, &status)) {
    7475          14 :                 return status;
    7476             :         }
    7477             : 
    7478           0 :         if (out != NULL) {
    7479           0 :                 *out = (DATA_BLOB) {
    7480           0 :                         .data = talloc_move(mem_ctx, &state->out.data),
    7481           0 :                         .length = state->out.length,
    7482             :                 };
    7483             :         }
    7484             : 
    7485           0 :         return NT_STATUS_OK;
    7486             : }

Generated by: LCOV version 1.14