LCOV - code coverage report
Current view: top level - source3/torture - torture.c (source / functions) Hit Total Coverage
Test: coverage report for master 70ed9daf Lines: 4429 8182 54.1 %
Date: 2024-01-11 09:59:51 Functions: 193 240 80.4 %

          Line data    Source code
       1             : /* 
       2             :    Unix SMB/CIFS implementation.
       3             :    SMB torture tester
       4             :    Copyright (C) Andrew Tridgell 1997-1998
       5             :    Copyright (C) Jeremy Allison 2009
       6             : 
       7             :    This program is free software; you can redistribute it and/or modify
       8             :    it under the terms of the GNU General Public License as published by
       9             :    the Free Software Foundation; either version 3 of the License, or
      10             :    (at your option) any later version.
      11             : 
      12             :    This program is distributed in the hope that it will be useful,
      13             :    but WITHOUT ANY WARRANTY; without even the implied warranty of
      14             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
      15             :    GNU General Public License for more details.
      16             : 
      17             :    You should have received a copy of the GNU General Public License
      18             :    along with this program.  If not, see <http://www.gnu.org/licenses/>.
      19             : */
      20             : 
      21             : #include "includes.h"
      22             : #include "system/shmem.h"
      23             : #include "libsmb/namequery.h"
      24             : #include "wbc_async.h"
      25             : #include "torture/proto.h"
      26             : #include "libcli/security/security.h"
      27             : #include "tldap.h"
      28             : #include "tldap_util.h"
      29             : #include "tldap_gensec_bind.h"
      30             : #include "../librpc/gen_ndr/svcctl.h"
      31             : #include "../lib/util/memcache.h"
      32             : #include "nsswitch/winbind_client.h"
      33             : #include "dbwrap/dbwrap.h"
      34             : #include "dbwrap/dbwrap_open.h"
      35             : #include "dbwrap/dbwrap_rbt.h"
      36             : #include "async_smb.h"
      37             : #include "libsmb/libsmb.h"
      38             : #include "libsmb/clirap.h"
      39             : #include "trans2.h"
      40             : #include "libsmb/nmblib.h"
      41             : #include "../lib/util/tevent_ntstatus.h"
      42             : #include "util_tdb.h"
      43             : #include "../libcli/smb/read_smb.h"
      44             : #include "../libcli/smb/smbXcli_base.h"
      45             : #include "lib/util/sys_rw_data.h"
      46             : #include "lib/util/base64.h"
      47             : #include "lib/util/time.h"
      48             : #include "lib/gencache.h"
      49             : #include "lib/util/sys_rw.h"
      50             : #include "lib/util/asn1.h"
      51             : #include "lib/param/param.h"
      52             : #include "auth/gensec/gensec.h"
      53             : #include "lib/util/string_wrappers.h"
      54             : #include "source3/lib/substitute.h"
      55             : 
      56             : #include <gnutls/gnutls.h>
      57             : #include <gnutls/crypto.h>
      58             : 
      59             : extern char *optarg;
      60             : extern int optind;
      61             : 
      62             : fstring host, workgroup, share, password, username, myname;
      63             : struct cli_credentials *torture_creds;
      64             : static const char *sockops="TCP_NODELAY";
      65             : int torture_nprocs=1;
      66             : static int port_to_use=0;
      67             : int torture_numops=100;
      68             : int torture_blocksize=1024*1024;
      69             : static int procnum; /* records process count number when forking */
      70             : static struct cli_state *current_cli;
      71             : static fstring randomfname;
      72             : static bool use_oplocks;
      73             : static bool use_level_II_oplocks;
      74             : static const char *client_txt = "client_oplocks.txt";
      75             : static bool disable_spnego;
      76             : static bool use_kerberos;
      77             : static bool force_dos_errors;
      78             : static fstring multishare_conn_fname;
      79             : static bool use_multishare_conn = False;
      80             : static bool do_encrypt;
      81             : static const char *local_path = NULL;
      82             : static enum smb_signing_setting signing_state = SMB_SIGNING_DEFAULT;
      83             : char *test_filename;
      84             : 
      85             : bool torture_showall = False;
      86             : 
      87             : static double create_procs(bool (*fn)(int), bool *result);
      88             : 
      89             : /********************************************************************
      90             :  Ensure a connection is encrypted.
      91             : ********************************************************************/
      92             : 
      93         200 : static bool force_cli_encryption(struct cli_state *c,
      94             :                         const char *sharename)
      95             : {
      96           0 :         uint16_t major, minor;
      97           0 :         uint32_t caplow, caphigh;
      98           0 :         NTSTATUS status;
      99             : 
     100         200 :         if (!SERVER_HAS_UNIX_CIFS(c)) {
     101           0 :                 d_printf("Encryption required and "
     102             :                         "server that doesn't support "
     103             :                         "UNIX extensions - failing connect\n");
     104           0 :                         return false;
     105             :         }
     106             : 
     107         200 :         status = cli_unix_extensions_version(c, &major, &minor, &caplow,
     108             :                                              &caphigh);
     109         200 :         if (!NT_STATUS_IS_OK(status)) {
     110           0 :                 d_printf("Encryption required and "
     111             :                         "can't get UNIX CIFS extensions "
     112             :                         "version from server: %s\n", nt_errstr(status));
     113           0 :                 return false;
     114             :         }
     115             : 
     116         200 :         if (!(caplow & CIFS_UNIX_TRANSPORT_ENCRYPTION_CAP)) {
     117           0 :                 d_printf("Encryption required and "
     118             :                         "share %s doesn't support "
     119             :                         "encryption.\n", sharename);
     120           0 :                 return false;
     121             :         }
     122             : 
     123         200 :         status = cli_smb1_setup_encryption(c, torture_creds);
     124         200 :         if (!NT_STATUS_IS_OK(status)) {
     125           0 :                 d_printf("Encryption required and "
     126             :                         "setup failed with error %s.\n",
     127             :                         nt_errstr(status));
     128           0 :                 return false;
     129             :         }
     130             : 
     131         200 :         return true;
     132             : }
     133             : 
     134             : 
     135         100 : static struct cli_state *open_nbt_connection(void)
     136             : {
     137           0 :         struct cli_state *c;
     138           0 :         NTSTATUS status;
     139         100 :         int flags = 0;
     140             : 
     141         100 :         if (disable_spnego) {
     142           0 :                 flags |= CLI_FULL_CONNECTION_DONT_SPNEGO;
     143             :         }
     144             : 
     145         100 :         if (use_oplocks) {
     146           0 :                 flags |= CLI_FULL_CONNECTION_OPLOCKS;
     147             :         }
     148             : 
     149         100 :         if (use_level_II_oplocks) {
     150           0 :                 flags |= CLI_FULL_CONNECTION_LEVEL_II_OPLOCKS;
     151             :         }
     152             : 
     153         100 :         if (force_dos_errors) {
     154           0 :                 flags |= CLI_FULL_CONNECTION_FORCE_DOS_ERRORS;
     155             :         }
     156             : 
     157         100 :         status = cli_connect_nb(host, NULL, port_to_use, 0x20, myname,
     158             :                                 signing_state, flags, &c);
     159         100 :         if (!NT_STATUS_IS_OK(status)) {
     160           0 :                 printf("Failed to connect with %s. Error %s\n", host, nt_errstr(status) );
     161           0 :                 return NULL;
     162             :         }
     163             : 
     164         100 :         cli_set_timeout(c, 120000); /* set a really long timeout (2 minutes) */
     165             : 
     166         100 :         return c;
     167             : }
     168             : 
     169             : /****************************************************************************
     170             :  Send a corrupt session request. See rfc1002.txt 4.3 and 4.3.2.
     171             : ****************************************************************************/
     172             : 
     173           4 : static bool cli_bad_session_request(int fd,
     174             :                          struct nmb_name *calling, struct nmb_name *called)
     175             : {
     176           0 :         TALLOC_CTX *frame;
     177           0 :         uint8_t len_buf[4];
     178           0 :         struct iovec iov[3];
     179           0 :         ssize_t len;
     180           0 :         uint8_t *inbuf;
     181           0 :         int err;
     182           4 :         bool ret = false;
     183           0 :         uint8_t message_type;
     184           0 :         uint8_t error;
     185           0 :         struct tevent_context *ev;
     186           0 :         struct tevent_req *req;
     187             : 
     188           4 :         frame = talloc_stackframe();
     189             : 
     190           4 :         iov[0].iov_base = len_buf;
     191           4 :         iov[0].iov_len  = sizeof(len_buf);
     192             : 
     193             :         /* put in the destination name */
     194             : 
     195           4 :         iov[1].iov_base = name_mangle(talloc_tos(), called->name,
     196           4 :                                       called->name_type);
     197           4 :         if (iov[1].iov_base == NULL) {
     198           0 :                 goto fail;
     199             :         }
     200           4 :         iov[1].iov_len = name_len((unsigned char *)iov[1].iov_base,
     201           4 :                                   talloc_get_size(iov[1].iov_base));
     202             : 
     203             :         /* and my name */
     204             : 
     205           4 :         iov[2].iov_base = name_mangle(talloc_tos(), calling->name,
     206           4 :                                       calling->name_type);
     207           4 :         if (iov[2].iov_base == NULL) {
     208           0 :                 goto fail;
     209             :         }
     210           4 :         iov[2].iov_len = name_len((unsigned char *)iov[2].iov_base,
     211           4 :                                   talloc_get_size(iov[2].iov_base));
     212             : 
     213             :         /* Deliberately corrupt the name len (first byte) */
     214           4 :         *((uint8_t *)iov[2].iov_base) = 100;
     215             : 
     216             :         /* send a session request (RFC 1002) */
     217             :         /* setup the packet length
     218             :          * Remove four bytes from the length count, since the length
     219             :          * field in the NBT Session Service header counts the number
     220             :          * of bytes which follow.  The cli_send_smb() function knows
     221             :          * about this and accounts for those four bytes.
     222             :          * CRH.
     223             :          */
     224             : 
     225           4 :         _smb_setlen(len_buf, iov[1].iov_len + iov[2].iov_len);
     226           4 :         SCVAL(len_buf,0,0x81);
     227             : 
     228           4 :         len = write_data_iov(fd, iov, 3);
     229           4 :         if (len == -1) {
     230           0 :                 goto fail;
     231             :         }
     232             : 
     233           4 :         ev = samba_tevent_context_init(frame);
     234           4 :         if (ev == NULL) {
     235           0 :                 goto fail;
     236             :         }
     237           4 :         req = read_smb_send(frame, ev, fd);
     238           4 :         if (req == NULL) {
     239           0 :                 goto fail;
     240             :         }
     241           4 :         if (!tevent_req_poll(req, ev)) {
     242           0 :                 goto fail;
     243             :         }
     244           4 :         len = read_smb_recv(req, talloc_tos(), &inbuf, &err);
     245           4 :         if (len == -1) {
     246           0 :                 errno = err;
     247           0 :                 goto fail;
     248             :         }
     249           4 :         TALLOC_FREE(ev);
     250             : 
     251           4 :         message_type = CVAL(inbuf, 0);
     252           4 :         if (message_type != 0x83) {
     253           0 :                 d_fprintf(stderr, "Expected msg type 0x83, got 0x%2.2x\n",
     254             :                           message_type);
     255           0 :                 goto fail;
     256             :         }
     257             : 
     258           4 :         if (smb_len(inbuf) != 1) {
     259           0 :                 d_fprintf(stderr, "Expected smb_len 1, got %d\n",
     260           0 :                           (int)smb_len(inbuf));
     261           0 :                 goto fail;
     262             :         }
     263             : 
     264           4 :         error = CVAL(inbuf, 4);
     265           4 :         if (error !=  0x82) {
     266           0 :                 d_fprintf(stderr, "Expected error 0x82, got %d\n",
     267             :                           (int)error);
     268           0 :                 goto fail;
     269             :         }
     270             : 
     271           4 :         ret = true;
     272           4 : fail:
     273           4 :         TALLOC_FREE(frame);
     274           4 :         return ret;
     275             : }
     276             : 
     277             : /* Insert a NULL at the first separator of the given path and return a pointer
     278             :  * to the remainder of the string.
     279             :  */
     280             : static char *
     281           0 : terminate_path_at_separator(char * path)
     282             : {
     283           0 :         char * p;
     284             : 
     285           0 :         if (!path) {
     286           0 :                 return NULL;
     287             :         }
     288             : 
     289           0 :         if ((p = strchr_m(path, '/'))) {
     290           0 :                 *p = '\0';
     291           0 :                 return p + 1;
     292             :         }
     293             : 
     294           0 :         if ((p = strchr_m(path, '\\'))) {
     295           0 :                 *p = '\0';
     296           0 :                 return p + 1;
     297             :         }
     298             : 
     299             :         /* No separator. */
     300           0 :         return NULL;
     301             : }
     302             : 
     303             : /*
     304             :   parse a //server/share type UNC name
     305             : */
     306           0 : bool smbcli_parse_unc(const char *unc_name, TALLOC_CTX *mem_ctx,
     307             :                       char **hostname, char **sharename)
     308             : {
     309           0 :         char *p;
     310             : 
     311           0 :         *hostname = *sharename = NULL;
     312             : 
     313           0 :         if (strncmp(unc_name, "\\\\", 2) &&
     314           0 :             strncmp(unc_name, "//", 2)) {
     315           0 :                 return False;
     316             :         }
     317             : 
     318           0 :         *hostname = talloc_strdup(mem_ctx, &unc_name[2]);
     319           0 :         p = terminate_path_at_separator(*hostname);
     320             : 
     321           0 :         if (p && *p) {
     322           0 :                 *sharename = talloc_strdup(mem_ctx, p);
     323           0 :                 terminate_path_at_separator(*sharename);
     324             :         }
     325             : 
     326           0 :         if (*hostname && *sharename) {
     327           0 :                 return True;
     328             :         }
     329             : 
     330           0 :         TALLOC_FREE(*hostname);
     331           0 :         TALLOC_FREE(*sharename);
     332           0 :         return False;
     333             : }
     334             : 
     335         497 : static bool torture_open_connection_share(struct cli_state **c,
     336             :                                    const char *hostname, 
     337             :                                    const char *sharename,
     338             :                                    int flags)
     339             : {
     340           0 :         NTSTATUS status;
     341             : 
     342         497 :         status = cli_full_connection_creds(c,
     343             :                                            myname,
     344             :                                            hostname,
     345             :                                            NULL, /* dest_ss */
     346             :                                            port_to_use,
     347             :                                            sharename,
     348             :                                            "?????",
     349             :                                            torture_creds,
     350             :                                            flags);
     351         497 :         if (!NT_STATUS_IS_OK(status)) {
     352           2 :                 printf("failed to open share connection: //%s/%s port:%d - %s\n",
     353             :                         hostname, sharename, port_to_use, nt_errstr(status));
     354           2 :                 return False;
     355             :         }
     356             : 
     357         495 :         cli_set_timeout(*c, 120000); /* set a really long timeout (2 minutes) */
     358             : 
     359         495 :         if (do_encrypt) {
     360         198 :                 return force_cli_encryption(*c,
     361             :                                         sharename);
     362             :         }
     363         297 :         return True;
     364             : }
     365             : 
     366         497 : bool torture_open_connection_flags(struct cli_state **c, int conn_index, int flags)
     367             : {
     368         497 :         char **unc_list = NULL;
     369         497 :         int num_unc_names = 0;
     370           0 :         bool result;
     371             : 
     372         497 :         if (use_multishare_conn==True) {
     373           0 :                 char *h, *s;
     374           0 :                 unc_list = file_lines_load(multishare_conn_fname, &num_unc_names, 0, NULL);
     375           0 :                 if (!unc_list || num_unc_names <= 0) {
     376           0 :                         printf("Failed to load unc names list from '%s'\n", multishare_conn_fname);
     377           0 :                         exit(1);
     378             :                 }
     379             : 
     380           0 :                 if (!smbcli_parse_unc(unc_list[conn_index % num_unc_names],
     381             :                                       NULL, &h, &s)) {
     382           0 :                         printf("Failed to parse UNC name %s\n",
     383           0 :                                unc_list[conn_index % num_unc_names]);
     384           0 :                         TALLOC_FREE(unc_list);
     385           0 :                         exit(1);
     386             :                 }
     387             : 
     388           0 :                 result = torture_open_connection_share(c, h, s, flags);
     389             : 
     390             :                 /* h, s were copied earlier */
     391           0 :                 TALLOC_FREE(unc_list);
     392           0 :                 return result;
     393             :         }
     394             : 
     395         497 :         return torture_open_connection_share(c, host, share, flags);
     396             : }
     397             : 
     398         485 : bool torture_open_connection(struct cli_state **c, int conn_index)
     399             : {
     400         485 :         int flags = CLI_FULL_CONNECTION_FORCE_SMB1;
     401             : 
     402         485 :         if (use_oplocks) {
     403           8 :                 flags |= CLI_FULL_CONNECTION_OPLOCKS;
     404             :         }
     405         485 :         if (use_level_II_oplocks) {
     406           8 :                 flags |= CLI_FULL_CONNECTION_LEVEL_II_OPLOCKS;
     407             :         }
     408             : 
     409         485 :         return torture_open_connection_flags(c, conn_index, flags);
     410             : }
     411             : 
     412          90 : bool torture_init_connection(struct cli_state **pcli)
     413             : {
     414           0 :         struct cli_state *cli;
     415             : 
     416          90 :         cli = open_nbt_connection();
     417          90 :         if (cli == NULL) {
     418           0 :                 return false;
     419             :         }
     420             : 
     421          90 :         *pcli = cli;
     422          90 :         return true;
     423             : }
     424             : 
     425           5 : bool torture_cli_session_setup2(struct cli_state *cli, uint16_t *new_vuid)
     426             : {
     427           5 :         uint16_t old_vuid = cli_state_get_uid(cli);
     428           0 :         NTSTATUS status;
     429           0 :         bool ret;
     430             : 
     431           5 :         cli_state_set_uid(cli, 0);
     432           5 :         status = cli_session_setup_creds(cli, torture_creds);
     433           5 :         ret = NT_STATUS_IS_OK(status);
     434           5 :         *new_vuid = cli_state_get_uid(cli);
     435           5 :         cli_state_set_uid(cli, old_vuid);
     436           5 :         return ret;
     437             : }
     438             : 
     439             : 
     440         421 : bool torture_close_connection(struct cli_state *c)
     441             : {
     442         421 :         bool ret = True;
     443           0 :         NTSTATUS status;
     444             : 
     445         421 :         status = cli_tdis(c);
     446         421 :         if (!NT_STATUS_IS_OK(status)) {
     447           5 :                 printf("tdis failed (%s)\n", nt_errstr(status));
     448           5 :                 ret = False;
     449             :         }
     450             : 
     451         421 :         cli_shutdown(c);
     452             : 
     453         421 :         return ret;
     454             : }
     455             : 
     456          48 : void torture_conn_set_sockopt(struct cli_state *cli)
     457             : {
     458          48 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
     459          48 : }
     460             : 
     461          58 : static NTSTATUS torture_delete_fn(struct file_info *finfo,
     462             :                                   const char *pattern,
     463             :                                   void *state)
     464             : {
     465           0 :         NTSTATUS status;
     466          58 :         char *filename = NULL;
     467          58 :         char *dirname = NULL;
     468          58 :         char *p = NULL;
     469          58 :         TALLOC_CTX *frame = talloc_stackframe();
     470          58 :         struct cli_state *cli = (struct cli_state *)state;
     471             : 
     472          58 :         if (ISDOT(finfo->name) || ISDOTDOT(finfo->name)) {
     473          36 :                 TALLOC_FREE(frame);
     474          36 :                 return NT_STATUS_OK;
     475             :         }
     476             : 
     477          22 :         dirname = talloc_strdup(frame, pattern);
     478          22 :         if (dirname == NULL) {
     479           0 :                 TALLOC_FREE(frame);
     480           0 :                 return NT_STATUS_NO_MEMORY;
     481             :         }
     482          22 :         p = strrchr_m(dirname, '\\');
     483          22 :         if (p != NULL) {
     484             :                 /* Remove the terminating '\' */
     485          22 :                 *p = '\0';
     486             :         }
     487          22 :         if (dirname[0] != '\0') {
     488          22 :                 filename = talloc_asprintf(frame,
     489             :                                            "%s\\%s",
     490             :                                            dirname,
     491             :                                            finfo->name);
     492             :         } else {
     493           0 :                 filename = talloc_asprintf(frame,
     494             :                                            "%s",
     495             :                                            finfo->name);
     496             :         }
     497          22 :         if (filename == NULL) {
     498           0 :                 TALLOC_FREE(frame);
     499           0 :                 return NT_STATUS_NO_MEMORY;
     500             :         }
     501          22 :         if (finfo->attr & FILE_ATTRIBUTE_DIRECTORY) {
     502           5 :                 char *subdirname = talloc_asprintf(frame,
     503             :                                                    "%s\\*",
     504             :                                                    filename);
     505           5 :                 if (subdirname == NULL) {
     506           0 :                         TALLOC_FREE(frame);
     507           0 :                         return NT_STATUS_NO_MEMORY;
     508             :                 }
     509           5 :                 status = cli_list(cli,
     510             :                                   subdirname,
     511             :                                   FILE_ATTRIBUTE_DIRECTORY |
     512             :                                           FILE_ATTRIBUTE_HIDDEN |
     513             :                                           FILE_ATTRIBUTE_SYSTEM,
     514             :                                   torture_delete_fn,
     515             :                                   cli);
     516           5 :                 if (!NT_STATUS_IS_OK(status)) {
     517           0 :                         printf("torture_delete_fn: cli_list "
     518             :                                 "of %s failed (%s)\n",
     519             :                                 subdirname,
     520             :                                 nt_errstr(status));
     521           0 :                         TALLOC_FREE(frame);
     522           0 :                         return status;
     523             :                 }
     524           5 :                 status = cli_rmdir(cli, filename);
     525             :         } else {
     526          17 :                 status = cli_unlink(cli,
     527             :                                     filename,
     528             :                                     FILE_ATTRIBUTE_SYSTEM |
     529             :                                         FILE_ATTRIBUTE_HIDDEN);
     530             :         }
     531          22 :         if (!NT_STATUS_IS_OK(status)) {
     532           0 :                 if (finfo->attr & FILE_ATTRIBUTE_DIRECTORY) {
     533           0 :                         printf("torture_delete_fn: cli_rmdir"
     534             :                                 " of %s failed (%s)\n",
     535             :                                 filename,
     536             :                                 nt_errstr(status));
     537             :                 } else {
     538           0 :                         printf("torture_delete_fn: cli_unlink"
     539             :                                 " of %s failed (%s)\n",
     540             :                                 filename,
     541             :                                 nt_errstr(status));
     542             :                 }
     543             :         }
     544          22 :         TALLOC_FREE(frame);
     545          22 :         return status;
     546             : }
     547             : 
     548          30 : void torture_deltree(struct cli_state *cli, const char *dname)
     549             : {
     550          30 :         char *mask = NULL;
     551           0 :         NTSTATUS status;
     552             : 
     553             :         /* It might be a file */
     554          30 :         (void)cli_unlink(cli,
     555             :                          dname,
     556             :                          FILE_ATTRIBUTE_SYSTEM |
     557             :                                 FILE_ATTRIBUTE_HIDDEN);
     558             : 
     559          30 :         mask = talloc_asprintf(cli,
     560             :                                "%s\\*",
     561             :                                dname);
     562          30 :         if (mask == NULL) {
     563           0 :                 printf("torture_deltree: talloc_asprintf failed\n");
     564           0 :                 return;
     565             :         }
     566             : 
     567          30 :         status = cli_list(cli,
     568             :                         mask,
     569             :                         FILE_ATTRIBUTE_DIRECTORY |
     570             :                                 FILE_ATTRIBUTE_HIDDEN|
     571             :                                 FILE_ATTRIBUTE_SYSTEM,
     572             :                         torture_delete_fn,
     573             :                         cli);
     574          30 :         if (!NT_STATUS_IS_OK(status)) {
     575          17 :                 printf("torture_deltree: cli_list of %s failed (%s)\n",
     576             :                         mask,
     577             :                         nt_errstr(status));
     578             :         }
     579          30 :         TALLOC_FREE(mask);
     580          30 :         status = cli_rmdir(cli, dname);
     581          30 :         if (!NT_STATUS_IS_OK(status)) {
     582          17 :                 printf("torture_deltree: cli_rmdir of %s failed (%s)\n",
     583             :                         dname,
     584             :                         nt_errstr(status));
     585             :         }
     586             : }
     587             : 
     588             : /* check if the server produced the expected dos or nt error code */
     589          53 : static bool check_both_error(int line, NTSTATUS status,
     590             :                              uint8_t eclass, uint32_t ecode, NTSTATUS nterr)
     591             : {
     592          53 :         if (NT_STATUS_IS_DOS(status)) {
     593           0 :                 uint8_t cclass;
     594           0 :                 uint32_t num;
     595             : 
     596             :                 /* Check DOS error */
     597           0 :                 cclass = NT_STATUS_DOS_CLASS(status);
     598           0 :                 num = NT_STATUS_DOS_CODE(status);
     599             : 
     600           0 :                 if (eclass != cclass || ecode != num) {
     601           0 :                         printf("unexpected error code class=%d code=%d\n",
     602             :                                (int)cclass, (int)num);
     603           0 :                         printf(" expected %d/%d %s (line=%d)\n",
     604             :                                (int)eclass, (int)ecode, nt_errstr(nterr), line);
     605           0 :                         return false;
     606             :                 }
     607             :         } else {
     608             :                 /* Check NT error */
     609          53 :                 if (!NT_STATUS_EQUAL(nterr, status)) {
     610           0 :                         printf("unexpected error code %s\n",
     611             :                                 nt_errstr(status));
     612           0 :                         printf(" expected %s (line=%d)\n",
     613             :                                 nt_errstr(nterr), line);
     614           0 :                         return false;
     615             :                 }
     616             :         }
     617             : 
     618          53 :         return true;
     619             : }
     620             : 
     621             : 
     622             : /* check if the server produced the expected error code */
     623          38 : static bool check_error(int line, NTSTATUS status,
     624             :                         uint8_t eclass, uint32_t ecode, NTSTATUS nterr)
     625             : {
     626          38 :         if (NT_STATUS_IS_DOS(status)) {
     627           0 :                 uint8_t cclass;
     628           0 :                 uint32_t num;
     629             : 
     630             :                 /* Check DOS error */
     631             : 
     632           0 :                 cclass = NT_STATUS_DOS_CLASS(status);
     633           0 :                 num = NT_STATUS_DOS_CODE(status);
     634             : 
     635           0 :                 if (eclass != cclass || ecode != num) {
     636           0 :                         printf("unexpected error code class=%d code=%d\n", 
     637             :                                (int)cclass, (int)num);
     638           0 :                         printf(" expected %d/%d %s (line=%d)\n", 
     639             :                                (int)eclass, (int)ecode, nt_errstr(nterr),
     640             :                                line);
     641           0 :                         return False;
     642             :                 }
     643             : 
     644             :         } else {
     645             :                 /* Check NT error */
     646             : 
     647          38 :                 if (NT_STATUS_V(nterr) != NT_STATUS_V(status)) {
     648           1 :                         printf("unexpected error code %s\n",
     649             :                                nt_errstr(status));
     650           1 :                         printf(" expected %s (line=%d)\n", nt_errstr(nterr),
     651             :                                line);
     652           1 :                         return False;
     653             :                 }
     654             :         }
     655             : 
     656          37 :         return True;
     657             : }
     658             : 
     659           9 : NTSTATUS cli_qpathinfo1(struct cli_state *cli,
     660             :                         const char *fname,
     661             :                         time_t *change_time,
     662             :                         time_t *access_time,
     663             :                         time_t *write_time,
     664             :                         off_t *size,
     665             :                         uint32_t *pattr)
     666             : {
     667           9 :         int timezone = smb1cli_conn_server_time_zone(cli->conn);
     668           9 :         time_t (*date_fn)(const void *buf, int serverzone) = NULL;
     669           9 :         uint8_t *rdata = NULL;
     670           0 :         uint32_t num_rdata;
     671           0 :         NTSTATUS status;
     672             : 
     673           9 :         status = cli_qpathinfo(talloc_tos(),
     674             :                                cli,
     675             :                                fname,
     676             :                                SMB_INFO_STANDARD,
     677             :                                22,
     678             :                                CLI_BUFFER_SIZE,
     679             :                                &rdata,
     680             :                                &num_rdata);
     681           9 :         if (!NT_STATUS_IS_OK(status)) {
     682           4 :                 return status;
     683             :         }
     684           5 :         if (cli->win95) {
     685           0 :                 date_fn = make_unix_date;
     686             :         } else {
     687           5 :                 date_fn = make_unix_date2;
     688             :         }
     689             : 
     690           5 :         if (change_time) {
     691           5 :                 *change_time = date_fn(rdata + 0, timezone);
     692             :         }
     693           5 :         if (access_time) {
     694           5 :                 *access_time = date_fn(rdata + 4, timezone);
     695             :         }
     696           5 :         if (write_time) {
     697           5 :                 *write_time = date_fn(rdata + 8, timezone);
     698             :         }
     699           5 :         if (size) {
     700           5 :                 *size = PULL_LE_U32(rdata, 12);
     701             :         }
     702           5 :         if (pattr) {
     703           0 :                 *pattr = PULL_LE_U16(rdata, l1_attrFile);
     704             :         }
     705           5 :         return NT_STATUS_OK;
     706             : }
     707             : 
     708           0 : static bool wait_lock(struct cli_state *c, int fnum, uint32_t offset, uint32_t len)
     709             : {
     710           0 :         NTSTATUS status;
     711             : 
     712           0 :         status = cli_lock32(c, fnum, offset, len, -1, WRITE_LOCK);
     713             : 
     714           0 :         while (!NT_STATUS_IS_OK(status)) {
     715           0 :                 if (!check_both_error(__LINE__, status, ERRDOS,
     716           0 :                                       ERRlock, NT_STATUS_LOCK_NOT_GRANTED)) {
     717           0 :                         return false;
     718             :                 }
     719             : 
     720           0 :                 status = cli_lock32(c, fnum, offset, len, -1, WRITE_LOCK);
     721             :         }
     722             : 
     723           0 :         return true;
     724             : }
     725             : 
     726             : 
     727           0 : static bool rw_torture(struct cli_state *c)
     728             : {
     729           0 :         const char *lockfname = "\\torture.lck";
     730           0 :         fstring fname;
     731           0 :         uint16_t fnum;
     732           0 :         uint16_t fnum2;
     733           0 :         pid_t pid2, pid = getpid();
     734           0 :         int i, j;
     735           0 :         char buf[1024];
     736           0 :         bool correct = True;
     737           0 :         size_t nread = 0;
     738           0 :         NTSTATUS status;
     739             : 
     740           0 :         memset(buf, '\0', sizeof(buf));
     741             : 
     742           0 :         status = cli_openx(c, lockfname, O_RDWR | O_CREAT | O_EXCL, 
     743             :                          DENY_NONE, &fnum2);
     744           0 :         if (!NT_STATUS_IS_OK(status)) {
     745           0 :                 status = cli_openx(c, lockfname, O_RDWR, DENY_NONE, &fnum2);
     746             :         }
     747           0 :         if (!NT_STATUS_IS_OK(status)) {
     748           0 :                 printf("open of %s failed (%s)\n",
     749             :                        lockfname, nt_errstr(status));
     750           0 :                 return False;
     751             :         }
     752             : 
     753           0 :         for (i=0;i<torture_numops;i++) {
     754           0 :                 unsigned n = (unsigned)sys_random()%10;
     755             : 
     756           0 :                 if (i % 10 == 0) {
     757           0 :                         printf("%d\r", i); fflush(stdout);
     758             :                 }
     759           0 :                 slprintf(fname, sizeof(fstring) - 1, "\\torture.%u", n);
     760             : 
     761           0 :                 if (!wait_lock(c, fnum2, n*sizeof(int), sizeof(int))) {
     762           0 :                         return False;
     763             :                 }
     764             : 
     765           0 :                 status = cli_openx(c, fname, O_RDWR | O_CREAT | O_TRUNC,
     766             :                                   DENY_ALL, &fnum);
     767           0 :                 if (!NT_STATUS_IS_OK(status)) {
     768           0 :                         printf("open failed (%s)\n", nt_errstr(status));
     769           0 :                         correct = False;
     770           0 :                         break;
     771             :                 }
     772             : 
     773           0 :                 status = cli_writeall(c, fnum, 0, (uint8_t *)&pid, 0,
     774             :                                       sizeof(pid), NULL);
     775           0 :                 if (!NT_STATUS_IS_OK(status)) {
     776           0 :                         printf("write failed (%s)\n", nt_errstr(status));
     777           0 :                         correct = False;
     778             :                 }
     779             : 
     780           0 :                 for (j=0;j<50;j++) {
     781           0 :                         status = cli_writeall(c, fnum, 0, (uint8_t *)buf,
     782           0 :                                               sizeof(pid)+(j*sizeof(buf)),
     783             :                                               sizeof(buf), NULL);
     784           0 :                         if (!NT_STATUS_IS_OK(status)) {
     785           0 :                                 printf("write failed (%s)\n",
     786             :                                        nt_errstr(status));
     787           0 :                                 correct = False;
     788             :                         }
     789             :                 }
     790             : 
     791           0 :                 pid2 = 0;
     792             : 
     793           0 :                 status = cli_read(c, fnum, (char *)&pid2, 0, sizeof(pid),
     794             :                                   &nread);
     795           0 :                 if (!NT_STATUS_IS_OK(status)) {
     796           0 :                         printf("read failed (%s)\n", nt_errstr(status));
     797           0 :                         correct = false;
     798           0 :                 } else if (nread != sizeof(pid)) {
     799           0 :                         printf("read/write compare failed: "
     800             :                                "recv %ld req %ld\n", (unsigned long)nread,
     801             :                                (unsigned long)sizeof(pid));
     802           0 :                         correct = false;
     803             :                 }
     804             : 
     805           0 :                 if (pid2 != pid) {
     806           0 :                         printf("data corruption!\n");
     807           0 :                         correct = False;
     808             :                 }
     809             : 
     810           0 :                 status = cli_close(c, fnum);
     811           0 :                 if (!NT_STATUS_IS_OK(status)) {
     812           0 :                         printf("close failed (%s)\n", nt_errstr(status));
     813           0 :                         correct = False;
     814             :                 }
     815             : 
     816           0 :                 status = cli_unlink(c, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
     817           0 :                 if (!NT_STATUS_IS_OK(status)) {
     818           0 :                         printf("unlink failed (%s)\n", nt_errstr(status));
     819           0 :                         correct = False;
     820             :                 }
     821             : 
     822           0 :                 status = cli_unlock(c, fnum2, n*sizeof(int), sizeof(int));
     823           0 :                 if (!NT_STATUS_IS_OK(status)) {
     824           0 :                         printf("unlock failed (%s)\n", nt_errstr(status));
     825           0 :                         correct = False;
     826             :                 }
     827             :         }
     828             : 
     829           0 :         cli_close(c, fnum2);
     830           0 :         cli_unlink(c, lockfname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
     831             : 
     832           0 :         printf("%d\n", i);
     833             : 
     834           0 :         return correct;
     835             : }
     836             : 
     837           0 : static bool run_torture(int dummy)
     838             : {
     839           0 :         struct cli_state *cli;
     840           0 :         bool ret;
     841             : 
     842           0 :         cli = current_cli;
     843             : 
     844           0 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
     845             : 
     846           0 :         ret = rw_torture(cli);
     847             : 
     848           0 :         if (!torture_close_connection(cli)) {
     849           0 :                 ret = False;
     850             :         }
     851             : 
     852           0 :         return ret;
     853             : }
     854             : 
     855           0 : static bool rw_torture3(struct cli_state *c, char *lockfname)
     856             : {
     857           0 :         uint16_t fnum = (uint16_t)-1;
     858           0 :         unsigned int i = 0;
     859           0 :         char buf[131072];
     860           0 :         char buf_rd[131072];
     861           0 :         unsigned count;
     862           0 :         unsigned countprev = 0;
     863           0 :         size_t sent = 0;
     864           0 :         bool correct = True;
     865           0 :         NTSTATUS status = NT_STATUS_OK;
     866             : 
     867           0 :         srandom(1);
     868           0 :         for (i = 0; i < sizeof(buf); i += sizeof(uint32_t))
     869             :         {
     870           0 :                 SIVAL(buf, i, sys_random());
     871             :         }
     872             : 
     873           0 :         if (procnum == 0)
     874             :         {
     875           0 :                 status = cli_unlink(
     876             :                         c, lockfname,
     877             :                         FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
     878           0 :                 if (!NT_STATUS_IS_OK(status)) {
     879           0 :                         printf("unlink failed (%s) (normal, this file should "
     880             :                                "not exist)\n", nt_errstr(status));
     881             :                 }
     882             : 
     883           0 :                 status = cli_openx(c, lockfname, O_RDWR | O_CREAT | O_EXCL,
     884             :                                   DENY_NONE, &fnum);
     885           0 :                 if (!NT_STATUS_IS_OK(status)) {
     886           0 :                         printf("first open read/write of %s failed (%s)\n",
     887             :                                         lockfname, nt_errstr(status));
     888           0 :                         return False;
     889             :                 }
     890             :         }
     891             :         else
     892             :         {
     893           0 :                 for (i = 0; i < 500 && fnum == (uint16_t)-1; i++)
     894             :                 {
     895           0 :                         status = cli_openx(c, lockfname, O_RDONLY, 
     896             :                                          DENY_NONE, &fnum);
     897           0 :                         if (NT_STATUS_IS_OK(status)) {
     898           0 :                                 break;
     899             :                         }
     900           0 :                         smb_msleep(10);
     901             :                 }
     902           0 :                 if (!NT_STATUS_IS_OK(status)) {
     903           0 :                         printf("second open read-only of %s failed (%s)\n",
     904             :                                         lockfname, nt_errstr(status));
     905           0 :                         return False;
     906             :                 }
     907             :         }
     908             : 
     909           0 :         i = 0;
     910           0 :         for (count = 0; count < sizeof(buf); count += sent)
     911             :         {
     912           0 :                 if (count >= countprev) {
     913           0 :                         printf("%d %8d\r", i, count);
     914           0 :                         fflush(stdout);
     915           0 :                         i++;
     916           0 :                         countprev += (sizeof(buf) / 20);
     917             :                 }
     918             : 
     919           0 :                 if (procnum == 0)
     920             :                 {
     921           0 :                         sent = ((unsigned)sys_random()%(20))+ 1;
     922           0 :                         if (sent > sizeof(buf) - count)
     923             :                         {
     924           0 :                                 sent = sizeof(buf) - count;
     925             :                         }
     926             : 
     927           0 :                         status = cli_writeall(c, fnum, 0, (uint8_t *)buf+count,
     928             :                                               count, sent, NULL);
     929           0 :                         if (!NT_STATUS_IS_OK(status)) {
     930           0 :                                 printf("write failed (%s)\n",
     931             :                                        nt_errstr(status));
     932           0 :                                 correct = False;
     933             :                         }
     934             :                 }
     935             :                 else
     936             :                 {
     937           0 :                         status = cli_read(c, fnum, buf_rd+count, count,
     938             :                                           sizeof(buf)-count, &sent);
     939           0 :                         if(!NT_STATUS_IS_OK(status)) {
     940           0 :                                 printf("read failed offset:%d size:%ld (%s)\n",
     941             :                                        count, (unsigned long)sizeof(buf)-count,
     942             :                                        nt_errstr(status));
     943           0 :                                 correct = False;
     944           0 :                                 sent = 0;
     945           0 :                         } else if (sent > 0) {
     946           0 :                                 if (memcmp(buf_rd+count, buf+count, sent) != 0)
     947             :                                 {
     948           0 :                                         printf("read/write compare failed\n");
     949           0 :                                         printf("offset: %d req %ld recvd %ld\n", count, (unsigned long)sizeof(buf)-count, (unsigned long)sent);
     950           0 :                                         correct = False;
     951           0 :                                         break;
     952             :                                 }
     953             :                         }
     954             :                 }
     955             : 
     956             :         }
     957             : 
     958           0 :         status = cli_close(c, fnum);
     959           0 :         if (!NT_STATUS_IS_OK(status)) {
     960           0 :                 printf("close failed (%s)\n", nt_errstr(status));
     961           0 :                 correct = False;
     962             :         }
     963             : 
     964           0 :         return correct;
     965             : }
     966             : 
     967          18 : static bool rw_torture2(struct cli_state *c1, struct cli_state *c2)
     968             : {
     969          18 :         const char *lockfname = "\\torture2.lck";
     970           0 :         uint16_t fnum1;
     971           0 :         uint16_t fnum2;
     972           0 :         int i;
     973           0 :         char buf[131072];
     974           0 :         char buf_rd[131072];
     975          18 :         bool correct = True;
     976           0 :         size_t bytes_read;
     977           0 :         NTSTATUS status;
     978             : 
     979          18 :         status = cli_unlink(c1, lockfname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
     980          18 :         if (!NT_STATUS_IS_OK(status)) {
     981          18 :                 printf("unlink failed (%s) (normal, this file should not exist)\n", nt_errstr(status));
     982             :         }
     983             : 
     984          18 :         status = cli_openx(c1, lockfname, O_RDWR | O_CREAT | O_EXCL,
     985             :                           DENY_NONE, &fnum1);
     986          18 :         if (!NT_STATUS_IS_OK(status)) {
     987           0 :                 printf("first open read/write of %s failed (%s)\n",
     988             :                                 lockfname, nt_errstr(status));
     989           0 :                 return False;
     990             :         }
     991             : 
     992          18 :         status = cli_openx(c2, lockfname, O_RDONLY, DENY_NONE, &fnum2);
     993          18 :         if (!NT_STATUS_IS_OK(status)) {
     994           0 :                 printf("second open read-only of %s failed (%s)\n",
     995             :                                 lockfname, nt_errstr(status));
     996           0 :                 cli_close(c1, fnum1);
     997           0 :                 return False;
     998             :         }
     999             : 
    1000        1818 :         for (i = 0; i < torture_numops; i++)
    1001             :         {
    1002        1800 :                 size_t buf_size = ((unsigned)sys_random()%(sizeof(buf)-1))+ 1;
    1003        1800 :                 if (i % 10 == 0) {
    1004         180 :                         printf("%d\r", i); fflush(stdout);
    1005             :                 }
    1006             : 
    1007        1800 :                 generate_random_buffer((unsigned char *)buf, buf_size);
    1008             : 
    1009        1800 :                 status = cli_writeall(c1, fnum1, 0, (uint8_t *)buf, 0,
    1010             :                                       buf_size, NULL);
    1011        1800 :                 if (!NT_STATUS_IS_OK(status)) {
    1012           0 :                         printf("write failed (%s)\n", nt_errstr(status));
    1013           0 :                         correct = False;
    1014           0 :                         break;
    1015             :                 }
    1016             : 
    1017        1800 :                 status = cli_read(c2, fnum2, buf_rd, 0, buf_size, &bytes_read);
    1018        1800 :                 if(!NT_STATUS_IS_OK(status)) {
    1019           0 :                         printf("read failed (%s)\n", nt_errstr(status));
    1020           0 :                         correct = false;
    1021           0 :                         break;
    1022        1800 :                 } else if (bytes_read != buf_size) {
    1023           0 :                         printf("read failed\n");
    1024           0 :                         printf("read %ld, expected %ld\n",
    1025             :                                (unsigned long)bytes_read,
    1026             :                                (unsigned long)buf_size); 
    1027           0 :                         correct = False;
    1028           0 :                         break;
    1029             :                 }
    1030             : 
    1031        1800 :                 if (memcmp(buf_rd, buf, buf_size) != 0)
    1032             :                 {
    1033           0 :                         printf("read/write compare failed\n");
    1034           0 :                         correct = False;
    1035           0 :                         break;
    1036             :                 }
    1037             :         }
    1038             : 
    1039          18 :         status = cli_close(c2, fnum2);
    1040          18 :         if (!NT_STATUS_IS_OK(status)) {
    1041           0 :                 printf("close failed (%s)\n", nt_errstr(status));
    1042           0 :                 correct = False;
    1043             :         }
    1044             : 
    1045          18 :         status = cli_close(c1, fnum1);
    1046          18 :         if (!NT_STATUS_IS_OK(status)) {
    1047           0 :                 printf("close failed (%s)\n", nt_errstr(status));
    1048           0 :                 correct = False;
    1049             :         }
    1050             : 
    1051          18 :         status = cli_unlink(c1, lockfname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    1052          18 :         if (!NT_STATUS_IS_OK(status)) {
    1053           0 :                 printf("unlink failed (%s)\n", nt_errstr(status));
    1054           0 :                 correct = False;
    1055             :         }
    1056             : 
    1057          18 :         return correct;
    1058             : }
    1059             : 
    1060           9 : static bool run_readwritetest(int dummy)
    1061             : {
    1062           0 :         struct cli_state *cli1, *cli2;
    1063           9 :         bool test1, test2 = False;
    1064             : 
    1065           9 :         if (!torture_open_connection(&cli1, 0) || !torture_open_connection(&cli2, 1)) {
    1066           0 :                 return False;
    1067             :         }
    1068           9 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    1069           9 :         smbXcli_conn_set_sockopt(cli2->conn, sockops);
    1070             : 
    1071           9 :         printf("starting readwritetest\n");
    1072             : 
    1073           9 :         test1 = rw_torture2(cli1, cli2);
    1074           9 :         printf("Passed readwritetest v1: %s\n", BOOLSTR(test1));
    1075             : 
    1076           9 :         if (test1) {
    1077           9 :                 test2 = rw_torture2(cli1, cli1);
    1078           9 :                 printf("Passed readwritetest v2: %s\n", BOOLSTR(test2));
    1079             :         }
    1080             : 
    1081           9 :         if (!torture_close_connection(cli1)) {
    1082           0 :                 test1 = False;
    1083             :         }
    1084             : 
    1085           9 :         if (!torture_close_connection(cli2)) {
    1086           0 :                 test2 = False;
    1087             :         }
    1088             : 
    1089           9 :         return (test1 && test2);
    1090             : }
    1091             : 
    1092           0 : static bool run_readwritemulti(int dummy)
    1093             : {
    1094           0 :         struct cli_state *cli;
    1095           0 :         bool test;
    1096             : 
    1097           0 :         cli = current_cli;
    1098             : 
    1099           0 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    1100             : 
    1101           0 :         printf("run_readwritemulti: fname %s\n", randomfname);
    1102           0 :         test = rw_torture3(cli, randomfname);
    1103             : 
    1104           0 :         if (!torture_close_connection(cli)) {
    1105           0 :                 test = False;
    1106             :         }
    1107             : 
    1108           0 :         return test;
    1109             : }
    1110             : 
    1111          14 : static bool run_readwritelarge_internal(void)
    1112             : {
    1113           0 :         static struct cli_state *cli1;
    1114           0 :         uint16_t fnum1;
    1115          14 :         const char *lockfname = "\\large.dat";
    1116           0 :         off_t fsize;
    1117           0 :         char buf[126*1024];
    1118          14 :         bool correct = True;
    1119           0 :         NTSTATUS status;
    1120             : 
    1121          14 :         if (!torture_open_connection(&cli1, 0)) {
    1122           0 :                 return False;
    1123             :         }
    1124          14 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    1125          14 :         memset(buf,'\0',sizeof(buf));
    1126             : 
    1127          14 :         printf("starting readwritelarge_internal\n");
    1128             : 
    1129          14 :         cli_unlink(cli1, lockfname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    1130             : 
    1131          14 :         status = cli_openx(cli1, lockfname, O_RDWR | O_CREAT | O_EXCL,
    1132             :                           DENY_NONE, &fnum1);
    1133          14 :         if (!NT_STATUS_IS_OK(status)) {
    1134           0 :                 printf("open read/write of %s failed (%s)\n", lockfname, nt_errstr(status));
    1135           0 :                 return False;
    1136             :         }
    1137             : 
    1138          14 :         cli_writeall(cli1, fnum1, 0, (uint8_t *)buf, 0, sizeof(buf), NULL);
    1139             : 
    1140          14 :         status = cli_qfileinfo_basic(cli1, fnum1, NULL, &fsize, NULL, NULL,
    1141             :                                      NULL, NULL, NULL);
    1142          14 :         if (!NT_STATUS_IS_OK(status)) {
    1143           0 :                 printf("qfileinfo failed (%s)\n", nt_errstr(status));
    1144           0 :                 correct = False;
    1145             :         }
    1146             : 
    1147          14 :         if (fsize == sizeof(buf))
    1148          14 :                 printf("readwritelarge_internal test 1 succeeded (size = %lx)\n",
    1149             :                        (unsigned long)fsize);
    1150             :         else {
    1151           0 :                 printf("readwritelarge_internal test 1 failed (size = %lx)\n",
    1152             :                        (unsigned long)fsize);
    1153           0 :                 correct = False;
    1154             :         }
    1155             : 
    1156          14 :         status = cli_close(cli1, fnum1);
    1157          14 :         if (!NT_STATUS_IS_OK(status)) {
    1158           0 :                 printf("close failed (%s)\n", nt_errstr(status));
    1159           0 :                 correct = False;
    1160             :         }
    1161             : 
    1162          14 :         status = cli_unlink(cli1, lockfname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    1163          14 :         if (!NT_STATUS_IS_OK(status)) {
    1164           0 :                 printf("unlink failed (%s)\n", nt_errstr(status));
    1165           0 :                 correct = False;
    1166             :         }
    1167             : 
    1168          14 :         status = cli_openx(cli1, lockfname, O_RDWR | O_CREAT | O_EXCL,
    1169             :                           DENY_NONE, &fnum1);
    1170          14 :         if (!NT_STATUS_IS_OK(status)) {
    1171           0 :                 printf("open read/write of %s failed (%s)\n", lockfname, nt_errstr(status));
    1172           0 :                 return False;
    1173             :         }
    1174             : 
    1175          14 :         cli_smbwrite(cli1, fnum1, buf, 0, sizeof(buf), NULL);
    1176             : 
    1177          14 :         status = cli_qfileinfo_basic(cli1, fnum1, NULL, &fsize, NULL, NULL,
    1178             :                                      NULL, NULL, NULL);
    1179          14 :         if (!NT_STATUS_IS_OK(status)) {
    1180           0 :                 printf("qfileinfo failed (%s)\n", nt_errstr(status));
    1181           0 :                 correct = False;
    1182             :         }
    1183             : 
    1184          14 :         if (fsize == sizeof(buf))
    1185          14 :                 printf("readwritelarge_internal test 2 succeeded (size = %lx)\n",
    1186             :                        (unsigned long)fsize);
    1187             :         else {
    1188           0 :                 printf("readwritelarge_internal test 2 failed (size = %lx)\n",
    1189             :                        (unsigned long)fsize);
    1190           0 :                 correct = False;
    1191             :         }
    1192             : 
    1193          14 :         status = cli_close(cli1, fnum1);
    1194          14 :         if (!NT_STATUS_IS_OK(status)) {
    1195           0 :                 printf("close failed (%s)\n", nt_errstr(status));
    1196           0 :                 correct = False;
    1197             :         }
    1198             : 
    1199          14 :         if (!torture_close_connection(cli1)) {
    1200           0 :                 correct = False;
    1201             :         }
    1202          14 :         return correct;
    1203             : }
    1204             : 
    1205           9 : static bool run_readwritelarge(int dummy)
    1206             : {
    1207           9 :         return run_readwritelarge_internal();
    1208             : }
    1209             : 
    1210           5 : static bool run_readwritelarge_signtest(int dummy)
    1211             : {
    1212           0 :         bool ret;
    1213           5 :         signing_state = SMB_SIGNING_REQUIRED;
    1214           5 :         ret = run_readwritelarge_internal();
    1215           5 :         signing_state = SMB_SIGNING_DEFAULT;
    1216           5 :         return ret;
    1217             : }
    1218             : 
    1219             : int line_count = 0;
    1220             : int nbio_id;
    1221             : 
    1222             : #define ival(s) strtol(s, NULL, 0)
    1223             : 
    1224             : /* run a test that simulates an approximate netbench client load */
    1225           0 : static bool run_netbench(int client)
    1226             : {
    1227           0 :         struct cli_state *cli;
    1228           0 :         int i;
    1229           0 :         char line[1024];
    1230           0 :         char cname[20];
    1231           0 :         FILE *f;
    1232           0 :         const char *params[20];
    1233           0 :         bool correct = True;
    1234             : 
    1235           0 :         cli = current_cli;
    1236             : 
    1237           0 :         nbio_id = client;
    1238             : 
    1239           0 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    1240             : 
    1241           0 :         nb_setup(cli);
    1242             : 
    1243           0 :         slprintf(cname,sizeof(cname)-1, "client%d", client);
    1244             : 
    1245           0 :         f = fopen(client_txt, "r");
    1246             : 
    1247           0 :         if (!f) {
    1248           0 :                 perror(client_txt);
    1249           0 :                 return False;
    1250             :         }
    1251             : 
    1252           0 :         while (fgets(line, sizeof(line)-1, f)) {
    1253           0 :                 char *saveptr;
    1254           0 :                 line_count++;
    1255             : 
    1256           0 :                 line[strlen(line)-1] = 0;
    1257             : 
    1258             :                 /* printf("[%d] %s\n", line_count, line); */
    1259             : 
    1260           0 :                 all_string_sub(line,"client1", cname, sizeof(line));
    1261             : 
    1262             :                 /* parse the command parameters */
    1263           0 :                 params[0] = strtok_r(line, " ", &saveptr);
    1264           0 :                 i = 0;
    1265           0 :                 while (params[i]) params[++i] = strtok_r(NULL, " ", &saveptr);
    1266             : 
    1267           0 :                 params[i] = "";
    1268             : 
    1269           0 :                 if (i < 2) continue;
    1270             : 
    1271           0 :                 if (!strncmp(params[0],"SMB", 3)) {
    1272           0 :                         printf("ERROR: You are using a dbench 1 load file\n");
    1273           0 :                         exit(1);
    1274             :                 }
    1275             : 
    1276           0 :                 if (!strcmp(params[0],"NTCreateX")) {
    1277           0 :                         nb_createx(params[1], ival(params[2]), ival(params[3]), 
    1278           0 :                                    ival(params[4]));
    1279           0 :                 } else if (!strcmp(params[0],"Close")) {
    1280           0 :                         nb_close(ival(params[1]));
    1281           0 :                 } else if (!strcmp(params[0],"Rename")) {
    1282           0 :                         nb_rename(params[1], params[2]);
    1283           0 :                 } else if (!strcmp(params[0],"Unlink")) {
    1284           0 :                         nb_unlink(params[1]);
    1285           0 :                 } else if (!strcmp(params[0],"Deltree")) {
    1286           0 :                         nb_deltree(params[1]);
    1287           0 :                 } else if (!strcmp(params[0],"Rmdir")) {
    1288           0 :                         nb_rmdir(params[1]);
    1289           0 :                 } else if (!strcmp(params[0],"QUERY_PATH_INFORMATION")) {
    1290           0 :                         nb_qpathinfo(params[1]);
    1291           0 :                 } else if (!strcmp(params[0],"QUERY_FILE_INFORMATION")) {
    1292           0 :                         nb_qfileinfo(ival(params[1]));
    1293           0 :                 } else if (!strcmp(params[0],"QUERY_FS_INFORMATION")) {
    1294           0 :                         nb_qfsinfo(ival(params[1]));
    1295           0 :                 } else if (!strcmp(params[0],"FIND_FIRST")) {
    1296           0 :                         nb_findfirst(params[1]);
    1297           0 :                 } else if (!strcmp(params[0],"WriteX")) {
    1298           0 :                         nb_writex(ival(params[1]), 
    1299           0 :                                   ival(params[2]), ival(params[3]), ival(params[4]));
    1300           0 :                 } else if (!strcmp(params[0],"ReadX")) {
    1301           0 :                         nb_readx(ival(params[1]), 
    1302           0 :                                   ival(params[2]), ival(params[3]), ival(params[4]));
    1303           0 :                 } else if (!strcmp(params[0],"Flush")) {
    1304           0 :                         nb_flush(ival(params[1]));
    1305             :                 } else {
    1306           0 :                         printf("Unknown operation %s\n", params[0]);
    1307           0 :                         exit(1);
    1308             :                 }
    1309             :         }
    1310           0 :         fclose(f);
    1311             : 
    1312           0 :         nb_cleanup();
    1313             : 
    1314           0 :         if (!torture_close_connection(cli)) {
    1315           0 :                 correct = False;
    1316             :         }
    1317             : 
    1318           0 :         return correct;
    1319             : }
    1320             : 
    1321             : 
    1322             : /* run a test that simulates an approximate netbench client load */
    1323           0 : static bool run_nbench(int dummy)
    1324             : {
    1325           0 :         double t;
    1326           0 :         bool correct = True;
    1327             : 
    1328           0 :         nbio_shmem(torture_nprocs);
    1329             : 
    1330           0 :         nbio_id = -1;
    1331             : 
    1332           0 :         signal(SIGALRM, nb_alarm);
    1333           0 :         alarm(1);
    1334           0 :         t = create_procs(run_netbench, &correct);
    1335           0 :         alarm(0);
    1336             : 
    1337           0 :         printf("\nThroughput %g MB/sec\n", 
    1338           0 :                1.0e-6 * nbio_total() / t);
    1339           0 :         return correct;
    1340             : }
    1341             : 
    1342             : 
    1343             : /*
    1344             :   This test checks for two things:
    1345             : 
    1346             :   1) correct support for retaining locks over a close (ie. the server
    1347             :      must not use posix semantics)
    1348             :   2) support for lock timeouts
    1349             :  */
    1350           5 : static bool run_locktest1(int dummy)
    1351             : {
    1352           0 :         struct cli_state *cli1, *cli2;
    1353           5 :         const char *fname = "\\lockt1.lck";
    1354           0 :         uint16_t fnum1, fnum2, fnum3;
    1355           0 :         time_t t1, t2;
    1356           0 :         unsigned lock_timeout;
    1357           0 :         NTSTATUS status;
    1358             : 
    1359           5 :         if (!torture_open_connection(&cli1, 0) || !torture_open_connection(&cli2, 1)) {
    1360           0 :                 return False;
    1361             :         }
    1362           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    1363           5 :         smbXcli_conn_set_sockopt(cli2->conn, sockops);
    1364             : 
    1365           5 :         printf("starting locktest1\n");
    1366             : 
    1367           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    1368             : 
    1369           5 :         status = cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE,
    1370             :                           &fnum1);
    1371           5 :         if (!NT_STATUS_IS_OK(status)) {
    1372           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    1373           0 :                 return False;
    1374             :         }
    1375             : 
    1376           5 :         status = cli_openx(cli1, fname, O_RDWR, DENY_NONE, &fnum2);
    1377           5 :         if (!NT_STATUS_IS_OK(status)) {
    1378           0 :                 printf("open2 of %s failed (%s)\n", fname, nt_errstr(status));
    1379           0 :                 return False;
    1380             :         }
    1381             : 
    1382           5 :         status = cli_openx(cli2, fname, O_RDWR, DENY_NONE, &fnum3);
    1383           5 :         if (!NT_STATUS_IS_OK(status)) {
    1384           0 :                 printf("open3 of %s failed (%s)\n", fname, nt_errstr(status));
    1385           0 :                 return False;
    1386             :         }
    1387             : 
    1388           5 :         status = cli_lock32(cli1, fnum1, 0, 4, 0, WRITE_LOCK);
    1389           5 :         if (!NT_STATUS_IS_OK(status)) {
    1390           0 :                 printf("lock1 failed (%s)\n", nt_errstr(status));
    1391           0 :                 return false;
    1392             :         }
    1393             : 
    1394           5 :         status = cli_lock32(cli2, fnum3, 0, 4, 0, WRITE_LOCK);
    1395           5 :         if (NT_STATUS_IS_OK(status)) {
    1396           0 :                 printf("lock2 succeeded! This is a locking bug\n");
    1397           0 :                 return false;
    1398             :         } else {
    1399           5 :                 if (!check_both_error(__LINE__, status, ERRDOS, ERRlock,
    1400           5 :                                       NT_STATUS_LOCK_NOT_GRANTED)) {
    1401           0 :                         return false;
    1402             :                 }
    1403             :         }
    1404             : 
    1405           5 :         lock_timeout = (1 + (random() % 20));
    1406           5 :         printf("Testing lock timeout with timeout=%u\n", lock_timeout);
    1407           5 :         t1 = time(NULL);
    1408           5 :         status = cli_lock32(cli2, fnum3, 0, 4, lock_timeout * 1000, WRITE_LOCK);
    1409           5 :         if (NT_STATUS_IS_OK(status)) {
    1410           0 :                 printf("lock3 succeeded! This is a locking bug\n");
    1411           0 :                 return false;
    1412             :         } else {
    1413           5 :                 if (!check_both_error(__LINE__, status, ERRDOS, ERRlock,
    1414           5 :                                       NT_STATUS_FILE_LOCK_CONFLICT)) {
    1415           0 :                         return false;
    1416             :                 }
    1417             :         }
    1418           5 :         t2 = time(NULL);
    1419             : 
    1420           5 :         if (ABS(t2 - t1) < lock_timeout-1) {
    1421           0 :                 printf("error: This server appears not to support timed lock requests\n");
    1422             :         }
    1423             : 
    1424           5 :         printf("server slept for %u seconds for a %u second timeout\n",
    1425             :                (unsigned int)(t2-t1), lock_timeout);
    1426             : 
    1427           5 :         status = cli_close(cli1, fnum2);
    1428           5 :         if (!NT_STATUS_IS_OK(status)) {
    1429           0 :                 printf("close1 failed (%s)\n", nt_errstr(status));
    1430           0 :                 return False;
    1431             :         }
    1432             : 
    1433           5 :         status = cli_lock32(cli2, fnum3, 0, 4, 0, WRITE_LOCK);
    1434           5 :         if (NT_STATUS_IS_OK(status)) {
    1435           0 :                 printf("lock4 succeeded! This is a locking bug\n");
    1436           0 :                 return false;
    1437             :         } else {
    1438           5 :                 if (!check_both_error(__LINE__, status, ERRDOS, ERRlock,
    1439           5 :                                       NT_STATUS_FILE_LOCK_CONFLICT)) {
    1440           0 :                         return false;
    1441             :                 }
    1442             :         }
    1443             : 
    1444           5 :         status = cli_close(cli1, fnum1);
    1445           5 :         if (!NT_STATUS_IS_OK(status)) {
    1446           0 :                 printf("close2 failed (%s)\n", nt_errstr(status));
    1447           0 :                 return False;
    1448             :         }
    1449             : 
    1450           5 :         status = cli_close(cli2, fnum3);
    1451           5 :         if (!NT_STATUS_IS_OK(status)) {
    1452           0 :                 printf("close3 failed (%s)\n", nt_errstr(status));
    1453           0 :                 return False;
    1454             :         }
    1455             : 
    1456           5 :         status = cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    1457           5 :         if (!NT_STATUS_IS_OK(status)) {
    1458           0 :                 printf("unlink failed (%s)\n", nt_errstr(status));
    1459           0 :                 return False;
    1460             :         }
    1461             : 
    1462             : 
    1463           5 :         if (!torture_close_connection(cli1)) {
    1464           0 :                 return False;
    1465             :         }
    1466             : 
    1467           5 :         if (!torture_close_connection(cli2)) {
    1468           0 :                 return False;
    1469             :         }
    1470             : 
    1471           5 :         printf("Passed locktest1\n");
    1472           5 :         return True;
    1473             : }
    1474             : 
    1475             : /*
    1476             :   this checks to see if a secondary tconx can use open files from an
    1477             :   earlier tconx
    1478             :  */
    1479           5 : static bool run_tcon_test(int dummy)
    1480             : {
    1481           0 :         static struct cli_state *cli;
    1482           5 :         const char *fname = "\\tcontest.tmp";
    1483           0 :         uint16_t fnum1;
    1484           0 :         uint32_t cnum1, cnum2, cnum3;
    1485           5 :         struct smbXcli_tcon *orig_tcon = NULL;
    1486           5 :         char *orig_share = NULL;
    1487           0 :         uint16_t vuid1, vuid2;
    1488           0 :         char buf[4];
    1489           5 :         bool ret = True;
    1490           0 :         NTSTATUS status;
    1491             : 
    1492           5 :         memset(buf, '\0', sizeof(buf));
    1493             : 
    1494           5 :         if (!torture_open_connection(&cli, 0)) {
    1495           0 :                 return False;
    1496             :         }
    1497           5 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    1498             : 
    1499           5 :         printf("starting tcontest\n");
    1500             : 
    1501           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    1502             : 
    1503           5 :         status = cli_openx(cli, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum1);
    1504           5 :         if (!NT_STATUS_IS_OK(status)) {
    1505           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    1506           0 :                 return False;
    1507             :         }
    1508             : 
    1509           5 :         cnum1 = cli_state_get_tid(cli);
    1510           5 :         vuid1 = cli_state_get_uid(cli);
    1511             : 
    1512           5 :         status = cli_writeall(cli, fnum1, 0, (uint8_t *)buf, 130, 4, NULL);
    1513           5 :         if (!NT_STATUS_IS_OK(status)) {
    1514           0 :                 printf("initial write failed (%s)", nt_errstr(status));
    1515           0 :                 return False;
    1516             :         }
    1517             : 
    1518           5 :         cli_state_save_tcon_share(cli, &orig_tcon, &orig_share);
    1519             : 
    1520           5 :         status = cli_tree_connect_creds(cli, share, "?????", torture_creds);
    1521           5 :         if (!NT_STATUS_IS_OK(status)) {
    1522           0 :                 printf("%s refused 2nd tree connect (%s)\n", host,
    1523             :                        nt_errstr(status));
    1524           0 :                 cli_state_restore_tcon_share(cli, orig_tcon, orig_share);
    1525           0 :                 cli_shutdown(cli);
    1526           0 :                 return False;
    1527             :         }
    1528             : 
    1529           5 :         cnum2 = cli_state_get_tid(cli);
    1530           5 :         cnum3 = MAX(cnum1, cnum2) + 1; /* any invalid number */
    1531           5 :         vuid2 = cli_state_get_uid(cli) + 1;
    1532             : 
    1533             :         /* try a write with the wrong tid */
    1534           5 :         cli_state_set_tid(cli, cnum2);
    1535             : 
    1536           5 :         status = cli_writeall(cli, fnum1, 0, (uint8_t *)buf, 130, 4, NULL);
    1537           5 :         if (NT_STATUS_IS_OK(status)) {
    1538           0 :                 printf("* server allows write with wrong TID\n");
    1539           0 :                 ret = False;
    1540             :         } else {
    1541           5 :                 printf("server fails write with wrong TID : %s\n",
    1542             :                        nt_errstr(status));
    1543             :         }
    1544             : 
    1545             : 
    1546             :         /* try a write with an invalid tid */
    1547           5 :         cli_state_set_tid(cli, cnum3);
    1548             : 
    1549           5 :         status = cli_writeall(cli, fnum1, 0, (uint8_t *)buf, 130, 4, NULL);
    1550           5 :         if (NT_STATUS_IS_OK(status)) {
    1551           0 :                 printf("* server allows write with invalid TID\n");
    1552           0 :                 ret = False;
    1553             :         } else {
    1554           5 :                 printf("server fails write with invalid TID : %s\n",
    1555             :                        nt_errstr(status));
    1556             :         }
    1557             : 
    1558             :         /* try a write with an invalid vuid */
    1559           5 :         cli_state_set_uid(cli, vuid2);
    1560           5 :         cli_state_set_tid(cli, cnum1);
    1561             : 
    1562           5 :         status = cli_writeall(cli, fnum1, 0, (uint8_t *)buf, 130, 4, NULL);
    1563           5 :         if (NT_STATUS_IS_OK(status)) {
    1564           0 :                 printf("* server allows write with invalid VUID\n");
    1565           0 :                 ret = False;
    1566             :         } else {
    1567           5 :                 printf("server fails write with invalid VUID : %s\n",
    1568             :                        nt_errstr(status));
    1569             :         }
    1570             : 
    1571           5 :         cli_state_set_tid(cli, cnum1);
    1572           5 :         cli_state_set_uid(cli, vuid1);
    1573             : 
    1574           5 :         status = cli_close(cli, fnum1);
    1575           5 :         if (!NT_STATUS_IS_OK(status)) {
    1576           0 :                 printf("close failed (%s)\n", nt_errstr(status));
    1577           0 :                 cli_state_restore_tcon_share(cli, orig_tcon, orig_share);
    1578           0 :                 cli_shutdown(cli);
    1579           0 :                 return False;
    1580             :         }
    1581             : 
    1582           5 :         cli_state_set_tid(cli, cnum2);
    1583             : 
    1584           5 :         status = cli_tdis(cli);
    1585           5 :         if (!NT_STATUS_IS_OK(status)) {
    1586           0 :                 printf("secondary tdis failed (%s)\n", nt_errstr(status));
    1587           0 :                 cli_state_restore_tcon_share(cli, orig_tcon, orig_share);
    1588           0 :                 cli_shutdown(cli);
    1589           0 :                 return False;
    1590             :         }
    1591             : 
    1592           5 :         cli_state_restore_tcon_share(cli, orig_tcon, orig_share);
    1593             : 
    1594           5 :         cli_state_set_tid(cli, cnum1);
    1595             : 
    1596           5 :         if (!torture_close_connection(cli)) {
    1597           0 :                 return False;
    1598             :         }
    1599             : 
    1600           5 :         return ret;
    1601             : }
    1602             : 
    1603             : 
    1604             : /*
    1605             :  checks for old style tcon support
    1606             :  */
    1607           5 : static bool run_tcon2_test(int dummy)
    1608             : {
    1609           0 :         static struct cli_state *cli;
    1610           0 :         uint16_t cnum, max_xmit;
    1611           0 :         char *service;
    1612           0 :         NTSTATUS status;
    1613             : 
    1614           5 :         if (!torture_open_connection(&cli, 0)) {
    1615           0 :                 return False;
    1616             :         }
    1617           5 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    1618             : 
    1619           5 :         printf("starting tcon2 test\n");
    1620             : 
    1621           5 :         if (asprintf(&service, "\\\\%s\\%s", host, share) == -1) {
    1622           0 :                 return false;
    1623             :         }
    1624             : 
    1625           5 :         status = cli_raw_tcon(cli, service, password, "?????", &max_xmit, &cnum);
    1626             : 
    1627           5 :         SAFE_FREE(service);
    1628             : 
    1629           5 :         if (!NT_STATUS_IS_OK(status)) {
    1630           5 :                 printf("tcon2 failed : %s\n", nt_errstr(status));
    1631             :         } else {
    1632           0 :                 printf("tcon OK : max_xmit=%d cnum=%d\n",
    1633             :                        (int)max_xmit, (int)cnum);
    1634             :         }
    1635             : 
    1636           5 :         if (!torture_close_connection(cli)) {
    1637           0 :                 return False;
    1638             :         }
    1639             : 
    1640           5 :         printf("Passed tcon2 test\n");
    1641           5 :         return True;
    1642             : }
    1643             : 
    1644          50 : static bool tcon_devtest(struct cli_state *cli,
    1645             :                          const char *myshare, const char *devtype,
    1646             :                          const char *return_devtype,
    1647             :                          NTSTATUS expected_error)
    1648             : {
    1649           0 :         NTSTATUS status;
    1650           0 :         bool ret;
    1651             : 
    1652          50 :         status = cli_tree_connect_creds(cli, myshare, devtype, torture_creds);
    1653             : 
    1654          50 :         if (NT_STATUS_IS_OK(expected_error)) {
    1655          20 :                 if (NT_STATUS_IS_OK(status)) {
    1656          40 :                         if (return_devtype != NULL &&
    1657          20 :                             strequal(cli->dev, return_devtype)) {
    1658          20 :                                 ret = True;
    1659             :                         } else { 
    1660           0 :                                 printf("tconX to share %s with type %s "
    1661             :                                        "succeeded but returned the wrong "
    1662             :                                        "device type (got [%s] but should have got [%s])\n",
    1663             :                                        myshare, devtype, cli->dev, return_devtype);
    1664           0 :                                 ret = False;
    1665             :                         }
    1666             :                 } else {
    1667           0 :                         printf("tconX to share %s with type %s "
    1668             :                                "should have succeeded but failed\n",
    1669             :                                myshare, devtype);
    1670           0 :                         ret = False;
    1671             :                 }
    1672          20 :                 cli_tdis(cli);
    1673             :         } else {
    1674          30 :                 if (NT_STATUS_IS_OK(status)) {
    1675           0 :                         printf("tconx to share %s with type %s "
    1676             :                                "should have failed but succeeded\n",
    1677             :                                myshare, devtype);
    1678           0 :                         ret = False;
    1679             :                 } else {
    1680          30 :                         if (NT_STATUS_EQUAL(status, expected_error)) {
    1681          30 :                                 ret = True;
    1682             :                         } else {
    1683           0 :                                 printf("Returned unexpected error\n");
    1684           0 :                                 ret = False;
    1685             :                         }
    1686             :                 }
    1687             :         }
    1688          50 :         return ret;
    1689             : }
    1690             : 
    1691             : /*
    1692             :  checks for correct tconX support
    1693             :  */
    1694           5 : static bool run_tcon_devtype_test(int dummy)
    1695             : {
    1696           0 :         static struct cli_state *cli1 = NULL;
    1697           5 :         int flags = CLI_FULL_CONNECTION_FORCE_SMB1;
    1698           0 :         NTSTATUS status;
    1699           5 :         bool ret = True;
    1700             : 
    1701           5 :         status = cli_full_connection_creds(&cli1,
    1702             :                                            myname,
    1703             :                                            host,
    1704             :                                            NULL, /* dest_ss */
    1705             :                                            port_to_use,
    1706             :                                            NULL, /* service */
    1707             :                                            NULL, /* service_type */
    1708             :                                            torture_creds,
    1709             :                                            flags);
    1710             : 
    1711           5 :         if (!NT_STATUS_IS_OK(status)) {
    1712           0 :                 printf("could not open connection\n");
    1713           0 :                 return False;
    1714             :         }
    1715             : 
    1716           5 :         if (!tcon_devtest(cli1, "IPC$", "A:", NULL, NT_STATUS_BAD_DEVICE_TYPE))
    1717           0 :                 ret = False;
    1718             : 
    1719           5 :         if (!tcon_devtest(cli1, "IPC$", "?????", "IPC", NT_STATUS_OK))
    1720           0 :                 ret = False;
    1721             : 
    1722           5 :         if (!tcon_devtest(cli1, "IPC$", "LPT:", NULL, NT_STATUS_BAD_DEVICE_TYPE))
    1723           0 :                 ret = False;
    1724             : 
    1725           5 :         if (!tcon_devtest(cli1, "IPC$", "IPC", "IPC", NT_STATUS_OK))
    1726           0 :                 ret = False;
    1727             : 
    1728           5 :         if (!tcon_devtest(cli1, "IPC$", "FOOBA", NULL, NT_STATUS_BAD_DEVICE_TYPE))
    1729           0 :                 ret = False;
    1730             : 
    1731           5 :         if (!tcon_devtest(cli1, share, "A:", "A:", NT_STATUS_OK))
    1732           0 :                 ret = False;
    1733             : 
    1734           5 :         if (!tcon_devtest(cli1, share, "?????", "A:", NT_STATUS_OK))
    1735           0 :                 ret = False;
    1736             : 
    1737           5 :         if (!tcon_devtest(cli1, share, "LPT:", NULL, NT_STATUS_BAD_DEVICE_TYPE))
    1738           0 :                 ret = False;
    1739             : 
    1740           5 :         if (!tcon_devtest(cli1, share, "IPC", NULL, NT_STATUS_BAD_DEVICE_TYPE))
    1741           0 :                 ret = False;
    1742             : 
    1743           5 :         if (!tcon_devtest(cli1, share, "FOOBA", NULL, NT_STATUS_BAD_DEVICE_TYPE))
    1744           0 :                 ret = False;
    1745             : 
    1746           5 :         cli_shutdown(cli1);
    1747             : 
    1748           5 :         if (ret)
    1749           5 :                 printf("Passed tcondevtest\n");
    1750             : 
    1751           5 :         return ret;
    1752             : }
    1753             : 
    1754             : 
    1755             : /*
    1756             :   This test checks that 
    1757             : 
    1758             :   1) the server supports multiple locking contexts on the one SMB
    1759             :   connection, distinguished by PID.  
    1760             : 
    1761             :   2) the server correctly fails overlapping locks made by the same PID (this
    1762             :      goes against POSIX behaviour, which is why it is tricky to implement)
    1763             : 
    1764             :   3) the server denies unlock requests by an incorrect client PID
    1765             : */
    1766           5 : static bool run_locktest2(int dummy)
    1767             : {
    1768           0 :         static struct cli_state *cli;
    1769           5 :         const char *fname = "\\lockt2.lck";
    1770           0 :         uint16_t fnum1, fnum2, fnum3;
    1771           5 :         bool correct = True;
    1772           0 :         NTSTATUS status;
    1773             : 
    1774           5 :         if (!torture_open_connection(&cli, 0)) {
    1775           0 :                 return False;
    1776             :         }
    1777             : 
    1778           5 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    1779             : 
    1780           5 :         printf("starting locktest2\n");
    1781             : 
    1782           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    1783             : 
    1784           5 :         cli_setpid(cli, 1);
    1785             : 
    1786           5 :         status = cli_openx(cli, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum1);
    1787           5 :         if (!NT_STATUS_IS_OK(status)) {
    1788           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    1789           0 :                 return False;
    1790             :         }
    1791             : 
    1792           5 :         status = cli_openx(cli, fname, O_RDWR, DENY_NONE, &fnum2);
    1793           5 :         if (!NT_STATUS_IS_OK(status)) {
    1794           0 :                 printf("open2 of %s failed (%s)\n", fname, nt_errstr(status));
    1795           0 :                 return False;
    1796             :         }
    1797             : 
    1798           5 :         cli_setpid(cli, 2);
    1799             : 
    1800           5 :         status = cli_openx(cli, fname, O_RDWR, DENY_NONE, &fnum3);
    1801           5 :         if (!NT_STATUS_IS_OK(status)) {
    1802           0 :                 printf("open3 of %s failed (%s)\n", fname, nt_errstr(status));
    1803           0 :                 return False;
    1804             :         }
    1805             : 
    1806           5 :         cli_setpid(cli, 1);
    1807             : 
    1808           5 :         status = cli_lock32(cli, fnum1, 0, 4, 0, WRITE_LOCK);
    1809           5 :         if (!NT_STATUS_IS_OK(status)) {
    1810           0 :                 printf("lock1 failed (%s)\n", nt_errstr(status));
    1811           0 :                 return false;
    1812             :         }
    1813             : 
    1814           5 :         status = cli_lock32(cli, fnum1, 0, 4, 0, WRITE_LOCK);
    1815           5 :         if (NT_STATUS_IS_OK(status)) {
    1816           0 :                 printf("WRITE lock1 succeeded! This is a locking bug\n");
    1817           0 :                 correct = false;
    1818             :         } else {
    1819           5 :                 if (!check_both_error(__LINE__, status, ERRDOS, ERRlock,
    1820           5 :                                       NT_STATUS_LOCK_NOT_GRANTED)) {
    1821           0 :                         return false;
    1822             :                 }
    1823             :         }
    1824             : 
    1825           5 :         status = cli_lock32(cli, fnum2, 0, 4, 0, WRITE_LOCK);
    1826           5 :         if (NT_STATUS_IS_OK(status)) {
    1827           0 :                 printf("WRITE lock2 succeeded! This is a locking bug\n");
    1828           0 :                 correct = false;
    1829             :         } else {
    1830           5 :                 if (!check_both_error(__LINE__, status, ERRDOS, ERRlock,
    1831           5 :                                       NT_STATUS_LOCK_NOT_GRANTED)) {
    1832           0 :                         return false;
    1833             :                 }
    1834             :         }
    1835             : 
    1836           5 :         status = cli_lock32(cli, fnum2, 0, 4, 0, READ_LOCK);
    1837           5 :         if (NT_STATUS_IS_OK(status)) {
    1838           0 :                 printf("READ lock2 succeeded! This is a locking bug\n");
    1839           0 :                 correct = false;
    1840             :         } else {
    1841           5 :                 if (!check_both_error(__LINE__, status, ERRDOS, ERRlock,
    1842           5 :                                  NT_STATUS_FILE_LOCK_CONFLICT)) {
    1843           0 :                         return false;
    1844             :                 }
    1845             :         }
    1846             : 
    1847           5 :         status = cli_lock32(cli, fnum1, 100, 4, 0, WRITE_LOCK);
    1848           5 :         if (!NT_STATUS_IS_OK(status)) {
    1849           0 :                 printf("lock at 100 failed (%s)\n", nt_errstr(status));
    1850             :         }
    1851           5 :         cli_setpid(cli, 2);
    1852           5 :         if (NT_STATUS_IS_OK(cli_unlock(cli, fnum1, 100, 4))) {
    1853           0 :                 printf("unlock at 100 succeeded! This is a locking bug\n");
    1854           0 :                 correct = False;
    1855             :         }
    1856             : 
    1857           5 :         status = cli_unlock(cli, fnum1, 0, 4);
    1858           5 :         if (NT_STATUS_IS_OK(status)) {
    1859           0 :                 printf("unlock1 succeeded! This is a locking bug\n");
    1860           0 :                 correct = false;
    1861             :         } else {
    1862           5 :                 if (!check_both_error(__LINE__, status, ERRDOS, ERRlock,
    1863           5 :                                       NT_STATUS_RANGE_NOT_LOCKED)) {
    1864           0 :                         return false;
    1865             :                 }
    1866             :         }
    1867             : 
    1868           5 :         status = cli_unlock(cli, fnum1, 0, 8);
    1869           5 :         if (NT_STATUS_IS_OK(status)) {
    1870           0 :                 printf("unlock2 succeeded! This is a locking bug\n");
    1871           0 :                 correct = false;
    1872             :         } else {
    1873           5 :                 if (!check_both_error(__LINE__, status, ERRDOS, ERRlock,
    1874           5 :                                       NT_STATUS_RANGE_NOT_LOCKED)) {
    1875           0 :                         return false;
    1876             :                 }
    1877             :         }
    1878             : 
    1879           5 :         status = cli_lock32(cli, fnum3, 0, 4, 0, WRITE_LOCK);
    1880           5 :         if (NT_STATUS_IS_OK(status)) {
    1881           0 :                 printf("lock3 succeeded! This is a locking bug\n");
    1882           0 :                 correct = false;
    1883             :         } else {
    1884           5 :                 if (!check_both_error(__LINE__, status, ERRDOS, ERRlock,
    1885           5 :                                       NT_STATUS_LOCK_NOT_GRANTED)) {
    1886           0 :                         return false;
    1887             :                 }
    1888             :         }
    1889             : 
    1890           5 :         cli_setpid(cli, 1);
    1891             : 
    1892           5 :         status = cli_close(cli, fnum1);
    1893           5 :         if (!NT_STATUS_IS_OK(status)) {
    1894           0 :                 printf("close1 failed (%s)\n", nt_errstr(status));
    1895           0 :                 return False;
    1896             :         }
    1897             : 
    1898           5 :         status = cli_close(cli, fnum2);
    1899           5 :         if (!NT_STATUS_IS_OK(status)) {
    1900           0 :                 printf("close2 failed (%s)\n", nt_errstr(status));
    1901           0 :                 return False;
    1902             :         }
    1903             : 
    1904           5 :         status = cli_close(cli, fnum3);
    1905           5 :         if (!NT_STATUS_IS_OK(status)) {
    1906           0 :                 printf("close3 failed (%s)\n", nt_errstr(status));
    1907           0 :                 return False;
    1908             :         }
    1909             : 
    1910           5 :         if (!torture_close_connection(cli)) {
    1911           0 :                 correct = False;
    1912             :         }
    1913             : 
    1914           5 :         printf("locktest2 finished\n");
    1915             : 
    1916           5 :         return correct;
    1917             : }
    1918             : 
    1919             : 
    1920             : /*
    1921             :   This test checks that 
    1922             : 
    1923             :   1) the server supports the full offset range in lock requests
    1924             : */
    1925           5 : static bool run_locktest3(int dummy)
    1926             : {
    1927           0 :         static struct cli_state *cli1, *cli2;
    1928           5 :         const char *fname = "\\lockt3.lck";
    1929           0 :         uint16_t fnum1, fnum2;
    1930           0 :         int i;
    1931           0 :         uint32_t offset;
    1932           5 :         bool correct = True;
    1933           0 :         NTSTATUS status;
    1934             : 
    1935             : #define NEXT_OFFSET offset += (~(uint32_t)0) / torture_numops
    1936             : 
    1937           5 :         if (!torture_open_connection(&cli1, 0) || !torture_open_connection(&cli2, 1)) {
    1938           0 :                 return False;
    1939             :         }
    1940           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    1941           5 :         smbXcli_conn_set_sockopt(cli2->conn, sockops);
    1942             : 
    1943           5 :         printf("starting locktest3\n");
    1944             : 
    1945           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    1946             : 
    1947           5 :         status = cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE,
    1948             :                          &fnum1);
    1949           5 :         if (!NT_STATUS_IS_OK(status)) {
    1950           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    1951           0 :                 return False;
    1952             :         }
    1953             : 
    1954           5 :         status = cli_openx(cli2, fname, O_RDWR, DENY_NONE, &fnum2);
    1955           5 :         if (!NT_STATUS_IS_OK(status)) {
    1956           0 :                 printf("open2 of %s failed (%s)\n", fname, nt_errstr(status));
    1957           0 :                 return False;
    1958             :         }
    1959             : 
    1960         505 :         for (offset=i=0;i<torture_numops;i++) {
    1961         500 :                 NEXT_OFFSET;
    1962             : 
    1963         500 :                 status = cli_lock32(cli1, fnum1, offset-1, 1, 0, WRITE_LOCK);
    1964         500 :                 if (!NT_STATUS_IS_OK(status)) {
    1965           0 :                         printf("lock1 %d failed (%s)\n", 
    1966             :                                i,
    1967             :                                nt_errstr(status));
    1968           0 :                         return False;
    1969             :                 }
    1970             : 
    1971         500 :                 status = cli_lock32(cli2, fnum2, offset-2, 1, 0, WRITE_LOCK);
    1972         500 :                 if (!NT_STATUS_IS_OK(status)) {
    1973           0 :                         printf("lock2 %d failed (%s)\n", 
    1974             :                                i,
    1975             :                                nt_errstr(status));
    1976           0 :                         return False;
    1977             :                 }
    1978             :         }
    1979             : 
    1980         505 :         for (offset=i=0;i<torture_numops;i++) {
    1981         500 :                 NEXT_OFFSET;
    1982             : 
    1983         500 :                 status = cli_lock32(cli1, fnum1, offset-2, 1, 0, WRITE_LOCK);
    1984         500 :                 if (NT_STATUS_IS_OK(status)) {
    1985           0 :                         printf("error: lock1 %d succeeded!\n", i);
    1986           0 :                         return False;
    1987             :                 }
    1988             : 
    1989         500 :                 status = cli_lock32(cli2, fnum2, offset-1, 1, 0, WRITE_LOCK);
    1990         500 :                 if (NT_STATUS_IS_OK(status)) {
    1991           0 :                         printf("error: lock2 %d succeeded!\n", i);
    1992           0 :                         return False;
    1993             :                 }
    1994             : 
    1995         500 :                 status = cli_lock32(cli1, fnum1, offset-1, 1, 0, WRITE_LOCK);
    1996         500 :                 if (NT_STATUS_IS_OK(status)) {
    1997           0 :                         printf("error: lock3 %d succeeded!\n", i);
    1998           0 :                         return False;
    1999             :                 }
    2000             : 
    2001         500 :                 status = cli_lock32(cli2, fnum2, offset-2, 1, 0, WRITE_LOCK);
    2002         500 :                 if (NT_STATUS_IS_OK(status)) {
    2003           0 :                         printf("error: lock4 %d succeeded!\n", i);
    2004           0 :                         return False;
    2005             :                 }
    2006             :         }
    2007             : 
    2008         505 :         for (offset=i=0;i<torture_numops;i++) {
    2009         500 :                 NEXT_OFFSET;
    2010             : 
    2011         500 :                 status = cli_unlock(cli1, fnum1, offset-1, 1);
    2012         500 :                 if (!NT_STATUS_IS_OK(status)) {
    2013           0 :                         printf("unlock1 %d failed (%s)\n", 
    2014             :                                i,
    2015             :                                nt_errstr(status));
    2016           0 :                         return False;
    2017             :                 }
    2018             : 
    2019         500 :                 status = cli_unlock(cli2, fnum2, offset-2, 1);
    2020         500 :                 if (!NT_STATUS_IS_OK(status)) {
    2021           0 :                         printf("unlock2 %d failed (%s)\n", 
    2022             :                                i,
    2023             :                                nt_errstr(status));
    2024           0 :                         return False;
    2025             :                 }
    2026             :         }
    2027             : 
    2028           5 :         status = cli_close(cli1, fnum1);
    2029           5 :         if (!NT_STATUS_IS_OK(status)) {
    2030           0 :                 printf("close1 failed (%s)\n", nt_errstr(status));
    2031           0 :                 return False;
    2032             :         }
    2033             : 
    2034           5 :         status = cli_close(cli2, fnum2);
    2035           5 :         if (!NT_STATUS_IS_OK(status)) {
    2036           0 :                 printf("close2 failed (%s)\n", nt_errstr(status));
    2037           0 :                 return False;
    2038             :         }
    2039             : 
    2040           5 :         status = cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    2041           5 :         if (!NT_STATUS_IS_OK(status)) {
    2042           0 :                 printf("unlink failed (%s)\n", nt_errstr(status));
    2043           0 :                 return False;
    2044             :         }
    2045             : 
    2046           5 :         if (!torture_close_connection(cli1)) {
    2047           0 :                 correct = False;
    2048             :         }
    2049             : 
    2050           5 :         if (!torture_close_connection(cli2)) {
    2051           0 :                 correct = False;
    2052             :         }
    2053             : 
    2054           5 :         printf("finished locktest3\n");
    2055             : 
    2056           5 :         return correct;
    2057             : }
    2058             : 
    2059          40 : static bool test_cli_read(struct cli_state *cli, uint16_t fnum,
    2060             :                            char *buf, off_t offset, size_t size,
    2061             :                            size_t *nread, size_t expect)
    2062             : {
    2063           0 :         NTSTATUS status;
    2064           0 :         size_t l_nread;
    2065             : 
    2066          40 :         status = cli_read(cli, fnum, buf, offset, size, &l_nread);
    2067             : 
    2068          40 :         if(!NT_STATUS_IS_OK(status)) {
    2069          20 :                 return false;
    2070          20 :         } else if (l_nread != expect) {
    2071           0 :                 return false;
    2072             :         }
    2073             : 
    2074          20 :         if (nread) {
    2075           0 :                 *nread = l_nread;
    2076             :         }
    2077             : 
    2078          20 :         return true;
    2079             : }
    2080             : 
    2081             : #define EXPECTED(ret, v) if ((ret) != (v)) { \
    2082             :         printf("** "); correct = False; \
    2083             :         }
    2084             : 
    2085             : /*
    2086             :   looks at overlapping locks
    2087             : */
    2088           5 : static bool run_locktest4(int dummy)
    2089             : {
    2090           0 :         static struct cli_state *cli1, *cli2;
    2091           5 :         const char *fname = "\\lockt4.lck";
    2092           0 :         uint16_t fnum1, fnum2, f;
    2093           0 :         bool ret;
    2094           0 :         char buf[1000];
    2095           5 :         bool correct = True;
    2096           0 :         NTSTATUS status;
    2097             : 
    2098           5 :         if (!torture_open_connection(&cli1, 0) || !torture_open_connection(&cli2, 1)) {
    2099           0 :                 return False;
    2100             :         }
    2101             : 
    2102           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    2103           5 :         smbXcli_conn_set_sockopt(cli2->conn, sockops);
    2104             : 
    2105           5 :         printf("starting locktest4\n");
    2106             : 
    2107           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    2108             : 
    2109           5 :         cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum1);
    2110           5 :         cli_openx(cli2, fname, O_RDWR, DENY_NONE, &fnum2);
    2111             : 
    2112           5 :         memset(buf, 0, sizeof(buf));
    2113             : 
    2114           5 :         status = cli_writeall(cli1, fnum1, 0, (uint8_t *)buf, 0, sizeof(buf),
    2115             :                               NULL);
    2116           5 :         if (!NT_STATUS_IS_OK(status)) {
    2117           0 :                 printf("Failed to create file: %s\n", nt_errstr(status));
    2118           0 :                 correct = False;
    2119           0 :                 goto fail;
    2120             :         }
    2121             : 
    2122          10 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 0, 4, 0, WRITE_LOCK)) &&
    2123           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 2, 4, 0, WRITE_LOCK));
    2124           5 :         EXPECTED(ret, False);
    2125           5 :         printf("the same process %s set overlapping write locks\n", ret?"can":"cannot");
    2126             : 
    2127          10 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 10, 4, 0, READ_LOCK)) &&
    2128           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 12, 4, 0, READ_LOCK));
    2129           5 :         EXPECTED(ret, True);
    2130           5 :         printf("the same process %s set overlapping read locks\n", ret?"can":"cannot");
    2131             : 
    2132          10 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 20, 4, 0, WRITE_LOCK)) &&
    2133           5 :               NT_STATUS_IS_OK(cli_lock32(cli2, fnum2, 22, 4, 0, WRITE_LOCK));
    2134           5 :         EXPECTED(ret, False);
    2135           5 :         printf("a different connection %s set overlapping write locks\n", ret?"can":"cannot");
    2136             : 
    2137          10 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 30, 4, 0, READ_LOCK)) &&
    2138           5 :               NT_STATUS_IS_OK(cli_lock32(cli2, fnum2, 32, 4, 0, READ_LOCK));
    2139           5 :         EXPECTED(ret, True);
    2140           5 :         printf("a different connection %s set overlapping read locks\n", ret?"can":"cannot");
    2141             : 
    2142           5 :         ret = (cli_setpid(cli1, 1),
    2143          10 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 40, 4, 0, WRITE_LOCK))) &&
    2144           5 :               (cli_setpid(cli1, 2),
    2145           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 42, 4, 0, WRITE_LOCK)));
    2146           5 :         EXPECTED(ret, False);
    2147           5 :         printf("a different pid %s set overlapping write locks\n", ret?"can":"cannot");
    2148             : 
    2149           5 :         ret = (cli_setpid(cli1, 1),
    2150          10 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 50, 4, 0, READ_LOCK))) &&
    2151           5 :               (cli_setpid(cli1, 2),
    2152           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 52, 4, 0, READ_LOCK)));
    2153           5 :         EXPECTED(ret, True);
    2154           5 :         printf("a different pid %s set overlapping read locks\n", ret?"can":"cannot");
    2155             : 
    2156          10 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 60, 4, 0, READ_LOCK)) &&
    2157           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 60, 4, 0, READ_LOCK));
    2158           5 :         EXPECTED(ret, True);
    2159           5 :         printf("the same process %s set the same read lock twice\n", ret?"can":"cannot");
    2160             : 
    2161          10 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 70, 4, 0, WRITE_LOCK)) &&
    2162           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 70, 4, 0, WRITE_LOCK));
    2163           5 :         EXPECTED(ret, False);
    2164           5 :         printf("the same process %s set the same write lock twice\n", ret?"can":"cannot");
    2165             : 
    2166          10 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 80, 4, 0, READ_LOCK)) &&
    2167           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 80, 4, 0, WRITE_LOCK));
    2168           5 :         EXPECTED(ret, False);
    2169           5 :         printf("the same process %s overlay a read lock with a write lock\n", ret?"can":"cannot");
    2170             : 
    2171          10 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 90, 4, 0, WRITE_LOCK)) &&
    2172           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 90, 4, 0, READ_LOCK));
    2173           5 :         EXPECTED(ret, True);
    2174           5 :         printf("the same process %s overlay a write lock with a read lock\n", ret?"can":"cannot");
    2175             : 
    2176           5 :         ret = (cli_setpid(cli1, 1),
    2177          10 :              NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 100, 4, 0, WRITE_LOCK))) &&
    2178           5 :              (cli_setpid(cli1, 2),
    2179           5 :              NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 100, 4, 0, READ_LOCK)));
    2180           5 :         EXPECTED(ret, False);
    2181           5 :         printf("a different pid %s overlay a write lock with a read lock\n", ret?"can":"cannot");
    2182             : 
    2183           5 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 110, 4, 0, READ_LOCK)) &&
    2184          10 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 112, 4, 0, READ_LOCK)) &&
    2185           5 :               NT_STATUS_IS_OK(cli_unlock(cli1, fnum1, 110, 6));
    2186           5 :         EXPECTED(ret, False);
    2187           5 :         printf("the same process %s coalesce read locks\n", ret?"can":"cannot");
    2188             : 
    2189             : 
    2190          10 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 120, 4, 0, WRITE_LOCK)) &&
    2191           5 :               test_cli_read(cli2, fnum2, buf, 120, 4, NULL, 4);
    2192           5 :         EXPECTED(ret, False);
    2193           5 :         printf("this server %s strict write locking\n", ret?"doesn't do":"does");
    2194             : 
    2195           5 :         status = cli_lock32(cli1, fnum1, 130, 4, 0, READ_LOCK);
    2196           5 :         ret = NT_STATUS_IS_OK(status);
    2197           5 :         if (ret) {
    2198           5 :                 status = cli_writeall(cli2, fnum2, 0, (uint8_t *)buf, 130, 4,
    2199             :                                       NULL);
    2200           5 :                 ret = NT_STATUS_IS_OK(status);
    2201             :         }
    2202           5 :         EXPECTED(ret, False);
    2203           5 :         printf("this server %s strict read locking\n", ret?"doesn't do":"does");
    2204             : 
    2205             : 
    2206           5 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 140, 4, 0, READ_LOCK)) &&
    2207           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 140, 4, 0, READ_LOCK)) &&
    2208          15 :               NT_STATUS_IS_OK(cli_unlock(cli1, fnum1, 140, 4)) &&
    2209           5 :               NT_STATUS_IS_OK(cli_unlock(cli1, fnum1, 140, 4));
    2210           5 :         EXPECTED(ret, True);
    2211           5 :         printf("this server %s do recursive read locking\n", ret?"does":"doesn't");
    2212             : 
    2213             : 
    2214           5 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 150, 4, 0, WRITE_LOCK)) &&
    2215           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 150, 4, 0, READ_LOCK)) &&
    2216          10 :               NT_STATUS_IS_OK(cli_unlock(cli1, fnum1, 150, 4)) &&
    2217           5 :               test_cli_read(cli2, fnum2, buf, 150, 4, NULL, 4) &&
    2218           5 :               !(NT_STATUS_IS_OK(cli_writeall(cli2, fnum2, 0, (uint8_t *)buf,
    2219          10 :                                              150, 4, NULL))) &&
    2220           5 :               NT_STATUS_IS_OK(cli_unlock(cli1, fnum1, 150, 4));
    2221           5 :         EXPECTED(ret, True);
    2222           5 :         printf("this server %s do recursive lock overlays\n", ret?"does":"doesn't");
    2223             : 
    2224           5 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 160, 4, 0, READ_LOCK)) &&
    2225           5 :               NT_STATUS_IS_OK(cli_unlock(cli1, fnum1, 160, 4)) &&
    2226           5 :               NT_STATUS_IS_OK(cli_writeall(cli2, fnum2, 0, (uint8_t *)buf,
    2227          10 :                                            160, 4, NULL)) &&
    2228           5 :               test_cli_read(cli2, fnum2, buf, 160, 4, NULL, 4);
    2229           5 :         EXPECTED(ret, True);
    2230           5 :         printf("the same process %s remove a read lock using write locking\n", ret?"can":"cannot");
    2231             : 
    2232           5 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 170, 4, 0, WRITE_LOCK)) &&
    2233           5 :               NT_STATUS_IS_OK(cli_unlock(cli1, fnum1, 170, 4)) &&
    2234           5 :               NT_STATUS_IS_OK(cli_writeall(cli2, fnum2, 0, (uint8_t *)buf,
    2235          10 :                                            170, 4, NULL)) &&
    2236           5 :               test_cli_read(cli2, fnum2, buf, 170, 4, NULL, 4);
    2237           5 :         EXPECTED(ret, True);
    2238           5 :         printf("the same process %s remove a write lock using read locking\n", ret?"can":"cannot");
    2239             : 
    2240           5 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 190, 4, 0, WRITE_LOCK)) &&
    2241           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 190, 4, 0, READ_LOCK)) &&
    2242           5 :               NT_STATUS_IS_OK(cli_unlock(cli1, fnum1, 190, 4)) &&
    2243           5 :               !NT_STATUS_IS_OK(cli_writeall(cli2, fnum2, 0, (uint8_t *)buf,
    2244          10 :                                             190, 4, NULL)) &&
    2245           5 :               test_cli_read(cli2, fnum2, buf, 190, 4, NULL, 4);
    2246           5 :         EXPECTED(ret, True);
    2247           5 :         printf("the same process %s remove the first lock first\n", ret?"does":"doesn't");
    2248             : 
    2249           5 :         cli_close(cli1, fnum1);
    2250           5 :         cli_close(cli2, fnum2);
    2251           5 :         cli_openx(cli1, fname, O_RDWR, DENY_NONE, &fnum1);
    2252           5 :         cli_openx(cli1, fname, O_RDWR, DENY_NONE, &f);
    2253           5 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 0, 8, 0, READ_LOCK)) &&
    2254           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, f, 0, 1, 0, READ_LOCK)) &&
    2255           5 :               NT_STATUS_IS_OK(cli_close(cli1, fnum1)) &&
    2256          15 :               NT_STATUS_IS_OK(cli_openx(cli1, fname, O_RDWR, DENY_NONE, &fnum1)) &&
    2257           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 7, 1, 0, WRITE_LOCK));
    2258           5 :         cli_close(cli1, f);
    2259           5 :         cli_close(cli1, fnum1);
    2260           5 :         EXPECTED(ret, True);
    2261           5 :         printf("the server %s have the NT byte range lock bug\n", !ret?"does":"doesn't");
    2262             : 
    2263           5 :  fail:
    2264           5 :         cli_close(cli1, fnum1);
    2265           5 :         cli_close(cli2, fnum2);
    2266           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    2267           5 :         torture_close_connection(cli1);
    2268           5 :         torture_close_connection(cli2);
    2269             : 
    2270           5 :         printf("finished locktest4\n");
    2271           5 :         return correct;
    2272             : }
    2273             : 
    2274             : /*
    2275             :   looks at lock upgrade/downgrade.
    2276             : */
    2277           5 : static bool run_locktest5(int dummy)
    2278             : {
    2279           0 :         static struct cli_state *cli1, *cli2;
    2280           5 :         const char *fname = "\\lockt5.lck";
    2281           0 :         uint16_t fnum1, fnum2, fnum3;
    2282           0 :         bool ret;
    2283           0 :         char buf[1000];
    2284           5 :         bool correct = True;
    2285           0 :         NTSTATUS status;
    2286             : 
    2287           5 :         if (!torture_open_connection(&cli1, 0) || !torture_open_connection(&cli2, 1)) {
    2288           0 :                 return False;
    2289             :         }
    2290             : 
    2291           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    2292           5 :         smbXcli_conn_set_sockopt(cli2->conn, sockops);
    2293             : 
    2294           5 :         printf("starting locktest5\n");
    2295             : 
    2296           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    2297             : 
    2298           5 :         cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum1);
    2299           5 :         cli_openx(cli2, fname, O_RDWR, DENY_NONE, &fnum2);
    2300           5 :         cli_openx(cli1, fname, O_RDWR, DENY_NONE, &fnum3);
    2301             : 
    2302           5 :         memset(buf, 0, sizeof(buf));
    2303             : 
    2304           5 :         status = cli_writeall(cli1, fnum1, 0, (uint8_t *)buf, 0, sizeof(buf),
    2305             :                               NULL);
    2306           5 :         if (!NT_STATUS_IS_OK(status)) {
    2307           0 :                 printf("Failed to create file: %s\n", nt_errstr(status));
    2308           0 :                 correct = False;
    2309           0 :                 goto fail;
    2310             :         }
    2311             : 
    2312             :         /* Check for NT bug... */
    2313          10 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 0, 8, 0, READ_LOCK)) &&
    2314           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum3, 0, 1, 0, READ_LOCK));
    2315           5 :         cli_close(cli1, fnum1);
    2316           5 :         cli_openx(cli1, fname, O_RDWR, DENY_NONE, &fnum1);
    2317           5 :         status = cli_lock32(cli1, fnum1, 7, 1, 0, WRITE_LOCK);
    2318           5 :         ret = NT_STATUS_IS_OK(status);
    2319           5 :         EXPECTED(ret, True);
    2320           5 :         printf("this server %s the NT locking bug\n", ret ? "doesn't have" : "has");
    2321           5 :         cli_close(cli1, fnum1);
    2322           5 :         cli_openx(cli1, fname, O_RDWR, DENY_NONE, &fnum1);
    2323           5 :         cli_unlock(cli1, fnum3, 0, 1);
    2324             : 
    2325          10 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 0, 4, 0, WRITE_LOCK)) &&
    2326           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 1, 1, 0, READ_LOCK));
    2327           5 :         EXPECTED(ret, True);
    2328           5 :         printf("the same process %s overlay a write with a read lock\n", ret?"can":"cannot");
    2329             : 
    2330           5 :         status = cli_lock32(cli2, fnum2, 0, 4, 0, READ_LOCK);
    2331           5 :         ret = NT_STATUS_IS_OK(status);
    2332           5 :         EXPECTED(ret, False);
    2333             : 
    2334           5 :         printf("a different process %s get a read lock on the first process lock stack\n", ret?"can":"cannot");
    2335             : 
    2336             :         /* Unlock the process 2 lock. */
    2337           5 :         cli_unlock(cli2, fnum2, 0, 4);
    2338             : 
    2339           5 :         status = cli_lock32(cli1, fnum3, 0, 4, 0, READ_LOCK);
    2340           5 :         ret = NT_STATUS_IS_OK(status);
    2341           5 :         EXPECTED(ret, False);
    2342             : 
    2343           5 :         printf("the same process on a different fnum %s get a read lock\n", ret?"can":"cannot");
    2344             : 
    2345             :         /* Unlock the process 1 fnum3 lock. */
    2346           5 :         cli_unlock(cli1, fnum3, 0, 4);
    2347             : 
    2348             :         /* Stack 2 more locks here. */
    2349          10 :         ret = NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 0, 4, 0, READ_LOCK)) &&
    2350           5 :               NT_STATUS_IS_OK(cli_lock32(cli1, fnum1, 0, 4, 0, READ_LOCK));
    2351             : 
    2352           5 :         EXPECTED(ret, True);
    2353           5 :         printf("the same process %s stack read locks\n", ret?"can":"cannot");
    2354             : 
    2355             :         /* Unlock the first process lock, then check this was the WRITE lock that was
    2356             :                 removed. */
    2357             : 
    2358          10 :         ret = NT_STATUS_IS_OK(cli_unlock(cli1, fnum1, 0, 4)) &&
    2359           5 :               NT_STATUS_IS_OK(cli_lock32(cli2, fnum2, 0, 4, 0, READ_LOCK));
    2360             : 
    2361           5 :         EXPECTED(ret, True);
    2362           5 :         printf("the first unlock removes the %s lock\n", ret?"WRITE":"READ");
    2363             : 
    2364             :         /* Unlock the process 2 lock. */
    2365           5 :         cli_unlock(cli2, fnum2, 0, 4);
    2366             : 
    2367             :         /* We should have 3 stacked locks here. Ensure we need to do 3 unlocks. */
    2368             : 
    2369           5 :         ret = NT_STATUS_IS_OK(cli_unlock(cli1, fnum1, 1, 1)) &&
    2370          10 :                   NT_STATUS_IS_OK(cli_unlock(cli1, fnum1, 0, 4)) &&
    2371           5 :                   NT_STATUS_IS_OK(cli_unlock(cli1, fnum1, 0, 4));
    2372             : 
    2373           5 :         EXPECTED(ret, True);
    2374           5 :         printf("the same process %s unlock the stack of 4 locks\n", ret?"can":"cannot"); 
    2375             : 
    2376             :         /* Ensure the next unlock fails. */
    2377           5 :         ret = NT_STATUS_IS_OK(cli_unlock(cli1, fnum1, 0, 4));
    2378           5 :         EXPECTED(ret, False);
    2379           5 :         printf("the same process %s count the lock stack\n", !ret?"can":"cannot"); 
    2380             : 
    2381             :         /* Ensure connection 2 can get a write lock. */
    2382           5 :         status = cli_lock32(cli2, fnum2, 0, 4, 0, WRITE_LOCK);
    2383           5 :         ret = NT_STATUS_IS_OK(status);
    2384           5 :         EXPECTED(ret, True);
    2385             : 
    2386           5 :         printf("a different process %s get a write lock on the unlocked stack\n", ret?"can":"cannot");
    2387             : 
    2388             : 
    2389           5 :  fail:
    2390           5 :         cli_close(cli1, fnum1);
    2391           5 :         cli_close(cli2, fnum2);
    2392           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    2393           5 :         if (!torture_close_connection(cli1)) {
    2394           0 :                 correct = False;
    2395             :         }
    2396           5 :         if (!torture_close_connection(cli2)) {
    2397           0 :                 correct = False;
    2398             :         }
    2399             : 
    2400           5 :         printf("finished locktest5\n");
    2401             : 
    2402           5 :         return correct;
    2403             : }
    2404             : 
    2405             : /*
    2406             :   tries the unusual lockingX locktype bits
    2407             : */
    2408           5 : static bool run_locktest6(int dummy)
    2409             : {
    2410           0 :         static struct cli_state *cli;
    2411           5 :         const char *fname[1] = { "\\lock6.txt" };
    2412           0 :         int i;
    2413           0 :         uint16_t fnum;
    2414           0 :         NTSTATUS status;
    2415             : 
    2416           5 :         if (!torture_open_connection(&cli, 0)) {
    2417           0 :                 return False;
    2418             :         }
    2419             : 
    2420           5 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    2421             : 
    2422           5 :         printf("starting locktest6\n");
    2423             : 
    2424          10 :         for (i=0;i<1;i++) {
    2425           5 :                 printf("Testing %s\n", fname[i]);
    2426             : 
    2427           5 :                 cli_unlink(cli, fname[i], FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    2428             : 
    2429           5 :                 cli_openx(cli, fname[i], O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum);
    2430           5 :                 status = cli_locktype(cli, fnum, 0, 8, 0, LOCKING_ANDX_CHANGE_LOCKTYPE);
    2431           5 :                 cli_close(cli, fnum);
    2432           5 :                 printf("CHANGE_LOCKTYPE gave %s\n", nt_errstr(status));
    2433             : 
    2434           5 :                 cli_openx(cli, fname[i], O_RDWR, DENY_NONE, &fnum);
    2435           5 :                 status = cli_locktype(cli, fnum, 0, 8, 0, LOCKING_ANDX_CANCEL_LOCK);
    2436           5 :                 cli_close(cli, fnum);
    2437           5 :                 printf("CANCEL_LOCK gave %s\n", nt_errstr(status));
    2438             : 
    2439           5 :                 cli_unlink(cli, fname[i], FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    2440             :         }
    2441             : 
    2442           5 :         torture_close_connection(cli);
    2443             : 
    2444           5 :         printf("finished locktest6\n");
    2445           5 :         return True;
    2446             : }
    2447             : 
    2448           5 : static bool run_locktest7(int dummy)
    2449             : {
    2450           0 :         struct cli_state *cli1;
    2451           5 :         const char *fname = "\\lockt7.lck";
    2452           0 :         uint16_t fnum1;
    2453           0 :         char buf[200];
    2454           5 :         bool correct = False;
    2455           0 :         size_t nread;
    2456           0 :         NTSTATUS status;
    2457             : 
    2458           5 :         if (!torture_open_connection(&cli1, 0)) {
    2459           0 :                 return False;
    2460             :         }
    2461             : 
    2462           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    2463             : 
    2464           5 :         printf("starting locktest7\n");
    2465             : 
    2466           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    2467             : 
    2468           5 :         cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum1);
    2469             : 
    2470           5 :         memset(buf, 0, sizeof(buf));
    2471             : 
    2472           5 :         status = cli_writeall(cli1, fnum1, 0, (uint8_t *)buf, 0, sizeof(buf),
    2473             :                               NULL);
    2474           5 :         if (!NT_STATUS_IS_OK(status)) {
    2475           0 :                 printf("Failed to create file: %s\n", nt_errstr(status));
    2476           0 :                 goto fail;
    2477             :         }
    2478             : 
    2479           5 :         cli_setpid(cli1, 1);
    2480             : 
    2481           5 :         status = cli_lock32(cli1, fnum1, 130, 4, 0, READ_LOCK);
    2482           5 :         if (!NT_STATUS_IS_OK(status)) {
    2483           0 :                 printf("Unable to apply read lock on range 130:4, "
    2484             :                        "error was %s\n", nt_errstr(status));
    2485           0 :                 goto fail;
    2486             :         } else {
    2487           5 :                 printf("pid1 successfully locked range 130:4 for READ\n");
    2488             :         }
    2489             : 
    2490           5 :         status = cli_read(cli1, fnum1, buf, 130, 4, &nread);
    2491           5 :         if (!NT_STATUS_IS_OK(status)) {
    2492           0 :                 printf("pid1 unable to read the range 130:4, error was %s\n",
    2493             :                       nt_errstr(status));
    2494           0 :                 goto fail;
    2495           5 :         } else if (nread != 4) {
    2496           0 :                 printf("pid1 unable to read the range 130:4, "
    2497             :                        "recv %ld req %d\n", (unsigned long)nread, 4);
    2498           0 :                 goto fail;
    2499             :         } else {
    2500           5 :                 printf("pid1 successfully read the range 130:4\n");
    2501             :         }
    2502             : 
    2503           5 :         status = cli_writeall(cli1, fnum1, 0, (uint8_t *)buf, 130, 4, NULL);
    2504           5 :         if (!NT_STATUS_IS_OK(status)) {
    2505           5 :                 printf("pid1 unable to write to the range 130:4, error was "
    2506             :                        "%s\n", nt_errstr(status));
    2507           5 :                 if (!NT_STATUS_EQUAL(status, NT_STATUS_FILE_LOCK_CONFLICT)) {
    2508           0 :                         printf("Incorrect error (should be NT_STATUS_FILE_LOCK_CONFLICT)\n");
    2509           0 :                         goto fail;
    2510             :                 }
    2511             :         } else {
    2512           0 :                 printf("pid1 successfully wrote to the range 130:4 (should be denied)\n");
    2513           0 :                 goto fail;
    2514             :         }
    2515             : 
    2516           5 :         cli_setpid(cli1, 2);
    2517             : 
    2518           5 :         status = cli_read(cli1, fnum1, buf, 130, 4, &nread);
    2519           5 :         if (!NT_STATUS_IS_OK(status)) {
    2520           0 :                 printf("pid2 unable to read the range 130:4, error was %s\n",
    2521             :                       nt_errstr(status));
    2522           0 :                 goto fail;
    2523           5 :         } else if (nread != 4) {
    2524           0 :                 printf("pid2 unable to read the range 130:4, "
    2525             :                        "recv %ld req %d\n", (unsigned long)nread, 4);
    2526           0 :                 goto fail;
    2527             :         } else {
    2528           5 :                 printf("pid2 successfully read the range 130:4\n");
    2529             :         }
    2530             : 
    2531           5 :         status = cli_writeall(cli1, fnum1, 0, (uint8_t *)buf, 130, 4, NULL);
    2532           5 :         if (!NT_STATUS_IS_OK(status)) {
    2533           5 :                 printf("pid2 unable to write to the range 130:4, error was "
    2534             :                        "%s\n", nt_errstr(status));
    2535           5 :                 if (!NT_STATUS_EQUAL(status, NT_STATUS_FILE_LOCK_CONFLICT)) {
    2536           0 :                         printf("Incorrect error (should be NT_STATUS_FILE_LOCK_CONFLICT)\n");
    2537           0 :                         goto fail;
    2538             :                 }
    2539             :         } else {
    2540           0 :                 printf("pid2 successfully wrote to the range 130:4 (should be denied)\n");
    2541           0 :                 goto fail;
    2542             :         }
    2543             : 
    2544           5 :         cli_setpid(cli1, 1);
    2545           5 :         cli_unlock(cli1, fnum1, 130, 4);
    2546             : 
    2547           5 :         status = cli_lock32(cli1, fnum1, 130, 4, 0, WRITE_LOCK);
    2548           5 :         if (!NT_STATUS_IS_OK(status)) {
    2549           0 :                 printf("Unable to apply write lock on range 130:4, error was %s\n", nt_errstr(status));
    2550           0 :                 goto fail;
    2551             :         } else {
    2552           5 :                 printf("pid1 successfully locked range 130:4 for WRITE\n");
    2553             :         }
    2554             : 
    2555           5 :         status = cli_read(cli1, fnum1, buf, 130, 4, &nread);
    2556           5 :         if (!NT_STATUS_IS_OK(status)) {
    2557           0 :                 printf("pid1 unable to read the range 130:4, error was %s\n",
    2558             :                       nt_errstr(status));
    2559           0 :                 goto fail;
    2560           5 :         } else if (nread != 4) {
    2561           0 :                 printf("pid1 unable to read the range 130:4, "
    2562             :                        "recv %ld req %d\n", (unsigned long)nread, 4);
    2563           0 :                 goto fail;
    2564             :         } else {
    2565           5 :                 printf("pid1 successfully read the range 130:4\n");
    2566             :         }
    2567             : 
    2568           5 :         status = cli_writeall(cli1, fnum1, 0, (uint8_t *)buf, 130, 4, NULL);
    2569           5 :         if (!NT_STATUS_IS_OK(status)) {
    2570           0 :                 printf("pid1 unable to write to the range 130:4, error was "
    2571             :                        "%s\n", nt_errstr(status));
    2572           0 :                 goto fail;
    2573             :         } else {
    2574           5 :                 printf("pid1 successfully wrote to the range 130:4\n");
    2575             :         }
    2576             : 
    2577           5 :         cli_setpid(cli1, 2);
    2578             : 
    2579           5 :         status = cli_read(cli1, fnum1, buf, 130, 4, &nread);
    2580           5 :         if (!NT_STATUS_IS_OK(status)) {
    2581           5 :                 printf("pid2 unable to read the range 130:4, error was "
    2582             :                        "%s\n", nt_errstr(status));
    2583           5 :                 if (!NT_STATUS_EQUAL(status, NT_STATUS_FILE_LOCK_CONFLICT)) {
    2584           0 :                         printf("Incorrect error (should be NT_STATUS_FILE_LOCK_CONFLICT)\n");
    2585           0 :                         goto fail;
    2586             :                 }
    2587             :         } else {
    2588           0 :                 printf("pid2 successfully read the range 130:4 (should be denied) recv %ld\n",
    2589             :                        (unsigned long)nread);
    2590           0 :                 goto fail;
    2591             :         }
    2592             : 
    2593           5 :         status = cli_writeall(cli1, fnum1, 0, (uint8_t *)buf, 130, 4, NULL);
    2594           5 :         if (!NT_STATUS_IS_OK(status)) {
    2595           5 :                 printf("pid2 unable to write to the range 130:4, error was "
    2596             :                        "%s\n", nt_errstr(status));
    2597           5 :                 if (!NT_STATUS_EQUAL(status, NT_STATUS_FILE_LOCK_CONFLICT)) {
    2598           0 :                         printf("Incorrect error (should be NT_STATUS_FILE_LOCK_CONFLICT)\n");
    2599           0 :                         goto fail;
    2600             :                 }
    2601             :         } else {
    2602           0 :                 printf("pid2 successfully wrote to the range 130:4 (should be denied)\n");
    2603           0 :                 goto fail;
    2604             :         }
    2605             : 
    2606           5 :         cli_unlock(cli1, fnum1, 130, 0);
    2607           5 :         correct = True;
    2608             : 
    2609           5 : fail:
    2610           5 :         cli_close(cli1, fnum1);
    2611           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    2612           5 :         torture_close_connection(cli1);
    2613             : 
    2614           5 :         printf("finished locktest7\n");
    2615           5 :         return correct;
    2616             : }
    2617             : 
    2618             : /*
    2619             :  * This demonstrates a problem with our use of GPFS share modes: A file
    2620             :  * descriptor sitting in the pending close queue holding a GPFS share mode
    2621             :  * blocks opening a file another time. Happens with Word 2007 temp files.
    2622             :  * With "posix locking = yes" and "gpfs:sharemodes = yes" enabled, the third
    2623             :  * open is denied with NT_STATUS_SHARING_VIOLATION.
    2624             :  */
    2625             : 
    2626           0 : static bool run_locktest8(int dummy)
    2627             : {
    2628           0 :         struct cli_state *cli1;
    2629           0 :         const char *fname = "\\lockt8.lck";
    2630           0 :         uint16_t fnum1, fnum2;
    2631           0 :         char buf[200];
    2632           0 :         bool correct = False;
    2633           0 :         NTSTATUS status;
    2634             : 
    2635           0 :         if (!torture_open_connection(&cli1, 0)) {
    2636           0 :                 return False;
    2637             :         }
    2638             : 
    2639           0 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    2640             : 
    2641           0 :         printf("starting locktest8\n");
    2642             : 
    2643           0 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    2644             : 
    2645           0 :         status = cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_WRITE,
    2646             :                           &fnum1);
    2647           0 :         if (!NT_STATUS_IS_OK(status)) {
    2648           0 :                 d_fprintf(stderr, "cli_openx returned %s\n", nt_errstr(status));
    2649           0 :                 return false;
    2650             :         }
    2651             : 
    2652           0 :         memset(buf, 0, sizeof(buf));
    2653             : 
    2654           0 :         status = cli_openx(cli1, fname, O_RDONLY, DENY_NONE, &fnum2);
    2655           0 :         if (!NT_STATUS_IS_OK(status)) {
    2656           0 :                 d_fprintf(stderr, "cli_openx second time returned %s\n",
    2657             :                           nt_errstr(status));
    2658           0 :                 goto fail;
    2659             :         }
    2660             : 
    2661           0 :         status = cli_lock32(cli1, fnum2, 1, 1, 0, READ_LOCK);
    2662           0 :         if (!NT_STATUS_IS_OK(status)) {
    2663           0 :                 printf("Unable to apply read lock on range 1:1, error was "
    2664             :                        "%s\n", nt_errstr(status));
    2665           0 :                 goto fail;
    2666             :         }
    2667             : 
    2668           0 :         status = cli_close(cli1, fnum1);
    2669           0 :         if (!NT_STATUS_IS_OK(status)) {
    2670           0 :                 d_fprintf(stderr, "cli_close(fnum1) %s\n", nt_errstr(status));
    2671           0 :                 goto fail;
    2672             :         }
    2673             : 
    2674           0 :         status = cli_openx(cli1, fname, O_RDWR, DENY_NONE, &fnum1);
    2675           0 :         if (!NT_STATUS_IS_OK(status)) {
    2676           0 :                 d_fprintf(stderr, "cli_openx third time returned %s\n",
    2677             :                           nt_errstr(status));
    2678           0 :                 goto fail;
    2679             :         }
    2680             : 
    2681           0 :         correct = true;
    2682             : 
    2683           0 : fail:
    2684           0 :         cli_close(cli1, fnum1);
    2685           0 :         cli_close(cli1, fnum2);
    2686           0 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    2687           0 :         torture_close_connection(cli1);
    2688             : 
    2689           0 :         printf("finished locktest8\n");
    2690           0 :         return correct;
    2691             : }
    2692             : 
    2693             : /*
    2694             :  * This test is designed to be run in conjunction with
    2695             :  * external NFS or POSIX locks taken in the filesystem.
    2696             :  * It checks that the smbd server will block until the
    2697             :  * lock is released and then acquire it. JRA.
    2698             :  */
    2699             : 
    2700             : static bool got_alarm;
    2701             : static struct cli_state *alarm_cli;
    2702             : 
    2703           0 : static void alarm_handler(int dummy)
    2704             : {
    2705           0 :         got_alarm = True;
    2706           0 : }
    2707             : 
    2708           0 : static void alarm_handler_parent(int dummy)
    2709             : {
    2710           0 :         smbXcli_conn_disconnect(alarm_cli->conn, NT_STATUS_LOCAL_DISCONNECT);
    2711           0 : }
    2712             : 
    2713           8 : static void do_local_lock(const char *fname, int read_fd, int write_fd)
    2714             : {
    2715           0 :         int fd;
    2716           8 :         char c = '\0';
    2717           0 :         struct flock lock;
    2718           8 :         const char *local_pathname = NULL;
    2719           0 :         int ret;
    2720             : 
    2721           8 :         local_pathname = talloc_asprintf(talloc_tos(),
    2722             :                         "%s/%s", local_path, fname);
    2723           8 :         if (!local_pathname) {
    2724           0 :                 printf("child: alloc fail\n");
    2725           0 :                 exit(1);
    2726             :         }
    2727             : 
    2728           8 :         unlink(local_pathname);
    2729           8 :         fd = open(local_pathname, O_RDWR|O_CREAT, 0666);
    2730           8 :         if (fd == -1) {
    2731           0 :                 printf("child: open of %s failed %s.\n",
    2732           0 :                         local_pathname, strerror(errno));
    2733           0 :                 exit(1);
    2734             :         }
    2735             : 
    2736             :         /* Now take a fcntl lock. */
    2737           8 :         lock.l_type = F_WRLCK;
    2738           8 :         lock.l_whence = SEEK_SET;
    2739           8 :         lock.l_start = 0;
    2740           8 :         lock.l_len = 4;
    2741           8 :         lock.l_pid = getpid();
    2742             : 
    2743           8 :         ret = fcntl(fd,F_SETLK,&lock);
    2744           8 :         if (ret == -1) {
    2745           0 :                 printf("child: failed to get lock 0:4 on file %s. Error %s\n",
    2746           0 :                         local_pathname, strerror(errno));
    2747           0 :                 exit(1);
    2748             :         } else {
    2749           8 :                 printf("child: got lock 0:4 on file %s.\n",
    2750             :                         local_pathname );
    2751           8 :                 fflush(stdout);
    2752             :         }
    2753             : 
    2754           8 :         CatchSignal(SIGALRM, alarm_handler);
    2755           8 :         alarm(5);
    2756             :         /* Signal the parent. */
    2757           8 :         if (write(write_fd, &c, 1) != 1) {
    2758           0 :                 printf("child: start signal fail %s.\n",
    2759           0 :                         strerror(errno));
    2760           0 :                 exit(1);
    2761             :         }
    2762           8 :         alarm(0);
    2763             : 
    2764           8 :         alarm(10);
    2765             :         /* Wait for the parent to be ready. */
    2766           8 :         if (read(read_fd, &c, 1) != 1) {
    2767           0 :                 printf("child: reply signal fail %s.\n",
    2768           0 :                         strerror(errno));
    2769           0 :                 exit(1);
    2770             :         }
    2771           8 :         alarm(0);
    2772             : 
    2773           8 :         sleep(5);
    2774           8 :         close(fd);
    2775           8 :         printf("child: released lock 0:4 on file %s.\n",
    2776             :                 local_pathname );
    2777           8 :         fflush(stdout);
    2778           8 :         exit(0);
    2779             : }
    2780             : 
    2781           8 : static bool _run_locktest9X(const char *fname, int timeout)
    2782             : {
    2783           0 :         struct cli_state *cli1;
    2784           8 :         char *fpath = talloc_asprintf(talloc_tos(), "\\%s", fname);
    2785           0 :         uint16_t fnum;
    2786           8 :         bool correct = False;
    2787           0 :         int pipe_in[2], pipe_out[2];
    2788           0 :         pid_t child_pid;
    2789           8 :         char c = '\0';
    2790           0 :         int ret;
    2791           0 :         struct timeval start;
    2792           0 :         double seconds;
    2793           0 :         NTSTATUS status;
    2794             : 
    2795           8 :         printf("starting locktest9X: %s\n", fname);
    2796             : 
    2797           8 :         if (local_path == NULL) {
    2798           0 :                 d_fprintf(stderr, "locktest9X must be given a local path via -l <localpath>\n");
    2799           0 :                 return false;
    2800             :         }
    2801             : 
    2802           8 :         if (pipe(pipe_in) == -1 || pipe(pipe_out) == -1) {
    2803           0 :                 return false;
    2804             :         }
    2805             : 
    2806           8 :         child_pid = fork();
    2807          16 :         if (child_pid == -1) {
    2808           0 :                 return false;
    2809             :         }
    2810             : 
    2811          16 :         if (child_pid == 0) {
    2812             :                 /* Child. */
    2813           8 :                 do_local_lock(fname, pipe_out[0], pipe_in[1]);
    2814           0 :                 exit(0);
    2815             :         }
    2816             : 
    2817           8 :         close(pipe_out[0]);
    2818           8 :         close(pipe_in[1]);
    2819           8 :         pipe_out[0] = -1;
    2820           8 :         pipe_in[1] = -1;
    2821             : 
    2822             :         /* Parent. */
    2823           8 :         ret = read(pipe_in[0], &c, 1);
    2824           8 :         if (ret != 1) {
    2825           0 :                 d_fprintf(stderr, "failed to read start signal from child. %s\n",
    2826           0 :                         strerror(errno));
    2827           0 :                 return false;
    2828             :         }
    2829             : 
    2830           8 :         if (!torture_open_connection(&cli1, 0)) {
    2831           0 :                 return false;
    2832             :         }
    2833             : 
    2834           8 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    2835             : 
    2836           8 :         status = cli_openx(cli1, fpath, O_RDWR, DENY_NONE,
    2837             :                           &fnum);
    2838           8 :         if (!NT_STATUS_IS_OK(status)) {
    2839           0 :                 d_fprintf(stderr, "cli_openx returned %s\n", nt_errstr(status));
    2840           0 :                 return false;
    2841             :         }
    2842             : 
    2843             :         /* Ensure the child has the lock. */
    2844           8 :         status = cli_lock32(cli1, fnum, 0, 4, 0, WRITE_LOCK);
    2845           8 :         if (NT_STATUS_IS_OK(status)) {
    2846           0 :                 d_fprintf(stderr, "Got the lock on range 0:4 - this should not happen !\n");
    2847           0 :                 goto fail;
    2848             :         } else {
    2849           8 :                 d_printf("Child has the lock.\n");
    2850             :         }
    2851             : 
    2852             :         /* Tell the child to wait 5 seconds then exit. */
    2853           8 :         ret = write(pipe_out[1], &c, 1);
    2854           8 :         if (ret != 1) {
    2855           0 :                 d_fprintf(stderr, "failed to send exit signal to child. %s\n",
    2856           0 :                         strerror(errno));
    2857           0 :                 goto fail;
    2858             :         }
    2859             : 
    2860             :         /* Wait 20 seconds for the lock. */
    2861           8 :         alarm_cli = cli1;
    2862           8 :         CatchSignal(SIGALRM, alarm_handler_parent);
    2863           8 :         alarm(20);
    2864             : 
    2865           8 :         start = timeval_current();
    2866             : 
    2867           8 :         status = cli_lock32(cli1, fnum, 0, 4, timeout, WRITE_LOCK);
    2868           8 :         if (!NT_STATUS_IS_OK(status)) {
    2869           0 :                 d_fprintf(stderr, "Unable to apply write lock on range 0:4, error was "
    2870             :                        "%s\n", nt_errstr(status));
    2871           0 :                 goto fail_nofd;
    2872             :         }
    2873           8 :         alarm(0);
    2874             : 
    2875           8 :         seconds = timeval_elapsed(&start);
    2876             : 
    2877           8 :         printf("Parent got the lock after %.2f seconds.\n",
    2878             :                 seconds);
    2879             : 
    2880           8 :         status = cli_close(cli1, fnum);
    2881           8 :         if (!NT_STATUS_IS_OK(status)) {
    2882           0 :                 d_fprintf(stderr, "cli_close(fnum1) %s\n", nt_errstr(status));
    2883           0 :                 goto fail;
    2884             :         }
    2885             : 
    2886           8 :         correct = true;
    2887             : 
    2888           8 : fail:
    2889           8 :         cli_close(cli1, fnum);
    2890           8 :         torture_close_connection(cli1);
    2891             : 
    2892           8 : fail_nofd:
    2893             : 
    2894           8 :         printf("finished locktest9X: %s\n", fname);
    2895           8 :         return correct;
    2896             : }
    2897             : 
    2898           4 : static bool run_locktest9a(int dummy)
    2899             : {
    2900           4 :         return _run_locktest9X("lock9a.dat", -1);
    2901             : }
    2902             : 
    2903           4 : static bool run_locktest9b(int dummy)
    2904             : {
    2905           4 :         return _run_locktest9X("lock9b.dat", 10000);
    2906             : }
    2907             : 
    2908             : struct locktest10_state {
    2909             :         bool ok;
    2910             :         bool done;
    2911             : };
    2912             : 
    2913             : static void locktest10_lockingx_done(struct tevent_req *subreq);
    2914             : static void locktest10_read_andx_done(struct tevent_req *subreq);
    2915             : 
    2916           5 : static bool run_locktest10(int dummy)
    2917             : {
    2918           5 :         struct tevent_context *ev = NULL;
    2919           5 :         struct cli_state *cli1 = NULL;
    2920           5 :         struct cli_state *cli2 = NULL;
    2921           5 :         struct smb1_lock_element lck = { 0 };
    2922           5 :         struct tevent_req *reqs[2] = { NULL };
    2923           5 :         struct tevent_req *smbreqs[2] = { NULL };
    2924           5 :         const char fname[] = "\\lockt10.lck";
    2925           0 :         uint16_t fnum1, fnum2;
    2926           5 :         bool ret = false;
    2927           0 :         bool ok;
    2928           5 :         uint8_t data = 1;
    2929           5 :         struct locktest10_state state = { .ok = true };
    2930           0 :         NTSTATUS status;
    2931             : 
    2932           5 :         printf("starting locktest10\n");
    2933             : 
    2934           5 :         ev = samba_tevent_context_init(NULL);
    2935           5 :         if (ev == NULL) {
    2936           0 :                 d_fprintf(stderr, "samba_tevent_context_init failed\n");
    2937           0 :                 goto done;
    2938             :         }
    2939             : 
    2940           5 :         ok = torture_open_connection(&cli1, 0);
    2941           5 :         if (!ok) {
    2942           0 :                 goto done;
    2943             :         }
    2944           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    2945             : 
    2946           5 :         ok = torture_open_connection(&cli2, 1);
    2947           5 :         if (!ok) {
    2948           0 :                 goto done;
    2949             :         }
    2950           5 :         smbXcli_conn_set_sockopt(cli2->conn, sockops);
    2951             : 
    2952           5 :         status = cli_openx(cli1, fname, O_CREAT|O_RDWR, DENY_NONE, &fnum1);
    2953           5 :         if (!NT_STATUS_IS_OK(status)) {
    2954           0 :                 d_fprintf(stderr,
    2955             :                           "cli_openx failed: %s\n",
    2956             :                           nt_errstr(status));
    2957           0 :                 goto done;
    2958             :         }
    2959             : 
    2960           5 :         status = cli_writeall(cli1, fnum1, 0, &data, 0, sizeof(data), NULL);
    2961           5 :         if (!NT_STATUS_IS_OK(status)) {
    2962           0 :                 d_fprintf(stderr,
    2963             :                           "cli_writeall failed: %s\n",
    2964             :                           nt_errstr(status));
    2965           0 :                 goto done;
    2966             :         }
    2967             : 
    2968           5 :         status = cli_openx(cli2, fname, O_CREAT|O_RDWR, DENY_NONE, &fnum2);
    2969           5 :         if (!NT_STATUS_IS_OK(status)) {
    2970           0 :                 d_fprintf(stderr,
    2971             :                           "cli_openx failed: %s\n",
    2972             :                           nt_errstr(status));
    2973           0 :                 goto done;
    2974             :         }
    2975             : 
    2976           5 :         status = cli_locktype(
    2977             :                 cli2, fnum2, 0, 1, 0, LOCKING_ANDX_EXCLUSIVE_LOCK);
    2978           5 :         if (!NT_STATUS_IS_OK(status)) {
    2979           0 :                 d_fprintf(stderr,
    2980             :                           "cli_locktype failed: %s\n",
    2981             :                           nt_errstr(status));
    2982           0 :                 goto done;
    2983             :         }
    2984             : 
    2985           5 :         lck = (struct smb1_lock_element) {
    2986           5 :                 .pid = cli_getpid(cli1), .offset = 0, .length = 1,
    2987             :         };
    2988             : 
    2989           5 :         reqs[0] = cli_lockingx_create(
    2990             :                 ev,                             /* mem_ctx */
    2991             :                 ev,                             /* tevent_context */
    2992             :                 cli1,                           /* cli */
    2993             :                 fnum1,                          /* fnum */
    2994             :                 LOCKING_ANDX_EXCLUSIVE_LOCK,    /* typeoflock */
    2995             :                 0,                              /* newoplocklevel */
    2996             :                 1,                              /* timeout */
    2997             :                 0,                              /* num_unlocks */
    2998             :                 NULL,                           /* unlocks */
    2999             :                 1,                              /* num_locks */
    3000             :                 &lck,                               /* locks */
    3001             :                 &smbreqs[0]);                       /* psmbreq */
    3002           5 :         if (reqs[0] == NULL) {
    3003           0 :                 d_fprintf(stderr, "cli_lockingx_create failed\n");
    3004           0 :                 goto done;
    3005             :         }
    3006           5 :         tevent_req_set_callback(reqs[0], locktest10_lockingx_done, &state);
    3007             : 
    3008           5 :         reqs[1] = cli_read_andx_create(
    3009             :                 ev,             /* mem_ctx */
    3010             :                 ev,             /* ev */
    3011             :                 cli1,           /* cli */
    3012             :                 fnum1,          /* fnum */
    3013             :                 0,              /* offset */
    3014             :                 1,              /* size */
    3015             :                 &smbreqs[1]);       /* psmbreq */
    3016           5 :         if (reqs[1] == NULL) {
    3017           0 :                 d_fprintf(stderr, "cli_read_andx_create failed\n");
    3018           0 :                 goto done;
    3019             :         }
    3020           5 :         tevent_req_set_callback(reqs[1], locktest10_read_andx_done, &state);
    3021             : 
    3022           5 :         status = smb1cli_req_chain_submit(smbreqs, ARRAY_SIZE(smbreqs));
    3023           5 :         if (!NT_STATUS_IS_OK(status)) {
    3024           0 :                 d_fprintf(stderr,
    3025             :                           "smb1cli_req_chain_submit failed: %s\n",
    3026             :                           nt_errstr(status));
    3027           0 :                 goto done;
    3028             :         }
    3029             : 
    3030          30 :         while (!state.done) {
    3031          25 :                 tevent_loop_once(ev);
    3032             :         }
    3033             : 
    3034           5 :         torture_close_connection(cli1);
    3035             : 
    3036           5 :         if (state.ok) {
    3037           5 :                 ret = true;
    3038             :         }
    3039           0 : done:
    3040           5 :         return ret;
    3041             : }
    3042             : 
    3043           5 : static void locktest10_lockingx_done(struct tevent_req *subreq)
    3044             : {
    3045           5 :         struct locktest10_state *state = tevent_req_callback_data_void(subreq);
    3046           0 :         NTSTATUS status;
    3047             : 
    3048           5 :         status = cli_lockingx_recv(subreq);
    3049           5 :         TALLOC_FREE(subreq);
    3050             : 
    3051           5 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_FILE_LOCK_CONFLICT)) {
    3052           0 :                 d_printf("cli_lockingx returned %s\n", nt_errstr(status));
    3053           0 :                 state->ok = false;
    3054             :         }
    3055           5 : }
    3056             : 
    3057           5 : static void locktest10_read_andx_done(struct tevent_req *subreq)
    3058             : {
    3059           5 :         struct locktest10_state *state = tevent_req_callback_data_void(subreq);
    3060           5 :         ssize_t received = -1;
    3061           5 :         uint8_t *rcvbuf = NULL;
    3062           0 :         NTSTATUS status;
    3063             : 
    3064           5 :         status = cli_read_andx_recv(subreq, &received, &rcvbuf);
    3065             : 
    3066           5 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_REQUEST_ABORTED)) {
    3067           0 :                 d_printf("cli_read_andx returned %s\n", nt_errstr(status));
    3068           0 :                 state->ok = false;
    3069             :         }
    3070             : 
    3071           5 :         state->done = true;
    3072           5 :         TALLOC_FREE(subreq);
    3073           5 : }
    3074             : 
    3075           5 : static bool run_locktest11(int dummy)
    3076             : {
    3077           0 :         struct cli_state *cli1;
    3078           5 :         const char *fname = "\\lockt11.lck";
    3079           0 :         NTSTATUS status;
    3080           0 :         uint16_t fnum;
    3081           5 :         bool ret = false;
    3082             : 
    3083           5 :         if (!torture_open_connection(&cli1, 0)) {
    3084           0 :                 return false;
    3085             :         }
    3086             : 
    3087           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    3088             : 
    3089           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    3090             : 
    3091           5 :         status = cli_openx(cli1, fname, O_CREAT|O_RDWR, DENY_NONE, &fnum);
    3092           5 :         if (!NT_STATUS_IS_OK(status)) {
    3093           0 :                 d_fprintf(stderr,
    3094             :                           "cli_openx returned %s\n",
    3095             :                           nt_errstr(status));
    3096           0 :                 return false;
    3097             :         }
    3098             : 
    3099             :         /*
    3100             :          * Test that LOCKING_ANDX_CANCEL_LOCK without any locks
    3101             :          * returns NT_STATUS_OK
    3102             :          */
    3103             : 
    3104           5 :         status = cli_lockingx(
    3105             :                 cli1,                           /* cli */
    3106             :                 fnum,                           /* fnum */
    3107             :                 LOCKING_ANDX_CANCEL_LOCK,       /* typeoflock */
    3108             :                 0,                              /* newoplocklevel */
    3109             :                 0,                              /* timeout */
    3110             :                 0,                              /* num_unlocks */
    3111             :                 NULL,                           /* unlocks */
    3112             :                 0,                              /* num_locks */
    3113             :                 NULL);                          /* locks */
    3114             : 
    3115           5 :         if (!NT_STATUS_IS_OK(status)) {
    3116           1 :                 d_printf("cli_lockingX returned %s\n", nt_errstr(status));
    3117           1 :                 goto fail;
    3118             :         }
    3119             : 
    3120           4 :         ret = true;
    3121           5 : fail:
    3122           5 :         cli_close(cli1, fnum);
    3123           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    3124             : 
    3125           5 :         return ret;
    3126             : }
    3127             : 
    3128             : struct deferred_close_state {
    3129             :         struct tevent_context *ev;
    3130             :         struct cli_state *cli;
    3131             :         uint16_t fnum;
    3132             : };
    3133             : 
    3134             : static void deferred_close_waited(struct tevent_req *subreq);
    3135             : static void deferred_close_done(struct tevent_req *subreq);
    3136             : 
    3137           5 : static struct tevent_req *deferred_close_send(
    3138             :         TALLOC_CTX *mem_ctx,
    3139             :         struct tevent_context *ev,
    3140             :         int wait_secs,
    3141             :         struct cli_state *cli,
    3142             :         uint16_t fnum)
    3143             : {
    3144           5 :         struct tevent_req *req = NULL, *subreq = NULL;
    3145           5 :         struct deferred_close_state *state = NULL;
    3146           5 :         struct timeval wakeup_time = timeval_current_ofs(wait_secs, 0);
    3147             : 
    3148           5 :         req = tevent_req_create(
    3149             :                 mem_ctx, &state, struct deferred_close_state);
    3150           5 :         if (req == NULL) {
    3151           0 :                 return NULL;
    3152             :         }
    3153           5 :         state->ev = ev;
    3154           5 :         state->cli = cli;
    3155           5 :         state->fnum = fnum;
    3156             : 
    3157           5 :         subreq = tevent_wakeup_send(state, state->ev, wakeup_time);
    3158           5 :         if (tevent_req_nomem(subreq, req)) {
    3159           0 :                 return tevent_req_post(req, ev);
    3160             :         }
    3161           5 :         tevent_req_set_callback(subreq, deferred_close_waited, req);
    3162           5 :         return req;
    3163             : }
    3164             : 
    3165           5 : static void deferred_close_waited(struct tevent_req *subreq)
    3166             : {
    3167           5 :         struct tevent_req *req = tevent_req_callback_data(
    3168             :                 subreq, struct tevent_req);
    3169           5 :         struct deferred_close_state *state = tevent_req_data(
    3170             :                 req, struct deferred_close_state);
    3171           0 :         bool ok;
    3172             : 
    3173           5 :         ok = tevent_wakeup_recv(subreq);
    3174           5 :         TALLOC_FREE(subreq);
    3175           5 :         if (!ok) {
    3176           0 :                 tevent_req_oom(req);
    3177           0 :                 return;
    3178             :         }
    3179             : 
    3180           5 :         subreq = cli_close_send(state, state->ev, state->cli, state->fnum, 0);
    3181           5 :         if (tevent_req_nomem(subreq, req)) {
    3182           0 :                 return;
    3183             :         }
    3184           5 :         tevent_req_set_callback(subreq, deferred_close_done, req);
    3185             : }
    3186             : 
    3187           5 : static void deferred_close_done(struct tevent_req *subreq)
    3188             : {
    3189           5 :         NTSTATUS status = cli_close_recv(subreq);
    3190           5 :         tevent_req_simple_finish_ntstatus(subreq, status);
    3191           5 : }
    3192             : 
    3193           5 : static NTSTATUS deferred_close_recv(struct tevent_req *req)
    3194             : {
    3195           5 :         return tevent_req_simple_recv_ntstatus(req);
    3196             : }
    3197             : 
    3198             : struct lockread_state {
    3199             :         struct smb1_lock_element lck;
    3200             :         struct tevent_req *reqs[2];
    3201             :         struct tevent_req *smbreqs[2];
    3202             :         NTSTATUS lock_status;
    3203             :         NTSTATUS read_status;
    3204             :         uint8_t *readbuf;
    3205             : };
    3206             : 
    3207             : static void lockread_lockingx_done(struct tevent_req *subreq);
    3208             : static void lockread_read_andx_done(struct tevent_req *subreq);
    3209             : 
    3210           5 : static struct tevent_req *lockread_send(
    3211             :         TALLOC_CTX *mem_ctx,
    3212             :         struct tevent_context *ev,
    3213             :         struct cli_state *cli,
    3214             :         uint16_t fnum)
    3215             : {
    3216           5 :         struct tevent_req *req = NULL;
    3217           5 :         struct lockread_state *state = NULL;
    3218           0 :         NTSTATUS status;
    3219             : 
    3220           5 :         req = tevent_req_create(mem_ctx, &state, struct lockread_state);
    3221           5 :         if (req == NULL) {
    3222           0 :                 return NULL;
    3223             :         }
    3224             : 
    3225          10 :         state->lck = (struct smb1_lock_element) {
    3226           5 :                 .pid = cli_getpid(cli), .offset = 0, .length = 1,
    3227             :         };
    3228             : 
    3229          10 :         state->reqs[0] = cli_lockingx_create(
    3230             :                 ev,                             /* mem_ctx */
    3231             :                 ev,                             /* tevent_context */
    3232             :                 cli,                            /* cli */
    3233             :                 fnum,                           /* fnum */
    3234             :                 LOCKING_ANDX_EXCLUSIVE_LOCK,    /* typeoflock */
    3235             :                 0,                              /* newoplocklevel */
    3236             :                 10000,                          /* timeout */
    3237             :                 0,                              /* num_unlocks */
    3238             :                 NULL,                           /* unlocks */
    3239             :                 1,                              /* num_locks */
    3240           5 :                 &state->lck,                     /* locks */
    3241           5 :                 &state->smbreqs[0]);             /* psmbreq */
    3242           5 :         if (tevent_req_nomem(state->reqs[0], req)) {
    3243           0 :                 return tevent_req_post(req, ev);
    3244             :         }
    3245           5 :         tevent_req_set_callback(
    3246             :                 state->reqs[0], lockread_lockingx_done, req);
    3247             : 
    3248          10 :         state->reqs[1] = cli_read_andx_create(
    3249             :                 ev,             /* mem_ctx */
    3250             :                 ev,             /* ev */
    3251             :                 cli,            /* cli */
    3252             :                 fnum,           /* fnum */
    3253             :                 0,              /* offset */
    3254             :                 1,              /* size */
    3255           5 :                 &state->smbreqs[1]);     /* psmbreq */
    3256           5 :         if (tevent_req_nomem(state->reqs[1], req)) {
    3257           0 :                 return tevent_req_post(req, ev);
    3258             :         }
    3259           5 :         tevent_req_set_callback(
    3260             :                 state->reqs[1], lockread_read_andx_done, req);
    3261             : 
    3262           5 :         status = smb1cli_req_chain_submit(state->smbreqs, 2);
    3263           5 :         if (tevent_req_nterror(req, status)) {
    3264           0 :                 return tevent_req_post(req, ev);
    3265             :         }
    3266           5 :         return req;
    3267             : }
    3268             : 
    3269           5 : static void lockread_lockingx_done(struct tevent_req *subreq)
    3270             : {
    3271           5 :         struct tevent_req *req = tevent_req_callback_data(
    3272             :                 subreq, struct tevent_req);
    3273           5 :         struct lockread_state *state = tevent_req_data(
    3274             :                 req, struct lockread_state);
    3275           5 :         state->lock_status = cli_lockingx_recv(subreq);
    3276           5 :         TALLOC_FREE(subreq);
    3277           5 :         d_fprintf(stderr,
    3278             :                   "lockingx returned %s\n",
    3279             :                   nt_errstr(state->lock_status));
    3280           5 : }
    3281             : 
    3282           5 : static void lockread_read_andx_done(struct tevent_req *subreq)
    3283             : {
    3284           5 :         struct tevent_req *req = tevent_req_callback_data(
    3285             :                 subreq, struct tevent_req);
    3286           5 :         struct lockread_state *state = tevent_req_data(
    3287             :                 req, struct lockread_state);
    3288           5 :         ssize_t received = -1;
    3289           5 :         uint8_t *rcvbuf = NULL;
    3290             : 
    3291           5 :         state->read_status = cli_read_andx_recv(subreq, &received, &rcvbuf);
    3292             : 
    3293           5 :         d_fprintf(stderr,
    3294             :                   "read returned %s\n",
    3295             :                   nt_errstr(state->read_status));
    3296             : 
    3297           5 :         if (!NT_STATUS_IS_OK(state->read_status)) {
    3298           4 :                 TALLOC_FREE(subreq);
    3299           4 :                 tevent_req_done(req);
    3300           4 :                 return;
    3301             :         }
    3302             : 
    3303           1 :         if (received > 0) {
    3304           1 :                 state->readbuf = talloc_memdup(state, rcvbuf, received);
    3305           1 :                 TALLOC_FREE(subreq);
    3306           1 :                 if (tevent_req_nomem(state->readbuf, req)) {
    3307           0 :                         return;
    3308             :                 }
    3309             :         }
    3310           1 :         TALLOC_FREE(subreq);
    3311           1 :         tevent_req_done(req);
    3312             : }
    3313             : 
    3314           5 : static NTSTATUS lockread_recv(
    3315             :         struct tevent_req *req,
    3316             :         NTSTATUS *lock_status,
    3317             :         NTSTATUS *read_status,
    3318             :         TALLOC_CTX *mem_ctx,
    3319             :         uint8_t **read_buf)
    3320             : {
    3321           5 :         struct lockread_state *state = tevent_req_data(
    3322             :                 req, struct lockread_state);
    3323           0 :         NTSTATUS status;
    3324             : 
    3325           5 :         if (tevent_req_is_nterror(req, &status)) {
    3326           0 :                 return status;
    3327             :         }
    3328             : 
    3329           5 :         *lock_status = state->lock_status;
    3330           5 :         *read_status = state->read_status;
    3331           5 :         if (state->readbuf != NULL) {
    3332           1 :                 *read_buf = talloc_move(mem_ctx, &state->readbuf);
    3333             :         } else {
    3334           4 :                 *read_buf = NULL;
    3335             :         }
    3336             : 
    3337           5 :         return NT_STATUS_OK;
    3338             : }
    3339             : 
    3340             : struct lock12_state {
    3341             :         uint8_t dummy;
    3342             : };
    3343             : 
    3344             : static void lock12_closed(struct tevent_req *subreq);
    3345             : static void lock12_read(struct tevent_req *subreq);
    3346             : 
    3347           5 : static struct tevent_req *lock12_send(
    3348             :         TALLOC_CTX *mem_ctx,
    3349             :         struct tevent_context *ev,
    3350             :         struct cli_state *cli,
    3351             :         uint16_t fnum1,
    3352             :         uint16_t fnum2)
    3353             : {
    3354           5 :         struct tevent_req *req = NULL, *subreq = NULL;
    3355           5 :         struct lock12_state *state = NULL;
    3356             : 
    3357           5 :         req = tevent_req_create(mem_ctx, &state, struct lock12_state);
    3358           5 :         if (req == NULL) {
    3359           0 :                 return NULL;
    3360             :         }
    3361             : 
    3362           5 :         subreq = deferred_close_send(state, ev, 1, cli, fnum1);
    3363           5 :         if (tevent_req_nomem(subreq, req)) {
    3364           0 :                 return tevent_req_post(req, ev);
    3365             :         }
    3366           5 :         tevent_req_set_callback(subreq, lock12_closed, req);
    3367             : 
    3368           5 :         subreq = lockread_send(state, ev, cli, fnum2);
    3369           5 :         if (tevent_req_nomem(subreq, req)) {
    3370           0 :                 return tevent_req_post(req, ev);
    3371             :         }
    3372           5 :         tevent_req_set_callback(subreq, lock12_read, req);
    3373             : 
    3374           5 :         return req;
    3375             : }
    3376             : 
    3377           5 : static void lock12_closed(struct tevent_req *subreq)
    3378             : {
    3379           5 :         struct tevent_req *req = tevent_req_callback_data(
    3380             :                 subreq, struct tevent_req);
    3381           0 :         NTSTATUS status;
    3382             : 
    3383           5 :         status = deferred_close_recv(subreq);
    3384           5 :         TALLOC_FREE(subreq);
    3385           5 :         DBG_DEBUG("close returned %s\n", nt_errstr(status));
    3386           5 :         if (tevent_req_nterror(req, status)) {
    3387           0 :                 return;
    3388             :         }
    3389             : }
    3390             : 
    3391           5 : static void lock12_read(struct tevent_req *subreq)
    3392             : {
    3393           5 :         struct tevent_req *req = tevent_req_callback_data(
    3394             :                 subreq, struct tevent_req);
    3395           5 :         struct lock12_state *state = tevent_req_data(
    3396             :                 req, struct lock12_state);
    3397           0 :         NTSTATUS status, lock_status, read_status;
    3398           5 :         uint8_t *buf = NULL;
    3399             : 
    3400           5 :         status = lockread_recv(
    3401             :                 subreq, &lock_status, &read_status, state, &buf);
    3402           5 :         TALLOC_FREE(subreq);
    3403          10 :         if (tevent_req_nterror(req, status) ||
    3404          10 :             tevent_req_nterror(req, lock_status) ||
    3405           5 :             tevent_req_nterror(req, read_status)) {
    3406           4 :                 return;
    3407             :         }
    3408           1 :         tevent_req_done(req);
    3409             : }
    3410             : 
    3411           5 : static NTSTATUS lock12_recv(struct tevent_req *req)
    3412             : 
    3413             : {
    3414           0 :         NTSTATUS status;
    3415             : 
    3416           5 :         if (tevent_req_is_nterror(req, &status)) {
    3417           4 :                 return status;
    3418             :         }
    3419           1 :         return NT_STATUS_OK;
    3420             : }
    3421             : 
    3422           5 : static bool run_locktest12(int dummy)
    3423             : {
    3424           5 :         struct tevent_context *ev = NULL;
    3425           5 :         struct tevent_req *req = NULL;
    3426           5 :         struct cli_state *cli = NULL;
    3427           5 :         const char fname[] = "\\lockt12.lck";
    3428           0 :         uint16_t fnum1, fnum2;
    3429           5 :         bool ret = false;
    3430           0 :         bool ok;
    3431           5 :         uint8_t data = 1;
    3432           0 :         NTSTATUS status;
    3433             : 
    3434           5 :         printf("starting locktest12\n");
    3435             : 
    3436           5 :         ev = samba_tevent_context_init(NULL);
    3437           5 :         if (ev == NULL) {
    3438           0 :                 d_fprintf(stderr, "samba_tevent_context_init failed\n");
    3439           0 :                 goto done;
    3440             :         }
    3441             : 
    3442           5 :         ok = torture_open_connection(&cli, 0);
    3443           5 :         if (!ok) {
    3444           0 :                 goto done;
    3445             :         }
    3446           5 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    3447             : 
    3448           5 :         status = cli_openx(cli, fname, O_CREAT|O_RDWR, DENY_NONE, &fnum1);
    3449           5 :         if (!NT_STATUS_IS_OK(status)) {
    3450           0 :                 d_fprintf(stderr,
    3451             :                           "cli_openx failed: %s\n",
    3452             :                           nt_errstr(status));
    3453           0 :                 goto done;
    3454             :         }
    3455             : 
    3456           5 :         status = cli_openx(cli, fname, O_CREAT|O_RDWR, DENY_NONE, &fnum2);
    3457           5 :         if (!NT_STATUS_IS_OK(status)) {
    3458           0 :                 d_fprintf(stderr,
    3459             :                           "cli_openx failed: %s\n",
    3460             :                           nt_errstr(status));
    3461           0 :                 goto done;
    3462             :         }
    3463             : 
    3464           5 :         status = cli_writeall(cli, fnum1, 0, &data, 0, sizeof(data), NULL);
    3465           5 :         if (!NT_STATUS_IS_OK(status)) {
    3466           0 :                 d_fprintf(stderr,
    3467             :                           "cli_writeall failed: %s\n",
    3468             :                           nt_errstr(status));
    3469           0 :                 goto done;
    3470             :         }
    3471             : 
    3472           5 :         status = cli_locktype(
    3473             :                 cli, fnum1, 0, 1, 0, LOCKING_ANDX_EXCLUSIVE_LOCK);
    3474           5 :         if (!NT_STATUS_IS_OK(status)) {
    3475           0 :                 d_fprintf(stderr,
    3476             :                           "cli_locktype failed: %s\n",
    3477             :                           nt_errstr(status));
    3478           0 :                 goto done;
    3479             :         }
    3480             : 
    3481           5 :         req = lock12_send(ev, ev, cli, fnum1, fnum2);
    3482           5 :         if (req == NULL) {
    3483           0 :                 d_fprintf(stderr, "lock12_send failed\n");
    3484           0 :                 goto done;
    3485             :         }
    3486             : 
    3487           5 :         ok = tevent_req_poll_ntstatus(req, ev, &status);
    3488           5 :         if (!ok) {
    3489           0 :                 d_fprintf(stderr, "tevent_req_poll_ntstatus failed\n");
    3490           0 :                 goto done;
    3491             :         }
    3492             : 
    3493           5 :         if (!NT_STATUS_IS_OK(status)) {
    3494           0 :                 d_fprintf(stderr,
    3495             :                           "tevent_req_poll_ntstatus returned %s\n",
    3496             :                           nt_errstr(status));
    3497           0 :                 goto done;
    3498             :         }
    3499             : 
    3500           5 :         status = lock12_recv(req);
    3501           5 :         if (!NT_STATUS_IS_OK(status)) {
    3502           4 :                 d_fprintf(stderr, "lock12 returned %s\n", nt_errstr(status));
    3503           4 :                 goto done;
    3504             :         }
    3505             : 
    3506           1 :         ret = true;
    3507           5 : done:
    3508           5 :         if (cli != NULL) {
    3509           5 :                 torture_close_connection(cli);
    3510             :         }
    3511           5 :         return ret;
    3512             : }
    3513             : 
    3514             : struct lock_ntcancel_state {
    3515             :         struct timeval start;
    3516             :         struct smb1_lock_element lck;
    3517             :         struct tevent_req *subreq;
    3518             : };
    3519             : 
    3520             : static void lock_ntcancel_waited(struct tevent_req *subreq);
    3521             : static void lock_ntcancel_done(struct tevent_req *subreq);
    3522             : 
    3523           5 : static struct tevent_req *lock_ntcancel_send(
    3524             :         TALLOC_CTX *mem_ctx,
    3525             :         struct tevent_context *ev,
    3526             :         struct cli_state *cli,
    3527             :         uint16_t fnum)
    3528             : {
    3529           5 :         struct tevent_req *req = NULL, *subreq = NULL;
    3530           5 :         struct lock_ntcancel_state *state = NULL;
    3531             : 
    3532           5 :         req = tevent_req_create(mem_ctx, &state, struct lock_ntcancel_state);
    3533           5 :         if (req == NULL) {
    3534           0 :                 return NULL;
    3535             :         }
    3536          10 :         state->lck = (struct smb1_lock_element) {
    3537           5 :                 .pid = cli_getpid(cli), .offset = 0, .length = 1,
    3538             :         };
    3539           5 :         state->start = timeval_current();
    3540             : 
    3541          10 :         state->subreq = cli_lockingx_send(
    3542             :                 state,                          /* mem_ctx */
    3543             :                 ev,                             /* tevent_context */
    3544             :                 cli,                            /* cli */
    3545             :                 fnum,                           /* fnum */
    3546             :                 LOCKING_ANDX_EXCLUSIVE_LOCK,    /* typeoflock */
    3547             :                 0,                              /* newoplocklevel */
    3548             :                 10000,                          /* timeout */
    3549             :                 0,                              /* num_unlocks */
    3550             :                 NULL,                           /* unlocks */
    3551             :                 1,                              /* num_locks */
    3552           5 :                 &state->lck);                    /* locks */
    3553           5 :         if (tevent_req_nomem(state->subreq, req)) {
    3554           0 :                 return tevent_req_post(req, ev);
    3555             :         }
    3556           5 :         tevent_req_set_callback(state->subreq, lock_ntcancel_done, req);
    3557             : 
    3558           5 :         subreq = tevent_wakeup_send(state, ev, timeval_current_ofs(1, 0));
    3559           5 :         if (tevent_req_nomem(subreq, req)) {
    3560           0 :                 return tevent_req_post(req, ev);
    3561             :         }
    3562           5 :         tevent_req_set_callback(subreq, lock_ntcancel_waited, req);
    3563           5 :         return req;
    3564             : }
    3565             : 
    3566           5 : static void lock_ntcancel_waited(struct tevent_req *subreq)
    3567             : {
    3568           5 :         struct tevent_req *req = tevent_req_callback_data(
    3569             :                 subreq, struct tevent_req);
    3570           5 :         struct lock_ntcancel_state *state = tevent_req_data(
    3571             :                 req, struct lock_ntcancel_state);
    3572           0 :         bool ok;
    3573             : 
    3574           5 :         ok = tevent_wakeup_recv(subreq);
    3575           5 :         TALLOC_FREE(subreq);
    3576           5 :         if (!ok) {
    3577           0 :                 tevent_req_oom(req);
    3578           0 :                 return;
    3579             :         }
    3580             : 
    3581           5 :         ok = tevent_req_cancel(state->subreq);
    3582           5 :         if (!ok) {
    3583           0 :                 d_fprintf(stderr, "Could not cancel subreq\n");
    3584           0 :                 tevent_req_oom(req);
    3585           0 :                 return;
    3586             :         }
    3587             : }
    3588             : 
    3589           5 : static void lock_ntcancel_done(struct tevent_req *subreq)
    3590             : {
    3591           5 :         struct tevent_req *req = tevent_req_callback_data(
    3592             :                 subreq, struct tevent_req);
    3593           5 :         struct lock_ntcancel_state *state = tevent_req_data(
    3594             :                 req, struct lock_ntcancel_state);
    3595           0 :         NTSTATUS status;
    3596           0 :         double elapsed;
    3597             : 
    3598           5 :         status = cli_lockingx_recv(subreq);
    3599           5 :         TALLOC_FREE(subreq);
    3600             : 
    3601           5 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_FILE_LOCK_CONFLICT)) {
    3602           0 :                 d_printf("cli_lockingx returned %s\n", nt_errstr(status));
    3603           0 :                 tevent_req_nterror(req, NT_STATUS_UNSUCCESSFUL);
    3604           0 :                 return;
    3605             :         }
    3606             : 
    3607           5 :         elapsed = timeval_elapsed(&state->start);
    3608             : 
    3609           5 :         if (elapsed > 3) {
    3610           0 :                 d_printf("cli_lockingx was too slow, cancel did not work\n");
    3611           0 :                 tevent_req_nterror(req, NT_STATUS_UNSUCCESSFUL);
    3612           0 :                 return;
    3613             :         }
    3614             : 
    3615           5 :         tevent_req_done(req);
    3616             : }
    3617             : 
    3618           5 : static NTSTATUS lock_ntcancel_recv(struct tevent_req *req)
    3619             : {
    3620           5 :         return tevent_req_simple_recv_ntstatus(req);
    3621             : }
    3622             : 
    3623           5 : static bool run_locktest13(int dummy)
    3624             : {
    3625           5 :         struct tevent_context *ev = NULL;
    3626           5 :         struct tevent_req *req = NULL;
    3627           5 :         struct cli_state *cli = NULL;
    3628           5 :         const char fname[] = "\\lockt13.lck";
    3629           0 :         uint16_t fnum1, fnum2;
    3630           5 :         bool ret = false;
    3631           0 :         bool ok;
    3632           5 :         uint8_t data = 1;
    3633           0 :         NTSTATUS status;
    3634             : 
    3635           5 :         printf("starting locktest13\n");
    3636             : 
    3637           5 :         ev = samba_tevent_context_init(NULL);
    3638           5 :         if (ev == NULL) {
    3639           0 :                 d_fprintf(stderr, "samba_tevent_context_init failed\n");
    3640           0 :                 goto done;
    3641             :         }
    3642             : 
    3643           5 :         ok = torture_open_connection(&cli, 0);
    3644           5 :         if (!ok) {
    3645           0 :                 goto done;
    3646             :         }
    3647           5 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    3648             : 
    3649           5 :         status = cli_openx(cli, fname, O_CREAT|O_RDWR, DENY_NONE, &fnum1);
    3650           5 :         if (!NT_STATUS_IS_OK(status)) {
    3651           0 :                 d_fprintf(stderr,
    3652             :                           "cli_openx failed: %s\n",
    3653             :                           nt_errstr(status));
    3654           0 :                 goto done;
    3655             :         }
    3656             : 
    3657           5 :         status = cli_openx(cli, fname, O_CREAT|O_RDWR, DENY_NONE, &fnum2);
    3658           5 :         if (!NT_STATUS_IS_OK(status)) {
    3659           0 :                 d_fprintf(stderr,
    3660             :                           "cli_openx failed: %s\n",
    3661             :                           nt_errstr(status));
    3662           0 :                 goto done;
    3663             :         }
    3664             : 
    3665           5 :         status = cli_writeall(cli, fnum1, 0, &data, 0, sizeof(data), NULL);
    3666           5 :         if (!NT_STATUS_IS_OK(status)) {
    3667           0 :                 d_fprintf(stderr,
    3668             :                           "cli_writeall failed: %s\n",
    3669             :                           nt_errstr(status));
    3670           0 :                 goto done;
    3671             :         }
    3672             : 
    3673           5 :         status = cli_locktype(
    3674             :                 cli, fnum1, 0, 1, 0, LOCKING_ANDX_EXCLUSIVE_LOCK);
    3675           5 :         if (!NT_STATUS_IS_OK(status)) {
    3676           0 :                 d_fprintf(stderr,
    3677             :                           "cli_locktype failed: %s\n",
    3678             :                           nt_errstr(status));
    3679           0 :                 goto done;
    3680             :         }
    3681             : 
    3682           5 :         req = lock_ntcancel_send(ev, ev, cli, fnum2);
    3683           5 :         if (req == NULL) {
    3684           0 :                 d_fprintf(stderr, "lock_ntcancel_send failed\n");
    3685           0 :                 goto done;
    3686             :         }
    3687             : 
    3688           5 :         ok = tevent_req_poll_ntstatus(req, ev, &status);
    3689           5 :         if (!ok) {
    3690           0 :                 d_fprintf(stderr, "tevent_req_poll_ntstatus failed\n");
    3691           0 :                 goto done;
    3692             :         }
    3693             : 
    3694           5 :         if (!NT_STATUS_IS_OK(status)) {
    3695           0 :                 d_fprintf(stderr,
    3696             :                           "tevent_req_poll_ntstatus returned %s\n",
    3697             :                           nt_errstr(status));
    3698           0 :                 goto done;
    3699             :         }
    3700             : 
    3701           5 :         status = lock_ntcancel_recv(req);
    3702           5 :         if (!NT_STATUS_IS_OK(status)) {
    3703           0 :                 d_fprintf(stderr,
    3704             :                           "lock_ntcancel returned %s\n",
    3705             :                           nt_errstr(status));
    3706           0 :                 goto done;
    3707             :         }
    3708             : 
    3709           5 :         ret = true;
    3710           5 : done:
    3711           5 :         if (cli != NULL) {
    3712           5 :                 torture_close_connection(cli);
    3713             :         }
    3714           5 :         return ret;
    3715             : }
    3716             : 
    3717             : /*
    3718             : test whether fnums and tids open on one VC are available on another (a major
    3719             : security hole)
    3720             : */
    3721           5 : static bool run_fdpasstest(int dummy)
    3722             : {
    3723           0 :         struct cli_state *cli1, *cli2;
    3724           5 :         const char *fname = "\\fdpass.tst";
    3725           0 :         uint16_t fnum1;
    3726           0 :         char buf[1024];
    3727           0 :         NTSTATUS status;
    3728             : 
    3729           5 :         if (!torture_open_connection(&cli1, 0) || !torture_open_connection(&cli2, 1)) {
    3730           0 :                 return False;
    3731             :         }
    3732           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    3733           5 :         smbXcli_conn_set_sockopt(cli2->conn, sockops);
    3734             : 
    3735           5 :         printf("starting fdpasstest\n");
    3736             : 
    3737           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    3738             : 
    3739           5 :         status = cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE,
    3740             :                           &fnum1);
    3741           5 :         if (!NT_STATUS_IS_OK(status)) {
    3742           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    3743           0 :                 return False;
    3744             :         }
    3745             : 
    3746           5 :         status = cli_writeall(cli1, fnum1, 0, (const uint8_t *)"hello world\n", 0,
    3747             :                               13, NULL);
    3748           5 :         if (!NT_STATUS_IS_OK(status)) {
    3749           0 :                 printf("write failed (%s)\n", nt_errstr(status));
    3750           0 :                 return False;
    3751             :         }
    3752             : 
    3753           5 :         cli_state_set_uid(cli2, cli_state_get_uid(cli1));
    3754           5 :         cli_state_set_tid(cli2, cli_state_get_tid(cli1));
    3755           5 :         cli_setpid(cli2, cli_getpid(cli1));
    3756             : 
    3757           5 :         if (test_cli_read(cli2, fnum1, buf, 0, 13, NULL, 13)) {
    3758           0 :                 printf("read succeeded! nasty security hole [%s]\n", buf);
    3759           0 :                 return false;
    3760             :         }
    3761             : 
    3762           5 :         cli_close(cli1, fnum1);
    3763           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    3764             : 
    3765           5 :         torture_close_connection(cli1);
    3766           5 :         torture_close_connection(cli2);
    3767             : 
    3768           5 :         printf("finished fdpasstest\n");
    3769           5 :         return True;
    3770             : }
    3771             : 
    3772           5 : static bool run_fdsesstest(int dummy)
    3773             : {
    3774           0 :         struct cli_state *cli;
    3775           0 :         uint16_t new_vuid;
    3776           0 :         uint16_t saved_vuid;
    3777           0 :         uint32_t new_cnum;
    3778           0 :         uint32_t saved_cnum;
    3779           5 :         const char *fname = "\\fdsess.tst";
    3780           5 :         const char *fname1 = "\\fdsess1.tst";
    3781           0 :         uint16_t fnum1;
    3782           0 :         uint16_t fnum2;
    3783           0 :         char buf[1024];
    3784           5 :         bool ret = True;
    3785           0 :         NTSTATUS status;
    3786             : 
    3787           5 :         if (!torture_open_connection(&cli, 0))
    3788           0 :                 return False;
    3789           5 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    3790             : 
    3791           5 :         if (!torture_cli_session_setup2(cli, &new_vuid))
    3792           0 :                 return False;
    3793             : 
    3794           5 :         saved_cnum = cli_state_get_tid(cli);
    3795           5 :         if (!NT_STATUS_IS_OK(cli_tree_connect(cli, share, "?????", NULL)))
    3796           0 :                 return False;
    3797           5 :         new_cnum = cli_state_get_tid(cli);
    3798           5 :         cli_state_set_tid(cli, saved_cnum);
    3799             : 
    3800           5 :         printf("starting fdsesstest\n");
    3801             : 
    3802           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    3803           5 :         cli_unlink(cli, fname1, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    3804             : 
    3805           5 :         status = cli_openx(cli, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum1);
    3806           5 :         if (!NT_STATUS_IS_OK(status)) {
    3807           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    3808           0 :                 return False;
    3809             :         }
    3810             : 
    3811           5 :         status = cli_writeall(cli, fnum1, 0, (const uint8_t *)"hello world\n", 0, 13,
    3812             :                               NULL);
    3813           5 :         if (!NT_STATUS_IS_OK(status)) {
    3814           0 :                 printf("write failed (%s)\n", nt_errstr(status));
    3815           0 :                 return False;
    3816             :         }
    3817             : 
    3818           5 :         saved_vuid = cli_state_get_uid(cli);
    3819           5 :         cli_state_set_uid(cli, new_vuid);
    3820             : 
    3821           5 :         if (test_cli_read(cli, fnum1, buf, 0, 13, NULL, 13)) {
    3822           0 :                 printf("read succeeded with different vuid! "
    3823             :                        "nasty security hole [%s]\n", buf);
    3824           0 :                 ret = false;
    3825             :         }
    3826             :         /* Try to open a file with different vuid, samba cnum. */
    3827           5 :         if (NT_STATUS_IS_OK(cli_openx(cli, fname1, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum2))) {
    3828           5 :                 printf("create with different vuid, same cnum succeeded.\n");
    3829           5 :                 cli_close(cli, fnum2);
    3830           5 :                 cli_unlink(cli, fname1, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    3831             :         } else {
    3832           0 :                 printf("create with different vuid, same cnum failed.\n");
    3833           0 :                 printf("This will cause problems with service clients.\n");
    3834           0 :                 ret = False;
    3835             :         }
    3836             : 
    3837           5 :         cli_state_set_uid(cli, saved_vuid);
    3838             : 
    3839             :         /* Try with same vuid, different cnum. */
    3840           5 :         cli_state_set_tid(cli, new_cnum);
    3841             : 
    3842           5 :         if (test_cli_read(cli, fnum1, buf, 0, 13, NULL, 13)) {
    3843           0 :                 printf("read succeeded with different cnum![%s]\n", buf);
    3844           0 :                 ret = false;
    3845             :         }
    3846             : 
    3847           5 :         cli_state_set_tid(cli, saved_cnum);
    3848           5 :         cli_close(cli, fnum1);
    3849           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    3850             : 
    3851           5 :         torture_close_connection(cli);
    3852             : 
    3853           5 :         printf("finished fdsesstest\n");
    3854           5 :         return ret;
    3855             : }
    3856             : 
    3857             : /*
    3858             :   This test checks that 
    3859             : 
    3860             :   1) the server does not allow an unlink on a file that is open
    3861             : */
    3862           5 : static bool run_unlinktest(int dummy)
    3863             : {
    3864           0 :         struct cli_state *cli;
    3865           5 :         const char *fname = "\\unlink.tst";
    3866           0 :         uint16_t fnum;
    3867           5 :         bool correct = True;
    3868           0 :         NTSTATUS status;
    3869             : 
    3870           5 :         if (!torture_open_connection(&cli, 0)) {
    3871           0 :                 return False;
    3872             :         }
    3873             : 
    3874           5 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    3875             : 
    3876           5 :         printf("starting unlink test\n");
    3877             : 
    3878           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    3879             : 
    3880           5 :         cli_setpid(cli, 1);
    3881             : 
    3882           5 :         status = cli_openx(cli, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum);
    3883           5 :         if (!NT_STATUS_IS_OK(status)) {
    3884           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    3885           0 :                 return False;
    3886             :         }
    3887             : 
    3888           5 :         status = cli_unlink(cli, fname,
    3889             :                             FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    3890           5 :         if (NT_STATUS_IS_OK(status)) {
    3891           0 :                 printf("error: server allowed unlink on an open file\n");
    3892           0 :                 correct = False;
    3893             :         } else {
    3894           5 :                 correct = check_error(__LINE__, status, ERRDOS, ERRbadshare,
    3895           5 :                                       NT_STATUS_SHARING_VIOLATION);
    3896             :         }
    3897             : 
    3898           5 :         cli_close(cli, fnum);
    3899           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    3900             : 
    3901           5 :         if (!torture_close_connection(cli)) {
    3902           0 :                 correct = False;
    3903             :         }
    3904             : 
    3905           5 :         printf("unlink test finished\n");
    3906             : 
    3907           5 :         return correct;
    3908             : }
    3909             : 
    3910             : 
    3911             : /*
    3912             : test how many open files this server supports on the one socket
    3913             : */
    3914           0 : static bool run_maxfidtest(int dummy)
    3915             : {
    3916           0 :         struct cli_state *cli;
    3917           0 :         fstring fname;
    3918           0 :         uint16_t fnums[0x11000];
    3919           0 :         int i;
    3920           0 :         int retries=4;
    3921           0 :         bool correct = True;
    3922           0 :         NTSTATUS status;
    3923             : 
    3924           0 :         cli = current_cli;
    3925             : 
    3926           0 :         if (retries <= 0) {
    3927           0 :                 printf("failed to connect\n");
    3928           0 :                 return False;
    3929             :         }
    3930             : 
    3931           0 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    3932             : 
    3933           0 :         for (i=0; i<0x11000; i++) {
    3934           0 :                 slprintf(fname,sizeof(fname)-1,"\\maxfid.%d.%d", i,(int)getpid());
    3935           0 :                 status = cli_openx(cli, fname, O_RDWR|O_CREAT|O_TRUNC, DENY_NONE,
    3936             :                                   &fnums[i]);
    3937           0 :                 if (!NT_STATUS_IS_OK(status)) {
    3938           0 :                         printf("open of %s failed (%s)\n", 
    3939             :                                fname, nt_errstr(status));
    3940           0 :                         printf("maximum fnum is %d\n", i);
    3941           0 :                         break;
    3942             :                 }
    3943           0 :                 printf("%6d\r", i);
    3944             :         }
    3945           0 :         printf("%6d\n", i);
    3946           0 :         i--;
    3947             : 
    3948           0 :         printf("cleaning up\n");
    3949           0 :         for (;i>=0;i--) {
    3950           0 :                 slprintf(fname,sizeof(fname)-1,"\\maxfid.%d.%d", i,(int)getpid());
    3951           0 :                 cli_close(cli, fnums[i]);
    3952             : 
    3953           0 :                 status = cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    3954           0 :                 if (!NT_STATUS_IS_OK(status)) {
    3955           0 :                         printf("unlink of %s failed (%s)\n", 
    3956             :                                fname, nt_errstr(status));
    3957           0 :                         correct = False;
    3958             :                 }
    3959           0 :                 printf("%6d\r", i);
    3960             :         }
    3961           0 :         printf("%6d\n", 0);
    3962             : 
    3963           0 :         printf("maxfid test finished\n");
    3964           0 :         if (!torture_close_connection(cli)) {
    3965           0 :                 correct = False;
    3966             :         }
    3967           0 :         return correct;
    3968             : }
    3969             : 
    3970             : /* generate a random buffer */
    3971           0 : static void rand_buf(char *buf, int len)
    3972             : {
    3973           0 :         while (len--) {
    3974           0 :                 *buf = (char)sys_random();
    3975           0 :                 buf++;
    3976             :         }
    3977           0 : }
    3978             : 
    3979             : /* send smb negprot commands, not reading the response */
    3980           0 : static bool run_negprot_nowait(int dummy)
    3981             : {
    3982           0 :         struct tevent_context *ev;
    3983           0 :         int i;
    3984           0 :         struct cli_state *cli;
    3985           0 :         bool correct = True;
    3986             : 
    3987           0 :         printf("starting negprot nowait test\n");
    3988             : 
    3989           0 :         ev = samba_tevent_context_init(talloc_tos());
    3990           0 :         if (ev == NULL) {
    3991           0 :                 return false;
    3992             :         }
    3993             : 
    3994           0 :         if (!(cli = open_nbt_connection())) {
    3995           0 :                 TALLOC_FREE(ev);
    3996           0 :                 return False;
    3997             :         }
    3998             : 
    3999           0 :         for (i=0;i<50000;i++) {
    4000           0 :                 struct tevent_req *req;
    4001             : 
    4002           0 :                 req = smbXcli_negprot_send(
    4003             :                         ev,
    4004             :                         ev,
    4005             :                         cli->conn,
    4006           0 :                         cli->timeout,
    4007             :                         PROTOCOL_CORE,
    4008             :                         PROTOCOL_NT1,
    4009             :                         0,
    4010             :                         NULL);
    4011           0 :                 if (req == NULL) {
    4012           0 :                         TALLOC_FREE(ev);
    4013           0 :                         return false;
    4014             :                 }
    4015           0 :                 if (!tevent_req_poll(req, ev)) {
    4016           0 :                         d_fprintf(stderr, "tevent_req_poll failed: %s\n",
    4017           0 :                                   strerror(errno));
    4018           0 :                         TALLOC_FREE(ev);
    4019           0 :                         return false;
    4020             :                 }
    4021           0 :                 TALLOC_FREE(req);
    4022             :         }
    4023             : 
    4024           0 :         if (torture_close_connection(cli)) {
    4025           0 :                 correct = False;
    4026             :         }
    4027             : 
    4028           0 :         printf("finished negprot nowait test\n");
    4029             : 
    4030           0 :         return correct;
    4031             : }
    4032             : 
    4033             : /* send smb negprot commands, not reading the response */
    4034           4 : static bool run_bad_nbt_session(int dummy)
    4035             : {
    4036           0 :         struct nmb_name called, calling;
    4037           0 :         struct sockaddr_storage ss;
    4038           0 :         NTSTATUS status;
    4039           0 :         int fd;
    4040           0 :         bool ret;
    4041             : 
    4042           4 :         printf("starting bad nbt session test\n");
    4043             : 
    4044           4 :         make_nmb_name(&calling, myname, 0x0);
    4045           4 :         make_nmb_name(&called , host, 0x20);
    4046             : 
    4047           4 :         if (!resolve_name(host, &ss, 0x20, true)) {
    4048           0 :                 d_fprintf(stderr, "Could not resolve name %s\n", host);
    4049           0 :                 return false;
    4050             :         }
    4051             : 
    4052           4 :         status = open_socket_out(&ss, NBT_SMB_PORT, 10000, &fd);
    4053           4 :         if (!NT_STATUS_IS_OK(status)) {
    4054           0 :                 d_fprintf(stderr, "open_socket_out failed: %s\n",
    4055             :                           nt_errstr(status));
    4056           0 :                 return false;
    4057             :         }
    4058             : 
    4059           4 :         ret = cli_bad_session_request(fd, &calling, &called);
    4060           4 :         close(fd);
    4061           4 :         if (!ret) {
    4062           0 :                 d_fprintf(stderr, "open_socket_out failed: %s\n",
    4063             :                           nt_errstr(status));
    4064           0 :                 return false;
    4065             :         }
    4066             : 
    4067           4 :         printf("finished bad nbt session test\n");
    4068           4 :         return true;
    4069             : }
    4070             : 
    4071             : /* send random IPC commands */
    4072           0 : static bool run_randomipc(int dummy)
    4073             : {
    4074           0 :         char *rparam = NULL;
    4075           0 :         char *rdata = NULL;
    4076           0 :         unsigned int rdrcnt,rprcnt;
    4077           0 :         char param[1024];
    4078           0 :         int api, param_len, i;
    4079           0 :         struct cli_state *cli;
    4080           0 :         bool correct = True;
    4081           0 :         int count = 50000;
    4082             : 
    4083           0 :         printf("starting random ipc test\n");
    4084             : 
    4085           0 :         if (!torture_open_connection(&cli, 0)) {
    4086           0 :                 return False;
    4087             :         }
    4088             : 
    4089           0 :         for (i=0;i<count;i++) {
    4090           0 :                 api = sys_random() % 500;
    4091           0 :                 param_len = (sys_random() % 64);
    4092             : 
    4093           0 :                 rand_buf(param, param_len);
    4094             : 
    4095           0 :                 SSVAL(param,0,api); 
    4096             : 
    4097           0 :                 cli_api(cli, 
    4098             :                         param, param_len, 8,  
    4099             :                         NULL, 0, CLI_BUFFER_SIZE,
    4100             :                         &rparam, &rprcnt,     
    4101             :                         &rdata, &rdrcnt);
    4102           0 :                 if (i % 100 == 0) {
    4103           0 :                         printf("%d/%d\r", i,count);
    4104             :                 }
    4105             :         }
    4106           0 :         printf("%d/%d\n", i, count);
    4107             : 
    4108           0 :         if (!torture_close_connection(cli)) {
    4109           0 :                 correct = False;
    4110             :         }
    4111             : 
    4112           0 :         SAFE_FREE(rparam);
    4113           0 :         SAFE_FREE(rdata);
    4114             : 
    4115           0 :         printf("finished random ipc test\n");
    4116             : 
    4117           0 :         return correct;
    4118             : }
    4119             : 
    4120             : 
    4121             : 
    4122           8 : static void browse_callback(const char *sname, uint32_t stype,
    4123             :                             const char *comment, void *state)
    4124             : {
    4125           8 :         printf("\t%20.20s %08x %s\n", sname, stype, comment);
    4126           8 : }
    4127             : 
    4128             : 
    4129             : 
    4130             : /*
    4131             :   This test checks the browse list code
    4132             : 
    4133             : */
    4134           5 : static bool run_browsetest(int dummy)
    4135             : {
    4136           0 :         static struct cli_state *cli;
    4137           5 :         bool correct = True;
    4138             : 
    4139           5 :         printf("starting browse test\n");
    4140             : 
    4141           5 :         if (!torture_open_connection(&cli, 0)) {
    4142           0 :                 return False;
    4143             :         }
    4144             : 
    4145           5 :         printf("domain list:\n");
    4146           5 :         cli_NetServerEnum(cli, cli->server_domain, 
    4147             :                           SV_TYPE_DOMAIN_ENUM,
    4148             :                           browse_callback, NULL);
    4149             : 
    4150           5 :         printf("machine list:\n");
    4151           5 :         cli_NetServerEnum(cli, cli->server_domain, 
    4152             :                           SV_TYPE_ALL,
    4153             :                           browse_callback, NULL);
    4154             : 
    4155           5 :         if (!torture_close_connection(cli)) {
    4156           0 :                 correct = False;
    4157             :         }
    4158             : 
    4159           5 :         printf("browse test finished\n");
    4160             : 
    4161           5 :         return correct;
    4162             : 
    4163             : }
    4164             : 
    4165          20 : static bool check_attributes(struct cli_state *cli,
    4166             :                                 const char *fname,
    4167             :                                 uint32_t expected_attrs)
    4168             : {
    4169          20 :         uint32_t attrs = 0;
    4170          20 :         NTSTATUS status = cli_getatr(cli,
    4171             :                                 fname,
    4172             :                                 &attrs,
    4173             :                                 NULL,
    4174             :                                 NULL);
    4175          20 :         if (!NT_STATUS_IS_OK(status)) {
    4176           0 :                 printf("cli_getatr failed with %s\n",
    4177             :                         nt_errstr(status));
    4178           0 :                 return false;
    4179             :         }
    4180          20 :         if (attrs != expected_attrs) {
    4181           0 :                 printf("Attributes incorrect 0x%x, should be 0x%x\n",
    4182             :                         (unsigned int)attrs,
    4183             :                         (unsigned int)expected_attrs);
    4184           0 :                 return false;
    4185             :         }
    4186          20 :         return true;
    4187             : }
    4188             : 
    4189             : /*
    4190             :   This checks how the getatr calls works
    4191             : */
    4192           5 : static bool run_attrtest(int dummy)
    4193             : {
    4194           0 :         struct cli_state *cli;
    4195           0 :         uint16_t fnum;
    4196           0 :         time_t t, t2;
    4197           5 :         const char *fname = "\\attrib123456789.tst";
    4198           5 :         bool correct = True;
    4199           0 :         NTSTATUS status;
    4200             : 
    4201           5 :         printf("starting attrib test\n");
    4202             : 
    4203           5 :         if (!torture_open_connection(&cli, 0)) {
    4204           0 :                 return False;
    4205             :         }
    4206             : 
    4207             :         /* Ensure we can't unlink with out-of-range (unknown) attribute. */
    4208           5 :         status = cli_unlink(cli, fname, 0x20000);
    4209           5 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_INVALID_PARAMETER)) {
    4210           0 :                 correct = false;
    4211           0 :                 goto out;
    4212             :         }
    4213             : 
    4214           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4215           5 :         cli_openx(cli, fname, 
    4216             :                         O_RDWR | O_CREAT | O_TRUNC, DENY_NONE, &fnum);
    4217           5 :         cli_close(cli, fnum);
    4218             : 
    4219           5 :         status = cli_getatr(cli, fname, NULL, NULL, &t);
    4220           5 :         if (!NT_STATUS_IS_OK(status)) {
    4221           0 :                 printf("getatr failed (%s)\n", nt_errstr(status));
    4222           0 :                 correct = False;
    4223             :         }
    4224             : 
    4225           5 :         if (labs(t - time(NULL)) > 60*60*24*10) {
    4226           0 :                 printf("ERROR: SMBgetatr bug. time is %s",
    4227             :                        ctime(&t));
    4228           0 :                 t = time(NULL);
    4229           0 :                 correct = True;
    4230             :         }
    4231             : 
    4232           5 :         t2 = t-60*60*24; /* 1 day ago */
    4233             : 
    4234             :         /* Ensure we can't set with out-of-range (unknown) attribute. */
    4235           5 :         status = cli_setatr(cli, fname, 0x20000, t2);
    4236           5 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_INVALID_PARAMETER)) {
    4237           0 :                 correct = false;
    4238           0 :                 goto out;
    4239             :         }
    4240             : 
    4241           5 :         status = cli_setatr(cli, fname, 0, t2);
    4242           5 :         if (!NT_STATUS_IS_OK(status)) {
    4243           0 :                 printf("setatr failed (%s)\n", nt_errstr(status));
    4244           0 :                 correct = True;
    4245             :         }
    4246             : 
    4247           5 :         status = cli_getatr(cli, fname, NULL, NULL, &t);
    4248           5 :         if (!NT_STATUS_IS_OK(status)) {
    4249           0 :                 printf("getatr failed (%s)\n", nt_errstr(status));
    4250           0 :                 correct = True;
    4251             :         }
    4252             : 
    4253           5 :         if (t != t2) {
    4254           0 :                 printf("ERROR: getatr/setatr bug. times are\n%s",
    4255             :                        ctime(&t));
    4256           0 :                 printf("%s", ctime(&t2));
    4257           0 :                 correct = True;
    4258             :         }
    4259             : 
    4260           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4261             : 
    4262             :         /* Check cli_setpathinfo_ext() */
    4263             :         /* Re-create the file. */
    4264           5 :         status = cli_openx(cli, fname,
    4265             :                         O_RDWR | O_CREAT | O_TRUNC, DENY_NONE, &fnum);
    4266           5 :         if (!NT_STATUS_IS_OK(status)) {
    4267           0 :                 printf("Failed to recreate %s (%s)\n",
    4268             :                         fname, nt_errstr(status));
    4269           0 :                 correct = false;
    4270             :         }
    4271           5 :         cli_close(cli, fnum);
    4272             : 
    4273           5 :         status = cli_setpathinfo_ext(
    4274             :                 cli,
    4275             :                 fname,
    4276           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* create */
    4277           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* access */
    4278           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* write */
    4279           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* change */
    4280             :                 FILE_ATTRIBUTE_SYSTEM |
    4281             :                 FILE_ATTRIBUTE_HIDDEN |
    4282             :                 FILE_ATTRIBUTE_READONLY);
    4283           5 :         if (!NT_STATUS_IS_OK(status)) {
    4284           0 :                 printf("cli_setpathinfo_ext failed with %s\n",
    4285             :                         nt_errstr(status));
    4286           0 :                 correct = false;
    4287             :         }
    4288             : 
    4289             :         /* Check attributes are correct. */
    4290           5 :         correct = check_attributes(cli,
    4291             :                         fname,
    4292             :                         FILE_ATTRIBUTE_SYSTEM |
    4293             :                         FILE_ATTRIBUTE_HIDDEN |
    4294             :                         FILE_ATTRIBUTE_READONLY);
    4295           5 :         if (correct == false) {
    4296           0 :                 goto out;
    4297             :         }
    4298             : 
    4299             :         /* Setting to FILE_ATTRIBUTE_NORMAL should be ignored. */
    4300           5 :         status = cli_setpathinfo_ext(
    4301             :                 cli,
    4302             :                 fname,
    4303           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* create */
    4304           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* access */
    4305           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* write */
    4306           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* change */
    4307             :                 FILE_ATTRIBUTE_NORMAL);
    4308           5 :         if (!NT_STATUS_IS_OK(status)) {
    4309           0 :                 printf("cli_setpathinfo_ext failed with %s\n",
    4310             :                         nt_errstr(status));
    4311           0 :                 correct = false;
    4312             :         }
    4313             : 
    4314             :         /* Check attributes are correct. */
    4315           5 :         correct = check_attributes(cli,
    4316             :                         fname,
    4317             :                         FILE_ATTRIBUTE_SYSTEM |
    4318             :                         FILE_ATTRIBUTE_HIDDEN |
    4319             :                         FILE_ATTRIBUTE_READONLY);
    4320           5 :         if (correct == false) {
    4321           0 :                 goto out;
    4322             :         }
    4323             : 
    4324             :         /* Setting to (uint16_t)-1 should also be ignored. */
    4325           5 :         status = cli_setpathinfo_ext(
    4326             :                 cli,
    4327             :                 fname,
    4328           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* create */
    4329           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* access */
    4330           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* write */
    4331           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* change */
    4332             :                 (uint32_t)-1);
    4333           5 :         if (!NT_STATUS_IS_OK(status)) {
    4334           0 :                 printf("cli_setpathinfo_ext failed with %s\n",
    4335             :                         nt_errstr(status));
    4336           0 :                 correct = false;
    4337             :         }
    4338             : 
    4339             :         /* Check attributes are correct. */
    4340           5 :         correct = check_attributes(cli,
    4341             :                         fname,
    4342             :                         FILE_ATTRIBUTE_SYSTEM |
    4343             :                         FILE_ATTRIBUTE_HIDDEN |
    4344             :                         FILE_ATTRIBUTE_READONLY);
    4345           5 :         if (correct == false) {
    4346           0 :                 goto out;
    4347             :         }
    4348             : 
    4349             :         /* Setting to 0 should clear them all. */
    4350           5 :         status = cli_setpathinfo_ext(
    4351             :                 cli,
    4352             :                 fname,
    4353           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* create */
    4354           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* access */
    4355           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* write */
    4356           5 :                 (struct timespec) { .tv_nsec = SAMBA_UTIME_OMIT }, /* change */
    4357             :                 0);
    4358           5 :         if (!NT_STATUS_IS_OK(status)) {
    4359           0 :                 printf("cli_setpathinfo_ext failed with %s\n",
    4360             :                         nt_errstr(status));
    4361           0 :                 correct = false;
    4362             :         }
    4363             : 
    4364             :         /* Check attributes are correct. */
    4365           5 :         correct = check_attributes(cli,
    4366             :                         fname,
    4367             :                         FILE_ATTRIBUTE_NORMAL);
    4368           5 :         if (correct == false) {
    4369           0 :                 goto out;
    4370             :         }
    4371             : 
    4372           5 :   out:
    4373             : 
    4374           5 :         cli_unlink(cli,
    4375             :                 fname,
    4376             :                 FILE_ATTRIBUTE_SYSTEM |
    4377             :                 FILE_ATTRIBUTE_HIDDEN|
    4378             :                 FILE_ATTRIBUTE_READONLY);
    4379             : 
    4380           5 :         if (!torture_close_connection(cli)) {
    4381           0 :                 correct = False;
    4382             :         }
    4383             : 
    4384           5 :         printf("attrib test finished\n");
    4385             : 
    4386           5 :         return correct;
    4387             : }
    4388             : 
    4389           5 : static NTSTATUS cli_qfilename(
    4390             :         struct cli_state *cli,
    4391             :         uint16_t fnum,
    4392             :         TALLOC_CTX *mem_ctx,
    4393             :         char **_name)
    4394             : {
    4395           0 :         uint16_t recv_flags2;
    4396           0 :         uint8_t *rdata;
    4397           0 :         uint32_t num_rdata;
    4398           0 :         NTSTATUS status;
    4399           5 :         char *name = NULL;
    4400           0 :         uint32_t namelen;
    4401             : 
    4402           5 :         status = cli_qfileinfo(talloc_tos(), cli, fnum,
    4403             :                                SMB_QUERY_FILE_NAME_INFO,
    4404             :                                4, CLI_BUFFER_SIZE, &recv_flags2,
    4405             :                                &rdata, &num_rdata);
    4406           5 :         if (!NT_STATUS_IS_OK(status)) {
    4407           0 :                 return status;
    4408             :         }
    4409             : 
    4410           5 :         namelen = IVAL(rdata, 0);
    4411           5 :         if (namelen > (num_rdata - 4)) {
    4412           0 :                 TALLOC_FREE(rdata);
    4413           0 :                 return NT_STATUS_INVALID_NETWORK_RESPONSE;
    4414             :         }
    4415             : 
    4416           5 :         pull_string_talloc(mem_ctx,
    4417             :                            (const char *)rdata,
    4418             :                            recv_flags2,
    4419             :                            &name,
    4420           5 :                            rdata + 4,
    4421             :                            namelen,
    4422             :                            STR_UNICODE);
    4423           5 :         if (name == NULL) {
    4424           0 :                 status = map_nt_error_from_unix(errno);
    4425           0 :                 TALLOC_FREE(rdata);
    4426           0 :                 return status;
    4427             :         }
    4428             : 
    4429           5 :         *_name = name;
    4430           5 :         TALLOC_FREE(rdata);
    4431           5 :         return NT_STATUS_OK;
    4432             : }
    4433             : 
    4434             : /*
    4435             :   This checks a couple of trans2 calls
    4436             : */
    4437           5 : static bool run_trans2test(int dummy)
    4438             : {
    4439           0 :         struct cli_state *cli;
    4440           0 :         uint16_t fnum;
    4441           0 :         off_t size;
    4442           0 :         time_t c_time, a_time, m_time;
    4443           0 :         struct timespec c_time_ts, a_time_ts, m_time_ts, w_time_ts, m_time2_ts;
    4444           5 :         const char *fname = "\\trans2.tst";
    4445           5 :         const char *dname = "\\trans2";
    4446           5 :         const char *fname2 = "\\trans2\\trans2.tst";
    4447           5 :         char *pname = NULL;
    4448           5 :         bool correct = True;
    4449           0 :         NTSTATUS status;
    4450           0 :         uint32_t fs_attr;
    4451           0 :         uint64_t ino;
    4452             : 
    4453           5 :         printf("starting trans2 test\n");
    4454             : 
    4455           5 :         if (!torture_open_connection(&cli, 0)) {
    4456           0 :                 return False;
    4457             :         }
    4458             : 
    4459           5 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    4460             :                 /* Ensure ino is zero, SMB2 gets a real one. */
    4461           0 :                 ino = 0;
    4462             :         } else {
    4463             :                 /* Ensure ino is -1, SMB1 never gets a real one. */
    4464           5 :                 ino = (uint64_t)-1;
    4465             :         }
    4466             : 
    4467           5 :         status = cli_get_fs_attr_info(cli, &fs_attr);
    4468           5 :         if (!NT_STATUS_IS_OK(status)) {
    4469           0 :                 printf("ERROR: cli_get_fs_attr_info returned %s\n",
    4470             :                        nt_errstr(status));
    4471           0 :                 correct = false;
    4472             :         }
    4473             : 
    4474           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4475           5 :         cli_openx(cli, fname, O_RDWR | O_CREAT | O_TRUNC, DENY_NONE, &fnum);
    4476           5 :         status = cli_qfileinfo_basic(cli, fnum, NULL, &size, &c_time_ts,
    4477             :                                      &a_time_ts, &w_time_ts, &m_time_ts, NULL);
    4478           5 :         if (!NT_STATUS_IS_OK(status)) {
    4479           0 :                 printf("ERROR: qfileinfo failed (%s)\n", nt_errstr(status));
    4480           0 :                 correct = False;
    4481             :         }
    4482             : 
    4483           5 :         status = cli_qfilename(cli, fnum, talloc_tos(), &pname);
    4484           5 :         if (!NT_STATUS_IS_OK(status)) {
    4485           0 :                 printf("ERROR: qfilename failed (%s)\n", nt_errstr(status));
    4486           0 :                 correct = False;
    4487             :         }
    4488           5 :         else if (strcmp(pname, fname)) {
    4489           0 :                 printf("qfilename gave different name? [%s] [%s]\n",
    4490             :                        fname, pname);
    4491           0 :                 correct = False;
    4492             :         }
    4493             : 
    4494           5 :         cli_close(cli, fnum);
    4495             : 
    4496           5 :         sleep(2);
    4497             : 
    4498           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4499           5 :         status = cli_openx(cli, fname, O_RDWR | O_CREAT | O_TRUNC, DENY_NONE,
    4500             :                           &fnum);
    4501           5 :         if (!NT_STATUS_IS_OK(status)) {
    4502           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    4503           0 :                 return False;
    4504             :         }
    4505           5 :         cli_close(cli, fnum);
    4506             : 
    4507           5 :         status = cli_qpathinfo1(cli, fname, &c_time, &a_time, &m_time, &size,
    4508             :                                 NULL);
    4509           5 :         if (!NT_STATUS_IS_OK(status)) {
    4510           0 :                 printf("ERROR: qpathinfo failed (%s)\n", nt_errstr(status));
    4511           0 :                 correct = False;
    4512             :         } else {
    4513           5 :                 time_t t = time(NULL);
    4514             : 
    4515           5 :                 if (c_time != m_time) {
    4516           0 :                         printf("create time=%s", ctime(&c_time));
    4517           0 :                         printf("modify time=%s", ctime(&m_time));
    4518           0 :                         printf("This system appears to have sticky create times\n");
    4519             :                 }
    4520           5 :                 if ((labs(a_time - t) > 60) && (a_time % (60*60) == 0)) {
    4521           0 :                         printf("access time=%s", ctime(&a_time));
    4522           0 :                         printf("This system appears to set a midnight access time\n");
    4523           0 :                         correct = False;
    4524             :                 }
    4525             : 
    4526           5 :                 if (labs(m_time - t) > 60*60*24*7) {
    4527           0 :                         printf("ERROR: totally incorrect times - maybe word reversed? mtime=%s", ctime(&m_time));
    4528           0 :                         correct = False;
    4529             :                 }
    4530             :         }
    4531             : 
    4532             : 
    4533           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4534           5 :         cli_openx(cli, fname, 
    4535             :                         O_RDWR | O_CREAT | O_TRUNC, DENY_NONE, &fnum);
    4536           5 :         cli_close(cli, fnum);
    4537           5 :         status = cli_qpathinfo2(cli,
    4538             :                                 fname,
    4539             :                                 &c_time_ts,
    4540             :                                 &a_time_ts,
    4541             :                                 &w_time_ts,
    4542             :                                 &m_time_ts,
    4543             :                                 &size,
    4544             :                                 NULL,
    4545             :                                 &ino,
    4546             :                                 NULL);
    4547           5 :         if (!NT_STATUS_IS_OK(status)) {
    4548           0 :                 printf("ERROR: qpathinfo2 failed (%s)\n", nt_errstr(status));
    4549           0 :                 correct = False;
    4550             :         } else {
    4551           5 :                 if (w_time_ts.tv_sec < 60*60*24*2) {
    4552           0 :                         printf("write time=%s", ctime(&w_time_ts.tv_sec));
    4553           0 :                         printf("This system appears to set a initial 0 write time\n");
    4554           0 :                         correct = False;
    4555             :                 }
    4556           5 :                 if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
    4557             :                         /* SMB2 should always return an inode. */
    4558           0 :                         if (ino == 0) {
    4559           0 :                                 printf("SMB2 bad inode (0)\n");
    4560           0 :                                 correct = false;
    4561             :                         }
    4562             :                 } else {
    4563             :                         /* SMB1 must always return zero here. */
    4564           5 :                         if (ino != 0) {
    4565           0 :                                 printf("SMB1 bad inode (!0)\n");
    4566           0 :                                 correct = false;
    4567             :                         }
    4568             :                 }
    4569             :         }
    4570             : 
    4571           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4572             : 
    4573             : 
    4574             :         /* check if the server updates the directory modification time
    4575             :            when creating a new file */
    4576           5 :         status = cli_mkdir(cli, dname);
    4577           5 :         if (!NT_STATUS_IS_OK(status)) {
    4578           0 :                 printf("ERROR: mkdir failed (%s)\n", nt_errstr(status));
    4579           0 :                 correct = False;
    4580             :         }
    4581           5 :         sleep(3);
    4582           5 :         status = cli_qpathinfo2(cli,
    4583             :                                 "\\trans2\\",
    4584             :                                 &c_time_ts,
    4585             :                                 &a_time_ts,
    4586             :                                 &w_time_ts,
    4587             :                                 &m_time_ts,
    4588             :                                 &size,
    4589             :                                 NULL,
    4590             :                                 NULL,
    4591             :                                 NULL);
    4592           5 :         if (!NT_STATUS_IS_OK(status)) {
    4593           0 :                 printf("ERROR: qpathinfo2 failed (%s)\n", nt_errstr(status));
    4594           0 :                 correct = False;
    4595             :         }
    4596             : 
    4597           5 :         cli_openx(cli, fname2, 
    4598             :                         O_RDWR | O_CREAT | O_TRUNC, DENY_NONE, &fnum);
    4599           5 :         cli_writeall(cli, fnum,  0, (uint8_t *)&fnum, 0, sizeof(fnum), NULL);
    4600           5 :         cli_close(cli, fnum);
    4601           5 :         status = cli_qpathinfo2(cli,
    4602             :                                 "\\trans2\\",
    4603             :                                 &c_time_ts,
    4604             :                                 &a_time_ts,
    4605             :                                 &w_time_ts,
    4606             :                                 &m_time2_ts,
    4607             :                                 &size,
    4608             :                                 NULL,
    4609             :                                 NULL,
    4610             :                                 NULL);
    4611           5 :         if (!NT_STATUS_IS_OK(status)) {
    4612           0 :                 printf("ERROR: qpathinfo2 failed (%s)\n", nt_errstr(status));
    4613           0 :                 correct = False;
    4614             :         } else {
    4615           5 :                 if (memcmp(&m_time_ts, &m_time2_ts, sizeof(struct timespec))
    4616             :                     == 0) {
    4617           0 :                         printf("This system does not update directory modification times\n");
    4618           0 :                         correct = False;
    4619             :                 }
    4620             :         }
    4621           5 :         cli_unlink(cli, fname2, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4622           5 :         cli_rmdir(cli, dname);
    4623             : 
    4624           5 :         if (!torture_close_connection(cli)) {
    4625           0 :                 correct = False;
    4626             :         }
    4627             : 
    4628           5 :         printf("trans2 test finished\n");
    4629             : 
    4630           5 :         return correct;
    4631             : }
    4632             : 
    4633             : /*
    4634             :   This checks new W2K calls.
    4635             : */
    4636             : 
    4637         180 : static NTSTATUS new_trans(struct cli_state *pcli, int fnum, int level)
    4638             : {
    4639         180 :         uint8_t *buf = NULL;
    4640           0 :         uint32_t len;
    4641           0 :         NTSTATUS status;
    4642             : 
    4643         180 :         status = cli_qfileinfo(talloc_tos(), pcli, fnum, level, 0,
    4644             :                                CLI_BUFFER_SIZE, NULL, &buf, &len);
    4645         180 :         if (!NT_STATUS_IS_OK(status)) {
    4646         180 :                 printf("ERROR: qfileinfo (%d) failed (%s)\n", level,
    4647             :                        nt_errstr(status));
    4648             :         } else {
    4649           0 :                 printf("qfileinfo: level %d, len = %u\n", level, len);
    4650           0 :                 dump_data(0, (uint8_t *)buf, len);
    4651           0 :                 printf("\n");
    4652             :         }
    4653         180 :         TALLOC_FREE(buf);
    4654         180 :         return status;
    4655             : }
    4656             : 
    4657           5 : static bool run_w2ktest(int dummy)
    4658             : {
    4659           0 :         struct cli_state *cli;
    4660           0 :         uint16_t fnum;
    4661           5 :         const char *fname = "\\w2ktest\\w2k.tst";
    4662           0 :         int level;
    4663           5 :         bool correct = True;
    4664             : 
    4665           5 :         printf("starting w2k test\n");
    4666             : 
    4667           5 :         if (!torture_open_connection(&cli, 0)) {
    4668           0 :                 return False;
    4669             :         }
    4670             : 
    4671           5 :         cli_openx(cli, fname, 
    4672             :                         O_RDWR | O_CREAT , DENY_NONE, &fnum);
    4673             : 
    4674         185 :         for (level = 1004; level < 1040; level++) {
    4675         180 :                 new_trans(cli, fnum, level);
    4676             :         }
    4677             : 
    4678           5 :         cli_close(cli, fnum);
    4679             : 
    4680           5 :         if (!torture_close_connection(cli)) {
    4681           0 :                 correct = False;
    4682             :         }
    4683             : 
    4684           5 :         printf("w2k test finished\n");
    4685             : 
    4686           5 :         return correct;
    4687             : }
    4688             : 
    4689             : 
    4690             : /*
    4691             :   this is a harness for some oplock tests
    4692             :  */
    4693           5 : static bool run_oplock1(int dummy)
    4694             : {
    4695           0 :         struct cli_state *cli1;
    4696           5 :         const char *fname = "\\lockt1.lck";
    4697           0 :         uint16_t fnum1;
    4698           5 :         bool correct = True;
    4699           0 :         NTSTATUS status;
    4700             : 
    4701           5 :         printf("starting oplock test 1\n");
    4702             : 
    4703           5 :         if (!torture_open_connection(&cli1, 0)) {
    4704           0 :                 return False;
    4705             :         }
    4706             : 
    4707           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4708             : 
    4709           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    4710             : 
    4711           5 :         cli1->use_oplocks = True;
    4712             : 
    4713           5 :         status = cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE,
    4714             :                           &fnum1);
    4715           5 :         if (!NT_STATUS_IS_OK(status)) {
    4716           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    4717           0 :                 return False;
    4718             :         }
    4719             : 
    4720           5 :         cli1->use_oplocks = False;
    4721             : 
    4722           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4723           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4724             : 
    4725           5 :         status = cli_close(cli1, fnum1);
    4726           5 :         if (!NT_STATUS_IS_OK(status)) {
    4727           0 :                 printf("close2 failed (%s)\n", nt_errstr(status));
    4728           0 :                 return False;
    4729             :         }
    4730             : 
    4731           5 :         status = cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4732           5 :         if (!NT_STATUS_IS_OK(status)) {
    4733           0 :                 printf("unlink failed (%s)\n", nt_errstr(status));
    4734           0 :                 return False;
    4735             :         }
    4736             : 
    4737           5 :         if (!torture_close_connection(cli1)) {
    4738           0 :                 correct = False;
    4739             :         }
    4740             : 
    4741           5 :         printf("finished oplock test 1\n");
    4742             : 
    4743           5 :         return correct;
    4744             : }
    4745             : 
    4746           4 : static bool run_oplock2(int dummy)
    4747             : {
    4748           0 :         struct cli_state *cli1, *cli2;
    4749           4 :         const char *fname = "\\lockt2.lck";
    4750           0 :         uint16_t fnum1, fnum2;
    4751           4 :         int saved_use_oplocks = use_oplocks;
    4752           0 :         char buf[4];
    4753           4 :         bool correct = True;
    4754           0 :         volatile bool *shared_correct;
    4755           0 :         size_t nread;
    4756           0 :         NTSTATUS status;
    4757             : 
    4758           4 :         shared_correct = (volatile bool *)anonymous_shared_allocate(sizeof(bool));
    4759           4 :         *shared_correct = True;
    4760             : 
    4761           4 :         use_level_II_oplocks = True;
    4762           4 :         use_oplocks = True;
    4763             : 
    4764           4 :         printf("starting oplock test 2\n");
    4765             : 
    4766           4 :         if (!torture_open_connection(&cli1, 0)) {
    4767           0 :                 use_level_II_oplocks = False;
    4768           0 :                 use_oplocks = saved_use_oplocks;
    4769           0 :                 return False;
    4770             :         }
    4771             : 
    4772           4 :         if (!torture_open_connection(&cli2, 1)) {
    4773           0 :                 use_level_II_oplocks = False;
    4774           0 :                 use_oplocks = saved_use_oplocks;
    4775           0 :                 return False;
    4776             :         }
    4777             : 
    4778           4 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4779             : 
    4780           4 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    4781           4 :         smbXcli_conn_set_sockopt(cli2->conn, sockops);
    4782             : 
    4783           4 :         status = cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE,
    4784             :                           &fnum1);
    4785           4 :         if (!NT_STATUS_IS_OK(status)) {
    4786           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    4787           0 :                 return False;
    4788             :         }
    4789             : 
    4790             :         /* Don't need the globals any more. */
    4791           4 :         use_level_II_oplocks = False;
    4792           4 :         use_oplocks = saved_use_oplocks;
    4793             : 
    4794           4 :         if (fork() == 0) {
    4795             :                 /* Child code */
    4796           4 :                 status = cli_openx(cli2, fname, O_RDWR, DENY_NONE, &fnum2);
    4797           4 :                 if (!NT_STATUS_IS_OK(status)) {
    4798           0 :                         printf("second open of %s failed (%s)\n", fname, nt_errstr(status));
    4799           0 :                         *shared_correct = False;
    4800           0 :                         exit(0);
    4801             :                 }
    4802             : 
    4803           4 :                 sleep(2);
    4804             : 
    4805           4 :                 status = cli_close(cli2, fnum2);
    4806           4 :                 if (!NT_STATUS_IS_OK(status)) {
    4807           0 :                         printf("close2 failed (%s)\n", nt_errstr(status));
    4808           0 :                         *shared_correct = False;
    4809             :                 }
    4810             : 
    4811           4 :                 exit(0);
    4812             :         }
    4813             : 
    4814           4 :         sleep(2);
    4815             : 
    4816             :         /* Ensure cli1 processes the break. Empty file should always return 0
    4817             :          * bytes.  */
    4818           4 :         status = cli_read(cli1, fnum1, buf, 0, 4, &nread);
    4819           4 :         if (!NT_STATUS_IS_OK(status)) {
    4820           0 :                 printf("read on fnum1 failed (%s)\n", nt_errstr(status));
    4821           0 :                 correct = false;
    4822           4 :         } else if (nread != 0) {
    4823           0 :                 printf("read on empty fnum1 failed. recv %ld expected %d\n",
    4824             :                       (unsigned long)nread, 0);
    4825           0 :                 correct = false;
    4826             :         }
    4827             : 
    4828             :         /* Should now be at level II. */
    4829             :         /* Test if sending a write locks causes a break to none. */
    4830           4 :         status = cli_lock32(cli1, fnum1, 0, 4, 0, READ_LOCK);
    4831           4 :         if (!NT_STATUS_IS_OK(status)) {
    4832           0 :                 printf("lock failed (%s)\n", nt_errstr(status));
    4833           0 :                 correct = False;
    4834             :         }
    4835             : 
    4836           4 :         cli_unlock(cli1, fnum1, 0, 4);
    4837             : 
    4838           4 :         sleep(2);
    4839             : 
    4840           4 :         status = cli_lock32(cli1, fnum1, 0, 4, 0, WRITE_LOCK);
    4841           4 :         if (!NT_STATUS_IS_OK(status)) {
    4842           0 :                 printf("lock failed (%s)\n", nt_errstr(status));
    4843           0 :                 correct = False;
    4844             :         }
    4845             : 
    4846           4 :         cli_unlock(cli1, fnum1, 0, 4);
    4847             : 
    4848           4 :         sleep(2);
    4849             : 
    4850           4 :         cli_read(cli1, fnum1, buf, 0, 4, NULL);
    4851             : 
    4852           4 :         status = cli_close(cli1, fnum1);
    4853           4 :         if (!NT_STATUS_IS_OK(status)) {
    4854           0 :                 printf("close1 failed (%s)\n", nt_errstr(status));
    4855           0 :                 correct = False;
    4856             :         }
    4857             : 
    4858           4 :         sleep(4);
    4859             : 
    4860           4 :         status = cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4861           4 :         if (!NT_STATUS_IS_OK(status)) {
    4862           0 :                 printf("unlink failed (%s)\n", nt_errstr(status));
    4863           0 :                 correct = False;
    4864             :         }
    4865             : 
    4866           4 :         if (!torture_close_connection(cli1)) {
    4867           0 :                 correct = False;
    4868             :         }
    4869             : 
    4870           4 :         if (!*shared_correct) {
    4871           0 :                 correct = False;
    4872             :         }
    4873             : 
    4874           4 :         printf("finished oplock test 2\n");
    4875             : 
    4876           4 :         return correct;
    4877             : }
    4878             : 
    4879             : struct oplock4_state {
    4880             :         struct tevent_context *ev;
    4881             :         struct cli_state *cli;
    4882             :         bool *got_break;
    4883             :         uint16_t *fnum2;
    4884             : };
    4885             : 
    4886             : static void oplock4_got_break(struct tevent_req *req);
    4887             : static void oplock4_got_open(struct tevent_req *req);
    4888             : 
    4889           5 : static bool run_oplock4(int dummy)
    4890             : {
    4891           0 :         struct tevent_context *ev;
    4892           0 :         struct cli_state *cli1, *cli2;
    4893           0 :         struct tevent_req *oplock_req, *open_req;
    4894           5 :         const char *fname = "\\lockt4.lck";
    4895           5 :         const char *fname_ln = "\\lockt4_ln.lck";
    4896           0 :         uint16_t fnum1, fnum2;
    4897           5 :         int saved_use_oplocks = use_oplocks;
    4898           0 :         NTSTATUS status;
    4899           5 :         bool correct = true;
    4900             : 
    4901           0 :         bool got_break;
    4902             : 
    4903           0 :         struct oplock4_state *state;
    4904             : 
    4905           5 :         printf("starting oplock test 4\n");
    4906             : 
    4907           5 :         if (!torture_open_connection(&cli1, 0)) {
    4908           0 :                 use_level_II_oplocks = false;
    4909           0 :                 use_oplocks = saved_use_oplocks;
    4910           0 :                 return false;
    4911             :         }
    4912             : 
    4913           5 :         if (!torture_open_connection(&cli2, 1)) {
    4914           0 :                 use_level_II_oplocks = false;
    4915           0 :                 use_oplocks = saved_use_oplocks;
    4916           0 :                 return false;
    4917             :         }
    4918             : 
    4919           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4920           5 :         cli_unlink(cli1, fname_ln, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    4921             : 
    4922           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    4923           5 :         smbXcli_conn_set_sockopt(cli2->conn, sockops);
    4924             : 
    4925             :         /* Create the file. */
    4926           5 :         status = cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE,
    4927             :                           &fnum1);
    4928           5 :         if (!NT_STATUS_IS_OK(status)) {
    4929           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    4930           0 :                 return false;
    4931             :         }
    4932             : 
    4933           5 :         status = cli_close(cli1, fnum1);
    4934           5 :         if (!NT_STATUS_IS_OK(status)) {
    4935           0 :                 printf("close1 failed (%s)\n", nt_errstr(status));
    4936           0 :                 return false;
    4937             :         }
    4938             : 
    4939             :         /* Now create a hardlink. */
    4940           5 :         status = cli_hardlink(cli1, fname, fname_ln);
    4941           5 :         if (!NT_STATUS_IS_OK(status)) {
    4942           0 :                 printf("nt hardlink failed (%s)\n", nt_errstr(status));
    4943           0 :                 return false;
    4944             :         }
    4945             : 
    4946             :         /* Prove that opening hardlinks cause deny modes to conflict. */
    4947           5 :         status = cli_openx(cli1, fname, O_RDWR, DENY_ALL, &fnum1);
    4948           5 :         if (!NT_STATUS_IS_OK(status)) {
    4949           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    4950           0 :                 return false;
    4951             :         }
    4952             : 
    4953           5 :         status = cli_openx(cli1, fname_ln, O_RDWR, DENY_NONE, &fnum2);
    4954           5 :         if (NT_STATUS_IS_OK(status)) {
    4955           0 :                 printf("open of %s succeeded - should fail with sharing violation.\n",
    4956             :                         fname_ln);
    4957           0 :                 return false;
    4958             :         }
    4959             : 
    4960           5 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_SHARING_VIOLATION)) {
    4961           0 :                 printf("open of %s should fail with sharing violation. Got %s\n",
    4962             :                         fname_ln, nt_errstr(status));
    4963           0 :                 return false;
    4964             :         }
    4965             : 
    4966           5 :         status = cli_close(cli1, fnum1);
    4967           5 :         if (!NT_STATUS_IS_OK(status)) {
    4968           0 :                 printf("close1 failed (%s)\n", nt_errstr(status));
    4969           0 :                 return false;
    4970             :         }
    4971             : 
    4972           5 :         cli1->use_oplocks = true;
    4973           5 :         cli2->use_oplocks = true;
    4974             : 
    4975           5 :         status = cli_openx(cli1, fname, O_RDWR, DENY_NONE, &fnum1);
    4976           5 :         if (!NT_STATUS_IS_OK(status)) {
    4977           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    4978           0 :                 return false;
    4979             :         }
    4980             : 
    4981           5 :         ev = samba_tevent_context_init(talloc_tos());
    4982           5 :         if (ev == NULL) {
    4983           0 :                 printf("tevent_context_init failed\n");
    4984           0 :                 return false;
    4985             :         }
    4986             : 
    4987           5 :         state = talloc(ev, struct oplock4_state);
    4988           5 :         if (state == NULL) {
    4989           0 :                 printf("talloc failed\n");
    4990           0 :                 return false;
    4991             :         }
    4992           5 :         state->ev = ev;
    4993           5 :         state->cli = cli1;
    4994           5 :         state->got_break = &got_break;
    4995           5 :         state->fnum2 = &fnum2;
    4996             : 
    4997           5 :         oplock_req = cli_smb_oplock_break_waiter_send(
    4998             :                 talloc_tos(), ev, cli1);
    4999           5 :         if (oplock_req == NULL) {
    5000           0 :                 printf("cli_smb_oplock_break_waiter_send failed\n");
    5001           0 :                 return false;
    5002             :         }
    5003           5 :         tevent_req_set_callback(oplock_req, oplock4_got_break, state);
    5004             : 
    5005           5 :         open_req = cli_openx_send(
    5006             :                 talloc_tos(), ev, cli2, fname_ln, O_RDWR, DENY_NONE);
    5007           5 :         if (open_req == NULL) {
    5008           0 :                 printf("cli_openx_send failed\n");
    5009           0 :                 return false;
    5010             :         }
    5011           5 :         tevent_req_set_callback(open_req, oplock4_got_open, state);
    5012             : 
    5013           5 :         got_break = false;
    5014           5 :         fnum2 = 0xffff;
    5015             : 
    5016          40 :         while (!got_break || fnum2 == 0xffff) {
    5017           0 :                 int ret;
    5018          35 :                 ret = tevent_loop_once(ev);
    5019          35 :                 if (ret == -1) {
    5020           0 :                         printf("tevent_loop_once failed: %s\n",
    5021           0 :                                strerror(errno));
    5022           0 :                         return false;
    5023             :                 }
    5024             :         }
    5025             : 
    5026           5 :         status = cli_close(cli2, fnum2);
    5027           5 :         if (!NT_STATUS_IS_OK(status)) {
    5028           0 :                 printf("close2 failed (%s)\n", nt_errstr(status));
    5029           0 :                 correct = false;
    5030             :         }
    5031             : 
    5032           5 :         status = cli_close(cli1, fnum1);
    5033           5 :         if (!NT_STATUS_IS_OK(status)) {
    5034           0 :                 printf("close1 failed (%s)\n", nt_errstr(status));
    5035           0 :                 correct = false;
    5036             :         }
    5037             : 
    5038           5 :         status = cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5039           5 :         if (!NT_STATUS_IS_OK(status)) {
    5040           0 :                 printf("unlink failed (%s)\n", nt_errstr(status));
    5041           0 :                 correct = false;
    5042             :         }
    5043             : 
    5044           5 :         status = cli_unlink(cli1, fname_ln, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5045           5 :         if (!NT_STATUS_IS_OK(status)) {
    5046           0 :                 printf("unlink failed (%s)\n", nt_errstr(status));
    5047           0 :                 correct = false;
    5048             :         }
    5049             : 
    5050           5 :         if (!torture_close_connection(cli1)) {
    5051           0 :                 correct = false;
    5052             :         }
    5053             : 
    5054           5 :         if (!got_break) {
    5055           0 :                 correct = false;
    5056             :         }
    5057             : 
    5058           5 :         printf("finished oplock test 4\n");
    5059             : 
    5060           5 :         return correct;
    5061             : }
    5062             : 
    5063           5 : static void oplock4_got_break(struct tevent_req *req)
    5064             : {
    5065           5 :         struct oplock4_state *state = tevent_req_callback_data(
    5066             :                 req, struct oplock4_state);
    5067           0 :         uint16_t fnum;
    5068           0 :         uint8_t level;
    5069           0 :         NTSTATUS status;
    5070             : 
    5071           5 :         status = cli_smb_oplock_break_waiter_recv(req, &fnum, &level);
    5072           5 :         TALLOC_FREE(req);
    5073           5 :         if (!NT_STATUS_IS_OK(status)) {
    5074           0 :                 printf("cli_smb_oplock_break_waiter_recv returned %s\n",
    5075             :                        nt_errstr(status));
    5076           0 :                 return;
    5077             :         }
    5078           5 :         *state->got_break = true;
    5079             : 
    5080           5 :         req = cli_oplock_ack_send(state, state->ev, state->cli, fnum,
    5081             :                                   NO_OPLOCK);
    5082           5 :         if (req == NULL) {
    5083           0 :                 printf("cli_oplock_ack_send failed\n");
    5084           0 :                 return;
    5085             :         }
    5086             : }
    5087             : 
    5088           5 : static void oplock4_got_open(struct tevent_req *req)
    5089             : {
    5090           5 :         struct oplock4_state *state = tevent_req_callback_data(
    5091             :                 req, struct oplock4_state);
    5092           0 :         NTSTATUS status;
    5093             : 
    5094           5 :         status = cli_openx_recv(req, state->fnum2);
    5095           5 :         if (!NT_STATUS_IS_OK(status)) {
    5096           0 :                 printf("cli_openx_recv returned %s\n", nt_errstr(status));
    5097           0 :                 *state->fnum2 = 0xffff;
    5098             :         }
    5099           5 : }
    5100             : 
    5101             : #ifdef HAVE_KERNEL_OPLOCKS_LINUX
    5102             : 
    5103             : struct oplock5_state {
    5104             :         int pipe_down_fd;
    5105             : };
    5106             : 
    5107             : /*
    5108             :  * Async open the file that has a kernel oplock, do an echo to get
    5109             :  * that 100% across, close the file to signal to the child fd that the
    5110             :  * oplock can be dropped, wait for the open reply.
    5111             :  */
    5112             : 
    5113             : static void oplock5_opened(struct tevent_req *subreq);
    5114             : static void oplock5_pong(struct tevent_req *subreq);
    5115             : static void oplock5_timedout(struct tevent_req *subreq);
    5116             : 
    5117           1 : static struct tevent_req *oplock5_send(
    5118             :         TALLOC_CTX *mem_ctx,
    5119             :         struct tevent_context *ev,
    5120             :         struct cli_state *cli,
    5121             :         const char *fname,
    5122             :         int pipe_down_fd)
    5123             : {
    5124           1 :         struct tevent_req *req = NULL, *subreq = NULL;
    5125           1 :         struct oplock5_state *state = NULL;
    5126           0 :         static uint8_t data = 0;
    5127             : 
    5128           1 :         req = tevent_req_create(mem_ctx, &state, struct oplock5_state);
    5129           1 :         if (req == NULL) {
    5130           0 :                 return NULL;
    5131             :         }
    5132           1 :         state->pipe_down_fd = pipe_down_fd;
    5133             : 
    5134           1 :         subreq = cli_ntcreate_send(
    5135             :                 state,
    5136             :                 ev,
    5137             :                 cli,
    5138             :                 fname,
    5139             :                 0,                      /* CreatFlags */
    5140             :                 SEC_FILE_READ_DATA,    /* DesiredAccess */
    5141             :                 FILE_ATTRIBUTE_NORMAL,  /* FileAttributes */
    5142             :                 FILE_SHARE_WRITE|FILE_SHARE_READ, /* ShareAccess */
    5143             :                 FILE_OPEN,               /* CreateDisposition */
    5144             :                 FILE_NON_DIRECTORY_FILE, /* CreateOptions */
    5145             :                 0,                       /* Impersonation */
    5146             :                 0);                      /* SecurityFlags */
    5147           1 :         if (tevent_req_nomem(subreq, req)) {
    5148           0 :                 return tevent_req_post(req, ev);
    5149             :         }
    5150           1 :         tevent_req_set_callback(subreq, oplock5_opened, req);
    5151             : 
    5152           1 :         subreq = cli_echo_send(
    5153             :                 state,
    5154             :                 ev,
    5155             :                 cli,
    5156             :                 1,
    5157           1 :                 (DATA_BLOB) { .data = &data, .length = sizeof(data) });
    5158           1 :         if (tevent_req_nomem(subreq, req)) {
    5159           0 :                 return tevent_req_post(req, ev);
    5160             :         }
    5161           1 :         tevent_req_set_callback(subreq, oplock5_pong, req);
    5162             : 
    5163           1 :         subreq = tevent_wakeup_send(state, ev, timeval_current_ofs(20, 0));
    5164           1 :         if (tevent_req_nomem(subreq, req)) {
    5165           0 :                 return tevent_req_post(req, ev);
    5166             :         }
    5167           1 :         tevent_req_set_callback(subreq, oplock5_timedout, req);
    5168             : 
    5169           1 :         return req;
    5170             : }
    5171             : 
    5172           1 : static void oplock5_opened(struct tevent_req *subreq)
    5173             : {
    5174           1 :         struct tevent_req *req = tevent_req_callback_data(
    5175             :                 subreq, struct tevent_req);
    5176           0 :         NTSTATUS status;
    5177           0 :         uint16_t fnum;
    5178             : 
    5179           1 :         status = cli_ntcreate_recv(subreq, &fnum, NULL);
    5180           1 :         TALLOC_FREE(subreq);
    5181           1 :         if (tevent_req_nterror(req, status)) {
    5182           0 :                 return;
    5183             :         }
    5184           1 :         tevent_req_done(req);
    5185             : }
    5186             : 
    5187           1 : static void oplock5_pong(struct tevent_req *subreq)
    5188             : {
    5189           1 :         struct tevent_req *req = tevent_req_callback_data(
    5190             :                 subreq, struct tevent_req);
    5191           1 :         struct oplock5_state *state = tevent_req_data(
    5192             :                 req, struct oplock5_state);
    5193           0 :         NTSTATUS status;
    5194             : 
    5195           1 :         status = cli_echo_recv(subreq);
    5196           1 :         TALLOC_FREE(subreq);
    5197           1 :         if (tevent_req_nterror(req, status)) {
    5198           0 :                 return;
    5199             :         }
    5200             : 
    5201           1 :         close(state->pipe_down_fd);
    5202             : }
    5203             : 
    5204           0 : static void oplock5_timedout(struct tevent_req *subreq)
    5205             : {
    5206           0 :         struct tevent_req *req = tevent_req_callback_data(
    5207             :                 subreq, struct tevent_req);
    5208           0 :         bool ok;
    5209             : 
    5210           0 :         ok = tevent_wakeup_recv(subreq);
    5211           0 :         TALLOC_FREE(subreq);
    5212           0 :         if (!ok) {
    5213           0 :                 tevent_req_oom(req);
    5214           0 :                 return;
    5215             :         }
    5216           0 :         tevent_req_nterror(req, NT_STATUS_TIMEOUT);
    5217             : }
    5218             : 
    5219           1 : static NTSTATUS oplock5_recv(struct tevent_req *req)
    5220             : {
    5221           1 :         return tevent_req_simple_recv_ntstatus(req);
    5222             : }
    5223             : 
    5224           1 : static bool run_oplock5(int dummy)
    5225             : {
    5226           1 :         struct tevent_context *ev = NULL;
    5227           1 :         struct tevent_req *req = NULL;
    5228           1 :         struct cli_state *cli = NULL;
    5229           1 :         const char *fname = "oplock5.txt";
    5230           0 :         int pipe_down[2], pipe_up[2];
    5231           0 :         pid_t child_pid;
    5232           1 :         uint8_t c = '\0';
    5233           0 :         NTSTATUS status;
    5234           0 :         int ret;
    5235           0 :         bool ok;
    5236             : 
    5237           1 :         printf("starting oplock5\n");
    5238             : 
    5239           1 :         if (local_path == NULL) {
    5240           0 :                 d_fprintf(stderr, "oplock5 must be given a local path via "
    5241             :                           "-l <localpath>\n");
    5242           0 :                 return false;
    5243             :         }
    5244             : 
    5245           1 :         ret = pipe(pipe_down);
    5246           1 :         if (ret == -1) {
    5247           0 :                 d_fprintf(stderr, "pipe() failed: %s\n", strerror(errno));
    5248           0 :                 return false;
    5249             :         }
    5250           1 :         ret = pipe(pipe_up);
    5251           1 :         if (ret == -1) {
    5252           0 :                 d_fprintf(stderr, "pipe() failed: %s\n", strerror(errno));
    5253           0 :                 return false;
    5254             :         }
    5255             : 
    5256           1 :         child_pid = fork();
    5257           2 :         if (child_pid == -1) {
    5258           0 :                 d_fprintf(stderr, "fork() failed: %s\n", strerror(errno));
    5259           0 :                 return false;
    5260             :         }
    5261             : 
    5262           2 :         if (child_pid == 0) {
    5263           1 :                 char *local_file = NULL;
    5264           0 :                 int fd;
    5265             : 
    5266           1 :                 close(pipe_down[1]);
    5267           1 :                 close(pipe_up[0]);
    5268             : 
    5269           1 :                 local_file = talloc_asprintf(
    5270           1 :                         talloc_tos(), "%s/%s", local_path, fname);
    5271           1 :                 if (local_file == 0) {
    5272           0 :                         c = 1;
    5273           0 :                         goto do_write;
    5274             :                 }
    5275           1 :                 fd = open(local_file, O_RDWR|O_CREAT, 0644);
    5276           1 :                 if (fd == -1) {
    5277           0 :                         d_fprintf(stderr,
    5278             :                                   "open(%s) in child failed: %s\n",
    5279             :                                   local_file,
    5280           0 :                                   strerror(errno));
    5281           0 :                         c = 2;
    5282           0 :                         goto do_write;
    5283             :                 }
    5284             : 
    5285           1 :                 signal(SIGIO, SIG_IGN);
    5286             : 
    5287           1 :                 ret = fcntl(fd, F_SETLEASE, F_WRLCK);
    5288           1 :                 if (ret == -1) {
    5289           0 :                         d_fprintf(stderr,
    5290             :                                   "SETLEASE in child failed: %s\n",
    5291           0 :                                   strerror(errno));
    5292           0 :                         c = 3;
    5293           0 :                         goto do_write;
    5294             :                 }
    5295             : 
    5296           1 :         do_write:
    5297           1 :                 ret = sys_write(pipe_up[1], &c, sizeof(c));
    5298           1 :                 if (ret == -1) {
    5299           0 :                         d_fprintf(stderr,
    5300             :                                   "sys_write failed: %s\n",
    5301           0 :                                   strerror(errno));
    5302           0 :                         exit(4);
    5303             :                 }
    5304           1 :                 ret = sys_read(pipe_down[0], &c, sizeof(c));
    5305           1 :                 if (ret == -1) {
    5306           0 :                         d_fprintf(stderr,
    5307             :                                   "sys_read failed: %s\n",
    5308           0 :                                   strerror(errno));
    5309           0 :                         exit(5);
    5310             :                 }
    5311           1 :                 exit(0);
    5312             :         }
    5313             : 
    5314           1 :         close(pipe_up[1]);
    5315           1 :         close(pipe_down[0]);
    5316             : 
    5317           1 :         ret = sys_read(pipe_up[0], &c, sizeof(c));
    5318           1 :         if (ret != 1) {
    5319           0 :                 d_fprintf(stderr,
    5320             :                           "sys_read failed: %s\n",
    5321           0 :                           strerror(errno));
    5322           0 :                 return false;
    5323             :         }
    5324           1 :         if (c != 0) {
    5325           0 :                 d_fprintf(stderr, "got error code %"PRIu8"\n", c);
    5326           0 :                 return false;
    5327             :         }
    5328             : 
    5329           1 :         ok = torture_open_connection(&cli, 0);
    5330           1 :         if (!ok) {
    5331           0 :                 d_fprintf(stderr, "torture_open_connection failed\n");
    5332           0 :                 return false;
    5333             :         }
    5334             : 
    5335           1 :         ev = samba_tevent_context_init(talloc_tos());
    5336           1 :         if (ev == NULL) {
    5337           0 :                 d_fprintf(stderr, "samba_tevent_context_init failed\n");
    5338           0 :                 return false;
    5339             :         }
    5340             : 
    5341           1 :         req = oplock5_send(ev, ev, cli, fname, pipe_down[1]);
    5342           1 :         if (req == NULL) {
    5343           0 :                 d_fprintf(stderr, "oplock5_send failed\n");
    5344           0 :                 return false;
    5345             :         }
    5346             : 
    5347           1 :         ok = tevent_req_poll_ntstatus(req, ev, &status);
    5348           1 :         if (!ok) {
    5349           0 :                 d_fprintf(stderr,
    5350             :                           "tevent_req_poll_ntstatus failed: %s\n",
    5351             :                           nt_errstr(status));
    5352           0 :                 return false;
    5353             :         }
    5354             : 
    5355           1 :         status = oplock5_recv(req);
    5356           1 :         TALLOC_FREE(req);
    5357           1 :         if (!NT_STATUS_IS_OK(status)) {
    5358           0 :                 d_fprintf(stderr,
    5359             :                           "oplock5 failed: %s\n",
    5360             :                           nt_errstr(status));
    5361           0 :                 return false;
    5362             :         }
    5363             : 
    5364           1 :         return true;
    5365             : }
    5366             : 
    5367             : #endif /* HAVE_KERNEL_OPLOCKS_LINUX */
    5368             : 
    5369             : /*
    5370             :   Test delete on close semantics.
    5371             :  */
    5372           5 : static bool run_deletetest(int dummy)
    5373             : {
    5374           5 :         struct cli_state *cli1 = NULL;
    5375           5 :         struct cli_state *cli2 = NULL;
    5376           5 :         const char *fname = "\\delete.file";
    5377           5 :         uint16_t fnum1 = (uint16_t)-1;
    5378           5 :         uint16_t fnum2 = (uint16_t)-1;
    5379           5 :         bool correct = false;
    5380           0 :         NTSTATUS status;
    5381             : 
    5382           5 :         printf("starting delete test\n");
    5383             : 
    5384           5 :         if (!torture_open_connection(&cli1, 0)) {
    5385           0 :                 return False;
    5386             :         }
    5387             : 
    5388           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    5389             : 
    5390             :         /* Test 1 - this should delete the file on close. */
    5391             : 
    5392           5 :         cli_setatr(cli1, fname, 0, 0);
    5393           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5394             : 
    5395           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_ALL_ACCESS|DELETE_ACCESS,
    5396             :                               FILE_ATTRIBUTE_NORMAL, 0, FILE_OVERWRITE_IF,
    5397             :                               FILE_DELETE_ON_CLOSE, 0, &fnum1, NULL);
    5398           5 :         if (!NT_STATUS_IS_OK(status)) {
    5399           0 :                 printf("[1] open of %s failed (%s)\n", fname, nt_errstr(status));
    5400           0 :                 goto fail;
    5401             :         }
    5402             : 
    5403           5 :         status = cli_close(cli1, fnum1);
    5404           5 :         if (!NT_STATUS_IS_OK(status)) {
    5405           0 :                 printf("[1] close failed (%s)\n", nt_errstr(status));
    5406           0 :                 goto fail;
    5407             :         }
    5408             : 
    5409           5 :         status = cli_openx(cli1, fname, O_RDWR, DENY_NONE, &fnum1);
    5410           5 :         if (NT_STATUS_IS_OK(status)) {
    5411           0 :                 printf("[1] open of %s succeeded (should fail)\n", fname);
    5412           0 :                 goto fail;
    5413             :         }
    5414             : 
    5415           5 :         printf("first delete on close test succeeded.\n");
    5416             : 
    5417             :         /* Test 2 - this should delete the file on close. */
    5418             : 
    5419           5 :         cli_setatr(cli1, fname, 0, 0);
    5420           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5421             : 
    5422           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_ALL_ACCESS,
    5423             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_NONE,
    5424             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    5425           5 :         if (!NT_STATUS_IS_OK(status)) {
    5426           0 :                 printf("[2] open of %s failed (%s)\n", fname, nt_errstr(status));
    5427           0 :                 goto fail;
    5428             :         }
    5429             : 
    5430           5 :         status = cli_nt_delete_on_close(cli1, fnum1, true);
    5431           5 :         if (!NT_STATUS_IS_OK(status)) {
    5432           0 :                 printf("[2] setting delete_on_close failed (%s)\n", nt_errstr(status));
    5433           0 :                 goto fail;
    5434             :         }
    5435             : 
    5436           5 :         status = cli_close(cli1, fnum1);
    5437           5 :         if (!NT_STATUS_IS_OK(status)) {
    5438           0 :                 printf("[2] close failed (%s)\n", nt_errstr(status));
    5439           0 :                 goto fail;
    5440             :         }
    5441             : 
    5442           5 :         status = cli_openx(cli1, fname, O_RDONLY, DENY_NONE, &fnum1);
    5443           5 :         if (NT_STATUS_IS_OK(status)) {
    5444           0 :                 printf("[2] open of %s succeeded should have been deleted on close !\n", fname);
    5445           0 :                 status = cli_close(cli1, fnum1);
    5446           0 :                 if (!NT_STATUS_IS_OK(status)) {
    5447           0 :                         printf("[2] close failed (%s)\n", nt_errstr(status));
    5448             :                 }
    5449           0 :                 cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5450           0 :                 goto fail;
    5451             :         }
    5452             : 
    5453           5 :         printf("second delete on close test succeeded.\n");
    5454             : 
    5455             :         /* Test 3 - ... */
    5456           5 :         cli_setatr(cli1, fname, 0, 0);
    5457           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5458             : 
    5459           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_ALL_ACCESS,
    5460             :                               FILE_ATTRIBUTE_NORMAL,
    5461             :                               FILE_SHARE_READ|FILE_SHARE_WRITE,
    5462             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    5463           5 :         if (!NT_STATUS_IS_OK(status)) {
    5464           0 :                 printf("[3] open - 1 of %s failed (%s)\n", fname, nt_errstr(status));
    5465           0 :                 goto fail;
    5466             :         }
    5467             : 
    5468             :         /* This should fail with a sharing violation - open for delete is only compatible
    5469             :            with SHARE_DELETE. */
    5470             : 
    5471           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_READ_ACCESS,
    5472             :                               FILE_ATTRIBUTE_NORMAL,
    5473             :                               FILE_SHARE_READ|FILE_SHARE_WRITE,
    5474             :                               FILE_OPEN, 0, 0, &fnum2, NULL);
    5475           5 :         if (NT_STATUS_IS_OK(status)) {
    5476           0 :                 printf("[3] open  - 2 of %s succeeded - should have failed.\n", fname);
    5477           0 :                 goto fail;
    5478             :         }
    5479             : 
    5480             :         /* This should succeed. */
    5481           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_READ_ACCESS,
    5482             :                              FILE_ATTRIBUTE_NORMAL,
    5483             :                              FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    5484             :                              FILE_OPEN, 0, 0, &fnum2, NULL);
    5485           5 :         if (!NT_STATUS_IS_OK(status)) {
    5486           0 :                 printf("[3] open  - 3 of %s failed (%s)\n", fname, nt_errstr(status));
    5487           0 :                 goto fail;
    5488             :         }
    5489             : 
    5490           5 :         status = cli_nt_delete_on_close(cli1, fnum1, true);
    5491           5 :         if (!NT_STATUS_IS_OK(status)) {
    5492           0 :                 printf("[3] setting delete_on_close failed (%s)\n", nt_errstr(status));
    5493           0 :                 goto fail;
    5494             :         }
    5495             : 
    5496           5 :         status = cli_close(cli1, fnum1);
    5497           5 :         if (!NT_STATUS_IS_OK(status)) {
    5498           0 :                 printf("[3] close 1 failed (%s)\n", nt_errstr(status));
    5499           0 :                 goto fail;
    5500             :         }
    5501             : 
    5502           5 :         status = cli_close(cli1, fnum2);
    5503           5 :         if (!NT_STATUS_IS_OK(status)) {
    5504           0 :                 printf("[3] close 2 failed (%s)\n", nt_errstr(status));
    5505           0 :                 goto fail;
    5506             :         }
    5507             : 
    5508             :         /* This should fail - file should no longer be there. */
    5509             : 
    5510           5 :         status = cli_openx(cli1, fname, O_RDONLY, DENY_NONE, &fnum1);
    5511           5 :         if (NT_STATUS_IS_OK(status)) {
    5512           0 :                 printf("[3] open of %s succeeded should have been deleted on close !\n", fname);
    5513           0 :                 status = cli_close(cli1, fnum1);
    5514           0 :                 if (!NT_STATUS_IS_OK(status)) {
    5515           0 :                         printf("[3] close failed (%s)\n", nt_errstr(status));
    5516             :                 }
    5517           0 :                 cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5518           0 :                 goto fail;
    5519             :         }
    5520             : 
    5521           5 :         printf("third delete on close test succeeded.\n");
    5522             : 
    5523             :         /* Test 4 ... */
    5524           5 :         cli_setatr(cli1, fname, 0, 0);
    5525           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5526             : 
    5527           5 :         status = cli_ntcreate(cli1, fname, 0,
    5528             :                               FILE_READ_DATA|FILE_WRITE_DATA|DELETE_ACCESS,
    5529             :                               FILE_ATTRIBUTE_NORMAL,
    5530             :                               FILE_SHARE_READ|FILE_SHARE_WRITE,
    5531             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    5532           5 :         if (!NT_STATUS_IS_OK(status)) {
    5533           0 :                 printf("[4] open of %s failed (%s)\n", fname, nt_errstr(status));
    5534           0 :                 goto fail;
    5535             :         }
    5536             : 
    5537             :         /* This should succeed. */
    5538           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_READ_ACCESS,
    5539             :                              FILE_ATTRIBUTE_NORMAL,
    5540             :                              FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    5541             :                              FILE_OPEN, 0, 0, &fnum2, NULL);
    5542           5 :         if (!NT_STATUS_IS_OK(status)) {
    5543           0 :                 printf("[4] open  - 2 of %s failed (%s)\n", fname, nt_errstr(status));
    5544           0 :                 goto fail;
    5545             :         }
    5546             : 
    5547           5 :         status = cli_close(cli1, fnum2);
    5548           5 :         if (!NT_STATUS_IS_OK(status)) {
    5549           0 :                 printf("[4] close - 1 failed (%s)\n", nt_errstr(status));
    5550           0 :                 goto fail;
    5551             :         }
    5552             : 
    5553           5 :         status = cli_nt_delete_on_close(cli1, fnum1, true);
    5554           5 :         if (!NT_STATUS_IS_OK(status)) {
    5555           0 :                 printf("[4] setting delete_on_close failed (%s)\n", nt_errstr(status));
    5556           0 :                 goto fail;
    5557             :         }
    5558             : 
    5559             :         /* This should fail - no more opens once delete on close set. */
    5560           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_READ_ACCESS,
    5561             :                               FILE_ATTRIBUTE_NORMAL,
    5562             :                               FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    5563             :                               FILE_OPEN, 0, 0, &fnum2, NULL);
    5564           5 :         if (NT_STATUS_IS_OK(status)) {
    5565           0 :                 printf("[4] open  - 3 of %s succeeded ! Should have failed.\n", fname );
    5566           0 :                 goto fail;
    5567             :         }
    5568             : 
    5569           5 :         status = cli_close(cli1, fnum1);
    5570           5 :         if (!NT_STATUS_IS_OK(status)) {
    5571           0 :                 printf("[4] close - 2 failed (%s)\n", nt_errstr(status));
    5572           0 :                 goto fail;
    5573             :         }
    5574             : 
    5575           5 :         printf("fourth delete on close test succeeded.\n");
    5576             : 
    5577             :         /* Test 5 ... */
    5578           5 :         cli_setatr(cli1, fname, 0, 0);
    5579           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5580             : 
    5581           5 :         status = cli_openx(cli1, fname, O_RDWR|O_CREAT, DENY_NONE, &fnum1);
    5582           5 :         if (!NT_STATUS_IS_OK(status)) {
    5583           0 :                 printf("[5] open of %s failed (%s)\n", fname, nt_errstr(status));
    5584           0 :                 goto fail;
    5585             :         }
    5586             : 
    5587             :         /* This should fail - only allowed on NT opens with DELETE access. */
    5588             : 
    5589           5 :         status = cli_nt_delete_on_close(cli1, fnum1, true);
    5590           5 :         if (NT_STATUS_IS_OK(status)) {
    5591           0 :                 printf("[5] setting delete_on_close on OpenX file succeeded - should fail !\n");
    5592           0 :                 goto fail;
    5593             :         }
    5594             : 
    5595           5 :         status = cli_close(cli1, fnum1);
    5596           5 :         if (!NT_STATUS_IS_OK(status)) {
    5597           0 :                 printf("[5] close failed (%s)\n", nt_errstr(status));
    5598           0 :                 goto fail;
    5599             :         }
    5600             : 
    5601           5 :         printf("fifth delete on close test succeeded.\n");
    5602             : 
    5603             :         /* Test 6 ... */
    5604           5 :         cli_setatr(cli1, fname, 0, 0);
    5605           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5606             : 
    5607           5 :         status = cli_ntcreate(cli1, fname, 0, FILE_READ_DATA|FILE_WRITE_DATA,
    5608             :                              FILE_ATTRIBUTE_NORMAL,
    5609             :                              FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    5610             :                              FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    5611           5 :         if (!NT_STATUS_IS_OK(status)) {
    5612           0 :                 printf("[6] open of %s failed (%s)\n", fname,
    5613             :                        nt_errstr(status));
    5614           0 :                 goto fail;
    5615             :         }
    5616             : 
    5617             :         /* This should fail - only allowed on NT opens with DELETE access. */
    5618             : 
    5619           5 :         status = cli_nt_delete_on_close(cli1, fnum1, true);
    5620           5 :         if (NT_STATUS_IS_OK(status)) {
    5621           0 :                 printf("[6] setting delete_on_close on file with no delete access succeeded - should fail !\n");
    5622           0 :                 goto fail;
    5623             :         }
    5624             : 
    5625           5 :         status = cli_close(cli1, fnum1);
    5626           5 :         if (!NT_STATUS_IS_OK(status)) {
    5627           0 :                 printf("[6] close failed (%s)\n", nt_errstr(status));
    5628           0 :                 goto fail;
    5629             :         }
    5630             : 
    5631           5 :         printf("sixth delete on close test succeeded.\n");
    5632             : 
    5633             :         /* Test 7 ... */
    5634           5 :         cli_setatr(cli1, fname, 0, 0);
    5635           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5636             : 
    5637           5 :         status = cli_ntcreate(cli1, fname, 0,
    5638             :                               FILE_READ_DATA|FILE_WRITE_DATA|DELETE_ACCESS,
    5639             :                               FILE_ATTRIBUTE_NORMAL, 0, FILE_OVERWRITE_IF,
    5640             :                               0, 0, &fnum1, NULL);
    5641           5 :         if (!NT_STATUS_IS_OK(status)) {
    5642           0 :                 printf("[7] open of %s failed (%s)\n", fname, nt_errstr(status));
    5643           0 :                 goto fail;
    5644             :         }
    5645             : 
    5646           5 :         status = cli_nt_delete_on_close(cli1, fnum1, true);
    5647           5 :         if (!NT_STATUS_IS_OK(status)) {
    5648           0 :                 printf("[7] setting delete_on_close on file failed !\n");
    5649           0 :                 goto fail;
    5650             :         }
    5651             : 
    5652           5 :         status = cli_nt_delete_on_close(cli1, fnum1, false);
    5653           5 :         if (!NT_STATUS_IS_OK(status)) {
    5654           0 :                 printf("[7] unsetting delete_on_close on file failed !\n");
    5655           0 :                 goto fail;
    5656             :         }
    5657             : 
    5658           5 :         status = cli_close(cli1, fnum1);
    5659           5 :         if (!NT_STATUS_IS_OK(status)) {
    5660           0 :                 printf("[7] close - 1 failed (%s)\n", nt_errstr(status));
    5661           0 :                 goto fail;
    5662             :         }
    5663             : 
    5664             :         /* This next open should succeed - we reset the flag. */
    5665           5 :         status = cli_openx(cli1, fname, O_RDONLY, DENY_NONE, &fnum1);
    5666           5 :         if (!NT_STATUS_IS_OK(status)) {
    5667           0 :                 printf("[7] open of %s failed (%s)\n", fname, nt_errstr(status));
    5668           0 :                 goto fail;
    5669             :         }
    5670             : 
    5671           5 :         status = cli_close(cli1, fnum1);
    5672           5 :         if (!NT_STATUS_IS_OK(status)) {
    5673           0 :                 printf("[7] close - 2 failed (%s)\n", nt_errstr(status));
    5674           0 :                 goto fail;
    5675             :         }
    5676             : 
    5677           5 :         printf("seventh delete on close test succeeded.\n");
    5678             : 
    5679             :         /* Test 8 ... */
    5680           5 :         cli_setatr(cli1, fname, 0, 0);
    5681           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5682             : 
    5683           5 :         if (!torture_open_connection(&cli2, 1)) {
    5684           0 :                 printf("[8] failed to open second connection.\n");
    5685           0 :                 goto fail;
    5686             :         }
    5687             : 
    5688           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    5689             : 
    5690           5 :         status = cli_ntcreate(cli1, fname, 0,
    5691             :                              FILE_READ_DATA|FILE_WRITE_DATA|DELETE_ACCESS,
    5692             :                              FILE_ATTRIBUTE_NORMAL,
    5693             :                              FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    5694             :                              FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    5695           5 :         if (!NT_STATUS_IS_OK(status)) {
    5696           0 :                 printf("[8] open 1 of %s failed (%s)\n", fname, nt_errstr(status));
    5697           0 :                 goto fail;
    5698             :         }
    5699             : 
    5700           5 :         status = cli_ntcreate(cli2, fname, 0,
    5701             :                              FILE_READ_DATA|FILE_WRITE_DATA|DELETE_ACCESS,
    5702             :                              FILE_ATTRIBUTE_NORMAL,
    5703             :                              FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    5704             :                              FILE_OPEN, 0, 0, &fnum2, NULL);
    5705           5 :         if (!NT_STATUS_IS_OK(status)) {
    5706           0 :                 printf("[8] open 2 of %s failed (%s)\n", fname, nt_errstr(status));
    5707           0 :                 goto fail;
    5708             :         }
    5709             : 
    5710           5 :         status = cli_nt_delete_on_close(cli1, fnum1, true);
    5711           5 :         if (!NT_STATUS_IS_OK(status)) {
    5712           0 :                 printf("[8] setting delete_on_close on file failed !\n");
    5713           0 :                 goto fail;
    5714             :         }
    5715             : 
    5716           5 :         status = cli_close(cli1, fnum1);
    5717           5 :         if (!NT_STATUS_IS_OK(status)) {
    5718           0 :                 printf("[8] close - 1 failed (%s)\n", nt_errstr(status));
    5719           0 :                 goto fail;
    5720             :         }
    5721             : 
    5722           5 :         status = cli_close(cli2, fnum2);
    5723           5 :         if (!NT_STATUS_IS_OK(status)) {
    5724           0 :                 printf("[8] close - 2 failed (%s)\n", nt_errstr(status));
    5725           0 :                 goto fail;
    5726             :         }
    5727             : 
    5728             :         /* This should fail.. */
    5729           5 :         status = cli_openx(cli1, fname, O_RDONLY, DENY_NONE, &fnum1);
    5730           5 :         if (NT_STATUS_IS_OK(status)) {
    5731           0 :                 printf("[8] open of %s succeeded should have been deleted on close !\n", fname);
    5732           0 :                 goto fail;
    5733             :         }
    5734             : 
    5735           5 :         printf("eighth delete on close test succeeded.\n");
    5736             : 
    5737             :         /* Test 9 ... */
    5738             : 
    5739             :         /* This should fail - we need to set DELETE_ACCESS. */
    5740           5 :         status = cli_ntcreate(cli1, fname, 0, FILE_READ_DATA|FILE_WRITE_DATA,
    5741             :                               FILE_ATTRIBUTE_NORMAL,
    5742             :                               FILE_SHARE_NONE,
    5743             :                               FILE_OVERWRITE_IF,
    5744             :                               FILE_DELETE_ON_CLOSE, 0, &fnum1, NULL);
    5745           5 :         if (NT_STATUS_IS_OK(status)) {
    5746           0 :                 printf("[9] open of %s succeeded should have failed!\n", fname);
    5747           0 :                 goto fail;
    5748             :         }
    5749             : 
    5750           5 :         printf("ninth delete on close test succeeded.\n");
    5751             : 
    5752             :         /* Test 10 ... */
    5753             : 
    5754           5 :         status = cli_ntcreate(cli1, fname, 0,
    5755             :                              FILE_READ_DATA|FILE_WRITE_DATA|DELETE_ACCESS,
    5756             :                              FILE_ATTRIBUTE_NORMAL, FILE_SHARE_NONE,
    5757             :                              FILE_OVERWRITE_IF, FILE_DELETE_ON_CLOSE,
    5758             :                              0, &fnum1, NULL);
    5759           5 :         if (!NT_STATUS_IS_OK(status)) {
    5760           0 :                 printf("[10] open of %s failed (%s)\n", fname, nt_errstr(status));
    5761           0 :                 goto fail;
    5762             :         }
    5763             : 
    5764             :         /* This should delete the file. */
    5765           5 :         status = cli_close(cli1, fnum1);
    5766           5 :         if (!NT_STATUS_IS_OK(status)) {
    5767           0 :                 printf("[10] close failed (%s)\n", nt_errstr(status));
    5768           0 :                 goto fail;
    5769             :         }
    5770             : 
    5771             :         /* This should fail.. */
    5772           5 :         status = cli_openx(cli1, fname, O_RDONLY, DENY_NONE, &fnum1);
    5773           5 :         if (NT_STATUS_IS_OK(status)) {
    5774           0 :                 printf("[10] open of %s succeeded should have been deleted on close !\n", fname);
    5775           0 :                 goto fail;
    5776             :         }
    5777             : 
    5778           5 :         printf("tenth delete on close test succeeded.\n");
    5779             : 
    5780             :         /* Test 11 ... */
    5781             : 
    5782           5 :         cli_setatr(cli1, fname, 0, 0);
    5783           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5784             : 
    5785             :         /* Can we open a read-only file with delete access? */
    5786             : 
    5787             :         /* Create a readonly file. */
    5788           5 :         status = cli_ntcreate(cli1, fname, 0, FILE_READ_DATA|FILE_WRITE_DATA,
    5789             :                               FILE_ATTRIBUTE_READONLY, FILE_SHARE_NONE,
    5790             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    5791           5 :         if (!NT_STATUS_IS_OK(status)) {
    5792           0 :                 printf("[11] open of %s failed (%s)\n", fname, nt_errstr(status));
    5793           0 :                 goto fail;
    5794             :         }
    5795             : 
    5796           5 :         status = cli_close(cli1, fnum1);
    5797           5 :         if (!NT_STATUS_IS_OK(status)) {
    5798           0 :                 printf("[11] close failed (%s)\n", nt_errstr(status));
    5799           0 :                 goto fail;
    5800             :         }
    5801             : 
    5802             :         /* Now try open for delete access. */
    5803           5 :         status = cli_ntcreate(cli1, fname, 0,
    5804             :                              FILE_READ_ATTRIBUTES|DELETE_ACCESS,
    5805             :                              0,
    5806             :                              FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    5807             :                              FILE_OPEN, 0, 0, &fnum1, NULL);
    5808           5 :         if (!NT_STATUS_IS_OK(status)) {
    5809           0 :                 printf("[11] open of %s failed: %s\n", fname, nt_errstr(status));
    5810           0 :                 goto fail;
    5811             :         }
    5812             : 
    5813           5 :         cli_close(cli1, fnum1);
    5814             : 
    5815           5 :         printf("eleventh delete on close test succeeded.\n");
    5816             : 
    5817             :         /*
    5818             :          * Test 12
    5819             :          * like test 4 but with initial delete on close
    5820             :          */
    5821             : 
    5822           5 :         cli_setatr(cli1, fname, 0, 0);
    5823           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5824             : 
    5825           5 :         status = cli_ntcreate(cli1, fname, 0,
    5826             :                               FILE_READ_DATA|FILE_WRITE_DATA|DELETE_ACCESS,
    5827             :                               FILE_ATTRIBUTE_NORMAL,
    5828             :                               FILE_SHARE_READ|FILE_SHARE_WRITE,
    5829             :                               FILE_OVERWRITE_IF,
    5830             :                               FILE_DELETE_ON_CLOSE, 0, &fnum1, NULL);
    5831           5 :         if (!NT_STATUS_IS_OK(status)) {
    5832           0 :                 printf("[12] open 1 of %s failed (%s)\n", fname, nt_errstr(status));
    5833           0 :                 goto fail;
    5834             :         }
    5835             : 
    5836           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_READ_ACCESS,
    5837             :                               FILE_ATTRIBUTE_NORMAL,
    5838             :                               FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    5839             :                               FILE_OPEN, 0, 0, &fnum2, NULL);
    5840           5 :         if (!NT_STATUS_IS_OK(status)) {
    5841           0 :                 printf("[12] open 2 of %s failed(%s).\n", fname, nt_errstr(status));
    5842           0 :                 goto fail;
    5843             :         }
    5844             : 
    5845           5 :         status = cli_close(cli1, fnum2);
    5846           5 :         if (!NT_STATUS_IS_OK(status)) {
    5847           0 :                 printf("[12] close 1 failed (%s)\n", nt_errstr(status));
    5848           0 :                 goto fail;
    5849             :         }
    5850             : 
    5851           5 :         status = cli_nt_delete_on_close(cli1, fnum1, true);
    5852           5 :         if (!NT_STATUS_IS_OK(status)) {
    5853           0 :                 printf("[12] setting delete_on_close failed (%s)\n", nt_errstr(status));
    5854           0 :                 goto fail;
    5855             :         }
    5856             : 
    5857             :         /* This should fail - no more opens once delete on close set. */
    5858           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_READ_ACCESS,
    5859             :                               FILE_ATTRIBUTE_NORMAL,
    5860             :                               FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    5861             :                               FILE_OPEN, 0, 0, &fnum2, NULL);
    5862           5 :         if (NT_STATUS_IS_OK(status)) {
    5863           0 :                 printf("[12] open 3 of %s succeeded - should fail).\n", fname);
    5864           0 :                 goto fail;
    5865             :         }
    5866             : 
    5867           5 :         status = cli_nt_delete_on_close(cli1, fnum1, false);
    5868           5 :         if (!NT_STATUS_IS_OK(status)) {
    5869           0 :                 printf("[12] unsetting delete_on_close failed (%s)\n", nt_errstr(status));
    5870           0 :                 goto fail;
    5871             :         }
    5872             : 
    5873           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_READ_ACCESS,
    5874             :                               FILE_ATTRIBUTE_NORMAL,
    5875             :                               FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    5876             :                               FILE_OPEN, 0, 0, &fnum2, NULL);
    5877           5 :         if (!NT_STATUS_IS_OK(status)) {
    5878           0 :                 printf("[12] open 4 of %s failed (%s)\n", fname, nt_errstr(status));
    5879           0 :                 goto fail;
    5880             :         }
    5881             : 
    5882           5 :         status = cli_close(cli1, fnum2);
    5883           5 :         if (!NT_STATUS_IS_OK(status)) {
    5884           0 :                 printf("[12] close 2 failed (%s)\n", nt_errstr(status));
    5885           0 :                 goto fail;
    5886             :         }
    5887             : 
    5888           5 :         status = cli_close(cli1, fnum1);
    5889           5 :         if (!NT_STATUS_IS_OK(status)) {
    5890           0 :                 printf("[12] close 3 failed (%s)\n", nt_errstr(status));
    5891           0 :                 goto fail;
    5892             :         }
    5893             : 
    5894             :         /*
    5895             :          * setting delete on close on the handle does
    5896             :          * not unset the initial delete on close...
    5897             :          */
    5898           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_READ_ACCESS,
    5899             :                               FILE_ATTRIBUTE_NORMAL,
    5900             :                               FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    5901             :                               FILE_OPEN, 0, 0, &fnum2, NULL);
    5902           5 :         if (NT_STATUS_IS_OK(status)) {
    5903           0 :                 printf("[12] open 5 of %s succeeded - should fail).\n", fname);
    5904           0 :                 goto fail;
    5905           5 :         } else if (!NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_NAME_NOT_FOUND)) {
    5906           0 :                 printf("ntcreate returned %s, expected "
    5907             :                        "NT_STATUS_OBJECT_NAME_NOT_FOUND\n",
    5908             :                        nt_errstr(status));
    5909           0 :                 goto fail;
    5910             :         }
    5911             : 
    5912           5 :         printf("twelfth delete on close test succeeded.\n");
    5913             : 
    5914             : 
    5915           5 :         printf("finished delete test\n");
    5916             : 
    5917           5 :         correct = true;
    5918             : 
    5919           5 :   fail:
    5920             :         /* FIXME: This will crash if we aborted before cli2 got
    5921             :          * initialized, because these functions don't handle
    5922             :          * uninitialized connections. */
    5923             : 
    5924           5 :         if (fnum1 != (uint16_t)-1) cli_close(cli1, fnum1);
    5925           5 :         if (fnum2 != (uint16_t)-1) cli_close(cli1, fnum2);
    5926           5 :         cli_setatr(cli1, fname, 0, 0);
    5927           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5928             : 
    5929           5 :         if (cli1 && !torture_close_connection(cli1)) {
    5930           0 :                 correct = False;
    5931             :         }
    5932           5 :         if (cli2 && !torture_close_connection(cli2)) {
    5933           0 :                 correct = False;
    5934             :         }
    5935           5 :         return correct;
    5936             : }
    5937             : 
    5938             : struct delete_stream_state {
    5939             :         bool closed;
    5940             : };
    5941             : 
    5942             : static void delete_stream_unlinked(struct tevent_req *subreq);
    5943             : static void delete_stream_closed(struct tevent_req *subreq);
    5944             : 
    5945           5 : static struct tevent_req *delete_stream_send(
    5946             :         TALLOC_CTX *mem_ctx,
    5947             :         struct tevent_context *ev,
    5948             :         struct cli_state *cli,
    5949             :         const char *base_fname,
    5950             :         uint16_t stream_fnum)
    5951             : {
    5952           5 :         struct tevent_req *req = NULL, *subreq = NULL;
    5953           5 :         struct delete_stream_state *state = NULL;
    5954             : 
    5955           5 :         req = tevent_req_create(
    5956             :                 mem_ctx, &state, struct delete_stream_state);
    5957           5 :         if (req == NULL) {
    5958           0 :                 return NULL;
    5959             :         }
    5960             : 
    5961           5 :         subreq = cli_unlink_send(
    5962             :                 state,
    5963             :                 ev,
    5964             :                 cli,
    5965             :                 base_fname,
    5966             :                 FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    5967           5 :         if (tevent_req_nomem(subreq, req)) {
    5968           0 :                 return tevent_req_post(req, ev);
    5969             :         }
    5970           5 :         tevent_req_set_callback(subreq, delete_stream_unlinked, req);
    5971             : 
    5972           5 :         subreq = cli_close_send(state, ev, cli, stream_fnum, 0);
    5973           5 :         if (tevent_req_nomem(subreq, req)) {
    5974           0 :                 return tevent_req_post(req, ev);
    5975             :         }
    5976           5 :         tevent_req_set_callback(subreq, delete_stream_closed, req);
    5977             : 
    5978           5 :         return req;
    5979             : }
    5980             : 
    5981           5 : static void delete_stream_unlinked(struct tevent_req *subreq)
    5982             : {
    5983           5 :         struct tevent_req *req = tevent_req_callback_data(
    5984             :                 subreq, struct tevent_req);
    5985           5 :         struct delete_stream_state *state = tevent_req_data(
    5986             :                 req, struct delete_stream_state);
    5987           0 :         NTSTATUS status;
    5988             : 
    5989           5 :         status = cli_unlink_recv(subreq);
    5990           5 :         TALLOC_FREE(subreq);
    5991           5 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_SHARING_VIOLATION)) {
    5992           0 :                 printf("cli_unlink returned %s\n",
    5993             :                        nt_errstr(status));
    5994           0 :                 tevent_req_nterror(req, NT_STATUS_UNSUCCESSFUL);
    5995           0 :                 return;
    5996             :         }
    5997           5 :         if (!state->closed) {
    5998             :                 /* close reply should have come in first */
    5999           0 :                 printf("Not closed\n");
    6000           0 :                 tevent_req_nterror(req, NT_STATUS_UNSUCCESSFUL);
    6001           0 :                 return;
    6002             :         }
    6003           5 :         tevent_req_done(req);
    6004             : }
    6005             : 
    6006           5 : static void delete_stream_closed(struct tevent_req *subreq)
    6007             : {
    6008           5 :         struct tevent_req *req = tevent_req_callback_data(
    6009             :                 subreq, struct tevent_req);
    6010           5 :         struct delete_stream_state *state = tevent_req_data(
    6011             :                 req, struct delete_stream_state);
    6012           0 :         NTSTATUS status;
    6013             : 
    6014           5 :         status = cli_close_recv(subreq);
    6015           5 :         TALLOC_FREE(subreq);
    6016           5 :         if (tevent_req_nterror(req, status)) {
    6017           0 :                 return;
    6018             :         }
    6019             :         /* also waiting for the unlink to come back */
    6020           5 :         state->closed = true;
    6021             : }
    6022             : 
    6023           5 : static NTSTATUS delete_stream_recv(struct tevent_req *req)
    6024             : {
    6025           5 :         return tevent_req_simple_recv_ntstatus(req);
    6026             : }
    6027             : 
    6028           5 : static bool run_delete_stream(int dummy)
    6029             : {
    6030           5 :         struct tevent_context *ev = NULL;
    6031           5 :         struct tevent_req *req = NULL;
    6032           5 :         struct cli_state *cli = NULL;
    6033           5 :         const char fname[] = "delete_stream";
    6034           5 :         const char fname_stream[] = "delete_stream:Zone.Identifier:$DATA";
    6035           0 :         uint16_t fnum1, fnum2;
    6036           0 :         NTSTATUS status;
    6037           0 :         bool ok;
    6038             : 
    6039           5 :         printf("Starting stream delete test\n");
    6040             : 
    6041           5 :         ok = torture_open_connection(&cli, 0);
    6042           5 :         if (!ok) {
    6043           0 :                 return false;
    6044             :         }
    6045             : 
    6046           5 :         cli_setatr(cli, fname, 0, 0);
    6047           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6048             : 
    6049             :         /* Create the file. */
    6050           5 :         status = cli_ntcreate(
    6051             :                 cli,
    6052             :                 fname,
    6053             :                 0,
    6054             :                 READ_CONTROL_ACCESS,
    6055             :                 0,
    6056             :                 FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    6057             :                 FILE_CREATE,
    6058             :                 0x0,
    6059             :                 0x0,
    6060             :                 &fnum1,
    6061             :                 NULL);
    6062           5 :         if (!NT_STATUS_IS_OK(status)) {
    6063           0 :                 d_fprintf(stderr,
    6064             :                           "cli_ntcreate of %s failed (%s)\n",
    6065             :                           fname,
    6066             :                           nt_errstr(status));
    6067           0 :                 return false;
    6068             :         }
    6069           5 :         status = cli_close(cli, fnum1);
    6070           5 :         if (!NT_STATUS_IS_OK(status)) {
    6071           0 :                 d_fprintf(stderr,
    6072             :                           "cli_close of %s failed (%s)\n",
    6073             :                           fname,
    6074             :                           nt_errstr(status));
    6075           0 :                 return false;
    6076             :         }
    6077             : 
    6078             :         /* Now create the stream. */
    6079           5 :         status = cli_ntcreate(
    6080             :                 cli,
    6081             :                 fname_stream,
    6082             :                 0,
    6083             :                 FILE_WRITE_DATA,
    6084             :                 0,
    6085             :                 FILE_SHARE_READ|FILE_SHARE_WRITE,
    6086             :                 FILE_CREATE,
    6087             :                 0x0,
    6088             :                 0x0,
    6089             :                 &fnum1,
    6090             :                 NULL);
    6091             : 
    6092           5 :         if (!NT_STATUS_IS_OK(status)) {
    6093           0 :                 d_fprintf(stderr,
    6094             :                           "cli_ntcreate of %s failed (%s)\n",
    6095             :                           fname_stream,
    6096             :                           nt_errstr(status));
    6097           0 :                 return false;
    6098             :         }
    6099             : 
    6100             :         /* open it a second time */
    6101             : 
    6102           5 :         status = cli_ntcreate(
    6103             :                 cli,
    6104             :                 fname_stream,
    6105             :                 0,
    6106             :                 FILE_WRITE_DATA,
    6107             :                 0,
    6108             :                 FILE_SHARE_READ|FILE_SHARE_WRITE,
    6109             :                 FILE_OPEN,
    6110             :                 0x0,
    6111             :                 0x0,
    6112             :                 &fnum2,
    6113             :                 NULL);
    6114             : 
    6115           5 :         if (!NT_STATUS_IS_OK(status)) {
    6116           0 :                 d_fprintf(stderr,
    6117             :                           "2nd cli_ntcreate of %s failed (%s)\n",
    6118             :                           fname_stream,
    6119             :                           nt_errstr(status));
    6120           0 :                 return false;
    6121             :         }
    6122             : 
    6123           5 :         ev = samba_tevent_context_init(talloc_tos());
    6124           5 :         if (ev == NULL) {
    6125           0 :                 d_fprintf(stderr, "samba_tevent_context_init failed\n");
    6126           0 :                 return false;
    6127             :         }
    6128             : 
    6129           5 :         req = delete_stream_send(ev, ev, cli, fname, fnum1);
    6130           5 :         if (req == NULL) {
    6131           0 :                 d_fprintf(stderr, "delete_stream_send failed\n");
    6132           0 :                 return false;
    6133             :         }
    6134             : 
    6135           5 :         ok = tevent_req_poll_ntstatus(req, ev, &status);
    6136           5 :         if (!ok) {
    6137           0 :                 d_fprintf(stderr,
    6138             :                           "tevent_req_poll_ntstatus failed: %s\n",
    6139             :                           nt_errstr(status));
    6140           0 :                 return false;
    6141             :         }
    6142             : 
    6143           5 :         status = delete_stream_recv(req);
    6144           5 :         TALLOC_FREE(req);
    6145           5 :         if (!NT_STATUS_IS_OK(status)) {
    6146           0 :                 d_fprintf(stderr,
    6147             :                           "delete_stream failed: %s\n",
    6148             :                           nt_errstr(status));
    6149           0 :                 return false;
    6150             :         }
    6151             : 
    6152           5 :         status = cli_close(cli, fnum2);
    6153           5 :         if (!NT_STATUS_IS_OK(status)) {
    6154           0 :                 d_fprintf(stderr,
    6155             :                           "close failed: %s\n",
    6156             :                           nt_errstr(status));
    6157           0 :                 return false;
    6158             :         }
    6159             : 
    6160           5 :         status = cli_unlink(
    6161             :                 cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6162           5 :         if (!NT_STATUS_IS_OK(status)) {
    6163           0 :                 d_fprintf(stderr,
    6164             :                           "unlink failed: %s\n",
    6165             :                           nt_errstr(status));
    6166           0 :                 return false;
    6167             :         }
    6168             : 
    6169           5 :         return true;
    6170             : }
    6171             : 
    6172             : /*
    6173             :   Exercise delete on close semantics - use on the PRINT1 share in torture
    6174             :   testing.
    6175             :  */
    6176           2 : static bool run_delete_print_test(int dummy)
    6177             : {
    6178           2 :         struct cli_state *cli1 = NULL;
    6179           2 :         const char *fname = "print_delete.file";
    6180           2 :         uint16_t fnum1 = (uint16_t)-1;
    6181           2 :         bool correct = false;
    6182           2 :         const char *buf = "print file data\n";
    6183           0 :         NTSTATUS status;
    6184             : 
    6185           2 :         printf("starting print delete test\n");
    6186             : 
    6187           2 :         if (!torture_open_connection(&cli1, 0)) {
    6188           0 :                 return false;
    6189             :         }
    6190             : 
    6191           2 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    6192             : 
    6193           2 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_ALL_ACCESS|DELETE_ACCESS,
    6194             :                               FILE_ATTRIBUTE_NORMAL, 0, FILE_OVERWRITE_IF,
    6195             :                               0, 0, &fnum1, NULL);
    6196           2 :         if (!NT_STATUS_IS_OK(status)) {
    6197           0 :                 printf("open of %s failed (%s)\n",
    6198             :                         fname,
    6199             :                         nt_errstr(status));
    6200           0 :                 goto fail;
    6201             :         }
    6202             : 
    6203           2 :         status = cli_writeall(cli1,
    6204             :                         fnum1,
    6205             :                         0,
    6206             :                         (const uint8_t *)buf,
    6207             :                         0, /* offset */
    6208             :                         strlen(buf), /* size */
    6209             :                         NULL);
    6210           2 :         if (!NT_STATUS_IS_OK(status)) {
    6211           0 :                 printf("writing print file data failed (%s)\n",
    6212             :                         nt_errstr(status));
    6213           0 :                 goto fail;
    6214             :         }
    6215             : 
    6216           2 :         status = cli_nt_delete_on_close(cli1, fnum1, true);
    6217           2 :         if (!NT_STATUS_IS_OK(status)) {
    6218           0 :                 printf("setting delete_on_close failed (%s)\n",
    6219             :                         nt_errstr(status));
    6220           0 :                 goto fail;
    6221             :         }
    6222             : 
    6223           2 :         status = cli_close(cli1, fnum1);
    6224           2 :         if (!NT_STATUS_IS_OK(status)) {
    6225           0 :                 printf("close failed (%s)\n", nt_errstr(status));
    6226           0 :                 goto fail;
    6227             :         }
    6228             : 
    6229           2 :         printf("finished print delete test\n");
    6230             : 
    6231           2 :         correct = true;
    6232             : 
    6233           2 :   fail:
    6234             : 
    6235           2 :         if (fnum1 != (uint16_t)-1) {
    6236           2 :                 cli_close(cli1, fnum1);
    6237             :         }
    6238             : 
    6239           2 :         if (cli1 && !torture_close_connection(cli1)) {
    6240           0 :                 correct = false;
    6241             :         }
    6242           2 :         return correct;
    6243             : }
    6244             : 
    6245           4 : static bool run_deletetest_ln(int dummy)
    6246             : {
    6247           0 :         struct cli_state *cli;
    6248           4 :         const char *fname = "\\delete1";
    6249           4 :         const char *fname_ln = "\\delete1_ln";
    6250           0 :         uint16_t fnum;
    6251           0 :         uint16_t fnum1;
    6252           0 :         NTSTATUS status;
    6253           4 :         bool correct = true;
    6254           0 :         time_t t;
    6255             : 
    6256           4 :         printf("starting deletetest-ln\n");
    6257             : 
    6258           4 :         if (!torture_open_connection(&cli, 0)) {
    6259           0 :                 return false;
    6260             :         }
    6261             : 
    6262           4 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6263           4 :         cli_unlink(cli, fname_ln, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6264             : 
    6265           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    6266             : 
    6267             :         /* Create the file. */
    6268           4 :         status = cli_openx(cli, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum);
    6269           4 :         if (!NT_STATUS_IS_OK(status)) {
    6270           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    6271           0 :                 return false;
    6272             :         }
    6273             : 
    6274           4 :         status = cli_close(cli, fnum);
    6275           4 :         if (!NT_STATUS_IS_OK(status)) {
    6276           0 :                 printf("close1 failed (%s)\n", nt_errstr(status));
    6277           0 :                 return false;
    6278             :         }
    6279             : 
    6280             :         /* Now create a hardlink. */
    6281           4 :         status = cli_hardlink(cli, fname, fname_ln);
    6282           4 :         if (!NT_STATUS_IS_OK(status)) {
    6283           0 :                 printf("nt hardlink failed (%s)\n", nt_errstr(status));
    6284           0 :                 return false;
    6285             :         }
    6286             : 
    6287             :         /* Open the original file. */
    6288           4 :         status = cli_ntcreate(cli, fname, 0, FILE_READ_DATA,
    6289             :                         FILE_ATTRIBUTE_NORMAL,
    6290             :                         FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    6291             :                         FILE_OPEN_IF, 0, 0, &fnum, NULL);
    6292           4 :         if (!NT_STATUS_IS_OK(status)) {
    6293           0 :                 printf("ntcreate of %s failed (%s)\n", fname, nt_errstr(status));
    6294           0 :                 return false;
    6295             :         }
    6296             : 
    6297             :         /* Unlink the hard link path. */
    6298           4 :         status = cli_ntcreate(cli, fname_ln, 0, DELETE_ACCESS,
    6299             :                         FILE_ATTRIBUTE_NORMAL,
    6300             :                         FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    6301             :                         FILE_OPEN_IF, 0, 0, &fnum1, NULL);
    6302           4 :         if (!NT_STATUS_IS_OK(status)) {
    6303           0 :                 printf("ntcreate of %s failed (%s)\n", fname_ln, nt_errstr(status));
    6304           0 :                 return false;
    6305             :         }
    6306           4 :         status = cli_nt_delete_on_close(cli, fnum1, true);
    6307           4 :         if (!NT_STATUS_IS_OK(status)) {
    6308           0 :                 d_printf("(%s) failed to set delete_on_close %s: %s\n",
    6309             :                         __location__, fname_ln, nt_errstr(status));
    6310           0 :                 return false;
    6311             :         }
    6312             : 
    6313           4 :         status = cli_close(cli, fnum1);
    6314           4 :         if (!NT_STATUS_IS_OK(status)) {
    6315           0 :                 printf("close %s failed (%s)\n",
    6316             :                         fname_ln, nt_errstr(status));
    6317           0 :                 return false;
    6318             :         }
    6319             : 
    6320           4 :         status = cli_close(cli, fnum);
    6321           4 :         if (!NT_STATUS_IS_OK(status)) {
    6322           0 :                 printf("close %s failed (%s)\n",
    6323             :                         fname, nt_errstr(status));
    6324           0 :                 return false;
    6325             :         }
    6326             : 
    6327             :         /* Ensure the original file is still there. */
    6328           4 :         status = cli_getatr(cli, fname, NULL, NULL, &t);
    6329           4 :         if (!NT_STATUS_IS_OK(status)) {
    6330           0 :                 printf("%s getatr on file %s failed (%s)\n",
    6331             :                         __location__,
    6332             :                         fname,
    6333             :                         nt_errstr(status));
    6334           0 :                 correct = False;
    6335             :         }
    6336             : 
    6337             :         /* Ensure the link path is gone. */
    6338           4 :         status = cli_getatr(cli, fname_ln, NULL, NULL, &t);
    6339           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_NAME_NOT_FOUND)) {
    6340           0 :                 printf("%s, getatr for file %s returned wrong error code %s "
    6341             :                         "- should have been deleted\n",
    6342             :                         __location__,
    6343             :                         fname_ln, nt_errstr(status));
    6344           0 :                 correct = False;
    6345             :         }
    6346             : 
    6347           4 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6348           4 :         cli_unlink(cli, fname_ln, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6349             : 
    6350           4 :         if (!torture_close_connection(cli)) {
    6351           0 :                 correct = false;
    6352             :         }
    6353             : 
    6354           4 :         printf("finished deletetest-ln\n");
    6355             : 
    6356           4 :         return correct;
    6357             : }
    6358             : 
    6359             : /*
    6360             :   print out server properties
    6361             :  */
    6362           5 : static bool run_properties(int dummy)
    6363             : {
    6364           0 :         struct cli_state *cli;
    6365           5 :         bool correct = True;
    6366             : 
    6367           5 :         printf("starting properties test\n");
    6368             : 
    6369           5 :         ZERO_STRUCT(cli);
    6370             : 
    6371           5 :         if (!torture_open_connection(&cli, 0)) {
    6372           0 :                 return False;
    6373             :         }
    6374             : 
    6375           5 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    6376             : 
    6377           5 :         d_printf("Capabilities 0x%08x\n", smb1cli_conn_capabilities(cli->conn));
    6378             : 
    6379           5 :         if (!torture_close_connection(cli)) {
    6380           0 :                 correct = False;
    6381             :         }
    6382             : 
    6383           5 :         return correct;
    6384             : }
    6385             : 
    6386             : 
    6387             : 
    6388             : /* FIRST_DESIRED_ACCESS   0xf019f */
    6389             : #define FIRST_DESIRED_ACCESS   FILE_READ_DATA|FILE_WRITE_DATA|FILE_APPEND_DATA|\
    6390             :                                FILE_READ_EA|                           /* 0xf */ \
    6391             :                                FILE_WRITE_EA|FILE_READ_ATTRIBUTES|     /* 0x90 */ \
    6392             :                                FILE_WRITE_ATTRIBUTES|                  /* 0x100 */ \
    6393             :                                DELETE_ACCESS|READ_CONTROL_ACCESS|\
    6394             :                                WRITE_DAC_ACCESS|WRITE_OWNER_ACCESS     /* 0xf0000 */
    6395             : /* SECOND_DESIRED_ACCESS  0xe0080 */
    6396             : #define SECOND_DESIRED_ACCESS  FILE_READ_ATTRIBUTES|                   /* 0x80 */ \
    6397             :                                READ_CONTROL_ACCESS|WRITE_DAC_ACCESS|\
    6398             :                                WRITE_OWNER_ACCESS                      /* 0xe0000 */
    6399             : 
    6400             : #if 0
    6401             : #define THIRD_DESIRED_ACCESS   FILE_READ_ATTRIBUTES|                   /* 0x80 */ \
    6402             :                                READ_CONTROL_ACCESS|WRITE_DAC_ACCESS|\
    6403             :                                FILE_READ_DATA|\
    6404             :                                WRITE_OWNER_ACCESS                      /* */
    6405             : #endif
    6406             : 
    6407             : /*
    6408             :   Test ntcreate calls made by xcopy
    6409             :  */
    6410           5 : static bool run_xcopy(int dummy)
    6411             : {
    6412           0 :         static struct cli_state *cli1;
    6413           5 :         const char *fname = "\\test.txt";
    6414           5 :         bool correct = True;
    6415           0 :         uint16_t fnum1, fnum2;
    6416           0 :         NTSTATUS status;
    6417             : 
    6418           5 :         printf("starting xcopy test\n");
    6419             : 
    6420           5 :         if (!torture_open_connection(&cli1, 0)) {
    6421           0 :                 return False;
    6422             :         }
    6423             : 
    6424           5 :         status = cli_ntcreate(cli1, fname, 0, FIRST_DESIRED_ACCESS,
    6425             :                               FILE_ATTRIBUTE_ARCHIVE, FILE_SHARE_NONE,
    6426             :                               FILE_OVERWRITE_IF, 0x4044, 0, &fnum1, NULL);
    6427           5 :         if (!NT_STATUS_IS_OK(status)) {
    6428           0 :                 printf("First open failed - %s\n", nt_errstr(status));
    6429           0 :                 return False;
    6430             :         }
    6431             : 
    6432           5 :         status = cli_ntcreate(cli1, fname, 0, SECOND_DESIRED_ACCESS, 0,
    6433             :                              FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    6434             :                              FILE_OPEN, 0x200000, 0, &fnum2, NULL);
    6435           5 :         if (!NT_STATUS_IS_OK(status)) {
    6436           0 :                 printf("second open failed - %s\n", nt_errstr(status));
    6437           0 :                 return False;
    6438             :         }
    6439             : 
    6440           5 :         if (!torture_close_connection(cli1)) {
    6441           0 :                 correct = False;
    6442             :         }
    6443             : 
    6444           5 :         return correct;
    6445             : }
    6446             : 
    6447             : /*
    6448             :   Test rename on files open with share delete and no share delete.
    6449             :  */
    6450           5 : static bool run_rename(int dummy)
    6451             : {
    6452           0 :         static struct cli_state *cli1;
    6453           5 :         const char *fname = "\\test.txt";
    6454           5 :         const char *fname1 = "\\test1.txt";
    6455           5 :         bool correct = True;
    6456           0 :         uint16_t fnum1;
    6457           0 :         uint32_t attr;
    6458           0 :         NTSTATUS status;
    6459             : 
    6460           5 :         printf("starting rename test\n");
    6461             : 
    6462           5 :         if (!torture_open_connection(&cli1, 0)) {
    6463           0 :                 return False;
    6464             :         }
    6465             : 
    6466           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6467           5 :         cli_unlink(cli1, fname1, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6468             : 
    6469           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_READ_ACCESS,
    6470             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_READ,
    6471             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    6472           5 :         if (!NT_STATUS_IS_OK(status)) {
    6473           0 :                 printf("First open failed - %s\n", nt_errstr(status));
    6474           0 :                 return False;
    6475             :         }
    6476             : 
    6477           5 :         status = cli_rename(cli1, fname, fname1, false);
    6478           5 :         if (!NT_STATUS_IS_OK(status)) {
    6479           5 :                 printf("First rename failed (SHARE_READ) (this is correct) - %s\n", nt_errstr(status));
    6480             :         } else {
    6481           0 :                 printf("First rename succeeded (SHARE_READ) - this should have failed !\n");
    6482           0 :                 correct = False;
    6483             :         }
    6484             : 
    6485           5 :         status = cli_close(cli1, fnum1);
    6486           5 :         if (!NT_STATUS_IS_OK(status)) {
    6487           0 :                 printf("close - 1 failed (%s)\n", nt_errstr(status));
    6488           0 :                 return False;
    6489             :         }
    6490             : 
    6491           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6492           5 :         cli_unlink(cli1, fname1, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6493           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_READ_ACCESS, FILE_ATTRIBUTE_NORMAL,
    6494             : #if 0
    6495             :                               FILE_SHARE_DELETE|FILE_SHARE_NONE,
    6496             : #else
    6497             :                               FILE_SHARE_DELETE|FILE_SHARE_READ,
    6498             : #endif
    6499             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    6500           5 :         if (!NT_STATUS_IS_OK(status)) {
    6501           0 :                 printf("Second open failed - %s\n", nt_errstr(status));
    6502           0 :                 return False;
    6503             :         }
    6504             : 
    6505           5 :         status = cli_rename(cli1, fname, fname1, false);
    6506           5 :         if (!NT_STATUS_IS_OK(status)) {
    6507           0 :                 printf("Second rename failed (SHARE_DELETE | SHARE_READ) - this should have succeeded - %s\n", nt_errstr(status));
    6508           0 :                 correct = False;
    6509             :         } else {
    6510           5 :                 printf("Second rename succeeded (SHARE_DELETE | SHARE_READ)\n");
    6511             :         }
    6512             : 
    6513           5 :         status = cli_close(cli1, fnum1);
    6514           5 :         if (!NT_STATUS_IS_OK(status)) {
    6515           0 :                 printf("close - 2 failed (%s)\n", nt_errstr(status));
    6516           0 :                 return False;
    6517             :         }
    6518             : 
    6519           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6520           5 :         cli_unlink(cli1, fname1, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6521             : 
    6522           5 :         status = cli_ntcreate(cli1, fname, 0, READ_CONTROL_ACCESS,
    6523             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_NONE,
    6524             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    6525           5 :         if (!NT_STATUS_IS_OK(status)) {
    6526           0 :                 printf("Third open failed - %s\n", nt_errstr(status));
    6527           0 :                 return False;
    6528             :         }
    6529             : 
    6530             : 
    6531           5 :         status = cli_rename(cli1, fname, fname1, false);
    6532           5 :         if (!NT_STATUS_IS_OK(status)) {
    6533           0 :                 printf("Third rename failed (SHARE_NONE) - this should have succeeded - %s\n", nt_errstr(status));
    6534           0 :                 correct = False;
    6535             :         } else {
    6536           5 :                 printf("Third rename succeeded (SHARE_NONE)\n");
    6537             :         }
    6538             : 
    6539           5 :         status = cli_close(cli1, fnum1);
    6540           5 :         if (!NT_STATUS_IS_OK(status)) {
    6541           0 :                 printf("close - 3 failed (%s)\n", nt_errstr(status));
    6542           0 :                 return False;
    6543             :         }
    6544             : 
    6545           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6546           5 :         cli_unlink(cli1, fname1, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6547             : 
    6548             :         /*----*/
    6549             : 
    6550           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_READ_ACCESS,
    6551             :                               FILE_ATTRIBUTE_NORMAL,
    6552             :                               FILE_SHARE_READ | FILE_SHARE_WRITE,
    6553             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    6554           5 :         if (!NT_STATUS_IS_OK(status)) {
    6555           0 :                 printf("Fourth open failed - %s\n", nt_errstr(status));
    6556           0 :                 return False;
    6557             :         }
    6558             : 
    6559           5 :         status = cli_rename(cli1, fname, fname1, false);
    6560           5 :         if (!NT_STATUS_IS_OK(status)) {
    6561           5 :                 printf("Fourth rename failed (SHARE_READ | SHARE_WRITE) (this is correct) - %s\n", nt_errstr(status));
    6562             :         } else {
    6563           0 :                 printf("Fourth rename succeeded (SHARE_READ | SHARE_WRITE) - this should have failed !\n");
    6564           0 :                 correct = False;
    6565             :         }
    6566             : 
    6567           5 :         status = cli_close(cli1, fnum1);
    6568           5 :         if (!NT_STATUS_IS_OK(status)) {
    6569           0 :                 printf("close - 4 failed (%s)\n", nt_errstr(status));
    6570           0 :                 return False;
    6571             :         }
    6572             : 
    6573           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6574           5 :         cli_unlink(cli1, fname1, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6575             : 
    6576             :         /*--*/
    6577             : 
    6578           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_READ_ACCESS,
    6579             :                          FILE_ATTRIBUTE_NORMAL,
    6580             :                          FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
    6581             :                          FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    6582           5 :         if (!NT_STATUS_IS_OK(status)) {
    6583           0 :                 printf("Fifth open failed - %s\n", nt_errstr(status));
    6584           0 :                 return False;
    6585             :         }
    6586             : 
    6587           5 :         status = cli_rename(cli1, fname, fname1, false);
    6588           5 :         if (!NT_STATUS_IS_OK(status)) {
    6589           0 :                 printf("Fifth rename failed (SHARE_READ | SHARE_WRITE | SHARE_DELETE) - this should have succeeded - %s ! \n", nt_errstr(status));
    6590           0 :                 correct = False;
    6591             :         } else {
    6592           5 :                 printf("Fifth rename succeeded (SHARE_READ | SHARE_WRITE | SHARE_DELETE) (this is correct) - %s\n", nt_errstr(status));
    6593             :         }
    6594             : 
    6595             :         /*--*/
    6596           5 :         status = cli_close(cli1, fnum1);
    6597           5 :         if (!NT_STATUS_IS_OK(status)) {
    6598           0 :                 printf("close - 5 failed (%s)\n", nt_errstr(status));
    6599           0 :                 return False;
    6600             :         }
    6601             : 
    6602             :         /* Check that the renamed file has FILE_ATTRIBUTE_ARCHIVE. */
    6603           5 :         status = cli_getatr(cli1, fname1, &attr, NULL, NULL);
    6604           5 :         if (!NT_STATUS_IS_OK(status)) {
    6605           0 :                 printf("getatr on file %s failed - %s ! \n",
    6606             :                         fname1, nt_errstr(status));
    6607           0 :                 correct = False;
    6608             :         } else {
    6609           5 :                 if (attr != FILE_ATTRIBUTE_ARCHIVE) {
    6610           0 :                         printf("Renamed file %s has wrong attr 0x%x "
    6611             :                                 "(should be 0x%x)\n",
    6612             :                                 fname1,
    6613             :                                 attr,
    6614             :                                 (unsigned int)FILE_ATTRIBUTE_ARCHIVE);
    6615           0 :                         correct = False;
    6616             :                 } else {
    6617           5 :                         printf("Renamed file %s has archive bit set\n", fname1);
    6618             :                 }
    6619             :         }
    6620             : 
    6621           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6622           5 :         cli_unlink(cli1, fname1, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6623             : 
    6624           5 :         if (!torture_close_connection(cli1)) {
    6625           0 :                 correct = False;
    6626             :         }
    6627             : 
    6628           5 :         return correct;
    6629             : }
    6630             : 
    6631             : /*
    6632             :   Test rename into a directory with an ACL denying it.
    6633             :  */
    6634           4 : static bool run_rename_access(int dummy)
    6635             : {
    6636           0 :         static struct cli_state *cli = NULL;
    6637           0 :         static struct cli_state *posix_cli = NULL;
    6638           4 :         const char *src = "test.txt";
    6639           4 :         const char *dname = "dir";
    6640           4 :         const char *dst = "dir\\test.txt";
    6641           4 :         const char *dsrc = "test.dir";
    6642           4 :         const char *ddst = "dir\\test.dir";
    6643           4 :         uint16_t fnum = (uint16_t)-1;
    6644           4 :         struct security_descriptor *sd = NULL;
    6645           4 :         struct security_descriptor *newsd = NULL;
    6646           0 :         NTSTATUS status;
    6647           4 :         TALLOC_CTX *frame = NULL;
    6648             : 
    6649           4 :         frame = talloc_stackframe();
    6650           4 :         printf("starting rename access test\n");
    6651             : 
    6652             :         /* Windows connection. */
    6653           4 :         if (!torture_open_connection(&cli, 0)) {
    6654           0 :                 goto fail;
    6655             :         }
    6656             : 
    6657           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    6658             : 
    6659             :         /* Posix connection. */
    6660           4 :         if (!torture_open_connection(&posix_cli, 0)) {
    6661           0 :                 goto fail;
    6662             :         }
    6663             : 
    6664           4 :         smbXcli_conn_set_sockopt(posix_cli->conn, sockops);
    6665             : 
    6666           4 :         status = torture_setup_unix_extensions(posix_cli);
    6667           4 :         if (!NT_STATUS_IS_OK(status)) {
    6668           0 :                 goto fail;
    6669             :         }
    6670             : 
    6671             :         /* Start with a clean slate. */
    6672           4 :         cli_unlink(cli, src, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6673           4 :         cli_unlink(cli, dst, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6674           4 :         cli_rmdir(cli, dsrc);
    6675           4 :         cli_rmdir(cli, ddst);
    6676           4 :         cli_rmdir(cli, dname);
    6677             : 
    6678             :         /*
    6679             :          * Setup the destination directory with a DENY ACE to
    6680             :          * prevent new files within it.
    6681             :          */
    6682           4 :         status = cli_ntcreate(cli,
    6683             :                                 dname,
    6684             :                                 0,
    6685             :                                 FILE_READ_ATTRIBUTES|READ_CONTROL_ACCESS|
    6686             :                                         WRITE_DAC_ACCESS|FILE_READ_DATA|
    6687             :                                         WRITE_OWNER_ACCESS,
    6688             :                                 FILE_ATTRIBUTE_DIRECTORY,
    6689             :                                 FILE_SHARE_READ|FILE_SHARE_WRITE,
    6690             :                                 FILE_CREATE,
    6691             :                                 FILE_DIRECTORY_FILE,
    6692             :                                 0,
    6693             :                                 &fnum,
    6694             :                                 NULL);
    6695           4 :         if (!NT_STATUS_IS_OK(status)) {
    6696           0 :                 printf("Create of %s - %s\n", dname, nt_errstr(status));
    6697           0 :                 goto fail;
    6698             :         }
    6699             : 
    6700           4 :         status = cli_query_secdesc(cli,
    6701             :                                 fnum,
    6702             :                                 frame,
    6703             :                                 &sd);
    6704           4 :         if (!NT_STATUS_IS_OK(status)) {
    6705           0 :                 printf("cli_query_secdesc failed for %s (%s)\n",
    6706             :                         dname, nt_errstr(status));
    6707           0 :                 goto fail;
    6708             :         }
    6709             : 
    6710           4 :         newsd = security_descriptor_dacl_create(frame,
    6711             :                                         0,
    6712             :                                         NULL,
    6713             :                                         NULL,
    6714             :                                         SID_WORLD,
    6715             :                                         SEC_ACE_TYPE_ACCESS_DENIED,
    6716             :                                         SEC_DIR_ADD_FILE|SEC_DIR_ADD_SUBDIR,
    6717             :                                         0,
    6718             :                                         NULL);
    6719           4 :         if (newsd == NULL) {
    6720           0 :                 goto fail;
    6721             :         }
    6722           8 :         sd->dacl = security_acl_concatenate(frame,
    6723           4 :                                         newsd->dacl,
    6724           4 :                                         sd->dacl);
    6725           4 :         if (sd->dacl == NULL) {
    6726           0 :                 goto fail;
    6727             :         }
    6728           4 :         status = cli_set_secdesc(cli, fnum, sd);
    6729           4 :         if (!NT_STATUS_IS_OK(status)) {
    6730           0 :                 printf("cli_set_secdesc failed for %s (%s)\n",
    6731             :                         dname, nt_errstr(status));
    6732           0 :                 goto fail;
    6733             :         }
    6734           4 :         status = cli_close(cli, fnum);
    6735           4 :         if (!NT_STATUS_IS_OK(status)) {
    6736           0 :                 printf("close failed for %s (%s)\n",
    6737             :                         dname, nt_errstr(status));
    6738           0 :                 goto fail;
    6739             :         }
    6740             :         /* Now go around the back and chmod to 777 via POSIX. */
    6741           4 :         status = cli_posix_chmod(posix_cli, dname, 0777);
    6742           4 :         if (!NT_STATUS_IS_OK(status)) {
    6743           0 :                 printf("cli_posix_chmod failed for %s (%s)\n",
    6744             :                         dname, nt_errstr(status));
    6745           0 :                 goto fail;
    6746             :         }
    6747             : 
    6748             :         /* Check we can't create a file within dname via Windows. */
    6749           4 :         status = cli_openx(cli, dst, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum);
    6750           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_ACCESS_DENIED)) {
    6751           0 :                 cli_close(posix_cli, fnum);
    6752           0 :                 printf("Create of %s should be ACCESS denied, was %s\n",
    6753             :                         dst, nt_errstr(status));
    6754           0 :                 goto fail;
    6755             :         }
    6756             : 
    6757             :         /* Make the sample file/directory. */
    6758           4 :         status = cli_openx(cli, src, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum);
    6759           4 :         if (!NT_STATUS_IS_OK(status)) {
    6760           0 :                 printf("open of %s failed (%s)\n", src, nt_errstr(status));
    6761           0 :                 goto fail;
    6762             :         }
    6763           4 :         status = cli_close(cli, fnum);
    6764           4 :         if (!NT_STATUS_IS_OK(status)) {
    6765           0 :                 printf("cli_close failed (%s)\n", nt_errstr(status));
    6766           0 :                 goto fail;
    6767             :         }
    6768             : 
    6769           4 :         status = cli_mkdir(cli, dsrc);
    6770           4 :         if (!NT_STATUS_IS_OK(status)) {
    6771           0 :                 printf("cli_mkdir of %s failed (%s)\n",
    6772             :                         dsrc, nt_errstr(status));
    6773           0 :                 goto fail;
    6774             :         }
    6775             : 
    6776             :         /*
    6777             :          * OK - renames of the new file and directory into the
    6778             :          * dst directory should fail.
    6779             :          */
    6780             : 
    6781           4 :         status = cli_rename(cli, src, dst, false);
    6782           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_ACCESS_DENIED)) {
    6783           0 :                 printf("rename of %s -> %s should be ACCESS denied, was %s\n",
    6784             :                         src, dst, nt_errstr(status));
    6785           0 :                 goto fail;
    6786             :         }
    6787           4 :         status = cli_rename(cli, dsrc, ddst, false);
    6788           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_ACCESS_DENIED)) {
    6789           0 :                 printf("rename of %s -> %s should be ACCESS denied, was %s\n",
    6790             :                         src, dst, nt_errstr(status));
    6791           0 :                 goto fail;
    6792             :         }
    6793             : 
    6794           4 :         TALLOC_FREE(frame);
    6795           4 :         return true;
    6796             : 
    6797           0 :   fail:
    6798             : 
    6799           0 :         if (posix_cli) {
    6800           0 :                 torture_close_connection(posix_cli);
    6801             :         }
    6802             : 
    6803           0 :         if (cli) {
    6804           0 :                 if (fnum != (uint16_t)-1) {
    6805           0 :                         cli_close(cli, fnum);
    6806             :                 }
    6807           0 :                 cli_unlink(cli, src,
    6808             :                         FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6809           0 :                 cli_unlink(cli, dst,
    6810             :                         FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6811           0 :                 cli_rmdir(cli, dsrc);
    6812           0 :                 cli_rmdir(cli, ddst);
    6813           0 :                 cli_rmdir(cli, dname);
    6814             : 
    6815           0 :                 torture_close_connection(cli);
    6816             :         }
    6817             : 
    6818           0 :         TALLOC_FREE(frame);
    6819           0 :         return false;
    6820             : }
    6821             : 
    6822             : /*
    6823             :   Test owner rights ACE.
    6824             :  */
    6825           4 : static bool run_owner_rights(int dummy)
    6826             : {
    6827           0 :         static struct cli_state *cli = NULL;
    6828           4 :         const char *fname = "owner_rights.txt";
    6829           4 :         uint16_t fnum = (uint16_t)-1;
    6830           4 :         struct security_descriptor *sd = NULL;
    6831           4 :         struct security_descriptor *newsd = NULL;
    6832           0 :         NTSTATUS status;
    6833           4 :         TALLOC_CTX *frame = NULL;
    6834             : 
    6835           4 :         frame = talloc_stackframe();
    6836           4 :         printf("starting owner rights test\n");
    6837             : 
    6838             :         /* Windows connection. */
    6839           4 :         if (!torture_open_connection(&cli, 0)) {
    6840           0 :                 goto fail;
    6841             :         }
    6842             : 
    6843           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    6844             : 
    6845             :         /* Start with a clean slate. */
    6846           4 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6847             : 
    6848             :         /* Create the test file. */
    6849             :         /* Now try and open for read and write-dac. */
    6850           4 :         status = cli_ntcreate(cli,
    6851             :                                 fname,
    6852             :                                 0,
    6853             :                                 GENERIC_ALL_ACCESS,
    6854             :                                 FILE_ATTRIBUTE_NORMAL,
    6855             :                                 FILE_SHARE_READ|FILE_SHARE_WRITE|
    6856             :                                         FILE_SHARE_DELETE,
    6857             :                                 FILE_CREATE,
    6858             :                                 0,
    6859             :                                 0,
    6860             :                                 &fnum,
    6861             :                                 NULL);
    6862           4 :         if (!NT_STATUS_IS_OK(status)) {
    6863           0 :                 printf("Create of %s - %s\n", fname, nt_errstr(status));
    6864           0 :                 goto fail;
    6865             :         }
    6866             : 
    6867             :         /* Get the original SD. */
    6868           4 :         status = cli_query_secdesc(cli,
    6869             :                                 fnum,
    6870             :                                 frame,
    6871             :                                 &sd);
    6872           4 :         if (!NT_STATUS_IS_OK(status)) {
    6873           0 :                 printf("cli_query_secdesc failed for %s (%s)\n",
    6874             :                         fname, nt_errstr(status));
    6875           0 :                 goto fail;
    6876             :         }
    6877             : 
    6878             :         /*
    6879             :          * Add an "owner-rights" ACE denying WRITE_DATA,
    6880             :          * and an "owner-rights" ACE allowing READ_DATA.
    6881             :          */
    6882             : 
    6883           4 :         newsd = security_descriptor_dacl_create(frame,
    6884             :                                         0,
    6885             :                                         NULL,
    6886             :                                         NULL,
    6887             :                                         SID_OWNER_RIGHTS,
    6888             :                                         SEC_ACE_TYPE_ACCESS_DENIED,
    6889             :                                         FILE_WRITE_DATA,
    6890             :                                         0,
    6891             :                                         SID_OWNER_RIGHTS,
    6892             :                                         SEC_ACE_TYPE_ACCESS_ALLOWED,
    6893             :                                         FILE_READ_DATA,
    6894             :                                         0,
    6895             :                                         NULL);
    6896           4 :         if (newsd == NULL) {
    6897           0 :                 goto fail;
    6898             :         }
    6899           8 :         sd->dacl = security_acl_concatenate(frame,
    6900           4 :                                         newsd->dacl,
    6901           4 :                                         sd->dacl);
    6902           4 :         if (sd->dacl == NULL) {
    6903           0 :                 goto fail;
    6904             :         }
    6905           4 :         status = cli_set_secdesc(cli, fnum, sd);
    6906           4 :         if (!NT_STATUS_IS_OK(status)) {
    6907           0 :                 printf("cli_set_secdesc failed for %s (%s)\n",
    6908             :                         fname, nt_errstr(status));
    6909           0 :                 goto fail;
    6910             :         }
    6911           4 :         status = cli_close(cli, fnum);
    6912           4 :         if (!NT_STATUS_IS_OK(status)) {
    6913           0 :                 printf("close failed for %s (%s)\n",
    6914             :                         fname, nt_errstr(status));
    6915           0 :                 goto fail;
    6916             :         }
    6917           4 :         fnum = (uint16_t)-1;
    6918             : 
    6919             :         /* Try and open for FILE_WRITE_DATA */
    6920           4 :         status = cli_ntcreate(cli,
    6921             :                                 fname,
    6922             :                                 0,
    6923             :                                 FILE_WRITE_DATA,
    6924             :                                 FILE_ATTRIBUTE_NORMAL,
    6925             :                                 FILE_SHARE_READ|FILE_SHARE_WRITE|
    6926             :                                         FILE_SHARE_DELETE,
    6927             :                                 FILE_OPEN,
    6928             :                                 0,
    6929             :                                 0,
    6930             :                                 &fnum,
    6931             :                                 NULL);
    6932           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_ACCESS_DENIED)) {
    6933           0 :                 printf("Open of %s - %s\n", fname, nt_errstr(status));
    6934           0 :                 goto fail;
    6935             :         }
    6936             : 
    6937             :         /* Now try and open for FILE_READ_DATA */
    6938           4 :         status = cli_ntcreate(cli,
    6939             :                                 fname,
    6940             :                                 0,
    6941             :                                 FILE_READ_DATA,
    6942             :                                 FILE_ATTRIBUTE_NORMAL,
    6943             :                                 FILE_SHARE_READ|FILE_SHARE_WRITE|
    6944             :                                         FILE_SHARE_DELETE,
    6945             :                                 FILE_OPEN,
    6946             :                                 0,
    6947             :                                 0,
    6948             :                                 &fnum,
    6949             :                                 NULL);
    6950           4 :         if (!NT_STATUS_IS_OK(status)) {
    6951           0 :                 printf("Open of %s - %s\n", fname, nt_errstr(status));
    6952           0 :                 goto fail;
    6953             :         }
    6954             : 
    6955           4 :         status = cli_close(cli, fnum);
    6956           4 :         if (!NT_STATUS_IS_OK(status)) {
    6957           0 :                 printf("close failed for %s (%s)\n",
    6958             :                         fname, nt_errstr(status));
    6959           0 :                 goto fail;
    6960             :         }
    6961             : 
    6962             :         /* Restore clean slate. */
    6963           4 :         TALLOC_FREE(sd);
    6964           4 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    6965             : 
    6966             :         /* Create the test file. */
    6967           4 :         status = cli_ntcreate(cli,
    6968             :                                 fname,
    6969             :                                 0,
    6970             :                                 GENERIC_ALL_ACCESS,
    6971             :                                 FILE_ATTRIBUTE_NORMAL,
    6972             :                                 FILE_SHARE_READ|FILE_SHARE_WRITE|
    6973             :                                         FILE_SHARE_DELETE,
    6974             :                                 FILE_CREATE,
    6975             :                                 0,
    6976             :                                 0,
    6977             :                                 &fnum,
    6978             :                                 NULL);
    6979           4 :         if (!NT_STATUS_IS_OK(status)) {
    6980           0 :                 printf("Create of %s - %s\n", fname, nt_errstr(status));
    6981           0 :                 goto fail;
    6982             :         }
    6983             : 
    6984             :         /* Get the original SD. */
    6985           4 :         status = cli_query_secdesc(cli,
    6986             :                                 fnum,
    6987             :                                 frame,
    6988             :                                 &sd);
    6989           4 :         if (!NT_STATUS_IS_OK(status)) {
    6990           0 :                 printf("cli_query_secdesc failed for %s (%s)\n",
    6991             :                         fname, nt_errstr(status));
    6992           0 :                 goto fail;
    6993             :         }
    6994             : 
    6995             :         /*
    6996             :          * Add an "owner-rights ACE denying WRITE_DATA,
    6997             :          * and an "owner-rights ACE allowing READ_DATA|WRITE_DATA.
    6998             :          */
    6999             : 
    7000           4 :         newsd = security_descriptor_dacl_create(frame,
    7001             :                                         0,
    7002             :                                         NULL,
    7003             :                                         NULL,
    7004             :                                         SID_OWNER_RIGHTS,
    7005             :                                         SEC_ACE_TYPE_ACCESS_DENIED,
    7006             :                                         FILE_WRITE_DATA,
    7007             :                                         0,
    7008             :                                         SID_OWNER_RIGHTS,
    7009             :                                         SEC_ACE_TYPE_ACCESS_ALLOWED,
    7010             :                                         FILE_READ_DATA|FILE_WRITE_DATA,
    7011             :                                         0,
    7012             :                                         NULL);
    7013           4 :         if (newsd == NULL) {
    7014           0 :                 goto fail;
    7015             :         }
    7016           8 :         sd->dacl = security_acl_concatenate(frame,
    7017           4 :                                         newsd->dacl,
    7018           4 :                                         sd->dacl);
    7019           4 :         if (sd->dacl == NULL) {
    7020           0 :                 goto fail;
    7021             :         }
    7022           4 :         status = cli_set_secdesc(cli, fnum, sd);
    7023           4 :         if (!NT_STATUS_IS_OK(status)) {
    7024           0 :                 printf("cli_set_secdesc failed for %s (%s)\n",
    7025             :                         fname, nt_errstr(status));
    7026           0 :                 goto fail;
    7027             :         }
    7028           4 :         status = cli_close(cli, fnum);
    7029           4 :         if (!NT_STATUS_IS_OK(status)) {
    7030           0 :                 printf("close failed for %s (%s)\n",
    7031             :                         fname, nt_errstr(status));
    7032           0 :                 goto fail;
    7033             :         }
    7034           4 :         fnum = (uint16_t)-1;
    7035             : 
    7036             :         /* Try and open for FILE_WRITE_DATA */
    7037           4 :         status = cli_ntcreate(cli,
    7038             :                                 fname,
    7039             :                                 0,
    7040             :                                 FILE_WRITE_DATA,
    7041             :                                 FILE_ATTRIBUTE_NORMAL,
    7042             :                                 FILE_SHARE_READ|FILE_SHARE_WRITE|
    7043             :                                         FILE_SHARE_DELETE,
    7044             :                                 FILE_OPEN,
    7045             :                                 0,
    7046             :                                 0,
    7047             :                                 &fnum,
    7048             :                                 NULL);
    7049           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_ACCESS_DENIED)) {
    7050           0 :                 printf("Open of %s - %s\n", fname, nt_errstr(status));
    7051           0 :                 goto fail;
    7052             :         }
    7053             : 
    7054             :         /* Now try and open for FILE_READ_DATA */
    7055           4 :         status = cli_ntcreate(cli,
    7056             :                                 fname,
    7057             :                                 0,
    7058             :                                 FILE_READ_DATA,
    7059             :                                 FILE_ATTRIBUTE_NORMAL,
    7060             :                                 FILE_SHARE_READ|FILE_SHARE_WRITE|
    7061             :                                         FILE_SHARE_DELETE,
    7062             :                                 FILE_OPEN,
    7063             :                                 0,
    7064             :                                 0,
    7065             :                                 &fnum,
    7066             :                                 NULL);
    7067           4 :         if (!NT_STATUS_IS_OK(status)) {
    7068           0 :                 printf("Open of %s - %s\n", fname, nt_errstr(status));
    7069           0 :                 goto fail;
    7070             :         }
    7071             : 
    7072           4 :         status = cli_close(cli, fnum);
    7073           4 :         if (!NT_STATUS_IS_OK(status)) {
    7074           0 :                 printf("close failed for %s (%s)\n",
    7075             :                         fname, nt_errstr(status));
    7076           0 :                 goto fail;
    7077             :         }
    7078             : 
    7079             :         /* Restore clean slate. */
    7080           4 :         TALLOC_FREE(sd);
    7081           4 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7082             : 
    7083             : 
    7084             :         /* Create the test file. */
    7085           4 :         status = cli_ntcreate(cli,
    7086             :                                 fname,
    7087             :                                 0,
    7088             :                                 GENERIC_ALL_ACCESS,
    7089             :                                 FILE_ATTRIBUTE_NORMAL,
    7090             :                                 FILE_SHARE_READ|FILE_SHARE_WRITE|
    7091             :                                         FILE_SHARE_DELETE,
    7092             :                                 FILE_CREATE,
    7093             :                                 0,
    7094             :                                 0,
    7095             :                                 &fnum,
    7096             :                                 NULL);
    7097           4 :         if (!NT_STATUS_IS_OK(status)) {
    7098           0 :                 printf("Create of %s - %s\n", fname, nt_errstr(status));
    7099           0 :                 goto fail;
    7100             :         }
    7101             : 
    7102             :         /* Get the original SD. */
    7103           4 :         status = cli_query_secdesc(cli,
    7104             :                                 fnum,
    7105             :                                 frame,
    7106             :                                 &sd);
    7107           4 :         if (!NT_STATUS_IS_OK(status)) {
    7108           0 :                 printf("cli_query_secdesc failed for %s (%s)\n",
    7109             :                         fname, nt_errstr(status));
    7110           0 :                 goto fail;
    7111             :         }
    7112             : 
    7113             :         /*
    7114             :          * Add an "authenticated users" ACE allowing READ_DATA,
    7115             :          * add an "owner-rights" denying READ_DATA,
    7116             :          * and an "authenticated users" ACE allowing WRITE_DATA.
    7117             :          */
    7118             : 
    7119           4 :         newsd = security_descriptor_dacl_create(frame,
    7120             :                                         0,
    7121             :                                         NULL,
    7122             :                                         NULL,
    7123             :                                         SID_NT_AUTHENTICATED_USERS,
    7124             :                                         SEC_ACE_TYPE_ACCESS_ALLOWED,
    7125             :                                         FILE_READ_DATA,
    7126             :                                         0,
    7127             :                                         SID_OWNER_RIGHTS,
    7128             :                                         SEC_ACE_TYPE_ACCESS_DENIED,
    7129             :                                         FILE_READ_DATA,
    7130             :                                         0,
    7131             :                                         SID_NT_AUTHENTICATED_USERS,
    7132             :                                         SEC_ACE_TYPE_ACCESS_ALLOWED,
    7133             :                                         FILE_WRITE_DATA,
    7134             :                                         0,
    7135             :                                         NULL);
    7136           4 :         if (newsd == NULL) {
    7137           0 :                 printf("newsd == NULL\n");
    7138           0 :                 goto fail;
    7139             :         }
    7140           8 :         sd->dacl = security_acl_concatenate(frame,
    7141           4 :                                         newsd->dacl,
    7142           4 :                                         sd->dacl);
    7143           4 :         if (sd->dacl == NULL) {
    7144           0 :                 printf("sd->dacl == NULL\n");
    7145           0 :                 goto fail;
    7146             :         }
    7147           4 :         status = cli_set_secdesc(cli, fnum, sd);
    7148           4 :         if (!NT_STATUS_IS_OK(status)) {
    7149           0 :                 printf("cli_set_secdesc failed for %s (%s)\n",
    7150             :                         fname, nt_errstr(status));
    7151           0 :                 goto fail;
    7152             :         }
    7153           4 :         status = cli_close(cli, fnum);
    7154           4 :         if (!NT_STATUS_IS_OK(status)) {
    7155           0 :                 printf("close failed for %s (%s)\n",
    7156             :                         fname, nt_errstr(status));
    7157           0 :                 goto fail;
    7158             :         }
    7159           4 :         fnum = (uint16_t)-1;
    7160             : 
    7161             :         /* Now try and open for FILE_READ_DATA|FILE_WRITE_DATA */
    7162           4 :         status = cli_ntcreate(cli,
    7163             :                                 fname,
    7164             :                                 0,
    7165             :                                 FILE_READ_DATA|FILE_WRITE_DATA,
    7166             :                                 FILE_ATTRIBUTE_NORMAL,
    7167             :                                 FILE_SHARE_READ|FILE_SHARE_WRITE|
    7168             :                                         FILE_SHARE_DELETE,
    7169             :                                 FILE_OPEN,
    7170             :                                 0,
    7171             :                                 0,
    7172             :                                 &fnum,
    7173             :                                 NULL);
    7174           4 :         if (!NT_STATUS_IS_OK(status)) {
    7175           0 :                 printf("Open of %s - %s\n", fname, nt_errstr(status));
    7176           0 :                 goto fail;
    7177             :         }
    7178             : 
    7179           4 :         status = cli_close(cli, fnum);
    7180           4 :         if (!NT_STATUS_IS_OK(status)) {
    7181           0 :                 printf("close failed for %s (%s)\n",
    7182             :                         fname, nt_errstr(status));
    7183           0 :                 goto fail;
    7184             :         }
    7185             : 
    7186           4 :         cli_unlink(cli, fname,
    7187             :                 FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7188             : 
    7189           4 :         TALLOC_FREE(frame);
    7190           4 :         return true;
    7191             : 
    7192           0 :   fail:
    7193             : 
    7194           0 :         if (cli) {
    7195           0 :                 if (fnum != (uint16_t)-1) {
    7196           0 :                         cli_close(cli, fnum);
    7197             :                 }
    7198           0 :                 cli_unlink(cli, fname,
    7199             :                         FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7200           0 :                 torture_close_connection(cli);
    7201             :         }
    7202             : 
    7203           0 :         TALLOC_FREE(frame);
    7204           0 :         return false;
    7205             : }
    7206             : 
    7207             : /*
    7208             :  * Test SMB1-specific open with SEC_FLAG_SYSTEM_SECURITY.
    7209             :  * Note this test only works with a user with SeSecurityPrivilege set.
    7210             :  *
    7211             :  * NB. This is also tested in samba3.base.createx_access
    7212             :  * but this makes it very explicit what we're looking for.
    7213             :  */
    7214           2 : static bool run_smb1_system_security(int dummy)
    7215             : {
    7216           0 :         static struct cli_state *cli = NULL;
    7217           2 :         const char *fname = "system_security.txt";
    7218           2 :         uint16_t fnum = (uint16_t)-1;
    7219           0 :         NTSTATUS status;
    7220           2 :         TALLOC_CTX *frame = NULL;
    7221             : 
    7222           2 :         frame = talloc_stackframe();
    7223           2 :         printf("starting smb1 system security test\n");
    7224             : 
    7225             :         /* SMB1 connection - torture_open_connection() forces this. */
    7226           2 :         if (!torture_open_connection(&cli, 0)) {
    7227           0 :                 goto fail;
    7228             :         }
    7229             : 
    7230           2 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    7231             : 
    7232             :         /* Start with a clean slate. */
    7233           2 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7234             : 
    7235             :         /* Create the test file. */
    7236           2 :         status = cli_ntcreate(cli,
    7237             :                                 fname,
    7238             :                                 0,
    7239             :                                 GENERIC_ALL_ACCESS,
    7240             :                                 FILE_ATTRIBUTE_NORMAL,
    7241             :                                 FILE_SHARE_READ|FILE_SHARE_WRITE|
    7242             :                                         FILE_SHARE_DELETE,
    7243             :                                 FILE_CREATE,
    7244             :                                 0,
    7245             :                                 0,
    7246             :                                 &fnum,
    7247             :                                 NULL);
    7248           2 :         if (!NT_STATUS_IS_OK(status)) {
    7249           0 :                 printf("Create of %s - %s\n", fname, nt_errstr(status));
    7250           0 :                 goto fail;
    7251             :         }
    7252             : 
    7253           2 :         status = cli_close(cli, fnum);
    7254             : 
    7255             :         /* Open with SEC_FLAG_SYSTEM_SECURITY only. */
    7256             :         /*
    7257             :          * On SMB1 this succeeds - SMB2 it fails,
    7258             :          * see the SMB2-SACL test.
    7259             :          */
    7260           2 :         status = cli_ntcreate(cli,
    7261             :                                 fname,
    7262             :                                 0,
    7263             :                                 SEC_FLAG_SYSTEM_SECURITY,
    7264             :                                 FILE_ATTRIBUTE_NORMAL,
    7265             :                                 FILE_SHARE_READ|FILE_SHARE_WRITE|
    7266             :                                         FILE_SHARE_DELETE,
    7267             :                                 FILE_OPEN,
    7268             :                                 0,
    7269             :                                 0,
    7270             :                                 &fnum,
    7271             :                                 NULL);
    7272           2 :         if (!NT_STATUS_IS_OK(status)) {
    7273           0 :                 printf("Open of %s - %s\n", fname, nt_errstr(status));
    7274           0 :                 goto fail;
    7275             :         }
    7276             : 
    7277           2 :         status = cli_close(cli, fnum);
    7278             : 
    7279           2 :         cli_unlink(cli, fname,
    7280             :                 FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7281             : 
    7282           2 :         torture_close_connection(cli);
    7283           2 :         TALLOC_FREE(frame);
    7284           2 :         return true;
    7285             : 
    7286           0 :   fail:
    7287             : 
    7288           0 :         if (cli) {
    7289           0 :                 if (fnum != (uint16_t)-1) {
    7290           0 :                         cli_close(cli, fnum);
    7291             :                 }
    7292           0 :                 cli_unlink(cli, fname,
    7293             :                         FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7294           0 :                 torture_close_connection(cli);
    7295             :         }
    7296             : 
    7297           0 :         TALLOC_FREE(frame);
    7298           0 :         return false;
    7299             : }
    7300             : 
    7301           0 : static bool run_pipe_number(int dummy)
    7302             : {
    7303           0 :         struct cli_state *cli1;
    7304           0 :         const char *pipe_name = "\\SPOOLSS";
    7305           0 :         uint16_t fnum;
    7306           0 :         int num_pipes = 0;
    7307           0 :         NTSTATUS status;
    7308             : 
    7309           0 :         printf("starting pipenumber test\n");
    7310           0 :         if (!torture_open_connection(&cli1, 0)) {
    7311           0 :                 return False;
    7312             :         }
    7313             : 
    7314           0 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    7315           0 :         while(1) {
    7316           0 :                 status = cli_ntcreate(cli1, pipe_name, 0, FILE_READ_DATA,
    7317             :                                       FILE_ATTRIBUTE_NORMAL,
    7318             :                                       FILE_SHARE_READ|FILE_SHARE_WRITE,
    7319             :                                       FILE_OPEN_IF, 0, 0, &fnum, NULL);
    7320           0 :                 if (!NT_STATUS_IS_OK(status)) {
    7321           0 :                         printf("Open of pipe %s failed with error (%s)\n", pipe_name, nt_errstr(status));
    7322           0 :                         break;
    7323             :                 }
    7324           0 :                 num_pipes++;
    7325           0 :                 printf("\r%6d", num_pipes);
    7326             :         }
    7327             : 
    7328           0 :         printf("pipe_number test - we can open %d %s pipes.\n", num_pipes, pipe_name );
    7329           0 :         torture_close_connection(cli1);
    7330           0 :         return True;
    7331             : }
    7332             : 
    7333             : /*
    7334             :   Test open mode returns on read-only files.
    7335             :  */
    7336           5 : static bool run_opentest(int dummy)
    7337             : {
    7338           0 :         static struct cli_state *cli1;
    7339           0 :         static struct cli_state *cli2;
    7340           5 :         const char *fname = "\\readonly.file";
    7341           0 :         uint16_t fnum1, fnum2;
    7342           0 :         char buf[20];
    7343           0 :         off_t fsize;
    7344           5 :         bool correct = True;
    7345           0 :         char *tmp_path;
    7346           0 :         NTSTATUS status;
    7347             : 
    7348           5 :         printf("starting open test\n");
    7349             : 
    7350           5 :         if (!torture_open_connection(&cli1, 0)) {
    7351           0 :                 return False;
    7352             :         }
    7353             : 
    7354           5 :         cli_setatr(cli1, fname, 0, 0);
    7355           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7356             : 
    7357           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    7358             : 
    7359           5 :         status = cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum1);
    7360           5 :         if (!NT_STATUS_IS_OK(status)) {
    7361           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    7362           0 :                 return False;
    7363             :         }
    7364             : 
    7365           5 :         status = cli_close(cli1, fnum1);
    7366           5 :         if (!NT_STATUS_IS_OK(status)) {
    7367           0 :                 printf("close2 failed (%s)\n", nt_errstr(status));
    7368           0 :                 return False;
    7369             :         }
    7370             : 
    7371           5 :         status = cli_setatr(cli1, fname, FILE_ATTRIBUTE_READONLY, 0);
    7372           5 :         if (!NT_STATUS_IS_OK(status)) {
    7373           0 :                 printf("cli_setatr failed (%s)\n", nt_errstr(status));
    7374           0 :                 return False;
    7375             :         }
    7376             : 
    7377           5 :         status = cli_openx(cli1, fname, O_RDONLY, DENY_WRITE, &fnum1);
    7378           5 :         if (!NT_STATUS_IS_OK(status)) {
    7379           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    7380           0 :                 return False;
    7381             :         }
    7382             : 
    7383             :         /* This will fail - but the error should be ERRnoaccess, not ERRbadshare. */
    7384           5 :         status = cli_openx(cli1, fname, O_RDWR, DENY_ALL, &fnum2);
    7385             : 
    7386           5 :         if (check_error(__LINE__, status, ERRDOS, ERRnoaccess,
    7387           5 :                         NT_STATUS_ACCESS_DENIED)) {
    7388           4 :                 printf("correct error code ERRDOS/ERRnoaccess returned\n");
    7389             :         }
    7390             : 
    7391           5 :         printf("finished open test 1\n");
    7392             : 
    7393           5 :         cli_close(cli1, fnum1);
    7394             : 
    7395             :         /* Now try not readonly and ensure ERRbadshare is returned. */
    7396             : 
    7397           5 :         cli_setatr(cli1, fname, 0, 0);
    7398             : 
    7399           5 :         status = cli_openx(cli1, fname, O_RDONLY, DENY_WRITE, &fnum1);
    7400           5 :         if (!NT_STATUS_IS_OK(status)) {
    7401           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    7402           0 :                 return False;
    7403             :         }
    7404             : 
    7405             :         /* This will fail - but the error should be ERRshare. */
    7406           5 :         status = cli_openx(cli1, fname, O_RDWR, DENY_ALL, &fnum2);
    7407             : 
    7408           5 :         if (check_error(__LINE__, status, ERRDOS, ERRbadshare,
    7409           5 :                         NT_STATUS_SHARING_VIOLATION)) {
    7410           5 :                 printf("correct error code ERRDOS/ERRbadshare returned\n");
    7411             :         }
    7412             : 
    7413           5 :         status = cli_close(cli1, fnum1);
    7414           5 :         if (!NT_STATUS_IS_OK(status)) {
    7415           0 :                 printf("close2 failed (%s)\n", nt_errstr(status));
    7416           0 :                 return False;
    7417             :         }
    7418             : 
    7419           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7420             : 
    7421           5 :         printf("finished open test 2\n");
    7422             : 
    7423             :         /* Test truncate open disposition on file opened for read. */
    7424           5 :         status = cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum1);
    7425           5 :         if (!NT_STATUS_IS_OK(status)) {
    7426           0 :                 printf("(3) open (1) of %s failed (%s)\n", fname, nt_errstr(status));
    7427           0 :                 return False;
    7428             :         }
    7429             : 
    7430             :         /* write 20 bytes. */
    7431             : 
    7432           5 :         memset(buf, '\0', 20);
    7433             : 
    7434           5 :         status = cli_writeall(cli1, fnum1, 0, (uint8_t *)buf, 0, 20, NULL);
    7435           5 :         if (!NT_STATUS_IS_OK(status)) {
    7436           0 :                 printf("write failed (%s)\n", nt_errstr(status));
    7437           0 :                 correct = False;
    7438             :         }
    7439             : 
    7440           5 :         status = cli_close(cli1, fnum1);
    7441           5 :         if (!NT_STATUS_IS_OK(status)) {
    7442           0 :                 printf("(3) close1 failed (%s)\n", nt_errstr(status));
    7443           0 :                 return False;
    7444             :         }
    7445             : 
    7446             :         /* Ensure size == 20. */
    7447           5 :         status = cli_getatr(cli1, fname, NULL, &fsize, NULL);
    7448           5 :         if (!NT_STATUS_IS_OK(status)) {
    7449           0 :                 printf("(3) getatr failed (%s)\n", nt_errstr(status));
    7450           0 :                 return False;
    7451             :         }
    7452             : 
    7453           5 :         if (fsize != 20) {
    7454           0 :                 printf("(3) file size != 20\n");
    7455           0 :                 return False;
    7456             :         }
    7457             : 
    7458             :         /* Now test if we can truncate a file opened for readonly. */
    7459           5 :         status = cli_openx(cli1, fname, O_RDONLY|O_TRUNC, DENY_NONE, &fnum1);
    7460           5 :         if (!NT_STATUS_IS_OK(status)) {
    7461           0 :                 printf("(3) open (2) of %s failed (%s)\n", fname, nt_errstr(status));
    7462           0 :                 return False;
    7463             :         }
    7464             : 
    7465           5 :         status = cli_close(cli1, fnum1);
    7466           5 :         if (!NT_STATUS_IS_OK(status)) {
    7467           0 :                 printf("close2 failed (%s)\n", nt_errstr(status));
    7468           0 :                 return False;
    7469             :         }
    7470             : 
    7471             :         /* Ensure size == 0. */
    7472           5 :         status = cli_getatr(cli1, fname, NULL, &fsize, NULL);
    7473           5 :         if (!NT_STATUS_IS_OK(status)) {
    7474           0 :                 printf("(3) getatr failed (%s)\n", nt_errstr(status));
    7475           0 :                 return False;
    7476             :         }
    7477             : 
    7478           5 :         if (fsize != 0) {
    7479           0 :                 printf("(3) file size != 0\n");
    7480           0 :                 return False;
    7481             :         }
    7482           5 :         printf("finished open test 3\n");
    7483             : 
    7484           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7485             : 
    7486           5 :         printf("Do ctemp tests\n");
    7487           5 :         status = cli_ctemp(cli1, talloc_tos(), "\\", &fnum1, &tmp_path);
    7488           5 :         if (!NT_STATUS_IS_OK(status)) {
    7489           0 :                 printf("ctemp failed (%s)\n", nt_errstr(status));
    7490           0 :                 return False;
    7491             :         }
    7492             : 
    7493           5 :         printf("ctemp gave path %s\n", tmp_path);
    7494           5 :         status = cli_close(cli1, fnum1);
    7495           5 :         if (!NT_STATUS_IS_OK(status)) {
    7496           0 :                 printf("close of temp failed (%s)\n", nt_errstr(status));
    7497             :         }
    7498             : 
    7499           5 :         status = cli_unlink(cli1, tmp_path, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7500           5 :         if (!NT_STATUS_IS_OK(status)) {
    7501           0 :                 printf("unlink of temp failed (%s)\n", nt_errstr(status));
    7502             :         }
    7503             : 
    7504             :         /* Test the non-io opens... */
    7505             : 
    7506           5 :         if (!torture_open_connection(&cli2, 1)) {
    7507           0 :                 return False;
    7508             :         }
    7509             : 
    7510           5 :         cli_setatr(cli2, fname, 0, 0);
    7511           5 :         cli_unlink(cli2, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7512             : 
    7513           5 :         smbXcli_conn_set_sockopt(cli2->conn, sockops);
    7514             : 
    7515           5 :         printf("TEST #1 testing 2 non-io opens (no delete)\n");
    7516           5 :         status = cli_ntcreate(cli1, fname, 0, FILE_READ_ATTRIBUTES,
    7517             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_NONE,
    7518             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    7519           5 :         if (!NT_STATUS_IS_OK(status)) {
    7520           0 :                 printf("TEST #1 open 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7521           0 :                 return False;
    7522             :         }
    7523             : 
    7524           5 :         status = cli_ntcreate(cli2, fname, 0, FILE_READ_ATTRIBUTES,
    7525             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_NONE,
    7526             :                               FILE_OPEN_IF, 0, 0, &fnum2, NULL);
    7527           5 :         if (!NT_STATUS_IS_OK(status)) {
    7528           0 :                 printf("TEST #1 open 2 of %s failed (%s)\n", fname, nt_errstr(status));
    7529           0 :                 return False;
    7530             :         }
    7531             : 
    7532           5 :         status = cli_close(cli1, fnum1);
    7533           5 :         if (!NT_STATUS_IS_OK(status)) {
    7534           0 :                 printf("TEST #1 close 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7535           0 :                 return False;
    7536             :         }
    7537             : 
    7538           5 :         status = cli_close(cli2, fnum2);
    7539           5 :         if (!NT_STATUS_IS_OK(status)) {
    7540           0 :                 printf("TEST #1 close 2 of %s failed (%s)\n", fname, nt_errstr(status));
    7541           0 :                 return False;
    7542             :         }
    7543             : 
    7544           5 :         printf("non-io open test #1 passed.\n");
    7545             : 
    7546           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7547             : 
    7548           5 :         printf("TEST #2 testing 2 non-io opens (first with delete)\n");
    7549             : 
    7550           5 :         status = cli_ntcreate(cli1, fname, 0,
    7551             :                               DELETE_ACCESS|FILE_READ_ATTRIBUTES,
    7552             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_NONE,
    7553             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    7554           5 :         if (!NT_STATUS_IS_OK(status)) {
    7555           0 :                 printf("TEST #2 open 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7556           0 :                 return False;
    7557             :         }
    7558             : 
    7559           5 :         status = cli_ntcreate(cli2, fname, 0, FILE_READ_ATTRIBUTES,
    7560             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_NONE,
    7561             :                               FILE_OPEN_IF, 0, 0, &fnum2, NULL);
    7562           5 :         if (!NT_STATUS_IS_OK(status)) {
    7563           0 :                 printf("TEST #2 open 2 of %s failed (%s)\n", fname, nt_errstr(status));
    7564           0 :                 return False;
    7565             :         }
    7566             : 
    7567           5 :         status = cli_close(cli1, fnum1);
    7568           5 :         if (!NT_STATUS_IS_OK(status)) {
    7569           0 :                 printf("TEST #2 close 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7570           0 :                 return False;
    7571             :         }
    7572             : 
    7573           5 :         status = cli_close(cli2, fnum2);
    7574           5 :         if (!NT_STATUS_IS_OK(status)) {
    7575           0 :                 printf("TEST #2 close 2 of %s failed (%s)\n", fname, nt_errstr(status));
    7576           0 :                 return False;
    7577             :         }
    7578             : 
    7579           5 :         printf("non-io open test #2 passed.\n");
    7580             : 
    7581           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7582             : 
    7583           5 :         printf("TEST #3 testing 2 non-io opens (second with delete)\n");
    7584             : 
    7585           5 :         status = cli_ntcreate(cli1, fname, 0, FILE_READ_ATTRIBUTES,
    7586             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_NONE,
    7587             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    7588           5 :         if (!NT_STATUS_IS_OK(status)) {
    7589           0 :                 printf("TEST #3 open 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7590           0 :                 return False;
    7591             :         }
    7592             : 
    7593           5 :         status = cli_ntcreate(cli2, fname, 0,
    7594             :                               DELETE_ACCESS|FILE_READ_ATTRIBUTES,
    7595             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_NONE,
    7596             :                               FILE_OPEN_IF, 0, 0, &fnum2, NULL);
    7597           5 :         if (!NT_STATUS_IS_OK(status)) {
    7598           0 :                 printf("TEST #3 open 2 of %s failed (%s)\n", fname, nt_errstr(status));
    7599           0 :                 return False;
    7600             :         }
    7601             : 
    7602           5 :         status = cli_close(cli1, fnum1);
    7603           5 :         if (!NT_STATUS_IS_OK(status)) {
    7604           0 :                 printf("TEST #3 close 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7605           0 :                 return False;
    7606             :         }
    7607             : 
    7608           5 :         status = cli_close(cli2, fnum2);
    7609           5 :         if (!NT_STATUS_IS_OK(status)) {
    7610           0 :                 printf("TEST #3 close 2 of %s failed (%s)\n", fname, nt_errstr(status));
    7611           0 :                 return False;
    7612             :         }
    7613             : 
    7614           5 :         printf("non-io open test #3 passed.\n");
    7615             : 
    7616           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7617             : 
    7618           5 :         printf("TEST #4 testing 2 non-io opens (both with delete)\n");
    7619             : 
    7620           5 :         status = cli_ntcreate(cli1, fname, 0,
    7621             :                                DELETE_ACCESS|FILE_READ_ATTRIBUTES,
    7622             :                                FILE_ATTRIBUTE_NORMAL, FILE_SHARE_NONE,
    7623             :                                FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    7624           5 :         if (!NT_STATUS_IS_OK(status)) {
    7625           0 :                 printf("TEST #4 open 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7626           0 :                 return False;
    7627             :         }
    7628             : 
    7629           5 :         status = cli_ntcreate(cli2, fname, 0,
    7630             :                               DELETE_ACCESS|FILE_READ_ATTRIBUTES,
    7631             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_NONE,
    7632             :                               FILE_OPEN_IF, 0, 0, &fnum2, NULL);
    7633           5 :         if (NT_STATUS_IS_OK(status)) {
    7634           0 :                 printf("TEST #4 open 2 of %s SUCCEEDED - should have failed (%s)\n", fname, nt_errstr(status));
    7635           0 :                 return False;
    7636             :         }
    7637             : 
    7638           5 :         printf("TEST #4 open 2 of %s gave %s (correct error should be %s)\n", fname, nt_errstr(status), "sharing violation");
    7639             : 
    7640           5 :         status = cli_close(cli1, fnum1);
    7641           5 :         if (!NT_STATUS_IS_OK(status)) {
    7642           0 :                 printf("TEST #4 close 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7643           0 :                 return False;
    7644             :         }
    7645             : 
    7646           5 :         printf("non-io open test #4 passed.\n");
    7647             : 
    7648           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7649             : 
    7650           5 :         printf("TEST #5 testing 2 non-io opens (both with delete - both with file share delete)\n");
    7651             : 
    7652           5 :         status = cli_ntcreate(cli1, fname, 0,
    7653             :                               DELETE_ACCESS|FILE_READ_ATTRIBUTES,
    7654             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_DELETE,
    7655             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    7656           5 :         if (!NT_STATUS_IS_OK(status)) {
    7657           0 :                 printf("TEST #5 open 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7658           0 :                 return False;
    7659             :         }
    7660             : 
    7661           5 :         status = cli_ntcreate(cli2, fname, 0,
    7662             :                               DELETE_ACCESS|FILE_READ_ATTRIBUTES,
    7663             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_DELETE,
    7664             :                               FILE_OPEN_IF, 0, 0, &fnum2, NULL);
    7665           5 :         if (!NT_STATUS_IS_OK(status)) {
    7666           0 :                 printf("TEST #5 open 2 of %s failed (%s)\n", fname, nt_errstr(status));
    7667           0 :                 return False;
    7668             :         }
    7669             : 
    7670           5 :         status = cli_close(cli1, fnum1);
    7671           5 :         if (!NT_STATUS_IS_OK(status)) {
    7672           0 :                 printf("TEST #5 close 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7673           0 :                 return False;
    7674             :         }
    7675             : 
    7676           5 :         status = cli_close(cli2, fnum2);
    7677           5 :         if (!NT_STATUS_IS_OK(status)) {
    7678           0 :                 printf("TEST #5 close 2 of %s failed (%s)\n", fname, nt_errstr(status));
    7679           0 :                 return False;
    7680             :         }
    7681             : 
    7682           5 :         printf("non-io open test #5 passed.\n");
    7683             : 
    7684           5 :         printf("TEST #6 testing 1 non-io open, one io open\n");
    7685             : 
    7686           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7687             : 
    7688           5 :         status = cli_ntcreate(cli1, fname, 0, FILE_READ_DATA,
    7689             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_NONE,
    7690             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    7691           5 :         if (!NT_STATUS_IS_OK(status)) {
    7692           0 :                 printf("TEST #6 open 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7693           0 :                 return False;
    7694             :         }
    7695             : 
    7696           5 :         status = cli_ntcreate(cli2, fname, 0, FILE_READ_ATTRIBUTES,
    7697             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_READ,
    7698             :                               FILE_OPEN_IF, 0, 0, &fnum2, NULL);
    7699           5 :         if (!NT_STATUS_IS_OK(status)) {
    7700           0 :                 printf("TEST #6 open 2 of %s failed (%s)\n", fname, nt_errstr(status));
    7701           0 :                 return False;
    7702             :         }
    7703             : 
    7704           5 :         status = cli_close(cli1, fnum1);
    7705           5 :         if (!NT_STATUS_IS_OK(status)) {
    7706           0 :                 printf("TEST #6 close 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7707           0 :                 return False;
    7708             :         }
    7709             : 
    7710           5 :         status = cli_close(cli2, fnum2);
    7711           5 :         if (!NT_STATUS_IS_OK(status)) {
    7712           0 :                 printf("TEST #6 close 2 of %s failed (%s)\n", fname, nt_errstr(status));
    7713           0 :                 return False;
    7714             :         }
    7715             : 
    7716           5 :         printf("non-io open test #6 passed.\n");
    7717             : 
    7718           5 :         printf("TEST #7 testing 1 non-io open, one io open with delete\n");
    7719             : 
    7720           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7721             : 
    7722           5 :         status = cli_ntcreate(cli1, fname, 0, FILE_READ_DATA,
    7723             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_NONE,
    7724             :                               FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    7725           5 :         if (!NT_STATUS_IS_OK(status)) {
    7726           0 :                 printf("TEST #7 open 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7727           0 :                 return False;
    7728             :         }
    7729             : 
    7730           5 :         status = cli_ntcreate(cli2, fname, 0,
    7731             :                               DELETE_ACCESS|FILE_READ_ATTRIBUTES,
    7732             :                               FILE_ATTRIBUTE_NORMAL,
    7733             :                               FILE_SHARE_READ|FILE_SHARE_DELETE,
    7734             :                               FILE_OPEN_IF, 0, 0, &fnum2, NULL);
    7735           5 :         if (NT_STATUS_IS_OK(status)) {
    7736           0 :                 printf("TEST #7 open 2 of %s SUCCEEDED - should have failed (%s)\n", fname, nt_errstr(status));
    7737           0 :                 return False;
    7738             :         }
    7739             : 
    7740           5 :         printf("TEST #7 open 2 of %s gave %s (correct error should be %s)\n", fname, nt_errstr(status), "sharing violation");
    7741             : 
    7742           5 :         status = cli_close(cli1, fnum1);
    7743           5 :         if (!NT_STATUS_IS_OK(status)) {
    7744           0 :                 printf("TEST #7 close 1 of %s failed (%s)\n", fname, nt_errstr(status));
    7745           0 :                 return False;
    7746             :         }
    7747             : 
    7748           5 :         printf("non-io open test #7 passed.\n");
    7749             : 
    7750           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    7751             : 
    7752           5 :         printf("TEST #8 testing open without WRITE_ATTRIBUTES, updating close write time.\n");
    7753           5 :         status = cli_ntcreate(cli1, fname, 0, FILE_WRITE_DATA, FILE_ATTRIBUTE_NORMAL,
    7754             :                                 FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    7755             :                                 FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    7756           5 :         if (!NT_STATUS_IS_OK(status)) {
    7757           0 :                 printf("TEST #8 open of %s failed (%s)\n", fname, nt_errstr(status));
    7758           0 :                 correct = false;
    7759           0 :                 goto out;
    7760             :         }
    7761             : 
    7762             :         /* Write to ensure we have to update the file time. */
    7763           5 :         status = cli_writeall(cli1, fnum1, 0, (const uint8_t *)"TEST DATA\n", 0, 10,
    7764             :                               NULL);
    7765           5 :         if (!NT_STATUS_IS_OK(status)) {
    7766           0 :                 printf("TEST #8 cli_write failed: %s\n", nt_errstr(status));
    7767           0 :                 correct = false;
    7768           0 :                 goto out;
    7769             :         }
    7770             : 
    7771           5 :         status = cli_close(cli1, fnum1);
    7772           5 :         if (!NT_STATUS_IS_OK(status)) {
    7773           0 :                 printf("TEST #8 close of %s failed (%s)\n", fname, nt_errstr(status));
    7774           0 :                 correct = false;
    7775             :         }
    7776             : 
    7777           5 :   out:
    7778             : 
    7779           5 :         if (!torture_close_connection(cli1)) {
    7780           0 :                 correct = False;
    7781             :         }
    7782           5 :         if (!torture_close_connection(cli2)) {
    7783           0 :                 correct = False;
    7784             :         }
    7785             : 
    7786           5 :         return correct;
    7787             : }
    7788             : 
    7789         102 : NTSTATUS torture_setup_unix_extensions(struct cli_state *cli)
    7790             : {
    7791           0 :         uint16_t major, minor;
    7792           0 :         uint32_t caplow, caphigh;
    7793           0 :         NTSTATUS status;
    7794             : 
    7795         102 :         if (!SERVER_HAS_UNIX_CIFS(cli)) {
    7796           0 :                 printf("Server doesn't support UNIX CIFS extensions.\n");
    7797           0 :                 return NT_STATUS_NOT_SUPPORTED;
    7798             :         }
    7799             : 
    7800         102 :         status = cli_unix_extensions_version(cli, &major, &minor, &caplow,
    7801             :                                              &caphigh);
    7802         102 :         if (!NT_STATUS_IS_OK(status)) {
    7803           0 :                 printf("Server didn't return UNIX CIFS extensions: %s\n",
    7804             :                        nt_errstr(status));
    7805           0 :                 return status;
    7806             :         }
    7807             : 
    7808         102 :         status = cli_set_unix_extensions_capabilities(cli, major, minor,
    7809             :                                                       caplow, caphigh);
    7810         102 :         if (!NT_STATUS_IS_OK(status)) {
    7811           0 :                 printf("Server doesn't support setting UNIX CIFS extensions: "
    7812             :                        "%s.\n", nt_errstr(status));
    7813           0 :                 return status;
    7814             :         }
    7815             : 
    7816         102 :         return NT_STATUS_OK;
    7817             : }
    7818             : 
    7819             : /*
    7820             :   Test POSIX open /mkdir calls.
    7821             :  */
    7822           4 : static bool run_simple_posix_open_test(int dummy)
    7823             : {
    7824           0 :         static struct cli_state *cli1;
    7825           4 :         const char *fname = "posix:file";
    7826           4 :         const char *hname = "posix:hlink";
    7827           4 :         const char *sname = "posix:symlink";
    7828           4 :         const char *dname = "posix:dir";
    7829           0 :         char buf[10];
    7830           4 :         char *target = NULL;
    7831           4 :         uint16_t fnum1 = (uint16_t)-1;
    7832           0 :         SMB_STRUCT_STAT sbuf;
    7833           4 :         bool correct = false;
    7834           0 :         NTSTATUS status;
    7835           0 :         size_t nread;
    7836           4 :         const char *fname_windows = "windows_file";
    7837           4 :         uint16_t fnum2 = (uint16_t)-1;
    7838           0 :         bool ok;
    7839             : 
    7840           4 :         printf("Starting simple POSIX open test\n");
    7841             : 
    7842           4 :         if (!torture_open_connection(&cli1, 0)) {
    7843           0 :                 return false;
    7844             :         }
    7845             : 
    7846           4 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    7847             : 
    7848           4 :         status = torture_setup_unix_extensions(cli1);
    7849           4 :         if (!NT_STATUS_IS_OK(status)) {
    7850           0 :                 return false;
    7851             :         }
    7852             : 
    7853           4 :         cli_setatr(cli1, fname, 0, 0);
    7854           4 :         cli_posix_unlink(cli1, fname);
    7855           4 :         cli_setatr(cli1, dname, 0, 0);
    7856           4 :         cli_posix_rmdir(cli1, dname);
    7857           4 :         cli_setatr(cli1, hname, 0, 0);
    7858           4 :         cli_posix_unlink(cli1, hname);
    7859           4 :         cli_setatr(cli1, sname, 0, 0);
    7860           4 :         cli_posix_unlink(cli1, sname);
    7861           4 :         cli_setatr(cli1, fname_windows, 0, 0);
    7862           4 :         cli_posix_unlink(cli1, fname_windows);
    7863             : 
    7864             :         /* Create a directory. */
    7865           4 :         status = cli_posix_mkdir(cli1, dname, 0777);
    7866           4 :         if (!NT_STATUS_IS_OK(status)) {
    7867           0 :                 printf("POSIX mkdir of %s failed (%s)\n", dname, nt_errstr(status));
    7868           0 :                 goto out;
    7869             :         }
    7870             : 
    7871           4 :         status = cli_posix_open(cli1, fname, O_RDWR|O_CREAT|O_EXCL,
    7872             :                                 0600, &fnum1);
    7873           4 :         if (!NT_STATUS_IS_OK(status)) {
    7874           0 :                 printf("POSIX create of %s failed (%s)\n", fname, nt_errstr(status));
    7875           0 :                 goto out;
    7876             :         }
    7877             : 
    7878             :         /* Test ftruncate - set file size. */
    7879           4 :         status = cli_ftruncate(cli1, fnum1, 1000);
    7880           4 :         if (!NT_STATUS_IS_OK(status)) {
    7881           0 :                 printf("ftruncate failed (%s)\n", nt_errstr(status));
    7882           0 :                 goto out;
    7883             :         }
    7884             : 
    7885             :         /* Ensure st_size == 1000 */
    7886           4 :         status = cli_posix_stat(cli1, fname, &sbuf);
    7887           4 :         if (!NT_STATUS_IS_OK(status)) {
    7888           0 :                 printf("stat failed (%s)\n", nt_errstr(status));
    7889           0 :                 goto out;
    7890             :         }
    7891             : 
    7892           4 :         if (sbuf.st_ex_size != 1000) {
    7893           0 :                 printf("ftruncate - stat size (%u) != 1000\n", (unsigned int)sbuf.st_ex_size);
    7894           0 :                 goto out;
    7895             :         }
    7896             : 
    7897             :         /* Ensure st_mode == 0600 */
    7898           4 :         if ((sbuf.st_ex_mode & 07777) != 0600) {
    7899           0 :                 printf("posix_open - bad permissions 0%o != 0600\n",
    7900           0 :                                 (unsigned int)(sbuf.st_ex_mode & 07777));
    7901           0 :                 goto out;
    7902             :         }
    7903             : 
    7904             :         /* Test ftruncate - set file size back to zero. */
    7905           4 :         status = cli_ftruncate(cli1, fnum1, 0);
    7906           4 :         if (!NT_STATUS_IS_OK(status)) {
    7907           0 :                 printf("ftruncate failed (%s)\n", nt_errstr(status));
    7908           0 :                 goto out;
    7909             :         }
    7910             : 
    7911           4 :         status = cli_close(cli1, fnum1);
    7912           4 :         if (!NT_STATUS_IS_OK(status)) {
    7913           0 :                 printf("close failed (%s)\n", nt_errstr(status));
    7914           0 :                 goto out;
    7915             :         }
    7916             : 
    7917             :         /* Now open the file again for read only. */
    7918           4 :         status = cli_posix_open(cli1, fname, O_RDONLY, 0, &fnum1);
    7919           4 :         if (!NT_STATUS_IS_OK(status)) {
    7920           0 :                 printf("POSIX open of %s failed (%s)\n", fname, nt_errstr(status));
    7921           0 :                 goto out;
    7922             :         }
    7923             : 
    7924             :         /* Now unlink while open. */
    7925           4 :         status = cli_posix_unlink(cli1, fname);
    7926           4 :         if (!NT_STATUS_IS_OK(status)) {
    7927           0 :                 printf("POSIX unlink of %s failed (%s)\n", fname, nt_errstr(status));
    7928           0 :                 goto out;
    7929             :         }
    7930             : 
    7931           4 :         status = cli_close(cli1, fnum1);
    7932           4 :         if (!NT_STATUS_IS_OK(status)) {
    7933           0 :                 printf("close(2) failed (%s)\n", nt_errstr(status));
    7934           0 :                 goto out;
    7935             :         }
    7936             : 
    7937             :         /* Ensure the file has gone. */
    7938           4 :         status = cli_posix_open(cli1, fname, O_RDONLY, 0, &fnum1);
    7939           4 :         if (NT_STATUS_IS_OK(status)) {
    7940           0 :                 printf("POSIX open of %s succeeded, should have been deleted.\n", fname);
    7941           0 :                 goto out;
    7942             :         }
    7943             : 
    7944             :         /* Create again to test open with O_TRUNC. */
    7945           4 :         status = cli_posix_open(cli1, fname, O_RDWR|O_CREAT|O_EXCL, 0600, &fnum1);
    7946           4 :         if (!NT_STATUS_IS_OK(status)) {
    7947           0 :                 printf("POSIX create of %s failed (%s)\n", fname, nt_errstr(status));
    7948           0 :                 goto out;
    7949             :         }
    7950             : 
    7951             :         /* Test ftruncate - set file size. */
    7952           4 :         status = cli_ftruncate(cli1, fnum1, 1000);
    7953           4 :         if (!NT_STATUS_IS_OK(status)) {
    7954           0 :                 printf("ftruncate failed (%s)\n", nt_errstr(status));
    7955           0 :                 goto out;
    7956             :         }
    7957             : 
    7958             :         /* Ensure st_size == 1000 */
    7959           4 :         status = cli_posix_stat(cli1, fname, &sbuf);
    7960           4 :         if (!NT_STATUS_IS_OK(status)) {
    7961           0 :                 printf("stat failed (%s)\n", nt_errstr(status));
    7962           0 :                 goto out;
    7963             :         }
    7964             : 
    7965           4 :         if (sbuf.st_ex_size != 1000) {
    7966           0 :                 printf("ftruncate - stat size (%u) != 1000\n", (unsigned int)sbuf.st_ex_size);
    7967           0 :                 goto out;
    7968             :         }
    7969             : 
    7970           4 :         status = cli_close(cli1, fnum1);
    7971           4 :         if (!NT_STATUS_IS_OK(status)) {
    7972           0 :                 printf("close(2) failed (%s)\n", nt_errstr(status));
    7973           0 :                 goto out;
    7974             :         }
    7975             : 
    7976             :         /* Re-open with O_TRUNC. */
    7977           4 :         status = cli_posix_open(cli1, fname, O_WRONLY|O_TRUNC, 0600, &fnum1);
    7978           4 :         if (!NT_STATUS_IS_OK(status)) {
    7979           0 :                 printf("POSIX create of %s failed (%s)\n", fname, nt_errstr(status));
    7980           0 :                 goto out;
    7981             :         }
    7982             : 
    7983             :         /* Ensure st_size == 0 */
    7984           4 :         status = cli_posix_stat(cli1, fname, &sbuf);
    7985           4 :         if (!NT_STATUS_IS_OK(status)) {
    7986           0 :                 printf("stat failed (%s)\n", nt_errstr(status));
    7987           0 :                 goto out;
    7988             :         }
    7989             : 
    7990           4 :         if (sbuf.st_ex_size != 0) {
    7991           0 :                 printf("O_TRUNC - stat size (%u) != 0\n", (unsigned int)sbuf.st_ex_size);
    7992           0 :                 goto out;
    7993             :         }
    7994             : 
    7995           4 :         status = cli_close(cli1, fnum1);
    7996           4 :         if (!NT_STATUS_IS_OK(status)) {
    7997           0 :                 printf("close failed (%s)\n", nt_errstr(status));
    7998           0 :                 goto out;
    7999             :         }
    8000             : 
    8001           4 :         status = cli_posix_unlink(cli1, fname);
    8002           4 :         if (!NT_STATUS_IS_OK(status)) {
    8003           0 :                 printf("POSIX unlink of %s failed (%s)\n", fname, nt_errstr(status));
    8004           0 :                 goto out;
    8005             :         }
    8006             : 
    8007           4 :         status = cli_posix_open(cli1, dname, O_RDONLY, 0, &fnum1);
    8008           4 :         if (!NT_STATUS_IS_OK(status)) {
    8009           0 :                 printf("POSIX open directory O_RDONLY of %s failed (%s)\n",
    8010             :                         dname, nt_errstr(status));
    8011           0 :                 goto out;
    8012             :         }
    8013             : 
    8014           4 :         cli_close(cli1, fnum1);
    8015             : 
    8016             :         /* What happens when we try and POSIX open a directory for write ? */
    8017           4 :         status = cli_posix_open(cli1, dname, O_RDWR, 0, &fnum1);
    8018           4 :         if (NT_STATUS_IS_OK(status)) {
    8019           0 :                 printf("POSIX open of directory %s succeeded, "
    8020             :                        "should have failed.\n",
    8021             :                        dname);
    8022           0 :                 goto out;
    8023             :         } else {
    8024           4 :                 if (!check_both_error(__LINE__, status, ERRDOS, EISDIR,
    8025           4 :                                 NT_STATUS_FILE_IS_A_DIRECTORY)) {
    8026           0 :                         goto out;
    8027             :                 }
    8028             :         }
    8029             : 
    8030             :         /* Create the file. */
    8031           4 :         status = cli_posix_open(cli1, fname, O_RDWR|O_CREAT|O_EXCL,
    8032             :                                 0600, &fnum1);
    8033           4 :         if (!NT_STATUS_IS_OK(status)) {
    8034           0 :                 printf("POSIX create of %s failed (%s)\n", fname, nt_errstr(status));
    8035           0 :                 goto out;
    8036             :         }
    8037             : 
    8038             :         /* Write some data into it. */
    8039           4 :         status = cli_writeall(cli1, fnum1, 0, (const uint8_t *)"TEST DATA\n", 0, 10,
    8040             :                               NULL);
    8041           4 :         if (!NT_STATUS_IS_OK(status)) {
    8042           0 :                 printf("cli_write failed: %s\n", nt_errstr(status));
    8043           0 :                 goto out;
    8044             :         }
    8045             : 
    8046           4 :         cli_close(cli1, fnum1);
    8047             : 
    8048             :         /* Now create a hardlink. */
    8049           4 :         status = cli_posix_hardlink(cli1, fname, hname);
    8050           4 :         if (!NT_STATUS_IS_OK(status)) {
    8051           0 :                 printf("POSIX hardlink of %s failed (%s)\n", hname, nt_errstr(status));
    8052           0 :                 goto out;
    8053             :         }
    8054             : 
    8055             :         /* Now create a symlink. */
    8056           4 :         status = cli_posix_symlink(cli1, fname, sname);
    8057           4 :         if (!NT_STATUS_IS_OK(status)) {
    8058           0 :                 printf("POSIX symlink of %s failed (%s)\n", sname, nt_errstr(status));
    8059           0 :                 goto out;
    8060             :         }
    8061             : 
    8062             :         /* Open the hardlink for read. */
    8063           4 :         status = cli_posix_open(cli1, hname, O_RDONLY, 0, &fnum1);
    8064           4 :         if (!NT_STATUS_IS_OK(status)) {
    8065           0 :                 printf("POSIX open of %s failed (%s)\n", hname, nt_errstr(status));
    8066           0 :                 goto out;
    8067             :         }
    8068             : 
    8069           4 :         status = cli_read(cli1, fnum1, buf, 0, 10, &nread);
    8070           4 :         if (!NT_STATUS_IS_OK(status)) {
    8071           0 :                 printf("POSIX read of %s failed (%s)\n", hname,
    8072             :                        nt_errstr(status));
    8073           0 :                 goto out;
    8074           4 :         } else if (nread != 10) {
    8075           0 :                 printf("POSIX read of %s failed. Received %ld, expected %d\n",
    8076             :                        hname, (unsigned long)nread, 10);
    8077           0 :                 goto out;
    8078             :         }
    8079             : 
    8080           4 :         if (memcmp(buf, "TEST DATA\n", 10)) {
    8081           0 :                 printf("invalid data read from hardlink\n");
    8082           0 :                 goto out;
    8083             :         }
    8084             : 
    8085             :         /* Do a POSIX lock/unlock. */
    8086           4 :         status = cli_posix_lock(cli1, fnum1, 0, 100, true, READ_LOCK);
    8087           4 :         if (!NT_STATUS_IS_OK(status)) {
    8088           0 :                 printf("POSIX lock failed %s\n", nt_errstr(status));
    8089           0 :                 goto out;
    8090             :         }
    8091             : 
    8092             :         /* Punch a hole in the locked area. */
    8093           4 :         status = cli_posix_unlock(cli1, fnum1, 10, 80);
    8094           4 :         if (!NT_STATUS_IS_OK(status)) {
    8095           0 :                 printf("POSIX unlock failed %s\n", nt_errstr(status));
    8096           0 :                 goto out;
    8097             :         }
    8098             : 
    8099           4 :         cli_close(cli1, fnum1);
    8100             : 
    8101             :         /* Open the symlink for read - this should fail. A POSIX
    8102             :            client should not be doing opens on a symlink. */
    8103           4 :         status = cli_posix_open(cli1, sname, O_RDONLY, 0, &fnum1);
    8104           4 :         if (NT_STATUS_IS_OK(status)) {
    8105           0 :                 printf("POSIX open of %s succeeded (should have failed)\n", sname);
    8106           0 :                 goto out;
    8107             :         }
    8108           4 :         ok = check_both_error(
    8109             :                 __LINE__, status, ERRDOS, ERRbadpath,
    8110           4 :                 NT_STATUS_OBJECT_NAME_NOT_FOUND);
    8111           4 :         if (!ok) {
    8112           0 :                 printf("POSIX open of %s should have failed "
    8113             :                        "with NT_STATUS_OBJECT_NAME_NOT_FOUND, "
    8114             :                        "failed with %s instead.\n",
    8115             :                        sname, nt_errstr(status));
    8116           0 :                 goto out;
    8117             :         }
    8118             : 
    8119           4 :         status = cli_readlink(cli1, sname, talloc_tos(), &target, NULL, NULL);
    8120           4 :         if (!NT_STATUS_IS_OK(status)) {
    8121           0 :                 printf("POSIX readlink on %s failed (%s)\n", sname, nt_errstr(status));
    8122           0 :                 goto out;
    8123             :         }
    8124             : 
    8125           4 :         if (strcmp(target, fname) != 0) {
    8126           0 :                 printf("POSIX readlink on %s failed to match name %s (read %s)\n",
    8127             :                         sname, fname, target);
    8128           0 :                 goto out;
    8129             :         }
    8130             : 
    8131           4 :         status = cli_posix_rmdir(cli1, dname);
    8132           4 :         if (!NT_STATUS_IS_OK(status)) {
    8133           0 :                 printf("POSIX rmdir failed (%s)\n", nt_errstr(status));
    8134           0 :                 goto out;
    8135             :         }
    8136             : 
    8137             :         /* Check directory opens with a specific permission. */
    8138           4 :         status = cli_posix_mkdir(cli1, dname, 0700);
    8139           4 :         if (!NT_STATUS_IS_OK(status)) {
    8140           0 :                 printf("POSIX mkdir of %s failed (%s)\n", dname, nt_errstr(status));
    8141           0 :                 goto out;
    8142             :         }
    8143             : 
    8144             :         /* Ensure st_mode == 0700 */
    8145           4 :         status = cli_posix_stat(cli1, dname, &sbuf);
    8146           4 :         if (!NT_STATUS_IS_OK(status)) {
    8147           0 :                 printf("stat failed (%s)\n", nt_errstr(status));
    8148           0 :                 goto out;
    8149             :         }
    8150             : 
    8151           4 :         if ((sbuf.st_ex_mode & 07777) != 0700) {
    8152           0 :                 printf("posix_mkdir - bad permissions 0%o != 0700\n",
    8153           0 :                                 (unsigned int)(sbuf.st_ex_mode & 07777));
    8154           0 :                 goto out;
    8155             :         }
    8156             : 
    8157             :         /*
    8158             :          * Now create a Windows file, and attempt a POSIX unlink.
    8159             :          * This should fail with a sharing violation but due to:
    8160             :          *
    8161             :          * [Bug 9571] Unlink after open causes smbd to panic
    8162             :          *
    8163             :          * ensure we've fixed the lock ordering violation.
    8164             :          */
    8165             : 
    8166           4 :         status = cli_ntcreate(cli1, fname_windows, 0,
    8167             :                         FILE_READ_DATA|FILE_WRITE_DATA, 0,
    8168             :                         FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    8169             :                         FILE_CREATE,
    8170             :                         0x0, 0x0, &fnum2, NULL);
    8171           4 :         if (!NT_STATUS_IS_OK(status)) {
    8172           0 :                 printf("Windows create of %s failed (%s)\n", fname_windows,
    8173             :                         nt_errstr(status));
    8174           0 :                 goto out;
    8175             :         }
    8176             : 
    8177             :         /* Now try posix_unlink. */
    8178           4 :         status = cli_posix_unlink(cli1, fname_windows);
    8179           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_SHARING_VIOLATION)) {
    8180           0 :                 printf("POSIX unlink of %s should fail "
    8181             :                         "with NT_STATUS_SHARING_VIOLATION "
    8182             :                         "got %s instead !\n",
    8183             :                         fname_windows,
    8184             :                         nt_errstr(status));
    8185           0 :                 goto out;
    8186             :         }
    8187             : 
    8188           4 :         cli_close(cli1, fnum2);
    8189             : 
    8190           4 :         printf("Simple POSIX open test passed\n");
    8191           4 :         correct = true;
    8192             : 
    8193           4 :   out:
    8194             : 
    8195           4 :         if (fnum1 != (uint16_t)-1) {
    8196           4 :                 cli_close(cli1, fnum1);
    8197           4 :                 fnum1 = (uint16_t)-1;
    8198             :         }
    8199             : 
    8200           4 :         if (fnum2 != (uint16_t)-1) {
    8201           4 :                 cli_close(cli1, fnum2);
    8202           4 :                 fnum2 = (uint16_t)-1;
    8203             :         }
    8204             : 
    8205           4 :         cli_setatr(cli1, sname, 0, 0);
    8206           4 :         cli_posix_unlink(cli1, sname);
    8207           4 :         cli_setatr(cli1, hname, 0, 0);
    8208           4 :         cli_posix_unlink(cli1, hname);
    8209           4 :         cli_setatr(cli1, fname, 0, 0);
    8210           4 :         cli_posix_unlink(cli1, fname);
    8211           4 :         cli_setatr(cli1, dname, 0, 0);
    8212           4 :         cli_posix_rmdir(cli1, dname);
    8213           4 :         cli_setatr(cli1, fname_windows, 0, 0);
    8214           4 :         cli_posix_unlink(cli1, fname_windows);
    8215             : 
    8216           4 :         if (!torture_close_connection(cli1)) {
    8217           0 :                 correct = false;
    8218             :         }
    8219             : 
    8220           4 :         return correct;
    8221             : }
    8222             : 
    8223             : /*
    8224             :   Test POSIX and Windows ACLs are rejected on symlinks.
    8225             :  */
    8226           4 : static bool run_acl_symlink_test(int dummy)
    8227             : {
    8228           0 :         static struct cli_state *cli;
    8229           4 :         const char *fname = "posix_file";
    8230           4 :         const char *sname = "posix_symlink";
    8231           4 :         uint16_t fnum = (uint16_t)-1;
    8232           4 :         bool correct = false;
    8233           0 :         NTSTATUS status;
    8234           4 :         char *posix_acl = NULL;
    8235           4 :         size_t posix_acl_len = 0;
    8236           4 :         char *posix_acl_sym = NULL;
    8237           4 :         size_t posix_acl_len_sym = 0;
    8238           4 :         struct security_descriptor *sd = NULL;
    8239           4 :         TALLOC_CTX *frame = NULL;
    8240             : 
    8241           4 :         frame = talloc_stackframe();
    8242             : 
    8243           4 :         printf("Starting acl symlink test\n");
    8244             : 
    8245           4 :         if (!torture_open_connection(&cli, 0)) {
    8246           0 :                 TALLOC_FREE(frame);
    8247           0 :                 return false;
    8248             :         }
    8249             : 
    8250           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    8251             : 
    8252           4 :         status = torture_setup_unix_extensions(cli);
    8253           4 :         if (!NT_STATUS_IS_OK(status)) {
    8254           0 :                 TALLOC_FREE(frame);
    8255           0 :                 return false;
    8256             :         }
    8257             : 
    8258           4 :         cli_setatr(cli, fname, 0, 0);
    8259           4 :         cli_posix_unlink(cli, fname);
    8260           4 :         cli_setatr(cli, sname, 0, 0);
    8261           4 :         cli_posix_unlink(cli, sname);
    8262             : 
    8263           4 :         status = cli_ntcreate(cli,
    8264             :                         fname,
    8265             :                         0,
    8266             :                         READ_CONTROL_ACCESS,
    8267             :                         0,
    8268             :                         FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    8269             :                         FILE_CREATE,
    8270             :                         0x0,
    8271             :                         0x0,
    8272             :                         &fnum,
    8273             :                         NULL);
    8274             : 
    8275           4 :         if (!NT_STATUS_IS_OK(status)) {
    8276           0 :                 printf("cli_ntcreate of %s failed (%s)\n",
    8277             :                         fname,
    8278             :                         nt_errstr(status));
    8279           0 :                 goto out;
    8280             :         }
    8281             : 
    8282             :         /* Get the Windows ACL on the file. */
    8283           4 :         status = cli_query_secdesc(cli,
    8284             :                                 fnum,
    8285             :                                 frame,
    8286             :                                 &sd);
    8287           4 :         if (!NT_STATUS_IS_OK(status)) {
    8288           0 :                 printf("cli_query_secdesc failed (%s)\n",
    8289             :                         nt_errstr(status));
    8290           0 :                 goto out;
    8291             :         }
    8292             : 
    8293             :         /* Get the POSIX ACL on the file. */
    8294           4 :         status = cli_posix_getacl(cli,
    8295             :                                 fname,
    8296             :                                 frame,
    8297             :                                 &posix_acl_len,
    8298             :                                 &posix_acl);
    8299             : 
    8300           4 :         if (!NT_STATUS_IS_OK(status)) {
    8301           0 :                 printf("cli_posix_getacl failed (%s)\n",
    8302             :                         nt_errstr(status));
    8303           0 :                 goto out;
    8304             :         }
    8305             : 
    8306           4 :         status = cli_close(cli, fnum);
    8307           4 :         if (!NT_STATUS_IS_OK(status)) {
    8308           0 :                 printf("close failed (%s)\n", nt_errstr(status));
    8309           0 :                 goto out;
    8310             :         }
    8311           4 :         fnum = (uint16_t)-1;
    8312             : 
    8313             :         /* Now create a symlink. */
    8314           4 :         status = cli_posix_symlink(cli, fname, sname);
    8315           4 :         if (!NT_STATUS_IS_OK(status)) {
    8316           0 :                 printf("cli_posix_symlink of %s -> %s failed (%s)\n",
    8317             :                         sname,
    8318             :                         fname,
    8319             :                         nt_errstr(status));
    8320           0 :                 goto out;
    8321             :         }
    8322             : 
    8323             :         /* Open a handle on the symlink for SD set/get should fail. */
    8324           4 :         status = cli_ntcreate(cli,
    8325             :                         sname,
    8326             :                         0,
    8327             :                         READ_CONTROL_ACCESS|SEC_STD_WRITE_DAC,
    8328             :                         0,
    8329             :                         FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    8330             :                         FILE_OPEN,
    8331             :                         0x0,
    8332             :                         0x0,
    8333             :                         &fnum,
    8334             :                         NULL);
    8335             : 
    8336           4 :         if (NT_STATUS_IS_OK(status)) {
    8337           0 :                 printf("Symlink open for getsd/setsd of %s "
    8338             :                         "succeeded (should fail)\n",
    8339             :                         sname);
    8340           0 :                 goto out;
    8341             :         }
    8342             : 
    8343             :         /* Try a stat-open on the symlink, should also fail. */
    8344           4 :         status = cli_ntcreate(cli,
    8345             :                         sname,
    8346             :                         0,
    8347             :                         FILE_READ_ATTRIBUTES|FILE_WRITE_ATTRIBUTES,
    8348             :                         0,
    8349             :                         FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    8350             :                         FILE_OPEN,
    8351             :                         0x0,
    8352             :                         0x0,
    8353             :                         &fnum,
    8354             :                         NULL);
    8355             : 
    8356           4 :         if (NT_STATUS_IS_OK(status)) {
    8357           0 :                 printf("Stat-open of symlink succeeded (should fail)\n");
    8358           0 :                 goto out;
    8359             :         }
    8360             : 
    8361             :         /* Get the POSIX ACL on the symlink pathname. Should fail. */
    8362           4 :         status = cli_posix_getacl(cli,
    8363             :                                 sname,
    8364             :                                 frame,
    8365             :                                 &posix_acl_len_sym,
    8366             :                                 &posix_acl_sym);
    8367             : 
    8368           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_ACCESS_DENIED)) {
    8369           0 :                 printf("cli_posix_getacl on a symlink gave %s. "
    8370             :                         "Should be NT_STATUS_ACCESS_DENIED.\n",
    8371             :                         nt_errstr(status));
    8372           0 :                 goto out;
    8373             :         }
    8374             : 
    8375             :         /* Set the POSIX ACL on the symlink pathname. Should fail. */
    8376           4 :         status = cli_posix_setacl(cli,
    8377             :                                 sname,
    8378             :                                 posix_acl,
    8379             :                                 posix_acl_len);
    8380             : 
    8381           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_ACCESS_DENIED)) {
    8382           0 :                 printf("cli_posix_setacl on a symlink gave %s. "
    8383             :                         "Should be NT_STATUS_ACCESS_DENIED.\n",
    8384             :                         nt_errstr(status));
    8385           0 :                 goto out;
    8386             :         }
    8387             : 
    8388           4 :         printf("ACL symlink test passed\n");
    8389           4 :         correct = true;
    8390             : 
    8391           4 :   out:
    8392             : 
    8393           4 :         if (fnum != (uint16_t)-1) {
    8394           0 :                 cli_close(cli, fnum);
    8395           0 :                 fnum = (uint16_t)-1;
    8396             :         }
    8397             : 
    8398           4 :         cli_setatr(cli, sname, 0, 0);
    8399           4 :         cli_posix_unlink(cli, sname);
    8400           4 :         cli_setatr(cli, fname, 0, 0);
    8401           4 :         cli_posix_unlink(cli, fname);
    8402             : 
    8403           4 :         if (!torture_close_connection(cli)) {
    8404           0 :                 correct = false;
    8405             :         }
    8406             : 
    8407           4 :         TALLOC_FREE(frame);
    8408           4 :         return correct;
    8409             : }
    8410             : 
    8411             : /*
    8412             :   Test POSIX can delete a file containing streams.
    8413             :  */
    8414           4 : static bool run_posix_stream_delete(int dummy)
    8415             : {
    8416           4 :         struct cli_state *cli1 = NULL;
    8417           4 :         struct cli_state *cli2 = NULL;
    8418           4 :         const char *fname = "streamfile";
    8419           4 :         const char *stream_fname = "streamfile:Zone.Identifier:$DATA";
    8420           4 :         uint16_t fnum1 = (uint16_t)-1;
    8421           4 :         bool correct = false;
    8422           0 :         NTSTATUS status;
    8423           4 :         TALLOC_CTX *frame = NULL;
    8424             : 
    8425           4 :         frame = talloc_stackframe();
    8426             : 
    8427           4 :         printf("Starting POSIX stream delete test\n");
    8428             : 
    8429           4 :         if (!torture_open_connection(&cli1, 0) ||
    8430           4 :                         !torture_open_connection(&cli2, 1)) {
    8431           0 :                 TALLOC_FREE(frame);
    8432           0 :                 return false;
    8433             :         }
    8434             : 
    8435           4 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    8436           4 :         smbXcli_conn_set_sockopt(cli2->conn, sockops);
    8437             : 
    8438           4 :         status = torture_setup_unix_extensions(cli2);
    8439           4 :         if (!NT_STATUS_IS_OK(status)) {
    8440           0 :                 goto out;
    8441             :         }
    8442             : 
    8443           4 :         cli_setatr(cli1, fname, 0, 0);
    8444           4 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    8445             : 
    8446             :         /* Create the file. */
    8447           4 :         status = cli_ntcreate(cli1,
    8448             :                         fname,
    8449             :                         0,
    8450             :                         READ_CONTROL_ACCESS,
    8451             :                         0,
    8452             :                         FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    8453             :                         FILE_CREATE,
    8454             :                         0x0,
    8455             :                         0x0,
    8456             :                         &fnum1,
    8457             :                         NULL);
    8458             : 
    8459           4 :         if (!NT_STATUS_IS_OK(status)) {
    8460           0 :                 printf("cli_ntcreate of %s failed (%s)\n",
    8461             :                         fname,
    8462             :                         nt_errstr(status));
    8463           0 :                 goto out;
    8464             :         }
    8465             : 
    8466           4 :         status = cli_close(cli1, fnum1);
    8467           4 :         if (!NT_STATUS_IS_OK(status)) {
    8468           0 :                 printf("cli_close of %s failed (%s)\n",
    8469             :                         fname,
    8470             :                         nt_errstr(status));
    8471           0 :                 goto out;
    8472             :         }
    8473           4 :         fnum1 = (uint16_t)-1;
    8474             : 
    8475             :         /* Now create the stream. */
    8476           4 :         status = cli_ntcreate(cli1,
    8477             :                         stream_fname,
    8478             :                         0,
    8479             :                         FILE_WRITE_DATA,
    8480             :                         0,
    8481             :                         FILE_SHARE_READ|FILE_SHARE_WRITE,
    8482             :                         FILE_CREATE,
    8483             :                         0x0,
    8484             :                         0x0,
    8485             :                         &fnum1,
    8486             :                         NULL);
    8487             : 
    8488           4 :         if (!NT_STATUS_IS_OK(status)) {
    8489           0 :                 printf("cli_ntcreate of %s failed (%s)\n",
    8490             :                         stream_fname,
    8491             :                         nt_errstr(status));
    8492           0 :                 goto out;
    8493             :         }
    8494             : 
    8495             :         /* Leave the stream handle open... */
    8496             : 
    8497             :         /* POSIX unlink should fail. */
    8498           4 :         status = cli_posix_unlink(cli2, fname);
    8499           4 :         if (NT_STATUS_IS_OK(status)) {
    8500           0 :                 printf("cli_posix_unlink of %s succeeded, should have failed\n",
    8501             :                         fname);
    8502           0 :                 goto out;
    8503             :         }
    8504             : 
    8505           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_SHARING_VIOLATION)) {
    8506           0 :                 printf("cli_posix_unlink of %s failed with (%s) "
    8507             :                         "should have been NT_STATUS_SHARING_VIOLATION\n",
    8508             :                         fname,
    8509             :                         nt_errstr(status));
    8510           0 :                 goto out;
    8511             :         }
    8512             : 
    8513             :         /* Close the stream handle. */
    8514           4 :         status = cli_close(cli1, fnum1);
    8515           4 :         if (!NT_STATUS_IS_OK(status)) {
    8516           0 :                 printf("cli_close of %s failed (%s)\n",
    8517             :                         stream_fname,
    8518             :                         nt_errstr(status));
    8519           0 :                 goto out;
    8520             :         }
    8521           4 :         fnum1 = (uint16_t)-1;
    8522             : 
    8523             :         /* POSIX unlink after stream handle closed should succeed. */
    8524           4 :         status = cli_posix_unlink(cli2, fname);
    8525           4 :         if (!NT_STATUS_IS_OK(status)) {
    8526           0 :                 printf("cli_posix_unlink of %s failed (%s)\n",
    8527             :                         fname,
    8528             :                         nt_errstr(status));
    8529           0 :                 goto out;
    8530             :         }
    8531             : 
    8532           4 :         printf("POSIX stream delete test passed\n");
    8533           4 :         correct = true;
    8534             : 
    8535           4 :   out:
    8536             : 
    8537           4 :         if (fnum1 != (uint16_t)-1) {
    8538           0 :                 cli_close(cli1, fnum1);
    8539           0 :                 fnum1 = (uint16_t)-1;
    8540             :         }
    8541             : 
    8542           4 :         cli_setatr(cli1, fname, 0, 0);
    8543           4 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    8544             : 
    8545           4 :         if (!torture_close_connection(cli1)) {
    8546           0 :                 correct = false;
    8547             :         }
    8548           4 :         if (!torture_close_connection(cli2)) {
    8549           0 :                 correct = false;
    8550             :         }
    8551             : 
    8552           4 :         TALLOC_FREE(frame);
    8553           4 :         return correct;
    8554             : }
    8555             : 
    8556             : /*
    8557             :   Test setting EA's are rejected on symlinks.
    8558             :  */
    8559           4 : static bool run_ea_symlink_test(int dummy)
    8560             : {
    8561           0 :         static struct cli_state *cli;
    8562           4 :         const char *fname = "posix_file_ea";
    8563           4 :         const char *sname = "posix_symlink_ea";
    8564           4 :         const char *ea_name = "testea_name";
    8565           4 :         const char *ea_value = "testea_value";
    8566           4 :         uint16_t fnum = (uint16_t)-1;
    8567           4 :         bool correct = false;
    8568           0 :         NTSTATUS status;
    8569           0 :         size_t i, num_eas;
    8570           4 :         struct ea_struct *eas = NULL;
    8571           4 :         TALLOC_CTX *frame = NULL;
    8572             : 
    8573           4 :         frame = talloc_stackframe();
    8574             : 
    8575           4 :         printf("Starting EA symlink test\n");
    8576             : 
    8577           4 :         if (!torture_open_connection(&cli, 0)) {
    8578           0 :                 TALLOC_FREE(frame);
    8579           0 :                 return false;
    8580             :         }
    8581             : 
    8582           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    8583             : 
    8584           4 :         status = torture_setup_unix_extensions(cli);
    8585           4 :         if (!NT_STATUS_IS_OK(status)) {
    8586           0 :                 TALLOC_FREE(frame);
    8587           0 :                 return false;
    8588             :         }
    8589             : 
    8590           4 :         cli_setatr(cli, fname, 0, 0);
    8591           4 :         cli_posix_unlink(cli, fname);
    8592           4 :         cli_setatr(cli, sname, 0, 0);
    8593           4 :         cli_posix_unlink(cli, sname);
    8594             : 
    8595           4 :         status = cli_ntcreate(cli,
    8596             :                         fname,
    8597             :                         0,
    8598             :                         READ_CONTROL_ACCESS,
    8599             :                         0,
    8600             :                         FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    8601             :                         FILE_CREATE,
    8602             :                         0x0,
    8603             :                         0x0,
    8604             :                         &fnum,
    8605             :                         NULL);
    8606             : 
    8607           4 :         if (!NT_STATUS_IS_OK(status)) {
    8608           0 :                 printf("cli_ntcreate of %s failed (%s)\n",
    8609             :                         fname,
    8610             :                         nt_errstr(status));
    8611           0 :                 goto out;
    8612             :         }
    8613             : 
    8614           4 :         status = cli_close(cli, fnum);
    8615           4 :         if (!NT_STATUS_IS_OK(status)) {
    8616           0 :                 printf("close failed (%s)\n",
    8617             :                         nt_errstr(status));
    8618           0 :                 goto out;
    8619             :         }
    8620           4 :         fnum = (uint16_t)-1;
    8621             : 
    8622             :         /* Set an EA on the path. */
    8623           4 :         status = cli_set_ea_path(cli,
    8624             :                                 fname,
    8625             :                                 ea_name,
    8626             :                                 ea_value,
    8627           4 :                                 strlen(ea_value)+1);
    8628             : 
    8629           4 :         if (!NT_STATUS_IS_OK(status)) {
    8630           0 :                 printf("cli_set_ea_path failed (%s)\n",
    8631             :                         nt_errstr(status));
    8632           0 :                 goto out;
    8633             :         }
    8634             : 
    8635             :         /* Now create a symlink. */
    8636           4 :         status = cli_posix_symlink(cli, fname, sname);
    8637           4 :         if (!NT_STATUS_IS_OK(status)) {
    8638           0 :                 printf("cli_posix_symlink of %s -> %s failed (%s)\n",
    8639             :                         sname,
    8640             :                         fname,
    8641             :                         nt_errstr(status));
    8642           0 :                 goto out;
    8643             :         }
    8644             : 
    8645             :         /* Get the EA list on the path. Should return value set. */
    8646           4 :         status = cli_get_ea_list_path(cli,
    8647             :                                 fname,
    8648             :                                 frame,
    8649             :                                 &num_eas,
    8650             :                                 &eas);
    8651             : 
    8652           4 :         if (!NT_STATUS_IS_OK(status)) {
    8653           0 :                 printf("cli_get_ea_list_path failed (%s)\n",
    8654             :                         nt_errstr(status));
    8655           0 :                 goto out;
    8656             :         }
    8657             : 
    8658             :         /* Ensure the EA we set is there. */
    8659           4 :         for (i=0; i<num_eas; i++) {
    8660           4 :                 if (strcmp(eas[i].name, ea_name) == 0 &&
    8661           4 :                                 eas[i].value.length == strlen(ea_value)+1 &&
    8662           4 :                                 memcmp(eas[i].value.data,
    8663             :                                         ea_value,
    8664           4 :                                         eas[i].value.length) == 0) {
    8665           4 :                         break;
    8666             :                 }
    8667             :         }
    8668             : 
    8669           4 :         if (i == num_eas) {
    8670           0 :                 printf("Didn't find EA on pathname %s\n",
    8671             :                         fname);
    8672           0 :                 goto out;
    8673             :         }
    8674             : 
    8675           4 :         num_eas = 0;
    8676           4 :         TALLOC_FREE(eas);
    8677             : 
    8678             :         /* Get the EA list on the symlink. Should return empty list. */
    8679           4 :         status = cli_get_ea_list_path(cli,
    8680             :                                 sname,
    8681             :                                 frame,
    8682             :                                 &num_eas,
    8683             :                                 &eas);
    8684             : 
    8685           4 :         if (!NT_STATUS_IS_OK(status)) {
    8686           0 :                 printf("cli_get_ea_list_path failed (%s)\n",
    8687             :                         nt_errstr(status));
    8688           0 :                 goto out;
    8689             :         }
    8690             : 
    8691           4 :         if (num_eas != 0) {
    8692           0 :                 printf("cli_get_ea_list_path failed (%s)\n",
    8693             :                         nt_errstr(status));
    8694           0 :                 goto out;
    8695             :         }
    8696             : 
    8697             :         /* Set an EA on the symlink. Should fail. */
    8698           4 :         status = cli_set_ea_path(cli,
    8699             :                                 sname,
    8700             :                                 ea_name,
    8701             :                                 ea_value,
    8702           4 :                                 strlen(ea_value)+1);
    8703             : 
    8704           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_ACCESS_DENIED)) {
    8705           0 :                 printf("cli_set_ea_path on a symlink gave %s. "
    8706             :                         "Should be NT_STATUS_ACCESS_DENIED.\n",
    8707             :                         nt_errstr(status));
    8708           0 :                 goto out;
    8709             :         }
    8710             : 
    8711           4 :         printf("EA symlink test passed\n");
    8712           4 :         correct = true;
    8713             : 
    8714           4 :   out:
    8715             : 
    8716           4 :         if (fnum != (uint16_t)-1) {
    8717           0 :                 cli_close(cli, fnum);
    8718           0 :                 fnum = (uint16_t)-1;
    8719             :         }
    8720             : 
    8721           4 :         cli_setatr(cli, sname, 0, 0);
    8722           4 :         cli_posix_unlink(cli, sname);
    8723           4 :         cli_setatr(cli, fname, 0, 0);
    8724           4 :         cli_posix_unlink(cli, fname);
    8725             : 
    8726           4 :         if (!torture_close_connection(cli)) {
    8727           0 :                 correct = false;
    8728             :         }
    8729             : 
    8730           4 :         TALLOC_FREE(frame);
    8731           4 :         return correct;
    8732             : }
    8733             : 
    8734             : /*
    8735             :   Test POSIX locks are OFD-locks.
    8736             :  */
    8737           4 : static bool run_posix_ofd_lock_test(int dummy)
    8738             : {
    8739           0 :         static struct cli_state *cli;
    8740           4 :         const char *fname = "posix_file";
    8741           4 :         uint16_t fnum1 = (uint16_t)-1;
    8742           4 :         uint16_t fnum2 = (uint16_t)-1;
    8743           4 :         bool correct = false;
    8744           0 :         NTSTATUS status;
    8745           4 :         TALLOC_CTX *frame = NULL;
    8746             : 
    8747           4 :         frame = talloc_stackframe();
    8748             : 
    8749           4 :         printf("Starting POSIX ofd-lock test\n");
    8750             : 
    8751           4 :         if (!torture_open_connection(&cli, 0)) {
    8752           0 :                 TALLOC_FREE(frame);
    8753           0 :                 return false;
    8754             :         }
    8755             : 
    8756           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    8757             : 
    8758           4 :         status = torture_setup_unix_extensions(cli);
    8759           4 :         if (!NT_STATUS_IS_OK(status)) {
    8760           0 :                 TALLOC_FREE(frame);
    8761           0 :                 return false;
    8762             :         }
    8763             : 
    8764           4 :         cli_setatr(cli, fname, 0, 0);
    8765           4 :         cli_posix_unlink(cli, fname);
    8766             : 
    8767             :         /* Open the file twice. */
    8768           4 :         status = cli_posix_open(cli, fname, O_RDWR|O_CREAT|O_EXCL,
    8769             :                                 0600, &fnum1);
    8770           4 :         if (!NT_STATUS_IS_OK(status)) {
    8771           0 :                 printf("First POSIX open of %s failed\n", fname);
    8772           0 :                 goto out;
    8773             :         }
    8774             : 
    8775           4 :         status = cli_posix_open(cli, fname, O_RDWR, 0, &fnum2);
    8776           4 :         if (!NT_STATUS_IS_OK(status)) {
    8777           0 :                 printf("First POSIX open of %s failed\n", fname);
    8778           0 :                 goto out;
    8779             :         }
    8780             : 
    8781             :         /* Set a 0-50 lock on fnum1. */
    8782           4 :         status = cli_posix_lock(cli, fnum1, 0, 50, false, WRITE_LOCK);
    8783           4 :         if (!NT_STATUS_IS_OK(status)) {
    8784           0 :                 printf("POSIX lock (1) failed %s\n", nt_errstr(status));
    8785           0 :                 goto out;
    8786             :         }
    8787             : 
    8788             :         /* Set a 60-100 lock on fnum2. */
    8789           4 :         status = cli_posix_lock(cli, fnum2, 60, 100, false, WRITE_LOCK);
    8790           4 :         if (!NT_STATUS_IS_OK(status)) {
    8791           0 :                 printf("POSIX lock (2) failed %s\n", nt_errstr(status));
    8792           0 :                 goto out;
    8793             :         }
    8794             : 
    8795             :         /* close fnum1 - 0-50 lock should go away. */
    8796           4 :         status = cli_close(cli, fnum1);
    8797           4 :         if (!NT_STATUS_IS_OK(status)) {
    8798           0 :                 printf("close failed (%s)\n",
    8799             :                         nt_errstr(status));
    8800           0 :                 goto out;
    8801             :         }
    8802           4 :         fnum1 = (uint16_t)-1;
    8803             : 
    8804             :         /* Change the lock context. */
    8805           4 :         cli_setpid(cli, cli_getpid(cli) + 1);
    8806             : 
    8807             :         /* Re-open fnum1. */
    8808           4 :         status = cli_posix_open(cli, fname, O_RDWR, 0, &fnum1);
    8809           4 :         if (!NT_STATUS_IS_OK(status)) {
    8810           0 :                 printf("Third POSIX open of %s failed\n", fname);
    8811           0 :                 goto out;
    8812             :         }
    8813             : 
    8814             :         /* 60-100 lock should still be there. */
    8815           4 :         status = cli_posix_lock(cli, fnum1, 60, 100, false, WRITE_LOCK);
    8816           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_FILE_LOCK_CONFLICT)) {
    8817           0 :                 printf("POSIX lock 60-100 not there %s\n", nt_errstr(status));
    8818           0 :                 goto out;
    8819             :         }
    8820             : 
    8821             :         /* 0-50 lock should be gone. */
    8822           4 :         status = cli_posix_lock(cli, fnum1, 0, 50, false, WRITE_LOCK);
    8823           4 :         if (!NT_STATUS_IS_OK(status)) {
    8824           0 :                 printf("POSIX lock 0-50 failed %s\n", nt_errstr(status));
    8825           0 :                 goto out;
    8826             :         }
    8827             : 
    8828           4 :         printf("POSIX OFD lock test passed\n");
    8829           4 :         correct = true;
    8830             : 
    8831           4 :   out:
    8832             : 
    8833           4 :         if (fnum1 != (uint16_t)-1) {
    8834           4 :                 cli_close(cli, fnum1);
    8835           4 :                 fnum1 = (uint16_t)-1;
    8836             :         }
    8837           4 :         if (fnum2 != (uint16_t)-1) {
    8838           4 :                 cli_close(cli, fnum2);
    8839           4 :                 fnum2 = (uint16_t)-1;
    8840             :         }
    8841             : 
    8842           4 :         cli_setatr(cli, fname, 0, 0);
    8843           4 :         cli_posix_unlink(cli, fname);
    8844             : 
    8845           4 :         if (!torture_close_connection(cli)) {
    8846           0 :                 correct = false;
    8847             :         }
    8848             : 
    8849           4 :         TALLOC_FREE(frame);
    8850           4 :         return correct;
    8851             : }
    8852             : 
    8853             : struct posix_blocking_state {
    8854             :         struct tevent_context *ev;
    8855             :         struct cli_state *cli1;
    8856             :         uint16_t fnum1;
    8857             :         struct cli_state *cli2;
    8858             :         uint16_t fnum2;
    8859             :         bool gotblocked;
    8860             :         bool gotecho;
    8861             : };
    8862             : 
    8863             : static void posix_blocking_locked(struct tevent_req *subreq);
    8864             : static void posix_blocking_gotblocked(struct tevent_req *subreq);
    8865             : static void posix_blocking_gotecho(struct tevent_req *subreq);
    8866             : static void posix_blocking_unlocked(struct tevent_req *subreq);
    8867             : 
    8868           4 : static struct tevent_req *posix_blocking_send(
    8869             :         TALLOC_CTX *mem_ctx,
    8870             :         struct tevent_context *ev,
    8871             :         struct cli_state *cli1,
    8872             :         uint16_t fnum1,
    8873             :         struct cli_state *cli2,
    8874             :         uint16_t fnum2)
    8875             : {
    8876           4 :         struct tevent_req *req = NULL, *subreq = NULL;
    8877           4 :         struct posix_blocking_state *state = NULL;
    8878             : 
    8879           4 :         req = tevent_req_create(mem_ctx, &state, struct posix_blocking_state);
    8880           4 :         if (req == NULL) {
    8881           0 :                 return NULL;
    8882             :         }
    8883           4 :         state->ev = ev;
    8884           4 :         state->cli1 = cli1;
    8885           4 :         state->fnum1 = fnum1;
    8886           4 :         state->cli2 = cli2;
    8887           4 :         state->fnum2 = fnum2;
    8888             : 
    8889           4 :         subreq = cli_posix_lock_send(
    8890             :                 state,
    8891           4 :                 state->ev,
    8892           4 :                 state->cli1,
    8893           4 :                 state->fnum1,
    8894             :                 0,
    8895             :                 1,
    8896             :                 false,
    8897             :                 WRITE_LOCK);
    8898           4 :         if (tevent_req_nomem(subreq, req)) {
    8899           0 :                 return tevent_req_post(req, ev);
    8900             :         }
    8901           4 :         tevent_req_set_callback(subreq, posix_blocking_locked, req);
    8902           4 :         return req;
    8903             : }
    8904             : 
    8905           4 : static void posix_blocking_locked(struct tevent_req *subreq)
    8906             : {
    8907           4 :         struct tevent_req *req = tevent_req_callback_data(
    8908             :                 subreq, struct tevent_req);
    8909           4 :         struct posix_blocking_state *state = tevent_req_data(
    8910             :                 req, struct posix_blocking_state);
    8911           0 :         NTSTATUS status;
    8912             : 
    8913           4 :         status = cli_posix_lock_recv(subreq);
    8914           4 :         TALLOC_FREE(subreq);
    8915           4 :         if (tevent_req_nterror(req, status)) {
    8916           0 :                 return;
    8917             :         }
    8918             : 
    8919           4 :         subreq = cli_posix_lock_send(
    8920             :                 state,
    8921             :                 state->ev,
    8922             :                 state->cli2,
    8923           4 :                 state->fnum2,
    8924             :                 0,
    8925             :                 1,
    8926             :                 true,
    8927             :                 WRITE_LOCK);
    8928           4 :         if (tevent_req_nomem(subreq, req)) {
    8929           0 :                 return;
    8930             :         }
    8931           4 :         tevent_req_set_callback(subreq, posix_blocking_gotblocked, req);
    8932             : 
    8933             :         /* Make sure the blocking request is delivered */
    8934           4 :         subreq = cli_echo_send(
    8935             :                 state,
    8936             :                 state->ev,
    8937             :                 state->cli2,
    8938             :                 1,
    8939           4 :                 (DATA_BLOB) { .data = (uint8_t *)state, .length = 1 });
    8940           4 :         if (tevent_req_nomem(subreq, req)) {
    8941           0 :                 return;
    8942             :         }
    8943           4 :         tevent_req_set_callback(subreq, posix_blocking_gotecho, req);
    8944             : }
    8945             : 
    8946           4 : static void posix_blocking_gotblocked(struct tevent_req *subreq)
    8947             : {
    8948           4 :         struct tevent_req *req = tevent_req_callback_data(
    8949             :                 subreq, struct tevent_req);
    8950           4 :         struct posix_blocking_state *state = tevent_req_data(
    8951             :                 req, struct posix_blocking_state);
    8952           0 :         NTSTATUS status;
    8953             : 
    8954           4 :         status = cli_posix_lock_recv(subreq);
    8955           4 :         TALLOC_FREE(subreq);
    8956           4 :         if (tevent_req_nterror(req, status)) {
    8957           0 :                 return;
    8958             :         }
    8959           4 :         if (!state->gotecho) {
    8960           0 :                 printf("blocked req got through before echo\n");
    8961           0 :                 tevent_req_nterror(req, NT_STATUS_INVALID_LOCK_SEQUENCE);
    8962           0 :                 return;
    8963             :         }
    8964           4 :         tevent_req_done(req);
    8965             : }
    8966             : 
    8967           4 : static void posix_blocking_gotecho(struct tevent_req *subreq)
    8968             : {
    8969           4 :         struct tevent_req *req = tevent_req_callback_data(
    8970             :                 subreq, struct tevent_req);
    8971           4 :         struct posix_blocking_state *state = tevent_req_data(
    8972             :                 req, struct posix_blocking_state);
    8973           0 :         NTSTATUS status;
    8974             : 
    8975           4 :         status = cli_echo_recv(subreq);
    8976           4 :         TALLOC_FREE(subreq);
    8977           4 :         if (tevent_req_nterror(req, status)) {
    8978           0 :                 return;
    8979             :         }
    8980           4 :         if (state->gotblocked) {
    8981           0 :                 printf("blocked req got through before echo\n");
    8982           0 :                 tevent_req_nterror(req, NT_STATUS_INVALID_LOCK_SEQUENCE);
    8983           0 :                 return;
    8984             :         }
    8985           4 :         state->gotecho = true;
    8986             : 
    8987           4 :         subreq = cli_posix_lock_send(
    8988             :                 state,
    8989             :                 state->ev,
    8990             :                 state->cli1,
    8991           4 :                 state->fnum1,
    8992             :                 0,
    8993             :                 1,
    8994             :                 false,
    8995             :                 UNLOCK_LOCK);
    8996           4 :         if (tevent_req_nomem(subreq, req)) {
    8997           0 :                 return;
    8998             :         }
    8999           4 :         tevent_req_set_callback(subreq, posix_blocking_unlocked, req);
    9000             : }
    9001             : 
    9002           4 : static void posix_blocking_unlocked(struct tevent_req *subreq)
    9003             : {
    9004           4 :         struct tevent_req *req = tevent_req_callback_data(
    9005             :                 subreq, struct tevent_req);
    9006           0 :         NTSTATUS status;
    9007             : 
    9008           4 :         status = cli_posix_lock_recv(subreq);
    9009           4 :         TALLOC_FREE(subreq);
    9010           4 :         if (tevent_req_nterror(req, status)) {
    9011           0 :                 return;
    9012             :         }
    9013             :         /* tevent_req_done in posix_blocking_gotlocked */
    9014             : }
    9015             : 
    9016           4 : static NTSTATUS posix_blocking_recv(struct tevent_req *req)
    9017             : {
    9018           4 :         return tevent_req_simple_recv_ntstatus(req);
    9019             : }
    9020             : 
    9021           4 : static bool run_posix_blocking_lock(int dummy)
    9022             : {
    9023           4 :         struct tevent_context *ev = NULL;
    9024           4 :         struct cli_state *cli1 = NULL, *cli2 = NULL;
    9025           4 :         const char *fname = "posix_blocking";
    9026           4 :         uint16_t fnum1 = UINT16_MAX, fnum2 = UINT16_MAX;
    9027           4 :         struct tevent_req *req = NULL;
    9028           0 :         NTSTATUS status;
    9029           4 :         bool ret = false;
    9030           0 :         bool ok;
    9031             : 
    9032           4 :         printf("Starting posix blocking lock test\n");
    9033             : 
    9034           4 :         ev = samba_tevent_context_init(NULL);
    9035           4 :         if (ev == NULL) {
    9036           0 :                 return false;
    9037             :         }
    9038             : 
    9039           4 :         ok = torture_open_connection(&cli1, 0);
    9040           4 :         if (!ok) {
    9041           0 :                 goto fail;
    9042             :         }
    9043           4 :         ok = torture_open_connection(&cli2, 0);
    9044           4 :         if (!ok) {
    9045           0 :                 goto fail;
    9046             :         }
    9047             : 
    9048           4 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    9049             : 
    9050           4 :         status = torture_setup_unix_extensions(cli1);
    9051           4 :         if (!NT_STATUS_IS_OK(status)) {
    9052           0 :                 return false;
    9053             :         }
    9054             : 
    9055           4 :         status = torture_setup_unix_extensions(cli2);
    9056           4 :         if (!NT_STATUS_IS_OK(status)) {
    9057           0 :                 return false;
    9058             :         }
    9059             : 
    9060           4 :         cli_setatr(cli1, fname, 0, 0);
    9061           4 :         cli_posix_unlink(cli1, fname);
    9062             : 
    9063           4 :         status = cli_posix_open(cli1, fname, O_RDWR|O_CREAT|O_EXCL,
    9064             :                                 0600, &fnum1);
    9065           4 :         if (!NT_STATUS_IS_OK(status)) {
    9066           0 :                 printf("First POSIX open of %s failed: %s\n",
    9067             :                        fname,
    9068             :                        nt_errstr(status));
    9069           0 :                 goto fail;
    9070             :         }
    9071             : 
    9072           4 :         status = cli_posix_open(cli2, fname, O_RDWR, 0600, &fnum2);
    9073           4 :         if (!NT_STATUS_IS_OK(status)) {
    9074           0 :                 printf("Second POSIX open of %s failed: %s\n",
    9075             :                        fname,
    9076             :                        nt_errstr(status));
    9077           0 :                 goto fail;
    9078             :         }
    9079             : 
    9080           4 :         req = posix_blocking_send(ev, ev, cli1, fnum1, cli2, fnum2);
    9081           4 :         if (req == NULL) {
    9082           0 :                 printf("cli_posix_blocking failed\n");
    9083           0 :                 goto fail;
    9084             :         }
    9085             : 
    9086           4 :         ok = tevent_req_poll_ntstatus(req, ev, &status);
    9087           4 :         if (!ok) {
    9088           0 :                 printf("tevent_req_poll_ntstatus failed: %s\n",
    9089             :                        nt_errstr(status));
    9090           0 :                 goto fail;
    9091             :         }
    9092           4 :         status = posix_blocking_recv(req);
    9093           4 :         TALLOC_FREE(req);
    9094           4 :         if (!NT_STATUS_IS_OK(status)) {
    9095           0 :                 printf("posix_blocking_recv returned %s\n",
    9096             :                        nt_errstr(status));
    9097           0 :                 goto fail;
    9098             :         }
    9099             : 
    9100           4 :         ret = true;
    9101           4 : fail:
    9102             : 
    9103           4 :         if (fnum1 != UINT16_MAX) {
    9104           4 :                 cli_close(cli1, fnum1);
    9105           4 :                 fnum1 = UINT16_MAX;
    9106             :         }
    9107           4 :         if (fnum2 != UINT16_MAX) {
    9108           4 :                 cli_close(cli2, fnum2);
    9109           4 :                 fnum2 = UINT16_MAX;
    9110             :         }
    9111             : 
    9112           4 :         if (cli1 != NULL) {
    9113           4 :                 cli_setatr(cli1, fname, 0, 0);
    9114           4 :                 cli_posix_unlink(cli1, fname);
    9115             :         }
    9116             : 
    9117           4 :         ok = true;
    9118             : 
    9119           4 :         if (cli1 != NULL) {
    9120           4 :                 ok &= torture_close_connection(cli1);
    9121           4 :                 cli1 = NULL;
    9122             :         }
    9123           4 :         if (cli2 != NULL) {
    9124           4 :                 ok &= torture_close_connection(cli2);
    9125           4 :                 cli2 = NULL;
    9126             :         }
    9127             : 
    9128           4 :         if (!ok) {
    9129           0 :                 ret = false;
    9130             :         }
    9131           4 :         TALLOC_FREE(ev);
    9132           4 :         return ret;
    9133             : }
    9134             : 
    9135             : /*
    9136             :   Test POSIX mkdir is case-sensitive.
    9137             :  */
    9138           4 : static bool run_posix_mkdir_test(int dummy)
    9139             : {
    9140           0 :         static struct cli_state *cli;
    9141           4 :         const char *fname_foo = "POSIX_foo";
    9142           4 :         const char *fname_foo_Foo = "POSIX_foo/Foo";
    9143           4 :         const char *fname_foo_foo = "POSIX_foo/foo";
    9144           4 :         const char *fname_Foo = "POSIX_Foo";
    9145           4 :         const char *fname_Foo_Foo = "POSIX_Foo/Foo";
    9146           4 :         const char *fname_Foo_foo = "POSIX_Foo/foo";
    9147           4 :         bool correct = false;
    9148           0 :         NTSTATUS status;
    9149           4 :         TALLOC_CTX *frame = NULL;
    9150           4 :         uint16_t fnum = (uint16_t)-1;
    9151             : 
    9152           4 :         frame = talloc_stackframe();
    9153             : 
    9154           4 :         printf("Starting POSIX mkdir test\n");
    9155             : 
    9156           4 :         if (!torture_open_connection(&cli, 0)) {
    9157           0 :                 TALLOC_FREE(frame);
    9158           0 :                 return false;
    9159             :         }
    9160             : 
    9161           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    9162             : 
    9163           4 :         status = torture_setup_unix_extensions(cli);
    9164           4 :         if (!NT_STATUS_IS_OK(status)) {
    9165           0 :                 TALLOC_FREE(frame);
    9166           0 :                 return false;
    9167             :         }
    9168             : 
    9169           4 :         cli_posix_rmdir(cli, fname_foo_foo);
    9170           4 :         cli_posix_rmdir(cli, fname_foo_Foo);
    9171           4 :         cli_posix_rmdir(cli, fname_foo);
    9172             : 
    9173           4 :         cli_posix_rmdir(cli, fname_Foo_foo);
    9174           4 :         cli_posix_rmdir(cli, fname_Foo_Foo);
    9175           4 :         cli_posix_rmdir(cli, fname_Foo);
    9176             : 
    9177             :         /*
    9178             :          * Create a file POSIX_foo then try
    9179             :          * and use it in a directory path by
    9180             :          * doing mkdir POSIX_foo/bar.
    9181             :          * The mkdir should fail with
    9182             :          * NT_STATUS_OBJECT_PATH_NOT_FOUND
    9183             :          */
    9184             : 
    9185           4 :         status = cli_posix_open(cli,
    9186             :                         fname_foo,
    9187             :                         O_RDWR|O_CREAT,
    9188             :                         0666,
    9189             :                         &fnum);
    9190           4 :         if (!NT_STATUS_IS_OK(status)) {
    9191           0 :                 printf("cli_posix_open of %s failed error %s\n",
    9192             :                         fname_foo,
    9193             :                         nt_errstr(status));
    9194           0 :                 goto out;
    9195             :         }
    9196             : 
    9197           4 :         status = cli_posix_mkdir(cli, fname_foo_foo, 0777);
    9198           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_PATH_NOT_FOUND)) {
    9199           0 :                 printf("cli_posix_mkdir of %s should fail with "
    9200             :                         "NT_STATUS_OBJECT_PATH_NOT_FOUND got "
    9201             :                         "%s instead\n",
    9202             :                         fname_foo_foo,
    9203             :                         nt_errstr(status));
    9204           0 :                 goto out;
    9205             :         }
    9206             : 
    9207           4 :         status = cli_close(cli, fnum);
    9208           4 :         if (!NT_STATUS_IS_OK(status)) {
    9209           0 :                 printf("cli_close failed %s\n", nt_errstr(status));
    9210           0 :                 goto out;
    9211             :         }
    9212           4 :         fnum = (uint16_t)-1;
    9213             : 
    9214           4 :         status = cli_posix_unlink(cli, fname_foo);
    9215           4 :         if (!NT_STATUS_IS_OK(status)) {
    9216           0 :                 printf("cli_posix_unlink of %s failed error %s\n",
    9217             :                         fname_foo,
    9218             :                         nt_errstr(status));
    9219           0 :                 goto out;
    9220             :         }
    9221             : 
    9222             :         /*
    9223             :          * Now we've deleted everything, posix_mkdir, posix_rmdir,
    9224             :          * posix_open, posix_unlink, on
    9225             :          * POSIX_foo/foo should return NT_STATUS_OBJECT_PATH_NOT_FOUND
    9226             :          * not silently create POSIX_foo/foo.
    9227             :          */
    9228             : 
    9229           4 :         status = cli_posix_mkdir(cli, fname_foo_foo, 0777);
    9230           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_PATH_NOT_FOUND)) {
    9231           0 :                 printf("cli_posix_mkdir of %s should fail with "
    9232             :                         "NT_STATUS_OBJECT_PATH_NOT_FOUND got "
    9233             :                         "%s instead\n",
    9234             :                         fname_foo_foo,
    9235             :                         nt_errstr(status));
    9236           0 :                 goto out;
    9237             :         }
    9238             : 
    9239           4 :         status = cli_posix_rmdir(cli, fname_foo_foo);
    9240           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_PATH_NOT_FOUND)) {
    9241           0 :                 printf("cli_posix_rmdir of %s should fail with "
    9242             :                         "NT_STATUS_OBJECT_PATH_NOT_FOUND got "
    9243             :                         "%s instead\n",
    9244             :                         fname_foo_foo,
    9245             :                         nt_errstr(status));
    9246           0 :                 goto out;
    9247             :         }
    9248             : 
    9249           4 :         status = cli_posix_open(cli,
    9250             :                         fname_foo_foo,
    9251             :                         O_RDWR|O_CREAT,
    9252             :                         0666,
    9253             :                         &fnum);
    9254           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_PATH_NOT_FOUND)) {
    9255           0 :                 printf("cli_posix_open of %s should fail with "
    9256             :                         "NT_STATUS_OBJECT_PATH_NOT_FOUND got "
    9257             :                         "%s instead\n",
    9258             :                         fname_foo_foo,
    9259             :                         nt_errstr(status));
    9260           0 :                 goto out;
    9261             :         }
    9262             : 
    9263           4 :         status = cli_posix_unlink(cli, fname_foo_foo);
    9264           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_PATH_NOT_FOUND)) {
    9265           0 :                 printf("cli_posix_unlink of %s should fail with "
    9266             :                         "NT_STATUS_OBJECT_PATH_NOT_FOUND got "
    9267             :                         "%s instead\n",
    9268             :                         fname_foo_foo,
    9269             :                         nt_errstr(status));
    9270           0 :                 goto out;
    9271             :         }
    9272             : 
    9273           4 :         status = cli_posix_mkdir(cli, fname_foo, 0777);
    9274           4 :         if (!NT_STATUS_IS_OK(status)) {
    9275           0 :                 printf("cli_posix_mkdir of %s failed\n", fname_foo);
    9276           0 :                 goto out;
    9277             :         }
    9278             : 
    9279           4 :         status = cli_posix_mkdir(cli, fname_Foo, 0777);
    9280           4 :         if (!NT_STATUS_IS_OK(status)) {
    9281           0 :                 printf("cli_posix_mkdir of %s failed\n", fname_Foo);
    9282           0 :                 goto out;
    9283             :         }
    9284             : 
    9285           4 :         status = cli_posix_mkdir(cli, fname_foo_foo, 0777);
    9286           4 :         if (!NT_STATUS_IS_OK(status)) {
    9287           0 :                 printf("cli_posix_mkdir of %s failed\n", fname_foo_foo);
    9288           0 :                 goto out;
    9289             :         }
    9290             : 
    9291           4 :         status = cli_posix_mkdir(cli, fname_foo_Foo, 0777);
    9292           4 :         if (!NT_STATUS_IS_OK(status)) {
    9293           0 :                 printf("cli_posix_mkdir of %s failed\n", fname_foo_Foo);
    9294           0 :                 goto out;
    9295             :         }
    9296             : 
    9297           4 :         status = cli_posix_mkdir(cli, fname_Foo_foo, 0777);
    9298           4 :         if (!NT_STATUS_IS_OK(status)) {
    9299           0 :                 printf("cli_posix_mkdir of %s failed\n", fname_Foo_foo);
    9300           0 :                 goto out;
    9301             :         }
    9302             : 
    9303           4 :         status = cli_posix_mkdir(cli, fname_Foo_Foo, 0777);
    9304           4 :         if (!NT_STATUS_IS_OK(status)) {
    9305           0 :                 printf("cli_posix_mkdir of %s failed\n", fname_Foo_Foo);
    9306           0 :                 goto out;
    9307             :         }
    9308             : 
    9309           4 :         printf("POSIX mkdir test passed\n");
    9310           4 :         correct = true;
    9311             : 
    9312           4 :   out:
    9313             : 
    9314           4 :         if (fnum != (uint16_t)-1) {
    9315           0 :                 cli_close(cli, fnum);
    9316           0 :                 fnum = (uint16_t)-1;
    9317             :         }
    9318             : 
    9319           4 :         cli_posix_rmdir(cli, fname_foo_foo);
    9320           4 :         cli_posix_rmdir(cli, fname_foo_Foo);
    9321           4 :         cli_posix_rmdir(cli, fname_foo);
    9322             : 
    9323           4 :         cli_posix_rmdir(cli, fname_Foo_foo);
    9324           4 :         cli_posix_rmdir(cli, fname_Foo_Foo);
    9325           4 :         cli_posix_rmdir(cli, fname_Foo);
    9326             : 
    9327           4 :         if (!torture_close_connection(cli)) {
    9328           0 :                 correct = false;
    9329             :         }
    9330             : 
    9331           4 :         TALLOC_FREE(frame);
    9332           4 :         return correct;
    9333             : }
    9334             : 
    9335             : struct posix_acl_oplock_state {
    9336             :         struct tevent_context *ev;
    9337             :         struct cli_state *cli;
    9338             :         bool *got_break;
    9339             :         bool *acl_ret;
    9340             :         NTSTATUS status;
    9341             : };
    9342             : 
    9343           4 : static void posix_acl_oplock_got_break(struct tevent_req *req)
    9344             : {
    9345           4 :         struct posix_acl_oplock_state *state = tevent_req_callback_data(
    9346             :                 req, struct posix_acl_oplock_state);
    9347           0 :         uint16_t fnum;
    9348           0 :         uint8_t level;
    9349           0 :         NTSTATUS status;
    9350             : 
    9351           4 :         status = cli_smb_oplock_break_waiter_recv(req, &fnum, &level);
    9352           4 :         TALLOC_FREE(req);
    9353           4 :         if (!NT_STATUS_IS_OK(status)) {
    9354           0 :                 printf("cli_smb_oplock_break_waiter_recv returned %s\n",
    9355             :                        nt_errstr(status));
    9356           0 :                 return;
    9357             :         }
    9358           4 :         *state->got_break = true;
    9359             : 
    9360           4 :         req = cli_oplock_ack_send(state, state->ev, state->cli, fnum,
    9361             :                                   NO_OPLOCK);
    9362           4 :         if (req == NULL) {
    9363           0 :                 printf("cli_oplock_ack_send failed\n");
    9364           0 :                 return;
    9365             :         }
    9366             : }
    9367             : 
    9368           4 : static void posix_acl_oplock_got_acl(struct tevent_req *req)
    9369             : {
    9370           4 :         struct posix_acl_oplock_state *state = tevent_req_callback_data(
    9371             :                 req, struct posix_acl_oplock_state);
    9372           4 :         size_t ret_size = 0;
    9373           4 :         char *ret_data = NULL;
    9374             : 
    9375           4 :         state->status = cli_posix_getacl_recv(req,
    9376             :                         state,
    9377             :                         &ret_size,
    9378             :                         &ret_data);
    9379             : 
    9380           4 :         if (!NT_STATUS_IS_OK(state->status)) {
    9381           0 :                 printf("cli_posix_getacl_recv returned %s\n",
    9382             :                         nt_errstr(state->status));
    9383             :         }
    9384           4 :         *state->acl_ret = true;
    9385           4 : }
    9386             : 
    9387           4 : static bool run_posix_acl_oplock_test(int dummy)
    9388             : {
    9389           0 :         struct tevent_context *ev;
    9390           0 :         struct cli_state *cli1, *cli2;
    9391           0 :         struct tevent_req *oplock_req, *getacl_req;
    9392           4 :         const char *fname = "posix_acl_oplock";
    9393           0 :         uint16_t fnum;
    9394           4 :         int saved_use_oplocks = use_oplocks;
    9395           0 :         NTSTATUS status;
    9396           4 :         bool correct = true;
    9397           4 :         bool got_break = false;
    9398           4 :         bool acl_ret = false;
    9399             : 
    9400           0 :         struct posix_acl_oplock_state *state;
    9401             : 
    9402           4 :         printf("starting posix_acl_oplock test\n");
    9403             : 
    9404           4 :         if (!torture_open_connection(&cli1, 0)) {
    9405           0 :                 use_level_II_oplocks = false;
    9406           0 :                 use_oplocks = saved_use_oplocks;
    9407           0 :                 return false;
    9408             :         }
    9409             : 
    9410           4 :         if (!torture_open_connection(&cli2, 1)) {
    9411           0 :                 use_level_II_oplocks = false;
    9412           0 :                 use_oplocks = saved_use_oplocks;
    9413           0 :                 return false;
    9414             :         }
    9415             : 
    9416             :         /* Setup posix on cli2 only. */
    9417           4 :         status = torture_setup_unix_extensions(cli2);
    9418           4 :         if (!NT_STATUS_IS_OK(status)) {
    9419           0 :                 return false;
    9420             :         }
    9421             : 
    9422           4 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    9423           4 :         smbXcli_conn_set_sockopt(cli2->conn, sockops);
    9424             : 
    9425           4 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    9426             : 
    9427             :         /* Create the file on the Windows connection. */
    9428           4 :         status = cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE,
    9429             :                           &fnum);
    9430           4 :         if (!NT_STATUS_IS_OK(status)) {
    9431           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    9432           0 :                 return false;
    9433             :         }
    9434             : 
    9435           4 :         status = cli_close(cli1, fnum);
    9436           4 :         if (!NT_STATUS_IS_OK(status)) {
    9437           0 :                 printf("close1 failed (%s)\n", nt_errstr(status));
    9438           0 :                 return false;
    9439             :         }
    9440             : 
    9441           4 :         cli1->use_oplocks = true;
    9442             : 
    9443             :         /* Open with oplock. */
    9444           4 :         status = cli_ntcreate(cli1,
    9445             :                         fname,
    9446             :                         0,
    9447             :                         FILE_READ_DATA,
    9448             :                         FILE_ATTRIBUTE_NORMAL,
    9449             :                         FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
    9450             :                         FILE_OPEN,
    9451             :                         0,
    9452             :                         0,
    9453             :                         &fnum,
    9454             :                         NULL);
    9455             : 
    9456           4 :         if (!NT_STATUS_IS_OK(status)) {
    9457           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    9458           0 :                 return false;
    9459             :         }
    9460             : 
    9461           4 :         ev = samba_tevent_context_init(talloc_tos());
    9462           4 :         if (ev == NULL) {
    9463           0 :                 printf("tevent_context_init failed\n");
    9464           0 :                 return false;
    9465             :         }
    9466             : 
    9467           4 :         state = talloc_zero(ev, struct posix_acl_oplock_state);
    9468           4 :         if (state == NULL) {
    9469           0 :                 printf("talloc failed\n");
    9470           0 :                 return false;
    9471             :         }
    9472           4 :         state->ev = ev;
    9473           4 :         state->cli = cli1;
    9474           4 :         state->got_break = &got_break;
    9475           4 :         state->acl_ret = &acl_ret;
    9476             : 
    9477           4 :         oplock_req = cli_smb_oplock_break_waiter_send(
    9478             :                 talloc_tos(), ev, cli1);
    9479           4 :         if (oplock_req == NULL) {
    9480           0 :                 printf("cli_smb_oplock_break_waiter_send failed\n");
    9481           0 :                 return false;
    9482             :         }
    9483           4 :         tevent_req_set_callback(oplock_req, posix_acl_oplock_got_break, state);
    9484             : 
    9485             :         /* Get ACL on POSIX connection - should break oplock. */
    9486           4 :         getacl_req = cli_posix_getacl_send(talloc_tos(),
    9487             :                                 ev,
    9488             :                                 cli2,
    9489             :                                 fname);
    9490           4 :         if (getacl_req == NULL) {
    9491           0 :                 printf("cli_posix_getacl_send failed\n");
    9492           0 :                 return false;
    9493             :         }
    9494           4 :         tevent_req_set_callback(getacl_req, posix_acl_oplock_got_acl, state);
    9495             : 
    9496          36 :         while (!got_break || !acl_ret) {
    9497           0 :                 int ret;
    9498          32 :                 ret = tevent_loop_once(ev);
    9499          32 :                 if (ret == -1) {
    9500           0 :                         printf("tevent_loop_once failed: %s\n",
    9501           0 :                                strerror(errno));
    9502           0 :                         return false;
    9503             :                 }
    9504             :         }
    9505             : 
    9506           4 :         if (!NT_STATUS_IS_OK(state->status)) {
    9507           0 :                 printf("getacl failed (%s)\n", nt_errstr(state->status));
    9508           0 :                 correct = false;
    9509             :         }
    9510             : 
    9511           4 :         status = cli_close(cli1, fnum);
    9512           4 :         if (!NT_STATUS_IS_OK(status)) {
    9513           0 :                 printf("close2 failed (%s)\n", nt_errstr(status));
    9514           0 :                 correct = false;
    9515             :         }
    9516             : 
    9517           4 :         status = cli_unlink(cli1,
    9518             :                         fname,
    9519             :                         FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    9520           4 :         if (!NT_STATUS_IS_OK(status)) {
    9521           0 :                 printf("unlink failed (%s)\n", nt_errstr(status));
    9522           0 :                 correct = false;
    9523             :         }
    9524             : 
    9525           4 :         if (!torture_close_connection(cli1)) {
    9526           0 :                 correct = false;
    9527             :         }
    9528           4 :         if (!torture_close_connection(cli2)) {
    9529           0 :                 correct = false;
    9530             :         }
    9531             : 
    9532           4 :         if (!got_break) {
    9533           0 :                 correct = false;
    9534             :         }
    9535             : 
    9536           4 :         printf("finished posix acl oplock test\n");
    9537             : 
    9538           4 :         return correct;
    9539             : }
    9540             : 
    9541           4 : static bool run_posix_acl_shareroot_test(int dummy)
    9542             : {
    9543           0 :         struct cli_state *cli;
    9544           0 :         NTSTATUS status;
    9545           4 :         bool correct = false;
    9546           4 :         char *posix_acl = NULL;
    9547           4 :         size_t posix_acl_len = 0;
    9548           4 :         uint16_t num_file_acls = 0;
    9549           4 :         uint16_t num_dir_acls = 0;
    9550           0 :         uint16_t i;
    9551           4 :         uint32_t expected_size = 0;
    9552           4 :         bool got_user = false;
    9553           4 :         bool got_group = false;
    9554           4 :         bool got_other = false;
    9555           4 :         TALLOC_CTX *frame = NULL;
    9556             : 
    9557           4 :         frame = talloc_stackframe();
    9558             : 
    9559           4 :         printf("starting posix_acl_shareroot test\n");
    9560             : 
    9561           4 :         if (!torture_open_connection(&cli, 0)) {
    9562           0 :                 TALLOC_FREE(frame);
    9563           0 :                 return false;
    9564             :         }
    9565             : 
    9566           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    9567             : 
    9568           4 :         status = torture_setup_unix_extensions(cli);
    9569           4 :         if (!NT_STATUS_IS_OK(status)) {
    9570           0 :                 printf("Failed to setup unix extensions\n");
    9571           0 :                 goto out;
    9572             :         }
    9573             : 
    9574             :         /* Get the POSIX ACL on the root of the share. */
    9575           4 :         status = cli_posix_getacl(cli,
    9576             :                                 ".",
    9577             :                                 frame,
    9578             :                                 &posix_acl_len,
    9579             :                                 &posix_acl);
    9580             : 
    9581           4 :         if (!NT_STATUS_IS_OK(status)) {
    9582           0 :                 printf("cli_posix_getacl of '.' failed (%s)\n",
    9583             :                         nt_errstr(status));
    9584           0 :                 goto out;
    9585             :         }
    9586             : 
    9587           4 :         if (posix_acl_len < 6 ||
    9588           4 :                         SVAL(posix_acl,0) != SMB_POSIX_ACL_VERSION) {
    9589           0 :                 printf("getfacl ., unknown POSIX acl version %u.\n",
    9590           0 :                         (unsigned int)CVAL(posix_acl,0) );
    9591           0 :                 goto out;
    9592             :         }
    9593             : 
    9594           4 :         num_file_acls = SVAL(posix_acl,2);
    9595           4 :         num_dir_acls = SVAL(posix_acl,4);
    9596           4 :         expected_size = SMB_POSIX_ACL_HEADER_SIZE +
    9597           4 :                                 SMB_POSIX_ACL_ENTRY_SIZE*
    9598           4 :                                 (num_file_acls+num_dir_acls);
    9599             : 
    9600           4 :         if (posix_acl_len != expected_size) {
    9601           0 :                 printf("incorrect POSIX acl buffer size "
    9602             :                         "(should be %u, was %u).\n",
    9603             :                         (unsigned int)expected_size,
    9604             :                         (unsigned int)posix_acl_len);
    9605           0 :                 goto out;
    9606             :         }
    9607             : 
    9608             :         /*
    9609             :          * We don't need to know what the ACL's are
    9610             :          * we just need to know we have at least 3
    9611             :          * file entries (u,g,o).
    9612             :          */
    9613             : 
    9614          16 :         for (i = 0; i < num_file_acls; i++) {
    9615          12 :                 unsigned char tagtype =
    9616          12 :                         CVAL(posix_acl,
    9617             :                                 SMB_POSIX_ACL_HEADER_SIZE+
    9618             :                                 (i*SMB_POSIX_ACL_ENTRY_SIZE));
    9619             : 
    9620          12 :                 switch(tagtype) {
    9621           4 :                         case SMB_POSIX_ACL_USER_OBJ:
    9622           4 :                                 got_user = true;
    9623           4 :                                 break;
    9624           4 :                         case SMB_POSIX_ACL_GROUP_OBJ:
    9625           4 :                                 got_group = true;
    9626           4 :                                 break;
    9627           4 :                         case SMB_POSIX_ACL_OTHER:
    9628           4 :                                 got_other = true;
    9629           4 :                                 break;
    9630           0 :                         default:
    9631           0 :                                 break;
    9632             :                 }
    9633             :         }
    9634             : 
    9635           4 :         if (!got_user) {
    9636           0 :                 printf("Missing user entry\n");
    9637           0 :                 goto out;
    9638             :         }
    9639             : 
    9640           4 :         if (!got_group) {
    9641           0 :                 printf("Missing group entry\n");
    9642           0 :                 goto out;
    9643             :         }
    9644             : 
    9645           4 :         if (!got_other) {
    9646           0 :                 printf("Missing other entry\n");
    9647           0 :                 goto out;
    9648             :         }
    9649             : 
    9650           4 :         correct = true;
    9651             : 
    9652           4 :   out:
    9653             : 
    9654           4 :         if (!torture_close_connection(cli)) {
    9655           0 :                 correct = false;
    9656             :         }
    9657             : 
    9658           4 :         printf("finished posix acl shareroot test\n");
    9659           4 :         TALLOC_FREE(frame);
    9660             : 
    9661           4 :         return correct;
    9662             : }
    9663             : 
    9664             : static uint32_t open_attrs_table[] = {
    9665             :                 FILE_ATTRIBUTE_NORMAL,
    9666             :                 FILE_ATTRIBUTE_ARCHIVE,
    9667             :                 FILE_ATTRIBUTE_READONLY,
    9668             :                 FILE_ATTRIBUTE_HIDDEN,
    9669             :                 FILE_ATTRIBUTE_SYSTEM,
    9670             : 
    9671             :                 FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY,
    9672             :                 FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN,
    9673             :                 FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM,
    9674             :                 FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_HIDDEN,
    9675             :                 FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_SYSTEM,
    9676             :                 FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN|FILE_ATTRIBUTE_SYSTEM,
    9677             : 
    9678             :                 FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_HIDDEN,
    9679             :                 FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_SYSTEM,
    9680             :                 FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_HIDDEN|FILE_ATTRIBUTE_SYSTEM,
    9681             :                 FILE_ATTRIBUTE_HIDDEN,FILE_ATTRIBUTE_SYSTEM,
    9682             : };
    9683             : 
    9684             : struct trunc_open_results {
    9685             :         unsigned int num;
    9686             :         uint32_t init_attr;
    9687             :         uint32_t trunc_attr;
    9688             :         uint32_t result_attr;
    9689             : };
    9690             : 
    9691             : static struct trunc_open_results attr_results[] = {
    9692             :         { 0, FILE_ATTRIBUTE_NORMAL, FILE_ATTRIBUTE_NORMAL, FILE_ATTRIBUTE_ARCHIVE },
    9693             :         { 1, FILE_ATTRIBUTE_NORMAL, FILE_ATTRIBUTE_ARCHIVE, FILE_ATTRIBUTE_ARCHIVE },
    9694             :         { 2, FILE_ATTRIBUTE_NORMAL, FILE_ATTRIBUTE_READONLY, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY },
    9695             :         { 16, FILE_ATTRIBUTE_ARCHIVE, FILE_ATTRIBUTE_NORMAL, FILE_ATTRIBUTE_ARCHIVE },
    9696             :         { 17, FILE_ATTRIBUTE_ARCHIVE, FILE_ATTRIBUTE_ARCHIVE, FILE_ATTRIBUTE_ARCHIVE },
    9697             :         { 18, FILE_ATTRIBUTE_ARCHIVE, FILE_ATTRIBUTE_READONLY, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY },
    9698             :         { 51, FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN },
    9699             :         { 54, FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN },
    9700             :         { 56, FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_HIDDEN },
    9701             :         { 68, FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM },
    9702             :         { 71, FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM },
    9703             :         { 73, FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_SYSTEM },
    9704             :         { 99, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_HIDDEN,FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN },
    9705             :         { 102, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN },
    9706             :         { 104, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_HIDDEN },
    9707             :         { 116, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM },
    9708             :         { 119,  FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM,  FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM },
    9709             :         { 121, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_SYSTEM },
    9710             :         { 170, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM|FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM|FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM|FILE_ATTRIBUTE_HIDDEN },
    9711             :         { 173, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM|FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_HIDDEN|FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_HIDDEN|FILE_ATTRIBUTE_SYSTEM },
    9712             :         { 227, FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN },
    9713             :         { 230, FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_HIDDEN },
    9714             :         { 232, FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_HIDDEN, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_HIDDEN },
    9715             :         { 244, FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM },
    9716             :         { 247, FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_SYSTEM },
    9717             :         { 249, FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_SYSTEM, FILE_ATTRIBUTE_ARCHIVE|FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_SYSTEM }
    9718             : };
    9719             : 
    9720           0 : static bool run_openattrtest(int dummy)
    9721             : {
    9722           0 :         static struct cli_state *cli1;
    9723           0 :         const char *fname = "\\openattr.file";
    9724           0 :         uint16_t fnum1;
    9725           0 :         bool correct = True;
    9726           0 :         uint32_t attr;
    9727           0 :         unsigned int i, j, k, l;
    9728           0 :         NTSTATUS status;
    9729             : 
    9730           0 :         printf("starting open attr test\n");
    9731             : 
    9732           0 :         if (!torture_open_connection(&cli1, 0)) {
    9733           0 :                 return False;
    9734             :         }
    9735             : 
    9736           0 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
    9737             : 
    9738           0 :         for (k = 0, i = 0; i < sizeof(open_attrs_table)/sizeof(uint32_t); i++) {
    9739           0 :                 cli_setatr(cli1, fname, 0, 0);
    9740           0 :                 cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    9741             : 
    9742           0 :                 status = cli_ntcreate(cli1, fname, 0, FILE_WRITE_DATA,
    9743             :                                        open_attrs_table[i], FILE_SHARE_NONE,
    9744             :                                        FILE_OVERWRITE_IF, 0, 0, &fnum1, NULL);
    9745           0 :                 if (!NT_STATUS_IS_OK(status)) {
    9746           0 :                         printf("open %d (1) of %s failed (%s)\n", i, fname, nt_errstr(status));
    9747           0 :                         return False;
    9748             :                 }
    9749             : 
    9750           0 :                 status = cli_close(cli1, fnum1);
    9751           0 :                 if (!NT_STATUS_IS_OK(status)) {
    9752           0 :                         printf("close %d (1) of %s failed (%s)\n", i, fname, nt_errstr(status));
    9753           0 :                         return False;
    9754             :                 }
    9755             : 
    9756           0 :                 for (j = 0; j < sizeof(open_attrs_table)/sizeof(uint32_t); j++) {
    9757           0 :                         status = cli_ntcreate(cli1, fname, 0,
    9758             :                                               FILE_READ_DATA|FILE_WRITE_DATA,
    9759             :                                               open_attrs_table[j],
    9760             :                                               FILE_SHARE_NONE, FILE_OVERWRITE,
    9761             :                                               0, 0, &fnum1, NULL);
    9762           0 :                         if (!NT_STATUS_IS_OK(status)) {
    9763           0 :                                 for (l = 0; l < sizeof(attr_results)/sizeof(struct trunc_open_results); l++) {
    9764           0 :                                         if (attr_results[l].num == k) {
    9765           0 :                                                 printf("[%d] trunc open 0x%x -> 0x%x of %s failed - should have succeeded !(0x%x:%s)\n",
    9766             :                                                                 k, open_attrs_table[i],
    9767             :                                                                 open_attrs_table[j],
    9768             :                                                                 fname, NT_STATUS_V(status), nt_errstr(status));
    9769           0 :                                                 correct = False;
    9770             :                                         }
    9771             :                                 }
    9772             : 
    9773           0 :                                 if (!NT_STATUS_EQUAL(status, NT_STATUS_ACCESS_DENIED)) {
    9774           0 :                                         printf("[%d] trunc open 0x%x -> 0x%x failed with wrong error code %s\n",
    9775             :                                                         k, open_attrs_table[i], open_attrs_table[j],
    9776             :                                                         nt_errstr(status));
    9777           0 :                                         correct = False;
    9778             :                                 }
    9779             : #if 0
    9780             :                                 printf("[%d] trunc open 0x%x -> 0x%x failed\n", k, open_attrs_table[i], open_attrs_table[j]);
    9781             : #endif
    9782           0 :                                 k++;
    9783           0 :                                 continue;
    9784             :                         }
    9785             : 
    9786           0 :                         status = cli_close(cli1, fnum1);
    9787           0 :                         if (!NT_STATUS_IS_OK(status)) {
    9788           0 :                                 printf("close %d (2) of %s failed (%s)\n", j, fname, nt_errstr(status));
    9789           0 :                                 return False;
    9790             :                         }
    9791             : 
    9792           0 :                         status = cli_getatr(cli1, fname, &attr, NULL, NULL);
    9793           0 :                         if (!NT_STATUS_IS_OK(status)) {
    9794           0 :                                 printf("getatr(2) failed (%s)\n", nt_errstr(status));
    9795           0 :                                 return False;
    9796             :                         }
    9797             : 
    9798             : #if 0
    9799             :                         printf("[%d] getatr check [0x%x] trunc [0x%x] got attr 0x%x\n",
    9800             :                                         k,  open_attrs_table[i],  open_attrs_table[j], attr );
    9801             : #endif
    9802             : 
    9803           0 :                         for (l = 0; l < sizeof(attr_results)/sizeof(struct trunc_open_results); l++) {
    9804           0 :                                 if (attr_results[l].num == k) {
    9805           0 :                                         if (attr != attr_results[l].result_attr ||
    9806           0 :                                                         open_attrs_table[i] != attr_results[l].init_attr ||
    9807           0 :                                                         open_attrs_table[j] != attr_results[l].trunc_attr) {
    9808           0 :                                                 printf("getatr check failed. [0x%x] trunc [0x%x] got attr 0x%x, should be 0x%x\n",
    9809             :                                                 open_attrs_table[i],
    9810             :                                                 open_attrs_table[j],
    9811             :                                                 (unsigned int)attr,
    9812             :                                                 attr_results[l].result_attr);
    9813           0 :                                                 correct = False;
    9814             :                                         }
    9815           0 :                                         break;
    9816             :                                 }
    9817             :                         }
    9818           0 :                         k++;
    9819             :                 }
    9820             :         }
    9821             : 
    9822           0 :         cli_setatr(cli1, fname, 0, 0);
    9823           0 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    9824             : 
    9825           0 :         printf("open attr test %s.\n", correct ? "passed" : "failed");
    9826             : 
    9827           0 :         if (!torture_close_connection(cli1)) {
    9828           0 :                 correct = False;
    9829             :         }
    9830           0 :         return correct;
    9831             : }
    9832             : 
    9833           0 : static NTSTATUS list_fn(struct file_info *finfo,
    9834             :                     const char *name, void *state)
    9835             : {
    9836           0 :         int *matched = (int *)state;
    9837           0 :         if (matched != NULL) {
    9838           0 :                 *matched += 1;
    9839             :         }
    9840           0 :         return NT_STATUS_OK;
    9841             : }
    9842             : 
    9843             : /*
    9844             :   test directory listing speed
    9845             :  */
    9846           5 : static bool run_dirtest(int dummy)
    9847             : {
    9848           0 :         int i;
    9849           0 :         static struct cli_state *cli;
    9850           0 :         uint16_t fnum;
    9851           0 :         struct timeval core_start;
    9852           5 :         bool correct = True;
    9853           0 :         int matched;
    9854             : 
    9855           5 :         printf("starting directory test\n");
    9856             : 
    9857           5 :         if (!torture_open_connection(&cli, 0)) {
    9858           0 :                 return False;
    9859             :         }
    9860             : 
    9861           5 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
    9862             : 
    9863           5 :         srandom(0);
    9864         505 :         for (i=0;i<torture_numops;i++) {
    9865           0 :                 fstring fname;
    9866         500 :                 slprintf(fname, sizeof(fname), "\\%x", (int)random());
    9867         500 :                 if (!NT_STATUS_IS_OK(cli_openx(cli, fname, O_RDWR|O_CREAT, DENY_NONE, &fnum))) {
    9868           0 :                         fprintf(stderr,"Failed to open %s\n", fname);
    9869           0 :                         return False;
    9870             :                 }
    9871         500 :                 cli_close(cli, fnum);
    9872             :         }
    9873             : 
    9874           5 :         core_start = timeval_current();
    9875             : 
    9876           5 :         matched = 0;
    9877           5 :         cli_list(cli, "a*.*", 0, list_fn, &matched);
    9878           5 :         printf("Matched %d\n", matched);
    9879             : 
    9880           5 :         matched = 0;
    9881           5 :         cli_list(cli, "b*.*", 0, list_fn, &matched);
    9882           5 :         printf("Matched %d\n", matched);
    9883             : 
    9884           5 :         matched = 0;
    9885           5 :         cli_list(cli, "xyzabc", 0, list_fn, &matched);
    9886           5 :         printf("Matched %d\n", matched);
    9887             : 
    9888           5 :         printf("dirtest core %g seconds\n", timeval_elapsed(&core_start));
    9889             : 
    9890           5 :         srandom(0);
    9891         505 :         for (i=0;i<torture_numops;i++) {
    9892           0 :                 fstring fname;
    9893         500 :                 slprintf(fname, sizeof(fname), "\\%x", (int)random());
    9894         500 :                 cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    9895             :         }
    9896             : 
    9897           5 :         if (!torture_close_connection(cli)) {
    9898           0 :                 correct = False;
    9899             :         }
    9900             : 
    9901           5 :         printf("finished dirtest\n");
    9902             : 
    9903           5 :         return correct;
    9904             : }
    9905             : 
    9906           0 : static NTSTATUS del_fn(struct file_info *finfo, const char *mask,
    9907             :                    void *state)
    9908             : {
    9909           0 :         struct cli_state *pcli = (struct cli_state *)state;
    9910           0 :         fstring fname;
    9911           0 :         slprintf(fname, sizeof(fname), "\\LISTDIR\\%s", finfo->name);
    9912             : 
    9913           0 :         if (strcmp(finfo->name, ".") == 0 || strcmp(finfo->name, "..") == 0)
    9914           0 :                 return NT_STATUS_OK;
    9915             : 
    9916           0 :         if (finfo->attr & FILE_ATTRIBUTE_DIRECTORY) {
    9917           0 :                 if (!NT_STATUS_IS_OK(cli_rmdir(pcli, fname)))
    9918           0 :                         printf("del_fn: failed to rmdir %s\n,", fname );
    9919             :         } else {
    9920           0 :                 if (!NT_STATUS_IS_OK(cli_unlink(pcli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN)))
    9921           0 :                         printf("del_fn: failed to unlink %s\n,", fname );
    9922             :         }
    9923           0 :         return NT_STATUS_OK;
    9924             : }
    9925             : 
    9926             : 
    9927             : /*
    9928             :    send a raw ioctl - used by the torture code
    9929             : */
    9930      327690 : static NTSTATUS cli_raw_ioctl(struct cli_state *cli,
    9931             :                               uint16_t fnum,
    9932             :                               uint32_t code,
    9933             :                               DATA_BLOB *blob)
    9934             : {
    9935           0 :         uint16_t vwv[3];
    9936           0 :         NTSTATUS status;
    9937             : 
    9938      327690 :         PUSH_LE_U16(vwv + 0, 0, fnum);
    9939      327690 :         PUSH_LE_U16(vwv + 1, 0, code >> 16);
    9940      327690 :         PUSH_LE_U16(vwv + 2, 0, (code & 0xFFFF));
    9941             : 
    9942      327690 :         status = cli_smb(talloc_tos(),
    9943             :                          cli,
    9944             :                          SMBioctl,
    9945             :                          0,
    9946             :                          3,
    9947             :                          vwv,
    9948             :                          0,
    9949             :                          NULL,
    9950             :                          NULL,
    9951             :                          0,
    9952             :                          NULL,
    9953             :                          NULL,
    9954             :                          NULL,
    9955             :                          NULL);
    9956      327690 :         if (!NT_STATUS_IS_OK(status)) {
    9957      327682 :                 return status;
    9958             :         }
    9959           8 :         *blob = data_blob_null;
    9960           8 :         return NT_STATUS_OK;
    9961             : }
    9962             : 
    9963             : /*
    9964             :   sees what IOCTLs are supported
    9965             :  */
    9966           5 : bool torture_ioctl_test(int dummy)
    9967             : {
    9968           0 :         static struct cli_state *cli;
    9969           0 :         uint16_t device, function;
    9970           0 :         uint16_t fnum;
    9971           5 :         const char *fname = "\\ioctl.dat";
    9972           0 :         DATA_BLOB blob;
    9973           0 :         NTSTATUS status;
    9974             : 
    9975           5 :         if (!torture_open_connection(&cli, 0)) {
    9976           0 :                 return False;
    9977             :         }
    9978             : 
    9979           5 :         printf("starting ioctl test\n");
    9980             : 
    9981           5 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
    9982             : 
    9983           5 :         status = cli_openx(cli, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum);
    9984           5 :         if (!NT_STATUS_IS_OK(status)) {
    9985           0 :                 printf("open of %s failed (%s)\n", fname, nt_errstr(status));
    9986           0 :                 return False;
    9987             :         }
    9988             : 
    9989           5 :         status = cli_raw_ioctl(cli, fnum, 0x2d0000 | (0x0420<<2), &blob);
    9990           5 :         printf("ioctl device info: %s\n", nt_errstr(status));
    9991             : 
    9992           5 :         status = cli_raw_ioctl(cli, fnum, IOCTL_QUERY_JOB_INFO, &blob);
    9993           5 :         printf("ioctl job info: %s\n", nt_errstr(status));
    9994             : 
    9995        1285 :         for (device=0;device<0x100;device++) {
    9996        1280 :                 printf("ioctl test with device = 0x%x\n", device);
    9997      328960 :                 for (function=0;function<0x100;function++) {
    9998      327680 :                         uint32_t code = (device<<16) | function;
    9999             : 
   10000      327680 :                         status = cli_raw_ioctl(cli, fnum, code, &blob);
   10001             : 
   10002      327680 :                         if (NT_STATUS_IS_OK(status)) {
   10003           4 :                                 printf("ioctl 0x%x OK : %d bytes\n", (int)code,
   10004           4 :                                        (int)blob.length);
   10005           4 :                                 data_blob_free(&blob);
   10006             :                         }
   10007             :                 }
   10008             :         }
   10009             : 
   10010           5 :         if (!torture_close_connection(cli)) {
   10011           0 :                 return False;
   10012             :         }
   10013             : 
   10014           5 :         return True;
   10015             : }
   10016             : 
   10017             : 
   10018             : /*
   10019             :   tries variants of chkpath
   10020             :  */
   10021           5 : bool torture_chkpath_test(int dummy)
   10022             : {
   10023           0 :         static struct cli_state *cli;
   10024           0 :         uint16_t fnum;
   10025           0 :         bool ret;
   10026           0 :         NTSTATUS status;
   10027             : 
   10028           5 :         if (!torture_open_connection(&cli, 0)) {
   10029           0 :                 return False;
   10030             :         }
   10031             : 
   10032           5 :         printf("starting chkpath test\n");
   10033             : 
   10034             :         /* cleanup from an old run */
   10035           5 :         torture_deltree(cli, "\\chkpath.dir");
   10036             : 
   10037           5 :         status = cli_mkdir(cli, "\\chkpath.dir");
   10038           5 :         if (!NT_STATUS_IS_OK(status)) {
   10039           0 :                 printf("mkdir1 failed : %s\n", nt_errstr(status));
   10040           0 :                 return False;
   10041             :         }
   10042             : 
   10043           5 :         status = cli_mkdir(cli, "\\chkpath.dir\\dir2");
   10044           5 :         if (!NT_STATUS_IS_OK(status)) {
   10045           0 :                 printf("mkdir2 failed : %s\n", nt_errstr(status));
   10046           0 :                 return False;
   10047             :         }
   10048             : 
   10049           5 :         status = cli_openx(cli, "\\chkpath.dir\\foo.txt", O_RDWR|O_CREAT|O_EXCL,
   10050             :                           DENY_NONE, &fnum);
   10051           5 :         if (!NT_STATUS_IS_OK(status)) {
   10052           0 :                 printf("open1 failed (%s)\n", nt_errstr(status));
   10053           0 :                 return False;
   10054             :         }
   10055           5 :         cli_close(cli, fnum);
   10056             : 
   10057           5 :         status = cli_chkpath(cli, "\\chkpath.dir");
   10058           5 :         if (!NT_STATUS_IS_OK(status)) {
   10059           0 :                 printf("chkpath1 failed: %s\n", nt_errstr(status));
   10060           0 :                 ret = False;
   10061             :         }
   10062             : 
   10063           5 :         status = cli_chkpath(cli, "\\chkpath.dir\\dir2");
   10064           5 :         if (!NT_STATUS_IS_OK(status)) {
   10065           0 :                 printf("chkpath2 failed: %s\n", nt_errstr(status));
   10066           0 :                 ret = False;
   10067             :         }
   10068             : 
   10069           5 :         status = cli_chkpath(cli, "\\chkpath.dir\\foo.txt");
   10070           5 :         if (!NT_STATUS_IS_OK(status)) {
   10071           5 :                 ret = check_error(__LINE__, status, ERRDOS, ERRbadpath,
   10072           5 :                                   NT_STATUS_NOT_A_DIRECTORY);
   10073             :         } else {
   10074           0 :                 printf("* chkpath on a file should fail\n");
   10075           0 :                 ret = False;
   10076             :         }
   10077             : 
   10078           5 :         status = cli_chkpath(cli, "\\chkpath.dir\\bar.txt");
   10079           5 :         if (!NT_STATUS_IS_OK(status)) {
   10080           5 :                 ret = check_error(__LINE__, status, ERRDOS, ERRbadfile,
   10081           5 :                                   NT_STATUS_OBJECT_NAME_NOT_FOUND);
   10082             :         } else {
   10083           0 :                 printf("* chkpath on a non existent file should fail\n");
   10084           0 :                 ret = False;
   10085             :         }
   10086             : 
   10087           5 :         status = cli_chkpath(cli, "\\chkpath.dir\\dirxx\\bar.txt");
   10088           5 :         if (!NT_STATUS_IS_OK(status)) {
   10089           5 :                 ret = check_error(__LINE__, status, ERRDOS, ERRbadpath,
   10090           5 :                                   NT_STATUS_OBJECT_PATH_NOT_FOUND);
   10091             :         } else {
   10092           0 :                 printf("* chkpath on a non existent component should fail\n");
   10093           0 :                 ret = False;
   10094             :         }
   10095             : 
   10096           5 :         torture_deltree(cli, "\\chkpath.dir");
   10097             : 
   10098           5 :         if (!torture_close_connection(cli)) {
   10099           0 :                 return False;
   10100             :         }
   10101             : 
   10102           5 :         return ret;
   10103             : }
   10104             : 
   10105           0 : static bool run_eatest(int dummy)
   10106             : {
   10107           0 :         static struct cli_state *cli;
   10108           0 :         const char *fname = "\\eatest.txt";
   10109           0 :         bool correct = True;
   10110           0 :         uint16_t fnum;
   10111           0 :         size_t i, num_eas;
   10112           0 :         struct ea_struct *ea_list = NULL;
   10113           0 :         TALLOC_CTX *mem_ctx = talloc_init("eatest");
   10114           0 :         NTSTATUS status;
   10115             : 
   10116           0 :         printf("starting eatest\n");
   10117             : 
   10118           0 :         if (!torture_open_connection(&cli, 0)) {
   10119           0 :                 talloc_destroy(mem_ctx);
   10120           0 :                 return False;
   10121             :         }
   10122             : 
   10123           0 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
   10124             : 
   10125           0 :         status = cli_ntcreate(cli, fname, 0,
   10126             :                               FIRST_DESIRED_ACCESS, FILE_ATTRIBUTE_ARCHIVE,
   10127             :                               FILE_SHARE_NONE, FILE_OVERWRITE_IF,
   10128             :                               0x4044, 0, &fnum, NULL);
   10129           0 :         if (!NT_STATUS_IS_OK(status)) {
   10130           0 :                 printf("open failed - %s\n", nt_errstr(status));
   10131           0 :                 talloc_destroy(mem_ctx);
   10132           0 :                 return False;
   10133             :         }
   10134             : 
   10135           0 :         for (i = 0; i < 10; i++) {
   10136           0 :                 fstring ea_name, ea_val;
   10137             : 
   10138           0 :                 slprintf(ea_name, sizeof(ea_name), "EA_%zu", i);
   10139           0 :                 memset(ea_val, (char)i+1, i+1);
   10140           0 :                 status = cli_set_ea_fnum(cli, fnum, ea_name, ea_val, i+1);
   10141           0 :                 if (!NT_STATUS_IS_OK(status)) {
   10142           0 :                         printf("ea_set of name %s failed - %s\n", ea_name,
   10143             :                                nt_errstr(status));
   10144           0 :                         talloc_destroy(mem_ctx);
   10145           0 :                         return False;
   10146             :                 }
   10147             :         }
   10148             : 
   10149           0 :         cli_close(cli, fnum);
   10150           0 :         for (i = 0; i < 10; i++) {
   10151           0 :                 fstring ea_name, ea_val;
   10152             : 
   10153           0 :                 slprintf(ea_name, sizeof(ea_name), "EA_%zu", i+10);
   10154           0 :                 memset(ea_val, (char)i+1, i+1);
   10155           0 :                 status = cli_set_ea_path(cli, fname, ea_name, ea_val, i+1);
   10156           0 :                 if (!NT_STATUS_IS_OK(status)) {
   10157           0 :                         printf("ea_set of name %s failed - %s\n", ea_name,
   10158             :                                nt_errstr(status));
   10159           0 :                         talloc_destroy(mem_ctx);
   10160           0 :                         return False;
   10161             :                 }
   10162             :         }
   10163             : 
   10164           0 :         status = cli_get_ea_list_path(cli, fname, mem_ctx, &num_eas, &ea_list);
   10165           0 :         if (!NT_STATUS_IS_OK(status)) {
   10166           0 :                 printf("ea_get list failed - %s\n", nt_errstr(status));
   10167           0 :                 correct = False;
   10168             :         }
   10169             : 
   10170           0 :         printf("num_eas = %d\n", (int)num_eas);
   10171             : 
   10172           0 :         if (num_eas != 20) {
   10173           0 :                 printf("Should be 20 EA's stored... failing.\n");
   10174           0 :                 correct = False;
   10175             :         }
   10176             : 
   10177           0 :         for (i = 0; i < num_eas; i++) {
   10178           0 :                 printf("%zu: ea_name = %s. Val = ", i, ea_list[i].name);
   10179           0 :                 dump_data(0, ea_list[i].value.data,
   10180           0 :                           ea_list[i].value.length);
   10181             :         }
   10182             : 
   10183             :         /* Setting EA's to zero length deletes them. Test this */
   10184           0 :         printf("Now deleting all EA's - case independent....\n");
   10185             : 
   10186             : #if 1
   10187           0 :         cli_set_ea_path(cli, fname, "", "", 0);
   10188             : #else
   10189             :         for (i = 0; i < 20; i++) {
   10190             :                 fstring ea_name;
   10191             :                 slprintf(ea_name, sizeof(ea_name), "ea_%d", i);
   10192             :                 status = cli_set_ea_path(cli, fname, ea_name, "", 0);
   10193             :                 if (!NT_STATUS_IS_OK(status)) {
   10194             :                         printf("ea_set of name %s failed - %s\n", ea_name,
   10195             :                                nt_errstr(status));
   10196             :                         talloc_destroy(mem_ctx);
   10197             :                         return False;
   10198             :                 }
   10199             :         }
   10200             : #endif
   10201             : 
   10202           0 :         status = cli_get_ea_list_path(cli, fname, mem_ctx, &num_eas, &ea_list);
   10203           0 :         if (!NT_STATUS_IS_OK(status)) {
   10204           0 :                 printf("ea_get list failed - %s\n", nt_errstr(status));
   10205           0 :                 correct = False;
   10206             :         }
   10207             : 
   10208           0 :         printf("num_eas = %d\n", (int)num_eas);
   10209           0 :         for (i = 0; i < num_eas; i++) {
   10210           0 :                 printf("%zu: ea_name = %s. Val = ", i, ea_list[i].name);
   10211           0 :                 dump_data(0, ea_list[i].value.data,
   10212           0 :                           ea_list[i].value.length);
   10213             :         }
   10214             : 
   10215           0 :         if (num_eas != 0) {
   10216           0 :                 printf("deleting EA's failed.\n");
   10217           0 :                 correct = False;
   10218             :         }
   10219             : 
   10220             :         /* Try and delete a non existent EA. */
   10221           0 :         status = cli_set_ea_path(cli, fname, "foo", "", 0);
   10222           0 :         if (!NT_STATUS_IS_OK(status)) {
   10223           0 :                 printf("deleting non-existent EA 'foo' should succeed. %s\n",
   10224             :                        nt_errstr(status));
   10225           0 :                 correct = False;
   10226             :         }
   10227             : 
   10228           0 :         talloc_destroy(mem_ctx);
   10229           0 :         if (!torture_close_connection(cli)) {
   10230           0 :                 correct = False;
   10231             :         }
   10232             : 
   10233           0 :         return correct;
   10234             : }
   10235             : 
   10236           0 : static bool run_dirtest1(int dummy)
   10237             : {
   10238           0 :         int i;
   10239           0 :         static struct cli_state *cli;
   10240           0 :         uint16_t fnum;
   10241           0 :         int num_seen;
   10242           0 :         bool correct = True;
   10243             : 
   10244           0 :         printf("starting directory test\n");
   10245             : 
   10246           0 :         if (!torture_open_connection(&cli, 0)) {
   10247           0 :                 return False;
   10248             :         }
   10249             : 
   10250           0 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
   10251             : 
   10252           0 :         cli_list(cli, "\\LISTDIR\\*", 0, del_fn, cli);
   10253           0 :         cli_list(cli, "\\LISTDIR\\*", FILE_ATTRIBUTE_DIRECTORY, del_fn, cli);
   10254           0 :         cli_rmdir(cli, "\\LISTDIR");
   10255           0 :         cli_mkdir(cli, "\\LISTDIR");
   10256             : 
   10257             :         /* Create 1000 files and 1000 directories. */
   10258           0 :         for (i=0;i<1000;i++) {
   10259           0 :                 fstring fname;
   10260           0 :                 slprintf(fname, sizeof(fname), "\\LISTDIR\\f%d", i);
   10261           0 :                 if (!NT_STATUS_IS_OK(cli_ntcreate(cli, fname, 0, GENERIC_ALL_ACCESS, FILE_ATTRIBUTE_ARCHIVE,
   10262             :                                    FILE_SHARE_READ|FILE_SHARE_WRITE, FILE_OVERWRITE_IF,
   10263             :                                    0, 0, &fnum, NULL))) {
   10264           0 :                         fprintf(stderr,"Failed to open %s\n", fname);
   10265           0 :                         return False;
   10266             :                 }
   10267           0 :                 cli_close(cli, fnum);
   10268             :         }
   10269           0 :         for (i=0;i<1000;i++) {
   10270           0 :                 fstring fname;
   10271           0 :                 slprintf(fname, sizeof(fname), "\\LISTDIR\\d%d", i);
   10272           0 :                 if (!NT_STATUS_IS_OK(cli_mkdir(cli, fname))) {
   10273           0 :                         fprintf(stderr,"Failed to open %s\n", fname);
   10274           0 :                         return False;
   10275             :                 }
   10276             :         }
   10277             : 
   10278             :         /* Now ensure that doing an old list sees both files and directories. */
   10279           0 :         num_seen = 0;
   10280           0 :         cli_list_old(cli, "\\LISTDIR\\*", FILE_ATTRIBUTE_DIRECTORY, list_fn, &num_seen);
   10281           0 :         printf("num_seen = %d\n", num_seen );
   10282             :         /* We should see 100 files + 1000 directories + . and .. */
   10283           0 :         if (num_seen != 2002)
   10284           0 :                 correct = False;
   10285             : 
   10286             :         /* Ensure if we have the "must have" bits we only see the
   10287             :          * relevant entries.
   10288             :          */
   10289           0 :         num_seen = 0;
   10290           0 :         cli_list_old(cli, "\\LISTDIR\\*", (FILE_ATTRIBUTE_DIRECTORY<<8)|FILE_ATTRIBUTE_DIRECTORY, list_fn, &num_seen);
   10291           0 :         printf("num_seen = %d\n", num_seen );
   10292           0 :         if (num_seen != 1002)
   10293           0 :                 correct = False;
   10294             : 
   10295           0 :         num_seen = 0;
   10296           0 :         cli_list_old(cli, "\\LISTDIR\\*", (FILE_ATTRIBUTE_ARCHIVE<<8)|FILE_ATTRIBUTE_DIRECTORY, list_fn, &num_seen);
   10297           0 :         printf("num_seen = %d\n", num_seen );
   10298           0 :         if (num_seen != 1000)
   10299           0 :                 correct = False;
   10300             : 
   10301             :         /* Delete everything. */
   10302           0 :         cli_list(cli, "\\LISTDIR\\*", 0, del_fn, cli);
   10303           0 :         cli_list(cli, "\\LISTDIR\\*", FILE_ATTRIBUTE_DIRECTORY, del_fn, cli);
   10304           0 :         cli_rmdir(cli, "\\LISTDIR");
   10305             : 
   10306             : #if 0
   10307             :         printf("Matched %d\n", cli_list(cli, "a*.*", 0, list_fn, NULL));
   10308             :         printf("Matched %d\n", cli_list(cli, "b*.*", 0, list_fn, NULL));
   10309             :         printf("Matched %d\n", cli_list(cli, "xyzabc", 0, list_fn, NULL));
   10310             : #endif
   10311             : 
   10312           0 :         if (!torture_close_connection(cli)) {
   10313           0 :                 correct = False;
   10314             :         }
   10315             : 
   10316           0 :         printf("finished dirtest1\n");
   10317             : 
   10318           0 :         return correct;
   10319             : }
   10320             : 
   10321           0 : static bool run_error_map_extract(int dummy) {
   10322             : 
   10323           0 :         static struct cli_state *c_dos;
   10324           0 :         static struct cli_state *c_nt;
   10325           0 :         NTSTATUS status;
   10326             : 
   10327           0 :         uint32_t error;
   10328             : 
   10329           0 :         uint32_t errnum;
   10330           0 :         uint8_t errclass;
   10331             : 
   10332           0 :         NTSTATUS nt_status;
   10333             : 
   10334           0 :         fstring user;
   10335             : 
   10336             :         /* NT-Error connection */
   10337             : 
   10338           0 :         disable_spnego = true;
   10339           0 :         if (!(c_nt = open_nbt_connection())) {
   10340           0 :                 disable_spnego = false;
   10341           0 :                 return False;
   10342             :         }
   10343           0 :         disable_spnego = false;
   10344             : 
   10345           0 :         status = smbXcli_negprot(c_nt->conn,
   10346           0 :                                  c_nt->timeout,
   10347             :                                  PROTOCOL_CORE,
   10348             :                                  PROTOCOL_NT1,
   10349             :                                  NULL,
   10350             :                                  NULL,
   10351             :                                  NULL);
   10352             : 
   10353           0 :         if (!NT_STATUS_IS_OK(status)) {
   10354           0 :                 printf("%s rejected the NT-error negprot (%s)\n", host,
   10355             :                        nt_errstr(status));
   10356           0 :                 cli_shutdown(c_nt);
   10357           0 :                 return False;
   10358             :         }
   10359             : 
   10360           0 :         status = cli_session_setup_anon(c_nt);
   10361           0 :         if (!NT_STATUS_IS_OK(status)) {
   10362           0 :                 printf("%s rejected the NT-error initial session setup (%s)\n",host, nt_errstr(status));
   10363           0 :                 return False;
   10364             :         }
   10365             : 
   10366             :         /* DOS-Error connection */
   10367             : 
   10368           0 :         disable_spnego = true;
   10369           0 :         force_dos_errors = true;
   10370           0 :         if (!(c_dos = open_nbt_connection())) {
   10371           0 :                 disable_spnego = false;
   10372           0 :                 force_dos_errors = false;
   10373           0 :                 return False;
   10374             :         }
   10375           0 :         disable_spnego = false;
   10376           0 :         force_dos_errors = false;
   10377             : 
   10378           0 :         status = smbXcli_negprot(c_dos->conn,
   10379           0 :                                  c_dos->timeout,
   10380             :                                  PROTOCOL_CORE,
   10381             :                                  PROTOCOL_NT1,
   10382             :                                  NULL,
   10383             :                                  NULL,
   10384             :                                  NULL);
   10385           0 :         if (!NT_STATUS_IS_OK(status)) {
   10386           0 :                 printf("%s rejected the DOS-error negprot (%s)\n", host,
   10387             :                        nt_errstr(status));
   10388           0 :                 cli_shutdown(c_dos);
   10389           0 :                 return False;
   10390             :         }
   10391             : 
   10392           0 :         status = cli_session_setup_anon(c_dos);
   10393           0 :         if (!NT_STATUS_IS_OK(status)) {
   10394           0 :                 printf("%s rejected the DOS-error initial session setup (%s)\n",
   10395             :                         host, nt_errstr(status));
   10396           0 :                 return False;
   10397             :         }
   10398             : 
   10399           0 :         c_nt->map_dos_errors = false;
   10400           0 :         c_dos->map_dos_errors = false;
   10401             : 
   10402           0 :         for (error=(0xc0000000 | 0x1); error < (0xc0000000| 0xFFF); error++) {
   10403           0 :                 struct cli_credentials *user_creds = NULL;
   10404             : 
   10405           0 :                 fstr_sprintf(user, "%X", error);
   10406             : 
   10407           0 :                 user_creds = cli_session_creds_init(talloc_tos(),
   10408             :                                                     user,
   10409             :                                                     workgroup,
   10410             :                                                     NULL, /* realm */
   10411             :                                                     password,
   10412             :                                                     false, /* use_kerberos */
   10413             :                                                     false, /* fallback_after_kerberos */
   10414             :                                                     false, /* use_ccache */
   10415             :                                                     false); /* password_is_nt_hash */
   10416           0 :                 if (user_creds == NULL) {
   10417           0 :                         printf("cli_session_creds_init(%s) failed\n", user);
   10418           0 :                         return false;
   10419             :                 }
   10420             : 
   10421           0 :                 status = cli_session_setup_creds(c_nt, user_creds);
   10422           0 :                 if (NT_STATUS_IS_OK(status)) {
   10423           0 :                         printf("/** Session setup succeeded.  This shouldn't happen...*/\n");
   10424             :                 }
   10425             : 
   10426             :                 /* Case #1: 32-bit NT errors */
   10427           0 :                 if (!NT_STATUS_IS_DOS(status)) {
   10428           0 :                         nt_status = status;
   10429             :                 } else {
   10430           0 :                         printf("/** Dos error on NT connection! (%s) */\n", 
   10431             :                                nt_errstr(status));
   10432           0 :                         nt_status = NT_STATUS(0xc0000000);
   10433             :                 }
   10434             : 
   10435           0 :                 status = cli_session_setup_creds(c_dos, user_creds);
   10436           0 :                 if (NT_STATUS_IS_OK(status)) {
   10437           0 :                         printf("/** Session setup succeeded.  This shouldn't happen...*/\n");
   10438             :                 }
   10439             : 
   10440             :                 /* Case #1: 32-bit NT errors */
   10441           0 :                 if (NT_STATUS_IS_DOS(status)) {
   10442           0 :                         printf("/** NT error on DOS connection! (%s) */\n", 
   10443             :                                nt_errstr(status));
   10444           0 :                         errnum = errclass = 0;
   10445             :                 } else {
   10446           0 :                         errclass = NT_STATUS_DOS_CLASS(status);
   10447           0 :                         errnum = NT_STATUS_DOS_CODE(status);
   10448             :                 }
   10449             : 
   10450           0 :                 if (NT_STATUS_V(nt_status) != error) { 
   10451           0 :                         printf("/*\t{ This NT error code was 'sqashed'\n\t from %s to %s \n\t during the session setup }\n*/\n", 
   10452           0 :                                get_nt_error_c_code(talloc_tos(), NT_STATUS(error)), 
   10453           0 :                                get_nt_error_c_code(talloc_tos(), nt_status));
   10454             :                 }
   10455             : 
   10456           0 :                 printf("\t{%s,\t%s,\t%s},\n", 
   10457             :                        smb_dos_err_class(errclass), 
   10458             :                        smb_dos_err_name(errclass, errnum), 
   10459           0 :                        get_nt_error_c_code(talloc_tos(), NT_STATUS(error)));
   10460             : 
   10461           0 :                 TALLOC_FREE(user_creds);
   10462             :         }
   10463           0 :         return True;
   10464             : }
   10465             : 
   10466           0 : static bool run_sesssetup_bench(int dummy)
   10467             : {
   10468           0 :         static struct cli_state *c;
   10469           0 :         const char *fname = "\\file.dat";
   10470           0 :         uint16_t fnum;
   10471           0 :         NTSTATUS status;
   10472           0 :         int i;
   10473             : 
   10474           0 :         if (!torture_open_connection(&c, 0)) {
   10475           0 :                 return false;
   10476             :         }
   10477             : 
   10478           0 :         status = cli_ntcreate(c, fname, 0, GENERIC_ALL_ACCESS|DELETE_ACCESS,
   10479             :                               FILE_ATTRIBUTE_NORMAL, 0, FILE_OVERWRITE_IF,
   10480             :                               FILE_DELETE_ON_CLOSE, 0, &fnum, NULL);
   10481           0 :         if (!NT_STATUS_IS_OK(status)) {
   10482           0 :                 d_printf("open %s failed: %s\n", fname, nt_errstr(status));
   10483           0 :                 return false;
   10484             :         }
   10485             : 
   10486           0 :         for (i=0; i<torture_numops; i++) {
   10487           0 :                 status = cli_session_setup_creds(c, torture_creds);
   10488           0 :                 if (!NT_STATUS_IS_OK(status)) {
   10489           0 :                         d_printf("(%s) cli_session_setup_creds failed: %s\n",
   10490             :                                  __location__, nt_errstr(status));
   10491           0 :                         return false;
   10492             :                 }
   10493             : 
   10494           0 :                 d_printf("\r%d   ", (int)cli_state_get_uid(c));
   10495             : 
   10496           0 :                 status = cli_ulogoff(c);
   10497           0 :                 if (!NT_STATUS_IS_OK(status)) {
   10498           0 :                         d_printf("(%s) cli_ulogoff failed: %s\n",
   10499             :                                  __location__, nt_errstr(status));
   10500           0 :                         return false;
   10501             :                 }
   10502             :         }
   10503             : 
   10504           0 :         return true;
   10505             : }
   10506             : 
   10507           9 : static bool subst_test(const char *str, const char *user, const char *domain,
   10508             :                        uid_t uid, gid_t gid, const char *expected)
   10509             : {
   10510           9 :         char *subst;
   10511           9 :         bool result = true;
   10512             : 
   10513           9 :         subst = talloc_sub_specified(talloc_tos(), str, user, NULL, domain, uid, gid);
   10514             : 
   10515           9 :         if (strcmp(subst, expected) != 0) {
   10516           0 :                 printf("sub_specified(%s, %s, %s, %d, %d) returned [%s], expected "
   10517             :                        "[%s]\n", str, user, domain, (int)uid, (int)gid, subst,
   10518             :                        expected);
   10519           0 :                 result = false;
   10520             :         }
   10521             : 
   10522           9 :         TALLOC_FREE(subst);
   10523           9 :         return result;
   10524             : }
   10525             : 
   10526           5 : static void chain1_open_completion(struct tevent_req *req)
   10527             : {
   10528           0 :         uint16_t fnum;
   10529           0 :         NTSTATUS status;
   10530           5 :         status = cli_openx_recv(req, &fnum);
   10531           5 :         TALLOC_FREE(req);
   10532             : 
   10533           5 :         d_printf("cli_openx_recv returned %s: %d\n",
   10534             :                  nt_errstr(status),
   10535           5 :                  NT_STATUS_IS_OK(status) ? fnum : -1);
   10536           5 : }
   10537             : 
   10538           5 : static void chain1_write_completion(struct tevent_req *req)
   10539             : {
   10540           0 :         size_t written;
   10541           0 :         NTSTATUS status;
   10542           5 :         status = cli_write_andx_recv(req, &written);
   10543           5 :         TALLOC_FREE(req);
   10544             : 
   10545           5 :         d_printf("cli_write_andx_recv returned %s: %d\n",
   10546             :                  nt_errstr(status),
   10547           5 :                  NT_STATUS_IS_OK(status) ? (int)written : -1);
   10548           5 : }
   10549             : 
   10550           5 : static void chain1_close_completion(struct tevent_req *req)
   10551             : {
   10552           0 :         NTSTATUS status;
   10553           5 :         bool *done = (bool *)tevent_req_callback_data_void(req);
   10554             : 
   10555           5 :         status = cli_close_recv(req);
   10556           5 :         *done = true;
   10557             : 
   10558           5 :         TALLOC_FREE(req);
   10559             : 
   10560           5 :         d_printf("cli_close returned %s\n", nt_errstr(status));
   10561           5 : }
   10562             : 
   10563           5 : static bool run_chain1(int dummy)
   10564             : {
   10565           0 :         struct cli_state *cli1;
   10566           5 :         struct tevent_context *evt = samba_tevent_context_init(NULL);
   10567           0 :         struct tevent_req *reqs[3], *smbreqs[3];
   10568           5 :         bool done = false;
   10569           5 :         const char *str = "foobar";
   10570           5 :         const char *fname = "\\test_chain";
   10571           0 :         NTSTATUS status;
   10572             : 
   10573           5 :         printf("starting chain1 test\n");
   10574           5 :         if (!torture_open_connection(&cli1, 0)) {
   10575           0 :                 return False;
   10576             :         }
   10577             : 
   10578           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
   10579             : 
   10580           5 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
   10581             : 
   10582           5 :         reqs[0] = cli_openx_create(talloc_tos(), evt, cli1, fname,
   10583             :                                   O_CREAT|O_RDWR, 0, &smbreqs[0]);
   10584           5 :         if (reqs[0] == NULL) return false;
   10585           5 :         tevent_req_set_callback(reqs[0], chain1_open_completion, NULL);
   10586             : 
   10587             : 
   10588           5 :         reqs[1] = cli_write_andx_create(talloc_tos(), evt, cli1, 0, 0,
   10589           5 :                                         (const uint8_t *)str, 0, strlen(str)+1,
   10590             :                                         smbreqs, 1, &smbreqs[1]);
   10591           5 :         if (reqs[1] == NULL) return false;
   10592           5 :         tevent_req_set_callback(reqs[1], chain1_write_completion, NULL);
   10593             : 
   10594           5 :         reqs[2] = cli_smb1_close_create(talloc_tos(), evt, cli1, 0, &smbreqs[2]);
   10595           5 :         if (reqs[2] == NULL) return false;
   10596           5 :         tevent_req_set_callback(reqs[2], chain1_close_completion, &done);
   10597             : 
   10598           5 :         status = smb1cli_req_chain_submit(smbreqs, ARRAY_SIZE(smbreqs));
   10599           5 :         if (!NT_STATUS_IS_OK(status)) {
   10600           0 :                 return false;
   10601             :         }
   10602             : 
   10603          35 :         while (!done) {
   10604          30 :                 tevent_loop_once(evt);
   10605             :         }
   10606             : 
   10607           5 :         torture_close_connection(cli1);
   10608           5 :         return True;
   10609             : }
   10610             : 
   10611           5 : static void chain2_sesssetup_completion(struct tevent_req *req)
   10612             : {
   10613           0 :         NTSTATUS status;
   10614           5 :         status = cli_session_setup_guest_recv(req);
   10615           5 :         d_printf("sesssetup returned %s\n", nt_errstr(status));
   10616           5 : }
   10617             : 
   10618           5 : static void chain2_tcon_completion(struct tevent_req *req)
   10619             : {
   10620           5 :         bool *done = (bool *)tevent_req_callback_data_void(req);
   10621           0 :         NTSTATUS status;
   10622           5 :         status = cli_tcon_andx_recv(req);
   10623           5 :         d_printf("tcon_and_x returned %s\n", nt_errstr(status));
   10624           5 :         *done = true;
   10625           5 : }
   10626             : 
   10627           5 : static bool run_chain2(int dummy)
   10628             : {
   10629           0 :         struct cli_state *cli1;
   10630           5 :         struct tevent_context *evt = samba_tevent_context_init(NULL);
   10631           0 :         struct tevent_req *reqs[2], *smbreqs[2];
   10632           5 :         bool done = false;
   10633           0 :         NTSTATUS status;
   10634           5 :         int flags = CLI_FULL_CONNECTION_FORCE_SMB1;
   10635             : 
   10636           5 :         printf("starting chain2 test\n");
   10637           5 :         status = cli_start_connection(&cli1, lp_netbios_name(), host, NULL,
   10638             :                                       port_to_use, SMB_SIGNING_DEFAULT, flags);
   10639           5 :         if (!NT_STATUS_IS_OK(status)) {
   10640           0 :                 return False;
   10641             :         }
   10642             : 
   10643           5 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
   10644             : 
   10645           5 :         reqs[0] = cli_session_setup_guest_create(talloc_tos(), evt, cli1,
   10646             :                                                  &smbreqs[0]);
   10647           5 :         if (reqs[0] == NULL) return false;
   10648           5 :         tevent_req_set_callback(reqs[0], chain2_sesssetup_completion, NULL);
   10649             : 
   10650           5 :         reqs[1] = cli_tcon_andx_create(talloc_tos(), evt, cli1, "IPC$",
   10651             :                                        "?????", NULL, 0, &smbreqs[1]);
   10652           5 :         if (reqs[1] == NULL) return false;
   10653           5 :         tevent_req_set_callback(reqs[1], chain2_tcon_completion, &done);
   10654             : 
   10655           5 :         status = smb1cli_req_chain_submit(smbreqs, ARRAY_SIZE(smbreqs));
   10656           5 :         if (!NT_STATUS_IS_OK(status)) {
   10657           0 :                 return false;
   10658             :         }
   10659             : 
   10660          30 :         while (!done) {
   10661          25 :                 tevent_loop_once(evt);
   10662             :         }
   10663             : 
   10664           5 :         torture_close_connection(cli1);
   10665           5 :         return True;
   10666             : }
   10667             : 
   10668             : 
   10669             : struct torture_createdel_state {
   10670             :         struct tevent_context *ev;
   10671             :         struct cli_state *cli;
   10672             : };
   10673             : 
   10674             : static void torture_createdel_created(struct tevent_req *subreq);
   10675             : static void torture_createdel_closed(struct tevent_req *subreq);
   10676             : 
   10677           0 : static struct tevent_req *torture_createdel_send(TALLOC_CTX *mem_ctx,
   10678             :                                                  struct tevent_context *ev,
   10679             :                                                  struct cli_state *cli,
   10680             :                                                  const char *name)
   10681             : {
   10682           0 :         struct tevent_req *req, *subreq;
   10683           0 :         struct torture_createdel_state *state;
   10684             : 
   10685           0 :         req = tevent_req_create(mem_ctx, &state,
   10686             :                                 struct torture_createdel_state);
   10687           0 :         if (req == NULL) {
   10688           0 :                 return NULL;
   10689             :         }
   10690           0 :         state->ev = ev;
   10691           0 :         state->cli = cli;
   10692             : 
   10693           0 :         subreq = cli_ntcreate_send(
   10694             :                 state, ev, cli, name, 0,
   10695             :                 FILE_READ_DATA|FILE_WRITE_DATA|DELETE_ACCESS,
   10696             :                 FILE_ATTRIBUTE_NORMAL,
   10697             :                 FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
   10698             :                 FILE_OPEN_IF, FILE_DELETE_ON_CLOSE,
   10699             :                 SMB2_IMPERSONATION_IMPERSONATION, 0);
   10700             : 
   10701           0 :         if (tevent_req_nomem(subreq, req)) {
   10702           0 :                 return tevent_req_post(req, ev);
   10703             :         }
   10704           0 :         tevent_req_set_callback(subreq, torture_createdel_created, req);
   10705           0 :         return req;
   10706             : }
   10707             : 
   10708           0 : static void torture_createdel_created(struct tevent_req *subreq)
   10709             : {
   10710           0 :         struct tevent_req *req = tevent_req_callback_data(
   10711             :                 subreq, struct tevent_req);
   10712           0 :         struct torture_createdel_state *state = tevent_req_data(
   10713             :                 req, struct torture_createdel_state);
   10714           0 :         NTSTATUS status;
   10715           0 :         uint16_t fnum;
   10716             : 
   10717           0 :         status = cli_ntcreate_recv(subreq, &fnum, NULL);
   10718           0 :         TALLOC_FREE(subreq);
   10719           0 :         if (tevent_req_nterror(req, status)) {
   10720           0 :                 DEBUG(10, ("cli_ntcreate_recv returned %s\n",
   10721             :                            nt_errstr(status)));
   10722           0 :                 return;
   10723             :         }
   10724             : 
   10725           0 :         subreq = cli_close_send(state, state->ev, state->cli, fnum, 0);
   10726           0 :         if (tevent_req_nomem(subreq, req)) {
   10727           0 :                 return;
   10728             :         }
   10729           0 :         tevent_req_set_callback(subreq, torture_createdel_closed, req);
   10730             : }
   10731             : 
   10732           0 : static void torture_createdel_closed(struct tevent_req *subreq)
   10733             : {
   10734           0 :         struct tevent_req *req = tevent_req_callback_data(
   10735             :                 subreq, struct tevent_req);
   10736           0 :         NTSTATUS status;
   10737             : 
   10738           0 :         status = cli_close_recv(subreq);
   10739           0 :         if (tevent_req_nterror(req, status)) {
   10740           0 :                 DEBUG(10, ("cli_close_recv returned %s\n", nt_errstr(status)));
   10741           0 :                 return;
   10742             :         }
   10743           0 :         tevent_req_done(req);
   10744             : }
   10745             : 
   10746           0 : static NTSTATUS torture_createdel_recv(struct tevent_req *req)
   10747             : {
   10748           0 :         return tevent_req_simple_recv_ntstatus(req);
   10749             : }
   10750             : 
   10751             : struct torture_createdels_state {
   10752             :         struct tevent_context *ev;
   10753             :         struct cli_state *cli;
   10754             :         const char *base_name;
   10755             :         int sent;
   10756             :         int received;
   10757             :         int num_files;
   10758             :         struct tevent_req **reqs;
   10759             : };
   10760             : 
   10761             : static void torture_createdels_done(struct tevent_req *subreq);
   10762             : 
   10763           0 : static struct tevent_req *torture_createdels_send(TALLOC_CTX *mem_ctx,
   10764             :                                                   struct tevent_context *ev,
   10765             :                                                   struct cli_state *cli,
   10766             :                                                   const char *base_name,
   10767             :                                                   int num_parallel,
   10768             :                                                   int num_files)
   10769             : {
   10770           0 :         struct tevent_req *req;
   10771           0 :         struct torture_createdels_state *state;
   10772           0 :         int i;
   10773             : 
   10774           0 :         req = tevent_req_create(mem_ctx, &state,
   10775             :                                 struct torture_createdels_state);
   10776           0 :         if (req == NULL) {
   10777           0 :                 return NULL;
   10778             :         }
   10779           0 :         state->ev = ev;
   10780           0 :         state->cli = cli;
   10781           0 :         state->base_name = talloc_strdup(state, base_name);
   10782           0 :         if (tevent_req_nomem(state->base_name, req)) {
   10783           0 :                 return tevent_req_post(req, ev);
   10784             :         }
   10785           0 :         state->num_files = MAX(num_parallel, num_files);
   10786           0 :         state->sent = 0;
   10787           0 :         state->received = 0;
   10788             : 
   10789           0 :         state->reqs = talloc_array(state, struct tevent_req *, num_parallel);
   10790           0 :         if (tevent_req_nomem(state->reqs, req)) {
   10791           0 :                 return tevent_req_post(req, ev);
   10792             :         }
   10793             : 
   10794           0 :         for (i=0; i<num_parallel; i++) {
   10795           0 :                 char *name;
   10796             : 
   10797           0 :                 name = talloc_asprintf(state, "%s%8.8d", state->base_name,
   10798           0 :                                        state->sent);
   10799           0 :                 if (tevent_req_nomem(name, req)) {
   10800           0 :                         return tevent_req_post(req, ev);
   10801             :                 }
   10802           0 :                 state->reqs[i] = torture_createdel_send(
   10803           0 :                         state->reqs, state->ev, state->cli, name);
   10804           0 :                 if (tevent_req_nomem(state->reqs[i], req)) {
   10805           0 :                         return tevent_req_post(req, ev);
   10806             :                 }
   10807           0 :                 name = talloc_move(state->reqs[i], &name);
   10808           0 :                 tevent_req_set_callback(state->reqs[i],
   10809             :                                         torture_createdels_done, req);
   10810           0 :                 state->sent += 1;
   10811             :         }
   10812           0 :         return req;
   10813             : }
   10814             : 
   10815           0 : static void torture_createdels_done(struct tevent_req *subreq)
   10816             : {
   10817           0 :         struct tevent_req *req = tevent_req_callback_data(
   10818             :                 subreq, struct tevent_req);
   10819           0 :         struct torture_createdels_state *state = tevent_req_data(
   10820             :                 req, struct torture_createdels_state);
   10821           0 :         size_t i, num_parallel = talloc_array_length(state->reqs);
   10822           0 :         NTSTATUS status;
   10823           0 :         char *name;
   10824             : 
   10825           0 :         status = torture_createdel_recv(subreq);
   10826           0 :         if (!NT_STATUS_IS_OK(status)){
   10827           0 :                 DEBUG(10, ("torture_createdel_recv returned %s\n",
   10828             :                            nt_errstr(status)));
   10829           0 :                 TALLOC_FREE(subreq);
   10830           0 :                 tevent_req_nterror(req, status);
   10831           0 :                 return;
   10832             :         }
   10833             : 
   10834           0 :         for (i=0; i<num_parallel; i++) {
   10835           0 :                 if (subreq == state->reqs[i]) {
   10836           0 :                         break;
   10837             :                 }
   10838             :         }
   10839           0 :         if (i == num_parallel) {
   10840           0 :                 DEBUG(10, ("received something we did not send\n"));
   10841           0 :                 tevent_req_nterror(req, NT_STATUS_INTERNAL_ERROR);
   10842           0 :                 return;
   10843             :         }
   10844           0 :         TALLOC_FREE(state->reqs[i]);
   10845             : 
   10846           0 :         if (state->sent >= state->num_files) {
   10847           0 :                 tevent_req_done(req);
   10848           0 :                 return;
   10849             :         }
   10850             : 
   10851           0 :         name = talloc_asprintf(state, "%s%8.8d", state->base_name,
   10852             :                                state->sent);
   10853           0 :         if (tevent_req_nomem(name, req)) {
   10854           0 :                 return;
   10855             :         }
   10856           0 :         state->reqs[i] = torture_createdel_send(state->reqs, state->ev,
   10857             :                                                 state->cli, name);
   10858           0 :         if (tevent_req_nomem(state->reqs[i], req)) {
   10859           0 :                 return;
   10860             :         }
   10861           0 :         name = talloc_move(state->reqs[i], &name);
   10862           0 :         tevent_req_set_callback(state->reqs[i],      torture_createdels_done, req);
   10863           0 :         state->sent += 1;
   10864             : }
   10865             : 
   10866           0 : static NTSTATUS torture_createdels_recv(struct tevent_req *req)
   10867             : {
   10868           0 :         return tevent_req_simple_recv_ntstatus(req);
   10869             : }
   10870             : 
   10871             : struct swallow_notify_state {
   10872             :         struct tevent_context *ev;
   10873             :         struct cli_state *cli;
   10874             :         uint16_t fnum;
   10875             :         uint32_t completion_filter;
   10876             :         bool recursive;
   10877             :         bool (*fn)(uint32_t action, const char *name, void *priv);
   10878             :         void *priv;
   10879             : };
   10880             : 
   10881             : static void swallow_notify_done(struct tevent_req *subreq);
   10882             : 
   10883           0 : static struct tevent_req *swallow_notify_send(TALLOC_CTX *mem_ctx,
   10884             :                                               struct tevent_context *ev,
   10885             :                                               struct cli_state *cli,
   10886             :                                               uint16_t fnum,
   10887             :                                               uint32_t completion_filter,
   10888             :                                               bool recursive,
   10889             :                                               bool (*fn)(uint32_t action,
   10890             :                                                          const char *name,
   10891             :                                                          void *priv),
   10892             :                                               void *priv)
   10893             : {
   10894           0 :         struct tevent_req *req, *subreq;
   10895           0 :         struct swallow_notify_state *state;
   10896             : 
   10897           0 :         req = tevent_req_create(mem_ctx, &state,
   10898             :                                 struct swallow_notify_state);
   10899           0 :         if (req == NULL) {
   10900           0 :                 return NULL;
   10901             :         }
   10902           0 :         state->ev = ev;
   10903           0 :         state->cli = cli;
   10904           0 :         state->fnum = fnum;
   10905           0 :         state->completion_filter = completion_filter;
   10906           0 :         state->recursive = recursive;
   10907           0 :         state->fn = fn;
   10908           0 :         state->priv = priv;
   10909             : 
   10910           0 :         subreq = cli_notify_send(state, state->ev, state->cli, state->fnum,
   10911           0 :                                  0xffff, state->completion_filter,
   10912           0 :                                  state->recursive);
   10913           0 :         if (tevent_req_nomem(subreq, req)) {
   10914           0 :                 return tevent_req_post(req, ev);
   10915             :         }
   10916           0 :         tevent_req_set_callback(subreq, swallow_notify_done, req);
   10917           0 :         return req;
   10918             : }
   10919             : 
   10920           0 : static void swallow_notify_done(struct tevent_req *subreq)
   10921             : {
   10922           0 :         struct tevent_req *req = tevent_req_callback_data(
   10923             :                 subreq, struct tevent_req);
   10924           0 :         struct swallow_notify_state *state = tevent_req_data(
   10925             :                 req, struct swallow_notify_state);
   10926           0 :         NTSTATUS status;
   10927           0 :         uint32_t i, num_changes;
   10928           0 :         struct notify_change *changes;
   10929             : 
   10930           0 :         status = cli_notify_recv(subreq, state, &num_changes, &changes);
   10931           0 :         TALLOC_FREE(subreq);
   10932           0 :         if (!NT_STATUS_IS_OK(status)) {
   10933           0 :                 DEBUG(10, ("cli_notify_recv returned %s\n",
   10934             :                            nt_errstr(status)));
   10935           0 :                 tevent_req_nterror(req, status);
   10936           0 :                 return;
   10937             :         }
   10938             : 
   10939           0 :         for (i=0; i<num_changes; i++) {
   10940           0 :                 state->fn(changes[i].action, changes[i].name, state->priv);
   10941             :         }
   10942           0 :         TALLOC_FREE(changes);
   10943             : 
   10944           0 :         subreq = cli_notify_send(state, state->ev, state->cli, state->fnum,
   10945             :                                  0xffff, state->completion_filter,
   10946           0 :                                  state->recursive);
   10947           0 :         if (tevent_req_nomem(subreq, req)) {
   10948           0 :                 return;
   10949             :         }
   10950           0 :         tevent_req_set_callback(subreq, swallow_notify_done, req);
   10951             : }
   10952             : 
   10953           0 : static bool print_notifies(uint32_t action, const char *name, void *priv)
   10954             : {
   10955           0 :         if (DEBUGLEVEL > 5) {
   10956           0 :                 d_printf("%d %s\n", (int)action, name);
   10957             :         }
   10958           0 :         return true;
   10959             : }
   10960             : 
   10961           0 : static void notify_bench_done(struct tevent_req *req)
   10962             : {
   10963           0 :         int *num_finished = (int *)tevent_req_callback_data_void(req);
   10964           0 :         *num_finished += 1;
   10965           0 : }
   10966             : 
   10967           0 : static bool run_notify_bench(int dummy)
   10968             : {
   10969           0 :         const char *dname = "\\notify-bench";
   10970           0 :         struct tevent_context *ev;
   10971           0 :         NTSTATUS status;
   10972           0 :         uint16_t dnum;
   10973           0 :         struct tevent_req *req1;
   10974           0 :         struct tevent_req *req2 = NULL;
   10975           0 :         int i, num_unc_names;
   10976           0 :         int num_finished = 0;
   10977             : 
   10978           0 :         printf("starting notify-bench test\n");
   10979             : 
   10980           0 :         if (use_multishare_conn) {
   10981           0 :                 char **unc_list;
   10982           0 :                 unc_list = file_lines_load(multishare_conn_fname,
   10983             :                                            &num_unc_names, 0, NULL);
   10984           0 :                 if (!unc_list || num_unc_names <= 0) {
   10985           0 :                         d_printf("Failed to load unc names list from '%s'\n",
   10986             :                                  multishare_conn_fname);
   10987           0 :                         return false;
   10988             :                 }
   10989           0 :                 TALLOC_FREE(unc_list);
   10990             :         } else {
   10991           0 :                 num_unc_names = 1;
   10992             :         }
   10993             : 
   10994           0 :         ev = samba_tevent_context_init(talloc_tos());
   10995           0 :         if (ev == NULL) {
   10996           0 :                 d_printf("tevent_context_init failed\n");
   10997           0 :                 return false;
   10998             :         }
   10999             : 
   11000           0 :         for (i=0; i<num_unc_names; i++) {
   11001           0 :                 struct cli_state *cli;
   11002           0 :                 char *base_fname;
   11003             : 
   11004           0 :                 base_fname = talloc_asprintf(talloc_tos(), "%s\\file%3.3d.",
   11005             :                                              dname, i);
   11006           0 :                 if (base_fname == NULL) {
   11007           0 :                         return false;
   11008             :                 }
   11009             : 
   11010           0 :                 if (!torture_open_connection(&cli, i)) {
   11011           0 :                         return false;
   11012             :                 }
   11013             : 
   11014           0 :                 status = cli_ntcreate(cli, dname, 0,
   11015             :                                       MAXIMUM_ALLOWED_ACCESS,
   11016             :                                       0, FILE_SHARE_READ|FILE_SHARE_WRITE|
   11017             :                                       FILE_SHARE_DELETE,
   11018             :                                       FILE_OPEN_IF, FILE_DIRECTORY_FILE, 0,
   11019             :                                       &dnum, NULL);
   11020             : 
   11021           0 :                 if (!NT_STATUS_IS_OK(status)) {
   11022           0 :                         d_printf("Could not create %s: %s\n", dname,
   11023             :                                  nt_errstr(status));
   11024           0 :                         return false;
   11025             :                 }
   11026             : 
   11027           0 :                 req1 = swallow_notify_send(talloc_tos(), ev, cli, dnum,
   11028             :                                            FILE_NOTIFY_CHANGE_FILE_NAME |
   11029             :                                            FILE_NOTIFY_CHANGE_DIR_NAME |
   11030             :                                            FILE_NOTIFY_CHANGE_ATTRIBUTES |
   11031             :                                            FILE_NOTIFY_CHANGE_LAST_WRITE,
   11032             :                                            false, print_notifies, NULL);
   11033           0 :                 if (req1 == NULL) {
   11034           0 :                         d_printf("Could not create notify request\n");
   11035           0 :                         return false;
   11036             :                 }
   11037             : 
   11038           0 :                 req2 = torture_createdels_send(talloc_tos(), ev, cli,
   11039             :                                                base_fname, 10, torture_numops);
   11040           0 :                 if (req2 == NULL) {
   11041           0 :                         d_printf("Could not create createdels request\n");
   11042           0 :                         return false;
   11043             :                 }
   11044           0 :                 TALLOC_FREE(base_fname);
   11045             : 
   11046           0 :                 tevent_req_set_callback(req2, notify_bench_done,
   11047             :                                         &num_finished);
   11048             :         }
   11049             : 
   11050           0 :         while (num_finished < num_unc_names) {
   11051           0 :                 int ret;
   11052           0 :                 ret = tevent_loop_once(ev);
   11053           0 :                 if (ret != 0) {
   11054           0 :                         d_printf("tevent_loop_once failed\n");
   11055           0 :                         return false;
   11056             :                 }
   11057             :         }
   11058             : 
   11059           0 :         if (!tevent_req_poll(req2, ev)) {
   11060           0 :                 d_printf("tevent_req_poll failed\n");
   11061             :         }
   11062             : 
   11063           0 :         status = torture_createdels_recv(req2);
   11064           0 :         d_printf("torture_createdels_recv returned %s\n", nt_errstr(status));
   11065             : 
   11066           0 :         return true;
   11067             : }
   11068             : 
   11069           0 : static bool run_mangle1(int dummy)
   11070             : {
   11071           0 :         struct cli_state *cli;
   11072           0 :         const char *fname = "this_is_a_long_fname_to_be_mangled.txt";
   11073           0 :         uint16_t fnum;
   11074           0 :         fstring alt_name;
   11075           0 :         NTSTATUS status;
   11076             : 
   11077           0 :         printf("starting mangle1 test\n");
   11078           0 :         if (!torture_open_connection(&cli, 0)) {
   11079           0 :                 return False;
   11080             :         }
   11081             : 
   11082           0 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
   11083             : 
   11084           0 :         status = cli_ntcreate(cli, fname, 0, GENERIC_ALL_ACCESS|DELETE_ACCESS,
   11085             :                               FILE_ATTRIBUTE_NORMAL, 0, FILE_OVERWRITE_IF,
   11086             :                               0, 0, &fnum, NULL);
   11087           0 :         if (!NT_STATUS_IS_OK(status)) {
   11088           0 :                 d_printf("open %s failed: %s\n", fname, nt_errstr(status));
   11089           0 :                 return false;
   11090             :         }
   11091           0 :         cli_close(cli, fnum);
   11092             : 
   11093           0 :         status = cli_qpathinfo_alt_name(cli, fname, alt_name);
   11094           0 :         if (!NT_STATUS_IS_OK(status)) {
   11095           0 :                 d_printf("cli_qpathinfo_alt_name failed: %s\n",
   11096             :                          nt_errstr(status));
   11097           0 :                 return false;
   11098             :         }
   11099           0 :         d_printf("alt_name: %s\n", alt_name);
   11100             : 
   11101           0 :         status = cli_openx(cli, alt_name, O_RDONLY, DENY_NONE, &fnum);
   11102           0 :         if (!NT_STATUS_IS_OK(status)) {
   11103           0 :                 d_printf("cli_openx(%s) failed: %s\n", alt_name,
   11104             :                          nt_errstr(status));
   11105           0 :                 return false;
   11106             :         }
   11107           0 :         cli_close(cli, fnum);
   11108             : 
   11109           0 :         status = cli_qpathinfo1(cli, alt_name, NULL, NULL, NULL, NULL, NULL);
   11110           0 :         if (!NT_STATUS_IS_OK(status)) {
   11111           0 :                 d_printf("cli_qpathinfo1(%s) failed: %s\n", alt_name,
   11112             :                          nt_errstr(status));
   11113           0 :                 return false;
   11114             :         }
   11115             : 
   11116           0 :         return true;
   11117             : }
   11118             : 
   11119           2 : static NTSTATUS mangle_illegal_list_shortname_fn(struct file_info *f,
   11120             :                                                  const char *mask,
   11121             :                                                  void *state)
   11122             : {
   11123           2 :         if (f->short_name == NULL) {
   11124           2 :                 return NT_STATUS_OK;
   11125             :         }
   11126             : 
   11127           0 :         if (strlen(f->short_name) == 0) {
   11128           0 :                 return NT_STATUS_OK;
   11129             :         }
   11130             : 
   11131           0 :         printf("unexpected shortname: %s\n", f->short_name);
   11132             : 
   11133           0 :         return NT_STATUS_OBJECT_NAME_INVALID;
   11134             : }
   11135             : 
   11136           2 : static NTSTATUS mangle_illegal_list_name_fn(struct file_info *f,
   11137             :                                             const char *mask,
   11138             :                                             void *state)
   11139             : {
   11140           2 :         char *name = state;
   11141             : 
   11142           2 :         printf("name: %s\n", f->name);
   11143           2 :         fstrcpy(name, f->name);
   11144           2 :         return NT_STATUS_OK;
   11145             : }
   11146             : 
   11147           2 : static bool run_mangle_illegal(int dummy)
   11148             : {
   11149           2 :         struct cli_state *cli = NULL;
   11150           2 :         struct cli_state *cli_posix = NULL;
   11151           2 :         const char *fname = "\\MANGLE_ILLEGAL\\this_is_a_long_fname_to_be_mangled.txt";
   11152           2 :         const char *illegal_fname = "MANGLE_ILLEGAL/foo:bar";
   11153           2 :         char *mangled_path = NULL;
   11154           0 :         uint16_t fnum;
   11155           0 :         fstring name;
   11156           0 :         fstring alt_name;
   11157           0 :         NTSTATUS status;
   11158             : 
   11159           2 :         printf("starting mangle-illegal test\n");
   11160             : 
   11161           2 :         if (!torture_open_connection(&cli, 0)) {
   11162           0 :                 return False;
   11163             :         }
   11164             : 
   11165           2 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
   11166             : 
   11167           2 :         if (!torture_open_connection(&cli_posix, 0)) {
   11168           0 :                 return false;
   11169             :         }
   11170             : 
   11171           2 :         smbXcli_conn_set_sockopt(cli_posix->conn, sockops);
   11172             : 
   11173           2 :         status = torture_setup_unix_extensions(cli_posix);
   11174           2 :         if (!NT_STATUS_IS_OK(status)) {
   11175           0 :                 return false;
   11176             :         }
   11177             : 
   11178           2 :         cli_rmdir(cli, "\\MANGLE_ILLEGAL");
   11179           2 :         status = cli_mkdir(cli, "\\MANGLE_ILLEGAL");
   11180           2 :         if (!NT_STATUS_IS_OK(status)) {
   11181           0 :                 printf("mkdir1 failed : %s\n", nt_errstr(status));
   11182           0 :                 return False;
   11183             :         }
   11184             : 
   11185             :         /*
   11186             :          * Create a file with illegal NTFS characters and test that we
   11187             :          * get a usable mangled name
   11188             :          */
   11189             : 
   11190           2 :         cli_setatr(cli_posix, illegal_fname, 0, 0);
   11191           2 :         cli_posix_unlink(cli_posix, illegal_fname);
   11192             : 
   11193           2 :         status = cli_posix_open(cli_posix, illegal_fname, O_RDWR|O_CREAT|O_EXCL,
   11194             :                                 0600, &fnum);
   11195           2 :         if (!NT_STATUS_IS_OK(status)) {
   11196           0 :                 printf("POSIX create of %s failed (%s)\n",
   11197             :                        illegal_fname, nt_errstr(status));
   11198           0 :                 return false;
   11199             :         }
   11200             : 
   11201           2 :         status = cli_close(cli_posix, fnum);
   11202           2 :         if (!NT_STATUS_IS_OK(status)) {
   11203           0 :                 printf("close failed (%s)\n", nt_errstr(status));
   11204           0 :                 return false;
   11205             :         }
   11206             : 
   11207           2 :         status = cli_list(cli, "\\MANGLE_ILLEGAL\\*", 0, mangle_illegal_list_name_fn, &name);
   11208           2 :         if (!NT_STATUS_IS_OK(status)) {
   11209           0 :                 d_printf("cli_list failed: %s\n", nt_errstr(status));
   11210           0 :                 return false;
   11211             :         }
   11212             : 
   11213           2 :         mangled_path = talloc_asprintf(talloc_tos(), "\\MANGLE_ILLEGAL\\%s", name);
   11214           2 :         if (mangled_path == NULL) {
   11215           0 :                 return false;
   11216             :         }
   11217             : 
   11218           2 :         status = cli_openx(cli, mangled_path, O_RDONLY, DENY_NONE, &fnum);
   11219           2 :         if (!NT_STATUS_IS_OK(status)) {
   11220           0 :                 d_printf("cli_openx(%s) failed: %s\n", mangled_path, nt_errstr(status));
   11221           0 :                 TALLOC_FREE(mangled_path);
   11222           0 :                 return false;
   11223             :         }
   11224           2 :         TALLOC_FREE(mangled_path);
   11225           2 :         cli_close(cli, fnum);
   11226             : 
   11227           2 :         cli_setatr(cli_posix, illegal_fname, 0, 0);
   11228           2 :         cli_posix_unlink(cli_posix, illegal_fname);
   11229             : 
   11230             :         /*
   11231             :          * Create a file with a long name and check that we got *no* short name.
   11232             :          */
   11233             : 
   11234           2 :         status = cli_ntcreate(cli, fname, 0, GENERIC_ALL_ACCESS|DELETE_ACCESS,
   11235             :                               FILE_ATTRIBUTE_NORMAL, 0, FILE_OVERWRITE_IF,
   11236             :                               0, 0, &fnum, NULL);
   11237           2 :         if (!NT_STATUS_IS_OK(status)) {
   11238           0 :                 d_printf("open %s failed: %s\n", fname, nt_errstr(status));
   11239           0 :                 return false;
   11240             :         }
   11241           2 :         cli_close(cli, fnum);
   11242             : 
   11243           2 :         status = cli_list(cli, fname, 0, mangle_illegal_list_shortname_fn, &alt_name);
   11244           2 :         if (!NT_STATUS_IS_OK(status)) {
   11245           0 :                 d_printf("cli_list failed\n");
   11246           0 :                 return false;
   11247             :         }
   11248             : 
   11249           2 :         cli_unlink(cli, fname, 0);
   11250           2 :         cli_rmdir(cli, "\\MANGLE_ILLEGAL");
   11251             : 
   11252           2 :         if (!torture_close_connection(cli_posix)) {
   11253           0 :                 return false;
   11254             :         }
   11255             : 
   11256           2 :         if (!torture_close_connection(cli)) {
   11257           0 :                 return false;
   11258             :         }
   11259             : 
   11260           2 :         return true;
   11261             : }
   11262             : 
   11263           0 : static size_t null_source(uint8_t *buf, size_t n, void *priv)
   11264             : {
   11265           0 :         size_t *to_pull = (size_t *)priv;
   11266           0 :         size_t thistime = *to_pull;
   11267             : 
   11268           0 :         thistime = MIN(thistime, n);
   11269           0 :         if (thistime == 0) {
   11270           0 :                 return 0;
   11271             :         }
   11272             : 
   11273           0 :         memset(buf, 0, thistime);
   11274           0 :         *to_pull -= thistime;
   11275           0 :         return thistime;
   11276             : }
   11277             : 
   11278           0 : static bool run_windows_write(int dummy)
   11279             : {
   11280           0 :         struct cli_state *cli1;
   11281           0 :         uint16_t fnum;
   11282           0 :         int i;
   11283           0 :         bool ret = false;
   11284           0 :         const char *fname = "\\writetest.txt";
   11285           0 :         struct timeval start_time;
   11286           0 :         double seconds;
   11287           0 :         double kbytes;
   11288           0 :         NTSTATUS status;
   11289             : 
   11290           0 :         printf("starting windows_write test\n");
   11291           0 :         if (!torture_open_connection(&cli1, 0)) {
   11292           0 :                 return False;
   11293             :         }
   11294             : 
   11295           0 :         status = cli_openx(cli1, fname, O_RDWR|O_CREAT|O_EXCL, DENY_NONE, &fnum);
   11296           0 :         if (!NT_STATUS_IS_OK(status)) {
   11297           0 :                 printf("open failed (%s)\n", nt_errstr(status));
   11298           0 :                 return False;
   11299             :         }
   11300             : 
   11301           0 :         smbXcli_conn_set_sockopt(cli1->conn, sockops);
   11302             : 
   11303           0 :         start_time = timeval_current();
   11304             : 
   11305           0 :         for (i=0; i<torture_numops; i++) {
   11306           0 :                 uint8_t c = 0;
   11307           0 :                 off_t start = i * torture_blocksize;
   11308           0 :                 size_t to_pull = torture_blocksize - 1;
   11309             : 
   11310           0 :                 status = cli_writeall(cli1, fnum, 0, &c,
   11311           0 :                                       start + torture_blocksize - 1, 1, NULL);
   11312           0 :                 if (!NT_STATUS_IS_OK(status)) {
   11313           0 :                         printf("cli_write failed: %s\n", nt_errstr(status));
   11314           0 :                         goto fail;
   11315             :                 }
   11316             : 
   11317           0 :                 status = cli_push(cli1, fnum, 0, i * torture_blocksize, torture_blocksize,
   11318             :                                   null_source, &to_pull);
   11319           0 :                 if (!NT_STATUS_IS_OK(status)) {
   11320           0 :                         printf("cli_push returned: %s\n", nt_errstr(status));
   11321           0 :                         goto fail;
   11322             :                 }
   11323             :         }
   11324             : 
   11325           0 :         seconds = timeval_elapsed(&start_time);
   11326           0 :         kbytes = (double)torture_blocksize * torture_numops;
   11327           0 :         kbytes /= 1024;
   11328             : 
   11329           0 :         printf("Wrote %d kbytes in %.2f seconds: %d kb/sec\n", (int)kbytes,
   11330           0 :                (double)seconds, (int)(kbytes/seconds));
   11331             : 
   11332           0 :         ret = true;
   11333           0 :  fail:
   11334           0 :         cli_close(cli1, fnum);
   11335           0 :         cli_unlink(cli1, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
   11336           0 :         torture_close_connection(cli1);
   11337           0 :         return ret;
   11338             : }
   11339             : 
   11340          64 : static size_t calc_expected_return(struct cli_state *cli, size_t len_requested)
   11341             : {
   11342          64 :         size_t max_pdu = 0x1FFFF;
   11343             : 
   11344          64 :         if (cli->server_posix_capabilities & CIFS_UNIX_LARGE_READ_CAP) {
   11345          32 :                 max_pdu = 0xFFFFFF;
   11346             :         }
   11347             : 
   11348          64 :         if (smb1cli_conn_signing_is_active(cli->conn)) {
   11349          32 :                 max_pdu = 0x1FFFF;
   11350             :         }
   11351             : 
   11352          64 :         if (smb1cli_conn_encryption_on(cli->conn)) {
   11353          16 :                 max_pdu = CLI_BUFFER_SIZE;
   11354             :         }
   11355             : 
   11356          64 :         if ((len_requested & 0xFFFF0000) == 0xFFFF0000) {
   11357           8 :                 len_requested &= 0xFFFF;
   11358             :         }
   11359             : 
   11360          64 :         return MIN(len_requested,
   11361             :                    max_pdu - (MIN_SMB_SIZE + VWV(12) + 1 /* padding byte */));
   11362             : }
   11363             : 
   11364          64 : static bool check_read_call(struct cli_state *cli,
   11365             :                             uint16_t fnum,
   11366             :                             uint8_t *buf,
   11367             :                             size_t len_requested)
   11368             : {
   11369           0 :         NTSTATUS status;
   11370          64 :         struct tevent_req *subreq = NULL;
   11371          64 :         ssize_t len_read = 0;
   11372          64 :         size_t len_expected = 0;
   11373          64 :         struct tevent_context *ev = NULL;
   11374             : 
   11375          64 :         ev = samba_tevent_context_init(talloc_tos());
   11376          64 :         if (ev == NULL) {
   11377           0 :                 return false;
   11378             :         }
   11379             : 
   11380          64 :         subreq = cli_read_andx_send(talloc_tos(),
   11381             :                                     ev,
   11382             :                                     cli,
   11383             :                                     fnum,
   11384             :                                     0,
   11385             :                                     len_requested);
   11386             : 
   11387          64 :         if (!tevent_req_poll_ntstatus(subreq, ev, &status)) {
   11388           0 :                 return false;
   11389             :         }
   11390             : 
   11391          64 :         status = cli_read_andx_recv(subreq, &len_read, &buf);
   11392          64 :         if (!NT_STATUS_IS_OK(status)) {
   11393           0 :                 d_printf("cli_read_andx_recv failed: %s\n", nt_errstr(status));
   11394           0 :                 return false;
   11395             :         }
   11396             : 
   11397          64 :         TALLOC_FREE(subreq);
   11398          64 :         TALLOC_FREE(ev);
   11399             : 
   11400          64 :         len_expected = calc_expected_return(cli, len_requested);
   11401             : 
   11402          64 :         if (len_expected > 0x10000 && len_read == 0x10000) {
   11403             :                 /* Windows servers only return a max of 0x10000,
   11404             :                    doesn't matter if you set CAP_LARGE_READX in
   11405             :                    the client sessionsetupX call or not. */
   11406          12 :                 d_printf("Windows server - returned 0x10000 on a read of 0x%x\n",
   11407             :                         (unsigned int)len_requested);
   11408          52 :         } else if (len_read != len_expected) {
   11409           0 :                 d_printf("read of 0x%x failed: got 0x%x, expected 0x%x\n",
   11410             :                         (unsigned int)len_requested,
   11411             :                         (unsigned int)len_read,
   11412             :                         (unsigned int)len_expected);
   11413           0 :                 return false;
   11414             :         } else {
   11415          52 :                 d_printf("Correct read reply.\n");
   11416             :         }
   11417             : 
   11418          64 :         return true;
   11419             : }
   11420             : 
   11421             : /* Test large readX variants. */
   11422           8 : static bool large_readx_tests(struct cli_state *cli,
   11423             :                                 uint16_t fnum,
   11424             :                                 uint8_t *buf)
   11425             : {
   11426             :         /* A read of 0xFFFF0001 should *always* return 1 byte. */
   11427           8 :         if (check_read_call(cli, fnum, buf, 0xFFFF0001) == false) {
   11428           0 :                 return false;
   11429             :         }
   11430             :         /* A read of 0x10000 should return 0x10000 bytes. */
   11431           8 :         if (check_read_call(cli, fnum, buf,    0x10000) == false) {
   11432           0 :                 return false;
   11433             :         }
   11434             :         /* A read of 0x10000 should return 0x10001 bytes. */
   11435           8 :         if (check_read_call(cli, fnum, buf,    0x10001) == false) {
   11436           0 :                 return false;
   11437             :         }
   11438             :         /* A read of 0x1FFFF - (MIN_SMB_SIZE + VWV(12) should return
   11439             :            the requested number of bytes. */
   11440           8 :         if (check_read_call(cli, fnum, buf, 0x1FFFF - (MIN_SMB_SIZE + VWV(12))) == false) {
   11441           0 :                 return false;
   11442             :         }
   11443             :         /* A read of 1MB should return 1MB bytes (on Samba). */
   11444           8 :         if (check_read_call(cli, fnum, buf,   0x100000) == false) {
   11445           0 :                 return false;
   11446             :         }
   11447             : 
   11448           8 :         if (check_read_call(cli, fnum, buf,    0x20001) == false) {
   11449           0 :                 return false;
   11450             :         }
   11451           8 :         if (check_read_call(cli, fnum, buf, 0x22000001) == false) {
   11452           0 :                 return false;
   11453             :         }
   11454           8 :         if (check_read_call(cli, fnum, buf, 0xFFFE0001) == false) {
   11455           0 :                 return false;
   11456             :         }
   11457           8 :         return true;
   11458             : }
   11459             : 
   11460           5 : static bool run_large_readx(int dummy)
   11461             : {
   11462           5 :         uint8_t *buf = NULL;
   11463           5 :         struct cli_state *cli1 = NULL;
   11464           5 :         struct cli_state *cli2 = NULL;
   11465           5 :         bool correct = false;
   11466           5 :         const char *fname = "\\large_readx.dat";
   11467           0 :         NTSTATUS status;
   11468           5 :         uint16_t fnum1 = UINT16_MAX;
   11469           5 :         uint32_t normal_caps = 0;
   11470           5 :         size_t file_size = 20*1024*1024;
   11471           5 :         TALLOC_CTX *frame = talloc_stackframe();
   11472           0 :         size_t i;
   11473           0 :         struct {
   11474             :                 const char *name;
   11475             :                 enum smb_signing_setting signing_setting;
   11476             :                 enum protocol_types protocol;
   11477           5 :         } runs[] = {
   11478             :                 {
   11479             :                         .name = "NT1",
   11480             :                         .signing_setting = SMB_SIGNING_IF_REQUIRED,
   11481             :                         .protocol = PROTOCOL_NT1,
   11482             :                 },{
   11483             :                         .name = "NT1 - SIGNING_REQUIRED",
   11484             :                         .signing_setting = SMB_SIGNING_REQUIRED,
   11485             :                         .protocol = PROTOCOL_NT1,
   11486             :                 },
   11487             :         };
   11488             : 
   11489           5 :         printf("starting large_readx test\n");
   11490             : 
   11491           5 :         if (!torture_open_connection(&cli1, 0)) {
   11492           0 :                 goto out;
   11493             :         }
   11494             : 
   11495           5 :         normal_caps = smb1cli_conn_capabilities(cli1->conn);
   11496             : 
   11497           5 :         if (!(normal_caps & CAP_LARGE_READX)) {
   11498           0 :                 d_printf("Server doesn't have CAP_LARGE_READX 0x%x\n",
   11499             :                         (unsigned int)normal_caps);
   11500           0 :                 goto out;
   11501             :         }
   11502             : 
   11503             :         /* Create a file of size 4MB. */
   11504           5 :         status = cli_ntcreate(cli1, fname, 0, GENERIC_ALL_ACCESS,
   11505             :                         FILE_ATTRIBUTE_NORMAL, 0, FILE_OVERWRITE_IF,
   11506             :                         0, 0, &fnum1, NULL);
   11507             : 
   11508           5 :         if (!NT_STATUS_IS_OK(status)) {
   11509           0 :                 d_printf("open %s failed: %s\n", fname, nt_errstr(status));
   11510           0 :                 goto out;
   11511             :         }
   11512             : 
   11513             :         /* Write file_size bytes. */
   11514           5 :         buf = talloc_zero_array(frame, uint8_t, file_size);
   11515           5 :         if (buf == NULL) {
   11516           0 :                 goto out;
   11517             :         }
   11518             : 
   11519           5 :         status = cli_writeall(cli1,
   11520             :                               fnum1,
   11521             :                               0,
   11522             :                               buf,
   11523             :                               0,
   11524             :                               file_size,
   11525             :                               NULL);
   11526           5 :         if (!NT_STATUS_IS_OK(status)) {
   11527           0 :                 d_printf("cli_writeall failed: %s\n", nt_errstr(status));
   11528           0 :                 goto out;
   11529             :         }
   11530             : 
   11531           5 :         status = cli_close(cli1, fnum1);
   11532           5 :         if (!NT_STATUS_IS_OK(status)) {
   11533           0 :                 d_printf("cli_close failed: %s\n", nt_errstr(status));
   11534           0 :                 goto out;
   11535             :         }
   11536             : 
   11537           5 :         fnum1 = UINT16_MAX;
   11538             : 
   11539          15 :         for (i=0; i < ARRAY_SIZE(runs); i++) {
   11540          10 :                 enum smb_signing_setting saved_signing_setting = signing_state;
   11541          10 :                 uint16_t fnum2 = -1;
   11542             : 
   11543          10 :                 if (do_encrypt &&
   11544           4 :                     (runs[i].signing_setting == SMB_SIGNING_REQUIRED))
   11545             :                 {
   11546           2 :                         d_printf("skip[%u] - %s\n", (unsigned)i, runs[i].name);
   11547           2 :                         continue;
   11548             :                 }
   11549             : 
   11550           8 :                 d_printf("run[%u] - %s\n", (unsigned)i, runs[i].name);
   11551             : 
   11552           8 :                 signing_state = runs[i].signing_setting;
   11553           8 :                 cli2 = open_nbt_connection();
   11554           8 :                 signing_state = saved_signing_setting;
   11555           8 :                 if (cli2 == NULL) {
   11556           0 :                         goto out;
   11557             :                 }
   11558             : 
   11559           8 :                 status = smbXcli_negprot(cli2->conn,
   11560           8 :                                          cli2->timeout,
   11561             :                                          runs[i].protocol,
   11562             :                                          runs[i].protocol,
   11563             :                                          NULL,
   11564             :                                          NULL,
   11565             :                                          NULL);
   11566           8 :                 if (!NT_STATUS_IS_OK(status)) {
   11567           0 :                         goto out;
   11568             :                 }
   11569             : 
   11570           8 :                 status = cli_session_setup_creds(cli2, torture_creds);
   11571           8 :                 if (!NT_STATUS_IS_OK(status)) {
   11572           0 :                         goto out;
   11573             :                 }
   11574             : 
   11575           8 :                 status = cli_tree_connect(cli2,
   11576             :                                         share,
   11577             :                                         "?????",
   11578             :                                         password);
   11579           8 :                 if (!NT_STATUS_IS_OK(status)) {
   11580           0 :                         goto out;
   11581             :                 }
   11582             : 
   11583           8 :                 cli_set_timeout(cli2, 120000); /* set a really long timeout (2 minutes) */
   11584             : 
   11585           8 :                 normal_caps = smb1cli_conn_capabilities(cli2->conn);
   11586             : 
   11587           8 :                 if (!(normal_caps & CAP_LARGE_READX)) {
   11588           0 :                         d_printf("Server doesn't have CAP_LARGE_READX 0x%x\n",
   11589             :                                 (unsigned int)normal_caps);
   11590           0 :                         goto out;
   11591             :                 }
   11592             : 
   11593           8 :                 if (do_encrypt) {
   11594           2 :                         if (force_cli_encryption(cli2, share) == false) {
   11595           0 :                                 goto out;
   11596             :                         }
   11597           6 :                 } else if (SERVER_HAS_UNIX_CIFS(cli2)) {
   11598           0 :                         uint16_t major, minor;
   11599           0 :                         uint32_t caplow, caphigh;
   11600             : 
   11601           4 :                         status = cli_unix_extensions_version(cli2,
   11602             :                                                              &major, &minor,
   11603             :                                                              &caplow, &caphigh);
   11604           4 :                         if (!NT_STATUS_IS_OK(status)) {
   11605           0 :                                 goto out;
   11606             :                         }
   11607             :                 }
   11608             : 
   11609           8 :                 status = cli_ntcreate(cli2, fname, 0, FILE_READ_DATA,
   11610             :                                 FILE_ATTRIBUTE_NORMAL, 0, FILE_OPEN,
   11611             :                                 0, 0, &fnum2, NULL);
   11612           8 :                 if (!NT_STATUS_IS_OK(status)) {
   11613           0 :                         d_printf("Second open %s failed: %s\n", fname, nt_errstr(status));
   11614           0 :                         goto out;
   11615             :                 }
   11616             : 
   11617             :                 /* All reads must return less than file_size bytes. */
   11618           8 :                 if (!large_readx_tests(cli2, fnum2, buf)) {
   11619           0 :                         goto out;
   11620             :                 }
   11621             : 
   11622           8 :                 status = cli_close(cli2, fnum2);
   11623           8 :                 if (!NT_STATUS_IS_OK(status)) {
   11624           0 :                         d_printf("cli_close failed: %s\n", nt_errstr(status));
   11625           0 :                         goto out;
   11626             :                 }
   11627           8 :                 fnum2 = -1;
   11628             : 
   11629           8 :                 if (!torture_close_connection(cli2)) {
   11630           0 :                         goto out;
   11631             :                 }
   11632           8 :                 cli2 = NULL;
   11633             :         }
   11634             : 
   11635           5 :         correct = true;
   11636           5 :         printf("Success on large_readx test\n");
   11637             : 
   11638           5 :   out:
   11639             : 
   11640           5 :         if (cli2) {
   11641           0 :                 if (!torture_close_connection(cli2)) {
   11642           0 :                         correct = false;
   11643             :                 }
   11644             :         }
   11645             : 
   11646           5 :         if (cli1) {
   11647           5 :                 if (fnum1 != UINT16_MAX) {
   11648           0 :                         status = cli_close(cli1, fnum1);
   11649           0 :                         if (!NT_STATUS_IS_OK(status)) {
   11650           0 :                                 d_printf("cli_close failed: %s\n", nt_errstr(status));
   11651             :                         }
   11652           0 :                         fnum1 = UINT16_MAX;
   11653             :                 }
   11654             : 
   11655           5 :                 status = cli_unlink(cli1, fname,
   11656             :                                     FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
   11657           5 :                 if (!NT_STATUS_IS_OK(status)) {
   11658           0 :                         printf("unlink failed (%s)\n", nt_errstr(status));
   11659             :                 }
   11660             : 
   11661           5 :                 if (!torture_close_connection(cli1)) {
   11662           0 :                         correct = false;
   11663             :                 }
   11664             :         }
   11665             : 
   11666           5 :         TALLOC_FREE(frame);
   11667             : 
   11668           5 :         printf("finished large_readx test\n");
   11669           5 :         return correct;
   11670             : }
   11671             : 
   11672          24 : static NTSTATUS msdfs_attribute_list_fn(struct file_info *finfo,
   11673             :                                   const char *mask,
   11674             :                                   void *private_data)
   11675             : {
   11676          24 :         uint32_t *p_attr = (uint32_t *)private_data;
   11677             : 
   11678          24 :         if (strequal(finfo->name, test_filename)) {
   11679           4 :                 *p_attr = finfo->attr;
   11680             :         }
   11681             : 
   11682          24 :         return NT_STATUS_OK;
   11683             : }
   11684             : 
   11685           4 : static bool run_msdfs_attribute(int dummy)
   11686             : {
   11687           0 :         static struct cli_state *cli;
   11688           4 :         bool correct = false;
   11689           4 :         uint32_t attr = 0;
   11690           0 :         NTSTATUS status;
   11691             : 
   11692           4 :         printf("Starting MSDFS-ATTRIBUTE test\n");
   11693             : 
   11694           4 :         if (test_filename == NULL || test_filename[0] == '\0') {
   11695           0 :                 printf("MSDFS-ATTRIBUTE test "
   11696             :                         "needs -f filename-of-msdfs-link\n");
   11697           0 :                 return false;
   11698             :         }
   11699             : 
   11700             :         /*
   11701             :          * NB. We use torture_open_connection_flags() not
   11702             :          * torture_open_connection() as the latter forces
   11703             :          * SMB1.
   11704             :          */
   11705           4 :         if (!torture_open_connection_flags(&cli, 0, 0)) {
   11706           0 :                 return false;
   11707             :         }
   11708             : 
   11709           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
   11710             : 
   11711           4 :         status = cli_list(cli,
   11712             :                         "*",
   11713             :                         FILE_ATTRIBUTE_DIRECTORY,
   11714             :                         msdfs_attribute_list_fn,
   11715             :                         &attr);
   11716             : 
   11717           4 :         if (!NT_STATUS_IS_OK(status)) {
   11718           0 :                 printf("cli_list failed with %s\n",
   11719             :                         nt_errstr(status));
   11720           0 :                 goto out;
   11721             :         }
   11722           4 :         if ((attr & FILE_ATTRIBUTE_REPARSE_POINT) == 0) {
   11723           0 :                 printf("file %s should have "
   11724             :                         "FILE_ATTRIBUTE_REPARSE_POINT set. attr = 0x%x\n",
   11725             :                         test_filename,
   11726             :                         (unsigned int)attr);
   11727           0 :                 goto out;
   11728             :         }
   11729             : 
   11730           4 :         if ((attr & FILE_ATTRIBUTE_DIRECTORY) == 0) {
   11731           0 :                 printf("file %s should have "
   11732             :                         "FILE_ATTRIBUTE_DIRECTORY set. attr = 0x%x\n",
   11733             :                         test_filename,
   11734             :                         (unsigned int)attr);
   11735           0 :                 goto out;
   11736             :         }
   11737             : 
   11738           4 :         correct = true;
   11739             : 
   11740           4 :   out:
   11741             : 
   11742           4 :         torture_close_connection(cli);
   11743           4 :         return correct;
   11744             : }
   11745             : 
   11746           0 : static bool run_cli_echo(int dummy)
   11747             : {
   11748           0 :         struct cli_state *cli;
   11749           0 :         NTSTATUS status;
   11750             : 
   11751           0 :         printf("starting cli_echo test\n");
   11752           0 :         if (!torture_open_connection(&cli, 0)) {
   11753           0 :                 return false;
   11754             :         }
   11755           0 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
   11756             : 
   11757           0 :         status = cli_echo(cli, 5, data_blob_const("hello", 5));
   11758             : 
   11759           0 :         d_printf("cli_echo returned %s\n", nt_errstr(status));
   11760             : 
   11761           0 :         torture_close_connection(cli);
   11762           0 :         return NT_STATUS_IS_OK(status);
   11763             : }
   11764             : 
   11765          14 : static int splice_status(off_t written, void *priv)
   11766             : {
   11767          14 :         return true;
   11768             : }
   11769             : 
   11770           7 : static bool run_cli_splice(int dummy)
   11771             : {
   11772           7 :         uint8_t *buf = NULL;
   11773           7 :         struct cli_state *cli1 = NULL;
   11774           7 :         bool correct = false;
   11775           7 :         const char *fname_src = "\\splice_src.dat";
   11776           7 :         const char *fname_dst = "\\splice_dst.dat";
   11777           0 :         NTSTATUS status;
   11778           7 :         uint16_t fnum1 = UINT16_MAX;
   11779           7 :         uint16_t fnum2 = UINT16_MAX;
   11780           7 :         size_t file_size = 2*1024*1024;
   11781           7 :         size_t splice_size = 1*1024*1024 + 713;
   11782           0 :         uint8_t digest1[16], digest2[16];
   11783           7 :         off_t written = 0;
   11784           7 :         size_t nread = 0;
   11785           7 :         TALLOC_CTX *frame = talloc_stackframe();
   11786             : 
   11787           7 :         printf("starting cli_splice test\n");
   11788             : 
   11789           7 :         if (!torture_open_connection(&cli1, 0)) {
   11790           0 :                 goto out;
   11791             :         }
   11792             : 
   11793           7 :         cli_unlink(cli1, fname_src,
   11794             :                 FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
   11795           7 :         cli_unlink(cli1, fname_dst,
   11796             :                 FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
   11797             : 
   11798             :         /* Create a file */
   11799           7 :         status = cli_ntcreate(cli1, fname_src, 0, GENERIC_ALL_ACCESS,
   11800             :                         FILE_ATTRIBUTE_NORMAL, 0, FILE_OVERWRITE_IF,
   11801             :                         0, 0, &fnum1, NULL);
   11802             : 
   11803           7 :         if (!NT_STATUS_IS_OK(status)) {
   11804           0 :                 d_printf("open %s failed: %s\n", fname_src, nt_errstr(status));
   11805           0 :                 goto out;
   11806             :         }
   11807             : 
   11808             :         /* Write file_size bytes - must be bigger than splice_size. */
   11809           7 :         buf = talloc_zero_array(frame, uint8_t, file_size);
   11810           7 :         if (buf == NULL) {
   11811           0 :                 d_printf("talloc_fail\n");
   11812           0 :                 goto out;
   11813             :         }
   11814             : 
   11815             :         /* Fill it with random numbers. */
   11816           7 :         generate_random_buffer(buf, file_size);
   11817             : 
   11818             :         /* MD5 the first 1MB + 713 bytes. */
   11819           7 :         gnutls_hash_fast(GNUTLS_DIG_MD5,
   11820             :                          buf,
   11821             :                          splice_size,
   11822             :                          digest1);
   11823             : 
   11824           7 :         status = cli_writeall(cli1,
   11825             :                               fnum1,
   11826             :                               0,
   11827             :                               buf,
   11828             :                               0,
   11829             :                               file_size,
   11830             :                               NULL);
   11831           7 :         if (!NT_STATUS_IS_OK(status)) {
   11832           0 :                 d_printf("cli_writeall failed: %s\n", nt_errstr(status));
   11833           0 :                 goto out;
   11834             :         }
   11835             : 
   11836           7 :         status = cli_ntcreate(cli1, fname_dst, 0, GENERIC_ALL_ACCESS,
   11837             :                         FILE_ATTRIBUTE_NORMAL, 0, FILE_OVERWRITE_IF,
   11838             :                         0, 0, &fnum2, NULL);
   11839             : 
   11840           7 :         if (!NT_STATUS_IS_OK(status)) {
   11841           0 :                 d_printf("open %s failed: %s\n", fname_dst, nt_errstr(status));
   11842           0 :                 goto out;
   11843             :         }
   11844             : 
   11845             :         /* Now splice 1MB + 713 bytes. */
   11846           7 :         status = cli_splice(cli1,
   11847             :                                 cli1,
   11848             :                                 fnum1,
   11849             :                                 fnum2,
   11850             :                                 splice_size,
   11851             :                                 0,
   11852             :                                 0,
   11853             :                                 &written,
   11854             :                                 splice_status,
   11855             :                                 NULL);
   11856             : 
   11857           7 :         if (!NT_STATUS_IS_OK(status)) {
   11858           0 :                 d_printf("cli_splice failed: %s\n", nt_errstr(status));
   11859           0 :                 goto out;
   11860             :         }
   11861             : 
   11862             :         /* Clear the old buffer. */
   11863           7 :         memset(buf, '\0', file_size);
   11864             : 
   11865             :         /* Read the new file. */
   11866           7 :         status = cli_read(cli1, fnum2, (char *)buf, 0, splice_size, &nread);
   11867           7 :         if (!NT_STATUS_IS_OK(status)) {
   11868           0 :                 d_printf("cli_read failed: %s\n", nt_errstr(status));
   11869           0 :                 goto out;
   11870             :         }
   11871           7 :         if (nread != splice_size) {
   11872           0 :                 d_printf("bad read of 0x%x, should be 0x%x\n",
   11873             :                         (unsigned int)nread,
   11874             :                         (unsigned int)splice_size);
   11875           0 :                 goto out;
   11876             :         }
   11877             : 
   11878             :         /* MD5 the first 1MB + 713 bytes. */
   11879           7 :         gnutls_hash_fast(GNUTLS_DIG_MD5,
   11880             :                          buf,
   11881             :                          splice_size,
   11882             :                          digest2);
   11883             : 
   11884             :         /* Must be the same. */
   11885           7 :         if (memcmp(digest1, digest2, 16) != 0) {
   11886           0 :                 d_printf("bad MD5 compare\n");
   11887           0 :                 goto out;
   11888             :         }
   11889             : 
   11890           7 :         correct = true;
   11891           7 :         printf("Success on cli_splice test\n");
   11892             : 
   11893           7 :   out:
   11894             : 
   11895           7 :         if (cli1) {
   11896           7 :                 if (fnum1 != UINT16_MAX) {
   11897           7 :                         cli_close(cli1, fnum1);
   11898             :                 }
   11899           7 :                 if (fnum2 != UINT16_MAX) {
   11900           7 :                         cli_close(cli1, fnum2);
   11901             :                 }
   11902             : 
   11903           7 :                 cli_unlink(cli1, fname_src,
   11904             :                         FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
   11905           7 :                 cli_unlink(cli1, fname_dst,
   11906             :                         FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
   11907             : 
   11908           7 :                 if (!torture_close_connection(cli1)) {
   11909           0 :                         correct = false;
   11910             :                 }
   11911             :         }
   11912             : 
   11913           7 :         TALLOC_FREE(frame);
   11914           7 :         return correct;
   11915             : }
   11916             : 
   11917           4 : static bool run_uid_regression_test(int dummy)
   11918             : {
   11919           0 :         static struct cli_state *cli;
   11920           0 :         int16_t old_vuid;
   11921           0 :         int32_t old_cnum;
   11922           4 :         bool correct = True;
   11923           4 :         struct smbXcli_tcon *tcon_copy = NULL;
   11924           0 :         NTSTATUS status;
   11925             : 
   11926           4 :         printf("starting uid regression test\n");
   11927             : 
   11928           4 :         if (!torture_open_connection(&cli, 0)) {
   11929           0 :                 return False;
   11930             :         }
   11931             : 
   11932           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
   11933             : 
   11934             :         /* Ok - now save then logoff our current user. */
   11935           4 :         old_vuid = cli_state_get_uid(cli);
   11936             : 
   11937           4 :         status = cli_ulogoff(cli);
   11938           4 :         if (!NT_STATUS_IS_OK(status)) {
   11939           0 :                 d_printf("(%s) cli_ulogoff failed: %s\n",
   11940             :                          __location__, nt_errstr(status));
   11941           0 :                 correct = false;
   11942           0 :                 goto out;
   11943             :         }
   11944             : 
   11945           4 :         cli_state_set_uid(cli, old_vuid);
   11946             : 
   11947             :         /* Try an operation. */
   11948           4 :         status = cli_mkdir(cli, "\\uid_reg_test");
   11949           4 :         if (NT_STATUS_IS_OK(status)) {
   11950           0 :                 d_printf("(%s) cli_mkdir succeeded\n",
   11951             :                          __location__);
   11952           0 :                 correct = false;
   11953           0 :                 goto out;
   11954             :         } else {
   11955             :                 /* Should be bad uid. */
   11956           4 :                 if (!check_error(__LINE__, status, ERRSRV, ERRbaduid,
   11957           4 :                                  NT_STATUS_USER_SESSION_DELETED)) {
   11958           0 :                         correct = false;
   11959           0 :                         goto out;
   11960             :                 }
   11961             :         }
   11962             : 
   11963           4 :         old_cnum = cli_state_get_tid(cli);
   11964             :         /*
   11965             :          * This is an SMB1-only test.
   11966             :          * Copy the tcon, not "save/restore".
   11967             :          *
   11968             :          * In SMB1 the cli_tdis() below frees
   11969             :          * cli->smb1.tcon so we need a copy
   11970             :          * of the struct to put back for the
   11971             :          * second tdis call with invalid vuid.
   11972             :          *
   11973             :          * This is a test-only hack. Real client code
   11974             :          * uses cli_state_save_tcon_share()/cli_state_restore_tcon_share().
   11975             :          */
   11976           4 :         tcon_copy = smbXcli_tcon_copy(cli, cli->smb1.tcon);
   11977           4 :         if (tcon_copy == NULL) {
   11978           0 :                 correct = false;
   11979           0 :                 goto out;
   11980             :         }
   11981             : 
   11982             :         /* Now try a SMBtdis with the invalid vuid set to zero. */
   11983           4 :         cli_state_set_uid(cli, 0);
   11984             : 
   11985             :         /* This should succeed. */
   11986           4 :         status = cli_tdis(cli);
   11987             : 
   11988           4 :         if (NT_STATUS_IS_OK(status)) {
   11989           4 :                 d_printf("First tdis with invalid vuid should succeed.\n");
   11990             :         } else {
   11991           0 :                 d_printf("First tdis failed (%s)\n", nt_errstr(status));
   11992           0 :                 correct = false;
   11993           0 :                 cli->smb1.tcon = tcon_copy;
   11994           0 :                 goto out;
   11995             :         }
   11996             : 
   11997           4 :         cli->smb1.tcon = tcon_copy;
   11998           4 :         cli_state_set_uid(cli, old_vuid);
   11999           4 :         cli_state_set_tid(cli, old_cnum);
   12000             : 
   12001             :         /* This should fail. */
   12002           4 :         status = cli_tdis(cli);
   12003           4 :         if (NT_STATUS_IS_OK(status)) {
   12004           0 :                 d_printf("Second tdis with invalid vuid should fail - succeeded instead !.\n");
   12005           0 :                 correct = false;
   12006           0 :                 goto out;
   12007             :         } else {
   12008             :                 /* Should be bad tid. */
   12009           4 :                 if (!check_error(__LINE__, status, ERRSRV, ERRinvnid,
   12010           4 :                                 NT_STATUS_NETWORK_NAME_DELETED)) {
   12011           0 :                         correct = false;
   12012           0 :                         goto out;
   12013             :                 }
   12014             :         }
   12015             : 
   12016           4 :         cli_rmdir(cli, "\\uid_reg_test");
   12017             : 
   12018           4 :   out:
   12019             : 
   12020           4 :         cli_shutdown(cli);
   12021           4 :         return correct;
   12022             : }
   12023             : 
   12024             : 
   12025             : static const char *illegal_chars = "*\\/?<>|\":";
   12026             : static char force_shortname_chars[] = " +,.[];=\177";
   12027             : 
   12028           8 : static NTSTATUS shortname_del_fn(struct file_info *finfo,
   12029             :                              const char *mask, void *state)
   12030             : {
   12031           8 :         struct cli_state *pcli = (struct cli_state *)state;
   12032           0 :         fstring fname;
   12033           8 :         NTSTATUS status = NT_STATUS_OK;
   12034             : 
   12035           8 :         slprintf(fname, sizeof(fname), "\\shortname\\%s", finfo->name);
   12036             : 
   12037           8 :         if (strcmp(finfo->name, ".") == 0 || strcmp(finfo->name, "..") == 0)
   12038           8 :                 return NT_STATUS_OK;
   12039             : 
   12040           0 :         if (finfo->attr & FILE_ATTRIBUTE_DIRECTORY) {
   12041           0 :                 status = cli_rmdir(pcli, fname);
   12042           0 :                 if (!NT_STATUS_IS_OK(status)) {
   12043           0 :                         printf("del_fn: failed to rmdir %s\n,", fname );
   12044             :                 }
   12045             :         } else {
   12046           0 :                 status = cli_unlink(pcli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
   12047           0 :                 if (!NT_STATUS_IS_OK(status)) {
   12048           0 :                         printf("del_fn: failed to unlink %s\n,", fname );
   12049             :                 }
   12050             :         }
   12051           0 :         return status;
   12052             : }
   12053             : 
   12054             : struct sn_state {
   12055             :         int matched;
   12056             :         int i;
   12057             :         bool val;
   12058             : };
   12059             : 
   12060         348 : static NTSTATUS shortname_list_fn(struct file_info *finfo,
   12061             :                               const char *name, void *state)
   12062             : {
   12063         348 :         struct sn_state *s = (struct sn_state  *)state;
   12064         348 :         int i = s->i;
   12065             : 
   12066             : #if 0
   12067             :         printf("shortname list: i = %d, name = |%s|, shortname = |%s|\n",
   12068             :                 i, finfo->name, finfo->short_name);
   12069             : #endif
   12070             : 
   12071         348 :         if (strchr(force_shortname_chars, i)) {
   12072          36 :                 if (!finfo->short_name) {
   12073             :                         /* Shortname not created when it should be. */
   12074           0 :                         d_printf("(%s) ERROR: Shortname was not created for file %s containing %d\n",
   12075             :                                 __location__, finfo->name, i);
   12076           0 :                         s->val = true;
   12077             :                 }
   12078         312 :         } else if (finfo->short_name){
   12079             :                 /* Shortname created when it should not be. */
   12080           0 :                 d_printf("(%s) ERROR: Shortname %s was created for file %s\n",
   12081             :                         __location__, finfo->short_name, finfo->name);
   12082           0 :                 s->val = true;
   12083             :         }
   12084         348 :         s->matched += 1;
   12085         348 :         return NT_STATUS_OK;
   12086             : }
   12087             : 
   12088           4 : static bool run_shortname_test(int dummy)
   12089             : {
   12090           0 :         static struct cli_state *cli;
   12091           4 :         bool correct = True;
   12092           0 :         int i;
   12093           0 :         struct sn_state s;
   12094           0 :         char fname[40];
   12095           0 :         NTSTATUS status;
   12096             : 
   12097           4 :         printf("starting shortname test\n");
   12098             : 
   12099           4 :         if (!torture_open_connection(&cli, 0)) {
   12100           0 :                 return False;
   12101             :         }
   12102             : 
   12103           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
   12104             : 
   12105           4 :         cli_list(cli, "\\shortname\\*", 0, shortname_del_fn, cli);
   12106           4 :         cli_list(cli, "\\shortname\\*", FILE_ATTRIBUTE_DIRECTORY, shortname_del_fn, cli);
   12107           4 :         cli_rmdir(cli, "\\shortname");
   12108             : 
   12109           4 :         status = cli_mkdir(cli, "\\shortname");
   12110           4 :         if (!NT_STATUS_IS_OK(status)) {
   12111           0 :                 d_printf("(%s) cli_mkdir of \\shortname failed: %s\n",
   12112             :                         __location__, nt_errstr(status));
   12113           0 :                 correct = false;
   12114           0 :                 goto out;
   12115             :         }
   12116             : 
   12117           4 :         if (strlcpy(fname, "\\shortname\\", sizeof(fname)) >= sizeof(fname)) {
   12118           0 :                 correct = false;
   12119           0 :                 goto out;
   12120             :         }
   12121           4 :         if (strlcat(fname, "test .txt", sizeof(fname)) >= sizeof(fname)) {
   12122           0 :                 correct = false;
   12123           0 :                 goto out;
   12124             :         }
   12125             : 
   12126           4 :         s.val = false;
   12127             : 
   12128         388 :         for (i = 32; i < 128; i++) {
   12129         384 :                 uint16_t fnum = (uint16_t)-1;
   12130             : 
   12131         384 :                 s.i = i;
   12132             : 
   12133         384 :                 if (strchr(illegal_chars, i)) {
   12134          36 :                         continue;
   12135             :                 }
   12136         348 :                 fname[15] = i;
   12137             : 
   12138         348 :                 status = cli_ntcreate(cli, fname, 0, GENERIC_ALL_ACCESS, FILE_ATTRIBUTE_NORMAL,
   12139             :                                    FILE_SHARE_READ|FILE_SHARE_WRITE,
   12140             :                                    FILE_OVERWRITE_IF, 0, 0, &fnum, NULL);
   12141         348 :                 if (!NT_STATUS_IS_OK(status)) {
   12142           0 :                         d_printf("(%s) cli_nt_create of %s failed: %s\n",
   12143             :                                 __location__, fname, nt_errstr(status));
   12144           0 :                         correct = false;
   12145           0 :                         goto out;
   12146             :                 }
   12147         348 :                 cli_close(cli, fnum);
   12148             : 
   12149         348 :                 s.matched = 0;
   12150         348 :                 status = cli_list(cli, "\\shortname\\test*.*", 0,
   12151             :                                   shortname_list_fn, &s);
   12152         348 :                 if (s.matched != 1) {
   12153           0 :                         d_printf("(%s) failed to list %s: %s\n",
   12154             :                                 __location__, fname, nt_errstr(status));
   12155           0 :                         correct = false;
   12156           0 :                         goto out;
   12157             :                 }
   12158             : 
   12159         348 :                 status = cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
   12160         348 :                 if (!NT_STATUS_IS_OK(status)) {
   12161           0 :                         d_printf("(%s) failed to delete %s: %s\n",
   12162             :                                 __location__, fname, nt_errstr(status));
   12163           0 :                         correct = false;
   12164           0 :                         goto out;
   12165             :                 }
   12166             : 
   12167         348 :                 if (s.val) {
   12168           0 :                         correct = false;
   12169           0 :                         goto out;
   12170             :                 }
   12171             :         }
   12172             : 
   12173           4 :   out:
   12174             : 
   12175           4 :         cli_list(cli, "\\shortname\\*", 0, shortname_del_fn, cli);
   12176           4 :         cli_list(cli, "\\shortname\\*", FILE_ATTRIBUTE_DIRECTORY, shortname_del_fn, cli);
   12177           4 :         cli_rmdir(cli, "\\shortname");
   12178           4 :         torture_close_connection(cli);
   12179           4 :         return correct;
   12180             : }
   12181             : 
   12182             : TLDAPRC callback_code;
   12183             : 
   12184         478 : static void pagedsearch_cb(struct tevent_req *req)
   12185             : {
   12186           0 :         TLDAPRC rc;
   12187           0 :         struct tldap_message *msg;
   12188           0 :         char *dn;
   12189             : 
   12190         478 :         rc = tldap_search_paged_recv(req, talloc_tos(), &msg);
   12191         478 :         if (!TLDAP_RC_IS_SUCCESS(rc)) {
   12192           0 :                 d_printf("tldap_search_paged_recv failed: %s\n",
   12193             :                          tldap_rc2string(rc));
   12194           0 :                 callback_code = rc;
   12195           0 :                 return;
   12196             :         }
   12197         478 :         if (tldap_msg_type(msg) != TLDAP_RES_SEARCH_ENTRY) {
   12198           8 :                 TALLOC_FREE(msg);
   12199           8 :                 return;
   12200             :         }
   12201         470 :         if (!tldap_entry_dn(msg, &dn)) {
   12202           0 :                 d_printf("tldap_entry_dn failed\n");
   12203           0 :                 return;
   12204             :         }
   12205         470 :         d_printf("%s\n", dn);
   12206         470 :         TALLOC_FREE(msg);
   12207             : }
   12208             : 
   12209             : enum tldap_extended_val {
   12210             :         EXTENDED_ZERO = 0,
   12211             :         EXTENDED_ONE = 1,
   12212             :         EXTENDED_NONE = 2,
   12213             : };
   12214             : 
   12215             : /*
   12216             :  * Construct an extended dn control with either no value, 0 or 1
   12217             :  *
   12218             :  * No value and 0 are equivalent (non-hyphenated GUID)
   12219             :  * 1 has the hyphenated GUID
   12220             :  */
   12221             : static struct tldap_control *
   12222           6 : tldap_build_extended_control(enum tldap_extended_val val)
   12223             : {
   12224           0 :         struct tldap_control empty_control;
   12225           0 :         struct asn1_data *data;
   12226             : 
   12227           6 :         ZERO_STRUCT(empty_control);
   12228             : 
   12229           6 :         if (val != EXTENDED_NONE) {
   12230           4 :                 data = asn1_init(talloc_tos(), ASN1_MAX_TREE_DEPTH);
   12231             : 
   12232           4 :                 if (!data) {
   12233           0 :                         return NULL;
   12234             :                 }
   12235             : 
   12236           4 :                 if (!asn1_push_tag(data, ASN1_SEQUENCE(0))) {
   12237           0 :                         return NULL;
   12238             :                 }
   12239             : 
   12240           4 :                 if (!asn1_write_Integer(data, (int)val)) {
   12241           0 :                         return NULL;
   12242             :                 }
   12243             : 
   12244           4 :                 if (!asn1_pop_tag(data)) {
   12245           0 :                         return NULL;
   12246             :                 }
   12247             : 
   12248           4 :                 if (!asn1_blob(data, &empty_control.value)) {
   12249           0 :                         return NULL;
   12250             :                 }
   12251             :         }
   12252             : 
   12253           6 :         empty_control.oid = "1.2.840.113556.1.4.529";
   12254           6 :         empty_control.critical = true;
   12255             : 
   12256           6 :         return tldap_add_control(talloc_tos(), NULL, 0, &empty_control);
   12257             : 
   12258             : }
   12259             : 
   12260           6 : static bool tldap_test_dn_guid_format(struct tldap_context *ld, const char *basedn,
   12261             :                                       enum tldap_extended_val control_val)
   12262             : {
   12263           6 :         struct tldap_control *control = tldap_build_extended_control(control_val);
   12264           6 :         char *dn = NULL;
   12265           0 :         struct tldap_message **msg;
   12266           0 :         TLDAPRC rc;
   12267             : 
   12268           6 :         rc = tldap_search(ld, basedn, TLDAP_SCOPE_BASE,
   12269             :                           "(objectClass=*)", NULL, 0, 0,
   12270             :                           control, 1, NULL,
   12271             :                           0, 0, 0, 0, talloc_tos(), &msg);
   12272           6 :         if (!TLDAP_RC_IS_SUCCESS(rc)) {
   12273           0 :                 d_printf("tldap_search for domain DN failed: %s\n",
   12274             :                          tldap_errstr(talloc_tos(), ld, rc));
   12275           0 :                 return false;
   12276             :         }
   12277             : 
   12278           6 :         if (!tldap_entry_dn(msg[0], &dn)) {
   12279           0 :                 d_printf("tldap_search domain DN fetch failed: %s\n",
   12280             :                          tldap_errstr(talloc_tos(), ld, rc));
   12281           0 :                 return false;
   12282             :         }
   12283             : 
   12284           6 :         d_printf("%s\n", dn);
   12285             :         {
   12286           0 :                 uint32_t time_low;
   12287           0 :                 uint32_t time_mid, time_hi_and_version;
   12288           0 :                 uint32_t clock_seq[2];
   12289           0 :                 uint32_t node[6];
   12290           0 :                 char next;
   12291             : 
   12292           6 :                 switch (control_val) {
   12293           4 :                 case EXTENDED_NONE:
   12294             :                 case EXTENDED_ZERO:
   12295             :                         /*
   12296             :                          * When reading GUIDs with hyphens, scanf will treat
   12297             :                          * hyphen as a hex character (and counts as part of the
   12298             :                          * width). This creates leftover GUID string which we
   12299             :                          * check will for with 'next' and closing '>'.
   12300             :                          */
   12301           4 :                         if (12 == sscanf(dn, "<GUID=%08x%04x%04x%02x%02x%02x%02x%02x%02x%02x%02x>%c",
   12302             :                                          &time_low, &time_mid,
   12303             :                                          &time_hi_and_version, &clock_seq[0],
   12304             :                                          &clock_seq[1], &node[0], &node[1],
   12305             :                                          &node[2], &node[3], &node[4],
   12306             :                                          &node[5], &next)) {
   12307             :                                 /* This GUID is good */
   12308             :                         } else {
   12309           0 :                                 d_printf("GUID format in control (no hyphens) doesn't match output\n");
   12310           0 :                                 return false;
   12311             :                         }
   12312             : 
   12313           4 :                         break;
   12314           2 :                 case EXTENDED_ONE:
   12315           2 :                         if (12 == sscanf(dn,
   12316             :                                          "<GUID=%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x>%c",
   12317             :                                          &time_low, &time_mid,
   12318             :                                          &time_hi_and_version, &clock_seq[0],
   12319             :                                          &clock_seq[1], &node[0], &node[1],
   12320             :                                          &node[2], &node[3], &node[4],
   12321             :                                          &node[5], &next)) {
   12322             :                                 /* This GUID is good */
   12323             :                         } else {
   12324           0 :                                 d_printf("GUID format in control (with hyphens) doesn't match output\n");
   12325           0 :                                 return false;
   12326             :                         }
   12327             : 
   12328           2 :                         break;
   12329           0 :                 default:
   12330           0 :                         return false;
   12331             :                 }
   12332             :         }
   12333             : 
   12334           6 :         return true;
   12335             : }
   12336             : 
   12337           2 : static bool run_tldap(int dummy)
   12338             : {
   12339           0 :         struct tldap_context *ld;
   12340           0 :         int fd;
   12341           0 :         TLDAPRC rc;
   12342           0 :         NTSTATUS status;
   12343           0 :         struct sockaddr_storage addr;
   12344           0 :         struct tevent_context *ev;
   12345           0 :         struct tevent_req *req;
   12346           0 :         char *basedn;
   12347           0 :         const char *filter;
   12348             : 
   12349           2 :         if (!resolve_name(host, &addr, 0, false)) {
   12350           0 :                 d_printf("could not find host %s\n", host);
   12351           0 :                 return false;
   12352             :         }
   12353           2 :         status = open_socket_out(&addr, 389, 9999, &fd);
   12354           2 :         if (!NT_STATUS_IS_OK(status)) {
   12355           0 :                 d_printf("open_socket_out failed: %s\n", nt_errstr(status));
   12356           0 :                 return false;
   12357             :         }
   12358             : 
   12359           2 :         ld = tldap_context_create(talloc_tos(), fd);
   12360           2 :         if (ld == NULL) {
   12361           0 :                 close(fd);
   12362           0 :                 d_printf("tldap_context_create failed\n");
   12363           0 :                 return false;
   12364             :         }
   12365             : 
   12366           2 :         rc = tldap_fetch_rootdse(ld);
   12367           2 :         if (!TLDAP_RC_IS_SUCCESS(rc)) {
   12368           0 :                 d_printf("tldap_fetch_rootdse failed: %s\n",
   12369             :                          tldap_errstr(talloc_tos(), ld, rc));
   12370           0 :                 return false;
   12371             :         }
   12372             : 
   12373           2 :         basedn = tldap_talloc_single_attribute(
   12374             :                 tldap_rootdse(ld), "defaultNamingContext", talloc_tos());
   12375           2 :         if (basedn == NULL) {
   12376           0 :                 d_printf("no defaultNamingContext\n");
   12377           0 :                 return false;
   12378             :         }
   12379           2 :         d_printf("defaultNamingContext: %s\n", basedn);
   12380             : 
   12381           2 :         ev = samba_tevent_context_init(talloc_tos());
   12382           2 :         if (ev == NULL) {
   12383           0 :                 d_printf("tevent_context_init failed\n");
   12384           0 :                 return false;
   12385             :         }
   12386             : 
   12387           2 :         rc = tldap_gensec_bind(ld, torture_creds, "ldap", host, NULL,
   12388             :                                loadparm_init_s3(talloc_tos(),
   12389             :                                                 loadparm_s3_helpers()),
   12390             :                                GENSEC_FEATURE_SIGN | GENSEC_FEATURE_SEAL);
   12391             : 
   12392           2 :         if (!TLDAP_RC_IS_SUCCESS(rc)) {
   12393           0 :                 d_printf("tldap_gensec_bind failed\n");
   12394           0 :                 return false;
   12395             :         }
   12396             : 
   12397           2 :         callback_code = TLDAP_SUCCESS;
   12398             : 
   12399           2 :         req = tldap_search_paged_send(talloc_tos(), ev, ld, basedn,
   12400             :                                       TLDAP_SCOPE_SUB, "(objectclass=*)",
   12401             :                                       NULL, 0, 0,
   12402             :                                       NULL, 0, NULL, 0, 0, 0, 0, 5);
   12403           2 :         if (req == NULL) {
   12404           0 :                 d_printf("tldap_search_paged_send failed\n");
   12405           0 :                 return false;
   12406             :         }
   12407           2 :         tevent_req_set_callback(req, pagedsearch_cb, NULL);
   12408             : 
   12409           2 :         tevent_req_poll(req, ev);
   12410             : 
   12411           2 :         TALLOC_FREE(req);
   12412             : 
   12413           2 :         rc = callback_code;
   12414             : 
   12415           2 :         if (!TLDAP_RC_IS_SUCCESS(rc)) {
   12416           0 :                 d_printf("tldap_search with paging failed: %s\n",
   12417             :                          tldap_errstr(talloc_tos(), ld, rc));
   12418           0 :                 return false;
   12419             :         }
   12420             : 
   12421             :         /* test search filters against rootDSE */
   12422           2 :         filter = "(&(|(name=samba)(nextRid<=10000000)(usnChanged>=10)(samba~=ambas)(!(name=s*m*a)))"
   12423             :                    "(|(name:=samba)(name:dn:2.5.13.5:=samba)(:dn:2.5.13.5:=samba)(!(name=*samba))))";
   12424             : 
   12425           2 :         rc = tldap_search(ld, "", TLDAP_SCOPE_BASE, filter,
   12426             :                           NULL, 0, 0, NULL, 0, NULL, 0, 0, 0, 0,
   12427             :                           talloc_tos(), NULL);
   12428           2 :         if (!TLDAP_RC_IS_SUCCESS(rc)) {
   12429           0 :                 d_printf("tldap_search with complex filter failed: %s\n",
   12430             :                          tldap_errstr(talloc_tos(), ld, rc));
   12431           0 :                 return false;
   12432             :         }
   12433             : 
   12434             :         /*
   12435             :          * Tests to check for regression of:
   12436             :          *
   12437             :          * https://bugzilla.samba.org/show_bug.cgi?id=14029
   12438             :          *
   12439             :          * TLDAP used here to pick apart the original string DN (with GUID)
   12440             :          */
   12441           2 :         if (!tldap_test_dn_guid_format(ld, basedn, EXTENDED_NONE)) {
   12442           0 :                 d_printf("tldap_search with extended dn (no val) failed: %s\n",
   12443             :                          tldap_errstr(talloc_tos(), ld, rc));
   12444           0 :                 return false;
   12445             :         }
   12446           2 :         if (!tldap_test_dn_guid_format(ld, basedn, EXTENDED_ZERO)) {
   12447           0 :                 d_printf("tldap_search with extended dn (0) failed: %s\n",
   12448             :                          tldap_errstr(talloc_tos(), ld, rc));
   12449           0 :                 return false;
   12450             :         }
   12451           2 :         if (!tldap_test_dn_guid_format(ld, basedn, EXTENDED_ONE)) {
   12452           0 :                 d_printf("tldap_search with extended dn (1) failed: %s\n",
   12453             :                          tldap_errstr(talloc_tos(), ld, rc));
   12454           0 :                 return false;
   12455             :         }
   12456             : 
   12457           2 :         TALLOC_FREE(ld);
   12458           2 :         return true;
   12459             : }
   12460             : 
   12461             : /* Torture test to ensure no regression of :
   12462             : https://bugzilla.samba.org/show_bug.cgi?id=7084
   12463             : */
   12464             : 
   12465           4 : static bool run_dir_createtime(int dummy)
   12466             : {
   12467           0 :         struct cli_state *cli;
   12468           4 :         const char *dname = "\\testdir_createtime";
   12469           4 :         const char *fname = "\\testdir_createtime\\testfile";
   12470           0 :         NTSTATUS status;
   12471           0 :         struct timespec create_time;
   12472           0 :         struct timespec create_time1;
   12473           0 :         uint16_t fnum;
   12474           4 :         bool ret = false;
   12475           0 :         uint64_t ino;
   12476             : 
   12477           4 :         if (!torture_open_connection(&cli, 0)) {
   12478           0 :                 return false;
   12479             :         }
   12480             : 
   12481           4 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
   12482             :                 /* Ensure ino is zero, SMB2 gets a real one. */
   12483           0 :                 ino = 0;
   12484             :         } else {
   12485             :                 /* Ensure ino is -1, SMB1 never gets a real one. */
   12486           4 :                 ino = (uint64_t)-1;
   12487             :         }
   12488             : 
   12489           4 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
   12490           4 :         cli_rmdir(cli, dname);
   12491             : 
   12492           4 :         status = cli_mkdir(cli, dname);
   12493           4 :         if (!NT_STATUS_IS_OK(status)) {
   12494           0 :                 printf("mkdir failed: %s\n", nt_errstr(status));
   12495           0 :                 goto out;
   12496             :         }
   12497             : 
   12498           4 :         status = cli_qpathinfo2(cli,
   12499             :                                 dname,
   12500             :                                 &create_time,
   12501             :                                 NULL,
   12502             :                                 NULL,
   12503             :                                 NULL,
   12504             :                                 NULL,
   12505             :                                 NULL,
   12506             :                                 &ino,
   12507             :                                 NULL);
   12508           4 :         if (!NT_STATUS_IS_OK(status)) {
   12509           0 :                 printf("cli_qpathinfo2 returned %s\n",
   12510             :                        nt_errstr(status));
   12511           0 :                 goto out;
   12512             :         }
   12513             : 
   12514           4 :         if (smbXcli_conn_protocol(cli->conn) >= PROTOCOL_SMB2_02) {
   12515             :                 /* SMB2 should always return an inode. */
   12516           0 :                 if (ino == 0) {
   12517           0 :                         printf("SMB2 bad inode (0)\n");
   12518           0 :                         goto out;
   12519             :                 }
   12520             :         } else {
   12521             :                 /* SMB1 must always return zero here. */
   12522           4 :                 if (ino != 0) {
   12523           0 :                         printf("SMB1 bad inode (!0)\n");
   12524           0 :                         goto out;
   12525             :                 }
   12526             :         }
   12527             : 
   12528             :         /* Sleep 3 seconds, then create a file. */
   12529           4 :         sleep(3);
   12530             : 
   12531           4 :         status = cli_openx(cli, fname, O_RDWR | O_CREAT | O_EXCL,
   12532             :                          DENY_NONE, &fnum);
   12533           4 :         if (!NT_STATUS_IS_OK(status)) {
   12534           0 :                 printf("cli_openx failed: %s\n", nt_errstr(status));
   12535           0 :                 goto out;
   12536             :         }
   12537             : 
   12538           4 :         status = cli_qpathinfo2(cli,
   12539             :                                 dname,
   12540             :                                 &create_time1,
   12541             :                                 NULL,
   12542             :                                 NULL,
   12543             :                                 NULL,
   12544             :                                 NULL,
   12545             :                                 NULL,
   12546             :                                 NULL,
   12547             :                                 NULL);
   12548           4 :         if (!NT_STATUS_IS_OK(status)) {
   12549           0 :                 printf("cli_qpathinfo2 (2) returned %s\n",
   12550             :                        nt_errstr(status));
   12551           0 :                 goto out;
   12552             :         }
   12553             : 
   12554           4 :         if (timespec_compare(&create_time1, &create_time)) {
   12555           0 :                 printf("run_dir_createtime: create time was updated (error)\n");
   12556             :         } else {
   12557           4 :                 printf("run_dir_createtime: create time was not updated (correct)\n");
   12558           4 :                 ret = true;
   12559             :         }
   12560             : 
   12561           4 :   out:
   12562             : 
   12563           4 :         cli_unlink(cli, fname, FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN);
   12564           4 :         cli_rmdir(cli, dname);
   12565           4 :         if (!torture_close_connection(cli)) {
   12566           0 :                 ret = false;
   12567             :         }
   12568           4 :         return ret;
   12569             : }
   12570             : 
   12571             : 
   12572           4 : static bool run_streamerror(int dummy)
   12573             : {
   12574           0 :         struct cli_state *cli;
   12575           4 :         const char *dname = "\\testdir_streamerror";
   12576           4 :         const char *streamname =
   12577             :                 "testdir_streamerror:{4c8cc155-6c1e-11d1-8e41-00c04fb9386d}:$DATA";
   12578           0 :         NTSTATUS status;
   12579           0 :         time_t change_time, access_time, write_time;
   12580           0 :         off_t size;
   12581           0 :         uint16_t fnum;
   12582           0 :         uint32_t attr;
   12583           4 :         bool ret = true;
   12584             : 
   12585           4 :         if (!torture_open_connection(&cli, 0)) {
   12586           0 :                 return false;
   12587             :         }
   12588             : 
   12589           4 :         torture_deltree(cli, dname);
   12590             : 
   12591           4 :         status = cli_mkdir(cli, dname);
   12592           4 :         if (!NT_STATUS_IS_OK(status)) {
   12593           0 :                 printf("mkdir failed: %s\n", nt_errstr(status));
   12594           0 :                 return false;
   12595             :         }
   12596             : 
   12597           4 :         status = cli_qpathinfo1(cli, streamname, &change_time, &access_time,
   12598             :                                 &write_time, &size, &attr);
   12599           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_NAME_NOT_FOUND)) {
   12600           0 :                 printf("pathinfo returned %s, expected "
   12601             :                        "NT_STATUS_OBJECT_NAME_NOT_FOUND\n",
   12602             :                        nt_errstr(status));
   12603           0 :                 ret = false;
   12604             :         }
   12605             : 
   12606           4 :         status = cli_ntcreate(cli, streamname, 0x16,
   12607             :                               FILE_READ_DATA|FILE_READ_EA|
   12608             :                               FILE_READ_ATTRIBUTES|READ_CONTROL_ACCESS,
   12609             :                               FILE_ATTRIBUTE_NORMAL, FILE_SHARE_READ,
   12610             :                               FILE_OPEN, 0, 0, &fnum, NULL);
   12611             : 
   12612           4 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_NAME_NOT_FOUND)) {
   12613           0 :                 printf("ntcreate returned %s, expected "
   12614             :                        "NT_STATUS_OBJECT_NAME_NOT_FOUND\n",
   12615             :                        nt_errstr(status));
   12616           0 :                 ret = false;
   12617             :         }
   12618             : 
   12619             : 
   12620           4 :         cli_rmdir(cli, dname);
   12621           4 :         return ret;
   12622             : }
   12623             : 
   12624             : struct pidtest_state {
   12625             :         bool success;
   12626             :         uint16_t vwv[1];
   12627             :         DATA_BLOB data;
   12628             : };
   12629             : 
   12630             : static void pid_echo_done(struct tevent_req *subreq);
   12631             : 
   12632           4 : static struct tevent_req *pid_echo_send(TALLOC_CTX *mem_ctx,
   12633             :                         struct tevent_context *ev,
   12634             :                         struct cli_state *cli)
   12635             : {
   12636           0 :         struct tevent_req *req, *subreq;
   12637           0 :         struct pidtest_state *state;
   12638             : 
   12639           4 :         req = tevent_req_create(mem_ctx, &state, struct pidtest_state);
   12640           4 :         if (req == NULL) {
   12641           0 :                 return NULL;
   12642             :         }
   12643             : 
   12644           4 :         SSVAL(state->vwv, 0, 1);
   12645           4 :         state->data = data_blob_const("hello", 5);
   12646             : 
   12647           4 :         subreq = smb1cli_req_send(state,
   12648             :                                 ev,
   12649             :                                 cli->conn,
   12650             :                                 SMBecho,
   12651             :                                 0, 0, /* *_flags */
   12652             :                                 0, 0, /* *_flags2 */
   12653           4 :                                 cli->timeout,
   12654             :                                 0xDEADBEEF, /* pid */
   12655             :                                 NULL, /* tcon */
   12656             :                                 NULL, /* session */
   12657           4 :                                 ARRAY_SIZE(state->vwv), state->vwv,
   12658           4 :                                 state->data.length, state->data.data);
   12659             : 
   12660           4 :         if (tevent_req_nomem(subreq, req)) {
   12661           0 :                 return tevent_req_post(req, ev);
   12662             :         }
   12663           4 :         tevent_req_set_callback(subreq, pid_echo_done, req);
   12664           4 :         return req;
   12665             : }
   12666             : 
   12667           4 : static void pid_echo_done(struct tevent_req *subreq)
   12668             : {
   12669           4 :         struct tevent_req *req = tevent_req_callback_data(
   12670             :                 subreq, struct tevent_req);
   12671           4 :         struct pidtest_state *state = tevent_req_data(
   12672             :                 req, struct pidtest_state);
   12673           0 :         NTSTATUS status;
   12674           0 :         uint32_t num_bytes;
   12675           4 :         uint8_t *bytes = NULL;
   12676           4 :         struct iovec *recv_iov = NULL;
   12677           4 :         uint8_t *phdr = NULL;
   12678           4 :         uint16_t pidlow = 0;
   12679           4 :         uint16_t pidhigh = 0;
   12680           4 :         struct smb1cli_req_expected_response expected[] = {
   12681             :         {
   12682             :                 .status = NT_STATUS_OK,
   12683             :                 .wct    = 1,
   12684             :         },
   12685             :         };
   12686             : 
   12687           4 :         status = smb1cli_req_recv(subreq, state,
   12688             :                                 &recv_iov,
   12689             :                                 &phdr,
   12690             :                                 NULL, /* pwct */
   12691             :                                 NULL, /* pvwv */
   12692             :                                 NULL, /* pvwv_offset */
   12693             :                                 &num_bytes,
   12694             :                                 &bytes,
   12695             :                                 NULL, /* pbytes_offset */
   12696             :                                 NULL, /* pinbuf */
   12697             :                                 expected, ARRAY_SIZE(expected));
   12698             : 
   12699           4 :         TALLOC_FREE(subreq);
   12700             : 
   12701           4 :         if (!NT_STATUS_IS_OK(status)) {
   12702           0 :                 tevent_req_nterror(req, status);
   12703           0 :                 return;
   12704             :         }
   12705             : 
   12706           4 :         if (num_bytes != state->data.length) {
   12707           0 :                 tevent_req_nterror(req, NT_STATUS_INVALID_NETWORK_RESPONSE);
   12708           0 :                 return;
   12709             :         }
   12710             : 
   12711           4 :         if (memcmp(bytes, state->data.data, num_bytes) != 0) {
   12712           0 :                 tevent_req_nterror(req, NT_STATUS_INVALID_NETWORK_RESPONSE);
   12713           0 :                 return;
   12714             :         }
   12715             : 
   12716             :         /* Check pid low/high == DEADBEEF */
   12717           4 :         pidlow = SVAL(phdr, HDR_PID);
   12718           4 :         if (pidlow != 0xBEEF){
   12719           0 :                 printf("Incorrect pidlow 0x%x, should be 0xBEEF\n",
   12720             :                         (unsigned int)pidlow);
   12721           0 :                 tevent_req_nterror(req, NT_STATUS_INVALID_NETWORK_RESPONSE);
   12722           0 :                 return;
   12723             :         }
   12724           4 :         pidhigh = SVAL(phdr, HDR_PIDHIGH);
   12725           4 :         if (pidhigh != 0xDEAD){
   12726           0 :                 printf("Incorrect pidhigh 0x%x, should be 0xDEAD\n",
   12727             :                         (unsigned int)pidhigh);
   12728           0 :                 tevent_req_nterror(req, NT_STATUS_INVALID_NETWORK_RESPONSE);
   12729           0 :                 return;
   12730             :         }
   12731             : 
   12732           4 :         tevent_req_done(req);
   12733             : }
   12734             : 
   12735           4 : static NTSTATUS pid_echo_recv(struct tevent_req *req)
   12736             : {
   12737           4 :         return tevent_req_simple_recv_ntstatus(req);
   12738             : }
   12739             : 
   12740           4 : static bool run_pidhigh(int dummy)
   12741             : {
   12742           4 :         bool success = false;
   12743           4 :         struct cli_state *cli = NULL;
   12744           0 :         NTSTATUS status;
   12745           4 :         struct tevent_context *ev = NULL;
   12746           4 :         struct tevent_req *req = NULL;
   12747           4 :         TALLOC_CTX *frame = talloc_stackframe();
   12748             : 
   12749           4 :         printf("starting pid high test\n");
   12750           4 :         if (!torture_open_connection(&cli, 0)) {
   12751           0 :                 return false;
   12752             :         }
   12753           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
   12754             : 
   12755           4 :         ev = samba_tevent_context_init(frame);
   12756           4 :         if (ev == NULL) {
   12757           0 :                 goto fail;
   12758             :         }
   12759             : 
   12760           4 :         req = pid_echo_send(frame, ev, cli);
   12761           4 :         if (req == NULL) {
   12762           0 :                 goto fail;
   12763             :         }
   12764             : 
   12765           4 :         if (!tevent_req_poll_ntstatus(req, ev, &status)) {
   12766           0 :                 goto fail;
   12767             :         }
   12768             : 
   12769           4 :         status = pid_echo_recv(req);
   12770           4 :         if (NT_STATUS_IS_OK(status)) {
   12771           4 :                 printf("pid high test ok\n");
   12772           4 :                 success = true;
   12773             :         }
   12774             : 
   12775           0 :  fail:
   12776             : 
   12777           4 :         TALLOC_FREE(frame);
   12778           4 :         torture_close_connection(cli);
   12779           4 :         return success;
   12780             : }
   12781             : 
   12782             : /*
   12783             :   Test Windows open on a bad POSIX symlink.
   12784             :  */
   12785           4 : static bool run_symlink_open_test(int dummy)
   12786             : {
   12787           0 :         static struct cli_state *cli;
   12788           4 :         const char *fname = "non_existant_file";
   12789           4 :         const char *sname = "dangling_symlink";
   12790           4 :         uint16_t fnum = (uint16_t)-1;
   12791           4 :         bool correct = false;
   12792           0 :         NTSTATUS status;
   12793           4 :         TALLOC_CTX *frame = NULL;
   12794             : 
   12795           4 :         frame = talloc_stackframe();
   12796             : 
   12797           4 :         printf("Starting Windows bad symlink open test\n");
   12798             : 
   12799           4 :         if (!torture_open_connection(&cli, 0)) {
   12800           0 :                 TALLOC_FREE(frame);
   12801           0 :                 return false;
   12802             :         }
   12803             : 
   12804           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
   12805             : 
   12806           4 :         status = torture_setup_unix_extensions(cli);
   12807           4 :         if (!NT_STATUS_IS_OK(status)) {
   12808           0 :                 TALLOC_FREE(frame);
   12809           0 :                 return false;
   12810             :         }
   12811             : 
   12812             :         /* Ensure nothing exists. */
   12813           4 :         cli_setatr(cli, fname, 0, 0);
   12814           4 :         cli_posix_unlink(cli, fname);
   12815           4 :         cli_setatr(cli, sname, 0, 0);
   12816           4 :         cli_posix_unlink(cli, sname);
   12817             : 
   12818             :         /* Create a symlink pointing nowhere. */
   12819           4 :         status = cli_posix_symlink(cli, fname, sname);
   12820           4 :         if (!NT_STATUS_IS_OK(status)) {
   12821           0 :                 printf("cli_posix_symlink of %s -> %s failed (%s)\n",
   12822             :                         sname,
   12823             :                         fname,
   12824             :                         nt_errstr(status));
   12825           0 :                 goto out;
   12826             :         }
   12827             : 
   12828             :         /* Now ensure that a Windows open doesn't hang. */
   12829           4 :         status = cli_ntcreate(cli,
   12830             :                         sname,
   12831             :                         0,
   12832             :                         FILE_READ_DATA|FILE_WRITE_DATA,
   12833             :                         0,
   12834             :                         FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
   12835             :                         FILE_OPEN_IF,
   12836             :                         0x0,
   12837             :                         0x0,
   12838             :                         &fnum,
   12839             :                         NULL);
   12840             : 
   12841             :         /*
   12842             :          * We get either NT_STATUS_OBJECT_NAME_NOT_FOUND or
   12843             :          * NT_STATUS_OBJECT_PATH_NOT_FOUND depending on if
   12844             :          * we use O_NOFOLLOW on the server or not.
   12845             :          */
   12846           4 :         if (NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_NAME_NOT_FOUND) ||
   12847           0 :             NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_PATH_NOT_FOUND))
   12848             :         {
   12849           4 :                 correct = true;
   12850             :         } else {
   12851           0 :                 printf("cli_ntcreate of %s returned %s - should return"
   12852             :                                 " either (%s) or (%s)\n",
   12853             :                         sname,
   12854             :                         nt_errstr(status),
   12855           0 :                         nt_errstr(NT_STATUS_OBJECT_NAME_NOT_FOUND),
   12856           0 :                         nt_errstr(NT_STATUS_OBJECT_PATH_NOT_FOUND));
   12857           0 :                 goto out;
   12858             :         }
   12859             : 
   12860           4 :         correct = true;
   12861             : 
   12862           4 :   out:
   12863             : 
   12864           4 :         if (fnum != (uint16_t)-1) {
   12865           0 :                 cli_close(cli, fnum);
   12866           0 :                 fnum = (uint16_t)-1;
   12867             :         }
   12868             : 
   12869           4 :         cli_setatr(cli, sname, 0, 0);
   12870           4 :         cli_posix_unlink(cli, sname);
   12871           4 :         cli_setatr(cli, fname, 0, 0);
   12872           4 :         cli_posix_unlink(cli, fname);
   12873             : 
   12874           4 :         if (!torture_close_connection(cli)) {
   12875           0 :                 correct = false;
   12876             :         }
   12877             : 
   12878           4 :         TALLOC_FREE(frame);
   12879           4 :         return correct;
   12880             : }
   12881             : 
   12882          16 : static NTSTATUS smb1_wild_mangle_list_fn(struct file_info *finfo,
   12883             :                                         const char *name,
   12884             :                                         void *state)
   12885             : {
   12886          16 :         char **mangled_name_return = (char **)state;
   12887          16 :         bool is_mangled = strchr(finfo->name, '~');
   12888             : 
   12889          16 :         if (is_mangled) {
   12890           8 :                 *mangled_name_return = talloc_strdup(NULL, finfo->name);
   12891           8 :                 if (*mangled_name_return == NULL) {
   12892           0 :                         return NT_STATUS_NO_MEMORY;
   12893             :                 }
   12894             :         }
   12895          16 :         return NT_STATUS_OK;
   12896             : }
   12897             : 
   12898           4 : static bool run_smb1_wild_mangle_unlink_test(int dummy)
   12899             : {
   12900           0 :         static struct cli_state *cli_posix = NULL;
   12901           0 :         static struct cli_state *cli = NULL;
   12902           4 :         uint16_t fnum = (uint16_t)-1;
   12903           4 :         bool correct = false;
   12904           4 :         const char *dname = "smb1_wild_mangle_unlink";
   12905           4 :         const char *aname = "smb1_wild_mangle_unlink/a";
   12906           4 :         const char *star_name = "smb1_wild_mangle_unlink/*";
   12907           4 :         char *windows_unlink_name = NULL;
   12908           4 :         char *mangled_name = NULL;
   12909           0 :         NTSTATUS status;
   12910             : 
   12911           4 :         printf("Starting SMB1 wild mangle unlink test\n");
   12912             : 
   12913             :         /* Open a Windows connection. */
   12914           4 :         if (!torture_open_connection(&cli, 0)) {
   12915           0 :                 return false;
   12916             :         }
   12917             : 
   12918           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
   12919             : 
   12920             :         /* Open a POSIX connection. */
   12921           4 :         if (!torture_open_connection(&cli_posix, 0)) {
   12922           0 :                 goto out;
   12923             :         }
   12924             : 
   12925           4 :         smbXcli_conn_set_sockopt(cli_posix->conn, sockops);
   12926             : 
   12927           4 :         status = torture_setup_unix_extensions(cli_posix);
   12928           4 :         if (!NT_STATUS_IS_OK(status)) {
   12929           0 :                 printf("server doesn't support POSIX\n");
   12930           0 :                 goto out;
   12931             :         }
   12932             : 
   12933             :         /* Start fresh. */
   12934           4 :         torture_deltree(cli, dname);
   12935             : 
   12936             :         /*
   12937             :          * Create two files - 'a' and '*'.
   12938             :          * We need POSIX extensions for this as '*'
   12939             :          * is not a valid Windows name.
   12940             :          */
   12941             : 
   12942           4 :         status = cli_mkdir(cli, dname);
   12943           4 :         if (!NT_STATUS_IS_OK(status)) {
   12944           0 :                 printf("cli_mkdir of %s returned %s\n",
   12945             :                         dname,
   12946             :                         nt_errstr(status));
   12947           0 :                 goto out;
   12948             :         }
   12949             : 
   12950           4 :         status = cli_posix_open(cli_posix,
   12951             :                                 aname,
   12952             :                                 O_RDWR|O_CREAT|O_EXCL,
   12953             :                                 0660,
   12954             :                                 &fnum);
   12955           4 :         if (!NT_STATUS_IS_OK(status)) {
   12956           0 :                 printf("cli_posix_open (create) of %s returned %s\n",
   12957             :                         aname,
   12958             :                         nt_errstr(status));
   12959           0 :                 goto out;
   12960             :         }
   12961           4 :         status = cli_close(cli_posix, fnum);
   12962           4 :         if (!NT_STATUS_IS_OK(status)) {
   12963           0 :                 goto out;
   12964             :         }
   12965           4 :         status = cli_posix_open(cli_posix,
   12966             :                                 star_name,
   12967             :                                 O_RDWR|O_CREAT|O_EXCL,
   12968             :                                 0660,
   12969             :                                 &fnum);
   12970           4 :         if (!NT_STATUS_IS_OK(status)) {
   12971           0 :                 printf("cli_posix_open (create) of %s returned %s\n",
   12972             :                         star_name,
   12973             :                         nt_errstr(status));
   12974           0 :                 goto out;
   12975             :         }
   12976           4 :         status = cli_close(cli_posix, fnum);
   12977           4 :         if (!NT_STATUS_IS_OK(status)) {
   12978           0 :                 goto out;
   12979             :         }
   12980             : 
   12981           4 :         status = cli_list(cli,
   12982             :                         star_name,
   12983             :                         0,
   12984             :                         smb1_wild_mangle_list_fn,
   12985             :                         &mangled_name);
   12986           4 :         if (!NT_STATUS_IS_OK(status)) {
   12987           0 :                 printf("cli_list of %s returned %s\n",
   12988             :                         star_name,
   12989             :                         nt_errstr(status));
   12990           0 :                 goto out;
   12991             :         }
   12992             : 
   12993           4 :         if (mangled_name == NULL) {
   12994           0 :                 goto out;
   12995             :         }
   12996             : 
   12997           4 :         printf("mangled_name = %s\n",
   12998             :                 mangled_name);
   12999             : 
   13000             :         /*
   13001             :          * Try a Windows unlink with the mangled name.
   13002             :          * This should *NOT* unlink the 'a' name.
   13003             :          */
   13004             : 
   13005           4 :         windows_unlink_name = talloc_asprintf(cli_posix,
   13006             :                                         "%s\\%s",
   13007             :                                         dname,
   13008             :                                         mangled_name);
   13009             : 
   13010           4 :         status = cli_unlink(cli, windows_unlink_name, 0);
   13011           4 :         if (!NT_STATUS_IS_OK(status)) {
   13012           0 :                 printf("cli_unlink of %s returned %s\n",
   13013             :                         windows_unlink_name,
   13014             :                         nt_errstr(status));
   13015           0 :                 goto out;
   13016             :         }
   13017             : 
   13018             :         /* Does 'a' still exist ? */
   13019           4 :         status = cli_posix_open(cli_posix,
   13020             :                                 aname,
   13021             :                                 O_RDONLY,
   13022             :                                 0,
   13023             :                                 &fnum);
   13024           4 :         if (!NT_STATUS_IS_OK(status)) {
   13025           0 :                 printf("cli_posix_open O_RNONLY of %s returned %s\n",
   13026             :                         aname,
   13027             :                         nt_errstr(status));
   13028           0 :                 goto out;
   13029             :         }
   13030             : 
   13031           4 :         status = cli_close(cli_posix, fnum);
   13032           4 :         if (!NT_STATUS_IS_OK(status)) {
   13033           0 :                 goto out;
   13034             :         }
   13035             : 
   13036           4 :         correct = true;
   13037             : 
   13038           4 :   out:
   13039             : 
   13040           4 :         TALLOC_FREE(windows_unlink_name);
   13041           4 :         TALLOC_FREE(mangled_name);
   13042             : 
   13043           4 :         if (cli != NULL) {
   13044           4 :                 torture_deltree(cli, dname);
   13045           4 :                 torture_close_connection(cli);
   13046             :         }
   13047             : 
   13048           4 :         if (cli_posix != NULL) {
   13049           4 :                 torture_close_connection(cli_posix);
   13050             :         }
   13051             : 
   13052           4 :         return correct;
   13053             : }
   13054             : 
   13055           4 : static bool run_smb1_wild_mangle_rename_test(int dummy)
   13056             : {
   13057           0 :         static struct cli_state *cli_posix = NULL;
   13058           0 :         static struct cli_state *cli = NULL;
   13059           4 :         uint16_t fnum = (uint16_t)-1;
   13060           4 :         bool correct = false;
   13061           4 :         const char *dname = "smb1_wild_mangle_rename";
   13062           4 :         const char *fooname = "smb1_wild_mangle_rename/foo";
   13063           4 :         const char *foostar_name = "smb1_wild_mangle_rename/fo*";
   13064           4 :         const char *wild_name = "smb1_wild_mangle_rename/*";
   13065           4 :         char *windows_rename_src = NULL;
   13066           4 :         const char *windows_rename_dst = "smb1_wild_mangle_rename\\bar";
   13067           4 :         char *mangled_name = NULL;
   13068           0 :         NTSTATUS status;
   13069             : 
   13070           4 :         printf("Starting SMB1 wild mangle rename test\n");
   13071             : 
   13072           4 :         if (!torture_open_connection(&cli_posix, 0)) {
   13073           0 :                 return false;
   13074             :         }
   13075             : 
   13076           4 :         smbXcli_conn_set_sockopt(cli_posix->conn, sockops);
   13077             : 
   13078           4 :         status = torture_setup_unix_extensions(cli_posix);
   13079           4 :         if (!NT_STATUS_IS_OK(status)) {
   13080           0 :                 printf("server doesn't support POSIX\n");
   13081           0 :                 return false;
   13082             :         }
   13083             : 
   13084             :         /* Open a Windows connection. */
   13085           4 :         if (!torture_open_connection(&cli, 0)) {
   13086           0 :                 goto out;
   13087             :         }
   13088             : 
   13089           4 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
   13090             : 
   13091             :         /* Ensure we start from fresh. */
   13092           4 :         torture_deltree(cli, dname);
   13093             : 
   13094             :         /*
   13095             :          * Create two files - 'foo' and 'fo*'.
   13096             :          * We need POSIX extensions for this as 'fo*'
   13097             :          * is not a valid Windows name.
   13098             :          */
   13099             : 
   13100           4 :         status = cli_posix_mkdir(cli_posix, dname, 0770);
   13101           4 :         if (!NT_STATUS_IS_OK(status)) {
   13102           0 :                 printf("cli_posix_mkdir of %s returned %s\n",
   13103             :                         dname,
   13104             :                         nt_errstr(status));
   13105           0 :                 goto out;
   13106             :         }
   13107             : 
   13108           4 :         status = cli_posix_open(cli_posix,
   13109             :                                 fooname,
   13110             :                                 O_RDWR|O_CREAT|O_EXCL,
   13111             :                                 0660,
   13112             :                                 &fnum);
   13113           4 :         if (!NT_STATUS_IS_OK(status)) {
   13114           0 :                 printf("cli_posix_open (create) of %s returned %s\n",
   13115             :                         fooname,
   13116             :                         nt_errstr(status));
   13117           0 :                 goto out;
   13118             :         }
   13119           4 :         status = cli_close(cli_posix, fnum);
   13120           4 :         if (!NT_STATUS_IS_OK(status)) {
   13121           0 :                 goto out;
   13122             :         }
   13123           4 :         status = cli_posix_open(cli_posix,
   13124             :                                 foostar_name,
   13125             :                                 O_RDWR|O_CREAT|O_EXCL,
   13126             :                                 0660,
   13127             :                                 &fnum);
   13128           4 :         if (!NT_STATUS_IS_OK(status)) {
   13129           0 :                 printf("cli_posix_open (create) of %s returned %s\n",
   13130             :                         foostar_name,
   13131             :                         nt_errstr(status));
   13132           0 :                 goto out;
   13133             :         }
   13134           4 :         status = cli_close(cli_posix, fnum);
   13135           4 :         if (!NT_STATUS_IS_OK(status)) {
   13136           0 :                 goto out;
   13137             :         }
   13138             : 
   13139             :         /*
   13140             :          * Get the mangled name. We can re-use the
   13141             :          * previous smb1_wild_mangle_list_fn for this.
   13142             :          */
   13143             : 
   13144           4 :         status = cli_list(cli,
   13145             :                         wild_name,
   13146             :                         0,
   13147             :                         smb1_wild_mangle_list_fn,
   13148             :                         &mangled_name);
   13149           4 :         if (!NT_STATUS_IS_OK(status)) {
   13150           0 :                 printf("cli_list of %s returned %s\n",
   13151             :                         wild_name,
   13152             :                         nt_errstr(status));
   13153           0 :                 goto out;
   13154             :         }
   13155             : 
   13156           4 :         if (mangled_name == NULL) {
   13157           0 :                 goto out;
   13158             :         }
   13159             : 
   13160           4 :         printf("mangled_name = %s\n",
   13161             :                 mangled_name);
   13162             : 
   13163             :         /*
   13164             :          * Try a Windows rename with the mangled name.
   13165             :          * This should *NOT* rename the 'foo' name.
   13166             :          */
   13167             : 
   13168           4 :         windows_rename_src = talloc_asprintf(cli_posix,
   13169             :                                         "%s\\%s",
   13170             :                                         dname,
   13171             :                                         mangled_name);
   13172             : 
   13173           4 :         status = cli_rename(cli,
   13174             :                         windows_rename_src,
   13175             :                         windows_rename_dst,
   13176             :                         false);
   13177           4 :         if (!NT_STATUS_IS_OK(status)) {
   13178           0 :                 printf("cli_rename of %s -> %s returned %s\n",
   13179             :                         windows_rename_src,
   13180             :                         windows_rename_dst,
   13181             :                         nt_errstr(status));
   13182           0 :                 goto out;
   13183             :         }
   13184             : 
   13185             :         /* Does 'foo' still exist ? */
   13186           4 :         status = cli_posix_open(cli_posix,
   13187             :                                 fooname,
   13188             :                                 O_RDONLY,
   13189             :                                 0,
   13190             :                                 &fnum);
   13191           4 :         if (!NT_STATUS_IS_OK(status)) {
   13192           0 :                 printf("cli_posix_open O_RNONLY of %s returned %s\n",
   13193             :                         fooname,
   13194             :                         nt_errstr(status));
   13195           0 :                 goto out;
   13196             :         }
   13197             : 
   13198           4 :         status = cli_close(cli_posix, fnum);
   13199           4 :         if (!NT_STATUS_IS_OK(status)) {
   13200           0 :                 goto out;
   13201             :         }
   13202             : 
   13203           4 :         correct = true;
   13204             : 
   13205           4 :   out:
   13206             : 
   13207           4 :         TALLOC_FREE(mangled_name);
   13208           4 :         TALLOC_FREE(windows_rename_src);
   13209             : 
   13210           4 :         if (cli != NULL) {
   13211           4 :                 torture_deltree(cli, dname);
   13212           4 :                 torture_close_connection(cli);
   13213             :         }
   13214             : 
   13215           4 :         torture_close_connection(cli_posix);
   13216             : 
   13217           4 :         return correct;
   13218             : }
   13219             : 
   13220             : /*
   13221             :  * Only testing minimal time strings, as the others
   13222             :  * need (locale-dependent) guessing at what strftime does and
   13223             :  * even may differ in builds.
   13224             :  */
   13225           1 : static bool timesubst_test(void)
   13226             : {
   13227           1 :         TALLOC_CTX *ctx = NULL;
   13228             :         /* Sa 23. Dez 04:33:20 CET 2017 */
   13229           1 :         const struct timeval tv = { 1514000000, 123 };
   13230           1 :         const char* expect_minimal = "20171223_033320";
   13231           1 :         const char* expect_minus   = "20171223_033320_000123";
   13232           1 :         char *s;
   13233           1 :         char *env_tz, *orig_tz = NULL;
   13234           1 :         bool result = true;
   13235             : 
   13236           1 :         ctx = talloc_new(NULL);
   13237             : 
   13238           1 :         env_tz = getenv("TZ");
   13239           1 :         if(env_tz) {
   13240           1 :                 orig_tz = talloc_strdup(ctx, env_tz);
   13241             :         }
   13242           1 :         setenv("TZ", "UTC", 1);
   13243             : 
   13244           1 :         s = minimal_timeval_string(ctx, &tv, false);
   13245             : 
   13246           1 :         if(!s || strcmp(s, expect_minimal)) {
   13247           0 :                 printf("minimal_timeval_string(ctx, tv, false) returned [%s], expected "
   13248             :                        "[%s]\n", s ? s : "<nil>", expect_minimal);
   13249           0 :                 result = false;
   13250             :         }
   13251           1 :         TALLOC_FREE(s);
   13252           1 :         s = minimal_timeval_string(ctx, &tv, true);
   13253           1 :         if(!s || strcmp(s, expect_minus)) {
   13254           0 :                 printf("minimal_timeval_string(ctx, tv, true) returned [%s], expected "
   13255             :                        "[%s]\n", s ? s : "<nil>", expect_minus);
   13256           0 :                 result = false;
   13257             :         }
   13258           1 :         TALLOC_FREE(s);
   13259             : 
   13260           1 :         if(orig_tz) {
   13261           1 :                 setenv("TZ", orig_tz, 1);
   13262             :         }
   13263             : 
   13264           1 :         TALLOC_FREE(ctx);
   13265           1 :         return result;
   13266             : }
   13267             : 
   13268           1 : static bool run_local_substitute(int dummy)
   13269             : {
   13270           1 :         bool ok = true;
   13271             : 
   13272           1 :         ok &= subst_test("%U", "bla", "", -1, -1, "bla");
   13273           1 :         ok &= subst_test("%u%U", "bla", "", -1, -1, "blabla");
   13274           1 :         ok &= subst_test("%g", "", "", -1, -1, "NO_GROUP");
   13275           1 :         ok &= subst_test("%G", "", "", -1, -1, "NO_GROUP");
   13276           1 :         ok &= subst_test("%g", "", "", -1, 0, gidtoname(0));
   13277           1 :         ok &= subst_test("%G", "", "", -1, 0, gidtoname(0));
   13278           1 :         ok &= subst_test("%D%u", "u", "dom", -1, 0, "domu");
   13279           1 :         ok &= subst_test("%i %I", "", "", -1, -1, "0.0.0.0 0.0.0.0");
   13280           1 :         ok &= subst_test("%j %J", "", "", -1, -1, "0_0_0_0 0_0_0_0");
   13281             :         /* Substitution depends on current time, so better test the underlying
   13282             :            formatting function. At least covers %t. */
   13283           1 :         ok &= timesubst_test();
   13284             : 
   13285             :         /* Different captialization rules in sub_basic... */
   13286             : 
   13287           1 :         ok &=  (strcmp(talloc_sub_basic(talloc_tos(), "BLA", "dom", "%U%D"),
   13288           1 :                        "blaDOM") == 0);
   13289             : 
   13290           1 :         return ok;
   13291             : }
   13292             : 
   13293           1 : static bool run_local_base64(int dummy)
   13294             : {
   13295           1 :         int i;
   13296           1 :         bool ret = true;
   13297             : 
   13298        2000 :         for (i=1; i<2000; i++) {
   13299        1999 :                 DATA_BLOB blob1, blob2;
   13300        1999 :                 char *b64;
   13301             : 
   13302        1999 :                 blob1.data = talloc_array(talloc_tos(), uint8_t, i);
   13303        1999 :                 blob1.length = i;
   13304        1999 :                 generate_random_buffer(blob1.data, blob1.length);
   13305             : 
   13306        1999 :                 b64 = base64_encode_data_blob(talloc_tos(), blob1);
   13307        1999 :                 if (b64 == NULL) {
   13308           0 :                         d_fprintf(stderr, "base64_encode_data_blob failed "
   13309             :                                   "for %d bytes\n", i);
   13310           0 :                         ret = false;
   13311             :                 }
   13312        1999 :                 blob2 = base64_decode_data_blob(b64);
   13313        1999 :                 TALLOC_FREE(b64);
   13314             : 
   13315        1999 :                 if (data_blob_cmp(&blob1, &blob2)) {
   13316           0 :                         d_fprintf(stderr, "data_blob_cmp failed for %d "
   13317             :                                   "bytes\n", i);
   13318           0 :                         ret = false;
   13319             :                 }
   13320        1999 :                 TALLOC_FREE(blob1.data);
   13321        1999 :                 data_blob_free(&blob2);
   13322             :         }
   13323           1 :         return ret;
   13324             : }
   13325             : 
   13326     1000000 : static void parse_fn(const struct gencache_timeout *t,
   13327             :                      DATA_BLOB blob,
   13328             :                      void *private_data)
   13329             : {
   13330     1000000 :         return;
   13331             : }
   13332             : 
   13333           1 : static bool run_local_gencache(int dummy)
   13334             : {
   13335           1 :         char *val;
   13336           1 :         time_t tm;
   13337           1 :         DATA_BLOB blob;
   13338           1 :         char v;
   13339           1 :         struct memcache *mem;
   13340           1 :         int i;
   13341             : 
   13342           1 :         mem = memcache_init(NULL, 0);
   13343           1 :         if (mem == NULL) {
   13344           0 :                 d_printf("%s: memcache_init failed\n", __location__);
   13345           0 :                 return false;
   13346             :         }
   13347           1 :         memcache_set_global(mem);
   13348             : 
   13349           1 :         if (!gencache_set("foo", "bar", time(NULL) + 1000)) {
   13350           0 :                 d_printf("%s: gencache_set() failed\n", __location__);
   13351           0 :                 return False;
   13352             :         }
   13353             : 
   13354           1 :         if (!gencache_get("foo", NULL, NULL, NULL)) {
   13355           0 :                 d_printf("%s: gencache_get() failed\n", __location__);
   13356           0 :                 return False;
   13357             :         }
   13358             : 
   13359     1000001 :         for (i=0; i<1000000; i++) {
   13360     1000000 :                 gencache_parse("foo", parse_fn, NULL);
   13361             :         }
   13362             : 
   13363           1 :         if (!gencache_get("foo", talloc_tos(), &val, &tm)) {
   13364           0 :                 d_printf("%s: gencache_get() failed\n", __location__);
   13365           0 :                 return False;
   13366             :         }
   13367           1 :         TALLOC_FREE(val);
   13368             : 
   13369           1 :         if (!gencache_get("foo", talloc_tos(), &val, &tm)) {
   13370           0 :                 d_printf("%s: gencache_get() failed\n", __location__);
   13371           0 :                 return False;
   13372             :         }
   13373             : 
   13374           1 :         if (strcmp(val, "bar") != 0) {
   13375           0 :                 d_printf("%s: gencache_get() returned %s, expected %s\n",
   13376             :                          __location__, val, "bar");
   13377           0 :                 TALLOC_FREE(val);
   13378           0 :                 return False;
   13379             :         }
   13380             : 
   13381           1 :         TALLOC_FREE(val);
   13382             : 
   13383           1 :         if (!gencache_del("foo")) {
   13384           0 :                 d_printf("%s: gencache_del() failed\n", __location__);
   13385           0 :                 return False;
   13386             :         }
   13387           1 :         if (gencache_del("foo")) {
   13388           0 :                 d_printf("%s: second gencache_del() succeeded\n",
   13389             :                          __location__);
   13390           0 :                 return False;
   13391             :         }
   13392             : 
   13393           1 :         if (gencache_get("foo", talloc_tos(), &val, &tm)) {
   13394           0 :                 d_printf("%s: gencache_get() on deleted entry "
   13395             :                          "succeeded\n", __location__);
   13396           0 :                 return False;
   13397             :         }
   13398             : 
   13399           1 :         blob = data_blob_string_const_null("bar");
   13400           1 :         tm = time(NULL) + 60;
   13401             : 
   13402           1 :         if (!gencache_set_data_blob("foo", blob, tm)) {
   13403           0 :                 d_printf("%s: gencache_set_data_blob() failed\n", __location__);
   13404           0 :                 return False;
   13405             :         }
   13406             : 
   13407           1 :         if (!gencache_get_data_blob("foo", talloc_tos(), &blob, NULL, NULL)) {
   13408           0 :                 d_printf("%s: gencache_get_data_blob() failed\n", __location__);
   13409           0 :                 return False;
   13410             :         }
   13411             : 
   13412           1 :         if (strcmp((const char *)blob.data, "bar") != 0) {
   13413           0 :                 d_printf("%s: gencache_get_data_blob() returned %s, expected %s\n",
   13414           0 :                          __location__, (const char *)blob.data, "bar");
   13415           0 :                 data_blob_free(&blob);
   13416           0 :                 return False;
   13417             :         }
   13418             : 
   13419           1 :         data_blob_free(&blob);
   13420             : 
   13421           1 :         if (!gencache_del("foo")) {
   13422           0 :                 d_printf("%s: gencache_del() failed\n", __location__);
   13423           0 :                 return False;
   13424             :         }
   13425           1 :         if (gencache_del("foo")) {
   13426           0 :                 d_printf("%s: second gencache_del() succeeded\n",
   13427             :                          __location__);
   13428           0 :                 return False;
   13429             :         }
   13430             : 
   13431           1 :         if (gencache_get_data_blob("foo", talloc_tos(), &blob, NULL, NULL)) {
   13432           0 :                 d_printf("%s: gencache_get_data_blob() on deleted entry "
   13433             :                          "succeeded\n", __location__);
   13434           0 :                 return False;
   13435             :         }
   13436             : 
   13437           1 :         v = 1;
   13438           1 :         blob.data = (uint8_t *)&v;
   13439           1 :         blob.length = sizeof(v);
   13440             : 
   13441           1 :         if (!gencache_set_data_blob("blob", blob, tm)) {
   13442           0 :                 d_printf("%s: gencache_set_data_blob() failed\n",
   13443             :                          __location__);
   13444           0 :                 return false;
   13445             :         }
   13446           1 :         if (gencache_get("blob", talloc_tos(), &val, &tm)) {
   13447           0 :                 d_printf("%s: gencache_get succeeded\n", __location__);
   13448           0 :                 return false;
   13449             :         }
   13450             : 
   13451           0 :         return True;
   13452             : }
   13453             : 
   13454           1 : static bool rbt_testflags(struct db_context *db, const char *key,
   13455             :                           const char *value)
   13456             : {
   13457           1 :         bool ret = false;
   13458           1 :         NTSTATUS status;
   13459           1 :         struct db_record *rec;
   13460             : 
   13461           1 :         rec = dbwrap_fetch_locked(db, db, string_tdb_data(key));
   13462           1 :         if (rec == NULL) {
   13463           0 :                 d_fprintf(stderr, "fetch_locked failed\n");
   13464           0 :                 goto done;
   13465             :         }
   13466             : 
   13467           1 :         status = dbwrap_record_store(rec, string_tdb_data(value), TDB_MODIFY);
   13468           1 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_NAME_NOT_FOUND)) {
   13469           0 :                 d_fprintf(stderr, "store TDB_MODIFY unexpected status: %s\n",
   13470             :                           nt_errstr(status));
   13471           0 :                 goto done;
   13472             :         }
   13473             : 
   13474           1 :         status = dbwrap_record_store(rec, string_tdb_data("overwriteme"),
   13475             :                                      TDB_INSERT);
   13476           1 :         if (!NT_STATUS_IS_OK(status)) {
   13477           0 :                 d_fprintf(stderr, "store TDB_INSERT failed: %s\n",
   13478             :                           nt_errstr(status));
   13479           0 :                 goto done;
   13480             :         }
   13481             : 
   13482           1 :         status = dbwrap_record_store(rec, string_tdb_data(value), TDB_INSERT);
   13483           1 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_OBJECT_NAME_COLLISION)) {
   13484           0 :                 d_fprintf(stderr, "store TDB_INSERT unexpected status: %s\n",
   13485             :                           nt_errstr(status));
   13486           0 :                 goto done;
   13487             :         }
   13488             : 
   13489           1 :         status = dbwrap_record_store(rec, string_tdb_data(value), TDB_MODIFY);
   13490           1 :         if (!NT_STATUS_IS_OK(status)) {
   13491           0 :                 d_fprintf(stderr, "store TDB_MODIFY failed: %s\n",
   13492             :                           nt_errstr(status));
   13493           0 :                 goto done;
   13494             :         }
   13495             : 
   13496           0 :         ret = true;
   13497           1 : done:
   13498           1 :         TALLOC_FREE(rec);
   13499           1 :         return ret;
   13500             : }
   13501             : 
   13502        1998 : static bool rbt_testval(struct db_context *db, const char *key,
   13503             :                         const char *value)
   13504             : {
   13505        1998 :         struct db_record *rec;
   13506        1998 :         TDB_DATA data = string_tdb_data(value);
   13507        1998 :         bool ret = false;
   13508        1998 :         NTSTATUS status;
   13509        1998 :         TDB_DATA dbvalue;
   13510             : 
   13511        1998 :         rec = dbwrap_fetch_locked(db, db, string_tdb_data(key));
   13512        1998 :         if (rec == NULL) {
   13513           0 :                 d_fprintf(stderr, "fetch_locked failed\n");
   13514           0 :                 goto done;
   13515             :         }
   13516        1998 :         status = dbwrap_record_store(rec, data, 0);
   13517        1998 :         if (!NT_STATUS_IS_OK(status)) {
   13518           0 :                 d_fprintf(stderr, "store failed: %s\n", nt_errstr(status));
   13519           0 :                 goto done;
   13520             :         }
   13521        1998 :         TALLOC_FREE(rec);
   13522             : 
   13523        1998 :         rec = dbwrap_fetch_locked(db, db, string_tdb_data(key));
   13524        1998 :         if (rec == NULL) {
   13525           0 :                 d_fprintf(stderr, "second fetch_locked failed\n");
   13526           0 :                 goto done;
   13527             :         }
   13528             : 
   13529        1998 :         dbvalue = dbwrap_record_get_value(rec);
   13530        1998 :         if ((dbvalue.dsize != data.dsize)
   13531        1998 :             || (memcmp(dbvalue.dptr, data.dptr, data.dsize) != 0)) {
   13532           0 :                 d_fprintf(stderr, "Got wrong data back\n");
   13533           0 :                 goto done;
   13534             :         }
   13535             : 
   13536           0 :         ret = true;
   13537        1998 :  done:
   13538        1998 :         TALLOC_FREE(rec);
   13539        1998 :         return ret;
   13540             : }
   13541             : 
   13542        1000 : static int local_rbtree_traverse_read(struct db_record *rec, void *private_data)
   13543             : {
   13544        1000 :         int *count2 = (int *)private_data;
   13545        1000 :         (*count2)++;
   13546        1000 :         return 0;
   13547             : }
   13548             : 
   13549        1000 : static int local_rbtree_traverse_delete(struct db_record *rec, void *private_data)
   13550             : {
   13551        1000 :         int *count2 = (int *)private_data;
   13552        1000 :         (*count2)++;
   13553        1000 :         dbwrap_record_delete(rec);
   13554        1000 :         return 0;
   13555             : }
   13556             : 
   13557           1 : static bool run_local_rbtree(int dummy)
   13558             : {
   13559           1 :         struct db_context *db;
   13560           1 :         bool ret = false;
   13561           1 :         int i;
   13562           1 :         NTSTATUS status;
   13563           1 :         int count = 0;
   13564           1 :         int count2 = 0;
   13565             : 
   13566           1 :         db = db_open_rbt(NULL);
   13567             : 
   13568           1 :         if (db == NULL) {
   13569           0 :                 d_fprintf(stderr, "db_open_rbt failed\n");
   13570           0 :                 return false;
   13571             :         }
   13572             : 
   13573           1 :         if (!rbt_testflags(db, "firstkey", "firstval")) {
   13574           0 :                 goto done;
   13575             :         }
   13576             : 
   13577        1000 :         for (i = 0; i < 999; i++) {
   13578         999 :                 char key[sizeof("key-9223372036854775807")];
   13579         999 :                 char value[sizeof("value-9223372036854775807")];
   13580             : 
   13581         999 :                 snprintf(key, sizeof(key), "key%ld", random());
   13582         999 :                 snprintf(value, sizeof(value) ,"value%ld", random());
   13583             : 
   13584         999 :                 if (!rbt_testval(db, key, value)) {
   13585           0 :                         goto done;
   13586             :                 }
   13587             : 
   13588         999 :                 snprintf(value, sizeof(value) ,"value%ld", random());
   13589             : 
   13590         999 :                 if (!rbt_testval(db, key, value)) {
   13591           0 :                         goto done;
   13592             :                 }
   13593             :         }
   13594             : 
   13595           1 :         ret = true;
   13596           1 :         count = 0; count2 = 0;
   13597           1 :         status = dbwrap_traverse_read(db, local_rbtree_traverse_read,
   13598             :                                       &count2, &count);
   13599           1 :         printf("%s: read1: %d %d, %s\n", __func__, count, count2, nt_errstr(status));
   13600           1 :         if ((count != count2) || (count != 1000)) {
   13601           0 :                 ret = false;
   13602             :         }
   13603           1 :         count = 0; count2 = 0;
   13604           1 :         status = dbwrap_traverse(db, local_rbtree_traverse_delete,
   13605             :                                  &count2, &count);
   13606           1 :         printf("%s: delete: %d %d, %s\n", __func__, count, count2, nt_errstr(status));
   13607           1 :         if ((count != count2) || (count != 1000)) {
   13608           0 :                 ret = false;
   13609             :         }
   13610           1 :         count = 0; count2 = 0;
   13611           1 :         status = dbwrap_traverse_read(db, local_rbtree_traverse_read,
   13612             :                                       &count2, &count);
   13613           1 :         printf("%s: read2: %d %d, %s\n", __func__, count, count2, nt_errstr(status));
   13614           1 :         if ((count != count2) || (count != 0)) {
   13615           0 :                 ret = false;
   13616             :         }
   13617             : 
   13618           1 :  done:
   13619           1 :         TALLOC_FREE(db);
   13620           1 :         return ret;
   13621             : }
   13622             : 
   13623             : 
   13624             : /*
   13625             :   local test for character set functions
   13626             : 
   13627             :   This is a very simple test for the functionality in convert_string_error()
   13628             :  */
   13629           1 : static bool run_local_convert_string(int dummy)
   13630             : {
   13631           1 :         TALLOC_CTX *tmp_ctx = talloc_new(NULL);
   13632           1 :         const char *test_strings[2] = { "March", "M\303\244rz" };
   13633           1 :         char dst[7];
   13634           1 :         int i;
   13635             : 
   13636           3 :         for (i=0; i<2; i++) {
   13637           2 :                 const char *str = test_strings[i];
   13638           2 :                 int len = strlen(str);
   13639           2 :                 size_t converted_size;
   13640           2 :                 bool ret;
   13641             : 
   13642           2 :                 memset(dst, 'X', sizeof(dst));
   13643             : 
   13644             :                 /* first try with real source length */
   13645           2 :                 ret = convert_string_error(CH_UNIX, CH_UTF8,
   13646             :                                            str, len,
   13647             :                                            dst, sizeof(dst),
   13648             :                                            &converted_size);
   13649           2 :                 if (ret != true) {
   13650           0 :                         d_fprintf(stderr, "Failed to convert '%s' to CH_DISPLAY\n", str);
   13651           0 :                         goto failed;
   13652             :                 }
   13653             : 
   13654           2 :                 if (converted_size != len) {
   13655           0 :                         d_fprintf(stderr, "Converted size of '%s' should be %d - got %d\n",
   13656             :                                   str, len, (int)converted_size);
   13657           0 :                         goto failed;
   13658             :                 }
   13659             : 
   13660           2 :                 if (strncmp(str, dst, converted_size) != 0) {
   13661           0 :                         d_fprintf(stderr, "Expected '%s' to match '%s'\n", str, dst);
   13662           0 :                         goto failed;
   13663             :                 }
   13664             : 
   13665           2 :                 if (strlen(str) != converted_size) {
   13666           0 :                         d_fprintf(stderr, "Expected '%s' length %d - got %d\n", str,
   13667           0 :                                   (int)strlen(str), (int)converted_size);
   13668           0 :                         goto failed;
   13669             :                 }
   13670             : 
   13671           2 :                 if (dst[converted_size] != 'X') {
   13672           0 :                         d_fprintf(stderr, "Expected no termination of '%s'\n", dst);
   13673           0 :                         goto failed;
   13674             :                 }
   13675             : 
   13676             :                 /* now with srclen==-1, this causes the nul to be
   13677             :                  * converted too */
   13678           2 :                 ret = convert_string_error(CH_UNIX, CH_UTF8,
   13679             :                                            str, -1,
   13680             :                                            dst, sizeof(dst),
   13681             :                                            &converted_size);
   13682           2 :                 if (ret != true) {
   13683           0 :                         d_fprintf(stderr, "Failed to convert '%s' to CH_DISPLAY\n", str);
   13684           0 :                         goto failed;
   13685             :                 }
   13686             : 
   13687           2 :                 if (converted_size != len+1) {
   13688           0 :                         d_fprintf(stderr, "Converted size of '%s' should be %d - got %d\n",
   13689             :                                   str, len, (int)converted_size);
   13690           0 :                         goto failed;
   13691             :                 }
   13692             : 
   13693           2 :                 if (strncmp(str, dst, converted_size) != 0) {
   13694           0 :                         d_fprintf(stderr, "Expected '%s' to match '%s'\n", str, dst);
   13695           0 :                         goto failed;
   13696             :                 }
   13697             : 
   13698           2 :                 if (len+1 != converted_size) {
   13699           0 :                         d_fprintf(stderr, "Expected '%s' length %d - got %d\n", str,
   13700             :                                   len+1, (int)converted_size);
   13701           0 :                         goto failed;
   13702             :                 }
   13703             : 
   13704           2 :                 if (dst[converted_size] != 'X') {
   13705           0 :                         d_fprintf(stderr, "Expected no termination of '%s'\n", dst);
   13706           0 :                         goto failed;
   13707             :                 }
   13708             : 
   13709             :         }
   13710             : 
   13711             : 
   13712           1 :         TALLOC_FREE(tmp_ctx);
   13713           0 :         return true;
   13714           0 : failed:
   13715           0 :         TALLOC_FREE(tmp_ctx);
   13716           0 :         return false;
   13717             : }
   13718             : 
   13719           1 : static bool run_local_string_to_sid(int dummy) {
   13720           1 :         struct dom_sid sid;
   13721             : 
   13722           1 :         if (string_to_sid(&sid, "S--1-5-32-545")) {
   13723           0 :                 printf("allowing S--1-5-32-545\n");
   13724           0 :                 return false;
   13725             :         }
   13726           1 :         if (string_to_sid(&sid, "S-1-5-32-+545")) {
   13727           0 :                 printf("allowing S-1-5-32-+545\n");
   13728           0 :                 return false;
   13729             :         }
   13730           1 :         if (string_to_sid(&sid, "S-1-2-3-4-5-6-7-8-9-0-1-2-3-4-5-6-7-8-9-0")) {
   13731           0 :                 printf("allowing S-1-2-3-4-5-6-7-8-9-0-1-2-3-4-5-6-7-8-9-0\n");
   13732           0 :                 return false;
   13733             :         }
   13734           1 :         if (string_to_sid(&sid, "S-1-5-32-545-abc")) {
   13735           0 :                 printf("allowing S-1-5-32-545-abc\n");
   13736           0 :                 return false;
   13737             :         }
   13738           1 :         if (string_to_sid(&sid, "S-300-5-32-545")) {
   13739           0 :                 printf("allowing S-300-5-32-545\n");
   13740           0 :                 return false;
   13741             :         }
   13742           1 :         if (string_to_sid(&sid, "S-1-0xfffffffffffffe-32-545")) {
   13743           0 :                 printf("allowing S-1-0xfffffffffffffe-32-545\n");
   13744           0 :                 return false;
   13745             :         }
   13746           1 :         if (string_to_sid(&sid, "S-1-0xffffffffffff-5294967297-545")) {
   13747           0 :                 printf("allowing S-1-0xffffffffffff-5294967297-545\n");
   13748           0 :                 return false;
   13749             :         }
   13750           1 :         if (!string_to_sid(&sid, "S-1-0xfffffffffffe-32-545")) {
   13751           0 :                 printf("could not parse S-1-0xfffffffffffe-32-545\n");
   13752           0 :                 return false;
   13753             :         }
   13754           1 :         if (!string_to_sid(&sid, "S-1-5-32-545")) {
   13755           0 :                 printf("could not parse S-1-5-32-545\n");
   13756           0 :                 return false;
   13757             :         }
   13758           1 :         if (!dom_sid_equal(&sid, &global_sid_Builtin_Users)) {
   13759           0 :                 struct dom_sid_buf buf;
   13760           0 :                 printf("mis-parsed S-1-5-32-545 as %s\n",
   13761             :                        dom_sid_str_buf(&sid, &buf));
   13762           0 :                 return false;
   13763             :         }
   13764           0 :         return true;
   13765             : }
   13766             : 
   13767           3 : static bool sid_to_string_test(const char *expected) {
   13768           3 :         char *str;
   13769           3 :         bool res = true;
   13770           3 :         struct dom_sid sid;
   13771             : 
   13772           3 :         if (!string_to_sid(&sid, expected)) {
   13773           0 :                 printf("could not parse %s\n", expected);
   13774           0 :                 return false;
   13775             :         }
   13776             : 
   13777           3 :         str = dom_sid_string(NULL, &sid);
   13778           3 :         if (strcmp(str, expected)) {
   13779           0 :                 printf("Comparison failed (%s != %s)\n", str, expected);
   13780           0 :                 res = false;
   13781             :         }
   13782           3 :         TALLOC_FREE(str);
   13783           3 :         return res;
   13784             : }
   13785             : 
   13786           1 : static bool run_local_sid_to_string(int dummy) {
   13787           1 :         if (!sid_to_string_test("S-1-0xffffffffffff-1-1-1-1-1-1-1-1-1-1-1-1"))
   13788           0 :                 return false;
   13789           1 :         if (!sid_to_string_test("S-1-545"))
   13790           0 :                 return false;
   13791           1 :         if (!sid_to_string_test("S-255-3840-1-1-1-1"))
   13792           0 :                 return false;
   13793           0 :         return true;
   13794             : }
   13795             : 
   13796           1 : static bool run_local_binary_to_sid(int dummy) {
   13797           1 :         ssize_t ret;
   13798           1 :         struct dom_sid *sid = talloc(NULL, struct dom_sid);
   13799           1 :         static const uint8_t good_binary_sid[] = {
   13800             :                 0x1, /* revision number */
   13801             :                 15, /* num auths */
   13802             :                 0x1, 0x1, 0x1, 0x1, 0x1, 0x1, /* id_auth */
   13803             :                 0x1, 0x1, 0x1, 0x1, /* auth[0] */
   13804             :                 0x1, 0x1, 0x1, 0x1, /* auth[1] */
   13805             :                 0x1, 0x1, 0x1, 0x1, /* auth[2] */
   13806             :                 0x1, 0x1, 0x1, 0x1, /* auth[3] */
   13807             :                 0x1, 0x1, 0x1, 0x1, /* auth[4] */
   13808             :                 0x1, 0x1, 0x1, 0x1, /* auth[5] */
   13809             :                 0x1, 0x1, 0x1, 0x1, /* auth[6] */
   13810             :                 0x1, 0x1, 0x1, 0x1, /* auth[7] */
   13811             :                 0x1, 0x1, 0x1, 0x1, /* auth[8] */
   13812             :                 0x1, 0x1, 0x1, 0x1, /* auth[9] */
   13813             :                 0x1, 0x1, 0x1, 0x1, /* auth[10] */
   13814             :                 0x1, 0x1, 0x1, 0x1, /* auth[11] */
   13815             :                 0x1, 0x1, 0x1, 0x1, /* auth[12] */
   13816             :                 0x1, 0x1, 0x1, 0x1, /* auth[13] */
   13817             :                 0x1, 0x1, 0x1, 0x1, /* auth[14] */
   13818             :         };
   13819             : 
   13820           1 :         static const uint8_t long_binary_sid[] = {
   13821             :                 0x1, /* revision number */
   13822             :                 15, /* num auths */
   13823             :                 0x1, 0x1, 0x1, 0x1, 0x1, 0x1, /* id_auth */
   13824             :                 0x1, 0x1, 0x1, 0x1, /* auth[0] */
   13825             :                 0x1, 0x1, 0x1, 0x1, /* auth[1] */
   13826             :                 0x1, 0x1, 0x1, 0x1, /* auth[2] */
   13827             :                 0x1, 0x1, 0x1, 0x1, /* auth[3] */
   13828             :                 0x1, 0x1, 0x1, 0x1, /* auth[4] */
   13829             :                 0x1, 0x1, 0x1, 0x1, /* auth[5] */
   13830             :                 0x1, 0x1, 0x1, 0x1, /* auth[6] */
   13831             :                 0x1, 0x1, 0x1, 0x1, /* auth[7] */
   13832             :                 0x1, 0x1, 0x1, 0x1, /* auth[8] */
   13833             :                 0x1, 0x1, 0x1, 0x1, /* auth[9] */
   13834             :                 0x1, 0x1, 0x1, 0x1, /* auth[10] */
   13835             :                 0x1, 0x1, 0x1, 0x1, /* auth[11] */
   13836             :                 0x1, 0x1, 0x1, 0x1, /* auth[12] */
   13837             :                 0x1, 0x1, 0x1, 0x1, /* auth[13] */
   13838             :                 0x1, 0x1, 0x1, 0x1, /* auth[14] */
   13839             :                 0x1, 0x1, 0x1, 0x1, /* auth[15] */
   13840             :                 0x1, 0x1, 0x1, 0x1, /* auth[16] */
   13841             :                 0x1, 0x1, 0x1, 0x1, /* auth[17] */
   13842             :         };
   13843             : 
   13844           1 :         static const uint8_t long_binary_sid2[] = {
   13845             :                 0x1, /* revision number */
   13846             :                 32, /* num auths */
   13847             :                 0x1, 0x1, 0x1, 0x1, 0x1, 0x1, /* id_auth */
   13848             :                 0x1, 0x1, 0x1, 0x1, /* auth[0] */
   13849             :                 0x1, 0x1, 0x1, 0x1, /* auth[1] */
   13850             :                 0x1, 0x1, 0x1, 0x1, /* auth[2] */
   13851             :                 0x1, 0x1, 0x1, 0x1, /* auth[3] */
   13852             :                 0x1, 0x1, 0x1, 0x1, /* auth[4] */
   13853             :                 0x1, 0x1, 0x1, 0x1, /* auth[5] */
   13854             :                 0x1, 0x1, 0x1, 0x1, /* auth[6] */
   13855             :                 0x1, 0x1, 0x1, 0x1, /* auth[7] */
   13856             :                 0x1, 0x1, 0x1, 0x1, /* auth[8] */
   13857             :                 0x1, 0x1, 0x1, 0x1, /* auth[9] */
   13858             :                 0x1, 0x1, 0x1, 0x1, /* auth[10] */
   13859             :                 0x1, 0x1, 0x1, 0x1, /* auth[11] */
   13860             :                 0x1, 0x1, 0x1, 0x1, /* auth[12] */
   13861             :                 0x1, 0x1, 0x1, 0x1, /* auth[13] */
   13862             :                 0x1, 0x1, 0x1, 0x1, /* auth[14] */
   13863             :                 0x1, 0x1, 0x1, 0x1, /* auth[15] */
   13864             :                 0x1, 0x1, 0x1, 0x1, /* auth[16] */
   13865             :                 0x1, 0x1, 0x1, 0x1, /* auth[17] */
   13866             :                 0x1, 0x1, 0x1, 0x1, /* auth[18] */
   13867             :                 0x1, 0x1, 0x1, 0x1, /* auth[19] */
   13868             :                 0x1, 0x1, 0x1, 0x1, /* auth[20] */
   13869             :                 0x1, 0x1, 0x1, 0x1, /* auth[21] */
   13870             :                 0x1, 0x1, 0x1, 0x1, /* auth[22] */
   13871             :                 0x1, 0x1, 0x1, 0x1, /* auth[23] */
   13872             :                 0x1, 0x1, 0x1, 0x1, /* auth[24] */
   13873             :                 0x1, 0x1, 0x1, 0x1, /* auth[25] */
   13874             :                 0x1, 0x1, 0x1, 0x1, /* auth[26] */
   13875             :                 0x1, 0x1, 0x1, 0x1, /* auth[27] */
   13876             :                 0x1, 0x1, 0x1, 0x1, /* auth[28] */
   13877             :                 0x1, 0x1, 0x1, 0x1, /* auth[29] */
   13878             :                 0x1, 0x1, 0x1, 0x1, /* auth[30] */
   13879             :                 0x1, 0x1, 0x1, 0x1, /* auth[31] */
   13880             :         };
   13881             : 
   13882           1 :         ret = sid_parse(good_binary_sid, sizeof(good_binary_sid), sid);
   13883           1 :         if (ret == -1) {
   13884           0 :                 return false;
   13885             :         }
   13886           1 :         ret = sid_parse(long_binary_sid2, sizeof(long_binary_sid2), sid);
   13887           1 :         if (ret != -1) {
   13888           0 :                 return false;
   13889             :         }
   13890           1 :         ret = sid_parse(long_binary_sid, sizeof(long_binary_sid), sid);
   13891           1 :         if (ret != -1) {
   13892           0 :                 return false;
   13893             :         }
   13894           0 :         return true;
   13895             : }
   13896             : 
   13897             : /* Split a path name into filename and stream name components. Canonicalise
   13898             :  * such that an implicit $DATA token is always explicit.
   13899             :  *
   13900             :  * The "specification" of this function can be found in the
   13901             :  * run_local_stream_name() function in torture.c, I've tried those
   13902             :  * combinations against a W2k3 server.
   13903             :  */
   13904             : 
   13905           8 : static NTSTATUS split_ntfs_stream_name(TALLOC_CTX *mem_ctx, const char *fname,
   13906             :                                        char **pbase, char **pstream)
   13907             : {
   13908           8 :         char *base = NULL;
   13909           8 :         char *stream = NULL;
   13910           8 :         char *sname; /* stream name */
   13911           8 :         const char *stype; /* stream type */
   13912             : 
   13913           8 :         DEBUG(10, ("split_ntfs_stream_name called for [%s]\n", fname));
   13914             : 
   13915           8 :         sname = strchr_m(fname, ':');
   13916             : 
   13917           8 :         if (sname == NULL) {
   13918           1 :                 if (pbase != NULL) {
   13919           1 :                         base = talloc_strdup(mem_ctx, fname);
   13920           1 :                         NT_STATUS_HAVE_NO_MEMORY(base);
   13921             :                 }
   13922           1 :                 goto done;
   13923             :         }
   13924             : 
   13925           7 :         if (pbase != NULL) {
   13926           7 :                 base = talloc_strndup(mem_ctx, fname, PTR_DIFF(sname, fname));
   13927           7 :                 NT_STATUS_HAVE_NO_MEMORY(base);
   13928             :         }
   13929             : 
   13930           7 :         sname += 1;
   13931             : 
   13932           7 :         stype = strchr_m(sname, ':');
   13933             : 
   13934           7 :         if (stype == NULL) {
   13935           2 :                 sname = talloc_strdup(mem_ctx, sname);
   13936           2 :                 stype = "$DATA";
   13937             :         }
   13938             :         else {
   13939           5 :                 if (strcasecmp_m(stype, ":$DATA") != 0) {
   13940             :                         /*
   13941             :                          * If there is an explicit stream type, so far we only
   13942             :                          * allow $DATA. Is there anything else allowed? -- vl
   13943             :                          */
   13944           3 :                         DEBUG(10, ("[%s] is an invalid stream type\n", stype));
   13945           3 :                         TALLOC_FREE(base);
   13946           3 :                         return NT_STATUS_OBJECT_NAME_INVALID;
   13947             :                 }
   13948           2 :                 sname = talloc_strndup(mem_ctx, sname, PTR_DIFF(stype, sname));
   13949           2 :                 stype += 1;
   13950             :         }
   13951             : 
   13952           4 :         if (sname == NULL) {
   13953           0 :                 TALLOC_FREE(base);
   13954           0 :                 return NT_STATUS_NO_MEMORY;
   13955             :         }
   13956             : 
   13957           4 :         if (sname[0] == '\0') {
   13958             :                 /*
   13959             :                  * no stream name, so no stream
   13960             :                  */
   13961           1 :                 goto done;
   13962             :         }
   13963             : 
   13964           3 :         if (pstream != NULL) {
   13965           3 :                 stream = talloc_asprintf(mem_ctx, "%s:%s", sname, stype);
   13966           3 :                 if (stream == NULL) {
   13967           0 :                         TALLOC_FREE(sname);
   13968           0 :                         TALLOC_FREE(base);
   13969           0 :                         return NT_STATUS_NO_MEMORY;
   13970             :                 }
   13971             :                 /*
   13972             :                  * upper-case the type field
   13973             :                  */
   13974           3 :                 (void)strupper_m(strchr_m(stream, ':')+1);
   13975             :         }
   13976             : 
   13977           0 :  done:
   13978           5 :         if (pbase != NULL) {
   13979           5 :                 *pbase = base;
   13980             :         }
   13981           5 :         if (pstream != NULL) {
   13982           5 :                 *pstream = stream;
   13983             :         }
   13984           5 :         return NT_STATUS_OK;
   13985             : }
   13986             : 
   13987           8 : static bool test_stream_name(const char *fname, const char *expected_base,
   13988             :                              const char *expected_stream,
   13989             :                              NTSTATUS expected_status)
   13990             : {
   13991           8 :         NTSTATUS status;
   13992           8 :         char *base = NULL;
   13993           8 :         char *stream = NULL;
   13994             : 
   13995           8 :         status = split_ntfs_stream_name(talloc_tos(), fname, &base, &stream);
   13996           8 :         if (!NT_STATUS_EQUAL(status, expected_status)) {
   13997           0 :                 goto error;
   13998             :         }
   13999             : 
   14000           8 :         if (!NT_STATUS_IS_OK(status)) {
   14001           0 :                 return true;
   14002             :         }
   14003             : 
   14004           5 :         if (base == NULL) goto error;
   14005             : 
   14006           5 :         if (strcmp(expected_base, base) != 0) goto error;
   14007             : 
   14008           5 :         if ((expected_stream != NULL) && (stream == NULL)) goto error;
   14009           5 :         if ((expected_stream == NULL) && (stream != NULL)) goto error;
   14010             : 
   14011           5 :         if ((stream != NULL) && (strcmp(expected_stream, stream) != 0))
   14012           0 :                 goto error;
   14013             : 
   14014           5 :         TALLOC_FREE(base);
   14015           5 :         TALLOC_FREE(stream);
   14016           0 :         return true;
   14017             : 
   14018           0 :  error:
   14019           0 :         d_fprintf(stderr, "Do test_stream(%s, %s, %s, %s)\n",
   14020             :                   fname, expected_base ? expected_base : "<NULL>",
   14021             :                   expected_stream ? expected_stream : "<NULL>",
   14022             :                   nt_errstr(expected_status));
   14023           0 :         d_fprintf(stderr, "-> base=%s, stream=%s, status=%s\n",
   14024           0 :                   base ? base : "<NULL>", stream ? stream : "<NULL>",
   14025             :                   nt_errstr(status));
   14026           0 :         TALLOC_FREE(base);
   14027           0 :         TALLOC_FREE(stream);
   14028           0 :         return false;
   14029             : }
   14030             : 
   14031           1 : static bool run_local_stream_name(int dummy)
   14032             : {
   14033           1 :         bool ret = true;
   14034             : 
   14035           2 :         ret &= test_stream_name(
   14036           1 :                 "bla", "bla", NULL, NT_STATUS_OK);
   14037           2 :         ret &= test_stream_name(
   14038           1 :                 "bla::$DATA", "bla", NULL, NT_STATUS_OK);
   14039           2 :         ret &= test_stream_name(
   14040           1 :                 "bla:blub:", "bla", NULL, NT_STATUS_OBJECT_NAME_INVALID);
   14041           2 :         ret &= test_stream_name(
   14042           1 :                 "bla::", NULL, NULL, NT_STATUS_OBJECT_NAME_INVALID);
   14043           2 :         ret &= test_stream_name(
   14044           1 :                 "bla::123", "bla", NULL, NT_STATUS_OBJECT_NAME_INVALID);
   14045           2 :         ret &= test_stream_name(
   14046           1 :                 "bla:$DATA", "bla", "$DATA:$DATA", NT_STATUS_OK);
   14047           2 :         ret &= test_stream_name(
   14048           1 :                 "bla:x:$DATA", "bla", "x:$DATA", NT_STATUS_OK);
   14049           2 :         ret &= test_stream_name(
   14050           1 :                 "bla:x", "bla", "x:$DATA", NT_STATUS_OK);
   14051             : 
   14052           1 :         return ret;
   14053             : }
   14054             : 
   14055           2 : static bool data_blob_equal(DATA_BLOB a, DATA_BLOB b)
   14056             : {
   14057           2 :         if (a.length != b.length) {
   14058           0 :                 printf("a.length=%d != b.length=%d\n",
   14059           0 :                        (int)a.length, (int)b.length);
   14060           0 :                 return false;
   14061             :         }
   14062           2 :         if (memcmp(a.data, b.data, a.length) != 0) {
   14063           0 :                 printf("a.data and b.data differ\n");
   14064           0 :                 return false;
   14065             :         }
   14066           0 :         return true;
   14067             : }
   14068             : 
   14069           1 : static bool run_local_memcache(int dummy)
   14070             : {
   14071           1 :         struct memcache *cache;
   14072           1 :         DATA_BLOB k1, k2, k3, k4, k5;
   14073           1 :         DATA_BLOB d1, d3;
   14074           1 :         DATA_BLOB v1, v3;
   14075             : 
   14076           1 :         TALLOC_CTX *mem_ctx;
   14077           1 :         char *ptr1 = NULL;
   14078           1 :         char *ptr2 = NULL;
   14079           1 :         char *ptr3 = NULL;
   14080             : 
   14081           1 :         char *str1, *str2;
   14082           1 :         size_t size1, size2;
   14083           1 :         bool ret = false;
   14084             : 
   14085           1 :         mem_ctx = talloc_init("foo");
   14086           1 :         if (mem_ctx == NULL) {
   14087           0 :                 return false;
   14088             :         }
   14089             : 
   14090             :         /* STAT_CACHE TESTS */
   14091             : 
   14092           1 :         cache = memcache_init(NULL, sizeof(void *) == 8 ? 200 : 100);
   14093             : 
   14094           1 :         if (cache == NULL) {
   14095           0 :                 printf("memcache_init failed\n");
   14096           0 :                 return false;
   14097             :         }
   14098             : 
   14099           1 :         d1 = data_blob_const("d1", 2);
   14100           1 :         d3 = data_blob_const("d3", 2);
   14101             : 
   14102           1 :         k1 = data_blob_const("d1", 2);
   14103           1 :         k2 = data_blob_const("d2", 2);
   14104           1 :         k3 = data_blob_const("d3", 2);
   14105           1 :         k4 = data_blob_const("d4", 2);
   14106           1 :         k5 = data_blob_const("d5", 2);
   14107             : 
   14108           1 :         memcache_add(cache, STAT_CACHE, k1, d1);
   14109             : 
   14110           1 :         if (!memcache_lookup(cache, STAT_CACHE, k1, &v1)) {
   14111           0 :                 printf("could not find k1\n");
   14112           0 :                 return false;
   14113             :         }
   14114           1 :         if (!data_blob_equal(d1, v1)) {
   14115           0 :                 return false;
   14116             :         }
   14117             : 
   14118           1 :         memcache_add(cache, STAT_CACHE, k1, d3);
   14119             : 
   14120           1 :         if (!memcache_lookup(cache, STAT_CACHE, k1, &v3)) {
   14121           0 :                 printf("could not find replaced k1\n");
   14122           0 :                 return false;
   14123             :         }
   14124           1 :         if (!data_blob_equal(d3, v3)) {
   14125           0 :                 return false;
   14126             :         }
   14127             : 
   14128           1 :         TALLOC_FREE(cache);
   14129             : 
   14130             :         /* GETWD_CACHE TESTS */
   14131           1 :         str1 = talloc_strdup(mem_ctx, "string1");
   14132           1 :         if (str1 == NULL) {
   14133           0 :                 return false;
   14134             :         }
   14135           1 :         ptr2 = str1; /* Keep an alias for comparison. */
   14136             : 
   14137           1 :         str2 = talloc_strdup(mem_ctx, "string2");
   14138           1 :         if (str2 == NULL) {
   14139           0 :                 return false;
   14140             :         }
   14141             : 
   14142           1 :         cache = memcache_init(NULL, sizeof(void *) == 8 ? 200 : 100);
   14143           1 :         if (cache == NULL) {
   14144           0 :                 printf("memcache_init failed\n");
   14145           0 :                 return false;
   14146             :         }
   14147             : 
   14148           1 :         memcache_add_talloc(cache, GETWD_CACHE, k2, &str1);
   14149             :         /* str1 == NULL now. */
   14150           1 :         ptr1 = memcache_lookup_talloc(cache, GETWD_CACHE, k2);
   14151           1 :         if (ptr1 == NULL) {
   14152           0 :                 printf("could not find k2\n");
   14153           0 :                 return false;
   14154             :         }
   14155           1 :         if (ptr1 != ptr2) {
   14156           0 :                 printf("fetch of k2 got wrong string\n");
   14157           0 :                 return false;
   14158             :         }
   14159             : 
   14160             :         /* Add a blob to ensure k2 gets purged. */
   14161           1 :         d3 = data_blob_talloc_zero(mem_ctx, 180);
   14162           1 :         memcache_add(cache, STAT_CACHE, k3, d3);
   14163             : 
   14164           1 :         ptr2 = memcache_lookup_talloc(cache, GETWD_CACHE, k2);
   14165           1 :         if (ptr2 != NULL) {
   14166           0 :                 printf("Did find k2, should have been purged\n");
   14167           0 :                 return false;
   14168             :         }
   14169             : 
   14170             :         /*
   14171             :          * Test that talloc size also is accounted in memcache and
   14172             :          * causes purge of other object.
   14173             :          */
   14174             : 
   14175           1 :         str1 = talloc_zero_size(mem_ctx, 100);
   14176           1 :         str2 = talloc_zero_size(mem_ctx, 100);
   14177             : 
   14178           1 :         memcache_add_talloc(cache, GETWD_CACHE, k4, &str1);
   14179           1 :         memcache_add_talloc(cache, GETWD_CACHE, k5, &str1);
   14180             : 
   14181           1 :         ptr3 = memcache_lookup_talloc(cache, GETWD_CACHE, k4);
   14182           1 :         if (ptr3 != NULL) {
   14183           0 :                 printf("Did find k4, should have been purged\n");
   14184           0 :                 return false;
   14185             :         }
   14186             : 
   14187             :         /*
   14188             :          * Test that adding a duplicate non-talloced
   14189             :          * key/value on top of a talloced key/value takes account
   14190             :          * of the talloc_freed value size.
   14191             :          */
   14192           1 :         TALLOC_FREE(cache);
   14193           1 :         TALLOC_FREE(mem_ctx);
   14194             : 
   14195           1 :         mem_ctx = talloc_init("key_replace");
   14196           1 :         if (mem_ctx == NULL) {
   14197           0 :                 return false;
   14198             :         }
   14199             : 
   14200           1 :         cache = memcache_init(NULL, sizeof(void *) == 8 ? 200 : 100);
   14201           1 :         if (cache == NULL) {
   14202           0 :                 return false;
   14203             :         }
   14204             : 
   14205             :         /*
   14206             :          * Add a 100 byte talloced string. This will
   14207             :          * store a (4 or 8 byte) pointer and record the
   14208             :          * total talloced size.
   14209             :          */
   14210           1 :         str1 = talloc_zero_size(mem_ctx, 100);
   14211           1 :         memcache_add_talloc(cache, GETWD_CACHE, k4, &str1);
   14212             :         /*
   14213             :          * Now overwrite with a small talloced
   14214             :          * value. This should fit in the existing size
   14215             :          * and the total talloced size should be removed
   14216             :          * from the cache size.
   14217             :          */
   14218           1 :         str1 = talloc_zero_size(mem_ctx, 2);
   14219           1 :         memcache_add_talloc(cache, GETWD_CACHE, k4, &str1);
   14220             :         /*
   14221             :          * Now store a 20 byte string. If the
   14222             :          * total talloced size wasn't accounted for
   14223             :          * and removed in the overwrite, then this
   14224             :          * will evict k4.
   14225             :          */
   14226           1 :         str2 = talloc_zero_size(mem_ctx, 20);
   14227           1 :         memcache_add_talloc(cache, GETWD_CACHE, k5, &str2);
   14228             : 
   14229           1 :         ptr3 = memcache_lookup_talloc(cache, GETWD_CACHE, k4);
   14230           1 :         if (ptr3 == NULL) {
   14231           0 :                 printf("Did not find k4, should not have been purged\n");
   14232           0 :                 return false;
   14233             :         }
   14234             : 
   14235           1 :         TALLOC_FREE(cache);
   14236           1 :         TALLOC_FREE(mem_ctx);
   14237             : 
   14238           1 :         mem_ctx = talloc_init("foo");
   14239           1 :         if (mem_ctx == NULL) {
   14240           0 :                 return false;
   14241             :         }
   14242             : 
   14243           1 :         cache = memcache_init(NULL, 0);
   14244           1 :         if (cache == NULL) {
   14245           0 :                 return false;
   14246             :         }
   14247             : 
   14248           1 :         str1 = talloc_strdup(mem_ctx, "string1");
   14249           1 :         if (str1 == NULL) {
   14250           0 :                 return false;
   14251             :         }
   14252           1 :         str2 = talloc_strdup(mem_ctx, "string2");
   14253           1 :         if (str2 == NULL) {
   14254           0 :                 return false;
   14255             :         }
   14256           1 :         memcache_add_talloc(cache, SINGLETON_CACHE_TALLOC,
   14257             :                             data_blob_string_const("torture"), &str1);
   14258           1 :         size1 = talloc_total_size(cache);
   14259             : 
   14260           1 :         memcache_add_talloc(cache, SINGLETON_CACHE_TALLOC,
   14261             :                             data_blob_string_const("torture"), &str2);
   14262           1 :         size2 = talloc_total_size(cache);
   14263             : 
   14264           1 :         printf("size1=%d, size2=%d\n", (int)size1, (int)size2);
   14265             : 
   14266           1 :         if (size2 > size1) {
   14267           0 :                 printf("memcache leaks memory!\n");
   14268           0 :                 goto fail;
   14269             :         }
   14270             : 
   14271           0 :         ret = true;
   14272           1 :  fail:
   14273           1 :         TALLOC_FREE(cache);
   14274           1 :         return ret;
   14275             : }
   14276             : 
   14277         200 : static void wbclient_done(struct tevent_req *req)
   14278             : {
   14279           0 :         wbcErr wbc_err;
   14280           0 :         struct winbindd_response *wb_resp;
   14281         200 :         int *i = (int *)tevent_req_callback_data_void(req);
   14282             : 
   14283         200 :         wbc_err = wb_trans_recv(req, req, &wb_resp);
   14284         200 :         TALLOC_FREE(req);
   14285         200 :         *i += 1;
   14286         200 :         d_printf("wb_trans_recv %d returned %s\n", *i, wbcErrorString(wbc_err));
   14287         200 : }
   14288             : 
   14289           2 : static bool run_wbclient_multi_ping(int dummy)
   14290             : {
   14291           0 :         struct tevent_context *ev;
   14292           0 :         struct wb_context **wb_ctx;
   14293           0 :         struct winbindd_request wb_req;
   14294           2 :         bool result = false;
   14295           0 :         int i, j;
   14296             : 
   14297           2 :         BlockSignals(True, SIGPIPE);
   14298             : 
   14299           2 :         ev = tevent_context_init(talloc_tos());
   14300           2 :         if (ev == NULL) {
   14301           0 :                 goto fail;
   14302             :         }
   14303             : 
   14304           2 :         wb_ctx = talloc_array(ev, struct wb_context *, torture_nprocs);
   14305           2 :         if (wb_ctx == NULL) {
   14306           0 :                 goto fail;
   14307             :         }
   14308             : 
   14309           2 :         ZERO_STRUCT(wb_req);
   14310           2 :         wb_req.cmd = WINBINDD_PING;
   14311             : 
   14312           2 :         d_printf("torture_nprocs=%d, numops=%d\n", (int)torture_nprocs, (int)torture_numops);
   14313             : 
   14314           4 :         for (i=0; i<torture_nprocs; i++) {
   14315           2 :                 wb_ctx[i] = wb_context_init(ev, NULL);
   14316           2 :                 if (wb_ctx[i] == NULL) {
   14317           0 :                         goto fail;
   14318             :                 }
   14319         202 :                 for (j=0; j<torture_numops; j++) {
   14320           0 :                         struct tevent_req *req;
   14321         200 :                         req = wb_trans_send(ev, ev, wb_ctx[i],
   14322         200 :                                             (j % 2) == 0, &wb_req);
   14323         200 :                         if (req == NULL) {
   14324           0 :                                 goto fail;
   14325             :                         }
   14326         200 :                         tevent_req_set_callback(req, wbclient_done, &i);
   14327             :                 }
   14328             :         }
   14329             : 
   14330           2 :         i = 0;
   14331             : 
   14332         818 :         while (i < torture_nprocs * torture_numops) {
   14333         816 :                 tevent_loop_once(ev);
   14334             :         }
   14335             : 
   14336           2 :         result = true;
   14337           2 :  fail:
   14338           2 :         TALLOC_FREE(ev);
   14339           2 :         return result;
   14340             : }
   14341             : 
   14342           0 : static bool dbtrans_inc(struct db_context *db)
   14343             : {
   14344           0 :         struct db_record *rec;
   14345           0 :         uint32_t val;
   14346           0 :         bool ret = false;
   14347           0 :         NTSTATUS status;
   14348           0 :         TDB_DATA value;
   14349             : 
   14350           0 :         rec = dbwrap_fetch_locked(db, db, string_term_tdb_data("transtest"));
   14351           0 :         if (rec == NULL) {
   14352           0 :                 printf(__location__ "fetch_lock failed\n");
   14353           0 :                 return false;
   14354             :         }
   14355             : 
   14356           0 :         value = dbwrap_record_get_value(rec);
   14357             : 
   14358           0 :         if (value.dsize != sizeof(uint32_t)) {
   14359           0 :                 printf(__location__ "value.dsize = %d\n",
   14360           0 :                        (int)value.dsize);
   14361           0 :                 goto fail;
   14362             :         }
   14363             : 
   14364           0 :         memcpy(&val, value.dptr, sizeof(val));
   14365           0 :         val += 1;
   14366             : 
   14367           0 :         status = dbwrap_record_store(
   14368             :                 rec, make_tdb_data((uint8_t *)&val, sizeof(val)), 0);
   14369           0 :         if (!NT_STATUS_IS_OK(status)) {
   14370           0 :                 printf(__location__ "store failed: %s\n",
   14371             :                        nt_errstr(status));
   14372           0 :                 goto fail;
   14373             :         }
   14374             : 
   14375           0 :         ret = true;
   14376           0 : fail:
   14377           0 :         TALLOC_FREE(rec);
   14378           0 :         return ret;
   14379             : }
   14380             : 
   14381           0 : static bool run_local_dbtrans(int dummy)
   14382             : {
   14383           0 :         struct db_context *db;
   14384           0 :         struct db_record *rec;
   14385           0 :         NTSTATUS status;
   14386           0 :         uint32_t initial;
   14387           0 :         int res;
   14388           0 :         TDB_DATA value;
   14389             : 
   14390           0 :         db = db_open(talloc_tos(), "transtest.tdb", 0, TDB_DEFAULT,
   14391             :                      O_RDWR|O_CREAT, 0600, DBWRAP_LOCK_ORDER_1,
   14392             :                      DBWRAP_FLAG_NONE);
   14393           0 :         if (db == NULL) {
   14394           0 :                 printf("Could not open transtest.db\n");
   14395           0 :                 return false;
   14396             :         }
   14397             : 
   14398           0 :         res = dbwrap_transaction_start(db);
   14399           0 :         if (res != 0) {
   14400           0 :                 printf(__location__ "transaction_start failed\n");
   14401           0 :                 return false;
   14402             :         }
   14403             : 
   14404           0 :         rec = dbwrap_fetch_locked(db, db, string_term_tdb_data("transtest"));
   14405           0 :         if (rec == NULL) {
   14406           0 :                 printf(__location__ "fetch_lock failed\n");
   14407           0 :                 return false;
   14408             :         }
   14409             : 
   14410           0 :         value = dbwrap_record_get_value(rec);
   14411             : 
   14412           0 :         if (value.dptr == NULL) {
   14413           0 :                 initial = 0;
   14414           0 :                 status = dbwrap_record_store(
   14415             :                         rec, make_tdb_data((uint8_t *)&initial,
   14416             :                                            sizeof(initial)),
   14417             :                         0);
   14418           0 :                 if (!NT_STATUS_IS_OK(status)) {
   14419           0 :                         printf(__location__ "store returned %s\n",
   14420             :                                nt_errstr(status));
   14421           0 :                         return false;
   14422             :                 }
   14423             :         }
   14424             : 
   14425           0 :         TALLOC_FREE(rec);
   14426             : 
   14427           0 :         res = dbwrap_transaction_commit(db);
   14428           0 :         if (res != 0) {
   14429           0 :                 printf(__location__ "transaction_commit failed\n");
   14430           0 :                 return false;
   14431             :         }
   14432             : 
   14433           0 :         while (true) {
   14434           0 :                 uint32_t val, val2;
   14435           0 :                 int i;
   14436             : 
   14437           0 :                 res = dbwrap_transaction_start(db);
   14438           0 :                 if (res != 0) {
   14439           0 :                         printf(__location__ "transaction_start failed\n");
   14440           0 :                         break;
   14441             :                 }
   14442             : 
   14443           0 :                 status = dbwrap_fetch_uint32_bystring(db, "transtest", &val);
   14444           0 :                 if (!NT_STATUS_IS_OK(status)) {
   14445           0 :                         printf(__location__ "dbwrap_fetch_uint32 failed: %s\n",
   14446             :                                nt_errstr(status));
   14447           0 :                         break;
   14448             :                 }
   14449             : 
   14450           0 :                 for (i=0; i<10; i++) {
   14451           0 :                         if (!dbtrans_inc(db)) {
   14452           0 :                                 return false;
   14453             :                         }
   14454             :                 }
   14455             : 
   14456           0 :                 status = dbwrap_fetch_uint32_bystring(db, "transtest", &val2);
   14457           0 :                 if (!NT_STATUS_IS_OK(status)) {
   14458           0 :                         printf(__location__ "dbwrap_fetch_uint32 failed: %s\n",
   14459             :                                nt_errstr(status));
   14460           0 :                         break;
   14461             :                 }
   14462             : 
   14463           0 :                 if (val2 != val + 10) {
   14464           0 :                         printf(__location__ "val=%d, val2=%d\n",
   14465             :                                (int)val, (int)val2);
   14466           0 :                         break;
   14467             :                 }
   14468             : 
   14469           0 :                 printf("val2=%d\r", val2);
   14470             : 
   14471           0 :                 res = dbwrap_transaction_commit(db);
   14472           0 :                 if (res != 0) {
   14473           0 :                         printf(__location__ "transaction_commit failed\n");
   14474           0 :                         break;
   14475             :                 }
   14476             :         }
   14477             : 
   14478           0 :         TALLOC_FREE(db);
   14479           0 :         return true;
   14480             : }
   14481             : 
   14482             : /*
   14483             :  * Just a dummy test to be run under a debugger. There's no real way
   14484             :  * to inspect the tevent_poll specific function from outside of
   14485             :  * tevent_poll.c.
   14486             :  */
   14487             : 
   14488           1 : static bool run_local_tevent_poll(int dummy)
   14489             : {
   14490           1 :         struct tevent_context *ev;
   14491           1 :         struct tevent_fd *fd1, *fd2;
   14492           1 :         bool result = false;
   14493             : 
   14494           1 :         ev = tevent_context_init_byname(NULL, "poll");
   14495           1 :         if (ev == NULL) {
   14496           0 :                 d_fprintf(stderr, "tevent_context_init_byname failed\n");
   14497           0 :                 goto fail;
   14498             :         }
   14499             : 
   14500           1 :         fd1 = tevent_add_fd(ev, ev, 2, 0, NULL, NULL);
   14501           1 :         if (fd1 == NULL) {
   14502           0 :                 d_fprintf(stderr, "tevent_add_fd failed\n");
   14503           0 :                 goto fail;
   14504             :         }
   14505           1 :         fd2 = tevent_add_fd(ev, ev, 3, 0, NULL, NULL);
   14506           1 :         if (fd2 == NULL) {
   14507           0 :                 d_fprintf(stderr, "tevent_add_fd failed\n");
   14508           0 :                 goto fail;
   14509             :         }
   14510           1 :         TALLOC_FREE(fd2);
   14511             : 
   14512           1 :         fd2 = tevent_add_fd(ev, ev, 1, 0, NULL, NULL);
   14513           1 :         if (fd2 == NULL) {
   14514           0 :                 d_fprintf(stderr, "tevent_add_fd failed\n");
   14515           0 :                 goto fail;
   14516             :         }
   14517             : 
   14518           0 :         result = true;
   14519           1 : fail:
   14520           1 :         TALLOC_FREE(ev);
   14521           1 :         return result;
   14522             : }
   14523             : 
   14524           1 : static bool run_local_hex_encode_buf(int dummy)
   14525             : {
   14526           1 :         char buf[17];
   14527           1 :         uint8_t src[8];
   14528           1 :         size_t i;
   14529             : 
   14530           9 :         for (i=0; i<sizeof(src); i++) {
   14531           8 :                 src[i] = i;
   14532             :         }
   14533           1 :         hex_encode_buf(buf, src, sizeof(src));
   14534           1 :         if (strcmp(buf, "0001020304050607") != 0) {
   14535           0 :                 return false;
   14536             :         }
   14537           1 :         hex_encode_buf(buf, NULL, 0);
   14538           1 :         if (buf[0] != '\0') {
   14539           0 :                 return false;
   14540             :         }
   14541           0 :         return true;
   14542             : }
   14543             : 
   14544             : static const char *remove_duplicate_addrs2_test_strings_vector[] = {
   14545             :         "0.0.0.0",
   14546             :         "::0",
   14547             :         "1.2.3.1",
   14548             :         "0.0.0.0",
   14549             :         "0.0.0.0",
   14550             :         "1.2.3.2",
   14551             :         "1.2.3.3",
   14552             :         "1.2.3.4",
   14553             :         "1.2.3.5",
   14554             :         "::0",
   14555             :         "1.2.3.6",
   14556             :         "1.2.3.7",
   14557             :         "::0",
   14558             :         "::0",
   14559             :         "::0",
   14560             :         "1.2.3.8",
   14561             :         "1.2.3.9",
   14562             :         "1.2.3.10",
   14563             :         "1.2.3.11",
   14564             :         "1.2.3.12",
   14565             :         "1.2.3.13",
   14566             :         "1001:1111:1111:1000:0:1111:1111:1111",
   14567             :         "1.2.3.1",
   14568             :         "1.2.3.2",
   14569             :         "1.2.3.3",
   14570             :         "1.2.3.12",
   14571             :         "::0",
   14572             :         "::0"
   14573             : };
   14574             : 
   14575             : static const char *remove_duplicate_addrs2_test_strings_result[] = {
   14576             :         "1.2.3.1",
   14577             :         "1.2.3.2",
   14578             :         "1.2.3.3",
   14579             :         "1.2.3.4",
   14580             :         "1.2.3.5",
   14581             :         "1.2.3.6",
   14582             :         "1.2.3.7",
   14583             :         "1.2.3.8",
   14584             :         "1.2.3.9",
   14585             :         "1.2.3.10",
   14586             :         "1.2.3.11",
   14587             :         "1.2.3.12",
   14588             :         "1.2.3.13",
   14589             :         "1001:1111:1111:1000:0:1111:1111:1111"
   14590             : };
   14591             : 
   14592           1 : static bool run_local_remove_duplicate_addrs2(int dummy)
   14593             : {
   14594           1 :         struct samba_sockaddr test_vector[28];
   14595           1 :         size_t count, i;
   14596             : 
   14597             :         /* Construct the sockaddr_storage test vector. */
   14598          29 :         for (i = 0; i < 28; i++) {
   14599          28 :                 struct addrinfo hints;
   14600          28 :                 struct addrinfo *res = NULL;
   14601          28 :                 int ret;
   14602             : 
   14603          28 :                 memset(&hints, '\0', sizeof(hints));
   14604          28 :                 hints.ai_flags = AI_NUMERICHOST;
   14605          28 :                 ret = getaddrinfo(remove_duplicate_addrs2_test_strings_vector[i],
   14606             :                                 NULL,
   14607             :                                 &hints,
   14608             :                                 &res);
   14609          28 :                 if (ret) {
   14610           0 :                         fprintf(stderr, "getaddrinfo failed on [%s]\n",
   14611             :                                 remove_duplicate_addrs2_test_strings_vector[i]);
   14612           0 :                         return false;
   14613             :                 }
   14614          28 :                 memset(&test_vector[i], '\0', sizeof(test_vector[i]));
   14615          28 :                 memcpy(&test_vector[i].u.ss,
   14616          28 :                         res->ai_addr,
   14617          28 :                         res->ai_addrlen);
   14618          28 :                 freeaddrinfo(res);
   14619             :         }
   14620             : 
   14621           1 :         count = remove_duplicate_addrs2(test_vector, i);
   14622             : 
   14623           1 :         if (count != 14) {
   14624           0 :                 fprintf(stderr, "count wrong (%zu) should be 14\n",
   14625             :                         count);
   14626           0 :                 return false;
   14627             :         }
   14628             : 
   14629          15 :         for (i = 0; i < count; i++) {
   14630          14 :                 char addr[INET6_ADDRSTRLEN];
   14631             : 
   14632          14 :                 print_sockaddr(addr, sizeof(addr), &test_vector[i].u.ss);
   14633             : 
   14634          14 :                 if (strcmp(addr, remove_duplicate_addrs2_test_strings_result[i]) != 0) {
   14635           0 :                         fprintf(stderr, "mismatch on [%zu] [%s] [%s]\n",
   14636             :                                 i,
   14637             :                                 addr,
   14638             :                                 remove_duplicate_addrs2_test_strings_result[i]);
   14639           0 :                         return false;
   14640             :                 }
   14641             :         }
   14642             : 
   14643           1 :         printf("run_local_remove_duplicate_addrs2: success\n");
   14644           1 :         return true;
   14645             : }
   14646             : 
   14647           0 : static bool run_local_tdb_opener(int dummy)
   14648             : {
   14649           0 :         TDB_CONTEXT *t;
   14650           0 :         unsigned v = 0;
   14651             : 
   14652           0 :         while (1) {
   14653           0 :                 t = tdb_open("test.tdb", 1000, TDB_CLEAR_IF_FIRST,
   14654             :                              O_RDWR|O_CREAT, 0755);
   14655           0 :                 if (t == NULL) {
   14656           0 :                         perror("tdb_open failed");
   14657           0 :                         return false;
   14658             :                 }
   14659           0 :                 tdb_close(t);
   14660             : 
   14661           0 :                 v += 1;
   14662           0 :                 printf("\r%u", v);
   14663             :         }
   14664             :         return true;
   14665             : }
   14666             : 
   14667           0 : static bool run_local_tdb_writer(int dummy)
   14668             : {
   14669           0 :         TDB_CONTEXT *t;
   14670           0 :         unsigned v = 0;
   14671           0 :         TDB_DATA val;
   14672             : 
   14673           0 :         t = tdb_open("test.tdb", 1000, 0, O_RDWR|O_CREAT, 0755);
   14674           0 :         if (t == 0) {
   14675           0 :                 perror("tdb_open failed");
   14676           0 :                 return 1;
   14677             :         }
   14678             : 
   14679           0 :         val.dptr = (uint8_t *)&v;
   14680           0 :         val.dsize = sizeof(v);
   14681             : 
   14682           0 :         while (1) {
   14683           0 :                 TDB_DATA data;
   14684           0 :                 int ret;
   14685             : 
   14686           0 :                 ret = tdb_store(t, val, val, 0);
   14687           0 :                 if (ret != 0) {
   14688           0 :                         printf("%s\n", tdb_errorstr(t));
   14689             :                 }
   14690           0 :                 v += 1;
   14691           0 :                 printf("\r%u", v);
   14692             : 
   14693           0 :                 data = tdb_fetch(t, val);
   14694           0 :                 if (data.dptr != NULL) {
   14695           0 :                         SAFE_FREE(data.dptr);
   14696             :                 }
   14697             :         }
   14698             :         return true;
   14699             : }
   14700             : 
   14701           1 : static bool run_local_canonicalize_path(int dummy)
   14702             : {
   14703           1 :         const char *src[] = {
   14704             :                         "/foo/..",
   14705             :                         "/..",
   14706             :                         "/foo/bar/../baz",
   14707             :                         "/foo/././",
   14708             :                         "/../foo",
   14709             :                         ".././././",
   14710             :                         ".././././../../../boo",
   14711             :                         "./..",
   14712             :                         "/",
   14713             :                         "/../../",
   14714             :                         "/foo/../",
   14715             :                         "/./././",
   14716             :                         "/./././.",
   14717             :                         "/.../././.",
   14718             :                         "/./././.foo",
   14719             :                         "/./././.foo.",
   14720             :                         "/./././foo.",
   14721             :                         "/foo/bar/..",
   14722             :                         "/foo/bar/../baz/",
   14723             :                         "////////////////",
   14724             :                         "/////////./././././.",
   14725             :                         "/./.././../.boo/../baz",
   14726             :                         "/a/component/path",
   14727             :                         "/a/component/path/",
   14728             :                         "/a/component/path/..",
   14729             :                         "/a/component/../path/",
   14730             :                         "///a/./././///component/../////path/",
   14731             :                         NULL
   14732             :                         };
   14733           1 :         const char *dst[] = {
   14734             :                         "/",
   14735             :                         "/",
   14736             :                         "/foo/baz",
   14737             :                         "/foo",
   14738             :                         "/foo",
   14739             :                         "/",
   14740             :                         "/boo",
   14741             :                         "/",
   14742             :                         "/",
   14743             :                         "/",
   14744             :                         "/",
   14745             :                         "/",
   14746             :                         "/",
   14747             :                         "/...",
   14748             :                         "/.foo",
   14749             :                         "/.foo.",
   14750             :                         "/foo.",
   14751             :                         "/foo",
   14752             :                         "/foo/baz",
   14753             :                         "/",
   14754             :                         "/",
   14755             :                         "/baz",
   14756             :                         "/a/component/path",
   14757             :                         "/a/component/path",
   14758             :                         "/a/component",
   14759             :                         "/a/path",
   14760             :                         "/a/path",
   14761             :                         NULL
   14762             :                         };
   14763           1 :         unsigned int i;
   14764             : 
   14765          28 :         for (i = 0; src[i] != NULL; i++) {
   14766          27 :                 char *d = canonicalize_absolute_path(talloc_tos(), src[i]);
   14767          27 :                 if (d == NULL) {
   14768           0 :                         perror("talloc fail\n");
   14769           0 :                         return false;
   14770             :                 }
   14771          27 :                 if (strcmp(d, dst[i]) != 0) {
   14772           0 :                         d_fprintf(stderr,
   14773             :                                 "canonicalize mismatch %s -> %s != %s",
   14774             :                                 src[i], d, dst[i]);
   14775           0 :                         return false;
   14776             :                 }
   14777          27 :                 talloc_free(d);
   14778             :         }
   14779           0 :         return true;
   14780             : }
   14781             : struct session_setup_nt1_truncated_state {
   14782             :         uint16_t vwv[13];
   14783             :         uint8_t bytes[20];
   14784             : };
   14785             : 
   14786             : static void smb1_session_setup_nt1_truncated_done(struct tevent_req *subreq);
   14787             : 
   14788           2 : static struct tevent_req *smb1_session_setup_nt1_truncated_send(
   14789             :                 TALLOC_CTX *mem_ctx,
   14790             :                 struct tevent_context *ev,
   14791             :                 struct smbXcli_conn *conn)
   14792             : {
   14793           2 :         uint16_t *vwv = NULL;
   14794           2 :         uint8_t *bytes = NULL;
   14795           2 :         const char *pass = "12345678";
   14796           2 :         const char *uname = "z";
   14797           2 :         struct session_setup_nt1_truncated_state *state = NULL;
   14798           2 :         struct tevent_req *req = NULL;
   14799           2 :         struct tevent_req *subreq = NULL;
   14800             : 
   14801           2 :         req = tevent_req_create(mem_ctx,
   14802             :                                 &state,
   14803             :                                 struct session_setup_nt1_truncated_state);
   14804           2 :         if (req == NULL) {
   14805           0 :                 return NULL;
   14806             :         }
   14807           2 :         vwv = &state->vwv[0];
   14808           2 :         bytes = &state->bytes[0];
   14809             : 
   14810           2 :         SCVAL(vwv+0,  0, 0xff);
   14811           2 :         SCVAL(vwv+0,  1, 0);
   14812           2 :         SSVAL(vwv+1,  0, 0);
   14813           2 :         SSVAL(vwv+2,  0, 8192);
   14814           2 :         SSVAL(vwv+3,  0, 2);
   14815           2 :         SSVAL(vwv+4,  0, 1);
   14816           2 :         SIVAL(vwv+5,  0, 0);
   14817           2 :         SSVAL(vwv+7,  0, strlen(pass)); /* OEMPasswordLen */
   14818           2 :         SSVAL(vwv+8,  0, 0); /* UnicodePasswordLen */
   14819           2 :         SSVAL(vwv+9,  0, 0); /* reserved */
   14820           2 :         SSVAL(vwv+10, 0, 0); /* reserved */
   14821           2 :         SIVAL(vwv+11, 0, CAP_STATUS32);
   14822             : 
   14823           2 :         memcpy(bytes, pass, strlen(pass));
   14824           2 :         bytes += strlen(pass);
   14825           2 :         memcpy(bytes, uname, strlen(uname)+1);
   14826             : 
   14827           2 :         subreq = smb1cli_req_send(state, ev, conn,
   14828             :                                   SMBsesssetupX,
   14829             :                                   0, /*  additional_flags */
   14830             :                                   0, /*  clear_flags */
   14831             :                                   0, /*  additional_flags2 */
   14832             :                                   0, /*  clear_flags2 */
   14833             :                                   10000, /* timeout_msec */
   14834           2 :                                   getpid(),
   14835             :                                   NULL, /* tcon */
   14836             :                                   NULL, /* session */
   14837             :                                   13, /* wct */
   14838           2 :                                   state->vwv,
   14839           2 :                                   strlen(pass), /* Truncate length at password. */
   14840           2 :                                   state->bytes);
   14841           2 :         if (tevent_req_nomem(subreq, req)) {
   14842           0 :                 return tevent_req_post(req, ev);
   14843             :         }
   14844           2 :         tevent_req_set_callback(subreq,
   14845             :                                 smb1_session_setup_nt1_truncated_done,
   14846             :                                 req);
   14847           2 :         return req;
   14848             : }
   14849             : 
   14850           2 : static void smb1_session_setup_nt1_truncated_done(struct tevent_req *subreq)
   14851             : {
   14852           0 :         struct tevent_req *req =
   14853           2 :                 tevent_req_callback_data(subreq,
   14854             :                 struct tevent_req);
   14855           0 :         struct session_setup_nt1_truncated_state *state =
   14856           2 :                 tevent_req_data(req,
   14857             :                 struct session_setup_nt1_truncated_state);
   14858           0 :         NTSTATUS status;
   14859           2 :         struct smb1cli_req_expected_response expected[] = {
   14860             :         {
   14861             :                 .status = NT_STATUS_OK,
   14862             :                 .wct    = 3,
   14863             :         },
   14864             :         };
   14865             : 
   14866           2 :         status = smb1cli_req_recv(subreq, state,
   14867             :                                   NULL,
   14868             :                                   NULL,
   14869             :                                   NULL,
   14870             :                                   NULL,
   14871             :                                   NULL, /* pvwv_offset */
   14872             :                                   NULL,
   14873             :                                   NULL,
   14874             :                                   NULL, /* pbytes_offset */
   14875             :                                   NULL,
   14876             :                                   expected, ARRAY_SIZE(expected));
   14877           2 :         TALLOC_FREE(subreq);
   14878           2 :         if (tevent_req_nterror(req, status)) {
   14879           0 :                 return;
   14880             :         }
   14881           2 :         tevent_req_done(req);
   14882             : }
   14883             : 
   14884           2 : static NTSTATUS smb1_session_setup_nt1_truncated_recv(struct tevent_req *req)
   14885             : {
   14886           2 :         return tevent_req_simple_recv_ntstatus(req);
   14887             : }
   14888             : 
   14889           2 : static bool run_smb1_truncated_sesssetup(int dummy)
   14890             : {
   14891           0 :         struct tevent_context *ev;
   14892           0 :         struct tevent_req *req;
   14893           0 :         struct smbXcli_conn *conn;
   14894           0 :         struct sockaddr_storage ss;
   14895           0 :         NTSTATUS status;
   14896           0 :         int fd;
   14897           0 :         bool ok;
   14898             : 
   14899           2 :         printf("Starting send truncated SMB1 sesssetup.\n");
   14900             : 
   14901           2 :         ok = resolve_name(host, &ss, 0x20, true);
   14902           2 :         if (!ok) {
   14903           0 :                 d_fprintf(stderr, "Could not resolve name %s\n", host);
   14904           0 :                 return false;
   14905             :         }
   14906             : 
   14907           2 :         status = open_socket_out(&ss, 445, 10000, &fd);
   14908           2 :         if (!NT_STATUS_IS_OK(status)) {
   14909           0 :                 d_fprintf(stderr, "open_socket_out failed: %s\n",
   14910             :                           nt_errstr(status));
   14911           0 :                 return false;
   14912             :         }
   14913             : 
   14914           2 :         conn = smbXcli_conn_create(talloc_tos(), fd, host, SMB_SIGNING_OFF, 0,
   14915             :                                    NULL, 0, NULL);
   14916           2 :         if (conn == NULL) {
   14917           0 :                 d_fprintf(stderr, "smbXcli_conn_create failed\n");
   14918           0 :                 return false;
   14919             :         }
   14920             : 
   14921           2 :         status = smbXcli_negprot(conn,
   14922             :                                  0,
   14923             :                                  PROTOCOL_NT1,
   14924             :                                  PROTOCOL_NT1,
   14925             :                                  NULL,
   14926             :                                  NULL,
   14927             :                                  NULL);
   14928           2 :         if (!NT_STATUS_IS_OK(status)) {
   14929           0 :                 d_fprintf(stderr, "smbXcli_negprot failed!\n");
   14930           0 :                 return false;
   14931             :         }
   14932             : 
   14933           2 :         ev = samba_tevent_context_init(talloc_tos());
   14934           2 :         if (ev == NULL) {
   14935           0 :                 d_fprintf(stderr, "samba_tevent_context_init failed\n");
   14936           0 :                 return false;
   14937             :         }
   14938             : 
   14939           2 :         req = smb1_session_setup_nt1_truncated_send(ev, ev, conn);
   14940           2 :         if (req == NULL) {
   14941           0 :                 d_fprintf(stderr, "smb1_session_setup_nt1_truncated_send failed\n");
   14942           0 :                 return false;
   14943             :         }
   14944             : 
   14945           2 :         ok = tevent_req_poll_ntstatus(req, ev, &status);
   14946           2 :         if (!ok) {
   14947           0 :                 d_fprintf(stderr, "tevent_req_poll failed with status %s\n",
   14948             :                         nt_errstr(status));
   14949           0 :                 return false;
   14950             :         }
   14951             : 
   14952           2 :         status = smb1_session_setup_nt1_truncated_recv(req);
   14953           2 :         if (!NT_STATUS_IS_OK(status)) {
   14954           0 :                 d_fprintf(stderr, "smb1_session_setup_nt1_truncated_recv returned "
   14955             :                           "%s, expected NT_STATUS_OK\n",
   14956             :                           nt_errstr(status));
   14957           0 :                 return false;
   14958             :         }
   14959             : 
   14960           2 :         TALLOC_FREE(conn);
   14961           2 :         return true;
   14962             : }
   14963             : 
   14964             : struct smb1_negotiate_exit_state {
   14965             :         int dummy;
   14966             : };
   14967             : 
   14968             : static void smb1_negotiate_exit_done(struct tevent_req *subreq);
   14969             : 
   14970           4 : static struct tevent_req *smb1_negotiate_exit_send(
   14971             :                 TALLOC_CTX *mem_ctx,
   14972             :                 struct tevent_context *ev,
   14973             :                 struct smbXcli_conn *conn)
   14974             : {
   14975           4 :         struct smb1_negotiate_exit_state *state = NULL;
   14976           4 :         struct tevent_req *req = NULL;
   14977           4 :         struct tevent_req *subreq = NULL;
   14978             : 
   14979           4 :         req = tevent_req_create(mem_ctx,
   14980             :                                 &state,
   14981             :                                 struct smb1_negotiate_exit_state);
   14982           4 :         if (req == NULL) {
   14983           0 :                 return NULL;
   14984             :         }
   14985           4 :         subreq = smb1cli_req_send(state, ev, conn,
   14986             :                                   SMBexit,
   14987             :                                   0, /*  additional_flags */
   14988             :                                   0, /*  clear_flags */
   14989             :                                   0, /*  additional_flags2 */
   14990             :                                   0, /*  clear_flags2 */
   14991             :                                   10000, /* timeout_msec */
   14992           4 :                                   getpid(),
   14993             :                                   NULL, /* tcon */
   14994             :                                   NULL, /* session */
   14995             :                                   0, /* wct */
   14996             :                                   NULL,
   14997             :                                   0,
   14998             :                                   NULL);
   14999           4 :         if (tevent_req_nomem(subreq, req)) {
   15000           0 :                 return tevent_req_post(req, ev);
   15001             :         }
   15002           4 :         tevent_req_set_callback(subreq,
   15003             :                                 smb1_negotiate_exit_done,
   15004             :                                 req);
   15005           4 :         return req;
   15006             : }
   15007             : 
   15008           4 : static void smb1_negotiate_exit_done(struct tevent_req *subreq)
   15009             : {
   15010           0 :         struct tevent_req *req =
   15011           4 :                 tevent_req_callback_data(subreq,
   15012             :                 struct tevent_req);
   15013           0 :         struct smb1_negotiate_exit_state *state =
   15014           4 :                 tevent_req_data(req,
   15015             :                 struct smb1_negotiate_exit_state);
   15016           0 :         NTSTATUS status;
   15017           4 :         struct smb1cli_req_expected_response expected[] = {
   15018             :         {
   15019             :                 .status = NT_STATUS_OK,
   15020             :                 .wct    = 0,
   15021             :         },
   15022             :         };
   15023             : 
   15024           4 :         status = smb1cli_req_recv(subreq, state,
   15025             :                                   NULL,
   15026             :                                   NULL,
   15027             :                                   NULL,
   15028             :                                   NULL,
   15029             :                                   NULL, /* pvwv_offset */
   15030             :                                   NULL,
   15031             :                                   NULL,
   15032             :                                   NULL, /* pbytes_offset */
   15033             :                                   NULL,
   15034             :                                   expected, ARRAY_SIZE(expected));
   15035           4 :         TALLOC_FREE(subreq);
   15036           4 :         if (tevent_req_nterror(req, status)) {
   15037           4 :                 return;
   15038             :         }
   15039           0 :         tevent_req_done(req);
   15040             : }
   15041             : 
   15042           4 : static NTSTATUS smb1_negotiate_exit_recv(struct tevent_req *req)
   15043             : {
   15044           4 :         return tevent_req_simple_recv_ntstatus(req);
   15045             : }
   15046             : 
   15047           4 : static bool do_smb1_exit(TALLOC_CTX *mem_ctx,
   15048             :                          struct tevent_context *ev,
   15049             :                          struct smbXcli_conn *conn)
   15050             : {
   15051           0 :         struct tevent_req *req;
   15052           0 :         bool ok;
   15053           0 :         NTSTATUS status;
   15054           4 :         NTSTATUS expected_status = NT_STATUS_DOS(ERRSRV, ERRinvnid);;
   15055             : 
   15056           4 :         req = smb1_negotiate_exit_send(ev, ev, conn);
   15057           4 :         if (req == NULL) {
   15058           0 :                 d_fprintf(stderr, "smb1_negotiate_exit_send failed\n");
   15059           0 :                 return false;
   15060             :         }
   15061             : 
   15062           4 :         ok = tevent_req_poll_ntstatus(req, ev, &status);
   15063           4 :         if (!ok) {
   15064           0 :                 d_fprintf(stderr, "tevent_req_poll failed with status %s\n",
   15065             :                         nt_errstr(status));
   15066           0 :                 return false;
   15067             :         }
   15068             : 
   15069           4 :         status = smb1_negotiate_exit_recv(req);
   15070           4 :         if (!NT_STATUS_EQUAL(status, expected_status)) {
   15071           0 :                 d_fprintf(stderr, "smb1_negotiate_exit_recv returned "
   15072             :                           "%s, expected ERRSRV, ERRinvnid\n",
   15073             :                           nt_errstr(status));
   15074           0 :                 return false;
   15075             :         }
   15076           4 :         return true;
   15077             : }
   15078             : 
   15079           2 : static bool run_smb1_negotiate_exit(int dummy)
   15080             : {
   15081           0 :         struct tevent_context *ev;
   15082           0 :         struct smbXcli_conn *conn;
   15083           0 :         struct sockaddr_storage ss;
   15084           0 :         NTSTATUS status;
   15085           0 :         int fd;
   15086           0 :         bool ok;
   15087             : 
   15088           2 :         printf("Starting send SMB1 negotiate+exit.\n");
   15089             : 
   15090           2 :         ok = resolve_name(host, &ss, 0x20, true);
   15091           2 :         if (!ok) {
   15092           0 :                 d_fprintf(stderr, "Could not resolve name %s\n", host);
   15093           0 :                 return false;
   15094             :         }
   15095             : 
   15096           2 :         status = open_socket_out(&ss, 445, 10000, &fd);
   15097           2 :         if (!NT_STATUS_IS_OK(status)) {
   15098           0 :                 d_fprintf(stderr, "open_socket_out failed: %s\n",
   15099             :                           nt_errstr(status));
   15100           0 :                 return false;
   15101             :         }
   15102             : 
   15103           2 :         conn = smbXcli_conn_create(talloc_tos(), fd, host, SMB_SIGNING_OFF, 0,
   15104             :                                    NULL, 0, NULL);
   15105           2 :         if (conn == NULL) {
   15106           0 :                 d_fprintf(stderr, "smbXcli_conn_create failed\n");
   15107           0 :                 return false;
   15108             :         }
   15109             : 
   15110           2 :         status = smbXcli_negprot(conn,
   15111             :                                  0,
   15112             :                                  PROTOCOL_NT1,
   15113             :                                  PROTOCOL_NT1,
   15114             :                                  NULL,
   15115             :                                  NULL,
   15116             :                                  NULL);
   15117           2 :         if (!NT_STATUS_IS_OK(status)) {
   15118           0 :                 d_fprintf(stderr, "smbXcli_negprot failed!\n");
   15119           0 :                 return false;
   15120             :         }
   15121             : 
   15122           2 :         ev = samba_tevent_context_init(talloc_tos());
   15123           2 :         if (ev == NULL) {
   15124           0 :                 d_fprintf(stderr, "samba_tevent_context_init failed\n");
   15125           0 :                 return false;
   15126             :         }
   15127             : 
   15128             :         /*
   15129             :          * Call do_smb1_exit twice to catch a server crash, the
   15130             :          * server sends the first return code then crashes.
   15131             :          */
   15132           2 :         ok = do_smb1_exit(ev, ev, conn);
   15133           2 :         if (!ok) {
   15134           0 :                 d_fprintf(stderr, "do_smb1_exit (1) failed\n");
   15135           0 :                 return false;
   15136             :         }
   15137           2 :         ok = do_smb1_exit(ev, ev, conn);
   15138           2 :         if (!ok) {
   15139           0 :                 d_fprintf(stderr, "do_smb1_exit (2) failed\n");
   15140           0 :                 return false;
   15141             :         }
   15142             : 
   15143           2 :         TALLOC_FREE(conn);
   15144           2 :         return true;
   15145             : }
   15146             : 
   15147           2 : static bool run_smb1_negotiate_tcon(int dummy)
   15148             : {
   15149           2 :         struct cli_state *cli = NULL;
   15150           2 :         uint16_t cnum = 0;
   15151           2 :         uint16_t max_xmit = 0;
   15152           0 :         NTSTATUS status;
   15153             : 
   15154           2 :         printf("Starting send SMB1 negotiate+tcon.\n");
   15155           2 :         cli = open_nbt_connection();
   15156           2 :         if (cli == NULL) {
   15157           0 :                 d_fprintf(stderr, "open_nbt_connection failed!\n");
   15158           0 :                 return false;
   15159             :         }
   15160           2 :         smbXcli_conn_set_sockopt(cli->conn, sockops);
   15161             : 
   15162           2 :         status = smbXcli_negprot(cli->conn,
   15163             :                                  0,
   15164             :                                  PROTOCOL_NT1,
   15165             :                                  PROTOCOL_NT1,
   15166             :                                  NULL,
   15167             :                                  NULL,
   15168             :                                  NULL);
   15169           2 :         if (!NT_STATUS_IS_OK(status)) {
   15170           0 :                 d_fprintf(stderr, "smbXcli_negprot failed %s!\n",
   15171             :                         nt_errstr(status));
   15172           0 :                 return false;
   15173             :         }
   15174           2 :         status = cli_raw_tcon(cli,
   15175             :                               share,
   15176             :                               "",
   15177             :                               "?????",
   15178             :                               &max_xmit,
   15179             :                               &cnum);
   15180           2 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_ACCESS_DENIED)) {
   15181           0 :                 d_fprintf(stderr, "cli_raw_tcon failed - got %s "
   15182             :                         "(should get NT_STATUS_ACCESS_DENIED)!\n",
   15183             :                         nt_errstr(status));
   15184           0 :                 return false;
   15185             :         }
   15186           2 :         return true;
   15187             : }
   15188             : 
   15189           0 : static bool run_ign_bad_negprot(int dummy)
   15190             : {
   15191           0 :         struct tevent_context *ev;
   15192           0 :         struct tevent_req *req;
   15193           0 :         struct smbXcli_conn *conn;
   15194           0 :         struct sockaddr_storage ss;
   15195           0 :         NTSTATUS status;
   15196           0 :         int fd;
   15197           0 :         bool ok;
   15198             : 
   15199           0 :         printf("starting ignore bad negprot\n");
   15200             : 
   15201           0 :         ok = resolve_name(host, &ss, 0x20, true);
   15202           0 :         if (!ok) {
   15203           0 :                 d_fprintf(stderr, "Could not resolve name %s\n", host);
   15204           0 :                 return false;
   15205             :         }
   15206             : 
   15207           0 :         status = open_socket_out(&ss, 445, 10000, &fd);
   15208           0 :         if (!NT_STATUS_IS_OK(status)) {
   15209           0 :                 d_fprintf(stderr, "open_socket_out failed: %s\n",
   15210             :                           nt_errstr(status));
   15211           0 :                 return false;
   15212             :         }
   15213             : 
   15214           0 :         conn = smbXcli_conn_create(talloc_tos(), fd, host, SMB_SIGNING_OFF, 0,
   15215             :                                    NULL, 0, NULL);
   15216           0 :         if (conn == NULL) {
   15217           0 :                 d_fprintf(stderr, "smbXcli_conn_create failed\n");
   15218           0 :                 return false;
   15219             :         }
   15220             : 
   15221           0 :         status = smbXcli_negprot(conn,
   15222             :                                  0,
   15223             :                                  PROTOCOL_CORE,
   15224             :                                  PROTOCOL_CORE,
   15225             :                                  NULL,
   15226             :                                  NULL,
   15227             :                                  NULL);
   15228           0 :         if (NT_STATUS_IS_OK(status)) {
   15229           0 :                 d_fprintf(stderr, "smbXcli_negprot succeeded!\n");
   15230           0 :                 return false;
   15231             :         }
   15232             : 
   15233           0 :         ev = samba_tevent_context_init(talloc_tos());
   15234           0 :         if (ev == NULL) {
   15235           0 :                 d_fprintf(stderr, "samba_tevent_context_init failed\n");
   15236           0 :                 return false;
   15237             :         }
   15238             : 
   15239           0 :         req = smb1cli_session_setup_nt1_send(
   15240           0 :                 ev, ev, conn, 0, getpid(), NULL, 65503, 2, 1, 0, "", "",
   15241             :                 data_blob_null, data_blob_null, 0x40,
   15242             :                 "Windows 2000 2195", "Windows 2000 5.0");
   15243           0 :         if (req == NULL) {
   15244           0 :                 d_fprintf(stderr, "smb1cli_session_setup_nt1_send failed\n");
   15245           0 :                 return false;
   15246             :         }
   15247             : 
   15248           0 :         ok = tevent_req_poll_ntstatus(req, ev, &status);
   15249           0 :         if (!ok) {
   15250           0 :                 d_fprintf(stderr, "tevent_req_poll failed\n");
   15251           0 :                 return false;
   15252             :         }
   15253             : 
   15254           0 :         status = smb1cli_session_setup_nt1_recv(req, NULL, NULL, NULL, NULL,
   15255             :                                                 NULL, NULL);
   15256           0 :         if (!NT_STATUS_EQUAL(status, NT_STATUS_CONNECTION_RESET)) {
   15257           0 :                 d_fprintf(stderr, "smb1cli_session_setup_nt1_recv returned "
   15258             :                           "%s, expected NT_STATUS_CONNECTION_RESET\n",
   15259             :                           nt_errstr(status));
   15260           0 :                 return false;
   15261             :         }
   15262             : 
   15263           0 :         TALLOC_FREE(conn);
   15264             : 
   15265           0 :         printf("starting ignore bad negprot\n");
   15266             : 
   15267           0 :         return true;
   15268             : }
   15269             : 
   15270             : 
   15271          16 : static double create_procs(bool (*fn)(int), bool *result)
   15272             : {
   15273           0 :         int i, status;
   15274           0 :         volatile pid_t *child_status;
   15275           0 :         volatile bool *child_status_out;
   15276           0 :         int synccount;
   15277          16 :         int tries = 8;
   15278           0 :         struct timeval start;
   15279             : 
   15280          16 :         synccount = 0;
   15281             : 
   15282          16 :         child_status = (volatile pid_t *)anonymous_shared_allocate(sizeof(pid_t)*torture_nprocs);
   15283          16 :         if (!child_status) {
   15284           0 :                 printf("Failed to setup shared memory\n");
   15285           0 :                 return -1;
   15286             :         }
   15287             : 
   15288          16 :         child_status_out = (volatile bool *)anonymous_shared_allocate(sizeof(bool)*torture_nprocs);
   15289          16 :         if (!child_status_out) {
   15290           0 :                 printf("Failed to setup result status shared memory\n");
   15291           0 :                 return -1;
   15292             :         }
   15293             : 
   15294          32 :         for (i = 0; i < torture_nprocs; i++) {
   15295          16 :                 child_status[i] = 0;
   15296          16 :                 child_status_out[i] = True;
   15297             :         }
   15298             : 
   15299          16 :         start = timeval_current();
   15300             : 
   15301          32 :         for (i=0;i<torture_nprocs;i++) {
   15302          16 :                 procnum = i;
   15303          16 :                 if (fork() == 0) {
   15304           0 :                         pid_t mypid = getpid();
   15305           0 :                         sys_srandom(((int)mypid) ^ ((int)time(NULL)));
   15306             : 
   15307           0 :                         slprintf(myname,sizeof(myname),"CLIENT%d", i);
   15308             : 
   15309           0 :                         while (1) {
   15310           0 :                                 if (torture_open_connection(&current_cli, i)) break;
   15311           0 :                                 if (tries-- == 0) {
   15312           0 :                                         printf("pid %d failed to start\n", (int)getpid());
   15313           0 :                                         _exit(1);
   15314             :                                 }
   15315           0 :                                 smb_msleep(10); 
   15316             :                         }
   15317             : 
   15318           0 :                         child_status[i] = getpid();
   15319             : 
   15320           0 :                         while (child_status[i] && timeval_elapsed(&start) < 5) smb_msleep(2);
   15321             : 
   15322           0 :                         child_status_out[i] = fn(i);
   15323           0 :                         _exit(0);
   15324             :                 }
   15325             :         }
   15326             : 
   15327           0 :         do {
   15328          74 :                 synccount = 0;
   15329         148 :                 for (i=0;i<torture_nprocs;i++) {
   15330          74 :                         if (child_status[i]) synccount++;
   15331             :                 }
   15332          74 :                 if (synccount == torture_nprocs) break;
   15333          58 :                 smb_msleep(10);
   15334          58 :         } while (timeval_elapsed(&start) < 30);
   15335             : 
   15336          16 :         if (synccount != torture_nprocs) {
   15337           0 :                 printf("FAILED TO START %d CLIENTS (started %d)\n", torture_nprocs, synccount);
   15338           0 :                 *result = False;
   15339           0 :                 return timeval_elapsed(&start);
   15340             :         }
   15341             : 
   15342             :         /* start the client load */
   15343          16 :         start = timeval_current();
   15344             : 
   15345          32 :         for (i=0;i<torture_nprocs;i++) {
   15346          16 :                 child_status[i] = 0;
   15347             :         }
   15348             : 
   15349          16 :         printf("%d clients started\n", torture_nprocs);
   15350             : 
   15351          32 :         for (i=0;i<torture_nprocs;i++) {
   15352          16 :                 while (waitpid(0, &status, 0) == -1 && errno == EINTR) /* noop */ ;
   15353             :         }
   15354             : 
   15355          16 :         printf("\n");
   15356             : 
   15357          32 :         for (i=0;i<torture_nprocs;i++) {
   15358          16 :                 if (!child_status_out[i]) {
   15359           2 :                         *result = False;
   15360             :                 }
   15361             :         }
   15362          16 :         return timeval_elapsed(&start);
   15363             : }
   15364             : 
   15365             : #define FLAG_MULTIPROC 1
   15366             : 
   15367             : static struct {
   15368             :         const char *name;
   15369             :         bool (*fn)(int);
   15370             :         unsigned flags;
   15371             : } torture_ops[] = {
   15372             :         {
   15373             :                 .name = "FDPASS",
   15374             :                 .fn   = run_fdpasstest,
   15375             :         },
   15376             :         {
   15377             :                 .name = "LOCK1",
   15378             :                 .fn   = run_locktest1,
   15379             :         },
   15380             :         {
   15381             :                 .name = "LOCK2",
   15382             :                 .fn   =  run_locktest2,
   15383             :         },
   15384             :         {
   15385             :                 .name = "LOCK3",
   15386             :                 .fn   =  run_locktest3,
   15387             :         },
   15388             :         {
   15389             :                 .name = "LOCK4",
   15390             :                 .fn   =  run_locktest4,
   15391             :         },
   15392             :         {
   15393             :                 .name = "LOCK5",
   15394             :                 .fn   =  run_locktest5,
   15395             :         },
   15396             :         {
   15397             :                 .name = "LOCK6",
   15398             :                 .fn   =  run_locktest6,
   15399             :         },
   15400             :         {
   15401             :                 .name = "LOCK7",
   15402             :                 .fn   =  run_locktest7,
   15403             :         },
   15404             :         {
   15405             :                 .name = "LOCK8",
   15406             :                 .fn   =  run_locktest8,
   15407             :         },
   15408             :         {
   15409             :                 .name = "LOCK9A",
   15410             :                 .fn   =  run_locktest9a,
   15411             :         },
   15412             :         {
   15413             :                 .name = "LOCK9B",
   15414             :                 .fn   =  run_locktest9b,
   15415             :         },
   15416             :         {
   15417             :                 .name = "LOCK10",
   15418             :                 .fn   =  run_locktest10,
   15419             :         },
   15420             :         {
   15421             :                 .name = "LOCK11",
   15422             :                 .fn   =  run_locktest11,
   15423             :         },
   15424             :         {
   15425             :                 .name = "LOCK12",
   15426             :                 .fn   =  run_locktest12,
   15427             :         },
   15428             :         {
   15429             :                 .name = "LOCK13",
   15430             :                 .fn   =  run_locktest13,
   15431             :         },
   15432             :         {
   15433             :                 .name = "UNLINK",
   15434             :                 .fn   = run_unlinktest,
   15435             :         },
   15436             :         {
   15437             :                 .name = "BROWSE",
   15438             :                 .fn   = run_browsetest,
   15439             :         },
   15440             :         {
   15441             :                 .name = "ATTR",
   15442             :                 .fn   =   run_attrtest,
   15443             :         },
   15444             :         {
   15445             :                 .name = "TRANS2",
   15446             :                 .fn   = run_trans2test,
   15447             :         },
   15448             :         {
   15449             :                 .name  = "MAXFID",
   15450             :                 .fn    = run_maxfidtest,
   15451             :                 .flags = FLAG_MULTIPROC,
   15452             :         },
   15453             :         {
   15454             :                 .name  = "TORTURE",
   15455             :                 .fn    = run_torture,
   15456             :                 .flags = FLAG_MULTIPROC,
   15457             :         },
   15458             :         {
   15459             :                 .name  = "RANDOMIPC",
   15460             :                 .fn    = run_randomipc,
   15461             :         },
   15462             :         {
   15463             :                 .name  = "NEGNOWAIT",
   15464             :                 .fn    = run_negprot_nowait,
   15465             :         },
   15466             :         {
   15467             :                 .name  = "NBENCH",
   15468             :                 .fn    =  run_nbench,
   15469             :         },
   15470             :         {
   15471             :                 .name  = "NBENCH2",
   15472             :                 .fn    = run_nbench2,
   15473             :         },
   15474             :         {
   15475             :                 .name  = "OPLOCK1",
   15476             :                 .fn    =  run_oplock1,
   15477             :         },
   15478             :         {
   15479             :                 .name  = "OPLOCK2",
   15480             :                 .fn    =  run_oplock2,
   15481             :         },
   15482             :         {
   15483             :                 .name  = "OPLOCK4",
   15484             :                 .fn    =  run_oplock4,
   15485             :         },
   15486             : #ifdef HAVE_KERNEL_OPLOCKS_LINUX
   15487             :         {
   15488             :                 .name  = "OPLOCK5",
   15489             :                 .fn    =  run_oplock5,
   15490             :         },
   15491             : #endif
   15492             :         {
   15493             :                 .name  = "DIR",
   15494             :                 .fn    =  run_dirtest,
   15495             :         },
   15496             :         {
   15497             :                 .name  = "DIR1",
   15498             :                 .fn    =  run_dirtest1,
   15499             :         },
   15500             :         {
   15501             :                 .name  = "DIR-CREATETIME",
   15502             :                 .fn    =  run_dir_createtime,
   15503             :         },
   15504             :         {
   15505             :                 .name  = "DENY1",
   15506             :                 .fn    =  torture_denytest1,
   15507             :         },
   15508             :         {
   15509             :                 .name  = "DENY2",
   15510             :                 .fn    =  torture_denytest2,
   15511             :         },
   15512             :         {
   15513             :                 .name  = "TCON",
   15514             :                 .fn    =  run_tcon_test,
   15515             :         },
   15516             :         {
   15517             :                 .name  = "TCONDEV",
   15518             :                 .fn    =  run_tcon_devtype_test,
   15519             :         },
   15520             :         {
   15521             :                 .name  = "RW1",
   15522             :                 .fn    =  run_readwritetest,
   15523             :         },
   15524             :         {
   15525             :                 .name  = "RW2",
   15526             :                 .fn    =  run_readwritemulti,
   15527             :                 .flags = FLAG_MULTIPROC
   15528             :         },
   15529             :         {
   15530             :                 .name  = "RW3",
   15531             :                 .fn    =  run_readwritelarge,
   15532             :         },
   15533             :         {
   15534             :                 .name  = "RW-SIGNING",
   15535             :                 .fn    =  run_readwritelarge_signtest,
   15536             :         },
   15537             :         {
   15538             :                 .name  = "OPEN",
   15539             :                 .fn    = run_opentest,
   15540             :         },
   15541             :         {
   15542             :                 .name  = "POSIX",
   15543             :                 .fn    = run_simple_posix_open_test,
   15544             :         },
   15545             :         {
   15546             :                 .name  = "POSIX-APPEND",
   15547             :                 .fn    = run_posix_append,
   15548             :         },
   15549             :         {
   15550             :                 .name  = "POSIX-SYMLINK-ACL",
   15551             :                 .fn    = run_acl_symlink_test,
   15552             :         },
   15553             :         {
   15554             :                 .name  = "POSIX-SYMLINK-EA",
   15555             :                 .fn    = run_ea_symlink_test,
   15556             :         },
   15557             :         {
   15558             :                 .name  = "POSIX-STREAM-DELETE",
   15559             :                 .fn    = run_posix_stream_delete,
   15560             :         },
   15561             :         {
   15562             :                 .name  = "POSIX-OFD-LOCK",
   15563             :                 .fn    = run_posix_ofd_lock_test,
   15564             :         },
   15565             :         {
   15566             :                 .name  = "POSIX-BLOCKING-LOCK",
   15567             :                 .fn    = run_posix_blocking_lock,
   15568             :         },
   15569             :         {
   15570             :                 .name  = "POSIX-MKDIR",
   15571             :                 .fn    = run_posix_mkdir_test,
   15572             :         },
   15573             :         {
   15574             :                 .name  = "POSIX-ACL-OPLOCK",
   15575             :                 .fn    = run_posix_acl_oplock_test,
   15576             :         },
   15577             :         {
   15578             :                 .name  = "POSIX-ACL-SHAREROOT",
   15579             :                 .fn    = run_posix_acl_shareroot_test,
   15580             :         },
   15581             :         {
   15582             :                 .name  = "POSIX-LS-WILDCARD",
   15583             :                 .fn    = run_posix_ls_wildcard_test,
   15584             :         },
   15585             :         {
   15586             :                 .name  = "POSIX-LS-SINGLE",
   15587             :                 .fn    = run_posix_ls_single_test,
   15588             :         },
   15589             :         {
   15590             :                 .name  = "POSIX-READLINK",
   15591             :                 .fn    = run_posix_readlink_test,
   15592             :         },
   15593             :         {
   15594             :                 .name  = "POSIX-STAT",
   15595             :                 .fn    = run_posix_stat_test,
   15596             :         },
   15597             :         {
   15598             :                 .name  = "POSIX-SYMLINK-PARENT",
   15599             :                 .fn    = run_posix_symlink_parent_test,
   15600             :         },
   15601             :         {
   15602             :                 .name  = "POSIX-SYMLINK-CHMOD",
   15603             :                 .fn    = run_posix_symlink_chmod_test,
   15604             :         },
   15605             :         {
   15606             :                 .name  = "POSIX-SYMLINK-RENAME",
   15607             :                 .fn    = run_posix_symlink_rename_test,
   15608             :         },
   15609             :         {
   15610             :                 .name  = "POSIX-DIR-DEFAULT-ACL",
   15611             :                 .fn    = run_posix_dir_default_acl_test,
   15612             :         },
   15613             :         {
   15614             :                 .name  = "POSIX-SYMLINK-GETPATHINFO",
   15615             :                 .fn    = run_posix_symlink_getpathinfo_test,
   15616             :         },
   15617             :         {
   15618             :                 .name  = "POSIX-SYMLINK-SETPATHINFO",
   15619             :                 .fn    = run_posix_symlink_setpathinfo_test,
   15620             :         },
   15621             :         {
   15622             :                 .name  = "WINDOWS-BAD-SYMLINK",
   15623             :                 .fn    = run_symlink_open_test,
   15624             :         },
   15625             :         {
   15626             :                 .name  = "SMB1-WILD-MANGLE-UNLINK",
   15627             :                 .fn    = run_smb1_wild_mangle_unlink_test,
   15628             :         },
   15629             :         {
   15630             :                 .name  = "SMB1-WILD-MANGLE-RENAME",
   15631             :                 .fn    = run_smb1_wild_mangle_rename_test,
   15632             :         },
   15633             :         {
   15634             :                 .name  = "CASE-INSENSITIVE-CREATE",
   15635             :                 .fn    = run_case_insensitive_create,
   15636             :         },
   15637             :         {
   15638             :                 .name  = "ASYNC-ECHO",
   15639             :                 .fn    = run_async_echo,
   15640             :         },
   15641             :         {
   15642             :                 .name  = "UID-REGRESSION-TEST",
   15643             :                 .fn    = run_uid_regression_test,
   15644             :         },
   15645             :         {
   15646             :                 .name  = "SHORTNAME-TEST",
   15647             :                 .fn    = run_shortname_test,
   15648             :         },
   15649             :         {
   15650             :                 .name  = "ADDRCHANGE",
   15651             :                 .fn    = run_addrchange,
   15652             :         },
   15653             : #if 1
   15654             :         {
   15655             :                 .name  = "OPENATTR",
   15656             :                 .fn    = run_openattrtest,
   15657             :         },
   15658             : #endif
   15659             :         {
   15660             :                 .name  = "XCOPY",
   15661             :                 .fn    = run_xcopy,
   15662             :         },
   15663             :         {
   15664             :                 .name  = "RENAME",
   15665             :                 .fn    = run_rename,
   15666             :         },
   15667             :         {
   15668             :                 .name  = "RENAME-ACCESS",
   15669             :                 .fn    = run_rename_access,
   15670             :         },
   15671             :         {
   15672             :                 .name  = "OWNER-RIGHTS",
   15673             :                 .fn    = run_owner_rights,
   15674             :         },
   15675             :         {
   15676             :                 .name  = "DELETE",
   15677             :                 .fn    = run_deletetest,
   15678             :         },
   15679             :         {
   15680             :                 .name  = "DELETE-STREAM",
   15681             :                 .fn    = run_delete_stream,
   15682             :         },
   15683             :         {
   15684             :                 .name  = "DELETE-PRINT",
   15685             :                 .fn    = run_delete_print_test,
   15686             :         },
   15687             :         {
   15688             :                 .name  = "DELETE-LN",
   15689             :                 .fn    = run_deletetest_ln,
   15690             :         },
   15691             :         {
   15692             :                 .name  = "PROPERTIES",
   15693             :                 .fn    = run_properties,
   15694             :         },
   15695             :         {
   15696             :                 .name  = "MANGLE",
   15697             :                 .fn    = torture_mangle,
   15698             :         },
   15699             :         {
   15700             :                 .name  = "MANGLE1",
   15701             :                 .fn    = run_mangle1,
   15702             :         },
   15703             :         {
   15704             :                 .name  = "MANGLE-ILLEGAL",
   15705             :                 .fn    = run_mangle_illegal,
   15706             :         },
   15707             :         {
   15708             :                 .name  = "W2K",
   15709             :                 .fn    = run_w2ktest,
   15710             :         },
   15711             :         {
   15712             :                 .name  = "TRANS2SCAN",
   15713             :                 .fn    = torture_trans2_scan,
   15714             :         },
   15715             :         {
   15716             :                 .name  = "NTTRANSSCAN",
   15717             :                 .fn    = torture_nttrans_scan,
   15718             :         },
   15719             :         {
   15720             :                 .name  = "UTABLE",
   15721             :                 .fn    = torture_utable,
   15722             :         },
   15723             :         {
   15724             :                 .name  = "CASETABLE",
   15725             :                 .fn    = torture_casetable,
   15726             :         },
   15727             :         {
   15728             :                 .name  = "ERRMAPEXTRACT",
   15729             :                 .fn    = run_error_map_extract,
   15730             :         },
   15731             :         {
   15732             :                 .name  = "PIPE_NUMBER",
   15733             :                 .fn    = run_pipe_number,
   15734             :         },
   15735             :         {
   15736             :                 .name  = "TCON2",
   15737             :                 .fn    =  run_tcon2_test,
   15738             :         },
   15739             :         {
   15740             :                 .name  = "IOCTL",
   15741             :                 .fn    =  torture_ioctl_test,
   15742             :         },
   15743             :         {
   15744             :                 .name  = "CHKPATH",
   15745             :                 .fn    =  torture_chkpath_test,
   15746             :         },
   15747             :         {
   15748             :                 .name  = "FDSESS",
   15749             :                 .fn    = run_fdsesstest,
   15750             :         },
   15751             :         {
   15752             :                 .name  = "EATEST",
   15753             :                 .fn    = run_eatest,
   15754             :         },
   15755             :         {
   15756             :                 .name  = "SESSSETUP_BENCH",
   15757             :                 .fn    = run_sesssetup_bench,
   15758             :         },
   15759             :         {
   15760             :                 .name  = "CHAIN1",
   15761             :                 .fn    = run_chain1,
   15762             :         },
   15763             :         {
   15764             :                 .name  = "CHAIN2",
   15765             :                 .fn    = run_chain2,
   15766             :         },
   15767             :         {
   15768             :                 .name  = "CHAIN3",
   15769             :                 .fn    = run_chain3,
   15770             :         },
   15771             :         {
   15772             :                 .name  = "WINDOWS-WRITE",
   15773             :                 .fn    = run_windows_write,
   15774             :         },
   15775             :         {
   15776             :                 .name  = "LARGE_READX",
   15777             :                 .fn    = run_large_readx,
   15778             :         },
   15779             :         {
   15780             :                 .name  = "MSDFS-ATTRIBUTE",
   15781             :                 .fn    = run_msdfs_attribute,
   15782             :         },
   15783             :         {
   15784             :                 .name  = "NTTRANS-CREATE",
   15785             :                 .fn    = run_nttrans_create,
   15786             :         },
   15787             :         {
   15788             :                 .name  = "NTTRANS-FSCTL",
   15789             :                 .fn    = run_nttrans_fsctl,
   15790             :         },
   15791             :         {
   15792             :                 .name  = "CLI_ECHO",
   15793             :                 .fn    = run_cli_echo,
   15794             :         },
   15795             :         {
   15796             :                 .name  = "CLI_SPLICE",
   15797             :                 .fn    = run_cli_splice,
   15798             :         },
   15799             :         {
   15800             :                 .name  = "TLDAP",
   15801             :                 .fn    = run_tldap,
   15802             :         },
   15803             :         {
   15804             :                 .name  = "STREAMERROR",
   15805             :                 .fn    = run_streamerror,
   15806             :         },
   15807             :         {
   15808             :                 .name  = "NOTIFY-BENCH",
   15809             :                 .fn    = run_notify_bench,
   15810             :         },
   15811             :         {
   15812             :                 .name  = "NOTIFY-BENCH2",
   15813             :                 .fn    = run_notify_bench2,
   15814             :         },
   15815             :         {
   15816             :                 .name  = "NOTIFY-BENCH3",
   15817             :                 .fn    = run_notify_bench3,
   15818             :         },
   15819             :         {
   15820             :                 .name  = "BAD-NBT-SESSION",
   15821             :                 .fn    = run_bad_nbt_session,
   15822             :         },
   15823             :         {
   15824             :                 .name  = "IGN-BAD-NEGPROT",
   15825             :                 .fn    = run_ign_bad_negprot,
   15826             :         },
   15827             :         {
   15828             :                 .name  = "SMB-ANY-CONNECT",
   15829             :                 .fn    = run_smb_any_connect,
   15830             :         },
   15831             :         {
   15832             :                 .name  = "NOTIFY-ONLINE",
   15833             :                 .fn    = run_notify_online,
   15834             :         },
   15835             :         {
   15836             :                 .name  = "SMB2-BASIC",
   15837             :                 .fn    = run_smb2_basic,
   15838             :         },
   15839             :         {
   15840             :                 .name  = "SMB2-NEGPROT",
   15841             :                 .fn    = run_smb2_negprot,
   15842             :         },
   15843             :         {
   15844             :                 .name  = "SMB2-ANONYMOUS",
   15845             :                 .fn    = run_smb2_anonymous,
   15846             :         },
   15847             :         {
   15848             :                 .name  = "SMB2-SESSION-RECONNECT",
   15849             :                 .fn    = run_smb2_session_reconnect,
   15850             :         },
   15851             :         {
   15852             :                 .name  = "SMB2-TCON-DEPENDENCE",
   15853             :                 .fn    = run_smb2_tcon_dependence,
   15854             :         },
   15855             :         {
   15856             :                 .name  = "SMB2-MULTI-CHANNEL",
   15857             :                 .fn    = run_smb2_multi_channel,
   15858             :         },
   15859             :         {
   15860             :                 .name  = "SMB2-SESSION-REAUTH",
   15861             :                 .fn    = run_smb2_session_reauth,
   15862             :         },
   15863             :         {
   15864             :                 .name  = "SMB2-FTRUNCATE",
   15865             :                 .fn    = run_smb2_ftruncate,
   15866             :         },
   15867             :         {
   15868             :                 .name  = "SMB2-DIR-FSYNC",
   15869             :                 .fn    = run_smb2_dir_fsync,
   15870             :         },
   15871             :         {
   15872             :                 .name  = "SMB2-PATH-SLASH",
   15873             :                 .fn    = run_smb2_path_slash,
   15874             :         },
   15875             :         {
   15876             :                 .name  = "SMB1-SYSTEM-SECURITY",
   15877             :                 .fn    = run_smb1_system_security,
   15878             :         },
   15879             :         {
   15880             :                 .name  = "SMB2-SACL",
   15881             :                 .fn    = run_smb2_sacl,
   15882             :         },
   15883             :         {
   15884             :                 .name  = "SMB2-QUOTA1",
   15885             :                 .fn    = run_smb2_quota1,
   15886             :         },
   15887             :         {
   15888             :                 .name  = "SMB2-INVALID-PIPENAME",
   15889             :                 .fn    = run_smb2_invalid_pipename,
   15890             :         },
   15891             :         {
   15892             :                 .name  = "SMB2-STREAM-ACL",
   15893             :                 .fn    = run_smb2_stream_acl,
   15894             :         },
   15895             :         {
   15896             :                 .name  = "SMB2-LIST-DIR-ASYNC",
   15897             :                 .fn    = run_list_dir_async_test,
   15898             :         },
   15899             :         {
   15900             :                 .name  = "SMB2-DEL-ON-CLOSE-NONEMPTY",
   15901             :                 .fn    = run_delete_on_close_non_empty,
   15902             :         },
   15903             :         {
   15904             :                 .name  = "SMB2-DEL-ON-CLOSE-NONWRITE-DELETE-YES",
   15905             :                 .fn    = run_delete_on_close_nonwrite_delete_yes_test,
   15906             :         },
   15907             :         {
   15908             :                 .name  = "SMB2-DEL-ON-CLOSE-NONWRITE-DELETE-NO",
   15909             :                 .fn    = run_delete_on_close_nonwrite_delete_no_test,
   15910             :         },
   15911             :         {
   15912             :                 .name  = "SMB2-DFS-PATHS",
   15913             :                 .fn    = run_smb2_dfs_paths,
   15914             :         },
   15915             :         {
   15916             :                 .name  = "SMB2-NON-DFS-SHARE",
   15917             :                 .fn    = run_smb2_non_dfs_share,
   15918             :         },
   15919             :         {
   15920             :                 .name  = "SMB2-DFS-SHARE-NON-DFS-PATH",
   15921             :                 .fn    = run_smb2_dfs_share_non_dfs_path,
   15922             :         },
   15923             :         {
   15924             :                 .name  = "SMB2-DFS-FILENAME-LEADING-BACKSLASH",
   15925             :                 .fn    = run_smb2_dfs_filename_leading_backslash,
   15926             :         },
   15927             :         {
   15928             :                 .name  = "SMB2-PIPE-READ-ASYNC-DISCONNECT",
   15929             :                 .fn    = run_smb2_pipe_read_async_disconnect,
   15930             :         },
   15931             :         {
   15932             :                 .name  = "SMB1-TRUNCATED-SESSSETUP",
   15933             :                 .fn    = run_smb1_truncated_sesssetup,
   15934             :         },
   15935             :         {
   15936             :                 .name  = "SMB1-NEGOTIATE-EXIT",
   15937             :                 .fn    = run_smb1_negotiate_exit,
   15938             :         },
   15939             :         {
   15940             :                 .name  = "SMB1-NEGOTIATE-TCON",
   15941             :                 .fn    = run_smb1_negotiate_tcon,
   15942             :         },
   15943             :         {
   15944             :                 .name  = "SMB1-DFS-PATHS",
   15945             :                 .fn    = run_smb1_dfs_paths,
   15946             :         },
   15947             :         {
   15948             :                 .name  = "SMB1-DFS-SEARCH-PATHS",
   15949             :                 .fn    = run_smb1_dfs_search_paths,
   15950             :         },
   15951             :         {
   15952             :                 .name  = "SMB1-DFS-OPERATIONS",
   15953             :                 .fn    = run_smb1_dfs_operations,
   15954             :         },
   15955             :         {
   15956             :                 .name  = "SMB1-DFS-BADPATH",
   15957             :                 .fn    = run_smb1_dfs_check_badpath,
   15958             :         },
   15959             :         {
   15960             :                 .name  = "CLEANUP1",
   15961             :                 .fn    = run_cleanup1,
   15962             :         },
   15963             :         {
   15964             :                 .name  = "CLEANUP2",
   15965             :                 .fn    = run_cleanup2,
   15966             :         },
   15967             :         {
   15968             :                 .name  = "CLEANUP4",
   15969             :                 .fn    = run_cleanup4,
   15970             :         },
   15971             :         {
   15972             :                 .name  = "OPLOCK-CANCEL",
   15973             :                 .fn    = run_oplock_cancel,
   15974             :         },
   15975             :         {
   15976             :                 .name  = "PIDHIGH",
   15977             :                 .fn    = run_pidhigh,
   15978             :         },
   15979             :         {
   15980             :                 .name  = "LOCAL-SUBSTITUTE",
   15981             :                 .fn    = run_local_substitute,
   15982             :         },
   15983             :         {
   15984             :                 .name  = "LOCAL-GENCACHE",
   15985             :                 .fn    = run_local_gencache,
   15986             :         },
   15987             :         {
   15988             :                 .name  = "LOCAL-DBWRAP-WATCH1",
   15989             :                 .fn    = run_dbwrap_watch1,
   15990             :         },
   15991             :         {
   15992             :                 .name  = "LOCAL-DBWRAP-WATCH2",
   15993             :                 .fn    = run_dbwrap_watch2,
   15994             :         },
   15995             :         {
   15996             :                 .name  = "LOCAL-DBWRAP-WATCH3",
   15997             :                 .fn    = run_dbwrap_watch3,
   15998             :         },
   15999             :         {
   16000             :                 .name  = "LOCAL-DBWRAP-WATCH4",
   16001             :                 .fn    = run_dbwrap_watch4,
   16002             :         },
   16003             :         {
   16004             :                 .name  = "LOCAL-DBWRAP-DO-LOCKED1",
   16005             :                 .fn    = run_dbwrap_do_locked1,
   16006             :         },
   16007             :         {
   16008             :                 .name  = "LOCAL-MESSAGING-READ1",
   16009             :                 .fn    = run_messaging_read1,
   16010             :         },
   16011             :         {
   16012             :                 .name  = "LOCAL-MESSAGING-READ2",
   16013             :                 .fn    = run_messaging_read2,
   16014             :         },
   16015             :         {
   16016             :                 .name  = "LOCAL-MESSAGING-READ3",
   16017             :                 .fn    = run_messaging_read3,
   16018             :         },
   16019             :         {
   16020             :                 .name  = "LOCAL-MESSAGING-READ4",
   16021             :                 .fn    = run_messaging_read4,
   16022             :         },
   16023             :         {
   16024             :                 .name  = "LOCAL-MESSAGING-FDPASS1",
   16025             :                 .fn    = run_messaging_fdpass1,
   16026             :         },
   16027             :         {
   16028             :                 .name  = "LOCAL-MESSAGING-FDPASS2",
   16029             :                 .fn    = run_messaging_fdpass2,
   16030             :         },
   16031             :         {
   16032             :                 .name  = "LOCAL-MESSAGING-FDPASS2a",
   16033             :                 .fn    = run_messaging_fdpass2a,
   16034             :         },
   16035             :         {
   16036             :                 .name  = "LOCAL-MESSAGING-FDPASS2b",
   16037             :                 .fn    = run_messaging_fdpass2b,
   16038             :         },
   16039             :         {
   16040             :                 .name  = "LOCAL-MESSAGING-SEND-ALL",
   16041             :                 .fn    = run_messaging_send_all,
   16042             :         },
   16043             :         {
   16044             :                 .name  = "LOCAL-BASE64",
   16045             :                 .fn    = run_local_base64,
   16046             :         },
   16047             :         {
   16048             :                 .name  = "LOCAL-RBTREE",
   16049             :                 .fn    = run_local_rbtree,
   16050             :         },
   16051             :         {
   16052             :                 .name  = "LOCAL-MEMCACHE",
   16053             :                 .fn    = run_local_memcache,
   16054             :         },
   16055             :         {
   16056             :                 .name  = "LOCAL-STREAM-NAME",
   16057             :                 .fn    = run_local_stream_name,
   16058             :         },
   16059             :         {
   16060             :                 .name  = "LOCAL-STR-MATCH-MSWILD",
   16061             :                 .fn    = run_str_match_mswild,
   16062             :         },
   16063             :         {
   16064             :                 .name  = "LOCAL-STR-MATCH-REGEX-SUB1",
   16065             :                 .fn    = run_str_match_regex_sub1,
   16066             :         },
   16067             :         {
   16068             :                 .name  = "WBCLIENT-MULTI-PING",
   16069             :                 .fn    = run_wbclient_multi_ping,
   16070             :         },
   16071             :         {
   16072             :                 .name  = "LOCAL-string_to_sid",
   16073             :                 .fn    = run_local_string_to_sid,
   16074             :         },
   16075             :         {
   16076             :                 .name  = "LOCAL-sid_to_string",
   16077             :                 .fn    = run_local_sid_to_string,
   16078             :         },
   16079             :         {
   16080             :                 .name  = "LOCAL-binary_to_sid",
   16081             :                 .fn    = run_local_binary_to_sid,
   16082             :         },
   16083             :         {
   16084             :                 .name  = "LOCAL-DBTRANS",
   16085             :                 .fn    = run_local_dbtrans,
   16086             :         },
   16087             :         {
   16088             :                 .name  = "LOCAL-TEVENT-POLL",
   16089             :                 .fn    = run_local_tevent_poll,
   16090             :         },
   16091             :         {
   16092             :                 .name  = "LOCAL-CONVERT-STRING",
   16093             :                 .fn    = run_local_convert_string,
   16094             :         },
   16095             :         {
   16096             :                 .name  = "LOCAL-CONV-AUTH-INFO",
   16097             :                 .fn    = run_local_conv_auth_info,
   16098             :         },
   16099             :         {
   16100             :                 .name  = "LOCAL-hex_encode_buf",
   16101             :                 .fn    = run_local_hex_encode_buf,
   16102             :         },
   16103             :         {
   16104             :                 .name  = "LOCAL-IDMAP-TDB-COMMON",
   16105             :                 .fn    = run_idmap_tdb_common_test,
   16106             :         },
   16107             :         {
   16108             :                 .name  = "LOCAL-remove_duplicate_addrs2",
   16109             :                 .fn    = run_local_remove_duplicate_addrs2,
   16110             :         },
   16111             :         {
   16112             :                 .name  = "local-tdb-opener",
   16113             :                 .fn    = run_local_tdb_opener,
   16114             :         },
   16115             :         {
   16116             :                 .name  = "local-tdb-writer",
   16117             :                 .fn    = run_local_tdb_writer,
   16118             :         },
   16119             :         {
   16120             :                 .name  = "LOCAL-DBWRAP-CTDB1",
   16121             :                 .fn    = run_local_dbwrap_ctdb1,
   16122             :         },
   16123             :         {
   16124             :                 .name  = "LOCAL-BENCH-PTHREADPOOL",
   16125             :                 .fn    = run_bench_pthreadpool,
   16126             :         },
   16127             :         {
   16128             :                 .name  = "LOCAL-PTHREADPOOL-TEVENT",
   16129             :                 .fn    = run_pthreadpool_tevent,
   16130             :         },
   16131             :         {
   16132             :                 .name  = "LOCAL-G-LOCK1",
   16133             :                 .fn    = run_g_lock1,
   16134             :         },
   16135             :         {
   16136             :                 .name  = "LOCAL-G-LOCK2",
   16137             :                 .fn    = run_g_lock2,
   16138             :         },
   16139             :         {
   16140             :                 .name  = "LOCAL-G-LOCK3",
   16141             :                 .fn    = run_g_lock3,
   16142             :         },
   16143             :         {
   16144             :                 .name  = "LOCAL-G-LOCK4",
   16145             :                 .fn    = run_g_lock4,
   16146             :         },
   16147             :         {
   16148             :                 .name  = "LOCAL-G-LOCK4A",
   16149             :                 .fn    = run_g_lock4a,
   16150             :         },
   16151             :         {
   16152             :                 .name  = "LOCAL-G-LOCK5",
   16153             :                 .fn    = run_g_lock5,
   16154             :         },
   16155             :         {
   16156             :                 .name  = "LOCAL-G-LOCK6",
   16157             :                 .fn    = run_g_lock6,
   16158             :         },
   16159             :         {
   16160             :                 .name  = "LOCAL-G-LOCK7",
   16161             :                 .fn    = run_g_lock7,
   16162             :         },
   16163             :         {
   16164             :                 .name  = "LOCAL-G-LOCK8",
   16165             :                 .fn    = run_g_lock8,
   16166             :         },
   16167             :         {
   16168             :                 .name  = "LOCAL-G-LOCK-PING-PONG",
   16169             :                 .fn    = run_g_lock_ping_pong,
   16170             :         },
   16171             :         {
   16172             :                 .name  = "LOCAL-CANONICALIZE-PATH",
   16173             :                 .fn    = run_local_canonicalize_path,
   16174             :         },
   16175             :         {
   16176             :                 .name  = "LOCAL-NAMEMAP-CACHE1",
   16177             :                 .fn    = run_local_namemap_cache1,
   16178             :         },
   16179             :         {
   16180             :                 .name  = "LOCAL-IDMAP-CACHE1",
   16181             :                 .fn    = run_local_idmap_cache1,
   16182             :         },
   16183             :         {
   16184             :                 .name  = "qpathinfo-bufsize",
   16185             :                 .fn    = run_qpathinfo_bufsize,
   16186             :         },
   16187             :         {
   16188             :                 .name  = "hide-new-files-timeout",
   16189             :                 .fn    = run_hidenewfiles,
   16190             :         },
   16191             :         {
   16192             :                 .name  = "hide-new-files-timeout-showdirs",
   16193             :                 .fn    = run_hidenewfiles_showdirs,
   16194             :         },
   16195             : #ifdef CLUSTER_SUPPORT
   16196             :         {
   16197             :                 .name  = "ctdbd-conn1",
   16198             :                 .fn    = run_ctdbd_conn1,
   16199             :         },
   16200             : #endif
   16201             :         {
   16202             :                 .name  = "readdir-timestamp",
   16203             :                 .fn    = run_readdir_timestamp,
   16204             :         },
   16205             :         {
   16206             :                 .name  = "rpc-scale",
   16207             :                 .fn    = run_rpc_scale,
   16208             :         },
   16209             :         {
   16210             :                 .name  = "LOCAL-TDB-VALIDATE",
   16211             :                 .fn    = run_tdb_validate,
   16212             :         },
   16213             :         {
   16214             :                 .name = NULL,
   16215             :         },
   16216             : };
   16217             : 
   16218             : /****************************************************************************
   16219             : run a specified test or "ALL"
   16220             : ****************************************************************************/
   16221         546 : static bool run_test(const char *name)
   16222             : {
   16223         546 :         bool ret = True;
   16224         546 :         bool result = True;
   16225         546 :         bool found = False;
   16226          45 :         int i;
   16227          45 :         double t;
   16228         546 :         if (strequal(name,"ALL")) {
   16229           0 :                 for (i=0;torture_ops[i].name;i++) {
   16230           0 :                         run_test(torture_ops[i].name);
   16231             :                 }
   16232           0 :                 found = True;
   16233             :         }
   16234             : 
   16235      113747 :         for (i=0;torture_ops[i].name;i++) {
   16236      113197 :                 fstr_sprintf(randomfname, "\\XX%x", 
   16237      113197 :                          (unsigned)random());
   16238             : 
   16239      113197 :                 if (strequal(name, torture_ops[i].name)) {
   16240         546 :                         found = True;
   16241         546 :                         printf("Running %s\n", name);
   16242         546 :                         if (torture_ops[i].flags & FLAG_MULTIPROC) {
   16243          16 :                                 t = create_procs(torture_ops[i].fn, &result);
   16244          16 :                                 if (!result) { 
   16245           2 :                                         ret = False;
   16246           2 :                                         printf("TEST %s FAILED!\n", name);
   16247             :                                 }
   16248             :                         } else {
   16249          45 :                                 struct timeval start;
   16250         530 :                                 start = timeval_current();
   16251         530 :                                 if (!torture_ops[i].fn(0)) {
   16252          14 :                                         ret = False;
   16253          14 :                                         printf("TEST %s FAILED!\n", name);
   16254             :                                 }
   16255         534 :                                 t = timeval_elapsed(&start);
   16256             :                         }
   16257        9995 :                         printf("%s took %g secs\n\n", name, t);
   16258             :                 }
   16259             :         }
   16260             : 
   16261         550 :         if (!found) {
   16262           0 :                 printf("Did not find a test named %s\n", name);
   16263           0 :                 ret = False;
   16264             :         }
   16265             : 
   16266         550 :         return ret;
   16267             : }
   16268             : 
   16269             : 
   16270           0 : static void usage(void)
   16271             : {
   16272           0 :         int i;
   16273             : 
   16274           0 :         printf("WARNING samba4 test suite is much more complete nowadays.\n");
   16275           0 :         printf("Please use samba4 torture.\n\n");
   16276             : 
   16277           0 :         printf("Usage: smbtorture //server/share <options> TEST1 TEST2 ...\n");
   16278             : 
   16279           0 :         printf("\t-d debuglevel\n");
   16280           0 :         printf("\t-U user%%pass\n");
   16281           0 :         printf("\t-k                    use kerberos\n");
   16282           0 :         printf("\t-N numprocs\n");
   16283           0 :         printf("\t-n my_netbios_name\n");
   16284           0 :         printf("\t-W workgroup\n");
   16285           0 :         printf("\t-o num_operations\n");
   16286           0 :         printf("\t-O socket_options\n");
   16287           0 :         printf("\t-m maximum protocol\n");
   16288           0 :         printf("\t-L use oplocks\n");
   16289           0 :         printf("\t-c CLIENT.TXT         specify client load file for NBENCH\n");
   16290           0 :         printf("\t-A showall\n");
   16291           0 :         printf("\t-p port\n");
   16292           0 :         printf("\t-s seed\n");
   16293           0 :         printf("\t-b unclist_filename   specify multiple shares for multiple connections\n");
   16294           0 :         printf("\t-f filename           filename to test\n");
   16295           0 :         printf("\t-e                    encrypt\n");
   16296           0 :         printf("\n\n");
   16297             : 
   16298           0 :         printf("tests are:");
   16299           0 :         for (i=0;torture_ops[i].name;i++) {
   16300           0 :                 printf(" %s", torture_ops[i].name);
   16301             :         }
   16302           0 :         printf("\n");
   16303             : 
   16304           0 :         printf("default test is ALL\n");
   16305             : 
   16306           0 :         exit(1);
   16307             : }
   16308             : 
   16309             : /****************************************************************************
   16310             :   main program
   16311             : ****************************************************************************/
   16312         546 :  int main(int argc,char *argv[])
   16313             : {
   16314          45 :         int opt, i;
   16315          45 :         char *p;
   16316         546 :         int gotuser = 0;
   16317         546 :         int gotpass = 0;
   16318         546 :         bool correct = True;
   16319         546 :         TALLOC_CTX *frame = talloc_stackframe();
   16320         546 :         int seed = time(NULL);
   16321         546 :         struct loadparm_context *lp_ctx = NULL;
   16322             : 
   16323             : #ifdef HAVE_SETBUFFER
   16324         546 :         setbuffer(stdout, NULL, 0);
   16325             : #endif
   16326             : 
   16327         546 :         setup_logging("smbtorture", DEBUG_STDOUT);
   16328             : 
   16329         546 :         smb_init_locale();
   16330         546 :         fault_setup();
   16331             : 
   16332         546 :         lp_ctx = loadparm_init_s3(frame, loadparm_s3_helpers());
   16333         546 :         if (lp_ctx == NULL) {
   16334           0 :                 fprintf(stderr,
   16335             :                         "Failed to initialise the global parameter structure.\n");
   16336           0 :                 return 1;
   16337             :         }
   16338             : 
   16339         546 :         if (is_default_dyn_CONFIGFILE()) {
   16340         546 :                 if(getenv("SMB_CONF_PATH")) {
   16341         546 :                         set_dyn_CONFIGFILE(getenv("SMB_CONF_PATH"));
   16342             :                 }
   16343             :         }
   16344         546 :         lp_load_global(get_dyn_CONFIGFILE());
   16345         546 :         load_interfaces();
   16346             : 
   16347         546 :         if (argc < 2) {
   16348           0 :                 usage();
   16349             :         }
   16350             : 
   16351       11065 :         for(p = argv[1]; *p; p++)
   16352       10519 :           if(*p == '\\')
   16353           0 :             *p = '/';
   16354             : 
   16355         546 :         if (strncmp(argv[1], "//", 2)) {
   16356           0 :                 usage();
   16357             :         }
   16358             : 
   16359         546 :         fstrcpy(host, &argv[1][2]);
   16360         546 :         p = strchr_m(&host[2],'/');
   16361         546 :         if (!p) {
   16362           0 :                 usage();
   16363             :         }
   16364         546 :         *p = 0;
   16365         546 :         fstrcpy(share, p+1);
   16366             : 
   16367         546 :         fstrcpy(myname, get_myname(talloc_tos()));
   16368         546 :         if (!*myname) {
   16369           0 :                 fprintf(stderr, "Failed to get my hostname.\n");
   16370           0 :                 return 1;
   16371             :         }
   16372             : 
   16373         546 :         if (*username == 0 && getenv("LOGNAME")) {
   16374           0 :           fstrcpy(username,getenv("LOGNAME"));
   16375             :         }
   16376             : 
   16377         546 :         argc--;
   16378         546 :         argv++;
   16379             : 
   16380         546 :         fstrcpy(workgroup, lp_workgroup());
   16381             : 
   16382        1729 :         while ((opt = getopt(argc, argv, "p:hW:U:n:N:O:o:m:Ll:d:Aec:ks:b:B:f:"))
   16383        1774 :                != EOF) {
   16384        1228 :                 switch (opt) {
   16385           0 :                 case 'p':
   16386           0 :                         port_to_use = atoi(optarg);
   16387           0 :                         break;
   16388           0 :                 case 's':
   16389           0 :                         seed = atoi(optarg);
   16390           0 :                         break;
   16391           0 :                 case 'W':
   16392           0 :                         fstrcpy(workgroup,optarg);
   16393           0 :                         break;
   16394          33 :                 case 'm':
   16395          33 :                         lpcfg_set_cmdline(lp_ctx, "client max protocol", optarg);
   16396          33 :                         break;
   16397           0 :                 case 'N':
   16398           0 :                         torture_nprocs = atoi(optarg);
   16399           0 :                         break;
   16400           0 :                 case 'o':
   16401           0 :                         torture_numops = atoi(optarg);
   16402           0 :                         break;
   16403           0 :                 case 'd':
   16404           0 :                         lpcfg_set_cmdline(lp_ctx, "log level", optarg);
   16405           0 :                         break;
   16406           0 :                 case 'O':
   16407           0 :                         sockops = optarg;
   16408           0 :                         break;
   16409           0 :                 case 'L':
   16410           0 :                         use_oplocks = True;
   16411           0 :                         break;
   16412         463 :                 case 'l':
   16413         463 :                         local_path = optarg;
   16414         463 :                         break;
   16415           0 :                 case 'A':
   16416           0 :                         torture_showall = True;
   16417           0 :                         break;
   16418           0 :                 case 'n':
   16419           0 :                         fstrcpy(myname, optarg);
   16420           0 :                         break;
   16421           0 :                 case 'c':
   16422           0 :                         client_txt = optarg;
   16423           0 :                         break;
   16424         182 :                 case 'e':
   16425         182 :                         do_encrypt = true;
   16426         182 :                         break;
   16427           0 :                 case 'k':
   16428             : #ifdef HAVE_KRB5
   16429           0 :                         use_kerberos = True;
   16430             : #else
   16431             :                         d_printf("No kerberos support compiled in\n");
   16432             :                         exit(1);
   16433             : #endif
   16434           0 :                         break;
   16435         546 :                 case 'U':
   16436         546 :                         gotuser = 1;
   16437         546 :                         fstrcpy(username,optarg);
   16438         546 :                         p = strchr_m(username,'%');
   16439         546 :                         if (p) {
   16440         546 :                                 *p = 0;
   16441         546 :                                 fstrcpy(password, p+1);
   16442         546 :                                 gotpass = 1;
   16443             :                         }
   16444         501 :                         break;
   16445           0 :                 case 'b':
   16446           0 :                         fstrcpy(multishare_conn_fname, optarg);
   16447           0 :                         use_multishare_conn = True;
   16448           0 :                         break;
   16449           0 :                 case 'B':
   16450           0 :                         torture_blocksize = atoi(optarg);
   16451           0 :                         break;
   16452           4 :                 case 'f':
   16453           4 :                         test_filename = SMB_STRDUP(optarg);
   16454           4 :                         break;
   16455           0 :                 default:
   16456           0 :                         printf("Unknown option %c (%d)\n", (char)opt, opt);
   16457           0 :                         usage();
   16458             :                 }
   16459             :         }
   16460             : 
   16461         546 :         d_printf("using seed %d\n", seed);
   16462             : 
   16463         546 :         srandom(seed);
   16464             : 
   16465         546 :         if(use_kerberos && !gotuser) gotpass = True;
   16466             : 
   16467         546 :         while (!gotpass) {
   16468           0 :                 char pwd[256] = {0};
   16469           0 :                 int rc;
   16470             : 
   16471           0 :                 rc = samba_getpass("Password:", pwd, sizeof(pwd), false, false);
   16472           0 :                 if (rc == 0) {
   16473           0 :                         fstrcpy(password, pwd);
   16474           0 :                         gotpass = 1;
   16475             :                 }
   16476             :         }
   16477             : 
   16478         546 :         printf("host=%s share=%s user=%s myname=%s\n", 
   16479             :                host, share, username, myname);
   16480             : 
   16481         546 :         torture_creds = cli_session_creds_init(frame,
   16482             :                                                username,
   16483             :                                                workgroup,
   16484             :                                                NULL, /* realm */
   16485             :                                                password,
   16486             :                                                use_kerberos,
   16487             :                                                false, /* fallback_after_kerberos */
   16488             :                                                false, /* use_ccache */
   16489             :                                                false); /* password_is_nt_hash */
   16490         546 :         if (torture_creds == NULL) {
   16491           0 :                 d_printf("cli_session_creds_init() failed.\n");
   16492           0 :                 exit(1);
   16493             :         }
   16494             : 
   16495         546 :         if (argc == optind) {
   16496           0 :                 correct = run_test("ALL");
   16497             :         } else {
   16498        1096 :                 for (i=optind;i<argc;i++) {
   16499         546 :                         if (!run_test(argv[i])) {
   16500          16 :                                 correct = False;
   16501             :                         }
   16502             :                 }
   16503             :         }
   16504             : 
   16505         550 :         TALLOC_FREE(frame);
   16506             : 
   16507         550 :         if (correct) {
   16508         485 :                 return(0);
   16509             :         } else {
   16510          16 :                 return(1);
   16511             :         }
   16512             : }

Generated by: LCOV version 1.14