1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
|
/*
* This file and its contents are supplied under the terms of the
* Common Development and Distribution License ("CDDL"), version 1.0.
* You may only use this file in accordance with the terms of version
* 1.0 of the CDDL.
*
* A full copy of the text of the CDDL should have accompanied this
* source. A copy of the CDDL is also available via the Internet at
* http://www.illumos.org/license/CDDL.
*/
/*
* Copyright 2017 Nexenta Systems, Inc. All rights reserved.
*/
/*
* Dispatch function for SMB2_SET_INFO
*/
#include <smbsrv/smb2_kproto.h>
#include <smbsrv/smb_fsops.h>
#include <smbsrv/ntifs.h>
smb_sdrc_t
smb2_set_info(smb_request_t *sr)
{
smb_setinfo_t sinfo;
uint16_t StructSize;
uint16_t iBufOffset;
uint32_t iBufLength;
uint32_t AddlInfo;
smb2fid_t smb2fid;
uint32_t status;
uint8_t InfoType, InfoClass;
int rc = 0;
bzero(&sinfo, sizeof (sinfo));
/*
* Decode SMB2 Set Info request
*/
rc = smb_mbc_decodef(
&sr->smb_data, "wbblw..lqq",
&StructSize, /* w */
&InfoType, /* b */
&InfoClass, /* b */
&iBufLength, /* l */
&iBufOffset, /* w */
/* reserved .. */
&AddlInfo, /* l */
&smb2fid.persistent, /* q */
&smb2fid.temporal); /* q */
if (rc || StructSize != 33)
return (SDRC_ERROR);
/*
* If there's an input buffer, setup a shadow.
*/
if (iBufLength) {
rc = MBC_SHADOW_CHAIN(&sinfo.si_data, &sr->smb_data,
sr->smb2_cmd_hdr + iBufOffset, iBufLength);
if (rc) {
return (SDRC_ERROR);
}
}
/* No output data. */
sr->raw_data.max_bytes = 0;
status = smb2sr_lookup_fid(sr, &smb2fid);
DTRACE_SMB2_START(op__SetInfo, smb_request_t *, sr);
if (status)
goto errout;
if (iBufLength > smb2_max_trans) {
status = NT_STATUS_INVALID_PARAMETER;
goto errout;
}
sinfo.si_node = sr->fid_ofile->f_node;
sr->user_cr = sr->fid_ofile->f_cr;
switch (InfoType) {
case SMB2_0_INFO_FILE:
status = smb2_setinfo_file(sr, &sinfo, InfoClass);
break;
case SMB2_0_INFO_FILESYSTEM:
status = smb2_setinfo_fs(sr, &sinfo, InfoClass);
break;
case SMB2_0_INFO_SECURITY:
status = smb2_setinfo_sec(sr, &sinfo, AddlInfo);
break;
case SMB2_0_INFO_QUOTA:
status = smb2_setinfo_quota(sr, &sinfo);
break;
default:
status = NT_STATUS_INVALID_PARAMETER;
break;
}
errout:
sr->smb2_status = status;
DTRACE_SMB2_DONE(op__SetInfo, smb_request_t *, sr);
if (status) {
smb2sr_put_error(sr, status);
return (SDRC_SUCCESS);
}
/*
* SMB2 Query Info reply
*/
(void) smb_mbc_encodef(
&sr->reply, "w..",
2); /* StructSize */ /* w */
return (SDRC_SUCCESS);
}
|