2 Unix SMB/CIFS implementation.
4 SMB2 client getinfo calls
6 Copyright (C) Andrew Tridgell 2005
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 2 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, write to the Free Software
20 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
24 #include "libcli/raw/libcliraw.h"
25 #include "libcli/smb2/smb2.h"
26 #include "libcli/smb2/smb2_calls.h"
29 send a getinfo request
31 struct smb2_request
*smb2_getinfo_send(struct smb2_tree
*tree
, struct smb2_getinfo
*io
)
33 struct smb2_request
*req
;
35 req
= smb2_request_init_tree(tree
, SMB2_OP_GETINFO
, 0x28, False
, 0);
36 if (req
== NULL
) return NULL
;
38 /* this seems to be a bug, they use 0x29 but only send 0x28 bytes */
39 SSVAL(req
->out
.body
, 0x00, 0x29);
41 SSVAL(req
->out
.body
, 0x02, io
->in
.level
);
42 SIVAL(req
->out
.body
, 0x04, io
->in
.max_response_size
);
43 SIVAL(req
->out
.body
, 0x08, io
->in
.unknown1
);
44 SIVAL(req
->out
.body
, 0x0C, io
->in
.unknown2
);
45 SIVAL(req
->out
.body
, 0x10, io
->in
.flags
);
46 SIVAL(req
->out
.body
, 0x14, io
->in
.flags2
);
47 smb2_push_handle(req
->out
.body
+0x18, &io
->in
.file
.handle
);
49 smb2_transport_send(req
);
58 NTSTATUS
smb2_getinfo_recv(struct smb2_request
*req
, TALLOC_CTX
*mem_ctx
,
59 struct smb2_getinfo
*io
)
63 if (!smb2_request_receive(req
) ||
64 smb2_request_is_error(req
)) {
65 return smb2_request_destroy(req
);
68 SMB2_CHECK_PACKET_RECV(req
, 0x08, True
);
70 status
= smb2_pull_o16s16_blob(&req
->in
, mem_ctx
, req
->in
.body
+0x02, &io
->out
.blob
);
71 if (!NT_STATUS_IS_OK(status
)) {
75 return smb2_request_destroy(req
);
81 NTSTATUS
smb2_getinfo(struct smb2_tree
*tree
, TALLOC_CTX
*mem_ctx
,
82 struct smb2_getinfo
*io
)
84 struct smb2_request
*req
= smb2_getinfo_send(tree
, io
);
85 return smb2_getinfo_recv(req
, mem_ctx
, io
);
90 map a generic info level to a SMB2 info level
92 uint16_t smb2_getinfo_map_level(uint16_t level
, uint8_t class)
94 if (class == SMB2_GETINFO_FILE
&&
95 level
== RAW_FILEINFO_SEC_DESC
) {
96 return SMB2_GETINFO_SECURITY
;
98 if ((level
& 0xFF) == class) {
100 } else if (level
> 1000) {
101 return ((level
-1000)<<8) | class;
103 DEBUG(0,("Unable to map SMB2 info level 0x%04x of class %d\n", level
, class));
108 level specific getinfo call - async send
110 struct smb2_request
*smb2_getinfo_file_send(struct smb2_tree
*tree
, union smb_fileinfo
*io
)
112 struct smb2_getinfo b
;
113 uint16_t smb2_level
= smb2_getinfo_map_level(io
->generic
.level
, SMB2_GETINFO_FILE
);
115 if (smb2_level
== 0) {
120 b
.in
.max_response_size
= 0x10000;
121 b
.in
.file
.handle
= io
->generic
.in
.file
.handle
;
122 b
.in
.level
= smb2_level
;
124 if (io
->generic
.level
== RAW_FILEINFO_SEC_DESC
) {
125 b
.in
.flags
= io
->query_secdesc
.in
.secinfo_flags
;
127 if (io
->generic
.level
== RAW_FILEINFO_SMB2_ALL_EAS
) {
128 b
.in
.flags2
= io
->all_eas
.in
.continue_flags
;
131 return smb2_getinfo_send(tree
, &b
);
135 recv a getinfo reply and parse the level info
137 NTSTATUS
smb2_getinfo_file_recv(struct smb2_request
*req
, TALLOC_CTX
*mem_ctx
,
138 union smb_fileinfo
*io
)
140 struct smb2_getinfo b
;
143 status
= smb2_getinfo_recv(req
, mem_ctx
, &b
);
144 NT_STATUS_NOT_OK_RETURN(status
);
146 status
= smb_raw_fileinfo_passthru_parse(&b
.out
.blob
, mem_ctx
, io
->generic
.level
, io
);
147 data_blob_free(&b
.out
.blob
);
153 level specific getinfo call
155 NTSTATUS
smb2_getinfo_file(struct smb2_tree
*tree
, TALLOC_CTX
*mem_ctx
,
156 union smb_fileinfo
*io
)
158 struct smb2_request
*req
= smb2_getinfo_file_send(tree
, io
);
159 return smb2_getinfo_file_recv(req
, mem_ctx
, io
);
164 level specific getinfo call - async send
166 struct smb2_request
*smb2_getinfo_fs_send(struct smb2_tree
*tree
, union smb_fsinfo
*io
)
168 struct smb2_getinfo b
;
169 uint16_t smb2_level
= smb2_getinfo_map_level(io
->generic
.level
, SMB2_GETINFO_FS
);
171 if (smb2_level
== 0) {
176 b
.in
.max_response_size
= 0x10000;
177 b
.in
.file
.handle
= io
->generic
.handle
;
178 b
.in
.level
= smb2_level
;
180 return smb2_getinfo_send(tree
, &b
);
184 recv a getinfo reply and parse the level info
186 NTSTATUS
smb2_getinfo_fs_recv(struct smb2_request
*req
, TALLOC_CTX
*mem_ctx
,
187 union smb_fsinfo
*io
)
189 struct smb2_getinfo b
;
192 status
= smb2_getinfo_recv(req
, mem_ctx
, &b
);
193 NT_STATUS_NOT_OK_RETURN(status
);
195 status
= smb_raw_fsinfo_passthru_parse(b
.out
.blob
, mem_ctx
, io
->generic
.level
, io
);
196 data_blob_free(&b
.out
.blob
);
202 level specific getinfo call
204 NTSTATUS
smb2_getinfo_fs(struct smb2_tree
*tree
, TALLOC_CTX
*mem_ctx
,
205 union smb_fsinfo
*io
)
207 struct smb2_request
*req
= smb2_getinfo_fs_send(tree
, io
);
208 return smb2_getinfo_fs_recv(req
, mem_ctx
, io
);