r11771: - split out the setinfo blob construction in the libcli/raw code
[samba.git] / source4 / libcli / smb2 / getinfo.c
1 /* 
2    Unix SMB/CIFS implementation.
3
4    SMB2 client getinfo calls
5
6    Copyright (C) Andrew Tridgell 2005
7    
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.
12    
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.
17    
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.
21 */
22
23 #include "includes.h"
24 #include "libcli/raw/libcliraw.h"
25 #include "libcli/smb2/smb2.h"
26 #include "libcli/smb2/smb2_calls.h"
27
28 /*
29   send a getinfo request
30 */
31 struct smb2_request *smb2_getinfo_send(struct smb2_tree *tree, struct smb2_getinfo *io)
32 {
33         struct smb2_request *req;
34
35         req = smb2_request_init_tree(tree, SMB2_OP_GETINFO, 0x28, 0);
36         if (req == NULL) return NULL;
37
38         /* this seems to be a bug, they use 0x29 but only send 0x28 bytes */
39         SSVAL(req->out.body, 0x00, 0x29);
40
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.unknown4);
47         smb2_push_handle(req->out.body+0x18, &io->in.handle);
48
49         smb2_transport_send(req);
50
51         return req;
52 }
53
54
55 /*
56   recv a getinfo reply
57 */
58 NTSTATUS smb2_getinfo_recv(struct smb2_request *req, TALLOC_CTX *mem_ctx,
59                            struct smb2_getinfo *io)
60 {
61         NTSTATUS status;
62
63         if (!smb2_request_receive(req) || 
64             smb2_request_is_error(req)) {
65                 return smb2_request_destroy(req);
66         }
67
68         SMB2_CHECK_PACKET_RECV(req, 0x08, True);
69
70         status = smb2_pull_o16s16_blob(&req->in, mem_ctx, req->in.body+0x02, &io->out.blob);
71         if (!NT_STATUS_IS_OK(status)) {
72                 return status;
73         }
74
75         return smb2_request_destroy(req);
76 }
77
78 /*
79   sync getinfo request
80 */
81 NTSTATUS smb2_getinfo(struct smb2_tree *tree, TALLOC_CTX *mem_ctx,
82                       struct smb2_getinfo *io)
83 {
84         struct smb2_request *req = smb2_getinfo_send(tree, io);
85         return smb2_getinfo_recv(req, mem_ctx, io);
86 }
87
88
89 /*
90   map a generic info level to a SMB2 info level
91 */
92 uint16_t smb2_getinfo_map_level(uint16_t level, uint8_t class)
93 {
94         if ((level & 0xFF) == class) {
95                 return level;
96         } else if (level > 1000) {
97                 return ((level-1000)<<8) | class;
98         }
99         DEBUG(0,("Unable to map SMB2 info level 0x%04x of class %d\n", level, class));
100         return 0;       
101 }
102
103 /*
104   level specific getinfo call - async send
105 */
106 struct smb2_request *smb2_getinfo_file_send(struct smb2_tree *tree, union smb_fileinfo *io)
107 {
108         struct smb2_getinfo b;
109         uint16_t smb2_level = smb2_getinfo_map_level(io->generic.level, SMB2_GETINFO_FILE);
110         
111         if (smb2_level == 0) {
112                 return NULL;
113         }
114
115         ZERO_STRUCT(b);
116         b.in.max_response_size = 0x10000;
117         b.in.handle            = io->generic.in.handle;
118         b.in.level             = smb2_level;
119
120         if (io->generic.level == RAW_FILEINFO_SEC_DESC) {
121                 b.in.flags = io->query_secdesc.secinfo_flags;
122         }
123
124         return smb2_getinfo_send(tree, &b);
125 }
126
127 /*
128   recv a getinfo reply and parse the level info
129 */
130 NTSTATUS smb2_getinfo_file_recv(struct smb2_request *req, TALLOC_CTX *mem_ctx,
131                                 union smb_fileinfo *io)
132 {
133         struct smb2_getinfo b;
134         NTSTATUS status;
135
136         status = smb2_getinfo_recv(req, mem_ctx, &b);
137         NT_STATUS_NOT_OK_RETURN(status);
138
139         status = smb_raw_fileinfo_passthru_parse(&b.out.blob, mem_ctx, io->generic.level, io);
140         data_blob_free(&b.out.blob);
141
142         return status;
143 }
144
145 /*
146   level specific getinfo call
147 */
148 NTSTATUS smb2_getinfo_file(struct smb2_tree *tree, TALLOC_CTX *mem_ctx, 
149                            union smb_fileinfo *io)
150 {
151         struct smb2_request *req = smb2_getinfo_file_send(tree, io);
152         return smb2_getinfo_file_recv(req, mem_ctx, io);
153 }
154
155
156 /*
157   level specific getinfo call - async send
158 */
159 struct smb2_request *smb2_getinfo_fs_send(struct smb2_tree *tree, union smb_fsinfo *io)
160 {
161         struct smb2_getinfo b;
162         uint16_t smb2_level = smb2_getinfo_map_level(io->generic.level, SMB2_GETINFO_FS);
163         
164         if (smb2_level == 0) {
165                 return NULL;
166         }
167         
168         ZERO_STRUCT(b);
169         b.in.max_response_size = 0x10000;
170         b.in.handle            = io->generic.handle;
171         b.in.level             = smb2_level;
172
173         return smb2_getinfo_send(tree, &b);
174 }
175
176 /*
177   recv a getinfo reply and parse the level info
178 */
179 NTSTATUS smb2_getinfo_fs_recv(struct smb2_request *req, TALLOC_CTX *mem_ctx,
180                                 union smb_fsinfo *io)
181 {
182         struct smb2_getinfo b;
183         NTSTATUS status;
184
185         status = smb2_getinfo_recv(req, mem_ctx, &b);
186         NT_STATUS_NOT_OK_RETURN(status);
187
188         status = smb_raw_fsinfo_passthru_parse(b.out.blob, mem_ctx, io->generic.level, io);
189         data_blob_free(&b.out.blob);
190
191         return status;
192 }
193
194 /*
195   level specific getinfo call
196 */
197 NTSTATUS smb2_getinfo_fs(struct smb2_tree *tree, TALLOC_CTX *mem_ctx, 
198                            union smb_fsinfo *io)
199 {
200         struct smb2_request *req = smb2_getinfo_fs_send(tree, io);
201         return smb2_getinfo_fs_recv(req, mem_ctx, io);
202 }
203