2 Unix SMB/CIFS implementation.
4 Copyright (C) Stefan Metzmacher 2011
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 3 of the License, or
9 (at your option) any later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with this program. If not, see <http://www.gnu.org/licenses/>.
21 #include "system/network.h"
22 #include "lib/util/tevent_ntstatus.h"
23 #include "smb_common.h"
24 #include "smbXcli_base.h"
25 #include "librpc/ndr/libndr.h"
27 struct smb2cli_ioctl_state
{
30 uint32_t max_input_length
;
31 uint32_t max_output_length
;
32 struct iovec
*recv_iov
;
33 DATA_BLOB out_input_buffer
;
34 DATA_BLOB out_output_buffer
;
37 static void smb2cli_ioctl_done(struct tevent_req
*subreq
);
39 struct tevent_req
*smb2cli_ioctl_send(TALLOC_CTX
*mem_ctx
,
40 struct tevent_context
*ev
,
41 struct smbXcli_conn
*conn
,
42 uint32_t timeout_msec
,
43 struct smbXcli_session
*session
,
44 struct smbXcli_tcon
*tcon
,
45 uint64_t in_fid_persistent
,
46 uint64_t in_fid_volatile
,
48 uint32_t in_max_input_length
,
49 const DATA_BLOB
*in_input_buffer
,
50 uint32_t in_max_output_length
,
51 const DATA_BLOB
*in_output_buffer
,
54 struct tevent_req
*req
, *subreq
;
55 struct smb2cli_ioctl_state
*state
;
59 uint32_t input_buffer_offset
= 0;
60 uint32_t input_buffer_length
= 0;
61 uint32_t output_buffer_offset
= 0;
62 uint32_t output_buffer_length
= 0;
63 uint32_t pad_length
= 0;
65 req
= tevent_req_create(mem_ctx
, &state
,
66 struct smb2cli_ioctl_state
);
70 state
->max_input_length
= in_max_input_length
;
71 state
->max_output_length
= in_max_output_length
;
73 if (in_input_buffer
) {
74 input_buffer_offset
= SMB2_HDR_BODY
+0x38;
75 input_buffer_length
= in_input_buffer
->length
;
78 if (in_output_buffer
) {
79 output_buffer_offset
= SMB2_HDR_BODY
+0x38;
80 if (input_buffer_length
> 0 && output_buffer_length
> 0) {
82 output_buffer_offset
+= input_buffer_length
;
83 tmp
= output_buffer_offset
;
84 output_buffer_offset
= NDR_ROUND(output_buffer_offset
, 8);
85 pad_length
= output_buffer_offset
- tmp
;
87 output_buffer_length
= in_output_buffer
->length
;
92 SSVAL(fixed
, 0x00, 0x39);
93 SSVAL(fixed
, 0x02, 0); /* reserved */
94 SIVAL(fixed
, 0x04, in_ctl_code
);
95 SBVAL(fixed
, 0x08, in_fid_persistent
);
96 SBVAL(fixed
, 0x10, in_fid_volatile
);
97 SIVAL(fixed
, 0x18, input_buffer_offset
);
98 SIVAL(fixed
, 0x1C, input_buffer_length
);
99 SIVAL(fixed
, 0x20, in_max_input_length
);
100 SIVAL(fixed
, 0x24, output_buffer_offset
);
101 SIVAL(fixed
, 0x28, output_buffer_length
);
102 SIVAL(fixed
, 0x2C, in_max_output_length
);
103 SIVAL(fixed
, 0x30, in_flags
);
104 SIVAL(fixed
, 0x34, 0); /* reserved */
106 if (input_buffer_length
> 0 && output_buffer_length
> 0) {
107 size_t avail
= UINT32_MAX
- (input_buffer_length
+ pad_length
);
108 size_t ofs
= output_buffer_offset
- input_buffer_offset
;
110 if (avail
< output_buffer_length
) {
111 tevent_req_nterror(req
, NT_STATUS_INVALID_PARAMETER_MIX
);
112 return tevent_req_post(req
, ev
);
115 dyn_len
= input_buffer_length
+ output_buffer_length
+ pad_length
;
117 dyn
= talloc_zero_array(state
, uint8_t, dyn_len
);
118 if (tevent_req_nomem(dyn
, req
)) {
119 return tevent_req_post(req
, ev
);
121 memcpy(dyn
, in_input_buffer
->data
,
122 in_input_buffer
->length
);
123 memcpy(dyn
+ ofs
, in_output_buffer
->data
,
124 in_output_buffer
->length
);
125 } else if (input_buffer_length
> 0) {
126 dyn
= in_input_buffer
->data
;
127 dyn_len
= in_input_buffer
->length
;
128 } else if (output_buffer_length
> 0) {
129 dyn
= in_output_buffer
->data
;
130 dyn_len
= in_output_buffer
->length
;
132 dyn
= state
->dyn_pad
;
133 dyn_len
= sizeof(state
->dyn_pad
);
136 subreq
= smb2cli_req_send(state
, ev
, conn
, SMB2_OP_IOCTL
,
141 state
->fixed
, sizeof(state
->fixed
),
143 if (tevent_req_nomem(subreq
, req
)) {
144 return tevent_req_post(req
, ev
);
146 tevent_req_set_callback(subreq
, smb2cli_ioctl_done
, req
);
150 static void smb2cli_ioctl_done(struct tevent_req
*subreq
)
152 struct tevent_req
*req
=
153 tevent_req_callback_data(subreq
,
155 struct smb2cli_ioctl_state
*state
=
157 struct smb2cli_ioctl_state
);
163 uint32_t dyn_ofs
= SMB2_HDR_BODY
+ 0x30;
164 uint32_t input_buffer_offset
;
165 uint32_t input_buffer_length
;
166 uint32_t output_buffer_offset
;
167 uint32_t output_buffer_length
;
168 static const struct smb2cli_req_expected_response expected
[] = {
170 .status
= NT_STATUS_OK
,
174 .status
= STATUS_BUFFER_OVERFLOW
,
179 status
= smb2cli_req_recv(subreq
, state
, &iov
,
180 expected
, ARRAY_SIZE(expected
));
181 if (tevent_req_nterror(req
, status
)) {
185 state
->recv_iov
= iov
;
186 fixed
= (uint8_t *)iov
[1].iov_base
;
187 dyn
= (uint8_t *)iov
[2].iov_base
;
188 dyn_len
= iov
[2].iov_len
;
190 input_buffer_offset
= IVAL(fixed
, 0x18);
191 input_buffer_length
= IVAL(fixed
, 0x1C);
192 output_buffer_offset
= IVAL(fixed
, 0x20);
193 output_buffer_length
= IVAL(fixed
, 0x24);
195 if ((input_buffer_offset
> 0) && (input_buffer_length
> 0)) {
198 if (input_buffer_offset
!= dyn_ofs
) {
200 req
, NT_STATUS_INVALID_NETWORK_RESPONSE
);
204 if (input_buffer_length
< dyn_len
) {
206 req
, NT_STATUS_INVALID_NETWORK_RESPONSE
);
210 if (input_buffer_length
> state
->max_input_length
) {
212 req
, NT_STATUS_INVALID_NETWORK_RESPONSE
);
216 state
->out_input_buffer
.data
= dyn
;
217 state
->out_input_buffer
.length
= input_buffer_length
;
219 ofs
= input_buffer_length
;
220 ofs
= NDR_ROUND(ofs
, 8);
227 if ((output_buffer_offset
> 0) && (output_buffer_length
> 0)) {
228 if (output_buffer_offset
!= dyn_ofs
) {
230 req
, NT_STATUS_INVALID_NETWORK_RESPONSE
);
234 if (output_buffer_length
< dyn_len
) {
236 req
, NT_STATUS_INVALID_NETWORK_RESPONSE
);
240 if (output_buffer_length
> state
->max_output_length
) {
242 req
, NT_STATUS_INVALID_NETWORK_RESPONSE
);
246 state
->out_output_buffer
.data
= dyn
;
247 state
->out_output_buffer
.length
= output_buffer_length
;
250 tevent_req_done(req
);
253 NTSTATUS
smb2cli_ioctl_recv(struct tevent_req
*req
,
255 DATA_BLOB
*out_input_buffer
,
256 DATA_BLOB
*out_output_buffer
)
258 struct smb2cli_ioctl_state
*state
=
260 struct smb2cli_ioctl_state
);
263 if (tevent_req_is_nterror(req
, &status
)) {
264 tevent_req_received(req
);
268 talloc_steal(mem_ctx
, state
->recv_iov
);
269 if (out_input_buffer
) {
270 *out_input_buffer
= state
->out_input_buffer
;
272 if (out_output_buffer
) {
273 *out_output_buffer
= state
->out_output_buffer
;
276 tevent_req_received(req
);
280 NTSTATUS
smb2cli_ioctl(struct smbXcli_conn
*conn
,
281 uint32_t timeout_msec
,
282 struct smbXcli_session
*session
,
283 struct smbXcli_tcon
*tcon
,
284 uint64_t in_fid_persistent
,
285 uint64_t in_fid_volatile
,
286 uint32_t in_ctl_code
,
287 uint32_t in_max_input_length
,
288 const DATA_BLOB
*in_input_buffer
,
289 uint32_t in_max_output_length
,
290 const DATA_BLOB
*in_output_buffer
,
293 DATA_BLOB
*out_input_buffer
,
294 DATA_BLOB
*out_output_buffer
)
296 TALLOC_CTX
*frame
= talloc_stackframe();
297 struct tevent_context
*ev
;
298 struct tevent_req
*req
;
299 NTSTATUS status
= NT_STATUS_NO_MEMORY
;
301 if (smbXcli_conn_has_async_calls(conn
)) {
303 * Can't use sync call while an async call is in flight
305 status
= NT_STATUS_INVALID_PARAMETER_MIX
;
308 ev
= tevent_context_init(frame
);
312 req
= smb2cli_ioctl_send(frame
, ev
, conn
, timeout_msec
,
319 in_max_output_length
,
325 if (!tevent_req_poll_ntstatus(req
, ev
, &status
)) {
328 status
= smb2cli_ioctl_recv(req
, mem_ctx
,