2 Unix SMB/CIFS implementation.
4 Copyright (C) Stefan Metzmacher 2012
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"
26 struct smb2cli_query_info_state
{
29 uint32_t max_output_length
;
30 struct iovec
*recv_iov
;
31 DATA_BLOB out_output_buffer
;
35 static void smb2cli_query_info_done(struct tevent_req
*subreq
);
37 struct tevent_req
*smb2cli_query_info_send(TALLOC_CTX
*mem_ctx
,
38 struct tevent_context
*ev
,
39 struct smbXcli_conn
*conn
,
40 uint32_t timeout_msec
,
41 struct smbXcli_session
*session
,
42 struct smbXcli_tcon
*tcon
,
44 uint8_t in_file_info_class
,
45 uint32_t in_max_output_length
,
46 const DATA_BLOB
*in_input_buffer
,
47 uint32_t in_additional_info
,
49 uint64_t in_fid_persistent
,
50 uint64_t in_fid_volatile
)
52 struct tevent_req
*req
, *subreq
;
53 struct smb2cli_query_info_state
*state
;
57 uint16_t input_buffer_offset
= 0;
58 uint32_t input_buffer_length
= 0;
60 req
= tevent_req_create(mem_ctx
, &state
,
61 struct smb2cli_query_info_state
);
65 state
->max_output_length
= in_max_output_length
;
67 if (in_input_buffer
) {
68 input_buffer_offset
= SMB2_HDR_BODY
+0x28;
69 input_buffer_length
= in_input_buffer
->length
;
74 SSVAL(fixed
, 0x00, 0x29);
75 SCVAL(fixed
, 0x02, in_info_type
);
76 SCVAL(fixed
, 0x03, in_file_info_class
); /* reserved */
77 SIVAL(fixed
, 0x04, in_max_output_length
);
78 SSVAL(fixed
, 0x08, input_buffer_offset
);
79 SSVAL(fixed
, 0x0A, 0); /* reserved */
80 SIVAL(fixed
, 0x0C, input_buffer_length
);
81 SIVAL(fixed
, 0x10, in_additional_info
);
82 SIVAL(fixed
, 0x14, in_flags
);
83 SBVAL(fixed
, 0x18, in_fid_persistent
);
84 SBVAL(fixed
, 0x20, in_fid_volatile
);
86 if (input_buffer_length
> 0) {
87 dyn
= in_input_buffer
->data
;
88 dyn_len
= in_input_buffer
->length
;
91 dyn_len
= sizeof(state
->dyn_pad
);
94 subreq
= smb2cli_req_send(state
, ev
, conn
, SMB2_OP_GETINFO
,
99 state
->fixed
, sizeof(state
->fixed
),
101 in_max_output_length
); /* max_dyn_len */
102 if (tevent_req_nomem(subreq
, req
)) {
103 return tevent_req_post(req
, ev
);
105 tevent_req_set_callback(subreq
, smb2cli_query_info_done
, req
);
109 static void smb2cli_query_info_done(struct tevent_req
*subreq
)
111 struct tevent_req
*req
=
112 tevent_req_callback_data(subreq
,
114 struct smb2cli_query_info_state
*state
=
116 struct smb2cli_query_info_state
);
122 uint32_t dyn_ofs
= SMB2_HDR_BODY
+ 0x08;
123 uint32_t output_buffer_offset
;
124 uint32_t output_buffer_length
;
125 static const struct smb2cli_req_expected_response expected
[] = {
127 .status
= NT_STATUS_OK
,
131 .status
= STATUS_BUFFER_OVERFLOW
,
136 status
= smb2cli_req_recv(subreq
, state
, &iov
,
137 expected
, ARRAY_SIZE(expected
));
139 if (NT_STATUS_EQUAL(status
, STATUS_BUFFER_OVERFLOW
)) {
142 if (tevent_req_nterror(req
, status
)) {
147 state
->recv_iov
= iov
;
148 fixed
= (uint8_t *)iov
[1].iov_base
;
149 dyn
= (uint8_t *)iov
[2].iov_base
;
150 dyn_len
= iov
[2].iov_len
;
152 output_buffer_offset
= SVAL(fixed
, 0x02);
153 output_buffer_length
= IVAL(fixed
, 0x04);
155 if ((output_buffer_offset
> 0) && (output_buffer_length
> 0)) {
156 if (output_buffer_offset
!= dyn_ofs
) {
158 req
, NT_STATUS_INVALID_NETWORK_RESPONSE
);
162 if (output_buffer_length
> dyn_len
) {
164 req
, NT_STATUS_INVALID_NETWORK_RESPONSE
);
168 if (output_buffer_length
> state
->max_output_length
) {
170 req
, NT_STATUS_INVALID_NETWORK_RESPONSE
);
174 state
->out_output_buffer
.data
= dyn
;
175 state
->out_output_buffer
.length
= output_buffer_length
;
178 state
->out_valid
= true;
180 if (tevent_req_nterror(req
, status
)) {
184 tevent_req_done(req
);
187 NTSTATUS
smb2cli_query_info_recv(struct tevent_req
*req
,
189 DATA_BLOB
*out_output_buffer
)
191 struct smb2cli_query_info_state
*state
=
193 struct smb2cli_query_info_state
);
194 NTSTATUS status
= NT_STATUS_OK
;
196 if (tevent_req_is_nterror(req
, &status
) && !state
->out_valid
) {
197 if (out_output_buffer
) {
198 *out_output_buffer
= data_blob_null
;
200 tevent_req_received(req
);
204 talloc_steal(mem_ctx
, state
->recv_iov
);
205 if (out_output_buffer
) {
206 *out_output_buffer
= state
->out_output_buffer
;
209 tevent_req_received(req
);
213 NTSTATUS
smb2cli_query_info(struct smbXcli_conn
*conn
,
214 uint32_t timeout_msec
,
215 struct smbXcli_session
*session
,
216 struct smbXcli_tcon
*tcon
,
217 uint8_t in_info_type
,
218 uint8_t in_file_info_class
,
219 uint32_t in_max_output_length
,
220 const DATA_BLOB
*in_input_buffer
,
221 uint32_t in_additional_info
,
223 uint64_t in_fid_persistent
,
224 uint64_t in_fid_volatile
,
226 DATA_BLOB
*out_output_buffer
)
228 TALLOC_CTX
*frame
= talloc_stackframe();
229 struct tevent_context
*ev
;
230 struct tevent_req
*req
;
231 NTSTATUS status
= NT_STATUS_NO_MEMORY
;
233 if (smbXcli_conn_has_async_calls(conn
)) {
235 * Can't use sync call while an async call is in flight
237 status
= NT_STATUS_INVALID_PARAMETER_MIX
;
240 ev
= samba_tevent_context_init(frame
);
244 req
= smb2cli_query_info_send(frame
, ev
,
249 in_max_output_length
,
258 if (!tevent_req_poll_ntstatus(req
, ev
, &status
)) {
261 status
= smb2cli_query_info_recv(req
, mem_ctx
,