2 * Unix SMB/CIFS implementation.
3 * Util functions valid in the SMB1 server
5 * Copyright (C) Volker Lendecke 2019
6 * Copyright by the authors of the functions moved here eventually
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 3 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program. If not, see <http://www.gnu.org/licenses/>.
23 #include "libcli/security/security.h"
24 #include "lib/util/sys_rw_data.h"
25 #include "smbd/fd_handle.h"
27 /****************************************************************************
28 Special FCB or DOS processing in the case of a sharing violation.
29 Try and find a duplicated file handle.
30 ****************************************************************************/
32 struct files_struct
*fcb_or_dos_open(
33 struct smb_request
*req
,
34 const struct smb_filename
*smb_fname
,
36 uint32_t create_options
,
37 uint32_t private_flags
)
39 struct connection_struct
*conn
= req
->conn
;
40 struct file_id id
= vfs_file_id_from_sbuf(conn
, &smb_fname
->st
);
41 struct files_struct
*fsp
= NULL
, *new_fsp
= NULL
;
45 (NTCREATEX_FLAG_DENY_DOS
|
46 NTCREATEX_FLAG_DENY_FCB
))
51 for(fsp
= file_find_di_first(conn
->sconn
, id
, true);
53 fsp
= file_find_di_next(fsp
, true)) {
55 DBG_DEBUG("Checking file %s, fd = %d, vuid = %"PRIu64
", "
56 "file_pid = %"PRIu16
", "
57 "private_options = 0x%"PRIx32
", "
58 "access_mask = 0x%"PRIx32
"\n",
60 fsp_get_pathref_fd(fsp
),
63 fh_get_private_options(fsp
->fh
),
66 if (fsp_get_pathref_fd(fsp
) != -1 &&
67 fsp
->vuid
== req
->vuid
&&
68 fsp
->file_pid
== req
->smbpid
&&
69 (fh_get_private_options(fsp
->fh
) &
70 (NTCREATEX_FLAG_DENY_DOS
|
71 NTCREATEX_FLAG_DENY_FCB
)) &&
72 (fsp
->access_mask
& FILE_WRITE_DATA
) &&
73 strequal(fsp
->fsp_name
->base_name
, smb_fname
->base_name
) &&
74 strequal(fsp
->fsp_name
->stream_name
,
75 smb_fname
->stream_name
)) {
76 DBG_DEBUG("file match\n");
85 /* quite an insane set of semantics ... */
86 if (is_executable(smb_fname
->base_name
) &&
87 (fh_get_private_options(fsp
->fh
) & NTCREATEX_FLAG_DENY_DOS
)) {
88 DBG_DEBUG("file fail due to is_executable.\n");
92 status
= file_new(req
, conn
, &new_fsp
);
93 if (!NT_STATUS_IS_OK(status
)) {
94 DBG_DEBUG("file_new failed: %s\n", nt_errstr(status
));
98 status
= dup_file_fsp(
105 if (!NT_STATUS_IS_OK(status
)) {
106 DBG_DEBUG("dup_file_fsp failed: %s\n", nt_errstr(status
));
107 file_free(req
, new_fsp
);
114 /****************************************************************************
115 Send a keepalive packet (rfc1002).
116 ****************************************************************************/
118 bool send_keepalive(int client
)
120 unsigned char buf
[4];
122 buf
[0] = NBSSkeepalive
;
123 buf
[1] = buf
[2] = buf
[3] = 0;
125 return(write_data(client
,(char *)buf
,4) == 4);