LCOV - code coverage report
Current view: top level - source4/torture/smb2 - durable_v2_open.c (source / functions) Hit Total Coverage
Test: coverage report for master 70ed9daf Lines: 1124 1265 88.9 %
Date: 2024-01-11 09:59:51 Functions: 25 25 100.0 %

          Line data    Source code
       1             : /*
       2             :    Unix SMB/CIFS implementation.
       3             : 
       4             :    test suite for SMB2 version two of durable opens
       5             : 
       6             :    Copyright (C) Michael Adam 2012
       7             : 
       8             :    This program is free software; you can redistribute it and/or modify
       9             :    it under the terms of the GNU General Public License as published by
      10             :    the Free Software Foundation; either version 3 of the License, or
      11             :    (at your option) any later version.
      12             : 
      13             :    This program is distributed in the hope that it will be useful,
      14             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      15             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
      16             :    GNU General Public License for more details.
      17             : 
      18             :    You should have received a copy of the GNU General Public License
      19             :    along with this program.  If not, see <http://www.gnu.org/licenses/>.
      20             : */
      21             : 
      22             : #include "includes.h"
      23             : #include "libcli/smb2/smb2.h"
      24             : #include "libcli/smb2/smb2_calls.h"
      25             : #include "../libcli/smb/smbXcli_base.h"
      26             : #include "torture/torture.h"
      27             : #include "torture/smb2/proto.h"
      28             : #include "librpc/ndr/libndr.h"
      29             : 
      30             : #define CHECK_VAL(v, correct) do { \
      31             :         if ((v) != (correct)) { \
      32             :                 torture_result(tctx, TORTURE_FAIL, "(%s): wrong value for %s got 0x%x - should be 0x%x\n", \
      33             :                                 __location__, #v, (int)v, (int)correct); \
      34             :                 ret = false; \
      35             :         }} while (0)
      36             : 
      37             : #define CHECK_STATUS(status, correct) do { \
      38             :         if (!NT_STATUS_EQUAL(status, correct)) { \
      39             :                 torture_result(tctx, TORTURE_FAIL, __location__": Incorrect status %s - should be %s", \
      40             :                        nt_errstr(status), nt_errstr(correct)); \
      41             :                 ret = false; \
      42             :                 goto done; \
      43             :         }} while (0)
      44             : 
      45             : #define CHECK_CREATED(__io, __created, __attribute)                     \
      46             :         do {                                                            \
      47             :                 CHECK_VAL((__io)->out.create_action, NTCREATEX_ACTION_ ## __created); \
      48             :                 CHECK_VAL((__io)->out.size, 0);                              \
      49             :                 CHECK_VAL((__io)->out.file_attr, (__attribute));     \
      50             :                 CHECK_VAL((__io)->out.reserved2, 0);                 \
      51             :         } while(0)
      52             : 
      53             : static struct {
      54             :         int count;
      55             :         struct smb2_close cl;
      56             : } break_info;
      57             : 
      58           4 : static void torture_oplock_close_callback(struct smb2_request *req)
      59             : {
      60           4 :         smb2_close_recv(req, &break_info.cl);
      61           4 : }
      62             : 
      63             : /* A general oplock break notification handler.  This should be used when a
      64             :  * test expects to break from batch or exclusive to a lower level. */
      65           4 : static bool torture_oplock_handler(struct smb2_transport *transport,
      66             :                                    const struct smb2_handle *handle,
      67             :                                    uint8_t level,
      68             :                                    void *private_data)
      69             : {
      70           4 :         struct smb2_tree *tree = private_data;
      71           0 :         struct smb2_request *req;
      72             : 
      73           4 :         break_info.count++;
      74             : 
      75           4 :         ZERO_STRUCT(break_info.cl);
      76           4 :         break_info.cl.in.file.handle = *handle;
      77             : 
      78           4 :         req = smb2_close_send(tree, &break_info.cl);
      79           4 :         req->async.fn = torture_oplock_close_callback;
      80           4 :         req->async.private_data = NULL;
      81           4 :         return true;
      82             : }
      83             : 
      84             : /**
      85             :  * testing various create blob combinations.
      86             :  */
      87           4 : bool test_durable_v2_open_create_blob(struct torture_context *tctx,
      88             :                                       struct smb2_tree *tree)
      89             : {
      90           0 :         NTSTATUS status;
      91           4 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
      92           0 :         char fname[256];
      93           0 :         struct smb2_handle _h;
      94           4 :         struct smb2_handle *h = NULL;
      95           0 :         struct smb2_create io;
      96           4 :         struct GUID create_guid = GUID_random();
      97           4 :         bool ret = true;
      98           0 :         struct smbcli_options options;
      99             : 
     100           4 :         options = tree->session->transport->options;
     101             : 
     102             :         /* Choose a random name in case the state is left a little funky. */
     103           4 :         snprintf(fname, 256, "durable_v2_open_create_blob_%s.dat",
     104             :                  generate_random_str(tctx, 8));
     105             : 
     106           4 :         smb2_util_unlink(tree, fname);
     107             : 
     108           4 :         smb2_oplock_create_share(&io, fname,
     109             :                                  smb2_util_share_access(""),
     110           4 :                                  smb2_util_oplock_level("b"));
     111           4 :         io.in.durable_open = false;
     112           4 :         io.in.durable_open_v2 = true;
     113           4 :         io.in.persistent_open = false;
     114           4 :         io.in.create_guid = create_guid;
     115           4 :         io.in.timeout = UINT32_MAX;
     116             : 
     117           4 :         status = smb2_create(tree, mem_ctx, &io);
     118           4 :         CHECK_STATUS(status, NT_STATUS_OK);
     119           4 :         _h = io.out.file.handle;
     120           4 :         h = &_h;
     121           4 :         CHECK_CREATED(&io, CREATED, FILE_ATTRIBUTE_ARCHIVE);
     122           4 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level("b"));
     123           4 :         CHECK_VAL(io.out.durable_open, false);
     124           4 :         CHECK_VAL(io.out.durable_open_v2, true);
     125           4 :         CHECK_VAL(io.out.persistent_open, false);
     126           4 :         CHECK_VAL(io.out.timeout, 300*1000);
     127             : 
     128             :         /* disconnect */
     129           4 :         TALLOC_FREE(tree);
     130             : 
     131             :         /* create a new session (same client_guid) */
     132           4 :         if (!torture_smb2_connection_ext(tctx, 0, &options, &tree)) {
     133           0 :                 torture_warning(tctx, "couldn't reconnect, bailing\n");
     134           0 :                 ret = false;
     135           0 :                 goto done;
     136             :         }
     137             : 
     138             :         /*
     139             :          * check invalid combinations of durable handle
     140             :          * request and reconnect blobs
     141             :          * See MS-SMB2: 3.3.5.9.12
     142             :          * Handling the SMB2_CREATE_DURABLE_HANDLE_RECONNECT_V2 Create Context
     143             :          */
     144           4 :         ZERO_STRUCT(io);
     145           4 :         io.in.fname = fname;
     146           4 :         io.in.durable_handle_v2 = h; /* durable v2 reconnect request */
     147           4 :         io.in.durable_open = true;   /* durable v1 handle request */
     148           4 :         io.in.create_guid = create_guid;
     149           4 :         status = smb2_create(tree, mem_ctx, &io);
     150           4 :         CHECK_STATUS(status, NT_STATUS_INVALID_PARAMETER);
     151             : 
     152           4 :         ZERO_STRUCT(io);
     153           4 :         io.in.fname = fname;
     154           4 :         io.in.durable_handle = h;     /* durable v1 reconnect request */
     155           4 :         io.in.durable_open_v2 = true; /* durable v2 handle request */
     156           4 :         io.in.create_guid = create_guid;
     157           4 :         status = smb2_create(tree, mem_ctx, &io);
     158           4 :         CHECK_STATUS(status, NT_STATUS_INVALID_PARAMETER);
     159             : 
     160           4 :         ZERO_STRUCT(io);
     161           4 :         io.in.fname = fname;
     162           4 :         io.in.durable_handle = h;    /* durable v1 reconnect request */
     163           4 :         io.in.durable_handle_v2 = h; /* durable v2 reconnect request */
     164           4 :         io.in.create_guid = create_guid;
     165           4 :         status = smb2_create(tree, mem_ctx, &io);
     166           4 :         CHECK_STATUS(status, NT_STATUS_INVALID_PARAMETER);
     167             : 
     168           4 :         ZERO_STRUCT(io);
     169           4 :         io.in.fname = fname;
     170           4 :         io.in.durable_handle_v2 = h;  /* durable v2 reconnect request */
     171           4 :         io.in.durable_open_v2 = true; /* durable v2 handle request */
     172           4 :         io.in.create_guid = create_guid;
     173           4 :         status = smb2_create(tree, mem_ctx, &io);
     174           4 :         CHECK_STATUS(status, NT_STATUS_INVALID_PARAMETER);
     175             : 
     176           4 : done:
     177           4 :         if (h != NULL) {
     178           4 :                 smb2_util_close(tree, *h);
     179             :         }
     180             : 
     181           4 :         smb2_util_unlink(tree, fname);
     182             : 
     183           4 :         talloc_free(tree);
     184             : 
     185           4 :         talloc_free(mem_ctx);
     186             : 
     187           4 :         return ret;
     188             : }
     189             : 
     190             : 
     191             : /**
     192             :  * basic durable_open test.
     193             :  * durable state should only be granted when requested
     194             :  * along with a batch oplock or a handle lease.
     195             :  *
     196             :  * This test tests durable open with all possible oplock types.
     197             :  */
     198             : 
     199             : struct durable_open_vs_oplock {
     200             :         const char *level;
     201             :         const char *share_mode;
     202             :         bool durable;
     203             :         bool persistent;
     204             : };
     205             : 
     206             : #define NUM_OPLOCK_TYPES 4
     207             : #define NUM_SHARE_MODES 8
     208             : #define NUM_OPLOCK_OPEN_TESTS ( NUM_OPLOCK_TYPES * NUM_SHARE_MODES )
     209             : static struct durable_open_vs_oplock durable_open_vs_oplock_table[NUM_OPLOCK_OPEN_TESTS] =
     210             : {
     211             :         { "", "", false, false },
     212             :         { "", "R", false, false },
     213             :         { "", "W", false, false },
     214             :         { "", "D", false, false },
     215             :         { "", "RD", false, false },
     216             :         { "", "RW", false, false },
     217             :         { "", "WD", false, false },
     218             :         { "", "RWD", false, false },
     219             : 
     220             :         { "s", "", false, false },
     221             :         { "s", "R", false, false },
     222             :         { "s", "W", false, false },
     223             :         { "s", "D", false, false },
     224             :         { "s", "RD", false, false },
     225             :         { "s", "RW", false, false },
     226             :         { "s", "WD", false, false },
     227             :         { "s", "RWD", false, false },
     228             : 
     229             :         { "x", "", false, false },
     230             :         { "x", "R", false, false },
     231             :         { "x", "W", false, false },
     232             :         { "x", "D", false, false },
     233             :         { "x", "RD", false, false },
     234             :         { "x", "RW", false, false },
     235             :         { "x", "WD", false, false },
     236             :         { "x", "RWD", false, false },
     237             : 
     238             :         { "b", "", true, false },
     239             :         { "b", "R", true, false },
     240             :         { "b", "W", true, false },
     241             :         { "b", "D", true, false },
     242             :         { "b", "RD", true, false },
     243             :         { "b", "RW", true, false },
     244             :         { "b", "WD", true, false },
     245             :         { "b", "RWD", true, false },
     246             : };
     247             : 
     248         256 : static bool test_one_durable_v2_open_oplock(struct torture_context *tctx,
     249             :                                             struct smb2_tree *tree,
     250             :                                             const char *fname,
     251             :                                             bool request_persistent,
     252             :                                             struct durable_open_vs_oplock test)
     253             : {
     254           0 :         NTSTATUS status;
     255         256 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
     256           0 :         struct smb2_handle _h;
     257         256 :         struct smb2_handle *h = NULL;
     258         256 :         bool ret = true;
     259           0 :         struct smb2_create io;
     260             : 
     261         256 :         smb2_util_unlink(tree, fname);
     262             : 
     263         256 :         smb2_oplock_create_share(&io, fname,
     264             :                                  smb2_util_share_access(test.share_mode),
     265         256 :                                  smb2_util_oplock_level(test.level));
     266         256 :         io.in.durable_open = false;
     267         256 :         io.in.durable_open_v2 = true;
     268         256 :         io.in.persistent_open = request_persistent;
     269         256 :         io.in.create_guid = GUID_random();
     270             : 
     271         256 :         status = smb2_create(tree, mem_ctx, &io);
     272         256 :         CHECK_STATUS(status, NT_STATUS_OK);
     273         256 :         _h = io.out.file.handle;
     274         256 :         h = &_h;
     275         256 :         CHECK_CREATED(&io, CREATED, FILE_ATTRIBUTE_ARCHIVE);
     276         256 :         CHECK_VAL(io.out.durable_open, false);
     277         256 :         CHECK_VAL(io.out.durable_open_v2, test.durable);
     278         256 :         CHECK_VAL(io.out.persistent_open, test.persistent);
     279         256 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level(test.level));
     280             : 
     281         256 : done:
     282         256 :         if (h != NULL) {
     283         256 :                 smb2_util_close(tree, *h);
     284             :         }
     285         256 :         smb2_util_unlink(tree, fname);
     286         256 :         talloc_free(mem_ctx);
     287             : 
     288         256 :         return ret;
     289             : }
     290             : 
     291           8 : static bool test_durable_v2_open_oplock_table(struct torture_context *tctx,
     292             :                                               struct smb2_tree *tree,
     293             :                                               const char *fname,
     294             :                                               bool request_persistent,
     295             :                                               struct durable_open_vs_oplock *table,
     296             :                                               uint8_t num_tests)
     297             : {
     298           8 :         bool ret = true;
     299           0 :         uint8_t i;
     300             : 
     301           8 :         smb2_util_unlink(tree, fname);
     302             : 
     303         264 :         for (i = 0; i < num_tests; i++) {
     304         256 :                 ret = test_one_durable_v2_open_oplock(tctx,
     305             :                                                       tree,
     306             :                                                       fname,
     307             :                                                       request_persistent,
     308         256 :                                                       table[i]);
     309         256 :                 if (ret == false) {
     310           0 :                         goto done;
     311             :                 }
     312             :         }
     313             : 
     314           8 : done:
     315           8 :         smb2_util_unlink(tree, fname);
     316             : 
     317           8 :         return ret;
     318             : }
     319             : 
     320           4 : bool test_durable_v2_open_oplock(struct torture_context *tctx,
     321             :                                  struct smb2_tree *tree)
     322             : {
     323           0 :         bool ret;
     324           0 :         char fname[256];
     325             : 
     326             :         /* Choose a random name in case the state is left a little funky. */
     327           4 :         snprintf(fname, 256, "durable_open_oplock_%s.dat",
     328             :                  generate_random_str(tctx, 8));
     329             : 
     330           4 :         ret = test_durable_v2_open_oplock_table(tctx, tree, fname,
     331             :                                                 false, /* request_persistent */
     332             :                                                 durable_open_vs_oplock_table,
     333             :                                                 NUM_OPLOCK_OPEN_TESTS);
     334             : 
     335           4 :         talloc_free(tree);
     336             : 
     337           4 :         return ret;
     338             : }
     339             : 
     340             : /**
     341             :  * basic durable handle open test.
     342             :  * persistent state should only be granted when requested
     343             :  * along with a batch oplock or a handle lease.
     344             :  *
     345             :  * This test tests persistent open with all valid lease types.
     346             :  */
     347             : 
     348             : struct durable_open_vs_lease {
     349             :         const char *type;
     350             :         const char *share_mode;
     351             :         bool durable;
     352             :         bool persistent;
     353             : };
     354             : 
     355             : #define NUM_LEASE_TYPES 5
     356             : #define NUM_LEASE_OPEN_TESTS ( NUM_LEASE_TYPES * NUM_SHARE_MODES )
     357             : static struct durable_open_vs_lease durable_open_vs_lease_table[NUM_LEASE_OPEN_TESTS] =
     358             : {
     359             :         { "", "", false, false },
     360             :         { "", "R", false, false },
     361             :         { "", "W", false, false },
     362             :         { "", "D", false, false },
     363             :         { "", "RW", false, false },
     364             :         { "", "RD", false, false },
     365             :         { "", "WD", false, false },
     366             :         { "", "RWD", false, false },
     367             : 
     368             :         { "R", "", false, false },
     369             :         { "R", "R", false, false },
     370             :         { "R", "W", false, false },
     371             :         { "R", "D", false, false },
     372             :         { "R", "RW", false, false },
     373             :         { "R", "RD", false, false },
     374             :         { "R", "DW", false, false },
     375             :         { "R", "RWD", false, false },
     376             : 
     377             :         { "RW", "", false, false },
     378             :         { "RW", "R", false, false },
     379             :         { "RW", "W", false, false },
     380             :         { "RW", "D", false, false },
     381             :         { "RW", "RW", false, false },
     382             :         { "RW", "RD", false, false },
     383             :         { "RW", "WD", false, false },
     384             :         { "RW", "RWD", false, false },
     385             : 
     386             :         { "RH", "", true, false },
     387             :         { "RH", "R", true, false },
     388             :         { "RH", "W", true, false },
     389             :         { "RH", "D", true, false },
     390             :         { "RH", "RW", true, false },
     391             :         { "RH", "RD", true, false },
     392             :         { "RH", "WD", true, false },
     393             :         { "RH", "RWD", true, false },
     394             : 
     395             :         { "RHW", "", true, false },
     396             :         { "RHW", "R", true, false },
     397             :         { "RHW", "W", true, false },
     398             :         { "RHW", "D", true, false },
     399             :         { "RHW", "RW", true, false },
     400             :         { "RHW", "RD", true, false },
     401             :         { "RHW", "WD", true, false },
     402             :         { "RHW", "RWD", true, false },
     403             : };
     404             : 
     405         160 : static bool test_one_durable_v2_open_lease(struct torture_context *tctx,
     406             :                                            struct smb2_tree *tree,
     407             :                                            const char *fname,
     408             :                                            bool request_persistent,
     409             :                                            struct durable_open_vs_lease test)
     410             : {
     411           0 :         NTSTATUS status;
     412         160 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
     413           0 :         struct smb2_handle _h;
     414         160 :         struct smb2_handle *h = NULL;
     415         160 :         bool ret = true;
     416           0 :         struct smb2_create io;
     417           0 :         struct smb2_lease ls;
     418           0 :         uint64_t lease;
     419             : 
     420         160 :         smb2_util_unlink(tree, fname);
     421             : 
     422         160 :         lease = random();
     423             : 
     424         160 :         smb2_lease_create_share(&io, &ls, false /* dir */, fname,
     425             :                                 smb2_util_share_access(test.share_mode),
     426             :                                 lease,
     427             :                                 smb2_util_lease_state(test.type));
     428         160 :         io.in.durable_open = false;
     429         160 :         io.in.durable_open_v2 = true;
     430         160 :         io.in.persistent_open = request_persistent;
     431         160 :         io.in.create_guid = GUID_random();
     432             : 
     433         160 :         status = smb2_create(tree, mem_ctx, &io);
     434         160 :         CHECK_STATUS(status, NT_STATUS_OK);
     435         160 :         _h = io.out.file.handle;
     436         160 :         h = &_h;
     437         160 :         CHECK_CREATED(&io, CREATED, FILE_ATTRIBUTE_ARCHIVE);
     438         160 :         CHECK_VAL(io.out.durable_open, false);
     439         160 :         CHECK_VAL(io.out.durable_open_v2, test.durable);
     440         160 :         CHECK_VAL(io.out.persistent_open, test.persistent);
     441         160 :         CHECK_VAL(io.out.oplock_level, SMB2_OPLOCK_LEVEL_LEASE);
     442         160 :         CHECK_VAL(io.out.lease_response.lease_key.data[0], lease);
     443         160 :         CHECK_VAL(io.out.lease_response.lease_key.data[1], ~lease);
     444         160 :         CHECK_VAL(io.out.lease_response.lease_state,
     445             :                   smb2_util_lease_state(test.type));
     446         160 : done:
     447         160 :         if (h != NULL) {
     448         160 :                 smb2_util_close(tree, *h);
     449             :         }
     450         160 :         smb2_util_unlink(tree, fname);
     451         160 :         talloc_free(mem_ctx);
     452             : 
     453         160 :         return ret;
     454             : }
     455             : 
     456           4 : static bool test_durable_v2_open_lease_table(struct torture_context *tctx,
     457             :                                              struct smb2_tree *tree,
     458             :                                              const char *fname,
     459             :                                              bool request_persistent,
     460             :                                              struct durable_open_vs_lease *table,
     461             :                                              uint8_t num_tests)
     462             : {
     463           4 :         bool ret = true;
     464           0 :         uint8_t i;
     465             : 
     466           4 :         smb2_util_unlink(tree, fname);
     467             : 
     468         164 :         for (i = 0; i < num_tests; i++) {
     469         160 :                 ret = test_one_durable_v2_open_lease(tctx,
     470             :                                                      tree,
     471             :                                                      fname,
     472             :                                                      request_persistent,
     473         160 :                                                      table[i]);
     474         160 :                 if (ret == false) {
     475           0 :                         goto done;
     476             :                 }
     477             :         }
     478             : 
     479           4 : done:
     480           4 :         smb2_util_unlink(tree, fname);
     481             : 
     482           4 :         return ret;
     483             : }
     484             : 
     485           4 : bool test_durable_v2_open_lease(struct torture_context *tctx,
     486             :                                 struct smb2_tree *tree)
     487             : {
     488           0 :         char fname[256];
     489           4 :         bool ret = true;
     490           0 :         uint32_t caps;
     491             : 
     492           4 :         caps = smb2cli_conn_server_capabilities(tree->session->transport->conn);
     493           4 :         if (!(caps & SMB2_CAP_LEASING)) {
     494           2 :                 torture_skip(tctx, "leases are not supported");
     495             :         }
     496             : 
     497             :         /* Choose a random name in case the state is left a little funky. */
     498           2 :         snprintf(fname, 256, "durable_open_lease_%s.dat", generate_random_str(tctx, 8));
     499             : 
     500           2 :         ret = test_durable_v2_open_lease_table(tctx, tree, fname,
     501             :                                                false, /* request_persistent */
     502             :                                                durable_open_vs_lease_table,
     503             :                                                NUM_LEASE_OPEN_TESTS);
     504             : 
     505           2 :         talloc_free(tree);
     506           2 :         return ret;
     507             : }
     508             : 
     509             : /**
     510             :  * basic test for doing a durable open
     511             :  * and do a durable reopen on the same connection
     512             :  * while the first open is still active (fails)
     513             :  */
     514           4 : bool test_durable_v2_open_reopen1(struct torture_context *tctx,
     515             :                                   struct smb2_tree *tree)
     516             : {
     517           0 :         NTSTATUS status;
     518           4 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
     519           0 :         char fname[256];
     520           0 :         struct smb2_handle _h;
     521           4 :         struct smb2_handle *h = NULL;
     522           0 :         struct smb2_create io;
     523           4 :         struct GUID create_guid = GUID_random();
     524           4 :         bool ret = true;
     525             : 
     526             :         /* Choose a random name in case the state is left a little funky. */
     527           4 :         snprintf(fname, 256, "durable_v2_open_reopen1_%s.dat",
     528             :                  generate_random_str(tctx, 8));
     529             : 
     530           4 :         smb2_util_unlink(tree, fname);
     531             : 
     532           4 :         smb2_oplock_create_share(&io, fname,
     533             :                                  smb2_util_share_access(""),
     534           4 :                                  smb2_util_oplock_level("b"));
     535           4 :         io.in.durable_open = false;
     536           4 :         io.in.durable_open_v2 = true;
     537           4 :         io.in.persistent_open = false;
     538           4 :         io.in.create_guid = create_guid;
     539           4 :         io.in.timeout = UINT32_MAX;
     540             : 
     541           4 :         status = smb2_create(tree, mem_ctx, &io);
     542           4 :         CHECK_STATUS(status, NT_STATUS_OK);
     543           4 :         _h = io.out.file.handle;
     544           4 :         h = &_h;
     545           4 :         CHECK_CREATED(&io, CREATED, FILE_ATTRIBUTE_ARCHIVE);
     546           4 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level("b"));
     547           4 :         CHECK_VAL(io.out.durable_open, false);
     548           4 :         CHECK_VAL(io.out.durable_open_v2, true);
     549           4 :         CHECK_VAL(io.out.persistent_open, false);
     550           4 :         CHECK_VAL(io.out.timeout, 300*1000);
     551             : 
     552             :         /* try a durable reconnect while the file is still open */
     553           4 :         ZERO_STRUCT(io);
     554           4 :         io.in.fname = "";
     555           4 :         io.in.durable_handle_v2 = h;
     556           4 :         io.in.create_guid = create_guid;
     557           4 :         status = smb2_create(tree, mem_ctx, &io);
     558           4 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
     559             : 
     560           4 : done:
     561           4 :         if (h != NULL) {
     562           4 :                 smb2_util_close(tree, *h);
     563             :         }
     564             : 
     565           4 :         smb2_util_unlink(tree, fname);
     566             : 
     567           4 :         talloc_free(tree);
     568             : 
     569           4 :         talloc_free(mem_ctx);
     570             : 
     571           4 :         return ret;
     572             : }
     573             : 
     574             : /**
     575             :  * Basic test for doing a durable open
     576             :  * and do a session reconnect while the first
     577             :  * session is still active and the handle is
     578             :  * still open in the client.
     579             :  * This closes the original session and  a
     580             :  * durable reconnect on the new session succeeds.
     581             :  */
     582           4 : bool test_durable_v2_open_reopen1a(struct torture_context *tctx,
     583             :                                    struct smb2_tree *tree)
     584             : {
     585           0 :         NTSTATUS status;
     586           4 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
     587           0 :         char fname[256];
     588           0 :         struct smb2_handle _h;
     589           4 :         struct smb2_handle *h = NULL;
     590           0 :         struct smb2_create io;
     591           4 :         struct GUID create_guid = GUID_random();
     592           4 :         bool ret = true;
     593           4 :         struct smb2_tree *tree2 = NULL;
     594           4 :         struct smb2_tree *tree3 = NULL;
     595           0 :         uint64_t previous_session_id;
     596           0 :         struct smbcli_options options;
     597           0 :         struct GUID orig_client_guid;
     598             : 
     599           4 :         options = tree->session->transport->options;
     600           4 :         orig_client_guid = options.client_guid;
     601             : 
     602             :         /* Choose a random name in case the state is left a little funky. */
     603           4 :         snprintf(fname, 256, "durable_v2_open_reopen1a_%s.dat",
     604             :                  generate_random_str(tctx, 8));
     605             : 
     606           4 :         smb2_util_unlink(tree, fname);
     607             : 
     608           4 :         smb2_oplock_create_share(&io, fname,
     609             :                                  smb2_util_share_access(""),
     610           4 :                                  smb2_util_oplock_level("b"));
     611           4 :         io.in.durable_open = false;
     612           4 :         io.in.durable_open_v2 = true;
     613           4 :         io.in.persistent_open = false;
     614           4 :         io.in.create_guid = create_guid;
     615           4 :         io.in.timeout = UINT32_MAX;
     616             : 
     617           4 :         status = smb2_create(tree, mem_ctx, &io);
     618           4 :         CHECK_STATUS(status, NT_STATUS_OK);
     619           4 :         _h = io.out.file.handle;
     620           4 :         h = &_h;
     621           4 :         CHECK_CREATED(&io, CREATED, FILE_ATTRIBUTE_ARCHIVE);
     622           4 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level("b"));
     623           4 :         CHECK_VAL(io.out.durable_open, false);
     624           4 :         CHECK_VAL(io.out.durable_open_v2, true);
     625           4 :         CHECK_VAL(io.out.persistent_open, false);
     626           4 :         CHECK_VAL(io.out.timeout, 300*1000);
     627             : 
     628             :         /*
     629             :          * a session reconnect on a second tcp connection
     630             :          */
     631             : 
     632           4 :         previous_session_id = smb2cli_session_current_id(tree->session->smbXcli);
     633             : 
     634             :         /* for oplocks, the client guid can be different: */
     635           4 :         options.client_guid = GUID_random();
     636             : 
     637           4 :         ret = torture_smb2_connection_ext(tctx, previous_session_id,
     638             :                                           &options, &tree2);
     639           4 :         torture_assert_goto(tctx, ret, ret, done, "couldn't reconnect");
     640             : 
     641             :         /*
     642             :          * check that this has deleted the old session
     643             :          */
     644             : 
     645           4 :         ZERO_STRUCT(io);
     646           4 :         io.in.fname = "";
     647           4 :         io.in.durable_handle_v2 = h;
     648           4 :         io.in.create_guid = create_guid;
     649           4 :         status = smb2_create(tree, mem_ctx, &io);
     650           4 :         CHECK_STATUS(status, NT_STATUS_USER_SESSION_DELETED);
     651             : 
     652           4 :         TALLOC_FREE(tree);
     653             : 
     654             :         /*
     655             :          * but a durable reconnect on the new session succeeds:
     656             :          */
     657             : 
     658           4 :         ZERO_STRUCT(io);
     659           4 :         io.in.fname = "";
     660           4 :         io.in.durable_handle_v2 = h;
     661           4 :         io.in.create_guid = create_guid;
     662           4 :         status = smb2_create(tree2, mem_ctx, &io);
     663           4 :         CHECK_STATUS(status, NT_STATUS_OK);
     664           4 :         CHECK_CREATED(&io, EXISTED, FILE_ATTRIBUTE_ARCHIVE);
     665           4 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level("b"));
     666           4 :         CHECK_VAL(io.out.durable_open, false);
     667           4 :         CHECK_VAL(io.out.durable_open_v2, false); /* no dh2q response blob */
     668           4 :         CHECK_VAL(io.out.persistent_open, false);
     669           4 :         CHECK_VAL(io.out.timeout, 0);
     670           4 :         _h = io.out.file.handle;
     671           4 :         h = &_h;
     672             : 
     673             :         /*
     674             :          * a session reconnect on a second tcp connection
     675             :          */
     676             : 
     677           4 :         previous_session_id = smb2cli_session_current_id(tree2->session->smbXcli);
     678             : 
     679             :         /* it works the same with the original guid */
     680           4 :         options.client_guid = orig_client_guid;
     681             : 
     682           4 :         ret = torture_smb2_connection_ext(tctx, previous_session_id,
     683             :                                           &options, &tree3);
     684           4 :         torture_assert_goto(tctx, ret, ret, done, "couldn't reconnect");
     685             : 
     686             :         /*
     687             :          * check that this has deleted the old session
     688             :          */
     689             : 
     690           4 :         ZERO_STRUCT(io);
     691           4 :         io.in.fname = "";
     692           4 :         io.in.durable_handle_v2 = h;
     693           4 :         io.in.create_guid = create_guid;
     694           4 :         status = smb2_create(tree2, mem_ctx, &io);
     695           4 :         CHECK_STATUS(status, NT_STATUS_USER_SESSION_DELETED);
     696           4 :         TALLOC_FREE(tree2);
     697             : 
     698             :         /*
     699             :          * but a durable reconnect on the new session succeeds:
     700             :          */
     701             : 
     702           4 :         ZERO_STRUCT(io);
     703           4 :         io.in.fname = "";
     704           4 :         io.in.durable_handle_v2 = h;
     705           4 :         io.in.create_guid = create_guid;
     706           4 :         status = smb2_create(tree3, mem_ctx, &io);
     707           4 :         CHECK_STATUS(status, NT_STATUS_OK);
     708           4 :         CHECK_CREATED(&io, EXISTED, FILE_ATTRIBUTE_ARCHIVE);
     709           4 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level("b"));
     710           4 :         CHECK_VAL(io.out.durable_open, false);
     711           4 :         CHECK_VAL(io.out.durable_open_v2, false); /* no dh2q response blob */
     712           4 :         CHECK_VAL(io.out.persistent_open, false);
     713           4 :         CHECK_VAL(io.out.timeout, 0);
     714           4 :         _h = io.out.file.handle;
     715           4 :         h = &_h;
     716             : 
     717           4 : done:
     718           4 :         if (tree == NULL) {
     719           4 :                 tree = tree2;
     720             :         }
     721             : 
     722           4 :         if (tree == NULL) {
     723           4 :                 tree = tree3;
     724             :         }
     725             : 
     726           4 :         if (tree != NULL) {
     727           4 :                 if (h != NULL) {
     728           4 :                         smb2_util_close(tree, *h);
     729             :                 }
     730             : 
     731           4 :                 smb2_util_unlink(tree, fname);
     732             : 
     733           4 :                 talloc_free(tree);
     734             :         }
     735             : 
     736           4 :         talloc_free(mem_ctx);
     737             : 
     738           4 :         return ret;
     739             : }
     740             : 
     741             : /**
     742             :  * lease variant of reopen1a
     743             :  *
     744             :  * Basic test for doing a durable open and doing a session
     745             :  * reconnect while the first session is still active and the
     746             :  * handle is still open in the client.
     747             :  * This closes the original session and  a durable reconnect on
     748             :  * the new session succeeds depending on the client guid:
     749             :  *
     750             :  * Durable reconnect on a session with a different client guid fails.
     751             :  * Durable reconnect on a session with the original client guid succeeds.
     752             :  */
     753           4 : bool test_durable_v2_open_reopen1a_lease(struct torture_context *tctx,
     754             :                                          struct smb2_tree *tree)
     755             : {
     756           0 :         NTSTATUS status;
     757           4 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
     758           0 :         char fname[256];
     759           0 :         struct smb2_handle _h;
     760           4 :         struct smb2_handle *h = NULL;
     761           0 :         struct smb2_create io;
     762           4 :         struct GUID create_guid = GUID_random();
     763           0 :         struct smb2_lease ls;
     764           0 :         uint64_t lease_key;
     765           4 :         bool ret = true;
     766           4 :         struct smb2_tree *tree2 = NULL;
     767           4 :         struct smb2_tree *tree3 = NULL;
     768           0 :         uint64_t previous_session_id;
     769           0 :         struct smbcli_options options;
     770           0 :         struct GUID orig_client_guid;
     771             : 
     772           4 :         options = tree->session->transport->options;
     773           4 :         orig_client_guid = options.client_guid;
     774             : 
     775             :         /* Choose a random name in case the state is left a little funky. */
     776           4 :         snprintf(fname, 256, "durable_v2_open_reopen1a_lease_%s.dat",
     777             :                  generate_random_str(tctx, 8));
     778             : 
     779           4 :         smb2_util_unlink(tree, fname);
     780             : 
     781           4 :         lease_key = random();
     782           4 :         smb2_lease_create(&io, &ls, false /* dir */, fname,
     783             :                           lease_key, smb2_util_lease_state("RWH"));
     784           4 :         io.in.durable_open = false;
     785           4 :         io.in.durable_open_v2 = true;
     786           4 :         io.in.persistent_open = false;
     787           4 :         io.in.create_guid = create_guid;
     788           4 :         io.in.timeout = UINT32_MAX;
     789             : 
     790           4 :         status = smb2_create(tree, mem_ctx, &io);
     791           4 :         CHECK_STATUS(status, NT_STATUS_OK);
     792           4 :         _h = io.out.file.handle;
     793           4 :         h = &_h;
     794           4 :         CHECK_CREATED(&io, CREATED, FILE_ATTRIBUTE_ARCHIVE);
     795           4 :         CHECK_VAL(io.out.durable_open, false);
     796           4 :         CHECK_VAL(io.out.durable_open_v2, true);
     797           4 :         CHECK_VAL(io.out.persistent_open, false);
     798           4 :         CHECK_VAL(io.out.timeout, 300*1000);
     799           4 :         CHECK_VAL(io.out.oplock_level, SMB2_OPLOCK_LEVEL_LEASE);
     800           4 :         CHECK_VAL(io.out.lease_response.lease_key.data[0], lease_key);
     801           4 :         CHECK_VAL(io.out.lease_response.lease_key.data[1], ~lease_key);
     802           4 :         CHECK_VAL(io.out.lease_response.lease_state,
     803             :                   smb2_util_lease_state("RWH"));
     804           4 :         CHECK_VAL(io.out.lease_response.lease_flags, 0);
     805           4 :         CHECK_VAL(io.out.lease_response.lease_duration, 0);
     806             : 
     807           4 :         previous_session_id = smb2cli_session_current_id(tree->session->smbXcli);
     808             : 
     809             :         /*
     810             :          * a session reconnect on a second tcp connection
     811             :          * with a different client_guid does not allow
     812             :          * the durable reconnect.
     813             :          */
     814             : 
     815           4 :         options.client_guid = GUID_random();
     816             : 
     817           4 :         ret = torture_smb2_connection_ext(tctx, previous_session_id,
     818             :                                           &options, &tree2);
     819           4 :         torture_assert_goto(tctx, ret, ret, done, "couldn't reconnect");
     820             : 
     821             :         /*
     822             :          * check that this has deleted the old session
     823             :          */
     824             : 
     825           4 :         ZERO_STRUCT(io);
     826           4 :         io.in.fname = fname;
     827           4 :         io.in.durable_handle_v2 = h;
     828           4 :         io.in.create_guid = create_guid;
     829           4 :         io.in.lease_request = &ls;
     830           4 :         status = smb2_create(tree, mem_ctx, &io);
     831           4 :         CHECK_STATUS(status, NT_STATUS_USER_SESSION_DELETED);
     832           4 :         TALLOC_FREE(tree);
     833             : 
     834             :         /*
     835             :          * but a durable reconnect on the new session with the wrong
     836             :          * client guid fails
     837             :          */
     838             : 
     839           4 :         ZERO_STRUCT(io);
     840           4 :         io.in.fname = fname;
     841           4 :         io.in.durable_handle_v2 = h;
     842           4 :         io.in.create_guid = create_guid;
     843           4 :         io.in.lease_request = &ls;
     844           4 :         status = smb2_create(tree2, mem_ctx, &io);
     845           4 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
     846             : 
     847             : 
     848             :         /*
     849             :          * now a session reconnect on a second tcp connection
     850             :          * with original client_guid allows the durable reconnect.
     851             :          */
     852             : 
     853           4 :         options.client_guid = orig_client_guid;
     854             :         //options.client_guid = GUID_random();
     855             : 
     856           4 :         ret = torture_smb2_connection_ext(tctx, previous_session_id,
     857             :                                           &options, &tree3);
     858           4 :         torture_assert_goto(tctx, ret, ret, done, "couldn't reconnect");
     859             : 
     860             :         /*
     861             :          * check that this has deleted the old session
     862             :          */
     863             : 
     864           4 :         ZERO_STRUCT(io);
     865           4 :         io.in.fname = fname;
     866           4 :         io.in.durable_handle_v2 = h;
     867           4 :         io.in.create_guid = create_guid;
     868           4 :         io.in.lease_request = &ls;
     869           4 :         status = smb2_create(tree2, mem_ctx, &io);
     870           4 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
     871           4 :         TALLOC_FREE(tree2);
     872             : 
     873             :         /*
     874             :          * but a durable reconnect on the new session succeeds:
     875             :          */
     876             : 
     877           4 :         ZERO_STRUCT(io);
     878           4 :         io.in.fname = fname;
     879           4 :         io.in.durable_handle_v2 = h;
     880           4 :         io.in.create_guid = create_guid;
     881           4 :         io.in.lease_request = &ls;
     882           4 :         status = smb2_create(tree3, mem_ctx, &io);
     883           4 :         CHECK_STATUS(status, NT_STATUS_OK);
     884           2 :         CHECK_CREATED(&io, EXISTED, FILE_ATTRIBUTE_ARCHIVE);
     885           2 :         CHECK_VAL(io.out.durable_open, false);
     886           2 :         CHECK_VAL(io.out.durable_open_v2, false); /* no dh2q response blob */
     887           2 :         CHECK_VAL(io.out.persistent_open, false);
     888           2 :         CHECK_VAL(io.out.timeout, 0);
     889           2 :         CHECK_VAL(io.out.oplock_level, SMB2_OPLOCK_LEVEL_LEASE);
     890           2 :         CHECK_VAL(io.out.lease_response.lease_key.data[0], lease_key);
     891           2 :         CHECK_VAL(io.out.lease_response.lease_key.data[1], ~lease_key);
     892           2 :         CHECK_VAL(io.out.lease_response.lease_state,
     893             :                   smb2_util_lease_state("RWH"));
     894           2 :         CHECK_VAL(io.out.lease_response.lease_flags, 0);
     895           2 :         CHECK_VAL(io.out.lease_response.lease_duration, 0);
     896           2 :         _h = io.out.file.handle;
     897           2 :         h = &_h;
     898             : 
     899           4 : done:
     900           4 :         if (tree == NULL) {
     901           4 :                 tree = tree2;
     902             :         }
     903             : 
     904           4 :         if (tree == NULL) {
     905           4 :                 tree = tree3;
     906             :         }
     907             : 
     908           4 :         if (tree != NULL) {
     909           4 :                 if (h != NULL) {
     910           4 :                         smb2_util_close(tree, *h);
     911             :                 }
     912             : 
     913           4 :                 smb2_util_unlink(tree, fname);
     914             : 
     915           4 :                 talloc_free(tree);
     916             :         }
     917             : 
     918           4 :         talloc_free(mem_ctx);
     919             : 
     920           4 :         return ret;
     921             : }
     922             : 
     923             : /**
     924             :  * basic test for doing a durable open
     925             :  * tcp disconnect, reconnect, do a durable reopen (succeeds)
     926             :  */
     927           4 : bool test_durable_v2_open_reopen2(struct torture_context *tctx,
     928             :                                   struct smb2_tree *tree)
     929             : {
     930           0 :         NTSTATUS status;
     931           4 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
     932           0 :         char fname[256];
     933           0 :         struct smb2_handle _h;
     934           4 :         struct smb2_handle *h = NULL;
     935           0 :         struct smb2_create io;
     936           4 :         struct GUID create_guid = GUID_random();
     937           4 :         struct GUID create_guid_invalid = GUID_random();
     938           4 :         bool ret = true;
     939             : 
     940             :         /* Choose a random name in case the state is left a little funky. */
     941           4 :         snprintf(fname, 256, "durable_v2_open_reopen2_%s.dat",
     942             :                  generate_random_str(tctx, 8));
     943             : 
     944           4 :         smb2_util_unlink(tree, fname);
     945             : 
     946           4 :         smb2_oplock_create_share(&io, fname,
     947             :                                  smb2_util_share_access(""),
     948           4 :                                  smb2_util_oplock_level("b"));
     949           4 :         io.in.durable_open = false;
     950           4 :         io.in.durable_open_v2 = true;
     951           4 :         io.in.persistent_open = false;
     952           4 :         io.in.create_guid = create_guid;
     953           4 :         io.in.timeout = UINT32_MAX;
     954             : 
     955           4 :         status = smb2_create(tree, mem_ctx, &io);
     956           4 :         CHECK_STATUS(status, NT_STATUS_OK);
     957           4 :         _h = io.out.file.handle;
     958           4 :         h = &_h;
     959           4 :         CHECK_CREATED(&io, CREATED, FILE_ATTRIBUTE_ARCHIVE);
     960           4 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level("b"));
     961           4 :         CHECK_VAL(io.out.durable_open, false);
     962           4 :         CHECK_VAL(io.out.durable_open_v2, true);
     963           4 :         CHECK_VAL(io.out.persistent_open, false);
     964           4 :         CHECK_VAL(io.out.timeout, 300*1000);
     965             : 
     966             :         /* disconnect, leaving the durable open */
     967           4 :         TALLOC_FREE(tree);
     968             : 
     969           4 :         if (!torture_smb2_connection(tctx, &tree)) {
     970           0 :                 torture_warning(tctx, "couldn't reconnect, bailing\n");
     971           0 :                 ret = false;
     972           0 :                 goto done;
     973             :         }
     974             : 
     975             :         /*
     976             :          * first a few failure cases
     977             :          */
     978             : 
     979           4 :         ZERO_STRUCT(io);
     980           4 :         io.in.fname = "";
     981           4 :         io.in.durable_handle_v2 = h;
     982           4 :         status = smb2_create(tree, mem_ctx, &io);
     983           4 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
     984             : 
     985           4 :         ZERO_STRUCT(io);
     986           4 :         io.in.fname = "__non_existing_fname__";
     987           4 :         io.in.durable_handle_v2 = h;
     988           4 :         status = smb2_create(tree, mem_ctx, &io);
     989           4 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
     990             : 
     991           4 :         ZERO_STRUCT(io);
     992           4 :         io.in.fname = fname;
     993           4 :         io.in.durable_handle_v2 = h;
     994           4 :         status = smb2_create(tree, mem_ctx, &io);
     995           4 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
     996             : 
     997             :         /* a non-zero but non-matching create_guid does not change it: */
     998           4 :         ZERO_STRUCT(io);
     999           4 :         io.in.fname = fname;
    1000           4 :         io.in.durable_handle_v2 = h;
    1001           4 :         io.in.create_guid = create_guid_invalid;
    1002           4 :         status = smb2_create(tree, mem_ctx, &io);
    1003           4 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
    1004             : 
    1005             :         /*
    1006             :          * now success:
    1007             :          * The important difference is that the create_guid is provided.
    1008             :          */
    1009           4 :         ZERO_STRUCT(io);
    1010           4 :         io.in.fname = fname;
    1011           4 :         io.in.durable_open_v2 = false;
    1012           4 :         io.in.durable_handle_v2 = h;
    1013           4 :         io.in.create_guid = create_guid;
    1014           4 :         h = NULL;
    1015             : 
    1016           4 :         status = smb2_create(tree, mem_ctx, &io);
    1017           4 :         CHECK_STATUS(status, NT_STATUS_OK);
    1018           4 :         CHECK_CREATED(&io, EXISTED, FILE_ATTRIBUTE_ARCHIVE);
    1019           4 :         CHECK_VAL(io.out.durable_open, false);
    1020           4 :         CHECK_VAL(io.out.durable_open_v2, false); /* no dh2q response blob */
    1021           4 :         CHECK_VAL(io.out.persistent_open, false);
    1022           4 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level("b"));
    1023           4 :         _h = io.out.file.handle;
    1024           4 :         h = &_h;
    1025             : 
    1026             :         /* disconnect one more time */
    1027           4 :         TALLOC_FREE(tree);
    1028             : 
    1029           4 :         if (!torture_smb2_connection(tctx, &tree)) {
    1030           0 :                 torture_warning(tctx, "couldn't reconnect, bailing\n");
    1031           0 :                 ret = false;
    1032           0 :                 goto done;
    1033             :         }
    1034             : 
    1035           4 :         ZERO_STRUCT(io);
    1036             :         /* These are completely ignored by the server */
    1037           4 :         io.in.security_flags = 0x78;
    1038           4 :         io.in.oplock_level = 0x78;
    1039           4 :         io.in.impersonation_level = 0x12345678;
    1040           4 :         io.in.create_flags = 0x12345678;
    1041           4 :         io.in.reserved = 0x12345678;
    1042           4 :         io.in.desired_access = 0x12345678;
    1043           4 :         io.in.file_attributes = 0x12345678;
    1044           4 :         io.in.share_access = 0x12345678;
    1045           4 :         io.in.create_disposition = 0x12345678;
    1046           4 :         io.in.create_options = 0x12345678;
    1047           4 :         io.in.fname = "__non_existing_fname__";
    1048             : 
    1049             :         /*
    1050             :          * only io.in.durable_handle_v2 and
    1051             :          * io.in.create_guid are checked
    1052             :          */
    1053           4 :         io.in.durable_open_v2 = false;
    1054           4 :         io.in.durable_handle_v2 = h;
    1055           4 :         io.in.create_guid = create_guid;
    1056           4 :         h = NULL;
    1057             : 
    1058           4 :         status = smb2_create(tree, mem_ctx, &io);
    1059           4 :         CHECK_STATUS(status, NT_STATUS_OK);
    1060           4 :         CHECK_CREATED(&io, EXISTED, FILE_ATTRIBUTE_ARCHIVE);
    1061           4 :         CHECK_VAL(io.out.durable_open, false);
    1062           4 :         CHECK_VAL(io.out.durable_open_v2, false); /* no dh2q response blob */
    1063           4 :         CHECK_VAL(io.out.persistent_open, false);
    1064           4 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level("b"));
    1065           4 :         _h = io.out.file.handle;
    1066           4 :         h = &_h;
    1067             : 
    1068           4 : done:
    1069           4 :         if (h != NULL) {
    1070           4 :                 smb2_util_close(tree, *h);
    1071             :         }
    1072             : 
    1073           4 :         smb2_util_unlink(tree, fname);
    1074             : 
    1075           4 :         talloc_free(tree);
    1076             : 
    1077           4 :         talloc_free(mem_ctx);
    1078             : 
    1079           4 :         return ret;
    1080             : }
    1081             : 
    1082             : /**
    1083             :  * durable reconnect test:
    1084             :  * connect with v2, reconnect with v1
    1085             :  */
    1086           4 : bool test_durable_v2_open_reopen2b(struct torture_context *tctx,
    1087             :                                    struct smb2_tree *tree)
    1088             : {
    1089           0 :         NTSTATUS status;
    1090           4 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
    1091           0 :         char fname[256];
    1092           0 :         struct smb2_handle _h;
    1093           4 :         struct smb2_handle *h = NULL;
    1094           0 :         struct smb2_create io;
    1095           4 :         struct GUID create_guid = GUID_random();
    1096           4 :         bool ret = true;
    1097           0 :         struct smbcli_options options;
    1098             : 
    1099           4 :         options = tree->session->transport->options;
    1100             : 
    1101             :         /* Choose a random name in case the state is left a little funky. */
    1102           4 :         snprintf(fname, 256, "durable_v2_open_reopen2b_%s.dat",
    1103             :                  generate_random_str(tctx, 8));
    1104             : 
    1105           4 :         smb2_util_unlink(tree, fname);
    1106             : 
    1107           4 :         smb2_oplock_create_share(&io, fname,
    1108             :                                  smb2_util_share_access(""),
    1109           4 :                                  smb2_util_oplock_level("b"));
    1110           4 :         io.in.durable_open = false;
    1111           4 :         io.in.durable_open_v2 = true;
    1112           4 :         io.in.persistent_open = false;
    1113           4 :         io.in.create_guid = create_guid;
    1114           4 :         io.in.timeout = UINT32_MAX;
    1115             : 
    1116           4 :         status = smb2_create(tree, mem_ctx, &io);
    1117           4 :         CHECK_STATUS(status, NT_STATUS_OK);
    1118           4 :         _h = io.out.file.handle;
    1119           4 :         h = &_h;
    1120           4 :         CHECK_CREATED(&io, CREATED, FILE_ATTRIBUTE_ARCHIVE);
    1121           4 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level("b"));
    1122           4 :         CHECK_VAL(io.out.durable_open, false);
    1123           4 :         CHECK_VAL(io.out.durable_open_v2, true);
    1124           4 :         CHECK_VAL(io.out.persistent_open, false);
    1125           4 :         CHECK_VAL(io.out.timeout, 300*1000);
    1126             : 
    1127             :         /* disconnect, leaving the durable open */
    1128           4 :         TALLOC_FREE(tree);
    1129             : 
    1130           4 :         if (!torture_smb2_connection_ext(tctx, 0, &options, &tree)) {
    1131           0 :                 torture_warning(tctx, "couldn't reconnect, bailing\n");
    1132           0 :                 ret = false;
    1133           0 :                 goto done;
    1134             :         }
    1135             : 
    1136           4 :         ZERO_STRUCT(io);
    1137           4 :         io.in.fname = fname;
    1138           4 :         io.in.durable_handle_v2 = h;     /* durable v2 reconnect */
    1139           4 :         io.in.create_guid = GUID_zero(); /* but zero create GUID */
    1140           4 :         status = smb2_create(tree, mem_ctx, &io);
    1141           4 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
    1142             : 
    1143           4 :         ZERO_STRUCT(io);
    1144           4 :         io.in.fname = fname;
    1145           4 :         io.in.durable_handle = h; /* durable v1 (!) reconnect */
    1146           4 :         h = NULL;
    1147             : 
    1148           4 :         status = smb2_create(tree, mem_ctx, &io);
    1149           4 :         CHECK_STATUS(status, NT_STATUS_OK);
    1150           4 :         CHECK_CREATED(&io, EXISTED, FILE_ATTRIBUTE_ARCHIVE);
    1151           4 :         CHECK_VAL(io.out.durable_open, false);
    1152           4 :         CHECK_VAL(io.out.durable_open_v2, false); /* no dh2q response blob */
    1153           4 :         CHECK_VAL(io.out.persistent_open, false);
    1154           4 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level("b"));
    1155           4 :         _h = io.out.file.handle;
    1156           4 :         h = &_h;
    1157             : 
    1158           4 : done:
    1159           4 :         if (h != NULL) {
    1160           4 :                 smb2_util_close(tree, *h);
    1161             :         }
    1162             : 
    1163           4 :         smb2_util_unlink(tree, fname);
    1164             : 
    1165           4 :         talloc_free(tree);
    1166             : 
    1167           4 :         talloc_free(mem_ctx);
    1168             : 
    1169           4 :         return ret;
    1170             : }
    1171             : /**
    1172             :  * durable reconnect test:
    1173             :  * connect with v1, reconnect with v2 : fails (no create_guid...)
    1174             :  */
    1175           4 : bool test_durable_v2_open_reopen2c(struct torture_context *tctx,
    1176             :                                    struct smb2_tree *tree)
    1177             : {
    1178           0 :         NTSTATUS status;
    1179           4 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
    1180           0 :         char fname[256];
    1181           0 :         struct smb2_handle _h;
    1182           4 :         struct smb2_handle *h = NULL;
    1183           0 :         struct smb2_create io;
    1184           4 :         struct GUID create_guid = GUID_random();
    1185           4 :         bool ret = true;
    1186           0 :         struct smbcli_options options;
    1187             : 
    1188           4 :         options = tree->session->transport->options;
    1189             : 
    1190             :         /* Choose a random name in case the state is left a little funky. */
    1191           4 :         snprintf(fname, 256, "durable_v2_open_reopen2c_%s.dat",
    1192             :                  generate_random_str(tctx, 8));
    1193             : 
    1194           4 :         smb2_util_unlink(tree, fname);
    1195             : 
    1196           4 :         smb2_oplock_create_share(&io, fname,
    1197             :                                  smb2_util_share_access(""),
    1198           4 :                                  smb2_util_oplock_level("b"));
    1199           4 :         io.in.durable_open = true;
    1200           4 :         io.in.durable_open_v2 = false;
    1201           4 :         io.in.persistent_open = false;
    1202           4 :         io.in.create_guid = create_guid;
    1203           4 :         io.in.timeout = UINT32_MAX;
    1204             : 
    1205           4 :         status = smb2_create(tree, mem_ctx, &io);
    1206           4 :         CHECK_STATUS(status, NT_STATUS_OK);
    1207           4 :         _h = io.out.file.handle;
    1208           4 :         h = &_h;
    1209           4 :         CHECK_CREATED(&io, CREATED, FILE_ATTRIBUTE_ARCHIVE);
    1210           4 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level("b"));
    1211           4 :         CHECK_VAL(io.out.durable_open, true);
    1212           4 :         CHECK_VAL(io.out.durable_open_v2, false);
    1213           4 :         CHECK_VAL(io.out.persistent_open, false);
    1214           4 :         CHECK_VAL(io.out.timeout, 0);
    1215             : 
    1216             :         /* disconnect, leaving the durable open */
    1217           4 :         TALLOC_FREE(tree);
    1218             : 
    1219           4 :         if (!torture_smb2_connection_ext(tctx, 0, &options, &tree)) {
    1220           0 :                 torture_warning(tctx, "couldn't reconnect, bailing\n");
    1221           0 :                 ret = false;
    1222           0 :                 goto done;
    1223             :         }
    1224             : 
    1225           4 :         ZERO_STRUCT(io);
    1226           4 :         io.in.fname = fname;
    1227           4 :         io.in.durable_handle_v2 = h;     /* durable v2 reconnect */
    1228           4 :         io.in.create_guid = create_guid;
    1229           4 :         status = smb2_create(tree, mem_ctx, &io);
    1230           4 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
    1231             : 
    1232           4 : done:
    1233           4 :         if (h != NULL) {
    1234           4 :                 smb2_util_close(tree, *h);
    1235             :         }
    1236             : 
    1237           4 :         smb2_util_unlink(tree, fname);
    1238             : 
    1239           4 :         talloc_free(tree);
    1240             : 
    1241           4 :         talloc_free(mem_ctx);
    1242             : 
    1243           4 :         return ret;
    1244             : }
    1245             : 
    1246             : /**
    1247             :  * lease variant of reopen2
    1248             :  * basic test for doing a durable open
    1249             :  * tcp disconnect, reconnect, do a durable reopen (succeeds)
    1250             :  */
    1251           4 : bool test_durable_v2_open_reopen2_lease(struct torture_context *tctx,
    1252             :                                         struct smb2_tree *tree)
    1253             : {
    1254           0 :         NTSTATUS status;
    1255           4 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
    1256           0 :         char fname[256];
    1257           0 :         struct smb2_handle _h;
    1258           4 :         struct smb2_handle *h = NULL;
    1259           0 :         struct smb2_create io;
    1260           4 :         struct GUID create_guid = GUID_random();
    1261           0 :         struct smb2_lease ls;
    1262           0 :         uint64_t lease_key;
    1263           4 :         bool ret = true;
    1264           0 :         struct smbcli_options options;
    1265           0 :         uint32_t caps;
    1266             : 
    1267           4 :         caps = smb2cli_conn_server_capabilities(tree->session->transport->conn);
    1268           4 :         if (!(caps & SMB2_CAP_LEASING)) {
    1269           2 :                 torture_skip(tctx, "leases are not supported");
    1270             :         }
    1271             : 
    1272           2 :         options = tree->session->transport->options;
    1273             : 
    1274             :         /* Choose a random name in case the state is left a little funky. */
    1275           2 :         snprintf(fname, 256, "durable_v2_open_reopen2_%s.dat",
    1276             :                  generate_random_str(tctx, 8));
    1277             : 
    1278           2 :         smb2_util_unlink(tree, fname);
    1279             : 
    1280           2 :         lease_key = generate_random_u64();
    1281           2 :         smb2_lease_create(&io, &ls, false /* dir */, fname,
    1282             :                           lease_key, smb2_util_lease_state("RWH"));
    1283           2 :         io.in.durable_open = false;
    1284           2 :         io.in.durable_open_v2 = true;
    1285           2 :         io.in.persistent_open = false;
    1286           2 :         io.in.create_guid = create_guid;
    1287           2 :         io.in.timeout = UINT32_MAX;
    1288             : 
    1289           2 :         status = smb2_create(tree, mem_ctx, &io);
    1290           2 :         CHECK_STATUS(status, NT_STATUS_OK);
    1291           2 :         _h = io.out.file.handle;
    1292           2 :         h = &_h;
    1293           2 :         CHECK_CREATED(&io, CREATED, FILE_ATTRIBUTE_ARCHIVE);
    1294           2 :         CHECK_VAL(io.out.durable_open, false);
    1295           2 :         CHECK_VAL(io.out.durable_open_v2, true);
    1296           2 :         CHECK_VAL(io.out.persistent_open, false);
    1297           2 :         CHECK_VAL(io.out.timeout, 300*1000);
    1298           2 :         CHECK_VAL(io.out.oplock_level, SMB2_OPLOCK_LEVEL_LEASE);
    1299           2 :         CHECK_VAL(io.out.lease_response.lease_key.data[0], lease_key);
    1300           2 :         CHECK_VAL(io.out.lease_response.lease_key.data[1], ~lease_key);
    1301           2 :         CHECK_VAL(io.out.lease_response.lease_state,
    1302             :                   smb2_util_lease_state("RWH"));
    1303           2 :         CHECK_VAL(io.out.lease_response.lease_flags, 0);
    1304           2 :         CHECK_VAL(io.out.lease_response.lease_duration, 0);
    1305             : 
    1306             :         /* disconnect, reconnect and then do durable reopen */
    1307           2 :         TALLOC_FREE(tree);
    1308             : 
    1309           2 :         if (!torture_smb2_connection_ext(tctx, 0, &options, &tree)) {
    1310           0 :                 torture_warning(tctx, "couldn't reconnect, bailing\n");
    1311           0 :                 ret = false;
    1312           0 :                 goto done;
    1313             :         }
    1314             : 
    1315             :         /* a few failure tests: */
    1316             : 
    1317             :         /*
    1318             :          * several attempts without lease attached:
    1319             :          * all fail with NT_STATUS_OBJECT_NAME_NOT_FOUND
    1320             :          * irrespective of file name provided
    1321             :          */
    1322             : 
    1323           2 :         ZERO_STRUCT(io);
    1324           2 :         io.in.fname = "";
    1325           2 :         io.in.durable_handle_v2 = h;
    1326           2 :         status = smb2_create(tree, mem_ctx, &io);
    1327           2 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
    1328             : 
    1329           2 :         ZERO_STRUCT(io);
    1330           2 :         io.in.fname = "__non_existing_fname__";
    1331           2 :         io.in.durable_handle_v2 = h;
    1332           2 :         status = smb2_create(tree, mem_ctx, &io);
    1333           2 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
    1334             : 
    1335           2 :         ZERO_STRUCT(io);
    1336           2 :         io.in.fname = fname;
    1337           2 :         io.in.durable_handle_v2 = h;
    1338           2 :         status = smb2_create(tree, mem_ctx, &io);
    1339           2 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
    1340             : 
    1341             :         /*
    1342             :          * attempt with lease provided, but
    1343             :          * with a changed lease key. => fails
    1344             :          */
    1345           2 :         ZERO_STRUCT(io);
    1346           2 :         io.in.fname = fname;
    1347           2 :         io.in.durable_open_v2 = false;
    1348           2 :         io.in.durable_handle_v2 = h;
    1349           2 :         io.in.create_guid = create_guid;
    1350           2 :         io.in.lease_request = &ls;
    1351           2 :         io.in.oplock_level = SMB2_OPLOCK_LEVEL_LEASE;
    1352             :         /* a wrong lease key lets the request fail */
    1353           2 :         ls.lease_key.data[0]++;
    1354             : 
    1355           2 :         status = smb2_create(tree, mem_ctx, &io);
    1356           2 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
    1357             : 
    1358             :         /* restore the correct lease key */
    1359           2 :         ls.lease_key.data[0]--;
    1360             : 
    1361             :         /*
    1362             :          * this last failing attempt is almost correct:
    1363             :          * only problem is: we use the wrong filename...
    1364             :          * Note that this gives INVALID_PARAMETER.
    1365             :          * This is different from oplocks!
    1366             :          */
    1367           2 :         ZERO_STRUCT(io);
    1368           2 :         io.in.fname = "__non_existing_fname__";
    1369           2 :         io.in.durable_open_v2 = false;
    1370           2 :         io.in.durable_handle_v2 = h;
    1371           2 :         io.in.create_guid = create_guid;
    1372           2 :         io.in.lease_request = &ls;
    1373           2 :         io.in.oplock_level = SMB2_OPLOCK_LEVEL_LEASE;
    1374             : 
    1375           2 :         status = smb2_create(tree, mem_ctx, &io);
    1376           2 :         CHECK_STATUS(status, NT_STATUS_INVALID_PARAMETER);
    1377             : 
    1378             :         /*
    1379             :          * Now for a succeeding reconnect:
    1380             :          */
    1381             : 
    1382           2 :         ZERO_STRUCT(io);
    1383           2 :         io.in.fname = fname;
    1384           2 :         io.in.durable_open_v2 = false;
    1385           2 :         io.in.durable_handle_v2 = h;
    1386           2 :         io.in.create_guid = create_guid;
    1387           2 :         io.in.lease_request = &ls;
    1388           2 :         io.in.oplock_level = SMB2_OPLOCK_LEVEL_LEASE;
    1389             : 
    1390             :         /* the requested lease state is irrelevant */
    1391           2 :         ls.lease_state = smb2_util_lease_state("");
    1392             : 
    1393           2 :         h = NULL;
    1394             : 
    1395           2 :         status = smb2_create(tree, mem_ctx, &io);
    1396           2 :         CHECK_STATUS(status, NT_STATUS_OK);
    1397             : 
    1398           2 :         CHECK_CREATED(&io, EXISTED, FILE_ATTRIBUTE_ARCHIVE);
    1399           2 :         CHECK_VAL(io.out.durable_open, false);
    1400           2 :         CHECK_VAL(io.out.durable_open_v2, false); /* no dh2q response blob */
    1401           2 :         CHECK_VAL(io.out.persistent_open, false);
    1402           2 :         CHECK_VAL(io.out.oplock_level, SMB2_OPLOCK_LEVEL_LEASE);
    1403           2 :         CHECK_VAL(io.out.lease_response.lease_key.data[0], lease_key);
    1404           2 :         CHECK_VAL(io.out.lease_response.lease_key.data[1], ~lease_key);
    1405           2 :         CHECK_VAL(io.out.lease_response.lease_state,
    1406             :                   smb2_util_lease_state("RWH"));
    1407           2 :         CHECK_VAL(io.out.lease_response.lease_flags, 0);
    1408           2 :         CHECK_VAL(io.out.lease_response.lease_duration, 0);
    1409           2 :         _h = io.out.file.handle;
    1410           2 :         h = &_h;
    1411             : 
    1412             :         /* disconnect one more time */
    1413           2 :         TALLOC_FREE(tree);
    1414             : 
    1415           2 :         if (!torture_smb2_connection_ext(tctx, 0, &options, &tree)) {
    1416           0 :                 torture_warning(tctx, "couldn't reconnect, bailing\n");
    1417           0 :                 ret = false;
    1418           0 :                 goto done;
    1419             :         }
    1420             : 
    1421             :         /*
    1422             :          * demonstrate that various parameters are ignored
    1423             :          * in the reconnect
    1424             :          */
    1425             : 
    1426           2 :         ZERO_STRUCT(io);
    1427             :         /*
    1428             :          * These are completely ignored by the server
    1429             :          */
    1430           2 :         io.in.security_flags = 0x78;
    1431           2 :         io.in.oplock_level = 0x78;
    1432           2 :         io.in.impersonation_level = 0x12345678;
    1433           2 :         io.in.create_flags = 0x12345678;
    1434           2 :         io.in.reserved = 0x12345678;
    1435           2 :         io.in.desired_access = 0x12345678;
    1436           2 :         io.in.file_attributes = 0x12345678;
    1437           2 :         io.in.share_access = 0x12345678;
    1438           2 :         io.in.create_disposition = 0x12345678;
    1439           2 :         io.in.create_options = 0x12345678;
    1440             : 
    1441             :         /*
    1442             :          * only these are checked:
    1443             :          * - io.in.fname
    1444             :          * - io.in.durable_handle_v2,
    1445             :          * - io.in.create_guid
    1446             :          * - io.in.lease_request->lease_key
    1447             :          */
    1448             : 
    1449           2 :         io.in.fname = fname;
    1450           2 :         io.in.durable_open_v2 = false;
    1451           2 :         io.in.durable_handle_v2 = h;
    1452           2 :         io.in.create_guid = create_guid;
    1453           2 :         io.in.lease_request = &ls;
    1454             : 
    1455             :         /* the requested lease state is irrelevant */
    1456           2 :         ls.lease_state = smb2_util_lease_state("");
    1457             : 
    1458           2 :         h = NULL;
    1459             : 
    1460           2 :         status = smb2_create(tree, mem_ctx, &io);
    1461           2 :         CHECK_STATUS(status, NT_STATUS_OK);
    1462             : 
    1463           2 :         CHECK_CREATED(&io, EXISTED, FILE_ATTRIBUTE_ARCHIVE);
    1464           2 :         CHECK_VAL(io.out.durable_open, false);
    1465           2 :         CHECK_VAL(io.out.durable_open_v2, false); /* no dh2q response blob */
    1466           2 :         CHECK_VAL(io.out.persistent_open, false);
    1467           2 :         CHECK_VAL(io.out.oplock_level, SMB2_OPLOCK_LEVEL_LEASE);
    1468           2 :         CHECK_VAL(io.out.lease_response.lease_key.data[0], lease_key);
    1469           2 :         CHECK_VAL(io.out.lease_response.lease_key.data[1], ~lease_key);
    1470           2 :         CHECK_VAL(io.out.lease_response.lease_state,
    1471             :                   smb2_util_lease_state("RWH"));
    1472           2 :         CHECK_VAL(io.out.lease_response.lease_flags, 0);
    1473           2 :         CHECK_VAL(io.out.lease_response.lease_duration, 0);
    1474             : 
    1475           2 :         _h = io.out.file.handle;
    1476           2 :         h = &_h;
    1477             : 
    1478           2 : done:
    1479           2 :         if (h != NULL) {
    1480           2 :                 smb2_util_close(tree, *h);
    1481             :         }
    1482             : 
    1483           2 :         smb2_util_unlink(tree, fname);
    1484             : 
    1485           2 :         talloc_free(tree);
    1486             : 
    1487           2 :         talloc_free(mem_ctx);
    1488             : 
    1489           2 :         return ret;
    1490             : }
    1491             : 
    1492             : /**
    1493             :  * lease_v2 variant of reopen2
    1494             :  * basic test for doing a durable open
    1495             :  * tcp disconnect, reconnect, do a durable reopen (succeeds)
    1496             :  */
    1497           4 : bool test_durable_v2_open_reopen2_lease_v2(struct torture_context *tctx,
    1498             :                                            struct smb2_tree *tree)
    1499             : {
    1500           0 :         NTSTATUS status;
    1501           4 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
    1502           0 :         char fname[256];
    1503           0 :         struct smb2_handle _h;
    1504           4 :         struct smb2_handle *h = NULL;
    1505           0 :         struct smb2_create io;
    1506           4 :         struct GUID create_guid = GUID_random();
    1507           0 :         struct smb2_lease ls;
    1508           0 :         uint64_t lease_key;
    1509           4 :         bool ret = true;
    1510           0 :         struct smbcli_options options;
    1511           0 :         uint32_t caps;
    1512             : 
    1513           4 :         caps = smb2cli_conn_server_capabilities(tree->session->transport->conn);
    1514           4 :         if (!(caps & SMB2_CAP_LEASING)) {
    1515           2 :                 torture_skip(tctx, "leases are not supported");
    1516             :         }
    1517             : 
    1518           2 :         options = tree->session->transport->options;
    1519             : 
    1520           2 :         smb2_deltree(tree, __func__);
    1521           2 :         status = torture_smb2_testdir(tree, __func__, &_h);
    1522           2 :         torture_assert_ntstatus_ok_goto(tctx, status, ret, done,
    1523             :                                         "torture_smb2_testdir failed\n");
    1524           2 :         smb2_util_close(tree, _h);
    1525             : 
    1526             :         /* Choose a random name in case the state is left a little funky. */
    1527           2 :         snprintf(fname, 256, "%s\\durable_v2_open_reopen2_%s.dat",
    1528             :                  __func__, generate_random_str(tctx, 8));
    1529             : 
    1530           2 :         smb2_util_unlink(tree, fname);
    1531             : 
    1532           2 :         lease_key = random();
    1533           2 :         smb2_lease_v2_create(&io, &ls, false /* dir */, fname,
    1534             :                              lease_key, 0, /* parent lease key */
    1535             :                              smb2_util_lease_state("RWH"), 0 /* lease epoch */);
    1536           2 :         io.in.durable_open = false;
    1537           2 :         io.in.durable_open_v2 = true;
    1538           2 :         io.in.persistent_open = false;
    1539           2 :         io.in.create_guid = create_guid;
    1540           2 :         io.in.timeout = UINT32_MAX;
    1541             : 
    1542           2 :         status = smb2_create(tree, mem_ctx, &io);
    1543           2 :         CHECK_STATUS(status, NT_STATUS_OK);
    1544           2 :         _h = io.out.file.handle;
    1545           2 :         h = &_h;
    1546           2 :         CHECK_CREATED(&io, CREATED, FILE_ATTRIBUTE_ARCHIVE);
    1547           2 :         CHECK_VAL(io.out.durable_open, false);
    1548           2 :         CHECK_VAL(io.out.durable_open_v2, true);
    1549           2 :         CHECK_VAL(io.out.persistent_open, false);
    1550           2 :         CHECK_VAL(io.out.timeout, 300*1000);
    1551           2 :         CHECK_VAL(io.out.oplock_level, SMB2_OPLOCK_LEVEL_LEASE);
    1552           2 :         CHECK_VAL(io.out.lease_response_v2.lease_key.data[0], lease_key);
    1553           2 :         CHECK_VAL(io.out.lease_response_v2.lease_key.data[1], ~lease_key);
    1554             : 
    1555             :         /* disconnect, reconnect and then do durable reopen */
    1556           2 :         TALLOC_FREE(tree);
    1557             : 
    1558           2 :         if (!torture_smb2_connection_ext(tctx, 0, &options, &tree)) {
    1559           0 :                 torture_warning(tctx, "couldn't reconnect, bailing\n");
    1560           0 :                 ret = false;
    1561           0 :                 goto done;
    1562             :         }
    1563             : 
    1564             :         /* a few failure tests: */
    1565             : 
    1566             :         /*
    1567             :          * several attempts without lease attached:
    1568             :          * all fail with NT_STATUS_OBJECT_NAME_NOT_FOUND
    1569             :          * irrespective of file name provided
    1570             :          */
    1571             : 
    1572           2 :         ZERO_STRUCT(io);
    1573           2 :         io.in.fname = "";
    1574           2 :         io.in.durable_handle_v2 = h;
    1575           2 :         status = smb2_create(tree, mem_ctx, &io);
    1576           2 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
    1577             : 
    1578           2 :         ZERO_STRUCT(io);
    1579           2 :         io.in.fname = "__non_existing_fname__";
    1580           2 :         io.in.durable_handle_v2 = h;
    1581           2 :         status = smb2_create(tree, mem_ctx, &io);
    1582           2 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
    1583             : 
    1584           2 :         ZERO_STRUCT(io);
    1585           2 :         io.in.fname = fname;
    1586           2 :         io.in.durable_handle_v2 = h;
    1587           2 :         status = smb2_create(tree, mem_ctx, &io);
    1588           2 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
    1589             : 
    1590             :         /*
    1591             :          * attempt with lease provided, but
    1592             :          * with a changed lease key. => fails
    1593             :          */
    1594           2 :         ZERO_STRUCT(io);
    1595           2 :         io.in.fname = fname;
    1596           2 :         io.in.durable_open_v2 = false;
    1597           2 :         io.in.durable_handle_v2 = h;
    1598           2 :         io.in.create_guid = create_guid;
    1599           2 :         io.in.lease_request_v2 = &ls;
    1600           2 :         io.in.oplock_level = SMB2_OPLOCK_LEVEL_LEASE;
    1601             :         /* a wrong lease key lets the request fail */
    1602           2 :         ls.lease_key.data[0]++;
    1603             : 
    1604           2 :         status = smb2_create(tree, mem_ctx, &io);
    1605           2 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
    1606             : 
    1607             :         /* restore the correct lease key */
    1608           2 :         ls.lease_key.data[0]--;
    1609             : 
    1610             : 
    1611             :         /*
    1612             :          * this last failing attempt is almost correct:
    1613             :          * only problem is: we use the wrong filename...
    1614             :          * Note that this gives INVALID_PARAMETER.
    1615             :          * This is different from oplocks!
    1616             :          */
    1617           2 :         ZERO_STRUCT(io);
    1618           2 :         io.in.fname = "__non_existing_fname__";
    1619           2 :         io.in.durable_open_v2 = false;
    1620           2 :         io.in.durable_handle_v2 = h;
    1621           2 :         io.in.create_guid = create_guid;
    1622           2 :         io.in.lease_request_v2 = &ls;
    1623           2 :         io.in.oplock_level = SMB2_OPLOCK_LEVEL_LEASE;
    1624             : 
    1625           2 :         status = smb2_create(tree, mem_ctx, &io);
    1626           2 :         CHECK_STATUS(status, NT_STATUS_INVALID_PARAMETER);
    1627             : 
    1628             :         /*
    1629             :          * Now for a succeeding reconnect:
    1630             :          */
    1631             : 
    1632           2 :         ZERO_STRUCT(io);
    1633           2 :         io.in.fname = fname;
    1634           2 :         io.in.durable_open_v2 = false;
    1635           2 :         io.in.durable_handle_v2 = h;
    1636           2 :         io.in.create_guid = create_guid;
    1637           2 :         io.in.lease_request_v2 = &ls;
    1638           2 :         io.in.oplock_level = SMB2_OPLOCK_LEVEL_LEASE;
    1639             : 
    1640             :         /* the requested lease state is irrelevant */
    1641           2 :         ls.lease_state = smb2_util_lease_state("");
    1642             : 
    1643           2 :         h = NULL;
    1644             : 
    1645           2 :         status = smb2_create(tree, mem_ctx, &io);
    1646           2 :         CHECK_STATUS(status, NT_STATUS_OK);
    1647             : 
    1648           2 :         CHECK_CREATED(&io, EXISTED, FILE_ATTRIBUTE_ARCHIVE);
    1649           2 :         CHECK_VAL(io.out.durable_open, false);
    1650           2 :         CHECK_VAL(io.out.durable_open_v2, false); /* no dh2q response blob */
    1651           2 :         CHECK_VAL(io.out.persistent_open, false);
    1652           2 :         CHECK_VAL(io.out.oplock_level, SMB2_OPLOCK_LEVEL_LEASE);
    1653           2 :         CHECK_VAL(io.out.lease_response_v2.lease_key.data[0], lease_key);
    1654           2 :         CHECK_VAL(io.out.lease_response_v2.lease_key.data[1], ~lease_key);
    1655           2 :         CHECK_VAL(io.out.lease_response_v2.lease_state,
    1656             :                   smb2_util_lease_state("RWH"));
    1657           2 :         CHECK_VAL(io.out.lease_response_v2.lease_flags, 0);
    1658           2 :         CHECK_VAL(io.out.lease_response_v2.lease_duration, 0);
    1659           2 :         _h = io.out.file.handle;
    1660           2 :         h = &_h;
    1661             : 
    1662             :         /* disconnect one more time */
    1663           2 :         TALLOC_FREE(tree);
    1664             : 
    1665           2 :         if (!torture_smb2_connection_ext(tctx, 0, &options, &tree)) {
    1666           0 :                 torture_warning(tctx, "couldn't reconnect, bailing\n");
    1667           0 :                 ret = false;
    1668           0 :                 goto done;
    1669             :         }
    1670             : 
    1671             :         /*
    1672             :          * demonstrate that various parameters are ignored
    1673             :          * in the reconnect
    1674             :          */
    1675             : 
    1676           2 :         ZERO_STRUCT(io);
    1677             :         /*
    1678             :          * These are completely ignored by the server
    1679             :          */
    1680           2 :         io.in.security_flags = 0x78;
    1681           2 :         io.in.oplock_level = 0x78;
    1682           2 :         io.in.impersonation_level = 0x12345678;
    1683           2 :         io.in.create_flags = 0x12345678;
    1684           2 :         io.in.reserved = 0x12345678;
    1685           2 :         io.in.desired_access = 0x12345678;
    1686           2 :         io.in.file_attributes = 0x12345678;
    1687           2 :         io.in.share_access = 0x12345678;
    1688           2 :         io.in.create_disposition = 0x12345678;
    1689           2 :         io.in.create_options = 0x12345678;
    1690           2 :         io.in.fname = "__non_existing_fname__";
    1691             : 
    1692             :         /*
    1693             :          * only these are checked:
    1694             :          * - io.in.fname
    1695             :          * - io.in.durable_handle_v2,
    1696             :          * - io.in.create_guid
    1697             :          * - io.in.lease_request_v2->lease_key
    1698             :          */
    1699             : 
    1700           2 :         io.in.fname = fname;
    1701           2 :         io.in.durable_open_v2 = false;
    1702           2 :         io.in.durable_handle_v2 = h;
    1703           2 :         io.in.create_guid = create_guid;
    1704           2 :         io.in.lease_request_v2 = &ls;
    1705             : 
    1706             :         /* the requested lease state is irrelevant */
    1707           2 :         ls.lease_state = smb2_util_lease_state("");
    1708             : 
    1709           2 :         h = NULL;
    1710             : 
    1711           2 :         status = smb2_create(tree, mem_ctx, &io);
    1712           2 :         CHECK_STATUS(status, NT_STATUS_OK);
    1713           2 :         CHECK_CREATED(&io, EXISTED, FILE_ATTRIBUTE_ARCHIVE);
    1714           2 :         CHECK_VAL(io.out.durable_open, false);
    1715           2 :         CHECK_VAL(io.out.durable_open_v2, false); /* no dh2q response blob */
    1716           2 :         CHECK_VAL(io.out.persistent_open, false);
    1717           2 :         CHECK_VAL(io.out.oplock_level, SMB2_OPLOCK_LEVEL_LEASE);
    1718           2 :         CHECK_VAL(io.out.lease_response_v2.lease_key.data[0], lease_key);
    1719           2 :         CHECK_VAL(io.out.lease_response_v2.lease_key.data[1], ~lease_key);
    1720           2 :         CHECK_VAL(io.out.lease_response_v2.lease_state,
    1721             :                   smb2_util_lease_state("RWH"));
    1722           2 :         CHECK_VAL(io.out.lease_response_v2.lease_flags, 0);
    1723           2 :         CHECK_VAL(io.out.lease_response_v2.lease_duration, 0);
    1724             : 
    1725           2 :         _h = io.out.file.handle;
    1726           2 :         h = &_h;
    1727             : 
    1728           2 : done:
    1729           2 :         if (h != NULL) {
    1730           2 :                 smb2_util_close(tree, *h);
    1731             :         }
    1732             : 
    1733           2 :         smb2_util_unlink(tree, fname);
    1734           2 :         smb2_deltree(tree, __func__);
    1735             : 
    1736           2 :         talloc_free(tree);
    1737             : 
    1738           2 :         talloc_free(mem_ctx);
    1739             : 
    1740           2 :         return ret;
    1741             : }
    1742             : 
    1743             : /**
    1744             :  * Test durable request / reconnect with AppInstanceId
    1745             :  */
    1746           4 : bool test_durable_v2_open_app_instance(struct torture_context *tctx,
    1747             :                                        struct smb2_tree *tree1,
    1748             :                                        struct smb2_tree *tree2)
    1749             : {
    1750           0 :         NTSTATUS status;
    1751           4 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
    1752           0 :         char fname[256];
    1753           0 :         struct smb2_handle _h1, _h2;
    1754           4 :         struct smb2_handle *h1 = NULL, *h2 = NULL;
    1755           0 :         struct smb2_create io1, io2;
    1756           4 :         bool ret = true;
    1757           4 :         struct GUID create_guid_1 = GUID_random();
    1758           4 :         struct GUID create_guid_2 = GUID_random();
    1759           4 :         struct GUID app_instance_id = GUID_random();
    1760             : 
    1761             :         /* Choose a random name in case the state is left a little funky. */
    1762           4 :         snprintf(fname, 256, "durable_v2_open_app_instance_%s.dat",
    1763             :                  generate_random_str(tctx, 8));
    1764             : 
    1765           4 :         smb2_util_unlink(tree1, fname);
    1766             : 
    1767           4 :         ZERO_STRUCT(break_info);
    1768           4 :         tree1->session->transport->oplock.handler = torture_oplock_handler;
    1769           4 :         tree1->session->transport->oplock.private_data = tree1;
    1770             : 
    1771           4 :         smb2_oplock_create_share(&io1, fname,
    1772             :                                  smb2_util_share_access(""),
    1773           4 :                                  smb2_util_oplock_level("b"));
    1774           4 :         io1.in.durable_open = false;
    1775           4 :         io1.in.durable_open_v2 = true;
    1776           4 :         io1.in.persistent_open = false;
    1777           4 :         io1.in.create_guid = create_guid_1;
    1778           4 :         io1.in.app_instance_id = &app_instance_id;
    1779           4 :         io1.in.timeout = UINT32_MAX;
    1780             : 
    1781           4 :         status = smb2_create(tree1, mem_ctx, &io1);
    1782           4 :         CHECK_STATUS(status, NT_STATUS_OK);
    1783           4 :         _h1 = io1.out.file.handle;
    1784           4 :         h1 = &_h1;
    1785           4 :         CHECK_CREATED(&io1, CREATED, FILE_ATTRIBUTE_ARCHIVE);
    1786           4 :         CHECK_VAL(io1.out.oplock_level, smb2_util_oplock_level("b"));
    1787           4 :         CHECK_VAL(io1.out.durable_open, false);
    1788           4 :         CHECK_VAL(io1.out.durable_open_v2, true);
    1789           4 :         CHECK_VAL(io1.out.persistent_open, false);
    1790           4 :         CHECK_VAL(io1.out.timeout, 300*1000);
    1791             : 
    1792             :         /*
    1793             :          * try to open the file as durable from a second tree with
    1794             :          * a different create guid but the same app_instance_id
    1795             :          * while the first handle is still open.
    1796             :          */
    1797             : 
    1798           4 :         smb2_oplock_create_share(&io2, fname,
    1799             :                                  smb2_util_share_access(""),
    1800           4 :                                  smb2_util_oplock_level("b"));
    1801           4 :         io2.in.durable_open = false;
    1802           4 :         io2.in.durable_open_v2 = true;
    1803           4 :         io2.in.persistent_open = false;
    1804           4 :         io2.in.create_guid = create_guid_2;
    1805           4 :         io2.in.app_instance_id = &app_instance_id;
    1806           4 :         io2.in.timeout = UINT32_MAX;
    1807             : 
    1808           4 :         status = smb2_create(tree2, mem_ctx, &io2);
    1809           4 :         CHECK_STATUS(status, NT_STATUS_OK);
    1810           4 :         _h2 = io2.out.file.handle;
    1811           4 :         h2 = &_h2;
    1812           4 :         CHECK_CREATED(&io2, EXISTED, FILE_ATTRIBUTE_ARCHIVE);
    1813           4 :         CHECK_VAL(io2.out.oplock_level, smb2_util_oplock_level("b"));
    1814           4 :         CHECK_VAL(io2.out.durable_open, false);
    1815           4 :         CHECK_VAL(io2.out.durable_open_v2, true);
    1816           4 :         CHECK_VAL(io2.out.persistent_open, false);
    1817           4 :         CHECK_VAL(io2.out.timeout, 300*1000);
    1818             : 
    1819           4 :         CHECK_VAL(break_info.count, 0);
    1820             : 
    1821           4 :         status = smb2_util_close(tree1, *h1);
    1822           4 :         CHECK_STATUS(status, NT_STATUS_FILE_CLOSED);
    1823           4 :         h1 = NULL;
    1824             : 
    1825           4 : done:
    1826           4 :         if (h1 != NULL) {
    1827           0 :                 smb2_util_close(tree1, *h1);
    1828             :         }
    1829           4 :         if (h2 != NULL) {
    1830           4 :                 smb2_util_close(tree2, *h2);
    1831             :         }
    1832             : 
    1833           4 :         smb2_util_unlink(tree2, fname);
    1834             : 
    1835           4 :         talloc_free(tree1);
    1836           4 :         talloc_free(tree2);
    1837             : 
    1838           4 :         talloc_free(mem_ctx);
    1839             : 
    1840           4 :         return ret;
    1841             : }
    1842             : 
    1843             : 
    1844             : /**
    1845             :  * basic persistent open test.
    1846             :  *
    1847             :  * This test tests durable open with all possible oplock types.
    1848             :  */
    1849             : 
    1850             : struct durable_open_vs_oplock persistent_open_oplock_ca_table[NUM_OPLOCK_OPEN_TESTS] =
    1851             : {
    1852             :         { "", "", true, true },
    1853             :         { "", "R", true, true },
    1854             :         { "", "W", true, true },
    1855             :         { "", "D", true, true },
    1856             :         { "", "RD", true, true },
    1857             :         { "", "RW", true, true },
    1858             :         { "", "WD", true, true },
    1859             :         { "", "RWD", true, true },
    1860             : 
    1861             :         { "s", "", true, true },
    1862             :         { "s", "R", true, true },
    1863             :         { "s", "W", true, true },
    1864             :         { "s", "D", true, true },
    1865             :         { "s", "RD", true, true },
    1866             :         { "s", "RW", true, true },
    1867             :         { "s", "WD", true, true },
    1868             :         { "s", "RWD", true, true },
    1869             : 
    1870             :         { "x", "", true, true },
    1871             :         { "x", "R", true, true },
    1872             :         { "x", "W", true, true },
    1873             :         { "x", "D", true, true },
    1874             :         { "x", "RD", true, true },
    1875             :         { "x", "RW", true, true },
    1876             :         { "x", "WD", true, true },
    1877             :         { "x", "RWD", true, true },
    1878             : 
    1879             :         { "b", "", true, true },
    1880             :         { "b", "R", true, true },
    1881             :         { "b", "W", true, true },
    1882             :         { "b", "D", true, true },
    1883             :         { "b", "RD", true, true },
    1884             :         { "b", "RW", true, true },
    1885             :         { "b", "WD", true, true },
    1886             :         { "b", "RWD", true, true },
    1887             : };
    1888             : 
    1889           4 : bool test_persistent_open_oplock(struct torture_context *tctx,
    1890             :                                  struct smb2_tree *tree)
    1891             : {
    1892           0 :         char fname[256];
    1893           4 :         bool ret = true;
    1894           0 :         uint32_t share_capabilities;
    1895           4 :         bool share_is_ca = false;
    1896           0 :         struct durable_open_vs_oplock *table;
    1897             : 
    1898             :         /* Choose a random name in case the state is left a little funky. */
    1899           4 :         snprintf(fname, 256, "persistent_open_oplock_%s.dat", generate_random_str(tctx, 8));
    1900             : 
    1901           4 :         share_capabilities = smb2cli_tcon_capabilities(tree->smbXcli);
    1902           4 :         share_is_ca = share_capabilities & SMB2_SHARE_CAP_CONTINUOUS_AVAILABILITY;
    1903             : 
    1904           4 :         if (share_is_ca) {
    1905           0 :                 table = persistent_open_oplock_ca_table;
    1906             :         } else {
    1907           4 :                 table = durable_open_vs_oplock_table;
    1908             :         }
    1909             : 
    1910           4 :         ret = test_durable_v2_open_oplock_table(tctx, tree, fname,
    1911             :                                                 true, /* request_persistent */
    1912             :                                                 table,
    1913             :                                                 NUM_OPLOCK_OPEN_TESTS);
    1914             : 
    1915           4 :         talloc_free(tree);
    1916             : 
    1917           4 :         return ret;
    1918             : }
    1919             : 
    1920             : /**
    1921             :  * basic persistent handle open test.
    1922             :  * persistent state should only be granted when requested
    1923             :  * along with a batch oplock or a handle lease.
    1924             :  *
    1925             :  * This test tests persistent open with all valid lease types.
    1926             :  */
    1927             : 
    1928             : struct durable_open_vs_lease persistent_open_lease_ca_table[NUM_LEASE_OPEN_TESTS] =
    1929             : {
    1930             :         { "", "", true, true },
    1931             :         { "", "R", true, true },
    1932             :         { "", "W", true, true },
    1933             :         { "", "D", true, true },
    1934             :         { "", "RW", true, true },
    1935             :         { "", "RD", true, true },
    1936             :         { "", "WD", true, true },
    1937             :         { "", "RWD", true, true },
    1938             : 
    1939             :         { "R", "", true, true },
    1940             :         { "R", "R", true, true },
    1941             :         { "R", "W", true, true },
    1942             :         { "R", "D", true, true },
    1943             :         { "R", "RW", true, true },
    1944             :         { "R", "RD", true, true },
    1945             :         { "R", "DW", true, true },
    1946             :         { "R", "RWD", true, true },
    1947             : 
    1948             :         { "RW", "", true, true },
    1949             :         { "RW", "R", true, true },
    1950             :         { "RW", "W", true, true },
    1951             :         { "RW", "D", true, true },
    1952             :         { "RW", "RW", true, true },
    1953             :         { "RW", "RD", true, true },
    1954             :         { "RW", "WD", true, true },
    1955             :         { "RW", "RWD", true, true },
    1956             : 
    1957             :         { "RH", "", true, true },
    1958             :         { "RH", "R", true, true },
    1959             :         { "RH", "W", true, true },
    1960             :         { "RH", "D", true, true },
    1961             :         { "RH", "RW", true, true },
    1962             :         { "RH", "RD", true, true },
    1963             :         { "RH", "WD", true, true },
    1964             :         { "RH", "RWD", true, true },
    1965             : 
    1966             :         { "RHW", "", true, true },
    1967             :         { "RHW", "R", true, true },
    1968             :         { "RHW", "W", true, true },
    1969             :         { "RHW", "D", true, true },
    1970             :         { "RHW", "RW", true, true },
    1971             :         { "RHW", "RD", true, true },
    1972             :         { "RHW", "WD", true, true },
    1973             :         { "RHW", "RWD", true, true },
    1974             : };
    1975             : 
    1976           4 : bool test_persistent_open_lease(struct torture_context *tctx,
    1977             :                                 struct smb2_tree *tree)
    1978             : {
    1979           0 :         char fname[256];
    1980           4 :         bool ret = true;
    1981           0 :         uint32_t caps;
    1982           0 :         uint32_t share_capabilities;
    1983           0 :         bool share_is_ca;
    1984           0 :         struct durable_open_vs_lease *table;
    1985             : 
    1986           4 :         caps = smb2cli_conn_server_capabilities(tree->session->transport->conn);
    1987           4 :         if (!(caps & SMB2_CAP_LEASING)) {
    1988           2 :                 torture_skip(tctx, "leases are not supported");
    1989             :         }
    1990             : 
    1991             :         /* Choose a random name in case the state is left a little funky. */
    1992           2 :         snprintf(fname, 256, "persistent_open_lease_%s.dat", generate_random_str(tctx, 8));
    1993             : 
    1994           2 :         share_capabilities = smb2cli_tcon_capabilities(tree->smbXcli);
    1995           2 :         share_is_ca = share_capabilities & SMB2_SHARE_CAP_CONTINUOUS_AVAILABILITY;
    1996             : 
    1997           2 :         if (share_is_ca) {
    1998           0 :                 table = persistent_open_lease_ca_table;
    1999             :         } else {
    2000           2 :                 table = durable_open_vs_lease_table;
    2001             :         }
    2002             : 
    2003           2 :         ret = test_durable_v2_open_lease_table(tctx, tree, fname,
    2004             :                                                true, /* request_persistent */
    2005             :                                                table,
    2006             :                                                NUM_LEASE_OPEN_TESTS);
    2007             : 
    2008           2 :         talloc_free(tree);
    2009             : 
    2010           2 :         return ret;
    2011             : }
    2012             : 
    2013             : /**
    2014             :  * setfileinfo test for doing a durable open
    2015             :  * create the file with lease and durable handle,
    2016             :  * write to it (via set end-of-file), tcp disconnect,
    2017             :  * reconnect, do a durable reopen - should succeed.
    2018             :  *
    2019             :  * BUG: https://bugzilla.samba.org/show_bug.cgi?id=15022
    2020             :  */
    2021           4 : bool test_durable_v2_setinfo(struct torture_context *tctx,
    2022             :                                            struct smb2_tree *tree)
    2023             : {
    2024           0 :         NTSTATUS status;
    2025           4 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
    2026           0 :         char fname[256];
    2027           0 :         struct smb2_handle _h;
    2028           4 :         struct smb2_handle *h = NULL;
    2029           0 :         struct smb2_create io;
    2030           0 :         union smb_setfileinfo si;
    2031           4 :         struct GUID create_guid = GUID_random();
    2032           0 :         struct smb2_lease ls;
    2033           0 :         uint64_t lease_key;
    2034           4 :         bool ret = true;
    2035           0 :         struct smbcli_options options;
    2036           0 :         uint32_t caps;
    2037             : 
    2038           4 :         caps = smb2cli_conn_server_capabilities(tree->session->transport->conn);
    2039           4 :         if (!(caps & SMB2_CAP_LEASING)) {
    2040           2 :                 torture_skip(tctx, "leases are not supported");
    2041             :         }
    2042             : 
    2043           2 :         options = tree->session->transport->options;
    2044             : 
    2045           2 :         smb2_deltree(tree, __func__);
    2046           2 :         status = torture_smb2_testdir(tree, __func__, &_h);
    2047           2 :         torture_assert_ntstatus_ok_goto(tctx, status, ret, done,
    2048             :                                         "torture_smb2_testdir failed\n");
    2049           2 :         smb2_util_close(tree, _h);
    2050             : 
    2051             :         /* Choose a random name in case the state is left a little funky. */
    2052           2 :         snprintf(fname, 256, "%s\\durable_v2_setinfo%s.dat",
    2053             :                  __func__, generate_random_str(tctx, 8));
    2054             : 
    2055           2 :         smb2_util_unlink(tree, fname);
    2056             : 
    2057           2 :         lease_key = random();
    2058           2 :         smb2_lease_v2_create(&io, &ls, false /* dir */, fname,
    2059             :                              lease_key, 0, /* parent lease key */
    2060             :                              smb2_util_lease_state("RWH"), 0 /* lease epoch */);
    2061           2 :         io.in.durable_open = false;
    2062           2 :         io.in.durable_open_v2 = true;
    2063           2 :         io.in.persistent_open = false;
    2064           2 :         io.in.create_guid = create_guid;
    2065           2 :         io.in.timeout = UINT32_MAX;
    2066             : 
    2067           2 :         status = smb2_create(tree, mem_ctx, &io);
    2068           2 :         CHECK_STATUS(status, NT_STATUS_OK);
    2069           2 :         _h = io.out.file.handle;
    2070           2 :         h = &_h;
    2071           2 :         CHECK_CREATED(&io, CREATED, FILE_ATTRIBUTE_ARCHIVE);
    2072           2 :         CHECK_VAL(io.out.durable_open, false);
    2073           2 :         CHECK_VAL(io.out.durable_open_v2, true);
    2074           2 :         CHECK_VAL(io.out.persistent_open, false);
    2075           2 :         CHECK_VAL(io.out.timeout, 300*1000);
    2076           2 :         CHECK_VAL(io.out.oplock_level, SMB2_OPLOCK_LEVEL_LEASE);
    2077           2 :         CHECK_VAL(io.out.lease_response_v2.lease_key.data[0], lease_key);
    2078           2 :         CHECK_VAL(io.out.lease_response_v2.lease_key.data[1], ~lease_key);
    2079             : 
    2080             :         /*
    2081             :          * Set EOF to 0x100000.
    2082             :          * Mimics an Apple client test, but most importantly
    2083             :          * causes the mtime timestamp on disk to be updated.
    2084             :          */
    2085           2 :         ZERO_STRUCT(si);
    2086           2 :         si.generic.level = SMB_SFILEINFO_END_OF_FILE_INFORMATION;
    2087           2 :         si.generic.in.file.handle = io.out.file.handle;
    2088           2 :         si.end_of_file_info.in.size = 0x100000;
    2089           2 :         status = smb2_setinfo_file(tree, &si);
    2090           2 :         CHECK_STATUS(status, NT_STATUS_OK);
    2091             : 
    2092             :         /* disconnect, reconnect and then do durable reopen */
    2093           2 :         TALLOC_FREE(tree);
    2094             : 
    2095           2 :         if (!torture_smb2_connection_ext(tctx, 0, &options, &tree)) {
    2096           0 :                 torture_warning(tctx, "couldn't reconnect, bailing\n");
    2097           0 :                 ret = false;
    2098           0 :                 goto done;
    2099             :         }
    2100             : 
    2101             :         /*
    2102             :          * Now for a succeeding reconnect:
    2103             :          */
    2104             : 
    2105           2 :         ZERO_STRUCT(io);
    2106           2 :         io.in.fname = fname;
    2107           2 :         io.in.durable_open_v2 = false;
    2108           2 :         io.in.durable_handle_v2 = h;
    2109           2 :         io.in.create_guid = create_guid;
    2110           2 :         io.in.lease_request_v2 = &ls;
    2111           2 :         io.in.oplock_level = SMB2_OPLOCK_LEVEL_LEASE;
    2112             : 
    2113             :         /* the requested lease state is irrelevant */
    2114           2 :         ls.lease_state = smb2_util_lease_state("");
    2115             : 
    2116           2 :         h = NULL;
    2117             : 
    2118           2 :         status = smb2_create(tree, mem_ctx, &io);
    2119           2 :         CHECK_STATUS(status, NT_STATUS_OK);
    2120             : 
    2121           2 :         CHECK_VAL(io.out.create_action, NTCREATEX_ACTION_EXISTED);
    2122           2 :         CHECK_VAL(io.out.size, 0x100000);                               \
    2123           2 :         CHECK_VAL(io.out.durable_open, false);
    2124           2 :         CHECK_VAL(io.out.durable_open_v2, false); /* no dh2q response blob */
    2125           2 :         CHECK_VAL(io.out.persistent_open, false);
    2126           2 :         CHECK_VAL(io.out.oplock_level, SMB2_OPLOCK_LEVEL_LEASE);
    2127           2 :         CHECK_VAL(io.out.lease_response_v2.lease_key.data[0], lease_key);
    2128           2 :         CHECK_VAL(io.out.lease_response_v2.lease_key.data[1], ~lease_key);
    2129           2 :         CHECK_VAL(io.out.lease_response_v2.lease_state,
    2130             :                   smb2_util_lease_state("RWH"));
    2131           2 :         CHECK_VAL(io.out.lease_response_v2.lease_flags, 0);
    2132           2 :         CHECK_VAL(io.out.lease_response_v2.lease_duration, 0);
    2133           2 :         _h = io.out.file.handle;
    2134           2 :         h = &_h;
    2135             : 
    2136           2 : done:
    2137             : 
    2138           2 :         if (h != NULL) {
    2139           2 :                 smb2_util_close(tree, *h);
    2140             :         }
    2141             : 
    2142           2 :         smb2_util_unlink(tree, fname);
    2143           2 :         smb2_deltree(tree, __func__);
    2144             : 
    2145           2 :         talloc_free(tree);
    2146             : 
    2147           2 :         talloc_free(mem_ctx);
    2148             : 
    2149           2 :         return ret;
    2150             : }
    2151             : 
    2152        2358 : struct torture_suite *torture_smb2_durable_v2_open_init(TALLOC_CTX *ctx)
    2153             : {
    2154         125 :         struct torture_suite *suite =
    2155        2358 :             torture_suite_create(ctx, "durable-v2-open");
    2156             : 
    2157        2358 :         torture_suite_add_1smb2_test(suite, "create-blob", test_durable_v2_open_create_blob);
    2158        2358 :         torture_suite_add_1smb2_test(suite, "open-oplock", test_durable_v2_open_oplock);
    2159        2358 :         torture_suite_add_1smb2_test(suite, "open-lease", test_durable_v2_open_lease);
    2160        2358 :         torture_suite_add_1smb2_test(suite, "reopen1", test_durable_v2_open_reopen1);
    2161        2358 :         torture_suite_add_1smb2_test(suite, "reopen1a", test_durable_v2_open_reopen1a);
    2162        2358 :         torture_suite_add_1smb2_test(suite, "reopen1a-lease", test_durable_v2_open_reopen1a_lease);
    2163        2358 :         torture_suite_add_1smb2_test(suite, "reopen2", test_durable_v2_open_reopen2);
    2164        2358 :         torture_suite_add_1smb2_test(suite, "reopen2b", test_durable_v2_open_reopen2b);
    2165        2358 :         torture_suite_add_1smb2_test(suite, "reopen2c", test_durable_v2_open_reopen2c);
    2166        2358 :         torture_suite_add_1smb2_test(suite, "reopen2-lease", test_durable_v2_open_reopen2_lease);
    2167        2358 :         torture_suite_add_1smb2_test(suite, "reopen2-lease-v2", test_durable_v2_open_reopen2_lease_v2);
    2168        2358 :         torture_suite_add_1smb2_test(suite, "durable-v2-setinfo", test_durable_v2_setinfo);
    2169        2358 :         torture_suite_add_2smb2_test(suite, "app-instance", test_durable_v2_open_app_instance);
    2170        2358 :         torture_suite_add_1smb2_test(suite, "persistent-open-oplock", test_persistent_open_oplock);
    2171        2358 :         torture_suite_add_1smb2_test(suite, "persistent-open-lease", test_persistent_open_lease);
    2172             : 
    2173        2358 :         suite->description = talloc_strdup(suite, "SMB2-DURABLE-V2-OPEN tests");
    2174             : 
    2175        2358 :         return suite;
    2176             : }
    2177             : 
    2178             : /**
    2179             :  * basic test for doing a durable open
    2180             :  * tcp disconnect, reconnect, do a durable reopen (succeeds)
    2181             :  */
    2182           6 : static bool test_durable_v2_reconnect_delay(struct torture_context *tctx,
    2183             :                                             struct smb2_tree *tree,
    2184             :                                             struct smb2_tree *tree2)
    2185             : {
    2186           0 :         NTSTATUS status;
    2187           6 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
    2188           0 :         char fname[256];
    2189           0 :         struct smb2_handle _h;
    2190           6 :         struct smb2_handle *h = NULL;
    2191           0 :         struct smb2_create io;
    2192           6 :         struct GUID create_guid = GUID_random();
    2193           0 :         struct smbcli_options options;
    2194           0 :         uint64_t previous_session_id;
    2195           6 :         uint8_t b = 0;
    2196           6 :         bool ret = true;
    2197           0 :         bool ok;
    2198             : 
    2199           6 :         options = tree->session->transport->options;
    2200           6 :         previous_session_id = smb2cli_session_current_id(tree->session->smbXcli);
    2201             : 
    2202             :         /* Choose a random name in case the state is left a little funky. */
    2203           6 :         snprintf(fname,
    2204             :                  sizeof(fname),
    2205             :                  "durable_v2_reconnect_delay_%s.dat",
    2206             :                  generate_random_str(tctx, 8));
    2207             : 
    2208           6 :         smb2_util_unlink(tree, fname);
    2209             : 
    2210           6 :         smb2_oplock_create_share(&io, fname,
    2211             :                                  smb2_util_share_access(""),
    2212           6 :                                  smb2_util_oplock_level("b"));
    2213           6 :         io.in.durable_open = false;
    2214           6 :         io.in.durable_open_v2 = true;
    2215           6 :         io.in.persistent_open = false;
    2216           6 :         io.in.create_guid = create_guid;
    2217           6 :         io.in.timeout = 0;
    2218             : 
    2219           6 :         status = smb2_create(tree, mem_ctx, &io);
    2220           6 :         CHECK_STATUS(status, NT_STATUS_OK);
    2221             : 
    2222           6 :         _h = io.out.file.handle;
    2223           6 :         h = &_h;
    2224           6 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level("b"));
    2225           6 :         CHECK_VAL(io.out.durable_open_v2, true);
    2226             : 
    2227           6 :         status = smb2_util_write(tree, *h, &b, 0, 1);
    2228           6 :         CHECK_STATUS(status, NT_STATUS_OK);
    2229             : 
    2230             :         /* disconnect, leaving the durable open */
    2231           6 :         TALLOC_FREE(tree);
    2232           6 :         h = NULL;
    2233             : 
    2234           6 :         ok = torture_smb2_connection_ext(tctx, previous_session_id,
    2235             :                                          &options, &tree);
    2236           6 :         torture_assert_goto(tctx, ok, ret, done, "couldn't reconnect, bailing\n");
    2237             : 
    2238           6 :         ZERO_STRUCT(io);
    2239           6 :         io.in.fname = fname;
    2240           6 :         io.in.durable_open_v2 = false;
    2241           6 :         io.in.durable_handle_v2 = &_h;
    2242           6 :         io.in.create_guid = create_guid;
    2243             : 
    2244           6 :         status = smb2_create(tree, mem_ctx, &io);
    2245           6 :         CHECK_STATUS(status, NT_STATUS_OK);
    2246           6 :         CHECK_VAL(io.out.oplock_level, smb2_util_oplock_level("b"));
    2247           6 :         _h = io.out.file.handle;
    2248           6 :         h = &_h;
    2249             : 
    2250           6 : done:
    2251           6 :         if (h != NULL) {
    2252           6 :                 smb2_util_close(tree, *h);
    2253             :         }
    2254           6 :         TALLOC_FREE(tree);
    2255             : 
    2256           6 :         smb2_util_unlink(tree2, fname);
    2257             : 
    2258           6 :         TALLOC_FREE(tree2);
    2259             : 
    2260           6 :         talloc_free(mem_ctx);
    2261             : 
    2262           6 :         return ret;
    2263             : }
    2264             : 
    2265             : /**
    2266             :  * basic test for doing a durable open with 1msec cleanup time
    2267             :  * tcp disconnect, wait a bit, reconnect, do a durable reopen (fails)
    2268             :  */
    2269           6 : static bool test_durable_v2_reconnect_delay_msec(struct torture_context *tctx,
    2270             :                                                  struct smb2_tree *tree,
    2271             :                                                  struct smb2_tree *tree2)
    2272             : {
    2273           0 :         NTSTATUS status;
    2274           6 :         TALLOC_CTX *mem_ctx = talloc_new(tctx);
    2275           0 :         char fname[256];
    2276           0 :         struct smb2_handle _h;
    2277           6 :         struct smb2_handle *h = NULL;
    2278           0 :         struct smb2_create io;
    2279           0 :         struct smb2_lease ls;
    2280           6 :         struct GUID create_guid = GUID_random();
    2281           0 :         struct smbcli_options options;
    2282           0 :         uint64_t previous_session_id;
    2283           6 :         uint8_t b = 0;
    2284           6 :         bool ret = true;
    2285           0 :         bool ok;
    2286             : 
    2287           6 :         options = tree->session->transport->options;
    2288           6 :         previous_session_id = smb2cli_session_current_id(tree->session->smbXcli);
    2289             : 
    2290             :         /* Choose a random name in case the state is left a little funky. */
    2291           6 :         snprintf(fname,
    2292             :                  sizeof(fname),
    2293             :                  "durable_v2_reconnect_delay_%s.dat",
    2294             :                  generate_random_str(tctx, 8));
    2295             : 
    2296           6 :         smb2_util_unlink(tree, fname);
    2297             : 
    2298           6 :         smb2_lease_create(
    2299             :                 &io,
    2300             :                 &ls,
    2301             :                 false /* dir */,
    2302             :                 fname,
    2303             :                 generate_random_u64(),
    2304             :                 smb2_util_lease_state("RWH"));
    2305           6 :         io.in.durable_open = false;
    2306           6 :         io.in.durable_open_v2 = true;
    2307           6 :         io.in.persistent_open = false;
    2308           6 :         io.in.create_guid = create_guid;
    2309           6 :         io.in.timeout = 1;
    2310             : 
    2311           6 :         status = smb2_create(tree, mem_ctx, &io);
    2312           6 :         CHECK_STATUS(status, NT_STATUS_OK);
    2313             : 
    2314           6 :         _h = io.out.file.handle;
    2315           6 :         h = &_h;
    2316           6 :         CHECK_VAL(io.out.oplock_level, SMB2_OPLOCK_LEVEL_LEASE);
    2317           6 :         CHECK_VAL(io.out.durable_open_v2, true);
    2318             : 
    2319           6 :         status = smb2_util_write(tree, *h, &b, 0, 1);
    2320           6 :         CHECK_STATUS(status, NT_STATUS_OK);
    2321             : 
    2322             :         /* disconnect, leaving the durable open */
    2323           6 :         TALLOC_FREE(tree);
    2324           6 :         h = NULL;
    2325             : 
    2326           6 :         ok = torture_smb2_connection_ext(tctx, previous_session_id,
    2327             :                                          &options, &tree);
    2328           6 :         torture_assert_goto(tctx, ok, ret, done, "couldn't reconnect, bailing\n");
    2329             : 
    2330           6 :         sleep(10);
    2331             : 
    2332           6 :         ZERO_STRUCT(io);
    2333           6 :         io.in.fname = fname;
    2334           6 :         io.in.durable_open_v2 = false;
    2335           6 :         io.in.durable_handle_v2 = &_h;
    2336           6 :         io.in.create_guid = create_guid;
    2337             : 
    2338           6 :         status = smb2_create(tree, mem_ctx, &io);
    2339           6 :         CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
    2340           6 :         _h = io.out.file.handle;
    2341           6 :         h = &_h;
    2342             : 
    2343           6 : done:
    2344           6 :         if (h != NULL) {
    2345           6 :                 smb2_util_close(tree, *h);
    2346             :         }
    2347           6 :         TALLOC_FREE(tree);
    2348             : 
    2349           6 :         smb2_util_unlink(tree2, fname);
    2350             : 
    2351           6 :         TALLOC_FREE(tree2);
    2352             : 
    2353           6 :         talloc_free(mem_ctx);
    2354             : 
    2355           6 :         return ret;
    2356             : }
    2357             : 
    2358        2358 : struct torture_suite *torture_smb2_durable_v2_delay_init(TALLOC_CTX *ctx)
    2359             : {
    2360         125 :         struct torture_suite *suite =
    2361        2358 :             torture_suite_create(ctx, "durable-v2-delay");
    2362             : 
    2363        2358 :         torture_suite_add_2smb2_test(suite,
    2364             :                                      "durable_v2_reconnect_delay",
    2365             :                                      test_durable_v2_reconnect_delay);
    2366        2358 :         torture_suite_add_2smb2_test(suite,
    2367             :                                      "durable_v2_reconnect_delay_msec",
    2368             :                                      test_durable_v2_reconnect_delay_msec);
    2369             : 
    2370        2358 :         return suite;
    2371             : }

Generated by: LCOV version 1.14