blob: eb74cceef480831cabd5249943c7f3487a633663 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * fs/cifs/cifssmb.c
3 *
Steve Frenchf19159d2010-04-21 04:12:10 +00004 * Copyright (C) International Business Machines Corp., 2002,2010
Linus Torvalds1da177e2005-04-16 15:20:36 -07005 * Author(s): Steve French (sfrench@us.ibm.com)
6 *
7 * Contains the routines for constructing the SMB PDUs themselves
8 *
9 * This library is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU Lesser General Public License as published
11 * by the Free Software Foundation; either version 2.1 of the License, or
12 * (at your option) any later version.
13 *
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
17 * the GNU Lesser General Public License for more details.
18 *
19 * You should have received a copy of the GNU Lesser General Public License
20 * along with this library; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 */
23
24 /* SMB/CIFS PDU handling routines here - except for leftovers in connect.c */
25 /* These are mostly routines that operate on a pathname, or on a tree id */
26 /* (mounted volume), but there are eight handle based routines which must be */
Steve French2dd29d32007-04-23 22:07:35 +000027 /* treated slightly differently for reconnection purposes since we never */
28 /* want to reuse a stale file handle and only the caller knows the file info */
Linus Torvalds1da177e2005-04-16 15:20:36 -070029
30#include <linux/fs.h>
31#include <linux/kernel.h>
32#include <linux/vfs.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090033#include <linux/slab.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070034#include <linux/posix_acl_xattr.h>
Jeff Laytonc28c89f2011-05-19 16:22:56 -040035#include <linux/pagemap.h>
Jeff Laytone28bc5b2011-10-19 15:30:07 -040036#include <linux/swap.h>
37#include <linux/task_io_accounting_ops.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070038#include <asm/uaccess.h>
39#include "cifspdu.h"
40#include "cifsglob.h"
Shirish Pargaonkard0d66c42007-10-03 18:22:19 +000041#include "cifsacl.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070042#include "cifsproto.h"
43#include "cifs_unicode.h"
44#include "cifs_debug.h"
Jeff Laytone28bc5b2011-10-19 15:30:07 -040045#include "fscache.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070046
47#ifdef CONFIG_CIFS_POSIX
48static struct {
49 int index;
50 char *name;
51} protocols[] = {
Steve French39798772006-05-31 22:40:51 +000052#ifdef CONFIG_CIFS_WEAK_PW_HASH
53 {LANMAN_PROT, "\2LM1.2X002"},
Steve French9ac00b72006-09-30 04:13:17 +000054 {LANMAN2_PROT, "\2LANMAN2.1"},
Steve French39798772006-05-31 22:40:51 +000055#endif /* weak password hashing for legacy clients */
Steve French50c2f752007-07-13 00:33:32 +000056 {CIFS_PROT, "\2NT LM 0.12"},
Steve French39798772006-05-31 22:40:51 +000057 {POSIX_PROT, "\2POSIX 2"},
Linus Torvalds1da177e2005-04-16 15:20:36 -070058 {BAD_PROT, "\2"}
59};
60#else
61static struct {
62 int index;
63 char *name;
64} protocols[] = {
Steve French39798772006-05-31 22:40:51 +000065#ifdef CONFIG_CIFS_WEAK_PW_HASH
66 {LANMAN_PROT, "\2LM1.2X002"},
Steve French18f75ca2006-10-01 03:13:01 +000067 {LANMAN2_PROT, "\2LANMAN2.1"},
Steve French39798772006-05-31 22:40:51 +000068#endif /* weak password hashing for legacy clients */
Steve French790fe572007-07-07 19:25:05 +000069 {CIFS_PROT, "\2NT LM 0.12"},
Linus Torvalds1da177e2005-04-16 15:20:36 -070070 {BAD_PROT, "\2"}
71};
72#endif
73
Steve French39798772006-05-31 22:40:51 +000074/* define the number of elements in the cifs dialect array */
75#ifdef CONFIG_CIFS_POSIX
76#ifdef CONFIG_CIFS_WEAK_PW_HASH
Steve French9ac00b72006-09-30 04:13:17 +000077#define CIFS_NUM_PROT 4
Steve French39798772006-05-31 22:40:51 +000078#else
79#define CIFS_NUM_PROT 2
80#endif /* CIFS_WEAK_PW_HASH */
81#else /* not posix */
82#ifdef CONFIG_CIFS_WEAK_PW_HASH
Steve French9ac00b72006-09-30 04:13:17 +000083#define CIFS_NUM_PROT 3
Steve French39798772006-05-31 22:40:51 +000084#else
85#define CIFS_NUM_PROT 1
86#endif /* CONFIG_CIFS_WEAK_PW_HASH */
87#endif /* CIFS_POSIX */
88
Jeff Layton3cf003c2012-07-11 09:09:36 -040089#ifdef CONFIG_HIGHMEM
90/*
91 * On arches that have high memory, kmap address space is limited. By
92 * serializing the kmap operations on those arches, we ensure that we don't
93 * end up with a bunch of threads in writeback with partially mapped page
94 * arrays, stuck waiting for kmap to come back. That situation prevents
95 * progress and can deadlock.
96 */
97static DEFINE_MUTEX(cifs_kmap_mutex);
98
99static inline void
100cifs_kmap_lock(void)
101{
102 mutex_lock(&cifs_kmap_mutex);
103}
104
105static inline void
106cifs_kmap_unlock(void)
107{
108 mutex_unlock(&cifs_kmap_mutex);
109}
110#else /* !CONFIG_HIGHMEM */
111#define cifs_kmap_lock() do { ; } while(0)
112#define cifs_kmap_unlock() do { ; } while(0)
113#endif /* CONFIG_HIGHMEM */
Jeff Laytone28bc5b2011-10-19 15:30:07 -0400114
Pavel Shilovskyaa24d1e2011-12-27 16:23:34 +0400115/*
116 * Mark as invalid, all open files on tree connections since they
117 * were closed when session to server was lost.
118 */
119void
120cifs_mark_open_files_invalid(struct cifs_tcon *tcon)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700121{
122 struct cifsFileInfo *open_file = NULL;
Steve French790fe572007-07-07 19:25:05 +0000123 struct list_head *tmp;
124 struct list_head *tmp1;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700125
Pavel Shilovskyaa24d1e2011-12-27 16:23:34 +0400126 /* list all files open on tree connection and mark them invalid */
Jeff Layton44772882010-10-15 15:34:03 -0400127 spin_lock(&cifs_file_list_lock);
Pavel Shilovskyaa24d1e2011-12-27 16:23:34 +0400128 list_for_each_safe(tmp, tmp1, &tcon->openFileList) {
Steve French790fe572007-07-07 19:25:05 +0000129 open_file = list_entry(tmp, struct cifsFileInfo, tlist);
Steve Frenchad8b15f2008-08-08 21:10:16 +0000130 open_file->invalidHandle = true;
Jeff Layton3bc303c2009-09-21 06:47:50 -0400131 open_file->oplock_break_cancelled = true;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700132 }
Jeff Layton44772882010-10-15 15:34:03 -0400133 spin_unlock(&cifs_file_list_lock);
Pavel Shilovskyaa24d1e2011-12-27 16:23:34 +0400134 /*
135 * BB Add call to invalidate_inodes(sb) for all superblocks mounted
136 * to this tcon.
137 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700138}
139
Jeff Layton9162ab22009-09-03 12:07:17 -0400140/* reconnect the socket, tcon, and smb session if needed */
141static int
Steve French96daf2b2011-05-27 04:34:02 +0000142cifs_reconnect_tcon(struct cifs_tcon *tcon, int smb_command)
Jeff Layton9162ab22009-09-03 12:07:17 -0400143{
Jeff Laytonc4a55342011-07-28 12:40:36 -0400144 int rc;
Steve French96daf2b2011-05-27 04:34:02 +0000145 struct cifs_ses *ses;
Jeff Layton9162ab22009-09-03 12:07:17 -0400146 struct TCP_Server_Info *server;
147 struct nls_table *nls_codepage;
148
149 /*
150 * SMBs NegProt, SessSetup, uLogoff do not have tcon yet so check for
151 * tcp and smb session status done differently for those three - in the
152 * calling routine
153 */
154 if (!tcon)
155 return 0;
156
157 ses = tcon->ses;
158 server = ses->server;
159
160 /*
161 * only tree disconnect, open, and write, (and ulogoff which does not
162 * have tcon) are allowed as we start force umount
163 */
164 if (tcon->tidStatus == CifsExiting) {
165 if (smb_command != SMB_COM_WRITE_ANDX &&
166 smb_command != SMB_COM_OPEN_ANDX &&
167 smb_command != SMB_COM_TREE_DISCONNECT) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000168 cFYI(1, "can not send cmd %d while umounting",
169 smb_command);
Jeff Layton9162ab22009-09-03 12:07:17 -0400170 return -ENODEV;
171 }
172 }
173
Jeff Layton9162ab22009-09-03 12:07:17 -0400174 /*
175 * Give demultiplex thread up to 10 seconds to reconnect, should be
176 * greater than cifs socket timeout which is 7 seconds
177 */
178 while (server->tcpStatus == CifsNeedReconnect) {
179 wait_event_interruptible_timeout(server->response_q,
Steve Frenchfd88ce92011-04-12 01:01:14 +0000180 (server->tcpStatus != CifsNeedReconnect), 10 * HZ);
Jeff Layton9162ab22009-09-03 12:07:17 -0400181
Steve Frenchfd88ce92011-04-12 01:01:14 +0000182 /* are we still trying to reconnect? */
Jeff Layton9162ab22009-09-03 12:07:17 -0400183 if (server->tcpStatus != CifsNeedReconnect)
184 break;
185
186 /*
187 * on "soft" mounts we wait once. Hard mounts keep
188 * retrying until process is killed or server comes
189 * back on-line
190 */
Jeff Laytond4025392011-02-07 08:54:35 -0500191 if (!tcon->retry) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000192 cFYI(1, "gave up waiting on reconnect in smb_init");
Jeff Layton9162ab22009-09-03 12:07:17 -0400193 return -EHOSTDOWN;
194 }
195 }
196
197 if (!ses->need_reconnect && !tcon->need_reconnect)
198 return 0;
199
200 nls_codepage = load_nls_default();
201
202 /*
203 * need to prevent multiple threads trying to simultaneously
204 * reconnect the same SMB session
205 */
Steve Frenchd7b619c2010-02-25 05:36:46 +0000206 mutex_lock(&ses->session_mutex);
Jeff Layton198b5682010-04-24 07:57:48 -0400207 rc = cifs_negotiate_protocol(0, ses);
208 if (rc == 0 && ses->need_reconnect)
Jeff Layton9162ab22009-09-03 12:07:17 -0400209 rc = cifs_setup_session(0, ses, nls_codepage);
210
211 /* do we need to reconnect tcon? */
212 if (rc || !tcon->need_reconnect) {
Steve Frenchd7b619c2010-02-25 05:36:46 +0000213 mutex_unlock(&ses->session_mutex);
Jeff Layton9162ab22009-09-03 12:07:17 -0400214 goto out;
215 }
216
Pavel Shilovskyaa24d1e2011-12-27 16:23:34 +0400217 cifs_mark_open_files_invalid(tcon);
Jeff Layton9162ab22009-09-03 12:07:17 -0400218 rc = CIFSTCon(0, ses, tcon->treeName, tcon, nls_codepage);
Steve Frenchd7b619c2010-02-25 05:36:46 +0000219 mutex_unlock(&ses->session_mutex);
Joe Perchesb6b38f72010-04-21 03:50:45 +0000220 cFYI(1, "reconnect tcon rc = %d", rc);
Jeff Layton9162ab22009-09-03 12:07:17 -0400221
222 if (rc)
223 goto out;
224
225 /*
226 * FIXME: check if wsize needs updated due to negotiated smb buffer
227 * size shrinking
228 */
229 atomic_inc(&tconInfoReconnectCount);
230
231 /* tell server Unix caps we support */
232 if (ses->capabilities & CAP_UNIX)
233 reset_cifs_unix_caps(0, tcon, NULL, NULL);
234
235 /*
236 * Removed call to reopen open files here. It is safer (and faster) to
237 * reopen files one at a time as needed in read and write.
238 *
239 * FIXME: what about file locks? don't we need to reclaim them ASAP?
240 */
241
242out:
243 /*
244 * Check if handle based operation so we know whether we can continue
245 * or not without returning to caller to reset file handle
246 */
247 switch (smb_command) {
248 case SMB_COM_READ_ANDX:
249 case SMB_COM_WRITE_ANDX:
250 case SMB_COM_CLOSE:
251 case SMB_COM_FIND_CLOSE2:
252 case SMB_COM_LOCKING_ANDX:
253 rc = -EAGAIN;
254 }
255
256 unload_nls(nls_codepage);
257 return rc;
258}
259
Steve Frenchad7a2922008-02-07 23:25:02 +0000260/* Allocate and return pointer to an SMB request buffer, and set basic
261 SMB information in the SMB header. If the return code is zero, this
262 function must have filled in request_buf pointer */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700263static int
Steve French96daf2b2011-05-27 04:34:02 +0000264small_smb_init(int smb_command, int wct, struct cifs_tcon *tcon,
Steve Frenchad7a2922008-02-07 23:25:02 +0000265 void **request_buf)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700266{
Jeff Laytonf5695992010-09-29 15:27:08 -0400267 int rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700268
Jeff Layton9162ab22009-09-03 12:07:17 -0400269 rc = cifs_reconnect_tcon(tcon, smb_command);
Steve French790fe572007-07-07 19:25:05 +0000270 if (rc)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700271 return rc;
272
273 *request_buf = cifs_small_buf_get();
274 if (*request_buf == NULL) {
275 /* BB should we add a retry in here if not a writepage? */
276 return -ENOMEM;
277 }
278
Steve French63135e02007-07-17 17:34:02 +0000279 header_assemble((struct smb_hdr *) *request_buf, smb_command,
Steve Frenchc18c8422007-07-18 23:21:09 +0000280 tcon, wct);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700281
Steve French790fe572007-07-07 19:25:05 +0000282 if (tcon != NULL)
283 cifs_stats_inc(&tcon->num_smbs_sent);
Steve Frencha4544342005-08-24 13:59:35 -0700284
Jeff Laytonf5695992010-09-29 15:27:08 -0400285 return 0;
Steve French5815449d2006-02-14 01:36:20 +0000286}
287
Steve French12b3b8f2006-02-09 21:12:47 +0000288int
Steve French50c2f752007-07-13 00:33:32 +0000289small_smb_init_no_tc(const int smb_command, const int wct,
Steve French96daf2b2011-05-27 04:34:02 +0000290 struct cifs_ses *ses, void **request_buf)
Steve French12b3b8f2006-02-09 21:12:47 +0000291{
292 int rc;
Steve French50c2f752007-07-13 00:33:32 +0000293 struct smb_hdr *buffer;
Steve French12b3b8f2006-02-09 21:12:47 +0000294
Steve French5815449d2006-02-14 01:36:20 +0000295 rc = small_smb_init(smb_command, wct, NULL, request_buf);
Steve French790fe572007-07-07 19:25:05 +0000296 if (rc)
Steve French12b3b8f2006-02-09 21:12:47 +0000297 return rc;
298
Steve French04fdabe2006-02-10 05:52:50 +0000299 buffer = (struct smb_hdr *)*request_buf;
Pavel Shilovsky88257362012-05-23 14:01:59 +0400300 buffer->Mid = get_next_mid(ses->server);
Steve French12b3b8f2006-02-09 21:12:47 +0000301 if (ses->capabilities & CAP_UNICODE)
302 buffer->Flags2 |= SMBFLG2_UNICODE;
Steve French04fdabe2006-02-10 05:52:50 +0000303 if (ses->capabilities & CAP_STATUS32)
Steve French12b3b8f2006-02-09 21:12:47 +0000304 buffer->Flags2 |= SMBFLG2_ERR_STATUS;
305
306 /* uid, tid can stay at zero as set in header assemble */
307
Steve French50c2f752007-07-13 00:33:32 +0000308 /* BB add support for turning on the signing when
Steve French12b3b8f2006-02-09 21:12:47 +0000309 this function is used after 1st of session setup requests */
310
311 return rc;
312}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700313
314/* If the return code is zero, this function must fill in request_buf pointer */
315static int
Steve French96daf2b2011-05-27 04:34:02 +0000316__smb_init(int smb_command, int wct, struct cifs_tcon *tcon,
Jeff Laytonf5695992010-09-29 15:27:08 -0400317 void **request_buf, void **response_buf)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700318{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700319 *request_buf = cifs_buf_get();
320 if (*request_buf == NULL) {
321 /* BB should we add a retry in here if not a writepage? */
322 return -ENOMEM;
323 }
324 /* Although the original thought was we needed the response buf for */
325 /* potential retries of smb operations it turns out we can determine */
326 /* from the mid flags when the request buffer can be resent without */
327 /* having to use a second distinct buffer for the response */
Steve French790fe572007-07-07 19:25:05 +0000328 if (response_buf)
Steve French50c2f752007-07-13 00:33:32 +0000329 *response_buf = *request_buf;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700330
331 header_assemble((struct smb_hdr *) *request_buf, smb_command, tcon,
Steve Frenchad7a2922008-02-07 23:25:02 +0000332 wct);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700333
Steve French790fe572007-07-07 19:25:05 +0000334 if (tcon != NULL)
335 cifs_stats_inc(&tcon->num_smbs_sent);
Steve Frencha4544342005-08-24 13:59:35 -0700336
Jeff Laytonf5695992010-09-29 15:27:08 -0400337 return 0;
338}
339
340/* If the return code is zero, this function must fill in request_buf pointer */
341static int
Steve French96daf2b2011-05-27 04:34:02 +0000342smb_init(int smb_command, int wct, struct cifs_tcon *tcon,
Jeff Laytonf5695992010-09-29 15:27:08 -0400343 void **request_buf, void **response_buf)
344{
345 int rc;
346
347 rc = cifs_reconnect_tcon(tcon, smb_command);
348 if (rc)
349 return rc;
350
351 return __smb_init(smb_command, wct, tcon, request_buf, response_buf);
352}
353
354static int
Steve French96daf2b2011-05-27 04:34:02 +0000355smb_init_no_reconnect(int smb_command, int wct, struct cifs_tcon *tcon,
Jeff Laytonf5695992010-09-29 15:27:08 -0400356 void **request_buf, void **response_buf)
357{
358 if (tcon->ses->need_reconnect || tcon->need_reconnect)
359 return -EHOSTDOWN;
360
361 return __smb_init(smb_command, wct, tcon, request_buf, response_buf);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700362}
363
Steve French50c2f752007-07-13 00:33:32 +0000364static int validate_t2(struct smb_t2_rsp *pSMB)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700365{
Jeff Layton12df83c2011-01-20 13:36:51 -0500366 unsigned int total_size;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700367
Jeff Layton12df83c2011-01-20 13:36:51 -0500368 /* check for plausible wct */
369 if (pSMB->hdr.WordCount < 10)
370 goto vt2_err;
371
Linus Torvalds1da177e2005-04-16 15:20:36 -0700372 /* check for parm and data offset going beyond end of smb */
Jeff Layton12df83c2011-01-20 13:36:51 -0500373 if (get_unaligned_le16(&pSMB->t2_rsp.ParameterOffset) > 1024 ||
374 get_unaligned_le16(&pSMB->t2_rsp.DataOffset) > 1024)
375 goto vt2_err;
376
Jeff Layton12df83c2011-01-20 13:36:51 -0500377 total_size = get_unaligned_le16(&pSMB->t2_rsp.ParameterCount);
378 if (total_size >= 512)
379 goto vt2_err;
380
Jeff Laytonfd5707e2011-03-31 17:22:07 -0400381 /* check that bcc is at least as big as parms + data, and that it is
382 * less than negotiated smb buffer
383 */
Jeff Layton12df83c2011-01-20 13:36:51 -0500384 total_size += get_unaligned_le16(&pSMB->t2_rsp.DataCount);
385 if (total_size > get_bcc(&pSMB->hdr) ||
386 total_size >= CIFSMaxBufSize + MAX_CIFS_HDR_SIZE)
387 goto vt2_err;
388
389 return 0;
390vt2_err:
Steve French50c2f752007-07-13 00:33:32 +0000391 cifs_dump_mem("Invalid transact2 SMB: ", (char *)pSMB,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700392 sizeof(struct smb_t2_rsp) + 16);
Jeff Layton12df83c2011-01-20 13:36:51 -0500393 return -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700394}
Jeff Layton690c5222011-01-20 13:36:51 -0500395
Linus Torvalds1da177e2005-04-16 15:20:36 -0700396int
Pavel Shilovsky286170a2012-05-25 10:43:58 +0400397CIFSSMBNegotiate(const unsigned int xid, struct cifs_ses *ses)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700398{
399 NEGOTIATE_REQ *pSMB;
400 NEGOTIATE_RSP *pSMBr;
401 int rc = 0;
402 int bytes_returned;
Steve French39798772006-05-31 22:40:51 +0000403 int i;
Steve French50c2f752007-07-13 00:33:32 +0000404 struct TCP_Server_Info *server;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700405 u16 count;
Steve French750d1152006-06-27 06:28:30 +0000406 unsigned int secFlags;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700407
Steve French790fe572007-07-07 19:25:05 +0000408 if (ses->server)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700409 server = ses->server;
410 else {
411 rc = -EIO;
412 return rc;
413 }
414 rc = smb_init(SMB_COM_NEGOTIATE, 0, NULL /* no tcon yet */ ,
415 (void **) &pSMB, (void **) &pSMBr);
416 if (rc)
417 return rc;
Steve French750d1152006-06-27 06:28:30 +0000418
419 /* if any of auth flags (ie not sign or seal) are overriden use them */
Steve French790fe572007-07-07 19:25:05 +0000420 if (ses->overrideSecFlg & (~(CIFSSEC_MUST_SIGN | CIFSSEC_MUST_SEAL)))
Steve French762e5ab2007-06-28 18:41:42 +0000421 secFlags = ses->overrideSecFlg; /* BB FIXME fix sign flags? */
Steve French750d1152006-06-27 06:28:30 +0000422 else /* if override flags set only sign/seal OR them with global auth */
Jeff Layton04912d62010-04-24 07:57:45 -0400423 secFlags = global_secflags | ses->overrideSecFlg;
Steve French750d1152006-06-27 06:28:30 +0000424
Joe Perchesb6b38f72010-04-21 03:50:45 +0000425 cFYI(1, "secFlags 0x%x", secFlags);
Steve Frenchf40c5622006-06-28 00:13:38 +0000426
Pavel Shilovsky88257362012-05-23 14:01:59 +0400427 pSMB->hdr.Mid = get_next_mid(server);
Yehuda Sadeh Weinraub100c1dd2007-06-05 21:31:16 +0000428 pSMB->hdr.Flags2 |= (SMBFLG2_UNICODE | SMBFLG2_ERR_STATUS);
Steve Frencha0136892007-10-04 20:05:09 +0000429
Yehuda Sadeh Weinraub100c1dd2007-06-05 21:31:16 +0000430 if ((secFlags & CIFSSEC_MUST_KRB5) == CIFSSEC_MUST_KRB5)
Steve French254e55e2006-06-04 05:53:15 +0000431 pSMB->hdr.Flags2 |= SMBFLG2_EXT_SEC;
Steve Frencha0136892007-10-04 20:05:09 +0000432 else if ((secFlags & CIFSSEC_AUTH_MASK) == CIFSSEC_MAY_KRB5) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000433 cFYI(1, "Kerberos only mechanism, enable extended security");
Steve Frencha0136892007-10-04 20:05:09 +0000434 pSMB->hdr.Flags2 |= SMBFLG2_EXT_SEC;
Jeff Laytonb4d6fcf2011-01-07 11:30:28 -0500435 } else if ((secFlags & CIFSSEC_MUST_NTLMSSP) == CIFSSEC_MUST_NTLMSSP)
Steve Frenchac683922009-05-06 04:16:04 +0000436 pSMB->hdr.Flags2 |= SMBFLG2_EXT_SEC;
437 else if ((secFlags & CIFSSEC_AUTH_MASK) == CIFSSEC_MAY_NTLMSSP) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000438 cFYI(1, "NTLMSSP only mechanism, enable extended security");
Steve Frenchac683922009-05-06 04:16:04 +0000439 pSMB->hdr.Flags2 |= SMBFLG2_EXT_SEC;
440 }
Steve French50c2f752007-07-13 00:33:32 +0000441
Steve French39798772006-05-31 22:40:51 +0000442 count = 0;
Steve French50c2f752007-07-13 00:33:32 +0000443 for (i = 0; i < CIFS_NUM_PROT; i++) {
Steve French39798772006-05-31 22:40:51 +0000444 strncpy(pSMB->DialectsArray+count, protocols[i].name, 16);
445 count += strlen(protocols[i].name) + 1;
446 /* null at end of source and target buffers anyway */
447 }
Steve Frenchbe8e3b02011-04-29 05:40:20 +0000448 inc_rfc1001_len(pSMB, count);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700449 pSMB->ByteCount = cpu_to_le16(count);
450
451 rc = SendReceive(xid, ses, (struct smb_hdr *) pSMB,
452 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Steve French50c2f752007-07-13 00:33:32 +0000453 if (rc != 0)
Steve French254e55e2006-06-04 05:53:15 +0000454 goto neg_err_exit;
455
Jeff Layton9bf67e52010-04-24 07:57:46 -0400456 server->dialect = le16_to_cpu(pSMBr->DialectIndex);
457 cFYI(1, "Dialect: %d", server->dialect);
Steve French254e55e2006-06-04 05:53:15 +0000458 /* Check wct = 1 error case */
Jeff Layton9bf67e52010-04-24 07:57:46 -0400459 if ((pSMBr->hdr.WordCount < 13) || (server->dialect == BAD_PROT)) {
Steve French254e55e2006-06-04 05:53:15 +0000460 /* core returns wct = 1, but we do not ask for core - otherwise
Steve French50c2f752007-07-13 00:33:32 +0000461 small wct just comes when dialect index is -1 indicating we
Steve French254e55e2006-06-04 05:53:15 +0000462 could not negotiate a common dialect */
463 rc = -EOPNOTSUPP;
464 goto neg_err_exit;
Steve French50c2f752007-07-13 00:33:32 +0000465#ifdef CONFIG_CIFS_WEAK_PW_HASH
Steve French790fe572007-07-07 19:25:05 +0000466 } else if ((pSMBr->hdr.WordCount == 13)
Jeff Layton9bf67e52010-04-24 07:57:46 -0400467 && ((server->dialect == LANMAN_PROT)
468 || (server->dialect == LANMAN2_PROT))) {
Steve Frenchb815f1e52006-10-02 05:53:29 +0000469 __s16 tmp;
Steve French50c2f752007-07-13 00:33:32 +0000470 struct lanman_neg_rsp *rsp = (struct lanman_neg_rsp *)pSMBr;
Steve French254e55e2006-06-04 05:53:15 +0000471
Steve French790fe572007-07-07 19:25:05 +0000472 if ((secFlags & CIFSSEC_MAY_LANMAN) ||
Steve French750d1152006-06-27 06:28:30 +0000473 (secFlags & CIFSSEC_MAY_PLNTXT))
Steve French254e55e2006-06-04 05:53:15 +0000474 server->secType = LANMAN;
475 else {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000476 cERROR(1, "mount failed weak security disabled"
477 " in /proc/fs/cifs/SecurityFlags");
Steve French39798772006-05-31 22:40:51 +0000478 rc = -EOPNOTSUPP;
479 goto neg_err_exit;
Steve French50c2f752007-07-13 00:33:32 +0000480 }
Pavel Shilovsky286170a2012-05-25 10:43:58 +0400481 server->sec_mode = le16_to_cpu(rsp->SecurityMode);
Pavel Shilovsky10b9b982012-03-20 12:55:09 +0300482 server->maxReq = min_t(unsigned int,
483 le16_to_cpu(rsp->MaxMpxCount),
484 cifs_max_pending);
Pavel Shilovsky45275782012-05-17 17:53:29 +0400485 set_credits(server, server->maxReq);
Jeff Laytonc974bef2011-10-11 06:41:32 -0400486 server->maxBuf = le16_to_cpu(rsp->MaxBufSize);
Steve Frencheca6acf2009-02-20 05:43:09 +0000487 server->max_vcs = le16_to_cpu(rsp->MaxNumberVcs);
Steve French254e55e2006-06-04 05:53:15 +0000488 /* even though we do not use raw we might as well set this
489 accurately, in case we ever find a need for it */
Steve French790fe572007-07-07 19:25:05 +0000490 if ((le16_to_cpu(rsp->RawMode) & RAW_ENABLE) == RAW_ENABLE) {
Steve Frencheca6acf2009-02-20 05:43:09 +0000491 server->max_rw = 0xFF00;
Steve French254e55e2006-06-04 05:53:15 +0000492 server->capabilities = CAP_MPX_MODE | CAP_RAW_MODE;
493 } else {
Steve Frencheca6acf2009-02-20 05:43:09 +0000494 server->max_rw = 0;/* do not need to use raw anyway */
Steve French254e55e2006-06-04 05:53:15 +0000495 server->capabilities = CAP_MPX_MODE;
496 }
Steve Frenchb815f1e52006-10-02 05:53:29 +0000497 tmp = (__s16)le16_to_cpu(rsp->ServerTimeZone);
Steve French1a70d652006-10-02 05:59:18 +0000498 if (tmp == -1) {
Steve French25ee4a92006-09-30 00:54:23 +0000499 /* OS/2 often does not set timezone therefore
500 * we must use server time to calc time zone.
Steve Frenchb815f1e52006-10-02 05:53:29 +0000501 * Could deviate slightly from the right zone.
502 * Smallest defined timezone difference is 15 minutes
503 * (i.e. Nepal). Rounding up/down is done to match
504 * this requirement.
Steve French25ee4a92006-09-30 00:54:23 +0000505 */
Steve Frenchb815f1e52006-10-02 05:53:29 +0000506 int val, seconds, remain, result;
Steve French25ee4a92006-09-30 00:54:23 +0000507 struct timespec ts, utc;
508 utc = CURRENT_TIME;
Jeff Laytonc4a2c082009-05-27 09:37:33 -0400509 ts = cnvrtDosUnixTm(rsp->SrvTime.Date,
510 rsp->SrvTime.Time, 0);
Joe Perchesb6b38f72010-04-21 03:50:45 +0000511 cFYI(1, "SrvTime %d sec since 1970 (utc: %d) diff: %d",
Steve French50c2f752007-07-13 00:33:32 +0000512 (int)ts.tv_sec, (int)utc.tv_sec,
Joe Perchesb6b38f72010-04-21 03:50:45 +0000513 (int)(utc.tv_sec - ts.tv_sec));
Steve Frenchb815f1e52006-10-02 05:53:29 +0000514 val = (int)(utc.tv_sec - ts.tv_sec);
Andre Haupt8594c152007-08-30 20:18:41 +0000515 seconds = abs(val);
Steve French947a5062006-10-02 05:55:25 +0000516 result = (seconds / MIN_TZ_ADJ) * MIN_TZ_ADJ;
Steve Frenchb815f1e52006-10-02 05:53:29 +0000517 remain = seconds % MIN_TZ_ADJ;
Steve French790fe572007-07-07 19:25:05 +0000518 if (remain >= (MIN_TZ_ADJ / 2))
Steve Frenchb815f1e52006-10-02 05:53:29 +0000519 result += MIN_TZ_ADJ;
Steve French790fe572007-07-07 19:25:05 +0000520 if (val < 0)
Steve Frenchad7a2922008-02-07 23:25:02 +0000521 result = -result;
Steve Frenchb815f1e52006-10-02 05:53:29 +0000522 server->timeAdj = result;
Steve French25ee4a92006-09-30 00:54:23 +0000523 } else {
Steve Frenchb815f1e52006-10-02 05:53:29 +0000524 server->timeAdj = (int)tmp;
525 server->timeAdj *= 60; /* also in seconds */
Steve French25ee4a92006-09-30 00:54:23 +0000526 }
Joe Perchesb6b38f72010-04-21 03:50:45 +0000527 cFYI(1, "server->timeAdj: %d seconds", server->timeAdj);
Steve French25ee4a92006-09-30 00:54:23 +0000528
Steve French39798772006-05-31 22:40:51 +0000529
Steve French254e55e2006-06-04 05:53:15 +0000530 /* BB get server time for time conversions and add
Steve French50c2f752007-07-13 00:33:32 +0000531 code to use it and timezone since this is not UTC */
Steve French39798772006-05-31 22:40:51 +0000532
Steve French50c2f752007-07-13 00:33:32 +0000533 if (rsp->EncryptionKeyLength ==
Steve French25ee4a92006-09-30 00:54:23 +0000534 cpu_to_le16(CIFS_CRYPTO_KEY_SIZE)) {
Shirish Pargaonkard3ba50b2010-10-27 15:20:36 -0500535 memcpy(ses->server->cryptkey, rsp->EncryptionKey,
Steve French254e55e2006-06-04 05:53:15 +0000536 CIFS_CRYPTO_KEY_SIZE);
Steve French96daf2b2011-05-27 04:34:02 +0000537 } else if (server->sec_mode & SECMODE_PW_ENCRYPT) {
Steve French254e55e2006-06-04 05:53:15 +0000538 rc = -EIO; /* need cryptkey unless plain text */
539 goto neg_err_exit;
540 }
Steve French39798772006-05-31 22:40:51 +0000541
Steve Frenchf19159d2010-04-21 04:12:10 +0000542 cFYI(1, "LANMAN negotiated");
Steve French254e55e2006-06-04 05:53:15 +0000543 /* we will not end up setting signing flags - as no signing
544 was in LANMAN and server did not return the flags on */
545 goto signing_check;
Steve French7c7b25b2006-06-01 19:20:10 +0000546#else /* weak security disabled */
Steve French790fe572007-07-07 19:25:05 +0000547 } else if (pSMBr->hdr.WordCount == 13) {
Steve Frenchf19159d2010-04-21 04:12:10 +0000548 cERROR(1, "mount failed, cifs module not built "
549 "with CIFS_WEAK_PW_HASH support");
Dan Carpenter8212cf72010-03-15 11:22:26 +0300550 rc = -EOPNOTSUPP;
Steve French7c7b25b2006-06-01 19:20:10 +0000551#endif /* WEAK_PW_HASH */
Steve French254e55e2006-06-04 05:53:15 +0000552 goto neg_err_exit;
Steve French790fe572007-07-07 19:25:05 +0000553 } else if (pSMBr->hdr.WordCount != 17) {
Steve French254e55e2006-06-04 05:53:15 +0000554 /* unknown wct */
555 rc = -EOPNOTSUPP;
556 goto neg_err_exit;
557 }
558 /* else wct == 17 NTLM */
Steve French96daf2b2011-05-27 04:34:02 +0000559 server->sec_mode = pSMBr->SecurityMode;
560 if ((server->sec_mode & SECMODE_USER) == 0)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000561 cFYI(1, "share mode security");
Steve French39798772006-05-31 22:40:51 +0000562
Steve French96daf2b2011-05-27 04:34:02 +0000563 if ((server->sec_mode & SECMODE_PW_ENCRYPT) == 0)
Steve Frenchbdc4bf6e2006-06-02 22:57:13 +0000564#ifdef CONFIG_CIFS_WEAK_PW_HASH
Steve French750d1152006-06-27 06:28:30 +0000565 if ((secFlags & CIFSSEC_MAY_PLNTXT) == 0)
Steve Frenchbdc4bf6e2006-06-02 22:57:13 +0000566#endif /* CIFS_WEAK_PW_HASH */
Joe Perchesb6b38f72010-04-21 03:50:45 +0000567 cERROR(1, "Server requests plain text password"
568 " but client support disabled");
Steve French9312f672006-06-04 22:21:07 +0000569
Steve French790fe572007-07-07 19:25:05 +0000570 if ((secFlags & CIFSSEC_MUST_NTLMV2) == CIFSSEC_MUST_NTLMV2)
Steve French254e55e2006-06-04 05:53:15 +0000571 server->secType = NTLMv2;
Steve French790fe572007-07-07 19:25:05 +0000572 else if (secFlags & CIFSSEC_MAY_NTLM)
Steve French254e55e2006-06-04 05:53:15 +0000573 server->secType = NTLM;
Steve French790fe572007-07-07 19:25:05 +0000574 else if (secFlags & CIFSSEC_MAY_NTLMV2)
Steve Frenchf40c5622006-06-28 00:13:38 +0000575 server->secType = NTLMv2;
Steve Frencha0136892007-10-04 20:05:09 +0000576 else if (secFlags & CIFSSEC_MAY_KRB5)
577 server->secType = Kerberos;
Steve Frenchac683922009-05-06 04:16:04 +0000578 else if (secFlags & CIFSSEC_MAY_NTLMSSP)
Steve Frenchf46c7232009-06-25 03:04:20 +0000579 server->secType = RawNTLMSSP;
Steve Frencha0136892007-10-04 20:05:09 +0000580 else if (secFlags & CIFSSEC_MAY_LANMAN)
581 server->secType = LANMAN;
Steve Frencha0136892007-10-04 20:05:09 +0000582 else {
583 rc = -EOPNOTSUPP;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000584 cERROR(1, "Invalid security type");
Steve Frencha0136892007-10-04 20:05:09 +0000585 goto neg_err_exit;
586 }
587 /* else ... any others ...? */
Steve French7c7b25b2006-06-01 19:20:10 +0000588
Steve French254e55e2006-06-04 05:53:15 +0000589 /* one byte, so no need to convert this or EncryptionKeyLen from
590 little endian */
Pavel Shilovsky10b9b982012-03-20 12:55:09 +0300591 server->maxReq = min_t(unsigned int, le16_to_cpu(pSMBr->MaxMpxCount),
592 cifs_max_pending);
Pavel Shilovsky45275782012-05-17 17:53:29 +0400593 set_credits(server, server->maxReq);
Steve French254e55e2006-06-04 05:53:15 +0000594 /* probably no need to store and check maxvcs */
Jeff Laytonc974bef2011-10-11 06:41:32 -0400595 server->maxBuf = le32_to_cpu(pSMBr->MaxBufferSize);
Steve Frencheca6acf2009-02-20 05:43:09 +0000596 server->max_rw = le32_to_cpu(pSMBr->MaxRawSize);
Joe Perchesb6b38f72010-04-21 03:50:45 +0000597 cFYI(DBG2, "Max buf = %d", ses->server->maxBuf);
Steve French254e55e2006-06-04 05:53:15 +0000598 server->capabilities = le32_to_cpu(pSMBr->Capabilities);
Steve Frenchb815f1e52006-10-02 05:53:29 +0000599 server->timeAdj = (int)(__s16)le16_to_cpu(pSMBr->ServerTimeZone);
600 server->timeAdj *= 60;
Steve French254e55e2006-06-04 05:53:15 +0000601 if (pSMBr->EncryptionKeyLength == CIFS_CRYPTO_KEY_SIZE) {
Shirish Pargaonkard3ba50b2010-10-27 15:20:36 -0500602 memcpy(ses->server->cryptkey, pSMBr->u.EncryptionKey,
Steve French254e55e2006-06-04 05:53:15 +0000603 CIFS_CRYPTO_KEY_SIZE);
Steve French07cc6cf2011-05-27 04:12:29 +0000604 } else if ((pSMBr->hdr.Flags2 & SMBFLG2_EXT_SEC ||
605 server->capabilities & CAP_EXTENDED_SECURITY) &&
606 (pSMBr->EncryptionKeyLength == 0)) {
Steve French254e55e2006-06-04 05:53:15 +0000607 /* decode security blob */
Jeff Layton820a8032011-05-04 08:05:26 -0400608 count = get_bcc(&pSMBr->hdr);
Jeff Laytone187e442007-10-16 17:10:44 +0000609 if (count < 16) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700610 rc = -EIO;
Jeff Laytone187e442007-10-16 17:10:44 +0000611 goto neg_err_exit;
612 }
Suresh Jayaraman3f9bcca2010-10-18 23:29:37 +0530613 spin_lock(&cifs_tcp_ses_lock);
Jeff Laytone7ddee92008-11-14 13:44:38 -0500614 if (server->srv_count > 1) {
Suresh Jayaraman3f9bcca2010-10-18 23:29:37 +0530615 spin_unlock(&cifs_tcp_ses_lock);
Jeff Laytone187e442007-10-16 17:10:44 +0000616 if (memcmp(server->server_GUID,
617 pSMBr->u.extended_response.
618 GUID, 16) != 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000619 cFYI(1, "server UID changed");
Steve French254e55e2006-06-04 05:53:15 +0000620 memcpy(server->server_GUID,
Jeff Laytone187e442007-10-16 17:10:44 +0000621 pSMBr->u.extended_response.GUID,
622 16);
623 }
Jeff Laytone7ddee92008-11-14 13:44:38 -0500624 } else {
Suresh Jayaraman3f9bcca2010-10-18 23:29:37 +0530625 spin_unlock(&cifs_tcp_ses_lock);
Jeff Laytone187e442007-10-16 17:10:44 +0000626 memcpy(server->server_GUID,
627 pSMBr->u.extended_response.GUID, 16);
Jeff Laytone7ddee92008-11-14 13:44:38 -0500628 }
Jeff Laytone187e442007-10-16 17:10:44 +0000629
630 if (count == 16) {
631 server->secType = RawNTLMSSP;
Steve French254e55e2006-06-04 05:53:15 +0000632 } else {
633 rc = decode_negTokenInit(pSMBr->u.extended_response.
Jeff Layton26efa0b2010-04-24 07:57:49 -0400634 SecurityBlob, count - 16,
635 server);
Shirish Pargaonkaref571ca2008-07-24 15:56:05 +0000636 if (rc == 1)
Jeff Laytone5459372007-11-03 05:11:06 +0000637 rc = 0;
Shirish Pargaonkaref571ca2008-07-24 15:56:05 +0000638 else
Steve French254e55e2006-06-04 05:53:15 +0000639 rc = -EINVAL;
Shirish Pargaonkar2b149f12010-09-18 22:02:18 -0500640 if (server->secType == Kerberos) {
641 if (!server->sec_kerberos &&
642 !server->sec_mskerberos)
643 rc = -EOPNOTSUPP;
644 } else if (server->secType == RawNTLMSSP) {
645 if (!server->sec_ntlmssp)
646 rc = -EOPNOTSUPP;
647 } else
648 rc = -EOPNOTSUPP;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700649 }
Steve French96daf2b2011-05-27 04:34:02 +0000650 } else if (server->sec_mode & SECMODE_PW_ENCRYPT) {
Steve French07cc6cf2011-05-27 04:12:29 +0000651 rc = -EIO; /* no crypt key only if plain text pwd */
652 goto neg_err_exit;
Steve French254e55e2006-06-04 05:53:15 +0000653 } else
654 server->capabilities &= ~CAP_EXTENDED_SECURITY;
655
Steve French6344a422006-06-12 04:18:35 +0000656#ifdef CONFIG_CIFS_WEAK_PW_HASH
Steve French254e55e2006-06-04 05:53:15 +0000657signing_check:
Steve French6344a422006-06-12 04:18:35 +0000658#endif
Steve French762e5ab2007-06-28 18:41:42 +0000659 if ((secFlags & CIFSSEC_MAY_SIGN) == 0) {
660 /* MUST_SIGN already includes the MAY_SIGN FLAG
661 so if this is zero it means that signing is disabled */
Joe Perchesb6b38f72010-04-21 03:50:45 +0000662 cFYI(1, "Signing disabled");
Steve French96daf2b2011-05-27 04:34:02 +0000663 if (server->sec_mode & SECMODE_SIGN_REQUIRED) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000664 cERROR(1, "Server requires "
Jeff Layton7111d212007-10-16 16:50:25 +0000665 "packet signing to be enabled in "
Joe Perchesb6b38f72010-04-21 03:50:45 +0000666 "/proc/fs/cifs/SecurityFlags.");
Steve Frenchabb63d62007-10-18 02:58:40 +0000667 rc = -EOPNOTSUPP;
668 }
Steve French96daf2b2011-05-27 04:34:02 +0000669 server->sec_mode &=
Steve French254e55e2006-06-04 05:53:15 +0000670 ~(SECMODE_SIGN_ENABLED | SECMODE_SIGN_REQUIRED);
Steve French762e5ab2007-06-28 18:41:42 +0000671 } else if ((secFlags & CIFSSEC_MUST_SIGN) == CIFSSEC_MUST_SIGN) {
672 /* signing required */
Joe Perchesb6b38f72010-04-21 03:50:45 +0000673 cFYI(1, "Must sign - secFlags 0x%x", secFlags);
Steve French96daf2b2011-05-27 04:34:02 +0000674 if ((server->sec_mode &
Steve French762e5ab2007-06-28 18:41:42 +0000675 (SECMODE_SIGN_ENABLED | SECMODE_SIGN_REQUIRED)) == 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000676 cERROR(1, "signing required but server lacks support");
Jeff38c10a12007-07-06 21:10:07 +0000677 rc = -EOPNOTSUPP;
Steve French762e5ab2007-06-28 18:41:42 +0000678 } else
Steve French96daf2b2011-05-27 04:34:02 +0000679 server->sec_mode |= SECMODE_SIGN_REQUIRED;
Steve French762e5ab2007-06-28 18:41:42 +0000680 } else {
681 /* signing optional ie CIFSSEC_MAY_SIGN */
Steve French96daf2b2011-05-27 04:34:02 +0000682 if ((server->sec_mode & SECMODE_SIGN_REQUIRED) == 0)
683 server->sec_mode &=
Steve French254e55e2006-06-04 05:53:15 +0000684 ~(SECMODE_SIGN_ENABLED | SECMODE_SIGN_REQUIRED);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700685 }
Steve French50c2f752007-07-13 00:33:32 +0000686
687neg_err_exit:
Steve French4a6d87f2005-08-13 08:15:54 -0700688 cifs_buf_release(pSMB);
Steve French254e55e2006-06-04 05:53:15 +0000689
Joe Perchesb6b38f72010-04-21 03:50:45 +0000690 cFYI(1, "negprot rc %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700691 return rc;
692}
693
694int
Pavel Shilovsky2e6e02a2012-05-25 11:11:39 +0400695CIFSSMBTDis(const unsigned int xid, struct cifs_tcon *tcon)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700696{
697 struct smb_hdr *smb_buffer;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700698 int rc = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700699
Joe Perchesb6b38f72010-04-21 03:50:45 +0000700 cFYI(1, "In tree disconnect");
Jeff Laytonf1987b42008-11-15 11:12:47 -0500701
702 /* BB: do we need to check this? These should never be NULL. */
703 if ((tcon->ses == NULL) || (tcon->ses->server == NULL))
704 return -EIO;
705
Linus Torvalds1da177e2005-04-16 15:20:36 -0700706 /*
Jeff Laytonf1987b42008-11-15 11:12:47 -0500707 * No need to return error on this operation if tid invalidated and
708 * closed on server already e.g. due to tcp session crashing. Also,
709 * the tcon is no longer on the list, so no need to take lock before
710 * checking this.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700711 */
Steve French268875b2009-06-25 00:29:21 +0000712 if ((tcon->need_reconnect) || (tcon->ses->need_reconnect))
Steve French50c2f752007-07-13 00:33:32 +0000713 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700714
Steve French50c2f752007-07-13 00:33:32 +0000715 rc = small_smb_init(SMB_COM_TREE_DISCONNECT, 0, tcon,
Steve French09d1db52005-04-28 22:41:08 -0700716 (void **)&smb_buffer);
Jeff Laytonf1987b42008-11-15 11:12:47 -0500717 if (rc)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700718 return rc;
Steve French133672e2007-11-13 22:41:37 +0000719
Pavel Shilovsky792af7b2012-03-23 14:28:02 -0400720 rc = SendReceiveNoRsp(xid, tcon->ses, (char *)smb_buffer, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700721 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000722 cFYI(1, "Tree disconnect failed %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700723
Steve French50c2f752007-07-13 00:33:32 +0000724 /* No need to return error on this operation if tid invalidated and
Jeff Laytonf1987b42008-11-15 11:12:47 -0500725 closed on server already e.g. due to tcp session crashing */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700726 if (rc == -EAGAIN)
727 rc = 0;
728
729 return rc;
730}
731
Jeff Layton766fdbb2011-01-11 07:24:21 -0500732/*
733 * This is a no-op for now. We're not really interested in the reply, but
734 * rather in the fact that the server sent one and that server->lstrp
735 * gets updated.
736 *
737 * FIXME: maybe we should consider checking that the reply matches request?
738 */
739static void
740cifs_echo_callback(struct mid_q_entry *mid)
741{
742 struct TCP_Server_Info *server = mid->callback_data;
743
744 DeleteMidQEntry(mid);
Pavel Shilovskya891f0f2012-05-23 16:14:34 +0400745 add_credits(server, 1, CIFS_ECHO_OP);
Jeff Layton766fdbb2011-01-11 07:24:21 -0500746}
747
748int
749CIFSSMBEcho(struct TCP_Server_Info *server)
750{
751 ECHO_REQ *smb;
752 int rc = 0;
Jeff Laytonfcc31cb2011-05-19 16:22:53 -0400753 struct kvec iov;
Jeff Layton766fdbb2011-01-11 07:24:21 -0500754
755 cFYI(1, "In echo request");
756
757 rc = small_smb_init(SMB_COM_ECHO, 0, NULL, (void **)&smb);
758 if (rc)
759 return rc;
760
761 /* set up echo request */
Steve French5443d132011-03-13 05:08:25 +0000762 smb->hdr.Tid = 0xffff;
Jeff Layton99d86c8f2011-01-20 21:19:25 -0500763 smb->hdr.WordCount = 1;
764 put_unaligned_le16(1, &smb->EchoCount);
Jeff Layton820a8032011-05-04 08:05:26 -0400765 put_bcc(1, &smb->hdr);
Jeff Layton766fdbb2011-01-11 07:24:21 -0500766 smb->Data[0] = 'a';
Steve Frenchbe8e3b02011-04-29 05:40:20 +0000767 inc_rfc1001_len(smb, 3);
Jeff Laytonfcc31cb2011-05-19 16:22:53 -0400768 iov.iov_base = smb;
769 iov.iov_len = be32_to_cpu(smb->hdr.smb_buf_length) + 4;
Jeff Layton766fdbb2011-01-11 07:24:21 -0500770
Jeff Layton44d22d82011-10-19 15:29:49 -0400771 rc = cifs_call_async(server, &iov, 1, NULL, cifs_echo_callback,
Pavel Shilovskya891f0f2012-05-23 16:14:34 +0400772 server, CIFS_ASYNC_OP | CIFS_ECHO_OP);
Jeff Layton766fdbb2011-01-11 07:24:21 -0500773 if (rc)
774 cFYI(1, "Echo request failed: %d", rc);
775
776 cifs_small_buf_release(smb);
777
778 return rc;
779}
780
Linus Torvalds1da177e2005-04-16 15:20:36 -0700781int
Pavel Shilovsky58c45c52012-05-25 10:54:49 +0400782CIFSSMBLogoff(const unsigned int xid, struct cifs_ses *ses)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700783{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700784 LOGOFF_ANDX_REQ *pSMB;
785 int rc = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700786
Joe Perchesb6b38f72010-04-21 03:50:45 +0000787 cFYI(1, "In SMBLogoff for session disconnect");
Jeff Layton14fbf502008-11-14 13:53:46 -0500788
789 /*
790 * BB: do we need to check validity of ses and server? They should
791 * always be valid since we have an active reference. If not, that
792 * should probably be a BUG()
793 */
794 if (!ses || !ses->server)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700795 return -EIO;
796
Steve Frenchd7b619c2010-02-25 05:36:46 +0000797 mutex_lock(&ses->session_mutex);
Steve French3b795212008-11-13 19:45:32 +0000798 if (ses->need_reconnect)
799 goto session_already_dead; /* no need to send SMBlogoff if uid
800 already closed due to reconnect */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700801 rc = small_smb_init(SMB_COM_LOGOFF_ANDX, 2, NULL, (void **)&pSMB);
802 if (rc) {
Steve Frenchd7b619c2010-02-25 05:36:46 +0000803 mutex_unlock(&ses->session_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700804 return rc;
805 }
806
Pavel Shilovsky88257362012-05-23 14:01:59 +0400807 pSMB->hdr.Mid = get_next_mid(ses->server);
Steve French1982c342005-08-17 12:38:22 -0700808
Steve French96daf2b2011-05-27 04:34:02 +0000809 if (ses->server->sec_mode &
Linus Torvalds1da177e2005-04-16 15:20:36 -0700810 (SECMODE_SIGN_REQUIRED | SECMODE_SIGN_ENABLED))
811 pSMB->hdr.Flags2 |= SMBFLG2_SECURITY_SIGNATURE;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700812
813 pSMB->hdr.Uid = ses->Suid;
814
815 pSMB->AndXCommand = 0xFF;
Pavel Shilovsky792af7b2012-03-23 14:28:02 -0400816 rc = SendReceiveNoRsp(xid, ses, (char *) pSMB, 0);
Steve French3b795212008-11-13 19:45:32 +0000817session_already_dead:
Steve Frenchd7b619c2010-02-25 05:36:46 +0000818 mutex_unlock(&ses->session_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700819
820 /* if session dead then we do not need to do ulogoff,
Steve French50c2f752007-07-13 00:33:32 +0000821 since server closed smb session, no sense reporting
Linus Torvalds1da177e2005-04-16 15:20:36 -0700822 error */
823 if (rc == -EAGAIN)
824 rc = 0;
825 return rc;
826}
827
828int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +0400829CIFSPOSIXDelFile(const unsigned int xid, struct cifs_tcon *tcon,
830 const char *fileName, __u16 type,
831 const struct nls_table *nls_codepage, int remap)
Steve French2d785a52007-07-15 01:48:57 +0000832{
833 TRANSACTION2_SPI_REQ *pSMB = NULL;
834 TRANSACTION2_SPI_RSP *pSMBr = NULL;
835 struct unlink_psx_rq *pRqD;
836 int name_len;
837 int rc = 0;
838 int bytes_returned = 0;
839 __u16 params, param_offset, offset, byte_count;
840
Joe Perchesb6b38f72010-04-21 03:50:45 +0000841 cFYI(1, "In POSIX delete");
Steve French2d785a52007-07-15 01:48:57 +0000842PsxDelete:
843 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
844 (void **) &pSMBr);
845 if (rc)
846 return rc;
847
848 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
849 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -0600850 cifsConvertToUTF16((__le16 *) pSMB->FileName, fileName,
851 PATH_MAX, nls_codepage, remap);
Steve French2d785a52007-07-15 01:48:57 +0000852 name_len++; /* trailing null */
853 name_len *= 2;
854 } else { /* BB add path length overrun check */
855 name_len = strnlen(fileName, PATH_MAX);
856 name_len++; /* trailing null */
857 strncpy(pSMB->FileName, fileName, name_len);
858 }
859
860 params = 6 + name_len;
861 pSMB->MaxParameterCount = cpu_to_le16(2);
862 pSMB->MaxDataCount = 0; /* BB double check this with jra */
863 pSMB->MaxSetupCount = 0;
864 pSMB->Reserved = 0;
865 pSMB->Flags = 0;
866 pSMB->Timeout = 0;
867 pSMB->Reserved2 = 0;
868 param_offset = offsetof(struct smb_com_transaction2_spi_req,
869 InformationLevel) - 4;
870 offset = param_offset + params;
871
872 /* Setup pointer to Request Data (inode type) */
873 pRqD = (struct unlink_psx_rq *)(((char *)&pSMB->hdr.Protocol) + offset);
874 pRqD->type = cpu_to_le16(type);
875 pSMB->ParameterOffset = cpu_to_le16(param_offset);
876 pSMB->DataOffset = cpu_to_le16(offset);
877 pSMB->SetupCount = 1;
878 pSMB->Reserved3 = 0;
879 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
880 byte_count = 3 /* pad */ + params + sizeof(struct unlink_psx_rq);
881
882 pSMB->DataCount = cpu_to_le16(sizeof(struct unlink_psx_rq));
883 pSMB->TotalDataCount = cpu_to_le16(sizeof(struct unlink_psx_rq));
884 pSMB->ParameterCount = cpu_to_le16(params);
885 pSMB->TotalParameterCount = pSMB->ParameterCount;
886 pSMB->InformationLevel = cpu_to_le16(SMB_POSIX_UNLINK);
887 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +0000888 inc_rfc1001_len(pSMB, byte_count);
Steve French2d785a52007-07-15 01:48:57 +0000889 pSMB->ByteCount = cpu_to_le16(byte_count);
890 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
891 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Steve Frenchad7a2922008-02-07 23:25:02 +0000892 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000893 cFYI(1, "Posix delete returned %d", rc);
Steve French2d785a52007-07-15 01:48:57 +0000894 cifs_buf_release(pSMB);
895
Pavel Shilovsky44c58182012-05-28 14:16:31 +0400896 cifs_stats_inc(&tcon->stats.cifs_stats.num_deletes);
Steve French2d785a52007-07-15 01:48:57 +0000897
898 if (rc == -EAGAIN)
899 goto PsxDelete;
900
901 return rc;
902}
903
904int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +0400905CIFSSMBDelFile(const unsigned int xid, struct cifs_tcon *tcon,
906 const char *fileName, const struct nls_table *nls_codepage,
907 int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700908{
909 DELETE_FILE_REQ *pSMB = NULL;
910 DELETE_FILE_RSP *pSMBr = NULL;
911 int rc = 0;
912 int bytes_returned;
913 int name_len;
914
915DelFileRetry:
916 rc = smb_init(SMB_COM_DELETE, 1, tcon, (void **) &pSMB,
917 (void **) &pSMBr);
918 if (rc)
919 return rc;
920
921 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
922 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -0600923 cifsConvertToUTF16((__le16 *) pSMB->fileName, fileName,
924 PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700925 name_len++; /* trailing null */
926 name_len *= 2;
Steve French09d1db52005-04-28 22:41:08 -0700927 } else { /* BB improve check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700928 name_len = strnlen(fileName, PATH_MAX);
929 name_len++; /* trailing null */
930 strncpy(pSMB->fileName, fileName, name_len);
931 }
932 pSMB->SearchAttributes =
933 cpu_to_le16(ATTR_READONLY | ATTR_HIDDEN | ATTR_SYSTEM);
934 pSMB->BufferFormat = 0x04;
Steve Frenchbe8e3b02011-04-29 05:40:20 +0000935 inc_rfc1001_len(pSMB, name_len + 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700936 pSMB->ByteCount = cpu_to_le16(name_len + 1);
937 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
938 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +0400939 cifs_stats_inc(&tcon->stats.cifs_stats.num_deletes);
Steve Frenchad7a2922008-02-07 23:25:02 +0000940 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000941 cFYI(1, "Error in RMFile = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700942
943 cifs_buf_release(pSMB);
944 if (rc == -EAGAIN)
945 goto DelFileRetry;
946
947 return rc;
948}
949
950int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +0400951CIFSSMBRmDir(const unsigned int xid, struct cifs_tcon *tcon,
952 const char *dirName, const struct nls_table *nls_codepage,
953 int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700954{
955 DELETE_DIRECTORY_REQ *pSMB = NULL;
956 DELETE_DIRECTORY_RSP *pSMBr = NULL;
957 int rc = 0;
958 int bytes_returned;
959 int name_len;
960
Joe Perchesb6b38f72010-04-21 03:50:45 +0000961 cFYI(1, "In CIFSSMBRmDir");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700962RmDirRetry:
963 rc = smb_init(SMB_COM_DELETE_DIRECTORY, 0, tcon, (void **) &pSMB,
964 (void **) &pSMBr);
965 if (rc)
966 return rc;
967
968 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
Steve Frenchacbbb762012-01-18 22:32:33 -0600969 name_len = cifsConvertToUTF16((__le16 *) pSMB->DirName, dirName,
970 PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700971 name_len++; /* trailing null */
972 name_len *= 2;
Steve French09d1db52005-04-28 22:41:08 -0700973 } else { /* BB improve check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700974 name_len = strnlen(dirName, PATH_MAX);
975 name_len++; /* trailing null */
976 strncpy(pSMB->DirName, dirName, name_len);
977 }
978
979 pSMB->BufferFormat = 0x04;
Steve Frenchbe8e3b02011-04-29 05:40:20 +0000980 inc_rfc1001_len(pSMB, name_len + 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700981 pSMB->ByteCount = cpu_to_le16(name_len + 1);
982 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
983 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +0400984 cifs_stats_inc(&tcon->stats.cifs_stats.num_rmdirs);
Steve Frenchad7a2922008-02-07 23:25:02 +0000985 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000986 cFYI(1, "Error in RMDir = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700987
988 cifs_buf_release(pSMB);
989 if (rc == -EAGAIN)
990 goto RmDirRetry;
991 return rc;
992}
993
994int
Pavel Shilovskyf4367202012-03-17 11:41:12 +0300995CIFSSMBMkDir(const unsigned int xid, struct cifs_tcon *tcon, const char *name,
996 struct cifs_sb_info *cifs_sb)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700997{
998 int rc = 0;
999 CREATE_DIRECTORY_REQ *pSMB = NULL;
1000 CREATE_DIRECTORY_RSP *pSMBr = NULL;
1001 int bytes_returned;
1002 int name_len;
Pavel Shilovskyf4367202012-03-17 11:41:12 +03001003 int remap = cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001004
Joe Perchesb6b38f72010-04-21 03:50:45 +00001005 cFYI(1, "In CIFSSMBMkDir");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001006MkDirRetry:
1007 rc = smb_init(SMB_COM_CREATE_DIRECTORY, 0, tcon, (void **) &pSMB,
1008 (void **) &pSMBr);
1009 if (rc)
1010 return rc;
1011
1012 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
Steve Frenchacbbb762012-01-18 22:32:33 -06001013 name_len = cifsConvertToUTF16((__le16 *) pSMB->DirName, name,
Pavel Shilovskyf4367202012-03-17 11:41:12 +03001014 PATH_MAX, cifs_sb->local_nls,
1015 remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001016 name_len++; /* trailing null */
1017 name_len *= 2;
Steve French09d1db52005-04-28 22:41:08 -07001018 } else { /* BB improve check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001019 name_len = strnlen(name, PATH_MAX);
1020 name_len++; /* trailing null */
1021 strncpy(pSMB->DirName, name, name_len);
1022 }
1023
1024 pSMB->BufferFormat = 0x04;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00001025 inc_rfc1001_len(pSMB, name_len + 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001026 pSMB->ByteCount = cpu_to_le16(name_len + 1);
1027 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
1028 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04001029 cifs_stats_inc(&tcon->stats.cifs_stats.num_mkdirs);
Steve Frenchad7a2922008-02-07 23:25:02 +00001030 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00001031 cFYI(1, "Error in Mkdir = %d", rc);
Steve Frencha5a2b482005-08-20 21:42:53 -07001032
Linus Torvalds1da177e2005-04-16 15:20:36 -07001033 cifs_buf_release(pSMB);
1034 if (rc == -EAGAIN)
1035 goto MkDirRetry;
1036 return rc;
1037}
1038
Steve French2dd29d32007-04-23 22:07:35 +00001039int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04001040CIFSPOSIXCreate(const unsigned int xid, struct cifs_tcon *tcon,
1041 __u32 posix_flags, __u64 mode, __u16 *netfid,
1042 FILE_UNIX_BASIC_INFO *pRetData, __u32 *pOplock,
1043 const char *name, const struct nls_table *nls_codepage,
1044 int remap)
Steve French2dd29d32007-04-23 22:07:35 +00001045{
1046 TRANSACTION2_SPI_REQ *pSMB = NULL;
1047 TRANSACTION2_SPI_RSP *pSMBr = NULL;
1048 int name_len;
1049 int rc = 0;
1050 int bytes_returned = 0;
Steve French2dd29d32007-04-23 22:07:35 +00001051 __u16 params, param_offset, offset, byte_count, count;
Steve Frenchad7a2922008-02-07 23:25:02 +00001052 OPEN_PSX_REQ *pdata;
1053 OPEN_PSX_RSP *psx_rsp;
Steve French2dd29d32007-04-23 22:07:35 +00001054
Joe Perchesb6b38f72010-04-21 03:50:45 +00001055 cFYI(1, "In POSIX Create");
Steve French2dd29d32007-04-23 22:07:35 +00001056PsxCreat:
1057 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
1058 (void **) &pSMBr);
1059 if (rc)
1060 return rc;
1061
1062 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
1063 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06001064 cifsConvertToUTF16((__le16 *) pSMB->FileName, name,
1065 PATH_MAX, nls_codepage, remap);
Steve French2dd29d32007-04-23 22:07:35 +00001066 name_len++; /* trailing null */
1067 name_len *= 2;
1068 } else { /* BB improve the check for buffer overruns BB */
1069 name_len = strnlen(name, PATH_MAX);
1070 name_len++; /* trailing null */
1071 strncpy(pSMB->FileName, name, name_len);
1072 }
1073
1074 params = 6 + name_len;
1075 count = sizeof(OPEN_PSX_REQ);
1076 pSMB->MaxParameterCount = cpu_to_le16(2);
1077 pSMB->MaxDataCount = cpu_to_le16(1000); /* large enough */
1078 pSMB->MaxSetupCount = 0;
1079 pSMB->Reserved = 0;
1080 pSMB->Flags = 0;
1081 pSMB->Timeout = 0;
1082 pSMB->Reserved2 = 0;
1083 param_offset = offsetof(struct smb_com_transaction2_spi_req,
Steve French50c2f752007-07-13 00:33:32 +00001084 InformationLevel) - 4;
Steve French2dd29d32007-04-23 22:07:35 +00001085 offset = param_offset + params;
Steve French2dd29d32007-04-23 22:07:35 +00001086 pdata = (OPEN_PSX_REQ *)(((char *)&pSMB->hdr.Protocol) + offset);
Cyril Gorcunov8f2376a2007-10-14 17:58:43 +00001087 pdata->Level = cpu_to_le16(SMB_QUERY_FILE_UNIX_BASIC);
Steve French2dd29d32007-04-23 22:07:35 +00001088 pdata->Permissions = cpu_to_le64(mode);
Steve French50c2f752007-07-13 00:33:32 +00001089 pdata->PosixOpenFlags = cpu_to_le32(posix_flags);
Steve French2dd29d32007-04-23 22:07:35 +00001090 pdata->OpenFlags = cpu_to_le32(*pOplock);
1091 pSMB->ParameterOffset = cpu_to_le16(param_offset);
1092 pSMB->DataOffset = cpu_to_le16(offset);
1093 pSMB->SetupCount = 1;
1094 pSMB->Reserved3 = 0;
1095 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
1096 byte_count = 3 /* pad */ + params + count;
1097
1098 pSMB->DataCount = cpu_to_le16(count);
1099 pSMB->ParameterCount = cpu_to_le16(params);
1100 pSMB->TotalDataCount = pSMB->DataCount;
1101 pSMB->TotalParameterCount = pSMB->ParameterCount;
1102 pSMB->InformationLevel = cpu_to_le16(SMB_POSIX_OPEN);
1103 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00001104 inc_rfc1001_len(pSMB, byte_count);
Steve French2dd29d32007-04-23 22:07:35 +00001105 pSMB->ByteCount = cpu_to_le16(byte_count);
1106 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
1107 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
1108 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001109 cFYI(1, "Posix create returned %d", rc);
Steve French2dd29d32007-04-23 22:07:35 +00001110 goto psx_create_err;
1111 }
1112
Joe Perchesb6b38f72010-04-21 03:50:45 +00001113 cFYI(1, "copying inode info");
Steve French2dd29d32007-04-23 22:07:35 +00001114 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
1115
Jeff Layton820a8032011-05-04 08:05:26 -04001116 if (rc || get_bcc(&pSMBr->hdr) < sizeof(OPEN_PSX_RSP)) {
Steve French2dd29d32007-04-23 22:07:35 +00001117 rc = -EIO; /* bad smb */
1118 goto psx_create_err;
1119 }
1120
1121 /* copy return information to pRetData */
Steve French50c2f752007-07-13 00:33:32 +00001122 psx_rsp = (OPEN_PSX_RSP *)((char *) &pSMBr->hdr.Protocol
Steve French2dd29d32007-04-23 22:07:35 +00001123 + le16_to_cpu(pSMBr->t2.DataOffset));
Steve French50c2f752007-07-13 00:33:32 +00001124
Steve French2dd29d32007-04-23 22:07:35 +00001125 *pOplock = le16_to_cpu(psx_rsp->OplockFlags);
Steve French790fe572007-07-07 19:25:05 +00001126 if (netfid)
Steve French2dd29d32007-04-23 22:07:35 +00001127 *netfid = psx_rsp->Fid; /* cifs fid stays in le */
1128 /* Let caller know file was created so we can set the mode. */
1129 /* Do we care about the CreateAction in any other cases? */
Steve French790fe572007-07-07 19:25:05 +00001130 if (cpu_to_le32(FILE_CREATE) == psx_rsp->CreateAction)
Steve French2dd29d32007-04-23 22:07:35 +00001131 *pOplock |= CIFS_CREATE_ACTION;
1132 /* check to make sure response data is there */
Cyril Gorcunov8f2376a2007-10-14 17:58:43 +00001133 if (psx_rsp->ReturnedLevel != cpu_to_le16(SMB_QUERY_FILE_UNIX_BASIC)) {
1134 pRetData->Type = cpu_to_le32(-1); /* unknown */
Joe Perchesb6b38f72010-04-21 03:50:45 +00001135 cFYI(DBG2, "unknown type");
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001136 } else {
Jeff Layton820a8032011-05-04 08:05:26 -04001137 if (get_bcc(&pSMBr->hdr) < sizeof(OPEN_PSX_RSP)
Steve French2dd29d32007-04-23 22:07:35 +00001138 + sizeof(FILE_UNIX_BASIC_INFO)) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001139 cERROR(1, "Open response data too small");
Cyril Gorcunov8f2376a2007-10-14 17:58:43 +00001140 pRetData->Type = cpu_to_le32(-1);
Steve French2dd29d32007-04-23 22:07:35 +00001141 goto psx_create_err;
1142 }
Steve French50c2f752007-07-13 00:33:32 +00001143 memcpy((char *) pRetData,
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001144 (char *)psx_rsp + sizeof(OPEN_PSX_RSP),
Steve French26f57362007-08-30 22:09:15 +00001145 sizeof(FILE_UNIX_BASIC_INFO));
Steve French2dd29d32007-04-23 22:07:35 +00001146 }
Steve French2dd29d32007-04-23 22:07:35 +00001147
1148psx_create_err:
1149 cifs_buf_release(pSMB);
1150
Steve French65bc98b2009-07-10 15:27:25 +00001151 if (posix_flags & SMB_O_DIRECTORY)
Pavel Shilovsky44c58182012-05-28 14:16:31 +04001152 cifs_stats_inc(&tcon->stats.cifs_stats.num_posixmkdirs);
Steve French65bc98b2009-07-10 15:27:25 +00001153 else
Pavel Shilovsky44c58182012-05-28 14:16:31 +04001154 cifs_stats_inc(&tcon->stats.cifs_stats.num_posixopens);
Steve French2dd29d32007-04-23 22:07:35 +00001155
1156 if (rc == -EAGAIN)
1157 goto PsxCreat;
1158
Steve French50c2f752007-07-13 00:33:32 +00001159 return rc;
Steve French2dd29d32007-04-23 22:07:35 +00001160}
1161
Steve Frencha9d02ad2005-08-24 23:06:05 -07001162static __u16 convert_disposition(int disposition)
1163{
1164 __u16 ofun = 0;
1165
1166 switch (disposition) {
1167 case FILE_SUPERSEDE:
1168 ofun = SMBOPEN_OCREATE | SMBOPEN_OTRUNC;
1169 break;
1170 case FILE_OPEN:
1171 ofun = SMBOPEN_OAPPEND;
1172 break;
1173 case FILE_CREATE:
1174 ofun = SMBOPEN_OCREATE;
1175 break;
1176 case FILE_OPEN_IF:
1177 ofun = SMBOPEN_OCREATE | SMBOPEN_OAPPEND;
1178 break;
1179 case FILE_OVERWRITE:
1180 ofun = SMBOPEN_OTRUNC;
1181 break;
1182 case FILE_OVERWRITE_IF:
1183 ofun = SMBOPEN_OCREATE | SMBOPEN_OTRUNC;
1184 break;
1185 default:
Joe Perchesb6b38f72010-04-21 03:50:45 +00001186 cFYI(1, "unknown disposition %d", disposition);
Steve Frencha9d02ad2005-08-24 23:06:05 -07001187 ofun = SMBOPEN_OAPPEND; /* regular open */
1188 }
1189 return ofun;
1190}
1191
Jeff Layton35fc37d2008-05-14 10:22:03 -07001192static int
1193access_flags_to_smbopen_mode(const int access_flags)
1194{
1195 int masked_flags = access_flags & (GENERIC_READ | GENERIC_WRITE);
1196
1197 if (masked_flags == GENERIC_READ)
1198 return SMBOPEN_READ;
1199 else if (masked_flags == GENERIC_WRITE)
1200 return SMBOPEN_WRITE;
1201
1202 /* just go for read/write */
1203 return SMBOPEN_READWRITE;
1204}
1205
Steve Frencha9d02ad2005-08-24 23:06:05 -07001206int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04001207SMBLegacyOpen(const unsigned int xid, struct cifs_tcon *tcon,
Steve Frencha9d02ad2005-08-24 23:06:05 -07001208 const char *fileName, const int openDisposition,
Steve Frenchad7a2922008-02-07 23:25:02 +00001209 const int access_flags, const int create_options, __u16 *netfid,
1210 int *pOplock, FILE_ALL_INFO *pfile_info,
Steve Frencha9d02ad2005-08-24 23:06:05 -07001211 const struct nls_table *nls_codepage, int remap)
1212{
1213 int rc = -EACCES;
1214 OPENX_REQ *pSMB = NULL;
1215 OPENX_RSP *pSMBr = NULL;
1216 int bytes_returned;
1217 int name_len;
1218 __u16 count;
1219
1220OldOpenRetry:
1221 rc = smb_init(SMB_COM_OPEN_ANDX, 15, tcon, (void **) &pSMB,
1222 (void **) &pSMBr);
1223 if (rc)
1224 return rc;
1225
1226 pSMB->AndXCommand = 0xFF; /* none */
1227
1228 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
1229 count = 1; /* account for one byte pad to word boundary */
1230 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06001231 cifsConvertToUTF16((__le16 *) (pSMB->fileName + 1),
1232 fileName, PATH_MAX, nls_codepage, remap);
Steve Frencha9d02ad2005-08-24 23:06:05 -07001233 name_len++; /* trailing null */
1234 name_len *= 2;
1235 } else { /* BB improve check for buffer overruns BB */
1236 count = 0; /* no pad */
1237 name_len = strnlen(fileName, PATH_MAX);
1238 name_len++; /* trailing null */
1239 strncpy(pSMB->fileName, fileName, name_len);
1240 }
1241 if (*pOplock & REQ_OPLOCK)
1242 pSMB->OpenFlags = cpu_to_le16(REQ_OPLOCK);
Steve French26f57362007-08-30 22:09:15 +00001243 else if (*pOplock & REQ_BATCHOPLOCK)
Steve Frencha9d02ad2005-08-24 23:06:05 -07001244 pSMB->OpenFlags = cpu_to_le16(REQ_BATCHOPLOCK);
Steve French26f57362007-08-30 22:09:15 +00001245
Steve Frencha9d02ad2005-08-24 23:06:05 -07001246 pSMB->OpenFlags |= cpu_to_le16(REQ_MORE_INFO);
Jeff Layton35fc37d2008-05-14 10:22:03 -07001247 pSMB->Mode = cpu_to_le16(access_flags_to_smbopen_mode(access_flags));
Steve Frencha9d02ad2005-08-24 23:06:05 -07001248 pSMB->Mode |= cpu_to_le16(0x40); /* deny none */
1249 /* set file as system file if special file such
1250 as fifo and server expecting SFU style and
1251 no Unix extensions */
1252
Steve French790fe572007-07-07 19:25:05 +00001253 if (create_options & CREATE_OPTION_SPECIAL)
1254 pSMB->FileAttributes = cpu_to_le16(ATTR_SYSTEM);
Steve Frenchad7a2922008-02-07 23:25:02 +00001255 else /* BB FIXME BB */
1256 pSMB->FileAttributes = cpu_to_le16(0/*ATTR_NORMAL*/);
Steve Frencha9d02ad2005-08-24 23:06:05 -07001257
Jeff Layton67750fb2008-05-09 22:28:02 +00001258 if (create_options & CREATE_OPTION_READONLY)
1259 pSMB->FileAttributes |= cpu_to_le16(ATTR_READONLY);
Steve Frencha9d02ad2005-08-24 23:06:05 -07001260
1261 /* BB FIXME BB */
Steve French50c2f752007-07-13 00:33:32 +00001262/* pSMB->CreateOptions = cpu_to_le32(create_options &
1263 CREATE_OPTIONS_MASK); */
Steve Frencha9d02ad2005-08-24 23:06:05 -07001264 /* BB FIXME END BB */
Steve French3e87d802005-09-18 20:49:21 -07001265
1266 pSMB->Sattr = cpu_to_le16(ATTR_HIDDEN | ATTR_SYSTEM | ATTR_DIRECTORY);
Steve French70ca7342005-09-22 16:32:06 -07001267 pSMB->OpenFunction = cpu_to_le16(convert_disposition(openDisposition));
Steve Frencha9d02ad2005-08-24 23:06:05 -07001268 count += name_len;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00001269 inc_rfc1001_len(pSMB, count);
Steve Frencha9d02ad2005-08-24 23:06:05 -07001270
1271 pSMB->ByteCount = cpu_to_le16(count);
1272 /* long_op set to 1 to allow for oplock break timeouts */
1273 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
Jeff Layton77499812011-01-11 07:24:23 -05001274 (struct smb_hdr *)pSMBr, &bytes_returned, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04001275 cifs_stats_inc(&tcon->stats.cifs_stats.num_opens);
Steve Frencha9d02ad2005-08-24 23:06:05 -07001276 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001277 cFYI(1, "Error in Open = %d", rc);
Steve Frencha9d02ad2005-08-24 23:06:05 -07001278 } else {
1279 /* BB verify if wct == 15 */
1280
Steve French582d21e2008-05-13 04:54:12 +00001281/* *pOplock = pSMBr->OplockLevel; */ /* BB take from action field*/
Steve Frencha9d02ad2005-08-24 23:06:05 -07001282
1283 *netfid = pSMBr->Fid; /* cifs fid stays in le */
1284 /* Let caller know file was created so we can set the mode. */
1285 /* Do we care about the CreateAction in any other cases? */
1286 /* BB FIXME BB */
Steve French790fe572007-07-07 19:25:05 +00001287/* if (cpu_to_le32(FILE_CREATE) == pSMBr->CreateAction)
Steve Frencha9d02ad2005-08-24 23:06:05 -07001288 *pOplock |= CIFS_CREATE_ACTION; */
1289 /* BB FIXME END */
1290
Steve French790fe572007-07-07 19:25:05 +00001291 if (pfile_info) {
Steve Frencha9d02ad2005-08-24 23:06:05 -07001292 pfile_info->CreationTime = 0; /* BB convert CreateTime*/
1293 pfile_info->LastAccessTime = 0; /* BB fixme */
1294 pfile_info->LastWriteTime = 0; /* BB fixme */
1295 pfile_info->ChangeTime = 0; /* BB fixme */
Steve French70ca7342005-09-22 16:32:06 -07001296 pfile_info->Attributes =
Steve French50c2f752007-07-13 00:33:32 +00001297 cpu_to_le32(le16_to_cpu(pSMBr->FileAttributes));
Steve Frencha9d02ad2005-08-24 23:06:05 -07001298 /* the file_info buf is endian converted by caller */
Steve French70ca7342005-09-22 16:32:06 -07001299 pfile_info->AllocationSize =
1300 cpu_to_le64(le32_to_cpu(pSMBr->EndOfFile));
1301 pfile_info->EndOfFile = pfile_info->AllocationSize;
Steve Frencha9d02ad2005-08-24 23:06:05 -07001302 pfile_info->NumberOfLinks = cpu_to_le32(1);
Jeff Layton9a8165f2008-10-17 21:03:20 -04001303 pfile_info->DeletePending = 0;
Steve Frencha9d02ad2005-08-24 23:06:05 -07001304 }
1305 }
1306
1307 cifs_buf_release(pSMB);
1308 if (rc == -EAGAIN)
1309 goto OldOpenRetry;
1310 return rc;
1311}
1312
Linus Torvalds1da177e2005-04-16 15:20:36 -07001313int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04001314CIFSSMBOpen(const unsigned int xid, struct cifs_tcon *tcon,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001315 const char *fileName, const int openDisposition,
Steve Frenchad7a2922008-02-07 23:25:02 +00001316 const int access_flags, const int create_options, __u16 *netfid,
1317 int *pOplock, FILE_ALL_INFO *pfile_info,
Steve French737b7582005-04-28 22:41:06 -07001318 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001319{
1320 int rc = -EACCES;
1321 OPEN_REQ *pSMB = NULL;
1322 OPEN_RSP *pSMBr = NULL;
1323 int bytes_returned;
1324 int name_len;
1325 __u16 count;
1326
1327openRetry:
1328 rc = smb_init(SMB_COM_NT_CREATE_ANDX, 24, tcon, (void **) &pSMB,
1329 (void **) &pSMBr);
1330 if (rc)
1331 return rc;
1332
1333 pSMB->AndXCommand = 0xFF; /* none */
1334
1335 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
1336 count = 1; /* account for one byte pad to word boundary */
1337 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06001338 cifsConvertToUTF16((__le16 *) (pSMB->fileName + 1),
1339 fileName, PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001340 name_len++; /* trailing null */
1341 name_len *= 2;
1342 pSMB->NameLength = cpu_to_le16(name_len);
Steve French09d1db52005-04-28 22:41:08 -07001343 } else { /* BB improve check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001344 count = 0; /* no pad */
1345 name_len = strnlen(fileName, PATH_MAX);
1346 name_len++; /* trailing null */
1347 pSMB->NameLength = cpu_to_le16(name_len);
1348 strncpy(pSMB->fileName, fileName, name_len);
1349 }
1350 if (*pOplock & REQ_OPLOCK)
1351 pSMB->OpenFlags = cpu_to_le32(REQ_OPLOCK);
Steve French26f57362007-08-30 22:09:15 +00001352 else if (*pOplock & REQ_BATCHOPLOCK)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001353 pSMB->OpenFlags = cpu_to_le32(REQ_BATCHOPLOCK);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001354 pSMB->DesiredAccess = cpu_to_le32(access_flags);
1355 pSMB->AllocationSize = 0;
Steve Frencheda3c0292005-07-21 15:20:28 -07001356 /* set file as system file if special file such
1357 as fifo and server expecting SFU style and
1358 no Unix extensions */
Steve French790fe572007-07-07 19:25:05 +00001359 if (create_options & CREATE_OPTION_SPECIAL)
Steve Frencheda3c0292005-07-21 15:20:28 -07001360 pSMB->FileAttributes = cpu_to_le32(ATTR_SYSTEM);
1361 else
1362 pSMB->FileAttributes = cpu_to_le32(ATTR_NORMAL);
Jeff Layton67750fb2008-05-09 22:28:02 +00001363
Linus Torvalds1da177e2005-04-16 15:20:36 -07001364 /* XP does not handle ATTR_POSIX_SEMANTICS */
1365 /* but it helps speed up case sensitive checks for other
1366 servers such as Samba */
1367 if (tcon->ses->capabilities & CAP_UNIX)
1368 pSMB->FileAttributes |= cpu_to_le32(ATTR_POSIX_SEMANTICS);
1369
Jeff Layton67750fb2008-05-09 22:28:02 +00001370 if (create_options & CREATE_OPTION_READONLY)
1371 pSMB->FileAttributes |= cpu_to_le32(ATTR_READONLY);
1372
Linus Torvalds1da177e2005-04-16 15:20:36 -07001373 pSMB->ShareAccess = cpu_to_le32(FILE_SHARE_ALL);
1374 pSMB->CreateDisposition = cpu_to_le32(openDisposition);
Steve Frencheda3c0292005-07-21 15:20:28 -07001375 pSMB->CreateOptions = cpu_to_le32(create_options & CREATE_OPTIONS_MASK);
Steve French09d1db52005-04-28 22:41:08 -07001376 /* BB Expirement with various impersonation levels and verify */
1377 pSMB->ImpersonationLevel = cpu_to_le32(SECURITY_IMPERSONATION);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001378 pSMB->SecurityFlags =
1379 SECURITY_CONTEXT_TRACKING | SECURITY_EFFECTIVE_ONLY;
1380
1381 count += name_len;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00001382 inc_rfc1001_len(pSMB, count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001383
1384 pSMB->ByteCount = cpu_to_le16(count);
1385 /* long_op set to 1 to allow for oplock break timeouts */
1386 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
Jeff Layton77499812011-01-11 07:24:23 -05001387 (struct smb_hdr *)pSMBr, &bytes_returned, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04001388 cifs_stats_inc(&tcon->stats.cifs_stats.num_opens);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001389 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001390 cFYI(1, "Error in Open = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001391 } else {
Steve French09d1db52005-04-28 22:41:08 -07001392 *pOplock = pSMBr->OplockLevel; /* 1 byte no need to le_to_cpu */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001393 *netfid = pSMBr->Fid; /* cifs fid stays in le */
1394 /* Let caller know file was created so we can set the mode. */
1395 /* Do we care about the CreateAction in any other cases? */
Steve French790fe572007-07-07 19:25:05 +00001396 if (cpu_to_le32(FILE_CREATE) == pSMBr->CreateAction)
Steve French50c2f752007-07-13 00:33:32 +00001397 *pOplock |= CIFS_CREATE_ACTION;
Steve French790fe572007-07-07 19:25:05 +00001398 if (pfile_info) {
Steve French61e74802008-12-03 00:57:54 +00001399 memcpy((char *)pfile_info, (char *)&pSMBr->CreationTime,
1400 36 /* CreationTime to Attributes */);
1401 /* the file_info buf is endian converted by caller */
1402 pfile_info->AllocationSize = pSMBr->AllocationSize;
1403 pfile_info->EndOfFile = pSMBr->EndOfFile;
1404 pfile_info->NumberOfLinks = cpu_to_le32(1);
1405 pfile_info->DeletePending = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001406 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001407 }
Steve Frencha5a2b482005-08-20 21:42:53 -07001408
Linus Torvalds1da177e2005-04-16 15:20:36 -07001409 cifs_buf_release(pSMB);
1410 if (rc == -EAGAIN)
1411 goto openRetry;
1412 return rc;
1413}
1414
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001415/*
1416 * Discard any remaining data in the current SMB. To do this, we borrow the
1417 * current bigbuf.
1418 */
1419static int
1420cifs_readv_discard(struct TCP_Server_Info *server, struct mid_q_entry *mid)
1421{
Pavel Shilovsky5ffef7b2012-03-23 14:28:03 -04001422 unsigned int rfclen = get_rfc1002_length(server->smallbuf);
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001423 int remaining = rfclen + 4 - server->total_read;
1424 struct cifs_readdata *rdata = mid->callback_data;
1425
1426 while (remaining > 0) {
1427 int length;
1428
1429 length = cifs_read_from_socket(server, server->bigbuf,
1430 min_t(unsigned int, remaining,
Pavel Shilovsky1887f602012-05-17 12:45:31 +04001431 CIFSMaxBufSize + MAX_HEADER_SIZE(server)));
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001432 if (length < 0)
1433 return length;
1434 server->total_read += length;
1435 remaining -= length;
1436 }
1437
1438 dequeue_mid(mid, rdata->result);
1439 return 0;
1440}
1441
1442static int
1443cifs_readv_receive(struct TCP_Server_Info *server, struct mid_q_entry *mid)
1444{
1445 int length, len;
Jeff Layton8d5ce4d2012-05-16 07:13:16 -04001446 unsigned int data_offset, data_len;
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001447 struct cifs_readdata *rdata = mid->callback_data;
Pavel Shilovsky5ffef7b2012-03-23 14:28:03 -04001448 char *buf = server->smallbuf;
1449 unsigned int buflen = get_rfc1002_length(buf) + 4;
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001450
Pavel Shilovsky7c9421e2012-03-23 14:28:03 -04001451 cFYI(1, "%s: mid=%llu offset=%llu bytes=%u", __func__,
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001452 mid->mid, rdata->offset, rdata->bytes);
1453
1454 /*
1455 * read the rest of READ_RSP header (sans Data array), or whatever we
1456 * can if there's not enough data. At this point, we've read down to
1457 * the Mid.
1458 */
Pavel Shilovskyeb378712012-05-17 13:02:51 +04001459 len = min_t(unsigned int, buflen, server->vals->read_rsp_size) -
Pavel Shilovsky1887f602012-05-17 12:45:31 +04001460 HEADER_SIZE(server) + 1;
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001461
Pavel Shilovsky1887f602012-05-17 12:45:31 +04001462 rdata->iov[0].iov_base = buf + HEADER_SIZE(server) - 1;
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001463 rdata->iov[0].iov_len = len;
1464
1465 length = cifs_readv_from_socket(server, rdata->iov, 1, len);
1466 if (length < 0)
1467 return length;
1468 server->total_read += length;
1469
1470 /* Was the SMB read successful? */
Pavel Shilovskyeb378712012-05-17 13:02:51 +04001471 rdata->result = server->ops->map_error(buf, false);
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001472 if (rdata->result != 0) {
1473 cFYI(1, "%s: server returned error %d", __func__,
1474 rdata->result);
1475 return cifs_readv_discard(server, mid);
1476 }
1477
1478 /* Is there enough to get to the rest of the READ_RSP header? */
Pavel Shilovskyeb378712012-05-17 13:02:51 +04001479 if (server->total_read < server->vals->read_rsp_size) {
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001480 cFYI(1, "%s: server returned short header. got=%u expected=%zu",
Pavel Shilovskyeb378712012-05-17 13:02:51 +04001481 __func__, server->total_read,
1482 server->vals->read_rsp_size);
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001483 rdata->result = -EIO;
1484 return cifs_readv_discard(server, mid);
1485 }
1486
Pavel Shilovskyeb378712012-05-17 13:02:51 +04001487 data_offset = server->ops->read_data_offset(buf) + 4;
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001488 if (data_offset < server->total_read) {
1489 /*
1490 * win2k8 sometimes sends an offset of 0 when the read
1491 * is beyond the EOF. Treat it as if the data starts just after
1492 * the header.
1493 */
1494 cFYI(1, "%s: data offset (%u) inside read response header",
1495 __func__, data_offset);
1496 data_offset = server->total_read;
1497 } else if (data_offset > MAX_CIFS_SMALL_BUFFER_SIZE) {
1498 /* data_offset is beyond the end of smallbuf */
1499 cFYI(1, "%s: data offset (%u) beyond end of smallbuf",
1500 __func__, data_offset);
1501 rdata->result = -EIO;
1502 return cifs_readv_discard(server, mid);
1503 }
1504
1505 cFYI(1, "%s: total_read=%u data_offset=%u", __func__,
1506 server->total_read, data_offset);
1507
1508 len = data_offset - server->total_read;
1509 if (len > 0) {
1510 /* read any junk before data into the rest of smallbuf */
Pavel Shilovsky5ffef7b2012-03-23 14:28:03 -04001511 rdata->iov[0].iov_base = buf + server->total_read;
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001512 rdata->iov[0].iov_len = len;
1513 length = cifs_readv_from_socket(server, rdata->iov, 1, len);
1514 if (length < 0)
1515 return length;
1516 server->total_read += length;
1517 }
1518
1519 /* set up first iov for signature check */
Pavel Shilovsky5ffef7b2012-03-23 14:28:03 -04001520 rdata->iov[0].iov_base = buf;
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001521 rdata->iov[0].iov_len = server->total_read;
1522 cFYI(1, "0: iov_base=%p iov_len=%zu",
1523 rdata->iov[0].iov_base, rdata->iov[0].iov_len);
1524
1525 /* how much data is in the response? */
Pavel Shilovskyeb378712012-05-17 13:02:51 +04001526 data_len = server->ops->read_data_length(buf);
Pavel Shilovsky5ffef7b2012-03-23 14:28:03 -04001527 if (data_offset + data_len > buflen) {
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001528 /* data_len is corrupt -- discard frame */
1529 rdata->result = -EIO;
1530 return cifs_readv_discard(server, mid);
1531 }
1532
1533 /* marshal up the page array */
Jeff Layton3cf003c2012-07-11 09:09:36 -04001534 cifs_kmap_lock();
Jeff Layton8d5ce4d2012-05-16 07:13:16 -04001535 len = rdata->marshal_iov(rdata, data_len);
Jeff Layton3cf003c2012-07-11 09:09:36 -04001536 cifs_kmap_unlock();
Jeff Layton8d5ce4d2012-05-16 07:13:16 -04001537 data_len -= len;
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001538
1539 /* issue the read if we have any iovecs left to fill */
1540 if (rdata->nr_iov > 1) {
1541 length = cifs_readv_from_socket(server, &rdata->iov[1],
1542 rdata->nr_iov - 1, len);
1543 if (length < 0)
1544 return length;
1545 server->total_read += length;
1546 } else {
1547 length = 0;
1548 }
1549
1550 rdata->bytes = length;
1551
Pavel Shilovsky5ffef7b2012-03-23 14:28:03 -04001552 cFYI(1, "total_read=%u buflen=%u remaining=%u", server->total_read,
Jeff Layton8d5ce4d2012-05-16 07:13:16 -04001553 buflen, data_len);
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001554
1555 /* discard anything left over */
Pavel Shilovsky5ffef7b2012-03-23 14:28:03 -04001556 if (server->total_read < buflen)
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001557 return cifs_readv_discard(server, mid);
1558
1559 dequeue_mid(mid, false);
1560 return length;
1561}
1562
1563static void
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001564cifs_readv_callback(struct mid_q_entry *mid)
1565{
1566 struct cifs_readdata *rdata = mid->callback_data;
1567 struct cifs_tcon *tcon = tlink_tcon(rdata->cfile->tlink);
1568 struct TCP_Server_Info *server = tcon->ses->server;
1569
Pavel Shilovsky7c9421e2012-03-23 14:28:03 -04001570 cFYI(1, "%s: mid=%llu state=%d result=%d bytes=%u", __func__,
1571 mid->mid, mid->mid_state, rdata->result, rdata->bytes);
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001572
Pavel Shilovsky7c9421e2012-03-23 14:28:03 -04001573 switch (mid->mid_state) {
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001574 case MID_RESPONSE_RECEIVED:
1575 /* result already set, check signature */
1576 if (server->sec_mode &
1577 (SECMODE_SIGN_REQUIRED | SECMODE_SIGN_ENABLED)) {
1578 if (cifs_verify_signature(rdata->iov, rdata->nr_iov,
1579 server, mid->sequence_number + 1))
1580 cERROR(1, "Unexpected SMB signature");
1581 }
1582 /* FIXME: should this be counted toward the initiating task? */
1583 task_io_account_read(rdata->bytes);
1584 cifs_stats_bytes_read(tcon, rdata->bytes);
1585 break;
1586 case MID_REQUEST_SUBMITTED:
1587 case MID_RETRY_NEEDED:
1588 rdata->result = -EAGAIN;
1589 break;
1590 default:
1591 rdata->result = -EIO;
1592 }
1593
Jeff Laytonda472fc2012-03-23 14:40:53 -04001594 queue_work(cifsiod_wq, &rdata->work);
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001595 DeleteMidQEntry(mid);
Pavel Shilovskya891f0f2012-05-23 16:14:34 +04001596 add_credits(server, 1, 0);
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001597}
1598
1599/* cifs_async_readv - send an async write, and set up mid to handle result */
1600int
1601cifs_async_readv(struct cifs_readdata *rdata)
1602{
1603 int rc;
1604 READ_REQ *smb = NULL;
1605 int wct;
1606 struct cifs_tcon *tcon = tlink_tcon(rdata->cfile->tlink);
1607
1608 cFYI(1, "%s: offset=%llu bytes=%u", __func__,
1609 rdata->offset, rdata->bytes);
1610
1611 if (tcon->ses->capabilities & CAP_LARGE_FILES)
1612 wct = 12;
1613 else {
1614 wct = 10; /* old style read */
1615 if ((rdata->offset >> 32) > 0) {
1616 /* can not handle this big offset for old */
1617 return -EIO;
1618 }
1619 }
1620
1621 rc = small_smb_init(SMB_COM_READ_ANDX, wct, tcon, (void **)&smb);
1622 if (rc)
1623 return rc;
1624
1625 smb->hdr.Pid = cpu_to_le16((__u16)rdata->pid);
1626 smb->hdr.PidHigh = cpu_to_le16((__u16)(rdata->pid >> 16));
1627
1628 smb->AndXCommand = 0xFF; /* none */
1629 smb->Fid = rdata->cfile->netfid;
1630 smb->OffsetLow = cpu_to_le32(rdata->offset & 0xFFFFFFFF);
1631 if (wct == 12)
1632 smb->OffsetHigh = cpu_to_le32(rdata->offset >> 32);
1633 smb->Remaining = 0;
1634 smb->MaxCount = cpu_to_le16(rdata->bytes & 0xFFFF);
1635 smb->MaxCountHigh = cpu_to_le32(rdata->bytes >> 16);
1636 if (wct == 12)
1637 smb->ByteCount = 0;
1638 else {
1639 /* old style read */
1640 struct smb_com_readx_req *smbr =
1641 (struct smb_com_readx_req *)smb;
1642 smbr->ByteCount = 0;
1643 }
1644
1645 /* 4 for RFC1001 length + 1 for BCC */
1646 rdata->iov[0].iov_base = smb;
1647 rdata->iov[0].iov_len = be32_to_cpu(smb->hdr.smb_buf_length) + 4;
1648
Jeff Layton6993f742012-05-16 07:13:17 -04001649 kref_get(&rdata->refcount);
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001650 rc = cifs_call_async(tcon->ses->server, rdata->iov, 1,
1651 cifs_readv_receive, cifs_readv_callback,
Pavel Shilovskya891f0f2012-05-23 16:14:34 +04001652 rdata, 0);
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001653
1654 if (rc == 0)
Pavel Shilovsky44c58182012-05-28 14:16:31 +04001655 cifs_stats_inc(&tcon->stats.cifs_stats.num_reads);
Jeff Layton6993f742012-05-16 07:13:17 -04001656 else
1657 kref_put(&rdata->refcount, cifs_readdata_release);
Jeff Laytone28bc5b2011-10-19 15:30:07 -04001658
1659 cifs_small_buf_release(smb);
1660 return rc;
1661}
1662
Linus Torvalds1da177e2005-04-16 15:20:36 -07001663int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04001664CIFSSMBRead(const unsigned int xid, struct cifs_io_parms *io_parms,
1665 unsigned int *nbytes, char **buf, int *pbuf_type)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001666{
1667 int rc = -EACCES;
1668 READ_REQ *pSMB = NULL;
1669 READ_RSP *pSMBr = NULL;
1670 char *pReadData = NULL;
Steve Frenchbfa0d752005-08-31 21:50:37 -07001671 int wct;
Steve Frenchec637e32005-12-12 20:53:18 -08001672 int resp_buf_type = 0;
1673 struct kvec iov[1];
Pavel Shilovskyd4ffff12011-05-26 06:02:00 +00001674 __u32 pid = io_parms->pid;
1675 __u16 netfid = io_parms->netfid;
1676 __u64 offset = io_parms->offset;
Steve French96daf2b2011-05-27 04:34:02 +00001677 struct cifs_tcon *tcon = io_parms->tcon;
Pavel Shilovskyd4ffff12011-05-26 06:02:00 +00001678 unsigned int count = io_parms->length;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001679
Joe Perchesb6b38f72010-04-21 03:50:45 +00001680 cFYI(1, "Reading %d bytes on fid %d", count, netfid);
Steve French790fe572007-07-07 19:25:05 +00001681 if (tcon->ses->capabilities & CAP_LARGE_FILES)
Steve Frenchbfa0d752005-08-31 21:50:37 -07001682 wct = 12;
Steve French4c3130e2008-12-09 00:28:16 +00001683 else {
Steve Frenchbfa0d752005-08-31 21:50:37 -07001684 wct = 10; /* old style read */
Pavel Shilovskyd4ffff12011-05-26 06:02:00 +00001685 if ((offset >> 32) > 0) {
Steve French4c3130e2008-12-09 00:28:16 +00001686 /* can not handle this big offset for old */
1687 return -EIO;
1688 }
1689 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001690
1691 *nbytes = 0;
Steve Frenchec637e32005-12-12 20:53:18 -08001692 rc = small_smb_init(SMB_COM_READ_ANDX, wct, tcon, (void **) &pSMB);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001693 if (rc)
1694 return rc;
1695
Pavel Shilovskyd4ffff12011-05-26 06:02:00 +00001696 pSMB->hdr.Pid = cpu_to_le16((__u16)pid);
1697 pSMB->hdr.PidHigh = cpu_to_le16((__u16)(pid >> 16));
1698
Linus Torvalds1da177e2005-04-16 15:20:36 -07001699 /* tcon and ses pointer are checked in smb_init */
1700 if (tcon->ses->server == NULL)
1701 return -ECONNABORTED;
1702
Steve Frenchec637e32005-12-12 20:53:18 -08001703 pSMB->AndXCommand = 0xFF; /* none */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001704 pSMB->Fid = netfid;
Pavel Shilovskyd4ffff12011-05-26 06:02:00 +00001705 pSMB->OffsetLow = cpu_to_le32(offset & 0xFFFFFFFF);
Steve French790fe572007-07-07 19:25:05 +00001706 if (wct == 12)
Pavel Shilovskyd4ffff12011-05-26 06:02:00 +00001707 pSMB->OffsetHigh = cpu_to_le32(offset >> 32);
Steve Frenchbfa0d752005-08-31 21:50:37 -07001708
Linus Torvalds1da177e2005-04-16 15:20:36 -07001709 pSMB->Remaining = 0;
1710 pSMB->MaxCount = cpu_to_le16(count & 0xFFFF);
1711 pSMB->MaxCountHigh = cpu_to_le32(count >> 16);
Steve French790fe572007-07-07 19:25:05 +00001712 if (wct == 12)
Steve Frenchbfa0d752005-08-31 21:50:37 -07001713 pSMB->ByteCount = 0; /* no need to do le conversion since 0 */
1714 else {
1715 /* old style read */
Steve French50c2f752007-07-13 00:33:32 +00001716 struct smb_com_readx_req *pSMBW =
Steve Frenchbfa0d752005-08-31 21:50:37 -07001717 (struct smb_com_readx_req *)pSMB;
Steve Frenchec637e32005-12-12 20:53:18 -08001718 pSMBW->ByteCount = 0;
Steve Frenchbfa0d752005-08-31 21:50:37 -07001719 }
Steve Frenchec637e32005-12-12 20:53:18 -08001720
1721 iov[0].iov_base = (char *)pSMB;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00001722 iov[0].iov_len = be32_to_cpu(pSMB->hdr.smb_buf_length) + 4;
Steve Frencha761ac52007-10-18 21:45:27 +00001723 rc = SendReceive2(xid, tcon->ses, iov, 1 /* num iovecs */,
Jeff Layton77499812011-01-11 07:24:23 -05001724 &resp_buf_type, CIFS_LOG_ERROR);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04001725 cifs_stats_inc(&tcon->stats.cifs_stats.num_reads);
Steve Frenchec637e32005-12-12 20:53:18 -08001726 pSMBr = (READ_RSP *)iov[0].iov_base;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001727 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001728 cERROR(1, "Send error in read = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001729 } else {
1730 int data_length = le16_to_cpu(pSMBr->DataLengthHigh);
1731 data_length = data_length << 16;
1732 data_length += le16_to_cpu(pSMBr->DataLength);
1733 *nbytes = data_length;
1734
1735 /*check that DataLength would not go beyond end of SMB */
Steve Frenchec637e32005-12-12 20:53:18 -08001736 if ((data_length > CIFSMaxBufSize)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001737 || (data_length > count)) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001738 cFYI(1, "bad length %d for count %d",
1739 data_length, count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001740 rc = -EIO;
1741 *nbytes = 0;
1742 } else {
Steve Frenchec637e32005-12-12 20:53:18 -08001743 pReadData = (char *) (&pSMBr->hdr.Protocol) +
Steve French26f57362007-08-30 22:09:15 +00001744 le16_to_cpu(pSMBr->DataOffset);
1745/* if (rc = copy_to_user(buf, pReadData, data_length)) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001746 cERROR(1, "Faulting on read rc = %d",rc);
Steve French50c2f752007-07-13 00:33:32 +00001747 rc = -EFAULT;
Steve French26f57362007-08-30 22:09:15 +00001748 }*/ /* can not use copy_to_user when using page cache*/
Steve French790fe572007-07-07 19:25:05 +00001749 if (*buf)
Steve French50c2f752007-07-13 00:33:32 +00001750 memcpy(*buf, pReadData, data_length);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001751 }
1752 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001753
Steve French4b8f9302006-02-26 16:41:18 +00001754/* cifs_small_buf_release(pSMB); */ /* Freed earlier now in SendReceive2 */
Steve French790fe572007-07-07 19:25:05 +00001755 if (*buf) {
1756 if (resp_buf_type == CIFS_SMALL_BUFFER)
Steve Frenchec637e32005-12-12 20:53:18 -08001757 cifs_small_buf_release(iov[0].iov_base);
Steve French790fe572007-07-07 19:25:05 +00001758 else if (resp_buf_type == CIFS_LARGE_BUFFER)
Steve Frenchec637e32005-12-12 20:53:18 -08001759 cifs_buf_release(iov[0].iov_base);
Steve French790fe572007-07-07 19:25:05 +00001760 } else if (resp_buf_type != CIFS_NO_BUFFER) {
Steve French50c2f752007-07-13 00:33:32 +00001761 /* return buffer to caller to free */
1762 *buf = iov[0].iov_base;
Steve French790fe572007-07-07 19:25:05 +00001763 if (resp_buf_type == CIFS_SMALL_BUFFER)
Steve Frenchec637e32005-12-12 20:53:18 -08001764 *pbuf_type = CIFS_SMALL_BUFFER;
Steve French790fe572007-07-07 19:25:05 +00001765 else if (resp_buf_type == CIFS_LARGE_BUFFER)
Steve Frenchec637e32005-12-12 20:53:18 -08001766 *pbuf_type = CIFS_LARGE_BUFFER;
Steve French6cec2ae2006-02-22 17:31:52 -06001767 } /* else no valid buffer on return - leave as null */
Steve Frenchec637e32005-12-12 20:53:18 -08001768
1769 /* Note: On -EAGAIN error only caller can retry on handle based calls
Linus Torvalds1da177e2005-04-16 15:20:36 -07001770 since file handle passed in no longer valid */
1771 return rc;
1772}
1773
Steve Frenchec637e32005-12-12 20:53:18 -08001774
Linus Torvalds1da177e2005-04-16 15:20:36 -07001775int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04001776CIFSSMBWrite(const unsigned int xid, struct cifs_io_parms *io_parms,
Pavel Shilovskyfa2989f2011-05-26 10:01:59 +04001777 unsigned int *nbytes, const char *buf,
Steve French50c2f752007-07-13 00:33:32 +00001778 const char __user *ubuf, const int long_op)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001779{
1780 int rc = -EACCES;
1781 WRITE_REQ *pSMB = NULL;
1782 WRITE_RSP *pSMBr = NULL;
Steve French1c955182005-08-30 20:58:07 -07001783 int bytes_returned, wct;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001784 __u32 bytes_sent;
1785 __u16 byte_count;
Pavel Shilovskyfa2989f2011-05-26 10:01:59 +04001786 __u32 pid = io_parms->pid;
1787 __u16 netfid = io_parms->netfid;
1788 __u64 offset = io_parms->offset;
Steve French96daf2b2011-05-27 04:34:02 +00001789 struct cifs_tcon *tcon = io_parms->tcon;
Pavel Shilovskyfa2989f2011-05-26 10:01:59 +04001790 unsigned int count = io_parms->length;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001791
Steve Frencha24e2d72010-04-03 17:20:21 +00001792 *nbytes = 0;
1793
Joe Perchesb6b38f72010-04-21 03:50:45 +00001794 /* cFYI(1, "write at %lld %d bytes", offset, count);*/
Steve French790fe572007-07-07 19:25:05 +00001795 if (tcon->ses == NULL)
Steve French1c955182005-08-30 20:58:07 -07001796 return -ECONNABORTED;
1797
Steve French790fe572007-07-07 19:25:05 +00001798 if (tcon->ses->capabilities & CAP_LARGE_FILES)
Steve French1c955182005-08-30 20:58:07 -07001799 wct = 14;
Steve French4c3130e2008-12-09 00:28:16 +00001800 else {
Steve French1c955182005-08-30 20:58:07 -07001801 wct = 12;
Steve French4c3130e2008-12-09 00:28:16 +00001802 if ((offset >> 32) > 0) {
1803 /* can not handle big offset for old srv */
1804 return -EIO;
1805 }
1806 }
Steve French1c955182005-08-30 20:58:07 -07001807
1808 rc = smb_init(SMB_COM_WRITE_ANDX, wct, tcon, (void **) &pSMB,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001809 (void **) &pSMBr);
1810 if (rc)
1811 return rc;
Pavel Shilovskyfa2989f2011-05-26 10:01:59 +04001812
1813 pSMB->hdr.Pid = cpu_to_le16((__u16)pid);
1814 pSMB->hdr.PidHigh = cpu_to_le16((__u16)(pid >> 16));
1815
Linus Torvalds1da177e2005-04-16 15:20:36 -07001816 /* tcon and ses pointer are checked in smb_init */
1817 if (tcon->ses->server == NULL)
1818 return -ECONNABORTED;
1819
1820 pSMB->AndXCommand = 0xFF; /* none */
1821 pSMB->Fid = netfid;
1822 pSMB->OffsetLow = cpu_to_le32(offset & 0xFFFFFFFF);
Steve French790fe572007-07-07 19:25:05 +00001823 if (wct == 14)
Steve French1c955182005-08-30 20:58:07 -07001824 pSMB->OffsetHigh = cpu_to_le32(offset >> 32);
Steve French50c2f752007-07-13 00:33:32 +00001825
Linus Torvalds1da177e2005-04-16 15:20:36 -07001826 pSMB->Reserved = 0xFFFFFFFF;
1827 pSMB->WriteMode = 0;
1828 pSMB->Remaining = 0;
1829
Steve French50c2f752007-07-13 00:33:32 +00001830 /* Can increase buffer size if buffer is big enough in some cases ie we
Linus Torvalds1da177e2005-04-16 15:20:36 -07001831 can send more if LARGE_WRITE_X capability returned by the server and if
1832 our buffer is big enough or if we convert to iovecs on socket writes
1833 and eliminate the copy to the CIFS buffer */
Steve French790fe572007-07-07 19:25:05 +00001834 if (tcon->ses->capabilities & CAP_LARGE_WRITE_X) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001835 bytes_sent = min_t(const unsigned int, CIFSMaxBufSize, count);
1836 } else {
1837 bytes_sent = (tcon->ses->server->maxBuf - MAX_CIFS_HDR_SIZE)
1838 & ~0xFF;
1839 }
1840
1841 if (bytes_sent > count)
1842 bytes_sent = count;
1843 pSMB->DataOffset =
Steve French50c2f752007-07-13 00:33:32 +00001844 cpu_to_le16(offsetof(struct smb_com_write_req, Data) - 4);
Steve French790fe572007-07-07 19:25:05 +00001845 if (buf)
Steve French61e74802008-12-03 00:57:54 +00001846 memcpy(pSMB->Data, buf, bytes_sent);
Steve French790fe572007-07-07 19:25:05 +00001847 else if (ubuf) {
1848 if (copy_from_user(pSMB->Data, ubuf, bytes_sent)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001849 cifs_buf_release(pSMB);
1850 return -EFAULT;
1851 }
Steve Frenche30dcf32005-09-20 20:49:16 -07001852 } else if (count != 0) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001853 /* No buffer */
1854 cifs_buf_release(pSMB);
1855 return -EINVAL;
Steve Frenche30dcf32005-09-20 20:49:16 -07001856 } /* else setting file size with write of zero bytes */
Steve French790fe572007-07-07 19:25:05 +00001857 if (wct == 14)
Steve Frenche30dcf32005-09-20 20:49:16 -07001858 byte_count = bytes_sent + 1; /* pad */
Steve Frenchad7a2922008-02-07 23:25:02 +00001859 else /* wct == 12 */
Steve Frenche30dcf32005-09-20 20:49:16 -07001860 byte_count = bytes_sent + 5; /* bigger pad, smaller smb hdr */
Steve Frenchad7a2922008-02-07 23:25:02 +00001861
Linus Torvalds1da177e2005-04-16 15:20:36 -07001862 pSMB->DataLengthLow = cpu_to_le16(bytes_sent & 0xFFFF);
1863 pSMB->DataLengthHigh = cpu_to_le16(bytes_sent >> 16);
Steve Frenchbe8e3b02011-04-29 05:40:20 +00001864 inc_rfc1001_len(pSMB, byte_count);
Steve French1c955182005-08-30 20:58:07 -07001865
Steve French790fe572007-07-07 19:25:05 +00001866 if (wct == 14)
Steve French1c955182005-08-30 20:58:07 -07001867 pSMB->ByteCount = cpu_to_le16(byte_count);
Steve French50c2f752007-07-13 00:33:32 +00001868 else { /* old style write has byte count 4 bytes earlier
1869 so 4 bytes pad */
1870 struct smb_com_writex_req *pSMBW =
Steve French1c955182005-08-30 20:58:07 -07001871 (struct smb_com_writex_req *)pSMB;
1872 pSMBW->ByteCount = cpu_to_le16(byte_count);
1873 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001874
1875 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
1876 (struct smb_hdr *) pSMBr, &bytes_returned, long_op);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04001877 cifs_stats_inc(&tcon->stats.cifs_stats.num_writes);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001878 if (rc) {
Steve Frenchf19159d2010-04-21 04:12:10 +00001879 cFYI(1, "Send error in write = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001880 } else {
1881 *nbytes = le16_to_cpu(pSMBr->CountHigh);
1882 *nbytes = (*nbytes) << 16;
1883 *nbytes += le16_to_cpu(pSMBr->Count);
Suresh Jayaraman6513a812010-03-31 12:00:03 +05301884
1885 /*
1886 * Mask off high 16 bits when bytes written as returned by the
1887 * server is greater than bytes requested by the client. Some
1888 * OS/2 servers are known to set incorrect CountHigh values.
1889 */
1890 if (*nbytes > count)
1891 *nbytes &= 0xFFFF;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001892 }
1893
1894 cifs_buf_release(pSMB);
1895
Steve French50c2f752007-07-13 00:33:32 +00001896 /* Note: On -EAGAIN error only caller can retry on handle based calls
Linus Torvalds1da177e2005-04-16 15:20:36 -07001897 since file handle passed in no longer valid */
1898
1899 return rc;
1900}
1901
Jeff Laytonc28c89f2011-05-19 16:22:56 -04001902void
1903cifs_writedata_release(struct kref *refcount)
1904{
1905 struct cifs_writedata *wdata = container_of(refcount,
1906 struct cifs_writedata, refcount);
1907
1908 if (wdata->cfile)
1909 cifsFileInfo_put(wdata->cfile);
1910
1911 kfree(wdata);
1912}
1913
1914/*
1915 * Write failed with a retryable error. Resend the write request. It's also
1916 * possible that the page was redirtied so re-clean the page.
1917 */
1918static void
1919cifs_writev_requeue(struct cifs_writedata *wdata)
1920{
1921 int i, rc;
1922 struct inode *inode = wdata->cfile->dentry->d_inode;
1923
1924 for (i = 0; i < wdata->nr_pages; i++) {
1925 lock_page(wdata->pages[i]);
1926 clear_page_dirty_for_io(wdata->pages[i]);
1927 }
1928
1929 do {
1930 rc = cifs_async_writev(wdata);
1931 } while (rc == -EAGAIN);
1932
1933 for (i = 0; i < wdata->nr_pages; i++) {
1934 if (rc != 0)
1935 SetPageError(wdata->pages[i]);
1936 unlock_page(wdata->pages[i]);
1937 }
1938
1939 mapping_set_error(inode->i_mapping, rc);
1940 kref_put(&wdata->refcount, cifs_writedata_release);
1941}
1942
Jeff Laytonc2e87642012-03-23 14:40:55 -04001943void
Jeff Laytonc28c89f2011-05-19 16:22:56 -04001944cifs_writev_complete(struct work_struct *work)
1945{
1946 struct cifs_writedata *wdata = container_of(work,
1947 struct cifs_writedata, work);
1948 struct inode *inode = wdata->cfile->dentry->d_inode;
1949 int i = 0;
1950
1951 if (wdata->result == 0) {
Jeff Layton597b0272012-03-23 14:40:56 -04001952 spin_lock(&inode->i_lock);
Jeff Laytonc28c89f2011-05-19 16:22:56 -04001953 cifs_update_eof(CIFS_I(inode), wdata->offset, wdata->bytes);
Jeff Layton597b0272012-03-23 14:40:56 -04001954 spin_unlock(&inode->i_lock);
Jeff Laytonc28c89f2011-05-19 16:22:56 -04001955 cifs_stats_bytes_written(tlink_tcon(wdata->cfile->tlink),
1956 wdata->bytes);
1957 } else if (wdata->sync_mode == WB_SYNC_ALL && wdata->result == -EAGAIN)
1958 return cifs_writev_requeue(wdata);
1959
1960 for (i = 0; i < wdata->nr_pages; i++) {
1961 struct page *page = wdata->pages[i];
1962 if (wdata->result == -EAGAIN)
1963 __set_page_dirty_nobuffers(page);
1964 else if (wdata->result < 0)
1965 SetPageError(page);
1966 end_page_writeback(page);
1967 page_cache_release(page);
1968 }
1969 if (wdata->result != -EAGAIN)
1970 mapping_set_error(inode->i_mapping, wdata->result);
1971 kref_put(&wdata->refcount, cifs_writedata_release);
1972}
1973
1974struct cifs_writedata *
Jeff Laytonc2e87642012-03-23 14:40:55 -04001975cifs_writedata_alloc(unsigned int nr_pages, work_func_t complete)
Jeff Laytonc28c89f2011-05-19 16:22:56 -04001976{
1977 struct cifs_writedata *wdata;
1978
1979 /* this would overflow */
1980 if (nr_pages == 0) {
1981 cERROR(1, "%s: called with nr_pages == 0!", __func__);
1982 return NULL;
1983 }
1984
1985 /* writedata + number of page pointers */
1986 wdata = kzalloc(sizeof(*wdata) +
1987 sizeof(struct page *) * (nr_pages - 1), GFP_NOFS);
1988 if (wdata != NULL) {
Jeff Laytonc28c89f2011-05-19 16:22:56 -04001989 kref_init(&wdata->refcount);
Jeff Laytonda82f7e2012-03-23 14:40:56 -04001990 INIT_LIST_HEAD(&wdata->list);
1991 init_completion(&wdata->done);
1992 INIT_WORK(&wdata->work, complete);
Jeff Laytonc28c89f2011-05-19 16:22:56 -04001993 }
1994 return wdata;
1995}
1996
1997/*
Pavel Shilovsky7c9421e2012-03-23 14:28:03 -04001998 * Check the mid_state and signature on received buffer (if any), and queue the
Jeff Laytonc28c89f2011-05-19 16:22:56 -04001999 * workqueue completion task.
2000 */
2001static void
2002cifs_writev_callback(struct mid_q_entry *mid)
2003{
2004 struct cifs_writedata *wdata = mid->callback_data;
Steve French96daf2b2011-05-27 04:34:02 +00002005 struct cifs_tcon *tcon = tlink_tcon(wdata->cfile->tlink);
Jeff Laytonc28c89f2011-05-19 16:22:56 -04002006 unsigned int written;
2007 WRITE_RSP *smb = (WRITE_RSP *)mid->resp_buf;
2008
Pavel Shilovsky7c9421e2012-03-23 14:28:03 -04002009 switch (mid->mid_state) {
Jeff Laytonc28c89f2011-05-19 16:22:56 -04002010 case MID_RESPONSE_RECEIVED:
2011 wdata->result = cifs_check_receive(mid, tcon->ses->server, 0);
2012 if (wdata->result != 0)
2013 break;
2014
2015 written = le16_to_cpu(smb->CountHigh);
2016 written <<= 16;
2017 written += le16_to_cpu(smb->Count);
2018 /*
2019 * Mask off high 16 bits when bytes written as returned
2020 * by the server is greater than bytes requested by the
2021 * client. OS/2 servers are known to set incorrect
2022 * CountHigh values.
2023 */
2024 if (written > wdata->bytes)
2025 written &= 0xFFFF;
2026
2027 if (written < wdata->bytes)
2028 wdata->result = -ENOSPC;
2029 else
2030 wdata->bytes = written;
2031 break;
2032 case MID_REQUEST_SUBMITTED:
2033 case MID_RETRY_NEEDED:
2034 wdata->result = -EAGAIN;
2035 break;
2036 default:
2037 wdata->result = -EIO;
2038 break;
2039 }
2040
Jeff Laytonda472fc2012-03-23 14:40:53 -04002041 queue_work(cifsiod_wq, &wdata->work);
Jeff Laytonc28c89f2011-05-19 16:22:56 -04002042 DeleteMidQEntry(mid);
Pavel Shilovskya891f0f2012-05-23 16:14:34 +04002043 add_credits(tcon->ses->server, 1, 0);
Jeff Laytonc28c89f2011-05-19 16:22:56 -04002044}
2045
2046/* cifs_async_writev - send an async write, and set up mid to handle result */
2047int
2048cifs_async_writev(struct cifs_writedata *wdata)
2049{
2050 int i, rc = -EACCES;
2051 WRITE_REQ *smb = NULL;
2052 int wct;
Steve French96daf2b2011-05-27 04:34:02 +00002053 struct cifs_tcon *tcon = tlink_tcon(wdata->cfile->tlink);
Jeff Laytonc28c89f2011-05-19 16:22:56 -04002054 struct kvec *iov = NULL;
2055
2056 if (tcon->ses->capabilities & CAP_LARGE_FILES) {
2057 wct = 14;
2058 } else {
2059 wct = 12;
2060 if (wdata->offset >> 32 > 0) {
2061 /* can not handle big offset for old srv */
2062 return -EIO;
2063 }
2064 }
2065
2066 rc = small_smb_init(SMB_COM_WRITE_ANDX, wct, tcon, (void **)&smb);
2067 if (rc)
2068 goto async_writev_out;
2069
2070 /* 1 iov per page + 1 for header */
2071 iov = kzalloc((wdata->nr_pages + 1) * sizeof(*iov), GFP_NOFS);
2072 if (iov == NULL) {
2073 rc = -ENOMEM;
2074 goto async_writev_out;
2075 }
2076
Jeff Laytonfe5f5d22012-03-23 14:40:55 -04002077 smb->hdr.Pid = cpu_to_le16((__u16)wdata->pid);
2078 smb->hdr.PidHigh = cpu_to_le16((__u16)(wdata->pid >> 16));
Pavel Shilovskyfa2989f2011-05-26 10:01:59 +04002079
Jeff Laytonc28c89f2011-05-19 16:22:56 -04002080 smb->AndXCommand = 0xFF; /* none */
2081 smb->Fid = wdata->cfile->netfid;
2082 smb->OffsetLow = cpu_to_le32(wdata->offset & 0xFFFFFFFF);
2083 if (wct == 14)
2084 smb->OffsetHigh = cpu_to_le32(wdata->offset >> 32);
2085 smb->Reserved = 0xFFFFFFFF;
2086 smb->WriteMode = 0;
2087 smb->Remaining = 0;
2088
2089 smb->DataOffset =
2090 cpu_to_le16(offsetof(struct smb_com_write_req, Data) - 4);
2091
2092 /* 4 for RFC1001 length + 1 for BCC */
2093 iov[0].iov_len = be32_to_cpu(smb->hdr.smb_buf_length) + 4 + 1;
2094 iov[0].iov_base = smb;
2095
Jeff Laytone9492872012-03-23 14:40:56 -04002096 /*
2097 * This function should marshal up the page array into the kvec
2098 * array, reserving [0] for the header. It should kmap the pages
2099 * and set the iov_len properly for each one. It may also set
2100 * wdata->bytes too.
2101 */
Jeff Layton3cf003c2012-07-11 09:09:36 -04002102 cifs_kmap_lock();
Jeff Laytone9492872012-03-23 14:40:56 -04002103 wdata->marshal_iov(iov, wdata);
Jeff Layton3cf003c2012-07-11 09:09:36 -04002104 cifs_kmap_unlock();
Jeff Laytonc28c89f2011-05-19 16:22:56 -04002105
2106 cFYI(1, "async write at %llu %u bytes", wdata->offset, wdata->bytes);
2107
2108 smb->DataLengthLow = cpu_to_le16(wdata->bytes & 0xFFFF);
2109 smb->DataLengthHigh = cpu_to_le16(wdata->bytes >> 16);
2110
2111 if (wct == 14) {
2112 inc_rfc1001_len(&smb->hdr, wdata->bytes + 1);
2113 put_bcc(wdata->bytes + 1, &smb->hdr);
2114 } else {
2115 /* wct == 12 */
2116 struct smb_com_writex_req *smbw =
2117 (struct smb_com_writex_req *)smb;
2118 inc_rfc1001_len(&smbw->hdr, wdata->bytes + 5);
2119 put_bcc(wdata->bytes + 5, &smbw->hdr);
2120 iov[0].iov_len += 4; /* pad bigger by four bytes */
2121 }
2122
2123 kref_get(&wdata->refcount);
2124 rc = cifs_call_async(tcon->ses->server, iov, wdata->nr_pages + 1,
Pavel Shilovskya891f0f2012-05-23 16:14:34 +04002125 NULL, cifs_writev_callback, wdata, 0);
Jeff Laytonc28c89f2011-05-19 16:22:56 -04002126
2127 if (rc == 0)
Pavel Shilovsky44c58182012-05-28 14:16:31 +04002128 cifs_stats_inc(&tcon->stats.cifs_stats.num_writes);
Jeff Laytonc28c89f2011-05-19 16:22:56 -04002129 else
2130 kref_put(&wdata->refcount, cifs_writedata_release);
2131
2132 /* send is done, unmap pages */
2133 for (i = 0; i < wdata->nr_pages; i++)
2134 kunmap(wdata->pages[i]);
2135
2136async_writev_out:
2137 cifs_small_buf_release(smb);
2138 kfree(iov);
2139 return rc;
2140}
2141
Steve Frenchd6e04ae2005-06-13 13:24:43 -05002142int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04002143CIFSSMBWrite2(const unsigned int xid, struct cifs_io_parms *io_parms,
Pavel Shilovskyfa2989f2011-05-26 10:01:59 +04002144 unsigned int *nbytes, struct kvec *iov, int n_vec,
2145 const int long_op)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002146{
2147 int rc = -EACCES;
2148 WRITE_REQ *pSMB = NULL;
Steve Frenchec637e32005-12-12 20:53:18 -08002149 int wct;
Steve Frenchd6e04ae2005-06-13 13:24:43 -05002150 int smb_hdr_len;
Steve Frenchec637e32005-12-12 20:53:18 -08002151 int resp_buf_type = 0;
Pavel Shilovskyfa2989f2011-05-26 10:01:59 +04002152 __u32 pid = io_parms->pid;
2153 __u16 netfid = io_parms->netfid;
2154 __u64 offset = io_parms->offset;
Steve French96daf2b2011-05-27 04:34:02 +00002155 struct cifs_tcon *tcon = io_parms->tcon;
Pavel Shilovskyfa2989f2011-05-26 10:01:59 +04002156 unsigned int count = io_parms->length;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002157
Jeff Laytonfbec9ab2009-04-03 13:44:00 -04002158 *nbytes = 0;
2159
Joe Perchesb6b38f72010-04-21 03:50:45 +00002160 cFYI(1, "write2 at %lld %d bytes", (long long)offset, count);
Steve Frenchff7feac2005-11-15 16:45:16 -08002161
Steve French4c3130e2008-12-09 00:28:16 +00002162 if (tcon->ses->capabilities & CAP_LARGE_FILES) {
Steve French8cc64c62005-10-03 13:49:43 -07002163 wct = 14;
Steve French4c3130e2008-12-09 00:28:16 +00002164 } else {
Steve French8cc64c62005-10-03 13:49:43 -07002165 wct = 12;
Steve French4c3130e2008-12-09 00:28:16 +00002166 if ((offset >> 32) > 0) {
2167 /* can not handle big offset for old srv */
2168 return -EIO;
2169 }
2170 }
Steve French8cc64c62005-10-03 13:49:43 -07002171 rc = small_smb_init(SMB_COM_WRITE_ANDX, wct, tcon, (void **) &pSMB);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002172 if (rc)
2173 return rc;
Pavel Shilovskyfa2989f2011-05-26 10:01:59 +04002174
2175 pSMB->hdr.Pid = cpu_to_le16((__u16)pid);
2176 pSMB->hdr.PidHigh = cpu_to_le16((__u16)(pid >> 16));
2177
Linus Torvalds1da177e2005-04-16 15:20:36 -07002178 /* tcon and ses pointer are checked in smb_init */
2179 if (tcon->ses->server == NULL)
2180 return -ECONNABORTED;
2181
Steve Frenchd6e04ae2005-06-13 13:24:43 -05002182 pSMB->AndXCommand = 0xFF; /* none */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002183 pSMB->Fid = netfid;
2184 pSMB->OffsetLow = cpu_to_le32(offset & 0xFFFFFFFF);
Steve French790fe572007-07-07 19:25:05 +00002185 if (wct == 14)
Steve French8cc64c62005-10-03 13:49:43 -07002186 pSMB->OffsetHigh = cpu_to_le32(offset >> 32);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002187 pSMB->Reserved = 0xFFFFFFFF;
2188 pSMB->WriteMode = 0;
2189 pSMB->Remaining = 0;
Steve Frenchd6e04ae2005-06-13 13:24:43 -05002190
Linus Torvalds1da177e2005-04-16 15:20:36 -07002191 pSMB->DataOffset =
Steve French50c2f752007-07-13 00:33:32 +00002192 cpu_to_le16(offsetof(struct smb_com_write_req, Data) - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002193
Steve French3e844692005-10-03 13:37:24 -07002194 pSMB->DataLengthLow = cpu_to_le16(count & 0xFFFF);
2195 pSMB->DataLengthHigh = cpu_to_le16(count >> 16);
Steve Frenchbe8e3b02011-04-29 05:40:20 +00002196 /* header + 1 byte pad */
2197 smb_hdr_len = be32_to_cpu(pSMB->hdr.smb_buf_length) + 1;
Steve French790fe572007-07-07 19:25:05 +00002198 if (wct == 14)
Steve Frenchbe8e3b02011-04-29 05:40:20 +00002199 inc_rfc1001_len(pSMB, count + 1);
Steve French8cc64c62005-10-03 13:49:43 -07002200 else /* wct == 12 */
Steve Frenchbe8e3b02011-04-29 05:40:20 +00002201 inc_rfc1001_len(pSMB, count + 5); /* smb data starts later */
Steve French790fe572007-07-07 19:25:05 +00002202 if (wct == 14)
Steve French8cc64c62005-10-03 13:49:43 -07002203 pSMB->ByteCount = cpu_to_le16(count + 1);
2204 else /* wct == 12 */ /* bigger pad, smaller smb hdr, keep offset ok */ {
Steve French50c2f752007-07-13 00:33:32 +00002205 struct smb_com_writex_req *pSMBW =
Steve French8cc64c62005-10-03 13:49:43 -07002206 (struct smb_com_writex_req *)pSMB;
2207 pSMBW->ByteCount = cpu_to_le16(count + 5);
2208 }
Steve French3e844692005-10-03 13:37:24 -07002209 iov[0].iov_base = pSMB;
Steve French790fe572007-07-07 19:25:05 +00002210 if (wct == 14)
Steve Frenchec637e32005-12-12 20:53:18 -08002211 iov[0].iov_len = smb_hdr_len + 4;
2212 else /* wct == 12 pad bigger by four bytes */
2213 iov[0].iov_len = smb_hdr_len + 8;
Steve French50c2f752007-07-13 00:33:32 +00002214
Steve French3e844692005-10-03 13:37:24 -07002215
Steve Frenchec637e32005-12-12 20:53:18 -08002216 rc = SendReceive2(xid, tcon->ses, iov, n_vec + 1, &resp_buf_type,
Steve French133672e2007-11-13 22:41:37 +00002217 long_op);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04002218 cifs_stats_inc(&tcon->stats.cifs_stats.num_writes);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002219 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00002220 cFYI(1, "Send error Write2 = %d", rc);
Steve French790fe572007-07-07 19:25:05 +00002221 } else if (resp_buf_type == 0) {
Steve Frenchec637e32005-12-12 20:53:18 -08002222 /* presumably this can not happen, but best to be safe */
2223 rc = -EIO;
Steve Frenchd6e04ae2005-06-13 13:24:43 -05002224 } else {
Steve Frenchad7a2922008-02-07 23:25:02 +00002225 WRITE_RSP *pSMBr = (WRITE_RSP *)iov[0].iov_base;
Steve Frenchd6e04ae2005-06-13 13:24:43 -05002226 *nbytes = le16_to_cpu(pSMBr->CountHigh);
2227 *nbytes = (*nbytes) << 16;
2228 *nbytes += le16_to_cpu(pSMBr->Count);
Suresh Jayaraman6513a812010-03-31 12:00:03 +05302229
2230 /*
2231 * Mask off high 16 bits when bytes written as returned by the
2232 * server is greater than bytes requested by the client. OS/2
2233 * servers are known to set incorrect CountHigh values.
2234 */
2235 if (*nbytes > count)
2236 *nbytes &= 0xFFFF;
Steve French50c2f752007-07-13 00:33:32 +00002237 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002238
Steve French4b8f9302006-02-26 16:41:18 +00002239/* cifs_small_buf_release(pSMB); */ /* Freed earlier now in SendReceive2 */
Steve French790fe572007-07-07 19:25:05 +00002240 if (resp_buf_type == CIFS_SMALL_BUFFER)
Steve Frenchec637e32005-12-12 20:53:18 -08002241 cifs_small_buf_release(iov[0].iov_base);
Steve French790fe572007-07-07 19:25:05 +00002242 else if (resp_buf_type == CIFS_LARGE_BUFFER)
Steve Frenchec637e32005-12-12 20:53:18 -08002243 cifs_buf_release(iov[0].iov_base);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002244
Steve French50c2f752007-07-13 00:33:32 +00002245 /* Note: On -EAGAIN error only caller can retry on handle based calls
Linus Torvalds1da177e2005-04-16 15:20:36 -07002246 since file handle passed in no longer valid */
2247
2248 return rc;
2249}
Steve Frenchd6e04ae2005-06-13 13:24:43 -05002250
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04002251int cifs_lockv(const unsigned int xid, struct cifs_tcon *tcon,
2252 const __u16 netfid, const __u8 lock_type, const __u32 num_unlock,
Pavel Shilovsky9ee305b2011-10-22 15:33:31 +04002253 const __u32 num_lock, LOCKING_ANDX_RANGE *buf)
2254{
2255 int rc = 0;
2256 LOCK_REQ *pSMB = NULL;
2257 struct kvec iov[2];
2258 int resp_buf_type;
2259 __u16 count;
2260
2261 cFYI(1, "cifs_lockv num lock %d num unlock %d", num_lock, num_unlock);
2262
2263 rc = small_smb_init(SMB_COM_LOCKING_ANDX, 8, tcon, (void **) &pSMB);
2264 if (rc)
2265 return rc;
2266
2267 pSMB->Timeout = 0;
2268 pSMB->NumberOfLocks = cpu_to_le16(num_lock);
2269 pSMB->NumberOfUnlocks = cpu_to_le16(num_unlock);
2270 pSMB->LockType = lock_type;
2271 pSMB->AndXCommand = 0xFF; /* none */
2272 pSMB->Fid = netfid; /* netfid stays le */
2273
2274 count = (num_unlock + num_lock) * sizeof(LOCKING_ANDX_RANGE);
2275 inc_rfc1001_len(pSMB, count);
2276 pSMB->ByteCount = cpu_to_le16(count);
2277
2278 iov[0].iov_base = (char *)pSMB;
2279 iov[0].iov_len = be32_to_cpu(pSMB->hdr.smb_buf_length) + 4 -
2280 (num_unlock + num_lock) * sizeof(LOCKING_ANDX_RANGE);
2281 iov[1].iov_base = (char *)buf;
2282 iov[1].iov_len = (num_unlock + num_lock) * sizeof(LOCKING_ANDX_RANGE);
2283
Pavel Shilovsky44c58182012-05-28 14:16:31 +04002284 cifs_stats_inc(&tcon->stats.cifs_stats.num_locks);
Pavel Shilovsky9ee305b2011-10-22 15:33:31 +04002285 rc = SendReceive2(xid, tcon->ses, iov, 2, &resp_buf_type, CIFS_NO_RESP);
2286 if (rc)
2287 cFYI(1, "Send error in cifs_lockv = %d", rc);
2288
2289 return rc;
2290}
Steve Frenchd6e04ae2005-06-13 13:24:43 -05002291
Linus Torvalds1da177e2005-04-16 15:20:36 -07002292int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04002293CIFSSMBLock(const unsigned int xid, struct cifs_tcon *tcon,
Pavel Shilovsky03776f42010-08-17 11:26:00 +04002294 const __u16 smb_file_id, const __u32 netpid, const __u64 len,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002295 const __u64 offset, const __u32 numUnlock,
Pavel Shilovsky12fed002011-01-17 20:15:44 +03002296 const __u32 numLock, const __u8 lockType,
2297 const bool waitFlag, const __u8 oplock_level)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002298{
2299 int rc = 0;
2300 LOCK_REQ *pSMB = NULL;
Steve Frenchaaa9bbe2008-05-23 17:38:32 +00002301/* LOCK_RSP *pSMBr = NULL; */ /* No response data other than rc to parse */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002302 int bytes_returned;
Pavel Shilovskya891f0f2012-05-23 16:14:34 +04002303 int flags = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002304 __u16 count;
2305
Joe Perchesb6b38f72010-04-21 03:50:45 +00002306 cFYI(1, "CIFSSMBLock timeout %d numLock %d", (int)waitFlag, numLock);
Steve French46810cb2005-04-28 22:41:09 -07002307 rc = small_smb_init(SMB_COM_LOCKING_ANDX, 8, tcon, (void **) &pSMB);
2308
Linus Torvalds1da177e2005-04-16 15:20:36 -07002309 if (rc)
2310 return rc;
2311
Steve French790fe572007-07-07 19:25:05 +00002312 if (lockType == LOCKING_ANDX_OPLOCK_RELEASE) {
Pavel Shilovskya891f0f2012-05-23 16:14:34 +04002313 /* no response expected */
2314 flags = CIFS_ASYNC_OP | CIFS_OBREAK_OP;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002315 pSMB->Timeout = 0;
Steve French4b18f2a2008-04-29 00:06:05 +00002316 } else if (waitFlag) {
Pavel Shilovskya891f0f2012-05-23 16:14:34 +04002317 flags = CIFS_BLOCKING_OP; /* blocking operation, no timeout */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002318 pSMB->Timeout = cpu_to_le32(-1);/* blocking - do not time out */
2319 } else {
2320 pSMB->Timeout = 0;
2321 }
2322
2323 pSMB->NumberOfLocks = cpu_to_le16(numLock);
2324 pSMB->NumberOfUnlocks = cpu_to_le16(numUnlock);
2325 pSMB->LockType = lockType;
Pavel Shilovsky12fed002011-01-17 20:15:44 +03002326 pSMB->OplockLevel = oplock_level;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002327 pSMB->AndXCommand = 0xFF; /* none */
2328 pSMB->Fid = smb_file_id; /* netfid stays le */
2329
Steve French790fe572007-07-07 19:25:05 +00002330 if ((numLock != 0) || (numUnlock != 0)) {
Pavel Shilovsky03776f42010-08-17 11:26:00 +04002331 pSMB->Locks[0].Pid = cpu_to_le16(netpid);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002332 /* BB where to store pid high? */
2333 pSMB->Locks[0].LengthLow = cpu_to_le32((u32)len);
2334 pSMB->Locks[0].LengthHigh = cpu_to_le32((u32)(len>>32));
2335 pSMB->Locks[0].OffsetLow = cpu_to_le32((u32)offset);
2336 pSMB->Locks[0].OffsetHigh = cpu_to_le32((u32)(offset>>32));
2337 count = sizeof(LOCKING_ANDX_RANGE);
2338 } else {
2339 /* oplock break */
2340 count = 0;
2341 }
Steve Frenchbe8e3b02011-04-29 05:40:20 +00002342 inc_rfc1001_len(pSMB, count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002343 pSMB->ByteCount = cpu_to_le16(count);
2344
Jeremy Allison7ee1af72006-08-02 21:56:33 +00002345 if (waitFlag) {
2346 rc = SendReceiveBlockingLock(xid, tcon, (struct smb_hdr *) pSMB,
Steve Frenchaaa9bbe2008-05-23 17:38:32 +00002347 (struct smb_hdr *) pSMB, &bytes_returned);
Steve French133672e2007-11-13 22:41:37 +00002348 cifs_small_buf_release(pSMB);
Jeremy Allison7ee1af72006-08-02 21:56:33 +00002349 } else {
Pavel Shilovskya891f0f2012-05-23 16:14:34 +04002350 rc = SendReceiveNoRsp(xid, tcon->ses, (char *)pSMB, flags);
Steve French133672e2007-11-13 22:41:37 +00002351 /* SMB buffer freed by function above */
Jeremy Allison7ee1af72006-08-02 21:56:33 +00002352 }
Pavel Shilovsky44c58182012-05-28 14:16:31 +04002353 cifs_stats_inc(&tcon->stats.cifs_stats.num_locks);
Steve Frenchad7a2922008-02-07 23:25:02 +00002354 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00002355 cFYI(1, "Send error in Lock = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002356
Steve French50c2f752007-07-13 00:33:32 +00002357 /* Note: On -EAGAIN error only caller can retry on handle based calls
Linus Torvalds1da177e2005-04-16 15:20:36 -07002358 since file handle passed in no longer valid */
2359 return rc;
2360}
2361
2362int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04002363CIFSSMBPosixLock(const unsigned int xid, struct cifs_tcon *tcon,
Jeff Laytonc5fd3632012-07-23 13:28:37 -04002364 const __u16 smb_file_id, const __u32 netpid,
2365 const loff_t start_offset, const __u64 len,
2366 struct file_lock *pLockData, const __u16 lock_type,
2367 const bool waitFlag)
Steve French08547b02006-02-28 22:39:25 +00002368{
2369 struct smb_com_transaction2_sfi_req *pSMB = NULL;
2370 struct smb_com_transaction2_sfi_rsp *pSMBr = NULL;
Steve French08547b02006-02-28 22:39:25 +00002371 struct cifs_posix_lock *parm_data;
2372 int rc = 0;
Steve French3a5ff612006-07-14 22:37:11 +00002373 int timeout = 0;
Steve French08547b02006-02-28 22:39:25 +00002374 int bytes_returned = 0;
Steve French133672e2007-11-13 22:41:37 +00002375 int resp_buf_type = 0;
Steve French08547b02006-02-28 22:39:25 +00002376 __u16 params, param_offset, offset, byte_count, count;
Steve French133672e2007-11-13 22:41:37 +00002377 struct kvec iov[1];
Steve French08547b02006-02-28 22:39:25 +00002378
Joe Perchesb6b38f72010-04-21 03:50:45 +00002379 cFYI(1, "Posix Lock");
Steve Frenchfc94cdb2006-05-30 18:03:32 +00002380
Steve French08547b02006-02-28 22:39:25 +00002381 rc = small_smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB);
2382
2383 if (rc)
2384 return rc;
2385
2386 pSMBr = (struct smb_com_transaction2_sfi_rsp *)pSMB;
2387
Steve French50c2f752007-07-13 00:33:32 +00002388 params = 6;
Steve French08547b02006-02-28 22:39:25 +00002389 pSMB->MaxSetupCount = 0;
2390 pSMB->Reserved = 0;
2391 pSMB->Flags = 0;
Steve French08547b02006-02-28 22:39:25 +00002392 pSMB->Reserved2 = 0;
2393 param_offset = offsetof(struct smb_com_transaction2_sfi_req, Fid) - 4;
2394 offset = param_offset + params;
2395
Steve French08547b02006-02-28 22:39:25 +00002396 count = sizeof(struct cifs_posix_lock);
2397 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve Frenchad7a2922008-02-07 23:25:02 +00002398 pSMB->MaxDataCount = cpu_to_le16(1000); /* BB find max SMB from sess */
Steve French08547b02006-02-28 22:39:25 +00002399 pSMB->SetupCount = 1;
2400 pSMB->Reserved3 = 0;
Jeff Laytonc5fd3632012-07-23 13:28:37 -04002401 if (pLockData)
Steve French08547b02006-02-28 22:39:25 +00002402 pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FILE_INFORMATION);
2403 else
2404 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FILE_INFORMATION);
2405 byte_count = 3 /* pad */ + params + count;
2406 pSMB->DataCount = cpu_to_le16(count);
2407 pSMB->ParameterCount = cpu_to_le16(params);
2408 pSMB->TotalDataCount = pSMB->DataCount;
2409 pSMB->TotalParameterCount = pSMB->ParameterCount;
2410 pSMB->ParameterOffset = cpu_to_le16(param_offset);
Steve French50c2f752007-07-13 00:33:32 +00002411 parm_data = (struct cifs_posix_lock *)
Steve French08547b02006-02-28 22:39:25 +00002412 (((char *) &pSMB->hdr.Protocol) + offset);
2413
2414 parm_data->lock_type = cpu_to_le16(lock_type);
Steve French790fe572007-07-07 19:25:05 +00002415 if (waitFlag) {
Steve French133672e2007-11-13 22:41:37 +00002416 timeout = CIFS_BLOCKING_OP; /* blocking operation, no timeout */
Steve Frenchcec6815a2006-05-30 18:07:17 +00002417 parm_data->lock_flags = cpu_to_le16(1);
Steve French3a5ff612006-07-14 22:37:11 +00002418 pSMB->Timeout = cpu_to_le32(-1);
2419 } else
2420 pSMB->Timeout = 0;
2421
Pavel Shilovsky4f6bcec2011-10-22 15:33:30 +04002422 parm_data->pid = cpu_to_le32(netpid);
Jeff Laytonc5fd3632012-07-23 13:28:37 -04002423 parm_data->start = cpu_to_le64(start_offset);
Steve Frenchcec6815a2006-05-30 18:07:17 +00002424 parm_data->length = cpu_to_le64(len); /* normalize negative numbers */
Steve French08547b02006-02-28 22:39:25 +00002425
2426 pSMB->DataOffset = cpu_to_le16(offset);
Steve Frenchf26282c2006-03-01 09:17:37 +00002427 pSMB->Fid = smb_file_id;
Steve French08547b02006-02-28 22:39:25 +00002428 pSMB->InformationLevel = cpu_to_le16(SMB_SET_POSIX_LOCK);
2429 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00002430 inc_rfc1001_len(pSMB, byte_count);
Steve French08547b02006-02-28 22:39:25 +00002431 pSMB->ByteCount = cpu_to_le16(byte_count);
Jeremy Allison7ee1af72006-08-02 21:56:33 +00002432 if (waitFlag) {
2433 rc = SendReceiveBlockingLock(xid, tcon, (struct smb_hdr *) pSMB,
2434 (struct smb_hdr *) pSMBr, &bytes_returned);
2435 } else {
Steve French133672e2007-11-13 22:41:37 +00002436 iov[0].iov_base = (char *)pSMB;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00002437 iov[0].iov_len = be32_to_cpu(pSMB->hdr.smb_buf_length) + 4;
Steve French133672e2007-11-13 22:41:37 +00002438 rc = SendReceive2(xid, tcon->ses, iov, 1 /* num iovecs */,
2439 &resp_buf_type, timeout);
2440 pSMB = NULL; /* request buf already freed by SendReceive2. Do
2441 not try to free it twice below on exit */
2442 pSMBr = (struct smb_com_transaction2_sfi_rsp *)iov[0].iov_base;
Jeremy Allison7ee1af72006-08-02 21:56:33 +00002443 }
2444
Steve French08547b02006-02-28 22:39:25 +00002445 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00002446 cFYI(1, "Send error in Posix Lock = %d", rc);
Jeff Laytonc5fd3632012-07-23 13:28:37 -04002447 } else if (pLockData) {
Steve Frenchfc94cdb2006-05-30 18:03:32 +00002448 /* lock structure can be returned on get */
2449 __u16 data_offset;
2450 __u16 data_count;
2451 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
Steve French08547b02006-02-28 22:39:25 +00002452
Jeff Layton820a8032011-05-04 08:05:26 -04002453 if (rc || get_bcc(&pSMBr->hdr) < sizeof(*parm_data)) {
Steve Frenchfc94cdb2006-05-30 18:03:32 +00002454 rc = -EIO; /* bad smb */
2455 goto plk_err_exit;
2456 }
Steve Frenchfc94cdb2006-05-30 18:03:32 +00002457 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
2458 data_count = le16_to_cpu(pSMBr->t2.DataCount);
Steve French790fe572007-07-07 19:25:05 +00002459 if (data_count < sizeof(struct cifs_posix_lock)) {
Steve Frenchfc94cdb2006-05-30 18:03:32 +00002460 rc = -EIO;
2461 goto plk_err_exit;
2462 }
2463 parm_data = (struct cifs_posix_lock *)
2464 ((char *)&pSMBr->hdr.Protocol + data_offset);
Pavel Shilovskyf05337c2010-04-05 09:59:14 +04002465 if (parm_data->lock_type == __constant_cpu_to_le16(CIFS_UNLCK))
Steve Frenchfc94cdb2006-05-30 18:03:32 +00002466 pLockData->fl_type = F_UNLCK;
Pavel Shilovskyf05337c2010-04-05 09:59:14 +04002467 else {
2468 if (parm_data->lock_type ==
2469 __constant_cpu_to_le16(CIFS_RDLCK))
2470 pLockData->fl_type = F_RDLCK;
2471 else if (parm_data->lock_type ==
2472 __constant_cpu_to_le16(CIFS_WRLCK))
2473 pLockData->fl_type = F_WRLCK;
2474
Steve French5443d132011-03-13 05:08:25 +00002475 pLockData->fl_start = le64_to_cpu(parm_data->start);
2476 pLockData->fl_end = pLockData->fl_start +
2477 le64_to_cpu(parm_data->length) - 1;
2478 pLockData->fl_pid = le32_to_cpu(parm_data->pid);
Pavel Shilovskyf05337c2010-04-05 09:59:14 +04002479 }
Steve Frenchfc94cdb2006-05-30 18:03:32 +00002480 }
Steve French50c2f752007-07-13 00:33:32 +00002481
Steve Frenchfc94cdb2006-05-30 18:03:32 +00002482plk_err_exit:
Steve French08547b02006-02-28 22:39:25 +00002483 if (pSMB)
2484 cifs_small_buf_release(pSMB);
2485
Steve French133672e2007-11-13 22:41:37 +00002486 if (resp_buf_type == CIFS_SMALL_BUFFER)
2487 cifs_small_buf_release(iov[0].iov_base);
2488 else if (resp_buf_type == CIFS_LARGE_BUFFER)
2489 cifs_buf_release(iov[0].iov_base);
2490
Steve French08547b02006-02-28 22:39:25 +00002491 /* Note: On -EAGAIN error only caller can retry on handle based calls
2492 since file handle passed in no longer valid */
2493
2494 return rc;
2495}
2496
2497
2498int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04002499CIFSSMBClose(const unsigned int xid, struct cifs_tcon *tcon, int smb_file_id)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002500{
2501 int rc = 0;
2502 CLOSE_REQ *pSMB = NULL;
Joe Perchesb6b38f72010-04-21 03:50:45 +00002503 cFYI(1, "In CIFSSMBClose");
Linus Torvalds1da177e2005-04-16 15:20:36 -07002504
2505/* do not retry on dead session on close */
2506 rc = small_smb_init(SMB_COM_CLOSE, 3, tcon, (void **) &pSMB);
Steve French790fe572007-07-07 19:25:05 +00002507 if (rc == -EAGAIN)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002508 return 0;
2509 if (rc)
2510 return rc;
2511
Linus Torvalds1da177e2005-04-16 15:20:36 -07002512 pSMB->FileID = (__u16) smb_file_id;
Steve Frenchb815f1e52006-10-02 05:53:29 +00002513 pSMB->LastWriteTime = 0xFFFFFFFF;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002514 pSMB->ByteCount = 0;
Pavel Shilovsky792af7b2012-03-23 14:28:02 -04002515 rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04002516 cifs_stats_inc(&tcon->stats.cifs_stats.num_closes);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002517 if (rc) {
Steve French790fe572007-07-07 19:25:05 +00002518 if (rc != -EINTR) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002519 /* EINTR is expected when user ctl-c to kill app */
Joe Perchesb6b38f72010-04-21 03:50:45 +00002520 cERROR(1, "Send error in Close = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002521 }
2522 }
2523
Linus Torvalds1da177e2005-04-16 15:20:36 -07002524 /* Since session is dead, file will be closed on server already */
Steve French790fe572007-07-07 19:25:05 +00002525 if (rc == -EAGAIN)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002526 rc = 0;
2527
2528 return rc;
2529}
2530
2531int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04002532CIFSSMBFlush(const unsigned int xid, struct cifs_tcon *tcon, int smb_file_id)
Steve Frenchb298f222009-02-21 21:17:43 +00002533{
2534 int rc = 0;
2535 FLUSH_REQ *pSMB = NULL;
Joe Perchesb6b38f72010-04-21 03:50:45 +00002536 cFYI(1, "In CIFSSMBFlush");
Steve Frenchb298f222009-02-21 21:17:43 +00002537
2538 rc = small_smb_init(SMB_COM_FLUSH, 1, tcon, (void **) &pSMB);
2539 if (rc)
2540 return rc;
2541
2542 pSMB->FileID = (__u16) smb_file_id;
2543 pSMB->ByteCount = 0;
Pavel Shilovsky792af7b2012-03-23 14:28:02 -04002544 rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04002545 cifs_stats_inc(&tcon->stats.cifs_stats.num_flushes);
Steve Frenchb298f222009-02-21 21:17:43 +00002546 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00002547 cERROR(1, "Send error in Flush = %d", rc);
Steve Frenchb298f222009-02-21 21:17:43 +00002548
2549 return rc;
2550}
2551
2552int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04002553CIFSSMBRename(const unsigned int xid, struct cifs_tcon *tcon,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002554 const char *fromName, const char *toName,
Steve French737b7582005-04-28 22:41:06 -07002555 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002556{
2557 int rc = 0;
2558 RENAME_REQ *pSMB = NULL;
2559 RENAME_RSP *pSMBr = NULL;
2560 int bytes_returned;
2561 int name_len, name_len2;
2562 __u16 count;
2563
Joe Perchesb6b38f72010-04-21 03:50:45 +00002564 cFYI(1, "In CIFSSMBRename");
Linus Torvalds1da177e2005-04-16 15:20:36 -07002565renameRetry:
2566 rc = smb_init(SMB_COM_RENAME, 1, tcon, (void **) &pSMB,
2567 (void **) &pSMBr);
2568 if (rc)
2569 return rc;
2570
2571 pSMB->BufferFormat = 0x04;
2572 pSMB->SearchAttributes =
2573 cpu_to_le16(ATTR_READONLY | ATTR_HIDDEN | ATTR_SYSTEM |
2574 ATTR_DIRECTORY);
2575
2576 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
2577 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06002578 cifsConvertToUTF16((__le16 *) pSMB->OldFileName, fromName,
2579 PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002580 name_len++; /* trailing null */
2581 name_len *= 2;
2582 pSMB->OldFileName[name_len] = 0x04; /* pad */
2583 /* protocol requires ASCII signature byte on Unicode string */
2584 pSMB->OldFileName[name_len + 1] = 0x00;
2585 name_len2 =
Steve Frenchacbbb762012-01-18 22:32:33 -06002586 cifsConvertToUTF16((__le16 *)&pSMB->OldFileName[name_len+2],
2587 toName, PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002588 name_len2 += 1 /* trailing null */ + 1 /* Signature word */ ;
2589 name_len2 *= 2; /* convert to bytes */
Steve French50c2f752007-07-13 00:33:32 +00002590 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002591 name_len = strnlen(fromName, PATH_MAX);
2592 name_len++; /* trailing null */
2593 strncpy(pSMB->OldFileName, fromName, name_len);
2594 name_len2 = strnlen(toName, PATH_MAX);
2595 name_len2++; /* trailing null */
2596 pSMB->OldFileName[name_len] = 0x04; /* 2nd buffer format */
2597 strncpy(&pSMB->OldFileName[name_len + 1], toName, name_len2);
2598 name_len2++; /* trailing null */
2599 name_len2++; /* signature byte */
2600 }
2601
2602 count = 1 /* 1st signature byte */ + name_len + name_len2;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00002603 inc_rfc1001_len(pSMB, count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002604 pSMB->ByteCount = cpu_to_le16(count);
2605
2606 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
2607 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04002608 cifs_stats_inc(&tcon->stats.cifs_stats.num_renames);
Steve Frenchad7a2922008-02-07 23:25:02 +00002609 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00002610 cFYI(1, "Send error in rename = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002611
Linus Torvalds1da177e2005-04-16 15:20:36 -07002612 cifs_buf_release(pSMB);
2613
2614 if (rc == -EAGAIN)
2615 goto renameRetry;
2616
2617 return rc;
2618}
2619
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04002620int CIFSSMBRenameOpenFile(const unsigned int xid, struct cifs_tcon *pTcon,
Jeff Layton391e5752008-09-24 11:32:59 -04002621 int netfid, const char *target_name,
Steve French50c2f752007-07-13 00:33:32 +00002622 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002623{
2624 struct smb_com_transaction2_sfi_req *pSMB = NULL;
2625 struct smb_com_transaction2_sfi_rsp *pSMBr = NULL;
Steve French50c2f752007-07-13 00:33:32 +00002626 struct set_file_rename *rename_info;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002627 char *data_offset;
2628 char dummy_string[30];
2629 int rc = 0;
2630 int bytes_returned = 0;
2631 int len_of_str;
2632 __u16 params, param_offset, offset, count, byte_count;
2633
Joe Perchesb6b38f72010-04-21 03:50:45 +00002634 cFYI(1, "Rename to File by handle");
Linus Torvalds1da177e2005-04-16 15:20:36 -07002635 rc = smb_init(SMB_COM_TRANSACTION2, 15, pTcon, (void **) &pSMB,
2636 (void **) &pSMBr);
2637 if (rc)
2638 return rc;
2639
2640 params = 6;
2641 pSMB->MaxSetupCount = 0;
2642 pSMB->Reserved = 0;
2643 pSMB->Flags = 0;
2644 pSMB->Timeout = 0;
2645 pSMB->Reserved2 = 0;
2646 param_offset = offsetof(struct smb_com_transaction2_sfi_req, Fid) - 4;
2647 offset = param_offset + params;
2648
2649 data_offset = (char *) (&pSMB->hdr.Protocol) + offset;
2650 rename_info = (struct set_file_rename *) data_offset;
2651 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve Frenchad7a2922008-02-07 23:25:02 +00002652 pSMB->MaxDataCount = cpu_to_le16(1000); /* BB find max SMB from sess */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002653 pSMB->SetupCount = 1;
2654 pSMB->Reserved3 = 0;
2655 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FILE_INFORMATION);
2656 byte_count = 3 /* pad */ + params;
2657 pSMB->ParameterCount = cpu_to_le16(params);
2658 pSMB->TotalParameterCount = pSMB->ParameterCount;
2659 pSMB->ParameterOffset = cpu_to_le16(param_offset);
2660 pSMB->DataOffset = cpu_to_le16(offset);
2661 /* construct random name ".cifs_tmp<inodenum><mid>" */
2662 rename_info->overwrite = cpu_to_le32(1);
2663 rename_info->root_fid = 0;
2664 /* unicode only call */
Steve French790fe572007-07-07 19:25:05 +00002665 if (target_name == NULL) {
Steve French50c2f752007-07-13 00:33:32 +00002666 sprintf(dummy_string, "cifs%x", pSMB->hdr.Mid);
Steve Frenchacbbb762012-01-18 22:32:33 -06002667 len_of_str =
2668 cifsConvertToUTF16((__le16 *)rename_info->target_name,
Steve French737b7582005-04-28 22:41:06 -07002669 dummy_string, 24, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002670 } else {
Steve Frenchacbbb762012-01-18 22:32:33 -06002671 len_of_str =
2672 cifsConvertToUTF16((__le16 *)rename_info->target_name,
Steve French50c2f752007-07-13 00:33:32 +00002673 target_name, PATH_MAX, nls_codepage,
2674 remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002675 }
2676 rename_info->target_name_len = cpu_to_le32(2 * len_of_str);
Jeff Layton391e5752008-09-24 11:32:59 -04002677 count = 12 /* sizeof(struct set_file_rename) */ + (2 * len_of_str);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002678 byte_count += count;
2679 pSMB->DataCount = cpu_to_le16(count);
2680 pSMB->TotalDataCount = pSMB->DataCount;
2681 pSMB->Fid = netfid;
2682 pSMB->InformationLevel =
2683 cpu_to_le16(SMB_SET_FILE_RENAME_INFORMATION);
2684 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00002685 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002686 pSMB->ByteCount = cpu_to_le16(byte_count);
2687 rc = SendReceive(xid, pTcon->ses, (struct smb_hdr *) pSMB,
Steve French50c2f752007-07-13 00:33:32 +00002688 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04002689 cifs_stats_inc(&pTcon->stats.cifs_stats.num_t2renames);
Steve Frenchad7a2922008-02-07 23:25:02 +00002690 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00002691 cFYI(1, "Send error in Rename (by file handle) = %d", rc);
Steve Frencha5a2b482005-08-20 21:42:53 -07002692
Linus Torvalds1da177e2005-04-16 15:20:36 -07002693 cifs_buf_release(pSMB);
2694
2695 /* Note: On -EAGAIN error only caller can retry on handle based calls
2696 since file handle passed in no longer valid */
2697
2698 return rc;
2699}
2700
2701int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04002702CIFSSMBCopy(const unsigned int xid, struct cifs_tcon *tcon,
2703 const char *fromName, const __u16 target_tid, const char *toName,
2704 const int flags, const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002705{
2706 int rc = 0;
2707 COPY_REQ *pSMB = NULL;
2708 COPY_RSP *pSMBr = NULL;
2709 int bytes_returned;
2710 int name_len, name_len2;
2711 __u16 count;
2712
Joe Perchesb6b38f72010-04-21 03:50:45 +00002713 cFYI(1, "In CIFSSMBCopy");
Linus Torvalds1da177e2005-04-16 15:20:36 -07002714copyRetry:
2715 rc = smb_init(SMB_COM_COPY, 1, tcon, (void **) &pSMB,
2716 (void **) &pSMBr);
2717 if (rc)
2718 return rc;
2719
2720 pSMB->BufferFormat = 0x04;
2721 pSMB->Tid2 = target_tid;
2722
2723 pSMB->Flags = cpu_to_le16(flags & COPY_TREE);
2724
2725 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
Steve Frenchacbbb762012-01-18 22:32:33 -06002726 name_len = cifsConvertToUTF16((__le16 *) pSMB->OldFileName,
2727 fromName, PATH_MAX, nls_codepage,
2728 remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002729 name_len++; /* trailing null */
2730 name_len *= 2;
2731 pSMB->OldFileName[name_len] = 0x04; /* pad */
2732 /* protocol requires ASCII signature byte on Unicode string */
2733 pSMB->OldFileName[name_len + 1] = 0x00;
Steve French50c2f752007-07-13 00:33:32 +00002734 name_len2 =
Steve Frenchacbbb762012-01-18 22:32:33 -06002735 cifsConvertToUTF16((__le16 *)&pSMB->OldFileName[name_len+2],
2736 toName, PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002737 name_len2 += 1 /* trailing null */ + 1 /* Signature word */ ;
2738 name_len2 *= 2; /* convert to bytes */
Steve French50c2f752007-07-13 00:33:32 +00002739 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002740 name_len = strnlen(fromName, PATH_MAX);
2741 name_len++; /* trailing null */
2742 strncpy(pSMB->OldFileName, fromName, name_len);
2743 name_len2 = strnlen(toName, PATH_MAX);
2744 name_len2++; /* trailing null */
2745 pSMB->OldFileName[name_len] = 0x04; /* 2nd buffer format */
2746 strncpy(&pSMB->OldFileName[name_len + 1], toName, name_len2);
2747 name_len2++; /* trailing null */
2748 name_len2++; /* signature byte */
2749 }
2750
2751 count = 1 /* 1st signature byte */ + name_len + name_len2;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00002752 inc_rfc1001_len(pSMB, count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002753 pSMB->ByteCount = cpu_to_le16(count);
2754
2755 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
2756 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
2757 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00002758 cFYI(1, "Send error in copy = %d with %d files copied",
2759 rc, le16_to_cpu(pSMBr->CopyCount));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002760 }
Steve French0d817bc2008-05-22 02:02:03 +00002761 cifs_buf_release(pSMB);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002762
2763 if (rc == -EAGAIN)
2764 goto copyRetry;
2765
2766 return rc;
2767}
2768
2769int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04002770CIFSUnixCreateSymLink(const unsigned int xid, struct cifs_tcon *tcon,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002771 const char *fromName, const char *toName,
2772 const struct nls_table *nls_codepage)
2773{
2774 TRANSACTION2_SPI_REQ *pSMB = NULL;
2775 TRANSACTION2_SPI_RSP *pSMBr = NULL;
2776 char *data_offset;
2777 int name_len;
2778 int name_len_target;
2779 int rc = 0;
2780 int bytes_returned = 0;
2781 __u16 params, param_offset, offset, byte_count;
2782
Joe Perchesb6b38f72010-04-21 03:50:45 +00002783 cFYI(1, "In Symlink Unix style");
Linus Torvalds1da177e2005-04-16 15:20:36 -07002784createSymLinkRetry:
2785 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
2786 (void **) &pSMBr);
2787 if (rc)
2788 return rc;
2789
2790 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
2791 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06002792 cifs_strtoUTF16((__le16 *) pSMB->FileName, fromName,
2793 /* find define for this maxpathcomponent */
2794 PATH_MAX, nls_codepage);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002795 name_len++; /* trailing null */
2796 name_len *= 2;
2797
Steve French50c2f752007-07-13 00:33:32 +00002798 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002799 name_len = strnlen(fromName, PATH_MAX);
2800 name_len++; /* trailing null */
2801 strncpy(pSMB->FileName, fromName, name_len);
2802 }
2803 params = 6 + name_len;
2804 pSMB->MaxSetupCount = 0;
2805 pSMB->Reserved = 0;
2806 pSMB->Flags = 0;
2807 pSMB->Timeout = 0;
2808 pSMB->Reserved2 = 0;
2809 param_offset = offsetof(struct smb_com_transaction2_spi_req,
Steve French50c2f752007-07-13 00:33:32 +00002810 InformationLevel) - 4;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002811 offset = param_offset + params;
2812
2813 data_offset = (char *) (&pSMB->hdr.Protocol) + offset;
2814 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
2815 name_len_target =
Steve Frenchacbbb762012-01-18 22:32:33 -06002816 cifs_strtoUTF16((__le16 *) data_offset, toName, PATH_MAX
2817 /* find define for this maxpathcomponent */
2818 , nls_codepage);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002819 name_len_target++; /* trailing null */
2820 name_len_target *= 2;
Steve French50c2f752007-07-13 00:33:32 +00002821 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002822 name_len_target = strnlen(toName, PATH_MAX);
2823 name_len_target++; /* trailing null */
2824 strncpy(data_offset, toName, name_len_target);
2825 }
2826
2827 pSMB->MaxParameterCount = cpu_to_le16(2);
2828 /* BB find exact max on data count below from sess */
2829 pSMB->MaxDataCount = cpu_to_le16(1000);
2830 pSMB->SetupCount = 1;
2831 pSMB->Reserved3 = 0;
2832 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
2833 byte_count = 3 /* pad */ + params + name_len_target;
2834 pSMB->DataCount = cpu_to_le16(name_len_target);
2835 pSMB->ParameterCount = cpu_to_le16(params);
2836 pSMB->TotalDataCount = pSMB->DataCount;
2837 pSMB->TotalParameterCount = pSMB->ParameterCount;
2838 pSMB->ParameterOffset = cpu_to_le16(param_offset);
2839 pSMB->DataOffset = cpu_to_le16(offset);
2840 pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_UNIX_LINK);
2841 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00002842 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002843 pSMB->ByteCount = cpu_to_le16(byte_count);
2844 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
2845 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04002846 cifs_stats_inc(&tcon->stats.cifs_stats.num_symlinks);
Steve Frenchad7a2922008-02-07 23:25:02 +00002847 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00002848 cFYI(1, "Send error in SetPathInfo create symlink = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002849
Steve French0d817bc2008-05-22 02:02:03 +00002850 cifs_buf_release(pSMB);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002851
2852 if (rc == -EAGAIN)
2853 goto createSymLinkRetry;
2854
2855 return rc;
2856}
2857
2858int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04002859CIFSUnixCreateHardLink(const unsigned int xid, struct cifs_tcon *tcon,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002860 const char *fromName, const char *toName,
Steve French737b7582005-04-28 22:41:06 -07002861 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002862{
2863 TRANSACTION2_SPI_REQ *pSMB = NULL;
2864 TRANSACTION2_SPI_RSP *pSMBr = NULL;
2865 char *data_offset;
2866 int name_len;
2867 int name_len_target;
2868 int rc = 0;
2869 int bytes_returned = 0;
2870 __u16 params, param_offset, offset, byte_count;
2871
Joe Perchesb6b38f72010-04-21 03:50:45 +00002872 cFYI(1, "In Create Hard link Unix style");
Linus Torvalds1da177e2005-04-16 15:20:36 -07002873createHardLinkRetry:
2874 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
2875 (void **) &pSMBr);
2876 if (rc)
2877 return rc;
2878
2879 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
Steve Frenchacbbb762012-01-18 22:32:33 -06002880 name_len = cifsConvertToUTF16((__le16 *) pSMB->FileName, toName,
2881 PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002882 name_len++; /* trailing null */
2883 name_len *= 2;
2884
Steve French50c2f752007-07-13 00:33:32 +00002885 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002886 name_len = strnlen(toName, PATH_MAX);
2887 name_len++; /* trailing null */
2888 strncpy(pSMB->FileName, toName, name_len);
2889 }
2890 params = 6 + name_len;
2891 pSMB->MaxSetupCount = 0;
2892 pSMB->Reserved = 0;
2893 pSMB->Flags = 0;
2894 pSMB->Timeout = 0;
2895 pSMB->Reserved2 = 0;
2896 param_offset = offsetof(struct smb_com_transaction2_spi_req,
Steve French50c2f752007-07-13 00:33:32 +00002897 InformationLevel) - 4;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002898 offset = param_offset + params;
2899
2900 data_offset = (char *) (&pSMB->hdr.Protocol) + offset;
2901 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
2902 name_len_target =
Steve Frenchacbbb762012-01-18 22:32:33 -06002903 cifsConvertToUTF16((__le16 *) data_offset, fromName,
2904 PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002905 name_len_target++; /* trailing null */
2906 name_len_target *= 2;
Steve French50c2f752007-07-13 00:33:32 +00002907 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002908 name_len_target = strnlen(fromName, PATH_MAX);
2909 name_len_target++; /* trailing null */
2910 strncpy(data_offset, fromName, name_len_target);
2911 }
2912
2913 pSMB->MaxParameterCount = cpu_to_le16(2);
2914 /* BB find exact max on data count below from sess*/
2915 pSMB->MaxDataCount = cpu_to_le16(1000);
2916 pSMB->SetupCount = 1;
2917 pSMB->Reserved3 = 0;
2918 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
2919 byte_count = 3 /* pad */ + params + name_len_target;
2920 pSMB->ParameterCount = cpu_to_le16(params);
2921 pSMB->TotalParameterCount = pSMB->ParameterCount;
2922 pSMB->DataCount = cpu_to_le16(name_len_target);
2923 pSMB->TotalDataCount = pSMB->DataCount;
2924 pSMB->ParameterOffset = cpu_to_le16(param_offset);
2925 pSMB->DataOffset = cpu_to_le16(offset);
2926 pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_UNIX_HLINK);
2927 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00002928 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002929 pSMB->ByteCount = cpu_to_le16(byte_count);
2930 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
2931 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04002932 cifs_stats_inc(&tcon->stats.cifs_stats.num_hardlinks);
Steve Frenchad7a2922008-02-07 23:25:02 +00002933 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00002934 cFYI(1, "Send error in SetPathInfo (hard link) = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002935
2936 cifs_buf_release(pSMB);
2937 if (rc == -EAGAIN)
2938 goto createHardLinkRetry;
2939
2940 return rc;
2941}
2942
2943int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04002944CIFSCreateHardLink(const unsigned int xid, struct cifs_tcon *tcon,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002945 const char *fromName, const char *toName,
Steve French737b7582005-04-28 22:41:06 -07002946 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002947{
2948 int rc = 0;
2949 NT_RENAME_REQ *pSMB = NULL;
2950 RENAME_RSP *pSMBr = NULL;
2951 int bytes_returned;
2952 int name_len, name_len2;
2953 __u16 count;
2954
Joe Perchesb6b38f72010-04-21 03:50:45 +00002955 cFYI(1, "In CIFSCreateHardLink");
Linus Torvalds1da177e2005-04-16 15:20:36 -07002956winCreateHardLinkRetry:
2957
2958 rc = smb_init(SMB_COM_NT_RENAME, 4, tcon, (void **) &pSMB,
2959 (void **) &pSMBr);
2960 if (rc)
2961 return rc;
2962
2963 pSMB->SearchAttributes =
2964 cpu_to_le16(ATTR_READONLY | ATTR_HIDDEN | ATTR_SYSTEM |
2965 ATTR_DIRECTORY);
2966 pSMB->Flags = cpu_to_le16(CREATE_HARD_LINK);
2967 pSMB->ClusterCount = 0;
2968
2969 pSMB->BufferFormat = 0x04;
2970
2971 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
2972 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06002973 cifsConvertToUTF16((__le16 *) pSMB->OldFileName, fromName,
2974 PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002975 name_len++; /* trailing null */
2976 name_len *= 2;
Jeff Laytonfcc7c092009-02-28 12:59:03 -05002977
2978 /* protocol specifies ASCII buffer format (0x04) for unicode */
2979 pSMB->OldFileName[name_len] = 0x04;
2980 pSMB->OldFileName[name_len + 1] = 0x00; /* pad */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002981 name_len2 =
Steve Frenchacbbb762012-01-18 22:32:33 -06002982 cifsConvertToUTF16((__le16 *)&pSMB->OldFileName[name_len+2],
2983 toName, PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002984 name_len2 += 1 /* trailing null */ + 1 /* Signature word */ ;
2985 name_len2 *= 2; /* convert to bytes */
Steve French50c2f752007-07-13 00:33:32 +00002986 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002987 name_len = strnlen(fromName, PATH_MAX);
2988 name_len++; /* trailing null */
2989 strncpy(pSMB->OldFileName, fromName, name_len);
2990 name_len2 = strnlen(toName, PATH_MAX);
2991 name_len2++; /* trailing null */
2992 pSMB->OldFileName[name_len] = 0x04; /* 2nd buffer format */
2993 strncpy(&pSMB->OldFileName[name_len + 1], toName, name_len2);
2994 name_len2++; /* trailing null */
2995 name_len2++; /* signature byte */
2996 }
2997
2998 count = 1 /* string type byte */ + name_len + name_len2;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00002999 inc_rfc1001_len(pSMB, count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003000 pSMB->ByteCount = cpu_to_le16(count);
3001
3002 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
3003 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04003004 cifs_stats_inc(&tcon->stats.cifs_stats.num_hardlinks);
Steve Frenchad7a2922008-02-07 23:25:02 +00003005 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00003006 cFYI(1, "Send error in hard link (NT rename) = %d", rc);
Steve Frenchad7a2922008-02-07 23:25:02 +00003007
Linus Torvalds1da177e2005-04-16 15:20:36 -07003008 cifs_buf_release(pSMB);
3009 if (rc == -EAGAIN)
3010 goto winCreateHardLinkRetry;
3011
3012 return rc;
3013}
3014
3015int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04003016CIFSSMBUnixQuerySymLink(const unsigned int xid, struct cifs_tcon *tcon,
Jeff Layton460b9692009-04-30 07:17:56 -04003017 const unsigned char *searchName, char **symlinkinfo,
Linus Torvalds1da177e2005-04-16 15:20:36 -07003018 const struct nls_table *nls_codepage)
3019{
3020/* SMB_QUERY_FILE_UNIX_LINK */
3021 TRANSACTION2_QPI_REQ *pSMB = NULL;
3022 TRANSACTION2_QPI_RSP *pSMBr = NULL;
3023 int rc = 0;
3024 int bytes_returned;
3025 int name_len;
3026 __u16 params, byte_count;
Jeff Layton460b9692009-04-30 07:17:56 -04003027 char *data_start;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003028
Joe Perchesb6b38f72010-04-21 03:50:45 +00003029 cFYI(1, "In QPathSymLinkInfo (Unix) for path %s", searchName);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003030
3031querySymLinkRetry:
3032 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
3033 (void **) &pSMBr);
3034 if (rc)
3035 return rc;
3036
3037 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
3038 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06003039 cifs_strtoUTF16((__le16 *) pSMB->FileName, searchName,
3040 PATH_MAX, nls_codepage);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003041 name_len++; /* trailing null */
3042 name_len *= 2;
Steve French50c2f752007-07-13 00:33:32 +00003043 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07003044 name_len = strnlen(searchName, PATH_MAX);
3045 name_len++; /* trailing null */
3046 strncpy(pSMB->FileName, searchName, name_len);
3047 }
3048
3049 params = 2 /* level */ + 4 /* rsrvd */ + name_len /* incl null */ ;
3050 pSMB->TotalDataCount = 0;
3051 pSMB->MaxParameterCount = cpu_to_le16(2);
Jeff Layton46a75742009-05-24 18:45:17 -04003052 pSMB->MaxDataCount = cpu_to_le16(CIFSMaxBufSize);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003053 pSMB->MaxSetupCount = 0;
3054 pSMB->Reserved = 0;
3055 pSMB->Flags = 0;
3056 pSMB->Timeout = 0;
3057 pSMB->Reserved2 = 0;
3058 pSMB->ParameterOffset = cpu_to_le16(offsetof(
Steve French50c2f752007-07-13 00:33:32 +00003059 struct smb_com_transaction2_qpi_req, InformationLevel) - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003060 pSMB->DataCount = 0;
3061 pSMB->DataOffset = 0;
3062 pSMB->SetupCount = 1;
3063 pSMB->Reserved3 = 0;
3064 pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_PATH_INFORMATION);
3065 byte_count = params + 1 /* pad */ ;
3066 pSMB->TotalParameterCount = cpu_to_le16(params);
3067 pSMB->ParameterCount = pSMB->TotalParameterCount;
3068 pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FILE_UNIX_LINK);
3069 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00003070 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003071 pSMB->ByteCount = cpu_to_le16(byte_count);
3072
3073 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
3074 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
3075 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00003076 cFYI(1, "Send error in QuerySymLinkInfo = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003077 } else {
3078 /* decode response */
3079
3080 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003081 /* BB also check enough total bytes returned */
Jeff Layton820a8032011-05-04 08:05:26 -04003082 if (rc || get_bcc(&pSMBr->hdr) < 2)
Jeff Layton460b9692009-04-30 07:17:56 -04003083 rc = -EIO;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003084 else {
Steve French0e0d2cf2009-05-01 05:27:32 +00003085 bool is_unicode;
Jeff Layton460b9692009-04-30 07:17:56 -04003086 u16 count = le16_to_cpu(pSMBr->t2.DataCount);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003087
Jeff Layton460b9692009-04-30 07:17:56 -04003088 data_start = ((char *) &pSMBr->hdr.Protocol) +
3089 le16_to_cpu(pSMBr->t2.DataOffset);
3090
Steve French0e0d2cf2009-05-01 05:27:32 +00003091 if (pSMBr->hdr.Flags2 & SMBFLG2_UNICODE)
3092 is_unicode = true;
3093 else
3094 is_unicode = false;
3095
Steve French737b7582005-04-28 22:41:06 -07003096 /* BB FIXME investigate remapping reserved chars here */
Steve Frenchacbbb762012-01-18 22:32:33 -06003097 *symlinkinfo = cifs_strndup_from_utf16(data_start,
3098 count, is_unicode, nls_codepage);
Jeff Layton8b6427a2009-05-19 09:57:03 -04003099 if (!*symlinkinfo)
Jeff Layton460b9692009-04-30 07:17:56 -04003100 rc = -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003101 }
3102 }
3103 cifs_buf_release(pSMB);
3104 if (rc == -EAGAIN)
3105 goto querySymLinkRetry;
3106 return rc;
3107}
3108
Steve Frenchc52a95542011-02-24 06:16:22 +00003109#ifdef CONFIG_CIFS_SYMLINK_EXPERIMENTAL
3110/*
3111 * Recent Windows versions now create symlinks more frequently
3112 * and they use the "reparse point" mechanism below. We can of course
3113 * do symlinks nicely to Samba and other servers which support the
3114 * CIFS Unix Extensions and we can also do SFU symlinks and "client only"
3115 * "MF" symlinks optionally, but for recent Windows we really need to
3116 * reenable the code below and fix the cifs_symlink callers to handle this.
3117 * In the interim this code has been moved to its own config option so
3118 * it is not compiled in by default until callers fixed up and more tested.
3119 */
Linus Torvalds1da177e2005-04-16 15:20:36 -07003120int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04003121CIFSSMBQueryReparseLinkInfo(const unsigned int xid, struct cifs_tcon *tcon,
Linus Torvalds1da177e2005-04-16 15:20:36 -07003122 const unsigned char *searchName,
Steve French50c2f752007-07-13 00:33:32 +00003123 char *symlinkinfo, const int buflen, __u16 fid,
Linus Torvalds1da177e2005-04-16 15:20:36 -07003124 const struct nls_table *nls_codepage)
3125{
3126 int rc = 0;
3127 int bytes_returned;
Steve French50c2f752007-07-13 00:33:32 +00003128 struct smb_com_transaction_ioctl_req *pSMB;
3129 struct smb_com_transaction_ioctl_rsp *pSMBr;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003130
Joe Perchesb6b38f72010-04-21 03:50:45 +00003131 cFYI(1, "In Windows reparse style QueryLink for path %s", searchName);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003132 rc = smb_init(SMB_COM_NT_TRANSACT, 23, tcon, (void **) &pSMB,
3133 (void **) &pSMBr);
3134 if (rc)
3135 return rc;
3136
3137 pSMB->TotalParameterCount = 0 ;
3138 pSMB->TotalDataCount = 0;
3139 pSMB->MaxParameterCount = cpu_to_le32(2);
3140 /* BB find exact data count max from sess structure BB */
Jeff Laytonc974bef2011-10-11 06:41:32 -04003141 pSMB->MaxDataCount = cpu_to_le32(CIFSMaxBufSize & 0xFFFFFF00);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003142 pSMB->MaxSetupCount = 4;
3143 pSMB->Reserved = 0;
3144 pSMB->ParameterOffset = 0;
3145 pSMB->DataCount = 0;
3146 pSMB->DataOffset = 0;
3147 pSMB->SetupCount = 4;
3148 pSMB->SubCommand = cpu_to_le16(NT_TRANSACT_IOCTL);
3149 pSMB->ParameterCount = pSMB->TotalParameterCount;
3150 pSMB->FunctionCode = cpu_to_le32(FSCTL_GET_REPARSE_POINT);
3151 pSMB->IsFsctl = 1; /* FSCTL */
3152 pSMB->IsRootFlag = 0;
3153 pSMB->Fid = fid; /* file handle always le */
3154 pSMB->ByteCount = 0;
3155
3156 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
3157 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
3158 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00003159 cFYI(1, "Send error in QueryReparseLinkInfo = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003160 } else { /* decode response */
3161 __u32 data_offset = le32_to_cpu(pSMBr->DataOffset);
3162 __u32 data_count = le32_to_cpu(pSMBr->DataCount);
Jeff Layton820a8032011-05-04 08:05:26 -04003163 if (get_bcc(&pSMBr->hdr) < 2 || data_offset > 512) {
3164 /* BB also check enough total bytes returned */
Linus Torvalds1da177e2005-04-16 15:20:36 -07003165 rc = -EIO; /* bad smb */
Steve Frenchafe48c32009-05-02 05:25:46 +00003166 goto qreparse_out;
3167 }
3168 if (data_count && (data_count < 2048)) {
3169 char *end_of_smb = 2 /* sizeof byte count */ +
Jeff Layton820a8032011-05-04 08:05:26 -04003170 get_bcc(&pSMBr->hdr) + (char *)&pSMBr->ByteCount;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003171
Steve Frenchafe48c32009-05-02 05:25:46 +00003172 struct reparse_data *reparse_buf =
Steve French50c2f752007-07-13 00:33:32 +00003173 (struct reparse_data *)
3174 ((char *)&pSMBr->hdr.Protocol
3175 + data_offset);
Steve Frenchafe48c32009-05-02 05:25:46 +00003176 if ((char *)reparse_buf >= end_of_smb) {
3177 rc = -EIO;
3178 goto qreparse_out;
3179 }
3180 if ((reparse_buf->LinkNamesBuf +
3181 reparse_buf->TargetNameOffset +
3182 reparse_buf->TargetNameLen) > end_of_smb) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00003183 cFYI(1, "reparse buf beyond SMB");
Steve Frenchafe48c32009-05-02 05:25:46 +00003184 rc = -EIO;
3185 goto qreparse_out;
3186 }
Steve French50c2f752007-07-13 00:33:32 +00003187
Steve Frenchafe48c32009-05-02 05:25:46 +00003188 if (pSMBr->hdr.Flags2 & SMBFLG2_UNICODE) {
3189 cifs_from_ucs2(symlinkinfo, (__le16 *)
Steve French50c2f752007-07-13 00:33:32 +00003190 (reparse_buf->LinkNamesBuf +
3191 reparse_buf->TargetNameOffset),
Steve Frenchafe48c32009-05-02 05:25:46 +00003192 buflen,
3193 reparse_buf->TargetNameLen,
3194 nls_codepage, 0);
3195 } else { /* ASCII names */
3196 strncpy(symlinkinfo,
3197 reparse_buf->LinkNamesBuf +
3198 reparse_buf->TargetNameOffset,
3199 min_t(const int, buflen,
3200 reparse_buf->TargetNameLen));
Linus Torvalds1da177e2005-04-16 15:20:36 -07003201 }
Steve Frenchafe48c32009-05-02 05:25:46 +00003202 } else {
3203 rc = -EIO;
Joe Perchesb6b38f72010-04-21 03:50:45 +00003204 cFYI(1, "Invalid return data count on "
3205 "get reparse info ioctl");
Linus Torvalds1da177e2005-04-16 15:20:36 -07003206 }
Steve Frenchafe48c32009-05-02 05:25:46 +00003207 symlinkinfo[buflen] = 0; /* just in case so the caller
3208 does not go off the end of the buffer */
Joe Perchesb6b38f72010-04-21 03:50:45 +00003209 cFYI(1, "readlink result - %s", symlinkinfo);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003210 }
Steve French989c7e52009-05-02 05:32:20 +00003211
Linus Torvalds1da177e2005-04-16 15:20:36 -07003212qreparse_out:
Steve French4a6d87f2005-08-13 08:15:54 -07003213 cifs_buf_release(pSMB);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003214
3215 /* Note: On -EAGAIN error only caller can retry on handle based calls
3216 since file handle passed in no longer valid */
3217
3218 return rc;
3219}
Steve Frenchc52a95542011-02-24 06:16:22 +00003220#endif /* CIFS_SYMLINK_EXPERIMENTAL */ /* BB temporarily unused */
Linus Torvalds1da177e2005-04-16 15:20:36 -07003221
3222#ifdef CONFIG_CIFS_POSIX
3223
3224/*Convert an Access Control Entry from wire format to local POSIX xattr format*/
Steve French50c2f752007-07-13 00:33:32 +00003225static void cifs_convert_ace(posix_acl_xattr_entry *ace,
3226 struct cifs_posix_ace *cifs_ace)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003227{
3228 /* u8 cifs fields do not need le conversion */
Steve Frenchff7feac2005-11-15 16:45:16 -08003229 ace->e_perm = cpu_to_le16(cifs_ace->cifs_e_perm);
3230 ace->e_tag = cpu_to_le16(cifs_ace->cifs_e_tag);
3231 ace->e_id = cpu_to_le32(le64_to_cpu(cifs_ace->cifs_uid));
Joe Perchesb6b38f72010-04-21 03:50:45 +00003232 /* cFYI(1, "perm %d tag %d id %d",ace->e_perm,ace->e_tag,ace->e_id); */
Linus Torvalds1da177e2005-04-16 15:20:36 -07003233
3234 return;
3235}
3236
3237/* Convert ACL from CIFS POSIX wire format to local Linux POSIX ACL xattr */
Steve French50c2f752007-07-13 00:33:32 +00003238static int cifs_copy_posix_acl(char *trgt, char *src, const int buflen,
3239 const int acl_type, const int size_of_data_area)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003240{
3241 int size = 0;
3242 int i;
3243 __u16 count;
Steve French50c2f752007-07-13 00:33:32 +00003244 struct cifs_posix_ace *pACE;
3245 struct cifs_posix_acl *cifs_acl = (struct cifs_posix_acl *)src;
3246 posix_acl_xattr_header *local_acl = (posix_acl_xattr_header *)trgt;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003247
3248 if (le16_to_cpu(cifs_acl->version) != CIFS_ACL_VERSION)
3249 return -EOPNOTSUPP;
3250
Steve French790fe572007-07-07 19:25:05 +00003251 if (acl_type & ACL_TYPE_ACCESS) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003252 count = le16_to_cpu(cifs_acl->access_entry_count);
3253 pACE = &cifs_acl->ace_array[0];
3254 size = sizeof(struct cifs_posix_acl);
3255 size += sizeof(struct cifs_posix_ace) * count;
3256 /* check if we would go beyond end of SMB */
Steve French790fe572007-07-07 19:25:05 +00003257 if (size_of_data_area < size) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00003258 cFYI(1, "bad CIFS POSIX ACL size %d vs. %d",
3259 size_of_data_area, size);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003260 return -EINVAL;
3261 }
Steve French790fe572007-07-07 19:25:05 +00003262 } else if (acl_type & ACL_TYPE_DEFAULT) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003263 count = le16_to_cpu(cifs_acl->access_entry_count);
3264 size = sizeof(struct cifs_posix_acl);
3265 size += sizeof(struct cifs_posix_ace) * count;
3266/* skip past access ACEs to get to default ACEs */
3267 pACE = &cifs_acl->ace_array[count];
3268 count = le16_to_cpu(cifs_acl->default_entry_count);
3269 size += sizeof(struct cifs_posix_ace) * count;
3270 /* check if we would go beyond end of SMB */
Steve French790fe572007-07-07 19:25:05 +00003271 if (size_of_data_area < size)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003272 return -EINVAL;
3273 } else {
3274 /* illegal type */
3275 return -EINVAL;
3276 }
3277
3278 size = posix_acl_xattr_size(count);
Steve French790fe572007-07-07 19:25:05 +00003279 if ((buflen == 0) || (local_acl == NULL)) {
Steve French50c2f752007-07-13 00:33:32 +00003280 /* used to query ACL EA size */
Steve French790fe572007-07-07 19:25:05 +00003281 } else if (size > buflen) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003282 return -ERANGE;
3283 } else /* buffer big enough */ {
Steve Frenchff7feac2005-11-15 16:45:16 -08003284 local_acl->a_version = cpu_to_le32(POSIX_ACL_XATTR_VERSION);
Steve French50c2f752007-07-13 00:33:32 +00003285 for (i = 0; i < count ; i++) {
3286 cifs_convert_ace(&local_acl->a_entries[i], pACE);
3287 pACE++;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003288 }
3289 }
3290 return size;
3291}
3292
Steve French50c2f752007-07-13 00:33:32 +00003293static __u16 convert_ace_to_cifs_ace(struct cifs_posix_ace *cifs_ace,
3294 const posix_acl_xattr_entry *local_ace)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003295{
3296 __u16 rc = 0; /* 0 = ACL converted ok */
3297
Steve Frenchff7feac2005-11-15 16:45:16 -08003298 cifs_ace->cifs_e_perm = le16_to_cpu(local_ace->e_perm);
3299 cifs_ace->cifs_e_tag = le16_to_cpu(local_ace->e_tag);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003300 /* BB is there a better way to handle the large uid? */
Steve French790fe572007-07-07 19:25:05 +00003301 if (local_ace->e_id == cpu_to_le32(-1)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003302 /* Probably no need to le convert -1 on any arch but can not hurt */
3303 cifs_ace->cifs_uid = cpu_to_le64(-1);
Steve French50c2f752007-07-13 00:33:32 +00003304 } else
Steve Frenchff7feac2005-11-15 16:45:16 -08003305 cifs_ace->cifs_uid = cpu_to_le64(le32_to_cpu(local_ace->e_id));
Joe Perchesb6b38f72010-04-21 03:50:45 +00003306 /*cFYI(1, "perm %d tag %d id %d",ace->e_perm,ace->e_tag,ace->e_id);*/
Linus Torvalds1da177e2005-04-16 15:20:36 -07003307 return rc;
3308}
3309
3310/* Convert ACL from local Linux POSIX xattr to CIFS POSIX ACL wire format */
Steve French50c2f752007-07-13 00:33:32 +00003311static __u16 ACL_to_cifs_posix(char *parm_data, const char *pACL,
3312 const int buflen, const int acl_type)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003313{
3314 __u16 rc = 0;
Steve French50c2f752007-07-13 00:33:32 +00003315 struct cifs_posix_acl *cifs_acl = (struct cifs_posix_acl *)parm_data;
3316 posix_acl_xattr_header *local_acl = (posix_acl_xattr_header *)pACL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003317 int count;
3318 int i;
3319
Steve French790fe572007-07-07 19:25:05 +00003320 if ((buflen == 0) || (pACL == NULL) || (cifs_acl == NULL))
Linus Torvalds1da177e2005-04-16 15:20:36 -07003321 return 0;
3322
3323 count = posix_acl_xattr_count((size_t)buflen);
Joe Perchesb6b38f72010-04-21 03:50:45 +00003324 cFYI(1, "setting acl with %d entries from buf of length %d and "
Steve French63135e02007-07-17 17:34:02 +00003325 "version of %d",
Joe Perchesb6b38f72010-04-21 03:50:45 +00003326 count, buflen, le32_to_cpu(local_acl->a_version));
Steve French790fe572007-07-07 19:25:05 +00003327 if (le32_to_cpu(local_acl->a_version) != 2) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00003328 cFYI(1, "unknown POSIX ACL version %d",
3329 le32_to_cpu(local_acl->a_version));
Linus Torvalds1da177e2005-04-16 15:20:36 -07003330 return 0;
3331 }
3332 cifs_acl->version = cpu_to_le16(1);
Steve French790fe572007-07-07 19:25:05 +00003333 if (acl_type == ACL_TYPE_ACCESS)
Steve Frenchff7feac2005-11-15 16:45:16 -08003334 cifs_acl->access_entry_count = cpu_to_le16(count);
Steve French790fe572007-07-07 19:25:05 +00003335 else if (acl_type == ACL_TYPE_DEFAULT)
Steve Frenchff7feac2005-11-15 16:45:16 -08003336 cifs_acl->default_entry_count = cpu_to_le16(count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003337 else {
Joe Perchesb6b38f72010-04-21 03:50:45 +00003338 cFYI(1, "unknown ACL type %d", acl_type);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003339 return 0;
3340 }
Steve French50c2f752007-07-13 00:33:32 +00003341 for (i = 0; i < count; i++) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003342 rc = convert_ace_to_cifs_ace(&cifs_acl->ace_array[i],
3343 &local_acl->a_entries[i]);
Steve French790fe572007-07-07 19:25:05 +00003344 if (rc != 0) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003345 /* ACE not converted */
3346 break;
3347 }
3348 }
Steve French790fe572007-07-07 19:25:05 +00003349 if (rc == 0) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003350 rc = (__u16)(count * sizeof(struct cifs_posix_ace));
3351 rc += sizeof(struct cifs_posix_acl);
3352 /* BB add check to make sure ACL does not overflow SMB */
3353 }
3354 return rc;
3355}
3356
3357int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04003358CIFSSMBGetPosixACL(const unsigned int xid, struct cifs_tcon *tcon,
Steve French50c2f752007-07-13 00:33:32 +00003359 const unsigned char *searchName,
3360 char *acl_inf, const int buflen, const int acl_type,
3361 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003362{
3363/* SMB_QUERY_POSIX_ACL */
3364 TRANSACTION2_QPI_REQ *pSMB = NULL;
3365 TRANSACTION2_QPI_RSP *pSMBr = NULL;
3366 int rc = 0;
3367 int bytes_returned;
3368 int name_len;
3369 __u16 params, byte_count;
Steve French50c2f752007-07-13 00:33:32 +00003370
Joe Perchesb6b38f72010-04-21 03:50:45 +00003371 cFYI(1, "In GetPosixACL (Unix) for path %s", searchName);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003372
3373queryAclRetry:
3374 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
3375 (void **) &pSMBr);
3376 if (rc)
3377 return rc;
Steve French50c2f752007-07-13 00:33:32 +00003378
Linus Torvalds1da177e2005-04-16 15:20:36 -07003379 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
3380 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06003381 cifsConvertToUTF16((__le16 *) pSMB->FileName,
3382 searchName, PATH_MAX, nls_codepage,
3383 remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003384 name_len++; /* trailing null */
3385 name_len *= 2;
3386 pSMB->FileName[name_len] = 0;
3387 pSMB->FileName[name_len+1] = 0;
Steve French50c2f752007-07-13 00:33:32 +00003388 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07003389 name_len = strnlen(searchName, PATH_MAX);
3390 name_len++; /* trailing null */
3391 strncpy(pSMB->FileName, searchName, name_len);
3392 }
3393
3394 params = 2 /* level */ + 4 /* rsrvd */ + name_len /* incl null */ ;
3395 pSMB->TotalDataCount = 0;
3396 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French50c2f752007-07-13 00:33:32 +00003397 /* BB find exact max data count below from sess structure BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07003398 pSMB->MaxDataCount = cpu_to_le16(4000);
3399 pSMB->MaxSetupCount = 0;
3400 pSMB->Reserved = 0;
3401 pSMB->Flags = 0;
3402 pSMB->Timeout = 0;
3403 pSMB->Reserved2 = 0;
3404 pSMB->ParameterOffset = cpu_to_le16(
Steve French50c2f752007-07-13 00:33:32 +00003405 offsetof(struct smb_com_transaction2_qpi_req,
3406 InformationLevel) - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003407 pSMB->DataCount = 0;
3408 pSMB->DataOffset = 0;
3409 pSMB->SetupCount = 1;
3410 pSMB->Reserved3 = 0;
3411 pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_PATH_INFORMATION);
3412 byte_count = params + 1 /* pad */ ;
3413 pSMB->TotalParameterCount = cpu_to_le16(params);
3414 pSMB->ParameterCount = pSMB->TotalParameterCount;
3415 pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_POSIX_ACL);
3416 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00003417 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003418 pSMB->ByteCount = cpu_to_le16(byte_count);
3419
3420 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
3421 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04003422 cifs_stats_inc(&tcon->stats.cifs_stats.num_acl_get);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003423 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00003424 cFYI(1, "Send error in Query POSIX ACL = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003425 } else {
3426 /* decode response */
Steve French50c2f752007-07-13 00:33:32 +00003427
Linus Torvalds1da177e2005-04-16 15:20:36 -07003428 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003429 /* BB also check enough total bytes returned */
Jeff Layton820a8032011-05-04 08:05:26 -04003430 if (rc || get_bcc(&pSMBr->hdr) < 2)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003431 rc = -EIO; /* bad smb */
3432 else {
3433 __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
3434 __u16 count = le16_to_cpu(pSMBr->t2.DataCount);
3435 rc = cifs_copy_posix_acl(acl_inf,
3436 (char *)&pSMBr->hdr.Protocol+data_offset,
Steve French50c2f752007-07-13 00:33:32 +00003437 buflen, acl_type, count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003438 }
3439 }
3440 cifs_buf_release(pSMB);
3441 if (rc == -EAGAIN)
3442 goto queryAclRetry;
3443 return rc;
3444}
3445
3446int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04003447CIFSSMBSetPosixACL(const unsigned int xid, struct cifs_tcon *tcon,
Steve French50c2f752007-07-13 00:33:32 +00003448 const unsigned char *fileName,
3449 const char *local_acl, const int buflen,
3450 const int acl_type,
3451 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003452{
3453 struct smb_com_transaction2_spi_req *pSMB = NULL;
3454 struct smb_com_transaction2_spi_rsp *pSMBr = NULL;
3455 char *parm_data;
3456 int name_len;
3457 int rc = 0;
3458 int bytes_returned = 0;
3459 __u16 params, byte_count, data_count, param_offset, offset;
3460
Joe Perchesb6b38f72010-04-21 03:50:45 +00003461 cFYI(1, "In SetPosixACL (Unix) for path %s", fileName);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003462setAclRetry:
3463 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
Steve French50c2f752007-07-13 00:33:32 +00003464 (void **) &pSMBr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003465 if (rc)
3466 return rc;
3467 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
3468 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06003469 cifsConvertToUTF16((__le16 *) pSMB->FileName, fileName,
3470 PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003471 name_len++; /* trailing null */
3472 name_len *= 2;
Steve French50c2f752007-07-13 00:33:32 +00003473 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07003474 name_len = strnlen(fileName, PATH_MAX);
3475 name_len++; /* trailing null */
3476 strncpy(pSMB->FileName, fileName, name_len);
3477 }
3478 params = 6 + name_len;
3479 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French582d21e2008-05-13 04:54:12 +00003480 /* BB find max SMB size from sess */
3481 pSMB->MaxDataCount = cpu_to_le16(1000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003482 pSMB->MaxSetupCount = 0;
3483 pSMB->Reserved = 0;
3484 pSMB->Flags = 0;
3485 pSMB->Timeout = 0;
3486 pSMB->Reserved2 = 0;
3487 param_offset = offsetof(struct smb_com_transaction2_spi_req,
Steve French50c2f752007-07-13 00:33:32 +00003488 InformationLevel) - 4;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003489 offset = param_offset + params;
3490 parm_data = ((char *) &pSMB->hdr.Protocol) + offset;
3491 pSMB->ParameterOffset = cpu_to_le16(param_offset);
3492
3493 /* convert to on the wire format for POSIX ACL */
Steve French50c2f752007-07-13 00:33:32 +00003494 data_count = ACL_to_cifs_posix(parm_data, local_acl, buflen, acl_type);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003495
Steve French790fe572007-07-07 19:25:05 +00003496 if (data_count == 0) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07003497 rc = -EOPNOTSUPP;
3498 goto setACLerrorExit;
3499 }
3500 pSMB->DataOffset = cpu_to_le16(offset);
3501 pSMB->SetupCount = 1;
3502 pSMB->Reserved3 = 0;
3503 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
3504 pSMB->InformationLevel = cpu_to_le16(SMB_SET_POSIX_ACL);
3505 byte_count = 3 /* pad */ + params + data_count;
3506 pSMB->DataCount = cpu_to_le16(data_count);
3507 pSMB->TotalDataCount = pSMB->DataCount;
3508 pSMB->ParameterCount = cpu_to_le16(params);
3509 pSMB->TotalParameterCount = pSMB->ParameterCount;
3510 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00003511 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003512 pSMB->ByteCount = cpu_to_le16(byte_count);
3513 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
Steve French50c2f752007-07-13 00:33:32 +00003514 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Steve Frenchad7a2922008-02-07 23:25:02 +00003515 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00003516 cFYI(1, "Set POSIX ACL returned %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003517
3518setACLerrorExit:
3519 cifs_buf_release(pSMB);
3520 if (rc == -EAGAIN)
3521 goto setAclRetry;
3522 return rc;
3523}
3524
Steve Frenchf654bac2005-04-28 22:41:04 -07003525/* BB fix tabs in this function FIXME BB */
3526int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04003527CIFSGetExtAttr(const unsigned int xid, struct cifs_tcon *tcon,
Steve Frenchad7a2922008-02-07 23:25:02 +00003528 const int netfid, __u64 *pExtAttrBits, __u64 *pMask)
Steve Frenchf654bac2005-04-28 22:41:04 -07003529{
Steve French50c2f752007-07-13 00:33:32 +00003530 int rc = 0;
3531 struct smb_t2_qfi_req *pSMB = NULL;
3532 struct smb_t2_qfi_rsp *pSMBr = NULL;
3533 int bytes_returned;
3534 __u16 params, byte_count;
Steve Frenchf654bac2005-04-28 22:41:04 -07003535
Joe Perchesb6b38f72010-04-21 03:50:45 +00003536 cFYI(1, "In GetExtAttr");
Steve French790fe572007-07-07 19:25:05 +00003537 if (tcon == NULL)
3538 return -ENODEV;
Steve Frenchf654bac2005-04-28 22:41:04 -07003539
3540GetExtAttrRetry:
Steve French790fe572007-07-07 19:25:05 +00003541 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
3542 (void **) &pSMBr);
3543 if (rc)
3544 return rc;
Steve Frenchf654bac2005-04-28 22:41:04 -07003545
Steve Frenchad7a2922008-02-07 23:25:02 +00003546 params = 2 /* level */ + 2 /* fid */;
Steve French790fe572007-07-07 19:25:05 +00003547 pSMB->t2.TotalDataCount = 0;
3548 pSMB->t2.MaxParameterCount = cpu_to_le16(4);
3549 /* BB find exact max data count below from sess structure BB */
3550 pSMB->t2.MaxDataCount = cpu_to_le16(4000);
3551 pSMB->t2.MaxSetupCount = 0;
3552 pSMB->t2.Reserved = 0;
3553 pSMB->t2.Flags = 0;
3554 pSMB->t2.Timeout = 0;
3555 pSMB->t2.Reserved2 = 0;
3556 pSMB->t2.ParameterOffset = cpu_to_le16(offsetof(struct smb_t2_qfi_req,
3557 Fid) - 4);
3558 pSMB->t2.DataCount = 0;
3559 pSMB->t2.DataOffset = 0;
3560 pSMB->t2.SetupCount = 1;
3561 pSMB->t2.Reserved3 = 0;
3562 pSMB->t2.SubCommand = cpu_to_le16(TRANS2_QUERY_FILE_INFORMATION);
3563 byte_count = params + 1 /* pad */ ;
3564 pSMB->t2.TotalParameterCount = cpu_to_le16(params);
3565 pSMB->t2.ParameterCount = pSMB->t2.TotalParameterCount;
3566 pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_ATTR_FLAGS);
3567 pSMB->Pad = 0;
Steve Frenchf654bac2005-04-28 22:41:04 -07003568 pSMB->Fid = netfid;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00003569 inc_rfc1001_len(pSMB, byte_count);
Steve French790fe572007-07-07 19:25:05 +00003570 pSMB->t2.ByteCount = cpu_to_le16(byte_count);
Steve Frenchf654bac2005-04-28 22:41:04 -07003571
Steve French790fe572007-07-07 19:25:05 +00003572 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
3573 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
3574 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00003575 cFYI(1, "error %d in GetExtAttr", rc);
Steve French790fe572007-07-07 19:25:05 +00003576 } else {
3577 /* decode response */
3578 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
Steve French790fe572007-07-07 19:25:05 +00003579 /* BB also check enough total bytes returned */
Jeff Layton820a8032011-05-04 08:05:26 -04003580 if (rc || get_bcc(&pSMBr->hdr) < 2)
Steve French790fe572007-07-07 19:25:05 +00003581 /* If rc should we check for EOPNOSUPP and
3582 disable the srvino flag? or in caller? */
3583 rc = -EIO; /* bad smb */
3584 else {
3585 __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
3586 __u16 count = le16_to_cpu(pSMBr->t2.DataCount);
3587 struct file_chattr_info *pfinfo;
3588 /* BB Do we need a cast or hash here ? */
3589 if (count != 16) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00003590 cFYI(1, "Illegal size ret in GetExtAttr");
Steve French790fe572007-07-07 19:25:05 +00003591 rc = -EIO;
3592 goto GetExtAttrOut;
3593 }
3594 pfinfo = (struct file_chattr_info *)
3595 (data_offset + (char *) &pSMBr->hdr.Protocol);
3596 *pExtAttrBits = le64_to_cpu(pfinfo->mode);
Steve Frenchf654bac2005-04-28 22:41:04 -07003597 *pMask = le64_to_cpu(pfinfo->mask);
Steve French790fe572007-07-07 19:25:05 +00003598 }
3599 }
Steve Frenchf654bac2005-04-28 22:41:04 -07003600GetExtAttrOut:
Steve French790fe572007-07-07 19:25:05 +00003601 cifs_buf_release(pSMB);
3602 if (rc == -EAGAIN)
3603 goto GetExtAttrRetry;
3604 return rc;
Steve Frenchf654bac2005-04-28 22:41:04 -07003605}
3606
Steve Frenchf654bac2005-04-28 22:41:04 -07003607#endif /* CONFIG_POSIX */
Linus Torvalds1da177e2005-04-16 15:20:36 -07003608
Jeff Layton79df1ba2010-12-06 12:52:08 -05003609#ifdef CONFIG_CIFS_ACL
3610/*
3611 * Initialize NT TRANSACT SMB into small smb request buffer. This assumes that
3612 * all NT TRANSACTS that we init here have total parm and data under about 400
3613 * bytes (to fit in small cifs buffer size), which is the case so far, it
3614 * easily fits. NB: Setup words themselves and ByteCount MaxSetupCount (size of
3615 * returned setup area) and MaxParameterCount (returned parms size) must be set
3616 * by caller
3617 */
3618static int
3619smb_init_nttransact(const __u16 sub_command, const int setup_count,
Steve French96daf2b2011-05-27 04:34:02 +00003620 const int parm_len, struct cifs_tcon *tcon,
Jeff Layton79df1ba2010-12-06 12:52:08 -05003621 void **ret_buf)
3622{
3623 int rc;
3624 __u32 temp_offset;
3625 struct smb_com_ntransact_req *pSMB;
3626
3627 rc = small_smb_init(SMB_COM_NT_TRANSACT, 19 + setup_count, tcon,
3628 (void **)&pSMB);
3629 if (rc)
3630 return rc;
3631 *ret_buf = (void *)pSMB;
3632 pSMB->Reserved = 0;
3633 pSMB->TotalParameterCount = cpu_to_le32(parm_len);
3634 pSMB->TotalDataCount = 0;
Jeff Laytonc974bef2011-10-11 06:41:32 -04003635 pSMB->MaxDataCount = cpu_to_le32(CIFSMaxBufSize & 0xFFFFFF00);
Jeff Layton79df1ba2010-12-06 12:52:08 -05003636 pSMB->ParameterCount = pSMB->TotalParameterCount;
3637 pSMB->DataCount = pSMB->TotalDataCount;
3638 temp_offset = offsetof(struct smb_com_ntransact_req, Parms) +
3639 (setup_count * 2) - 4 /* for rfc1001 length itself */;
3640 pSMB->ParameterOffset = cpu_to_le32(temp_offset);
3641 pSMB->DataOffset = cpu_to_le32(temp_offset + parm_len);
3642 pSMB->SetupCount = setup_count; /* no need to le convert byte fields */
3643 pSMB->SubCommand = cpu_to_le16(sub_command);
3644 return 0;
3645}
3646
3647static int
3648validate_ntransact(char *buf, char **ppparm, char **ppdata,
3649 __u32 *pparmlen, __u32 *pdatalen)
3650{
3651 char *end_of_smb;
3652 __u32 data_count, data_offset, parm_count, parm_offset;
3653 struct smb_com_ntransact_rsp *pSMBr;
Jeff Layton820a8032011-05-04 08:05:26 -04003654 u16 bcc;
Jeff Layton79df1ba2010-12-06 12:52:08 -05003655
3656 *pdatalen = 0;
3657 *pparmlen = 0;
3658
3659 if (buf == NULL)
3660 return -EINVAL;
3661
3662 pSMBr = (struct smb_com_ntransact_rsp *)buf;
3663
Jeff Layton820a8032011-05-04 08:05:26 -04003664 bcc = get_bcc(&pSMBr->hdr);
3665 end_of_smb = 2 /* sizeof byte count */ + bcc +
Jeff Layton79df1ba2010-12-06 12:52:08 -05003666 (char *)&pSMBr->ByteCount;
3667
3668 data_offset = le32_to_cpu(pSMBr->DataOffset);
3669 data_count = le32_to_cpu(pSMBr->DataCount);
3670 parm_offset = le32_to_cpu(pSMBr->ParameterOffset);
3671 parm_count = le32_to_cpu(pSMBr->ParameterCount);
3672
3673 *ppparm = (char *)&pSMBr->hdr.Protocol + parm_offset;
3674 *ppdata = (char *)&pSMBr->hdr.Protocol + data_offset;
3675
3676 /* should we also check that parm and data areas do not overlap? */
3677 if (*ppparm > end_of_smb) {
3678 cFYI(1, "parms start after end of smb");
3679 return -EINVAL;
3680 } else if (parm_count + *ppparm > end_of_smb) {
3681 cFYI(1, "parm end after end of smb");
3682 return -EINVAL;
3683 } else if (*ppdata > end_of_smb) {
3684 cFYI(1, "data starts after end of smb");
3685 return -EINVAL;
3686 } else if (data_count + *ppdata > end_of_smb) {
3687 cFYI(1, "data %p + count %d (%p) past smb end %p start %p",
3688 *ppdata, data_count, (data_count + *ppdata),
3689 end_of_smb, pSMBr);
3690 return -EINVAL;
Jeff Layton820a8032011-05-04 08:05:26 -04003691 } else if (parm_count + data_count > bcc) {
Jeff Layton79df1ba2010-12-06 12:52:08 -05003692 cFYI(1, "parm count and data count larger than SMB");
3693 return -EINVAL;
3694 }
3695 *pdatalen = data_count;
3696 *pparmlen = parm_count;
3697 return 0;
3698}
3699
Steve French0a4b92c2006-01-12 15:44:21 -08003700/* Get Security Descriptor (by handle) from remote server for a file or dir */
3701int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04003702CIFSSMBGetCIFSACL(const unsigned int xid, struct cifs_tcon *tcon, __u16 fid,
Steve French630f3f0c2007-10-25 21:17:17 +00003703 struct cifs_ntsd **acl_inf, __u32 *pbuflen)
Steve French0a4b92c2006-01-12 15:44:21 -08003704{
3705 int rc = 0;
3706 int buf_type = 0;
Steve Frenchad7a2922008-02-07 23:25:02 +00003707 QUERY_SEC_DESC_REQ *pSMB;
Steve French0a4b92c2006-01-12 15:44:21 -08003708 struct kvec iov[1];
3709
Joe Perchesb6b38f72010-04-21 03:50:45 +00003710 cFYI(1, "GetCifsACL");
Steve French0a4b92c2006-01-12 15:44:21 -08003711
Steve French630f3f0c2007-10-25 21:17:17 +00003712 *pbuflen = 0;
3713 *acl_inf = NULL;
3714
Steve Frenchb9c7a2b2007-10-26 23:40:20 +00003715 rc = smb_init_nttransact(NT_TRANSACT_QUERY_SECURITY_DESC, 0,
Steve French0a4b92c2006-01-12 15:44:21 -08003716 8 /* parm len */, tcon, (void **) &pSMB);
3717 if (rc)
3718 return rc;
3719
3720 pSMB->MaxParameterCount = cpu_to_le32(4);
3721 /* BB TEST with big acls that might need to be e.g. larger than 16K */
3722 pSMB->MaxSetupCount = 0;
3723 pSMB->Fid = fid; /* file handle always le */
3724 pSMB->AclFlags = cpu_to_le32(CIFS_ACL_OWNER | CIFS_ACL_GROUP |
3725 CIFS_ACL_DACL);
3726 pSMB->ByteCount = cpu_to_le16(11); /* 3 bytes pad + 8 bytes parm */
Steve Frenchbe8e3b02011-04-29 05:40:20 +00003727 inc_rfc1001_len(pSMB, 11);
Steve French0a4b92c2006-01-12 15:44:21 -08003728 iov[0].iov_base = (char *)pSMB;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00003729 iov[0].iov_len = be32_to_cpu(pSMB->hdr.smb_buf_length) + 4;
Steve French0a4b92c2006-01-12 15:44:21 -08003730
Steve Frencha761ac52007-10-18 21:45:27 +00003731 rc = SendReceive2(xid, tcon->ses, iov, 1 /* num iovec */, &buf_type,
Jeff Layton77499812011-01-11 07:24:23 -05003732 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04003733 cifs_stats_inc(&tcon->stats.cifs_stats.num_acl_get);
Steve French0a4b92c2006-01-12 15:44:21 -08003734 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00003735 cFYI(1, "Send error in QuerySecDesc = %d", rc);
Steve French0a4b92c2006-01-12 15:44:21 -08003736 } else { /* decode response */
Steve Frenchad7a2922008-02-07 23:25:02 +00003737 __le32 *parm;
Steve French630f3f0c2007-10-25 21:17:17 +00003738 __u32 parm_len;
3739 __u32 acl_len;
Steve French50c2f752007-07-13 00:33:32 +00003740 struct smb_com_ntransact_rsp *pSMBr;
Steve French630f3f0c2007-10-25 21:17:17 +00003741 char *pdata;
Steve French0a4b92c2006-01-12 15:44:21 -08003742
3743/* validate_nttransact */
Steve French50c2f752007-07-13 00:33:32 +00003744 rc = validate_ntransact(iov[0].iov_base, (char **)&parm,
Steve French630f3f0c2007-10-25 21:17:17 +00003745 &pdata, &parm_len, pbuflen);
Steve French790fe572007-07-07 19:25:05 +00003746 if (rc)
Steve French0a4b92c2006-01-12 15:44:21 -08003747 goto qsec_out;
3748 pSMBr = (struct smb_com_ntransact_rsp *)iov[0].iov_base;
3749
Joe Perchesb6b38f72010-04-21 03:50:45 +00003750 cFYI(1, "smb %p parm %p data %p", pSMBr, parm, *acl_inf);
Steve French0a4b92c2006-01-12 15:44:21 -08003751
3752 if (le32_to_cpu(pSMBr->ParameterCount) != 4) {
3753 rc = -EIO; /* bad smb */
Steve French630f3f0c2007-10-25 21:17:17 +00003754 *pbuflen = 0;
Steve French0a4b92c2006-01-12 15:44:21 -08003755 goto qsec_out;
3756 }
3757
3758/* BB check that data area is minimum length and as big as acl_len */
3759
Steve Frenchaf6f4612007-10-16 18:40:37 +00003760 acl_len = le32_to_cpu(*parm);
Steve French630f3f0c2007-10-25 21:17:17 +00003761 if (acl_len != *pbuflen) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00003762 cERROR(1, "acl length %d does not match %d",
3763 acl_len, *pbuflen);
Steve French630f3f0c2007-10-25 21:17:17 +00003764 if (*pbuflen > acl_len)
3765 *pbuflen = acl_len;
3766 }
Steve French0a4b92c2006-01-12 15:44:21 -08003767
Steve French630f3f0c2007-10-25 21:17:17 +00003768 /* check if buffer is big enough for the acl
3769 header followed by the smallest SID */
3770 if ((*pbuflen < sizeof(struct cifs_ntsd) + 8) ||
3771 (*pbuflen >= 64 * 1024)) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00003772 cERROR(1, "bad acl length %d", *pbuflen);
Steve French630f3f0c2007-10-25 21:17:17 +00003773 rc = -EINVAL;
3774 *pbuflen = 0;
3775 } else {
3776 *acl_inf = kmalloc(*pbuflen, GFP_KERNEL);
3777 if (*acl_inf == NULL) {
3778 *pbuflen = 0;
3779 rc = -ENOMEM;
3780 }
3781 memcpy(*acl_inf, pdata, *pbuflen);
3782 }
Steve French0a4b92c2006-01-12 15:44:21 -08003783 }
3784qsec_out:
Steve French790fe572007-07-07 19:25:05 +00003785 if (buf_type == CIFS_SMALL_BUFFER)
Steve French0a4b92c2006-01-12 15:44:21 -08003786 cifs_small_buf_release(iov[0].iov_base);
Steve French790fe572007-07-07 19:25:05 +00003787 else if (buf_type == CIFS_LARGE_BUFFER)
Steve French0a4b92c2006-01-12 15:44:21 -08003788 cifs_buf_release(iov[0].iov_base);
Steve French4b8f9302006-02-26 16:41:18 +00003789/* cifs_small_buf_release(pSMB); */ /* Freed earlier now in SendReceive2 */
Steve French0a4b92c2006-01-12 15:44:21 -08003790 return rc;
3791}
Steve French97837582007-12-31 07:47:21 +00003792
3793int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04003794CIFSSMBSetCIFSACL(const unsigned int xid, struct cifs_tcon *tcon, __u16 fid,
Shirish Pargaonkara5ff3762011-10-13 10:26:03 -05003795 struct cifs_ntsd *pntsd, __u32 acllen, int aclflag)
Steve French97837582007-12-31 07:47:21 +00003796{
3797 __u16 byte_count, param_count, data_count, param_offset, data_offset;
3798 int rc = 0;
3799 int bytes_returned = 0;
3800 SET_SEC_DESC_REQ *pSMB = NULL;
Jeff Laytonb2a3ad92012-03-26 09:55:29 -04003801 void *pSMBr;
Steve French97837582007-12-31 07:47:21 +00003802
3803setCifsAclRetry:
Jeff Laytonb2a3ad92012-03-26 09:55:29 -04003804 rc = smb_init(SMB_COM_NT_TRANSACT, 19, tcon, (void **) &pSMB, &pSMBr);
Steve French97837582007-12-31 07:47:21 +00003805 if (rc)
Jeff Laytonb2a3ad92012-03-26 09:55:29 -04003806 return rc;
Steve French97837582007-12-31 07:47:21 +00003807
3808 pSMB->MaxSetupCount = 0;
3809 pSMB->Reserved = 0;
3810
3811 param_count = 8;
3812 param_offset = offsetof(struct smb_com_transaction_ssec_req, Fid) - 4;
3813 data_count = acllen;
3814 data_offset = param_offset + param_count;
3815 byte_count = 3 /* pad */ + param_count;
3816
3817 pSMB->DataCount = cpu_to_le32(data_count);
3818 pSMB->TotalDataCount = pSMB->DataCount;
3819 pSMB->MaxParameterCount = cpu_to_le32(4);
3820 pSMB->MaxDataCount = cpu_to_le32(16384);
3821 pSMB->ParameterCount = cpu_to_le32(param_count);
3822 pSMB->ParameterOffset = cpu_to_le32(param_offset);
3823 pSMB->TotalParameterCount = pSMB->ParameterCount;
3824 pSMB->DataOffset = cpu_to_le32(data_offset);
3825 pSMB->SetupCount = 0;
3826 pSMB->SubCommand = cpu_to_le16(NT_TRANSACT_SET_SECURITY_DESC);
3827 pSMB->ByteCount = cpu_to_le16(byte_count+data_count);
3828
3829 pSMB->Fid = fid; /* file handle always le */
3830 pSMB->Reserved2 = 0;
Shirish Pargaonkara5ff3762011-10-13 10:26:03 -05003831 pSMB->AclFlags = cpu_to_le32(aclflag);
Steve French97837582007-12-31 07:47:21 +00003832
3833 if (pntsd && acllen) {
Jeff Laytonb2a3ad92012-03-26 09:55:29 -04003834 memcpy((char *)pSMBr + offsetof(struct smb_hdr, Protocol) +
3835 data_offset, pntsd, acllen);
Steve Frenchbe8e3b02011-04-29 05:40:20 +00003836 inc_rfc1001_len(pSMB, byte_count + data_count);
Steve French97837582007-12-31 07:47:21 +00003837 } else
Steve Frenchbe8e3b02011-04-29 05:40:20 +00003838 inc_rfc1001_len(pSMB, byte_count);
Steve French97837582007-12-31 07:47:21 +00003839
3840 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
3841 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
3842
Joe Perchesb6b38f72010-04-21 03:50:45 +00003843 cFYI(1, "SetCIFSACL bytes_returned: %d, rc: %d", bytes_returned, rc);
Steve French97837582007-12-31 07:47:21 +00003844 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00003845 cFYI(1, "Set CIFS ACL returned %d", rc);
Steve French97837582007-12-31 07:47:21 +00003846 cifs_buf_release(pSMB);
3847
3848 if (rc == -EAGAIN)
3849 goto setCifsAclRetry;
3850
3851 return (rc);
3852}
3853
Jeff Layton79df1ba2010-12-06 12:52:08 -05003854#endif /* CONFIG_CIFS_ACL */
Steve French0a4b92c2006-01-12 15:44:21 -08003855
Steve French6b8edfe2005-08-23 20:26:03 -07003856/* Legacy Query Path Information call for lookup to old servers such
3857 as Win9x/WinME */
Pavel Shilovsky68889f22012-05-25 14:40:22 +04003858int
3859SMBQueryInformation(const unsigned int xid, struct cifs_tcon *tcon,
3860 const char *search_name, FILE_ALL_INFO *data,
3861 const struct nls_table *nls_codepage, int remap)
Steve French6b8edfe2005-08-23 20:26:03 -07003862{
Steve Frenchad7a2922008-02-07 23:25:02 +00003863 QUERY_INFORMATION_REQ *pSMB;
3864 QUERY_INFORMATION_RSP *pSMBr;
Steve French6b8edfe2005-08-23 20:26:03 -07003865 int rc = 0;
3866 int bytes_returned;
3867 int name_len;
3868
Pavel Shilovsky68889f22012-05-25 14:40:22 +04003869 cFYI(1, "In SMBQPath path %s", search_name);
Steve French6b8edfe2005-08-23 20:26:03 -07003870QInfRetry:
3871 rc = smb_init(SMB_COM_QUERY_INFORMATION, 0, tcon, (void **) &pSMB,
Steve French50c2f752007-07-13 00:33:32 +00003872 (void **) &pSMBr);
Steve French6b8edfe2005-08-23 20:26:03 -07003873 if (rc)
3874 return rc;
3875
3876 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
3877 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06003878 cifsConvertToUTF16((__le16 *) pSMB->FileName,
Pavel Shilovsky68889f22012-05-25 14:40:22 +04003879 search_name, PATH_MAX, nls_codepage,
Steve Frenchacbbb762012-01-18 22:32:33 -06003880 remap);
Steve French6b8edfe2005-08-23 20:26:03 -07003881 name_len++; /* trailing null */
3882 name_len *= 2;
Steve French50c2f752007-07-13 00:33:32 +00003883 } else {
Pavel Shilovsky68889f22012-05-25 14:40:22 +04003884 name_len = strnlen(search_name, PATH_MAX);
Steve French6b8edfe2005-08-23 20:26:03 -07003885 name_len++; /* trailing null */
Pavel Shilovsky68889f22012-05-25 14:40:22 +04003886 strncpy(pSMB->FileName, search_name, name_len);
Steve French6b8edfe2005-08-23 20:26:03 -07003887 }
3888 pSMB->BufferFormat = 0x04;
Steve French50c2f752007-07-13 00:33:32 +00003889 name_len++; /* account for buffer type byte */
Steve Frenchbe8e3b02011-04-29 05:40:20 +00003890 inc_rfc1001_len(pSMB, (__u16)name_len);
Steve French6b8edfe2005-08-23 20:26:03 -07003891 pSMB->ByteCount = cpu_to_le16(name_len);
3892
3893 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
Steve French50c2f752007-07-13 00:33:32 +00003894 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Steve French6b8edfe2005-08-23 20:26:03 -07003895 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00003896 cFYI(1, "Send error in QueryInfo = %d", rc);
Pavel Shilovsky68889f22012-05-25 14:40:22 +04003897 } else if (data) {
Steve French1bd5bbc2006-09-28 03:35:57 +00003898 struct timespec ts;
3899 __u32 time = le32_to_cpu(pSMBr->last_write_time);
Steve Frenchad7a2922008-02-07 23:25:02 +00003900
3901 /* decode response */
Steve French1bd5bbc2006-09-28 03:35:57 +00003902 /* BB FIXME - add time zone adjustment BB */
Pavel Shilovsky68889f22012-05-25 14:40:22 +04003903 memset(data, 0, sizeof(FILE_ALL_INFO));
Steve French1bd5bbc2006-09-28 03:35:57 +00003904 ts.tv_nsec = 0;
3905 ts.tv_sec = time;
3906 /* decode time fields */
Pavel Shilovsky68889f22012-05-25 14:40:22 +04003907 data->ChangeTime = cpu_to_le64(cifs_UnixTimeToNT(ts));
3908 data->LastWriteTime = data->ChangeTime;
3909 data->LastAccessTime = 0;
3910 data->AllocationSize =
Steve French70ca7342005-09-22 16:32:06 -07003911 cpu_to_le64(le32_to_cpu(pSMBr->size));
Pavel Shilovsky68889f22012-05-25 14:40:22 +04003912 data->EndOfFile = data->AllocationSize;
3913 data->Attributes =
Steve French70ca7342005-09-22 16:32:06 -07003914 cpu_to_le32(le16_to_cpu(pSMBr->attr));
Steve French6b8edfe2005-08-23 20:26:03 -07003915 } else
3916 rc = -EIO; /* bad buffer passed in */
3917
3918 cifs_buf_release(pSMB);
3919
3920 if (rc == -EAGAIN)
3921 goto QInfRetry;
3922
3923 return rc;
3924}
3925
Jeff Laytonbcd53572010-02-12 07:44:16 -05003926int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04003927CIFSSMBQFileInfo(const unsigned int xid, struct cifs_tcon *tcon,
Jeff Laytonbcd53572010-02-12 07:44:16 -05003928 u16 netfid, FILE_ALL_INFO *pFindData)
3929{
3930 struct smb_t2_qfi_req *pSMB = NULL;
3931 struct smb_t2_qfi_rsp *pSMBr = NULL;
3932 int rc = 0;
3933 int bytes_returned;
3934 __u16 params, byte_count;
Steve French6b8edfe2005-08-23 20:26:03 -07003935
Jeff Laytonbcd53572010-02-12 07:44:16 -05003936QFileInfoRetry:
3937 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
3938 (void **) &pSMBr);
3939 if (rc)
3940 return rc;
Steve French6b8edfe2005-08-23 20:26:03 -07003941
Jeff Laytonbcd53572010-02-12 07:44:16 -05003942 params = 2 /* level */ + 2 /* fid */;
3943 pSMB->t2.TotalDataCount = 0;
3944 pSMB->t2.MaxParameterCount = cpu_to_le16(4);
3945 /* BB find exact max data count below from sess structure BB */
3946 pSMB->t2.MaxDataCount = cpu_to_le16(CIFSMaxBufSize);
3947 pSMB->t2.MaxSetupCount = 0;
3948 pSMB->t2.Reserved = 0;
3949 pSMB->t2.Flags = 0;
3950 pSMB->t2.Timeout = 0;
3951 pSMB->t2.Reserved2 = 0;
3952 pSMB->t2.ParameterOffset = cpu_to_le16(offsetof(struct smb_t2_qfi_req,
3953 Fid) - 4);
3954 pSMB->t2.DataCount = 0;
3955 pSMB->t2.DataOffset = 0;
3956 pSMB->t2.SetupCount = 1;
3957 pSMB->t2.Reserved3 = 0;
3958 pSMB->t2.SubCommand = cpu_to_le16(TRANS2_QUERY_FILE_INFORMATION);
3959 byte_count = params + 1 /* pad */ ;
3960 pSMB->t2.TotalParameterCount = cpu_to_le16(params);
3961 pSMB->t2.ParameterCount = pSMB->t2.TotalParameterCount;
3962 pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FILE_ALL_INFO);
3963 pSMB->Pad = 0;
3964 pSMB->Fid = netfid;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00003965 inc_rfc1001_len(pSMB, byte_count);
Jeff Laytonbcd53572010-02-12 07:44:16 -05003966
3967 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
3968 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
3969 if (rc) {
Steve Frenchf19159d2010-04-21 04:12:10 +00003970 cFYI(1, "Send error in QPathInfo = %d", rc);
Jeff Laytonbcd53572010-02-12 07:44:16 -05003971 } else { /* decode response */
3972 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
3973
3974 if (rc) /* BB add auto retry on EOPNOTSUPP? */
3975 rc = -EIO;
Jeff Layton820a8032011-05-04 08:05:26 -04003976 else if (get_bcc(&pSMBr->hdr) < 40)
Jeff Laytonbcd53572010-02-12 07:44:16 -05003977 rc = -EIO; /* bad smb */
3978 else if (pFindData) {
3979 __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
3980 memcpy((char *) pFindData,
3981 (char *) &pSMBr->hdr.Protocol +
3982 data_offset, sizeof(FILE_ALL_INFO));
3983 } else
3984 rc = -ENOMEM;
3985 }
3986 cifs_buf_release(pSMB);
3987 if (rc == -EAGAIN)
3988 goto QFileInfoRetry;
3989
3990 return rc;
3991}
Steve French6b8edfe2005-08-23 20:26:03 -07003992
Linus Torvalds1da177e2005-04-16 15:20:36 -07003993int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04003994CIFSSMBQPathInfo(const unsigned int xid, struct cifs_tcon *tcon,
Pavel Shilovsky68889f22012-05-25 14:40:22 +04003995 const char *search_name, FILE_ALL_INFO *data,
Steve Frenchacf1a1b2006-10-12 03:28:28 +00003996 int legacy /* old style infolevel */,
Steve French737b7582005-04-28 22:41:06 -07003997 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003998{
Pavel Shilovsky68889f22012-05-25 14:40:22 +04003999 /* level 263 SMB_QUERY_FILE_ALL_INFO */
Linus Torvalds1da177e2005-04-16 15:20:36 -07004000 TRANSACTION2_QPI_REQ *pSMB = NULL;
4001 TRANSACTION2_QPI_RSP *pSMBr = NULL;
4002 int rc = 0;
4003 int bytes_returned;
4004 int name_len;
4005 __u16 params, byte_count;
4006
Pavel Shilovsky68889f22012-05-25 14:40:22 +04004007 /* cFYI(1, "In QPathInfo path %s", search_name); */
Linus Torvalds1da177e2005-04-16 15:20:36 -07004008QPathInfoRetry:
4009 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
4010 (void **) &pSMBr);
4011 if (rc)
4012 return rc;
4013
4014 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
4015 name_len =
Pavel Shilovsky68889f22012-05-25 14:40:22 +04004016 cifsConvertToUTF16((__le16 *) pSMB->FileName, search_name,
Steve Frenchacbbb762012-01-18 22:32:33 -06004017 PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004018 name_len++; /* trailing null */
4019 name_len *= 2;
Steve French50c2f752007-07-13 00:33:32 +00004020 } else { /* BB improve the check for buffer overruns BB */
Pavel Shilovsky68889f22012-05-25 14:40:22 +04004021 name_len = strnlen(search_name, PATH_MAX);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004022 name_len++; /* trailing null */
Pavel Shilovsky68889f22012-05-25 14:40:22 +04004023 strncpy(pSMB->FileName, search_name, name_len);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004024 }
4025
Steve French50c2f752007-07-13 00:33:32 +00004026 params = 2 /* level */ + 4 /* reserved */ + name_len /* includes NUL */;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004027 pSMB->TotalDataCount = 0;
4028 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French582d21e2008-05-13 04:54:12 +00004029 /* BB find exact max SMB PDU from sess structure BB */
4030 pSMB->MaxDataCount = cpu_to_le16(4000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004031 pSMB->MaxSetupCount = 0;
4032 pSMB->Reserved = 0;
4033 pSMB->Flags = 0;
4034 pSMB->Timeout = 0;
4035 pSMB->Reserved2 = 0;
4036 pSMB->ParameterOffset = cpu_to_le16(offsetof(
Steve French50c2f752007-07-13 00:33:32 +00004037 struct smb_com_transaction2_qpi_req, InformationLevel) - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004038 pSMB->DataCount = 0;
4039 pSMB->DataOffset = 0;
4040 pSMB->SetupCount = 1;
4041 pSMB->Reserved3 = 0;
4042 pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_PATH_INFORMATION);
4043 byte_count = params + 1 /* pad */ ;
4044 pSMB->TotalParameterCount = cpu_to_le16(params);
4045 pSMB->ParameterCount = pSMB->TotalParameterCount;
Steve French790fe572007-07-07 19:25:05 +00004046 if (legacy)
Steve Frenchacf1a1b2006-10-12 03:28:28 +00004047 pSMB->InformationLevel = cpu_to_le16(SMB_INFO_STANDARD);
4048 else
4049 pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FILE_ALL_INFO);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004050 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00004051 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004052 pSMB->ByteCount = cpu_to_le16(byte_count);
4053
4054 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
4055 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
4056 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00004057 cFYI(1, "Send error in QPathInfo = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004058 } else { /* decode response */
4059 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
4060
Steve Frenchacf1a1b2006-10-12 03:28:28 +00004061 if (rc) /* BB add auto retry on EOPNOTSUPP? */
4062 rc = -EIO;
Jeff Layton820a8032011-05-04 08:05:26 -04004063 else if (!legacy && get_bcc(&pSMBr->hdr) < 40)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004064 rc = -EIO; /* bad smb */
Jeff Layton820a8032011-05-04 08:05:26 -04004065 else if (legacy && get_bcc(&pSMBr->hdr) < 24)
Steve French50c2f752007-07-13 00:33:32 +00004066 rc = -EIO; /* 24 or 26 expected but we do not read
4067 last field */
Pavel Shilovsky68889f22012-05-25 14:40:22 +04004068 else if (data) {
Steve Frenchacf1a1b2006-10-12 03:28:28 +00004069 int size;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004070 __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
Steve Frenchad7a2922008-02-07 23:25:02 +00004071
Pavel Shilovsky68889f22012-05-25 14:40:22 +04004072 /*
4073 * On legacy responses we do not read the last field,
4074 * EAsize, fortunately since it varies by subdialect and
4075 * also note it differs on Set vs Get, ie two bytes or 4
4076 * bytes depending but we don't care here.
4077 */
Steve Frenchad7a2922008-02-07 23:25:02 +00004078 if (legacy)
Steve Frenchacf1a1b2006-10-12 03:28:28 +00004079 size = sizeof(FILE_INFO_STANDARD);
4080 else
4081 size = sizeof(FILE_ALL_INFO);
Pavel Shilovsky68889f22012-05-25 14:40:22 +04004082 memcpy((char *) data, (char *) &pSMBr->hdr.Protocol +
Steve Frenchacf1a1b2006-10-12 03:28:28 +00004083 data_offset, size);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004084 } else
4085 rc = -ENOMEM;
4086 }
4087 cifs_buf_release(pSMB);
4088 if (rc == -EAGAIN)
4089 goto QPathInfoRetry;
4090
4091 return rc;
4092}
4093
4094int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04004095CIFSSMBUnixQFileInfo(const unsigned int xid, struct cifs_tcon *tcon,
Jeff Laytonc8634fd2010-02-12 07:44:17 -05004096 u16 netfid, FILE_UNIX_BASIC_INFO *pFindData)
4097{
4098 struct smb_t2_qfi_req *pSMB = NULL;
4099 struct smb_t2_qfi_rsp *pSMBr = NULL;
4100 int rc = 0;
4101 int bytes_returned;
4102 __u16 params, byte_count;
4103
4104UnixQFileInfoRetry:
4105 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
4106 (void **) &pSMBr);
4107 if (rc)
4108 return rc;
4109
4110 params = 2 /* level */ + 2 /* fid */;
4111 pSMB->t2.TotalDataCount = 0;
4112 pSMB->t2.MaxParameterCount = cpu_to_le16(4);
4113 /* BB find exact max data count below from sess structure BB */
4114 pSMB->t2.MaxDataCount = cpu_to_le16(CIFSMaxBufSize);
4115 pSMB->t2.MaxSetupCount = 0;
4116 pSMB->t2.Reserved = 0;
4117 pSMB->t2.Flags = 0;
4118 pSMB->t2.Timeout = 0;
4119 pSMB->t2.Reserved2 = 0;
4120 pSMB->t2.ParameterOffset = cpu_to_le16(offsetof(struct smb_t2_qfi_req,
4121 Fid) - 4);
4122 pSMB->t2.DataCount = 0;
4123 pSMB->t2.DataOffset = 0;
4124 pSMB->t2.SetupCount = 1;
4125 pSMB->t2.Reserved3 = 0;
4126 pSMB->t2.SubCommand = cpu_to_le16(TRANS2_QUERY_FILE_INFORMATION);
4127 byte_count = params + 1 /* pad */ ;
4128 pSMB->t2.TotalParameterCount = cpu_to_le16(params);
4129 pSMB->t2.ParameterCount = pSMB->t2.TotalParameterCount;
4130 pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FILE_UNIX_BASIC);
4131 pSMB->Pad = 0;
4132 pSMB->Fid = netfid;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00004133 inc_rfc1001_len(pSMB, byte_count);
Jeff Laytonc8634fd2010-02-12 07:44:17 -05004134
4135 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
4136 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
4137 if (rc) {
Steve Frenchf19159d2010-04-21 04:12:10 +00004138 cFYI(1, "Send error in QPathInfo = %d", rc);
Jeff Laytonc8634fd2010-02-12 07:44:17 -05004139 } else { /* decode response */
4140 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
4141
Jeff Layton820a8032011-05-04 08:05:26 -04004142 if (rc || get_bcc(&pSMBr->hdr) < sizeof(FILE_UNIX_BASIC_INFO)) {
Jeff Laytonac3aa2f2012-07-23 13:14:28 -04004143 cERROR(1, "Malformed FILE_UNIX_BASIC_INFO response. "
Jeff Laytonc8634fd2010-02-12 07:44:17 -05004144 "Unix Extensions can be disabled on mount "
Steve Frenchf19159d2010-04-21 04:12:10 +00004145 "by specifying the nosfu mount option.");
Jeff Laytonc8634fd2010-02-12 07:44:17 -05004146 rc = -EIO; /* bad smb */
4147 } else {
4148 __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
4149 memcpy((char *) pFindData,
4150 (char *) &pSMBr->hdr.Protocol +
4151 data_offset,
4152 sizeof(FILE_UNIX_BASIC_INFO));
4153 }
4154 }
4155
4156 cifs_buf_release(pSMB);
4157 if (rc == -EAGAIN)
4158 goto UnixQFileInfoRetry;
4159
4160 return rc;
4161}
4162
4163int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04004164CIFSSMBUnixQPathInfo(const unsigned int xid, struct cifs_tcon *tcon,
Linus Torvalds1da177e2005-04-16 15:20:36 -07004165 const unsigned char *searchName,
Steve French582d21e2008-05-13 04:54:12 +00004166 FILE_UNIX_BASIC_INFO *pFindData,
Steve French737b7582005-04-28 22:41:06 -07004167 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004168{
4169/* SMB_QUERY_FILE_UNIX_BASIC */
4170 TRANSACTION2_QPI_REQ *pSMB = NULL;
4171 TRANSACTION2_QPI_RSP *pSMBr = NULL;
4172 int rc = 0;
4173 int bytes_returned = 0;
4174 int name_len;
4175 __u16 params, byte_count;
4176
Joe Perchesb6b38f72010-04-21 03:50:45 +00004177 cFYI(1, "In QPathInfo (Unix) the path %s", searchName);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004178UnixQPathInfoRetry:
4179 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
4180 (void **) &pSMBr);
4181 if (rc)
4182 return rc;
4183
4184 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
4185 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06004186 cifsConvertToUTF16((__le16 *) pSMB->FileName, searchName,
4187 PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004188 name_len++; /* trailing null */
4189 name_len *= 2;
Steve French50c2f752007-07-13 00:33:32 +00004190 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07004191 name_len = strnlen(searchName, PATH_MAX);
4192 name_len++; /* trailing null */
4193 strncpy(pSMB->FileName, searchName, name_len);
4194 }
4195
Steve French50c2f752007-07-13 00:33:32 +00004196 params = 2 /* level */ + 4 /* reserved */ + name_len /* includes NUL */;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004197 pSMB->TotalDataCount = 0;
4198 pSMB->MaxParameterCount = cpu_to_le16(2);
4199 /* BB find exact max SMB PDU from sess structure BB */
Steve French50c2f752007-07-13 00:33:32 +00004200 pSMB->MaxDataCount = cpu_to_le16(4000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004201 pSMB->MaxSetupCount = 0;
4202 pSMB->Reserved = 0;
4203 pSMB->Flags = 0;
4204 pSMB->Timeout = 0;
4205 pSMB->Reserved2 = 0;
4206 pSMB->ParameterOffset = cpu_to_le16(offsetof(
Steve French50c2f752007-07-13 00:33:32 +00004207 struct smb_com_transaction2_qpi_req, InformationLevel) - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004208 pSMB->DataCount = 0;
4209 pSMB->DataOffset = 0;
4210 pSMB->SetupCount = 1;
4211 pSMB->Reserved3 = 0;
4212 pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_PATH_INFORMATION);
4213 byte_count = params + 1 /* pad */ ;
4214 pSMB->TotalParameterCount = cpu_to_le16(params);
4215 pSMB->ParameterCount = pSMB->TotalParameterCount;
4216 pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FILE_UNIX_BASIC);
4217 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00004218 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004219 pSMB->ByteCount = cpu_to_le16(byte_count);
4220
4221 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
4222 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
4223 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00004224 cFYI(1, "Send error in QPathInfo = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004225 } else { /* decode response */
4226 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
4227
Jeff Layton820a8032011-05-04 08:05:26 -04004228 if (rc || get_bcc(&pSMBr->hdr) < sizeof(FILE_UNIX_BASIC_INFO)) {
Jeff Laytonac3aa2f2012-07-23 13:14:28 -04004229 cERROR(1, "Malformed FILE_UNIX_BASIC_INFO response. "
Steve French1e71f252007-09-20 15:30:07 +00004230 "Unix Extensions can be disabled on mount "
Joe Perchesb6b38f72010-04-21 03:50:45 +00004231 "by specifying the nosfu mount option.");
Linus Torvalds1da177e2005-04-16 15:20:36 -07004232 rc = -EIO; /* bad smb */
4233 } else {
4234 __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
4235 memcpy((char *) pFindData,
4236 (char *) &pSMBr->hdr.Protocol +
4237 data_offset,
Steve French630f3f0c2007-10-25 21:17:17 +00004238 sizeof(FILE_UNIX_BASIC_INFO));
Linus Torvalds1da177e2005-04-16 15:20:36 -07004239 }
4240 }
4241 cifs_buf_release(pSMB);
4242 if (rc == -EAGAIN)
4243 goto UnixQPathInfoRetry;
4244
4245 return rc;
4246}
4247
Linus Torvalds1da177e2005-04-16 15:20:36 -07004248/* xid, tcon, searchName and codepage are input parms, rest are returned */
4249int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04004250CIFSFindFirst(const unsigned int xid, struct cifs_tcon *tcon,
Steve French50c2f752007-07-13 00:33:32 +00004251 const char *searchName,
Linus Torvalds1da177e2005-04-16 15:20:36 -07004252 const struct nls_table *nls_codepage,
Shirish Pargaonkar2608bee2012-05-15 10:19:16 -05004253 __u16 *pnetfid, __u16 search_flags,
Steve French50c2f752007-07-13 00:33:32 +00004254 struct cifs_search_info *psrch_inf, int remap, const char dirsep)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004255{
4256/* level 257 SMB_ */
4257 TRANSACTION2_FFIRST_REQ *pSMB = NULL;
4258 TRANSACTION2_FFIRST_RSP *pSMBr = NULL;
Steve Frenchad7a2922008-02-07 23:25:02 +00004259 T2_FFIRST_RSP_PARMS *parms;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004260 int rc = 0;
4261 int bytes_returned = 0;
4262 int name_len;
4263 __u16 params, byte_count;
4264
Joe Perchesb6b38f72010-04-21 03:50:45 +00004265 cFYI(1, "In FindFirst for %s", searchName);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004266
4267findFirstRetry:
4268 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
4269 (void **) &pSMBr);
4270 if (rc)
4271 return rc;
4272
4273 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
4274 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06004275 cifsConvertToUTF16((__le16 *) pSMB->FileName, searchName,
4276 PATH_MAX, nls_codepage, remap);
Steve French737b7582005-04-28 22:41:06 -07004277 /* We can not add the asterik earlier in case
4278 it got remapped to 0xF03A as if it were part of the
4279 directory name instead of a wildcard */
Linus Torvalds1da177e2005-04-16 15:20:36 -07004280 name_len *= 2;
Jeremy Allisonac670552005-06-22 17:26:35 -07004281 pSMB->FileName[name_len] = dirsep;
Steve French737b7582005-04-28 22:41:06 -07004282 pSMB->FileName[name_len+1] = 0;
4283 pSMB->FileName[name_len+2] = '*';
4284 pSMB->FileName[name_len+3] = 0;
4285 name_len += 4; /* now the trailing null */
Linus Torvalds1da177e2005-04-16 15:20:36 -07004286 pSMB->FileName[name_len] = 0; /* null terminate just in case */
4287 pSMB->FileName[name_len+1] = 0;
Steve French737b7582005-04-28 22:41:06 -07004288 name_len += 2;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004289 } else { /* BB add check for overrun of SMB buf BB */
4290 name_len = strnlen(searchName, PATH_MAX);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004291/* BB fix here and in unicode clause above ie
Steve French790fe572007-07-07 19:25:05 +00004292 if (name_len > buffersize-header)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004293 free buffer exit; BB */
4294 strncpy(pSMB->FileName, searchName, name_len);
Jeremy Allisonac670552005-06-22 17:26:35 -07004295 pSMB->FileName[name_len] = dirsep;
Steve French68575472005-04-30 11:10:57 -07004296 pSMB->FileName[name_len+1] = '*';
4297 pSMB->FileName[name_len+2] = 0;
4298 name_len += 3;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004299 }
4300
4301 params = 12 + name_len /* includes null */ ;
4302 pSMB->TotalDataCount = 0; /* no EAs */
4303 pSMB->MaxParameterCount = cpu_to_le16(10);
Jeff Laytonc974bef2011-10-11 06:41:32 -04004304 pSMB->MaxDataCount = cpu_to_le16(CIFSMaxBufSize & 0xFFFFFF00);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004305 pSMB->MaxSetupCount = 0;
4306 pSMB->Reserved = 0;
4307 pSMB->Flags = 0;
4308 pSMB->Timeout = 0;
4309 pSMB->Reserved2 = 0;
4310 byte_count = params + 1 /* pad */ ;
4311 pSMB->TotalParameterCount = cpu_to_le16(params);
4312 pSMB->ParameterCount = pSMB->TotalParameterCount;
4313 pSMB->ParameterOffset = cpu_to_le16(
Steve French88274812006-03-09 22:21:45 +00004314 offsetof(struct smb_com_transaction2_ffirst_req, SearchAttributes)
4315 - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004316 pSMB->DataCount = 0;
4317 pSMB->DataOffset = 0;
4318 pSMB->SetupCount = 1; /* one byte, no need to make endian neutral */
4319 pSMB->Reserved3 = 0;
4320 pSMB->SubCommand = cpu_to_le16(TRANS2_FIND_FIRST);
4321 pSMB->SearchAttributes =
4322 cpu_to_le16(ATTR_READONLY | ATTR_HIDDEN | ATTR_SYSTEM |
4323 ATTR_DIRECTORY);
Steve French50c2f752007-07-13 00:33:32 +00004324 pSMB->SearchCount = cpu_to_le16(CIFSMaxBufSize/sizeof(FILE_UNIX_INFO));
Shirish Pargaonkar2608bee2012-05-15 10:19:16 -05004325 pSMB->SearchFlags = cpu_to_le16(search_flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004326 pSMB->InformationLevel = cpu_to_le16(psrch_inf->info_level);
4327
4328 /* BB what should we set StorageType to? Does it matter? BB */
4329 pSMB->SearchStorageType = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00004330 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004331 pSMB->ByteCount = cpu_to_le16(byte_count);
4332
4333 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
4334 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04004335 cifs_stats_inc(&tcon->stats.cifs_stats.num_ffirst);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004336
Steve French88274812006-03-09 22:21:45 +00004337 if (rc) {/* BB add logic to retry regular search if Unix search
4338 rejected unexpectedly by server */
Linus Torvalds1da177e2005-04-16 15:20:36 -07004339 /* BB Add code to handle unsupported level rc */
Joe Perchesb6b38f72010-04-21 03:50:45 +00004340 cFYI(1, "Error in FindFirst = %d", rc);
Steve French1982c342005-08-17 12:38:22 -07004341
Steve French88274812006-03-09 22:21:45 +00004342 cifs_buf_release(pSMB);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004343
4344 /* BB eventually could optimize out free and realloc of buf */
4345 /* for this case */
4346 if (rc == -EAGAIN)
4347 goto findFirstRetry;
4348 } else { /* decode response */
4349 /* BB remember to free buffer if error BB */
4350 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
Steve French790fe572007-07-07 19:25:05 +00004351 if (rc == 0) {
Steve Frenchb77d7532008-10-08 19:13:46 +00004352 unsigned int lnoff;
4353
Linus Torvalds1da177e2005-04-16 15:20:36 -07004354 if (pSMBr->hdr.Flags2 & SMBFLG2_UNICODE)
Steve French4b18f2a2008-04-29 00:06:05 +00004355 psrch_inf->unicode = true;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004356 else
Steve French4b18f2a2008-04-29 00:06:05 +00004357 psrch_inf->unicode = false;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004358
4359 psrch_inf->ntwrk_buf_start = (char *)pSMBr;
Steve Frenchd47d7c12006-02-28 03:45:48 +00004360 psrch_inf->smallBuf = 0;
Steve French50c2f752007-07-13 00:33:32 +00004361 psrch_inf->srch_entries_start =
4362 (char *) &pSMBr->hdr.Protocol +
Linus Torvalds1da177e2005-04-16 15:20:36 -07004363 le16_to_cpu(pSMBr->t2.DataOffset);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004364 parms = (T2_FFIRST_RSP_PARMS *)((char *) &pSMBr->hdr.Protocol +
4365 le16_to_cpu(pSMBr->t2.ParameterOffset));
4366
Steve French790fe572007-07-07 19:25:05 +00004367 if (parms->EndofSearch)
Steve French4b18f2a2008-04-29 00:06:05 +00004368 psrch_inf->endOfSearch = true;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004369 else
Steve French4b18f2a2008-04-29 00:06:05 +00004370 psrch_inf->endOfSearch = false;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004371
Steve French50c2f752007-07-13 00:33:32 +00004372 psrch_inf->entries_in_buffer =
4373 le16_to_cpu(parms->SearchCount);
Steve French60808232006-04-22 15:53:05 +00004374 psrch_inf->index_of_last_entry = 2 /* skip . and .. */ +
Linus Torvalds1da177e2005-04-16 15:20:36 -07004375 psrch_inf->entries_in_buffer;
Steve Frenchb77d7532008-10-08 19:13:46 +00004376 lnoff = le16_to_cpu(parms->LastNameOffset);
Jeff Laytonc974bef2011-10-11 06:41:32 -04004377 if (CIFSMaxBufSize < lnoff) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00004378 cERROR(1, "ignoring corrupt resume name");
Steve Frenchb77d7532008-10-08 19:13:46 +00004379 psrch_inf->last_entry = NULL;
4380 return rc;
4381 }
4382
Steve French0752f152008-10-07 20:03:33 +00004383 psrch_inf->last_entry = psrch_inf->srch_entries_start +
Steve Frenchb77d7532008-10-08 19:13:46 +00004384 lnoff;
4385
Linus Torvalds1da177e2005-04-16 15:20:36 -07004386 *pnetfid = parms->SearchHandle;
4387 } else {
4388 cifs_buf_release(pSMB);
4389 }
4390 }
4391
4392 return rc;
4393}
4394
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04004395int CIFSFindNext(const unsigned int xid, struct cifs_tcon *tcon,
4396 __u16 searchHandle, __u16 search_flags,
4397 struct cifs_search_info *psrch_inf)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004398{
4399 TRANSACTION2_FNEXT_REQ *pSMB = NULL;
4400 TRANSACTION2_FNEXT_RSP *pSMBr = NULL;
Steve Frenchad7a2922008-02-07 23:25:02 +00004401 T2_FNEXT_RSP_PARMS *parms;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004402 char *response_data;
4403 int rc = 0;
Jeff Layton9438fab2011-08-23 07:21:28 -04004404 int bytes_returned;
4405 unsigned int name_len;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004406 __u16 params, byte_count;
4407
Joe Perchesb6b38f72010-04-21 03:50:45 +00004408 cFYI(1, "In FindNext");
Linus Torvalds1da177e2005-04-16 15:20:36 -07004409
Steve French4b18f2a2008-04-29 00:06:05 +00004410 if (psrch_inf->endOfSearch)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004411 return -ENOENT;
4412
4413 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
4414 (void **) &pSMBr);
4415 if (rc)
4416 return rc;
4417
Steve French50c2f752007-07-13 00:33:32 +00004418 params = 14; /* includes 2 bytes of null string, converted to LE below*/
Linus Torvalds1da177e2005-04-16 15:20:36 -07004419 byte_count = 0;
4420 pSMB->TotalDataCount = 0; /* no EAs */
4421 pSMB->MaxParameterCount = cpu_to_le16(8);
Jeff Laytonc974bef2011-10-11 06:41:32 -04004422 pSMB->MaxDataCount = cpu_to_le16(CIFSMaxBufSize & 0xFFFFFF00);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004423 pSMB->MaxSetupCount = 0;
4424 pSMB->Reserved = 0;
4425 pSMB->Flags = 0;
4426 pSMB->Timeout = 0;
4427 pSMB->Reserved2 = 0;
4428 pSMB->ParameterOffset = cpu_to_le16(
4429 offsetof(struct smb_com_transaction2_fnext_req,SearchHandle) - 4);
4430 pSMB->DataCount = 0;
4431 pSMB->DataOffset = 0;
4432 pSMB->SetupCount = 1;
4433 pSMB->Reserved3 = 0;
4434 pSMB->SubCommand = cpu_to_le16(TRANS2_FIND_NEXT);
4435 pSMB->SearchHandle = searchHandle; /* always kept as le */
4436 pSMB->SearchCount =
Steve French630f3f0c2007-10-25 21:17:17 +00004437 cpu_to_le16(CIFSMaxBufSize / sizeof(FILE_UNIX_INFO));
Linus Torvalds1da177e2005-04-16 15:20:36 -07004438 pSMB->InformationLevel = cpu_to_le16(psrch_inf->info_level);
4439 pSMB->ResumeKey = psrch_inf->resume_key;
Shirish Pargaonkar2608bee2012-05-15 10:19:16 -05004440 pSMB->SearchFlags = cpu_to_le16(search_flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004441
4442 name_len = psrch_inf->resume_name_len;
4443 params += name_len;
Steve French790fe572007-07-07 19:25:05 +00004444 if (name_len < PATH_MAX) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07004445 memcpy(pSMB->ResumeFileName, psrch_inf->presume_name, name_len);
4446 byte_count += name_len;
Steve Frenchef6724e2005-08-02 21:31:05 -07004447 /* 14 byte parm len above enough for 2 byte null terminator */
4448 pSMB->ResumeFileName[name_len] = 0;
4449 pSMB->ResumeFileName[name_len+1] = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004450 } else {
4451 rc = -EINVAL;
4452 goto FNext2_err_exit;
4453 }
4454 byte_count = params + 1 /* pad */ ;
4455 pSMB->TotalParameterCount = cpu_to_le16(params);
4456 pSMB->ParameterCount = pSMB->TotalParameterCount;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00004457 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004458 pSMB->ByteCount = cpu_to_le16(byte_count);
Steve French50c2f752007-07-13 00:33:32 +00004459
Linus Torvalds1da177e2005-04-16 15:20:36 -07004460 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
4461 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Pavel Shilovsky44c58182012-05-28 14:16:31 +04004462 cifs_stats_inc(&tcon->stats.cifs_stats.num_fnext);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004463 if (rc) {
4464 if (rc == -EBADF) {
Steve French4b18f2a2008-04-29 00:06:05 +00004465 psrch_inf->endOfSearch = true;
Jeff Layton63534502008-05-12 19:56:05 -07004466 cifs_buf_release(pSMB);
Steve French50c2f752007-07-13 00:33:32 +00004467 rc = 0; /* search probably was closed at end of search*/
Linus Torvalds1da177e2005-04-16 15:20:36 -07004468 } else
Joe Perchesb6b38f72010-04-21 03:50:45 +00004469 cFYI(1, "FindNext returned = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004470 } else { /* decode response */
4471 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
Steve French50c2f752007-07-13 00:33:32 +00004472
Steve French790fe572007-07-07 19:25:05 +00004473 if (rc == 0) {
Steve Frenchb77d7532008-10-08 19:13:46 +00004474 unsigned int lnoff;
4475
Linus Torvalds1da177e2005-04-16 15:20:36 -07004476 /* BB fixme add lock for file (srch_info) struct here */
4477 if (pSMBr->hdr.Flags2 & SMBFLG2_UNICODE)
Steve French4b18f2a2008-04-29 00:06:05 +00004478 psrch_inf->unicode = true;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004479 else
Steve French4b18f2a2008-04-29 00:06:05 +00004480 psrch_inf->unicode = false;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004481 response_data = (char *) &pSMBr->hdr.Protocol +
4482 le16_to_cpu(pSMBr->t2.ParameterOffset);
4483 parms = (T2_FNEXT_RSP_PARMS *)response_data;
4484 response_data = (char *)&pSMBr->hdr.Protocol +
4485 le16_to_cpu(pSMBr->t2.DataOffset);
Steve French790fe572007-07-07 19:25:05 +00004486 if (psrch_inf->smallBuf)
Steve Frenchd47d7c12006-02-28 03:45:48 +00004487 cifs_small_buf_release(
4488 psrch_inf->ntwrk_buf_start);
4489 else
4490 cifs_buf_release(psrch_inf->ntwrk_buf_start);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004491 psrch_inf->srch_entries_start = response_data;
4492 psrch_inf->ntwrk_buf_start = (char *)pSMB;
Steve Frenchd47d7c12006-02-28 03:45:48 +00004493 psrch_inf->smallBuf = 0;
Steve French790fe572007-07-07 19:25:05 +00004494 if (parms->EndofSearch)
Steve French4b18f2a2008-04-29 00:06:05 +00004495 psrch_inf->endOfSearch = true;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004496 else
Steve French4b18f2a2008-04-29 00:06:05 +00004497 psrch_inf->endOfSearch = false;
Steve French50c2f752007-07-13 00:33:32 +00004498 psrch_inf->entries_in_buffer =
4499 le16_to_cpu(parms->SearchCount);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004500 psrch_inf->index_of_last_entry +=
4501 psrch_inf->entries_in_buffer;
Steve Frenchb77d7532008-10-08 19:13:46 +00004502 lnoff = le16_to_cpu(parms->LastNameOffset);
Jeff Laytonc974bef2011-10-11 06:41:32 -04004503 if (CIFSMaxBufSize < lnoff) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00004504 cERROR(1, "ignoring corrupt resume name");
Steve Frenchb77d7532008-10-08 19:13:46 +00004505 psrch_inf->last_entry = NULL;
4506 return rc;
4507 } else
4508 psrch_inf->last_entry =
4509 psrch_inf->srch_entries_start + lnoff;
4510
Joe Perchesb6b38f72010-04-21 03:50:45 +00004511/* cFYI(1, "fnxt2 entries in buf %d index_of_last %d",
4512 psrch_inf->entries_in_buffer, psrch_inf->index_of_last_entry); */
Linus Torvalds1da177e2005-04-16 15:20:36 -07004513
4514 /* BB fixme add unlock here */
4515 }
4516
4517 }
4518
4519 /* BB On error, should we leave previous search buf (and count and
4520 last entry fields) intact or free the previous one? */
4521
4522 /* Note: On -EAGAIN error only caller can retry on handle based calls
4523 since file handle passed in no longer valid */
4524FNext2_err_exit:
4525 if (rc != 0)
4526 cifs_buf_release(pSMB);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004527 return rc;
4528}
4529
4530int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04004531CIFSFindClose(const unsigned int xid, struct cifs_tcon *tcon,
Steve French50c2f752007-07-13 00:33:32 +00004532 const __u16 searchHandle)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004533{
4534 int rc = 0;
4535 FINDCLOSE_REQ *pSMB = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004536
Joe Perchesb6b38f72010-04-21 03:50:45 +00004537 cFYI(1, "In CIFSSMBFindClose");
Linus Torvalds1da177e2005-04-16 15:20:36 -07004538 rc = small_smb_init(SMB_COM_FIND_CLOSE2, 1, tcon, (void **)&pSMB);
4539
4540 /* no sense returning error if session restarted
4541 as file handle has been closed */
Steve French790fe572007-07-07 19:25:05 +00004542 if (rc == -EAGAIN)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004543 return 0;
4544 if (rc)
4545 return rc;
4546
Linus Torvalds1da177e2005-04-16 15:20:36 -07004547 pSMB->FileID = searchHandle;
4548 pSMB->ByteCount = 0;
Pavel Shilovsky792af7b2012-03-23 14:28:02 -04004549 rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
Steve Frenchad7a2922008-02-07 23:25:02 +00004550 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00004551 cERROR(1, "Send error in FindClose = %d", rc);
Steve Frenchad7a2922008-02-07 23:25:02 +00004552
Pavel Shilovsky44c58182012-05-28 14:16:31 +04004553 cifs_stats_inc(&tcon->stats.cifs_stats.num_fclose);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004554
4555 /* Since session is dead, search handle closed on server already */
4556 if (rc == -EAGAIN)
4557 rc = 0;
4558
4559 return rc;
4560}
4561
Linus Torvalds1da177e2005-04-16 15:20:36 -07004562int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04004563CIFSGetSrvInodeNumber(const unsigned int xid, struct cifs_tcon *tcon,
Pavel Shilovsky1208ef12012-05-27 17:34:43 +04004564 const char *search_name, __u64 *inode_number,
Steve French50c2f752007-07-13 00:33:32 +00004565 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004566{
4567 int rc = 0;
4568 TRANSACTION2_QPI_REQ *pSMB = NULL;
4569 TRANSACTION2_QPI_RSP *pSMBr = NULL;
4570 int name_len, bytes_returned;
4571 __u16 params, byte_count;
4572
Pavel Shilovsky1208ef12012-05-27 17:34:43 +04004573 cFYI(1, "In GetSrvInodeNum for %s", search_name);
Steve French790fe572007-07-07 19:25:05 +00004574 if (tcon == NULL)
Steve French50c2f752007-07-13 00:33:32 +00004575 return -ENODEV;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004576
4577GetInodeNumberRetry:
4578 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
Steve French50c2f752007-07-13 00:33:32 +00004579 (void **) &pSMBr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004580 if (rc)
4581 return rc;
4582
Linus Torvalds1da177e2005-04-16 15:20:36 -07004583 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
4584 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06004585 cifsConvertToUTF16((__le16 *) pSMB->FileName,
Pavel Shilovsky1208ef12012-05-27 17:34:43 +04004586 search_name, PATH_MAX, nls_codepage,
Steve Frenchacbbb762012-01-18 22:32:33 -06004587 remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004588 name_len++; /* trailing null */
4589 name_len *= 2;
Steve French50c2f752007-07-13 00:33:32 +00004590 } else { /* BB improve the check for buffer overruns BB */
Pavel Shilovsky1208ef12012-05-27 17:34:43 +04004591 name_len = strnlen(search_name, PATH_MAX);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004592 name_len++; /* trailing null */
Pavel Shilovsky1208ef12012-05-27 17:34:43 +04004593 strncpy(pSMB->FileName, search_name, name_len);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004594 }
4595
4596 params = 2 /* level */ + 4 /* rsrvd */ + name_len /* incl null */ ;
4597 pSMB->TotalDataCount = 0;
4598 pSMB->MaxParameterCount = cpu_to_le16(2);
4599 /* BB find exact max data count below from sess structure BB */
4600 pSMB->MaxDataCount = cpu_to_le16(4000);
4601 pSMB->MaxSetupCount = 0;
4602 pSMB->Reserved = 0;
4603 pSMB->Flags = 0;
4604 pSMB->Timeout = 0;
4605 pSMB->Reserved2 = 0;
4606 pSMB->ParameterOffset = cpu_to_le16(offsetof(
Steve French50c2f752007-07-13 00:33:32 +00004607 struct smb_com_transaction2_qpi_req, InformationLevel) - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004608 pSMB->DataCount = 0;
4609 pSMB->DataOffset = 0;
4610 pSMB->SetupCount = 1;
4611 pSMB->Reserved3 = 0;
4612 pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_PATH_INFORMATION);
4613 byte_count = params + 1 /* pad */ ;
4614 pSMB->TotalParameterCount = cpu_to_le16(params);
4615 pSMB->ParameterCount = pSMB->TotalParameterCount;
4616 pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FILE_INTERNAL_INFO);
4617 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00004618 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004619 pSMB->ByteCount = cpu_to_le16(byte_count);
4620
4621 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
4622 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
4623 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00004624 cFYI(1, "error %d in QueryInternalInfo", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004625 } else {
4626 /* decode response */
4627 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004628 /* BB also check enough total bytes returned */
Jeff Layton820a8032011-05-04 08:05:26 -04004629 if (rc || get_bcc(&pSMBr->hdr) < 2)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004630 /* If rc should we check for EOPNOSUPP and
4631 disable the srvino flag? or in caller? */
4632 rc = -EIO; /* bad smb */
Steve French50c2f752007-07-13 00:33:32 +00004633 else {
Linus Torvalds1da177e2005-04-16 15:20:36 -07004634 __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
4635 __u16 count = le16_to_cpu(pSMBr->t2.DataCount);
Steve French50c2f752007-07-13 00:33:32 +00004636 struct file_internal_info *pfinfo;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004637 /* BB Do we need a cast or hash here ? */
Steve French790fe572007-07-07 19:25:05 +00004638 if (count < 8) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00004639 cFYI(1, "Illegal size ret in QryIntrnlInf");
Linus Torvalds1da177e2005-04-16 15:20:36 -07004640 rc = -EIO;
4641 goto GetInodeNumOut;
4642 }
4643 pfinfo = (struct file_internal_info *)
4644 (data_offset + (char *) &pSMBr->hdr.Protocol);
Steve French85a6dac2009-04-01 05:22:00 +00004645 *inode_number = le64_to_cpu(pfinfo->UniqueId);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004646 }
4647 }
4648GetInodeNumOut:
4649 cifs_buf_release(pSMB);
4650 if (rc == -EAGAIN)
4651 goto GetInodeNumberRetry;
4652 return rc;
4653}
Linus Torvalds1da177e2005-04-16 15:20:36 -07004654
Igor Mammedovfec45852008-05-16 13:06:30 +04004655/* parses DFS refferal V3 structure
4656 * caller is responsible for freeing target_nodes
4657 * returns:
4658 * on success - 0
4659 * on failure - errno
4660 */
4661static int
Steve Frencha1fe78f2008-05-16 18:48:38 +00004662parse_DFS_referrals(TRANSACTION2_GET_DFS_REFER_RSP *pSMBr,
Igor Mammedovfec45852008-05-16 13:06:30 +04004663 unsigned int *num_of_nodes,
4664 struct dfs_info3_param **target_nodes,
Igor Mammedov2c556082008-10-23 13:58:42 +04004665 const struct nls_table *nls_codepage, int remap,
4666 const char *searchName)
Igor Mammedovfec45852008-05-16 13:06:30 +04004667{
4668 int i, rc = 0;
4669 char *data_end;
4670 bool is_unicode;
4671 struct dfs_referral_level_3 *ref;
4672
Harvey Harrison5ca33c62008-07-23 17:45:58 -07004673 if (pSMBr->hdr.Flags2 & SMBFLG2_UNICODE)
4674 is_unicode = true;
4675 else
4676 is_unicode = false;
Igor Mammedovfec45852008-05-16 13:06:30 +04004677 *num_of_nodes = le16_to_cpu(pSMBr->NumberOfReferrals);
4678
4679 if (*num_of_nodes < 1) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00004680 cERROR(1, "num_referrals: must be at least > 0,"
Jeff Laytonac3aa2f2012-07-23 13:14:28 -04004681 "but we get num_referrals = %d", *num_of_nodes);
Igor Mammedovfec45852008-05-16 13:06:30 +04004682 rc = -EINVAL;
Steve Frencha1fe78f2008-05-16 18:48:38 +00004683 goto parse_DFS_referrals_exit;
Igor Mammedovfec45852008-05-16 13:06:30 +04004684 }
4685
4686 ref = (struct dfs_referral_level_3 *) &(pSMBr->referrals);
Al Viro1d92cfd2008-06-02 10:59:02 +01004687 if (ref->VersionNumber != cpu_to_le16(3)) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00004688 cERROR(1, "Referrals of V%d version are not supported,"
4689 "should be V3", le16_to_cpu(ref->VersionNumber));
Igor Mammedovfec45852008-05-16 13:06:30 +04004690 rc = -EINVAL;
Steve Frencha1fe78f2008-05-16 18:48:38 +00004691 goto parse_DFS_referrals_exit;
Igor Mammedovfec45852008-05-16 13:06:30 +04004692 }
4693
4694 /* get the upper boundary of the resp buffer */
4695 data_end = (char *)(&(pSMBr->PathConsumed)) +
4696 le16_to_cpu(pSMBr->t2.DataCount);
4697
Jeff Laytonac3aa2f2012-07-23 13:14:28 -04004698 cFYI(1, "num_referrals: %d dfs flags: 0x%x ...",
Igor Mammedovfec45852008-05-16 13:06:30 +04004699 *num_of_nodes,
Joe Perchesb6b38f72010-04-21 03:50:45 +00004700 le32_to_cpu(pSMBr->DFSFlags));
Igor Mammedovfec45852008-05-16 13:06:30 +04004701
4702 *target_nodes = kzalloc(sizeof(struct dfs_info3_param) *
4703 *num_of_nodes, GFP_KERNEL);
4704 if (*target_nodes == NULL) {
Jeff Laytonac3aa2f2012-07-23 13:14:28 -04004705 cERROR(1, "Failed to allocate buffer for target_nodes");
Igor Mammedovfec45852008-05-16 13:06:30 +04004706 rc = -ENOMEM;
Steve Frencha1fe78f2008-05-16 18:48:38 +00004707 goto parse_DFS_referrals_exit;
Igor Mammedovfec45852008-05-16 13:06:30 +04004708 }
4709
Daniel Mack3ad2f3f2010-02-03 08:01:28 +08004710 /* collect necessary data from referrals */
Igor Mammedovfec45852008-05-16 13:06:30 +04004711 for (i = 0; i < *num_of_nodes; i++) {
4712 char *temp;
4713 int max_len;
4714 struct dfs_info3_param *node = (*target_nodes)+i;
4715
Steve French0e0d2cf2009-05-01 05:27:32 +00004716 node->flags = le32_to_cpu(pSMBr->DFSFlags);
Igor Mammedov2c556082008-10-23 13:58:42 +04004717 if (is_unicode) {
Jeff Layton331c3132008-12-17 06:31:53 -05004718 __le16 *tmp = kmalloc(strlen(searchName)*2 + 2,
4719 GFP_KERNEL);
Steve French2920ee22009-08-31 15:27:26 +00004720 if (tmp == NULL) {
4721 rc = -ENOMEM;
4722 goto parse_DFS_referrals_exit;
4723 }
Steve Frenchacbbb762012-01-18 22:32:33 -06004724 cifsConvertToUTF16((__le16 *) tmp, searchName,
4725 PATH_MAX, nls_codepage, remap);
4726 node->path_consumed = cifs_utf16_bytes(tmp,
Jeff Layton69f801f2009-04-30 06:46:32 -04004727 le16_to_cpu(pSMBr->PathConsumed),
Igor Mammedov2c556082008-10-23 13:58:42 +04004728 nls_codepage);
4729 kfree(tmp);
4730 } else
4731 node->path_consumed = le16_to_cpu(pSMBr->PathConsumed);
4732
Igor Mammedovfec45852008-05-16 13:06:30 +04004733 node->server_type = le16_to_cpu(ref->ServerType);
4734 node->ref_flag = le16_to_cpu(ref->ReferralEntryFlags);
4735
4736 /* copy DfsPath */
4737 temp = (char *)ref + le16_to_cpu(ref->DfsPathOffset);
4738 max_len = data_end - temp;
Steve Frenchacbbb762012-01-18 22:32:33 -06004739 node->path_name = cifs_strndup_from_utf16(temp, max_len,
4740 is_unicode, nls_codepage);
Jeff Laytond8e2f532009-05-14 07:46:59 -04004741 if (!node->path_name) {
4742 rc = -ENOMEM;
Steve Frencha1fe78f2008-05-16 18:48:38 +00004743 goto parse_DFS_referrals_exit;
Jeff Layton066ce682009-04-30 07:16:14 -04004744 }
Igor Mammedovfec45852008-05-16 13:06:30 +04004745
4746 /* copy link target UNC */
4747 temp = (char *)ref + le16_to_cpu(ref->NetworkAddressOffset);
4748 max_len = data_end - temp;
Steve Frenchacbbb762012-01-18 22:32:33 -06004749 node->node_name = cifs_strndup_from_utf16(temp, max_len,
4750 is_unicode, nls_codepage);
Stefan Metzmacherd8f27992012-05-04 00:19:28 +02004751 if (!node->node_name) {
Jeff Laytond8e2f532009-05-14 07:46:59 -04004752 rc = -ENOMEM;
Stefan Metzmacherd8f27992012-05-04 00:19:28 +02004753 goto parse_DFS_referrals_exit;
4754 }
4755
4756 ref++;
Igor Mammedovfec45852008-05-16 13:06:30 +04004757 }
4758
Steve Frencha1fe78f2008-05-16 18:48:38 +00004759parse_DFS_referrals_exit:
Igor Mammedovfec45852008-05-16 13:06:30 +04004760 if (rc) {
4761 free_dfs_info_array(*target_nodes, *num_of_nodes);
4762 *target_nodes = NULL;
4763 *num_of_nodes = 0;
4764 }
4765 return rc;
4766}
4767
Linus Torvalds1da177e2005-04-16 15:20:36 -07004768int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04004769CIFSGetDFSRefer(const unsigned int xid, struct cifs_ses *ses,
Pavel Shilovskyb669f332012-05-27 20:21:53 +04004770 const char *search_name, struct dfs_info3_param **target_nodes,
Steve Frenchc2cf07d2008-05-15 06:20:02 +00004771 unsigned int *num_of_nodes,
Steve French737b7582005-04-28 22:41:06 -07004772 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004773{
4774/* TRANS2_GET_DFS_REFERRAL */
4775 TRANSACTION2_GET_DFS_REFER_REQ *pSMB = NULL;
4776 TRANSACTION2_GET_DFS_REFER_RSP *pSMBr = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004777 int rc = 0;
4778 int bytes_returned;
4779 int name_len;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004780 __u16 params, byte_count;
Steve Frenchc2cf07d2008-05-15 06:20:02 +00004781 *num_of_nodes = 0;
4782 *target_nodes = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004783
Pavel Shilovskyb669f332012-05-27 20:21:53 +04004784 cFYI(1, "In GetDFSRefer the path %s", search_name);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004785 if (ses == NULL)
4786 return -ENODEV;
4787getDFSRetry:
4788 rc = smb_init(SMB_COM_TRANSACTION2, 15, NULL, (void **) &pSMB,
4789 (void **) &pSMBr);
4790 if (rc)
4791 return rc;
Steve French50c2f752007-07-13 00:33:32 +00004792
4793 /* server pointer checked in called function,
Steve French1982c342005-08-17 12:38:22 -07004794 but should never be null here anyway */
Pavel Shilovsky88257362012-05-23 14:01:59 +04004795 pSMB->hdr.Mid = get_next_mid(ses->server);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004796 pSMB->hdr.Tid = ses->ipc_tid;
4797 pSMB->hdr.Uid = ses->Suid;
Steve French26f57362007-08-30 22:09:15 +00004798 if (ses->capabilities & CAP_STATUS32)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004799 pSMB->hdr.Flags2 |= SMBFLG2_ERR_STATUS;
Steve French26f57362007-08-30 22:09:15 +00004800 if (ses->capabilities & CAP_DFS)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004801 pSMB->hdr.Flags2 |= SMBFLG2_DFS;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004802
4803 if (ses->capabilities & CAP_UNICODE) {
4804 pSMB->hdr.Flags2 |= SMBFLG2_UNICODE;
4805 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06004806 cifsConvertToUTF16((__le16 *) pSMB->RequestFileName,
Pavel Shilovskyb669f332012-05-27 20:21:53 +04004807 search_name, PATH_MAX, nls_codepage,
Steve Frenchacbbb762012-01-18 22:32:33 -06004808 remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004809 name_len++; /* trailing null */
4810 name_len *= 2;
Steve French50c2f752007-07-13 00:33:32 +00004811 } else { /* BB improve the check for buffer overruns BB */
Pavel Shilovskyb669f332012-05-27 20:21:53 +04004812 name_len = strnlen(search_name, PATH_MAX);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004813 name_len++; /* trailing null */
Pavel Shilovskyb669f332012-05-27 20:21:53 +04004814 strncpy(pSMB->RequestFileName, search_name, name_len);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004815 }
4816
Steve French790fe572007-07-07 19:25:05 +00004817 if (ses->server) {
Steve French96daf2b2011-05-27 04:34:02 +00004818 if (ses->server->sec_mode &
Steve French1a4e15a2006-10-12 21:33:51 +00004819 (SECMODE_SIGN_REQUIRED | SECMODE_SIGN_ENABLED))
4820 pSMB->hdr.Flags2 |= SMBFLG2_SECURITY_SIGNATURE;
4821 }
4822
Steve French50c2f752007-07-13 00:33:32 +00004823 pSMB->hdr.Uid = ses->Suid;
Steve French1a4e15a2006-10-12 21:33:51 +00004824
Linus Torvalds1da177e2005-04-16 15:20:36 -07004825 params = 2 /* level */ + name_len /*includes null */ ;
4826 pSMB->TotalDataCount = 0;
4827 pSMB->DataCount = 0;
4828 pSMB->DataOffset = 0;
4829 pSMB->MaxParameterCount = 0;
Steve French582d21e2008-05-13 04:54:12 +00004830 /* BB find exact max SMB PDU from sess structure BB */
4831 pSMB->MaxDataCount = cpu_to_le16(4000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004832 pSMB->MaxSetupCount = 0;
4833 pSMB->Reserved = 0;
4834 pSMB->Flags = 0;
4835 pSMB->Timeout = 0;
4836 pSMB->Reserved2 = 0;
4837 pSMB->ParameterOffset = cpu_to_le16(offsetof(
Steve French50c2f752007-07-13 00:33:32 +00004838 struct smb_com_transaction2_get_dfs_refer_req, MaxReferralLevel) - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004839 pSMB->SetupCount = 1;
4840 pSMB->Reserved3 = 0;
4841 pSMB->SubCommand = cpu_to_le16(TRANS2_GET_DFS_REFERRAL);
4842 byte_count = params + 3 /* pad */ ;
4843 pSMB->ParameterCount = cpu_to_le16(params);
4844 pSMB->TotalParameterCount = pSMB->ParameterCount;
4845 pSMB->MaxReferralLevel = cpu_to_le16(3);
Steve Frenchbe8e3b02011-04-29 05:40:20 +00004846 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004847 pSMB->ByteCount = cpu_to_le16(byte_count);
4848
4849 rc = SendReceive(xid, ses, (struct smb_hdr *) pSMB,
4850 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
4851 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00004852 cFYI(1, "Send error in GetDFSRefer = %d", rc);
Steve Frenchc2cf07d2008-05-15 06:20:02 +00004853 goto GetDFSRefExit;
4854 }
4855 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004856
Steve Frenchc2cf07d2008-05-15 06:20:02 +00004857 /* BB Also check if enough total bytes returned? */
Jeff Layton820a8032011-05-04 08:05:26 -04004858 if (rc || get_bcc(&pSMBr->hdr) < 17) {
Steve Frenchc2cf07d2008-05-15 06:20:02 +00004859 rc = -EIO; /* bad smb */
Igor Mammedovfec45852008-05-16 13:06:30 +04004860 goto GetDFSRefExit;
Linus Torvalds1da177e2005-04-16 15:20:36 -07004861 }
Igor Mammedovfec45852008-05-16 13:06:30 +04004862
Joe Perchesb6b38f72010-04-21 03:50:45 +00004863 cFYI(1, "Decoding GetDFSRefer response BCC: %d Offset %d",
Jeff Layton820a8032011-05-04 08:05:26 -04004864 get_bcc(&pSMBr->hdr),
Joe Perchesb6b38f72010-04-21 03:50:45 +00004865 le16_to_cpu(pSMBr->t2.DataOffset));
Igor Mammedovfec45852008-05-16 13:06:30 +04004866
4867 /* parse returned result into more usable form */
Steve Frencha1fe78f2008-05-16 18:48:38 +00004868 rc = parse_DFS_referrals(pSMBr, num_of_nodes,
Igor Mammedov2c556082008-10-23 13:58:42 +04004869 target_nodes, nls_codepage, remap,
Pavel Shilovskyb669f332012-05-27 20:21:53 +04004870 search_name);
Igor Mammedovfec45852008-05-16 13:06:30 +04004871
Linus Torvalds1da177e2005-04-16 15:20:36 -07004872GetDFSRefExit:
Steve French0d817bc2008-05-22 02:02:03 +00004873 cifs_buf_release(pSMB);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004874
4875 if (rc == -EAGAIN)
4876 goto getDFSRetry;
4877
4878 return rc;
4879}
4880
Steve French20962432005-09-21 22:05:57 -07004881/* Query File System Info such as free space to old servers such as Win 9x */
4882int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04004883SMBOldQFSInfo(const unsigned int xid, struct cifs_tcon *tcon,
4884 struct kstatfs *FSData)
Steve French20962432005-09-21 22:05:57 -07004885{
4886/* level 0x01 SMB_QUERY_FILE_SYSTEM_INFO */
4887 TRANSACTION2_QFSI_REQ *pSMB = NULL;
4888 TRANSACTION2_QFSI_RSP *pSMBr = NULL;
4889 FILE_SYSTEM_ALLOC_INFO *response_data;
4890 int rc = 0;
4891 int bytes_returned = 0;
4892 __u16 params, byte_count;
4893
Joe Perchesb6b38f72010-04-21 03:50:45 +00004894 cFYI(1, "OldQFSInfo");
Steve French20962432005-09-21 22:05:57 -07004895oldQFSInfoRetry:
4896 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
4897 (void **) &pSMBr);
4898 if (rc)
4899 return rc;
Steve French20962432005-09-21 22:05:57 -07004900
4901 params = 2; /* level */
4902 pSMB->TotalDataCount = 0;
4903 pSMB->MaxParameterCount = cpu_to_le16(2);
4904 pSMB->MaxDataCount = cpu_to_le16(1000);
4905 pSMB->MaxSetupCount = 0;
4906 pSMB->Reserved = 0;
4907 pSMB->Flags = 0;
4908 pSMB->Timeout = 0;
4909 pSMB->Reserved2 = 0;
4910 byte_count = params + 1 /* pad */ ;
4911 pSMB->TotalParameterCount = cpu_to_le16(params);
4912 pSMB->ParameterCount = pSMB->TotalParameterCount;
4913 pSMB->ParameterOffset = cpu_to_le16(offsetof(
4914 struct smb_com_transaction2_qfsi_req, InformationLevel) - 4);
4915 pSMB->DataCount = 0;
4916 pSMB->DataOffset = 0;
4917 pSMB->SetupCount = 1;
4918 pSMB->Reserved3 = 0;
4919 pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FS_INFORMATION);
4920 pSMB->InformationLevel = cpu_to_le16(SMB_INFO_ALLOCATION);
Steve Frenchbe8e3b02011-04-29 05:40:20 +00004921 inc_rfc1001_len(pSMB, byte_count);
Steve French20962432005-09-21 22:05:57 -07004922 pSMB->ByteCount = cpu_to_le16(byte_count);
4923
4924 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
4925 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
4926 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00004927 cFYI(1, "Send error in QFSInfo = %d", rc);
Steve French20962432005-09-21 22:05:57 -07004928 } else { /* decode response */
4929 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
4930
Jeff Layton820a8032011-05-04 08:05:26 -04004931 if (rc || get_bcc(&pSMBr->hdr) < 18)
Steve French20962432005-09-21 22:05:57 -07004932 rc = -EIO; /* bad smb */
4933 else {
4934 __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
Joe Perchesb6b38f72010-04-21 03:50:45 +00004935 cFYI(1, "qfsinf resp BCC: %d Offset %d",
Jeff Layton820a8032011-05-04 08:05:26 -04004936 get_bcc(&pSMBr->hdr), data_offset);
Steve French20962432005-09-21 22:05:57 -07004937
Steve French50c2f752007-07-13 00:33:32 +00004938 response_data = (FILE_SYSTEM_ALLOC_INFO *)
Steve French20962432005-09-21 22:05:57 -07004939 (((char *) &pSMBr->hdr.Protocol) + data_offset);
4940 FSData->f_bsize =
4941 le16_to_cpu(response_data->BytesPerSector) *
4942 le32_to_cpu(response_data->
4943 SectorsPerAllocationUnit);
4944 FSData->f_blocks =
Steve French50c2f752007-07-13 00:33:32 +00004945 le32_to_cpu(response_data->TotalAllocationUnits);
Steve French20962432005-09-21 22:05:57 -07004946 FSData->f_bfree = FSData->f_bavail =
4947 le32_to_cpu(response_data->FreeAllocationUnits);
Joe Perchesb6b38f72010-04-21 03:50:45 +00004948 cFYI(1, "Blocks: %lld Free: %lld Block size %ld",
4949 (unsigned long long)FSData->f_blocks,
4950 (unsigned long long)FSData->f_bfree,
4951 FSData->f_bsize);
Steve French20962432005-09-21 22:05:57 -07004952 }
4953 }
4954 cifs_buf_release(pSMB);
4955
4956 if (rc == -EAGAIN)
4957 goto oldQFSInfoRetry;
4958
4959 return rc;
4960}
4961
Linus Torvalds1da177e2005-04-16 15:20:36 -07004962int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04004963CIFSSMBQFSInfo(const unsigned int xid, struct cifs_tcon *tcon,
4964 struct kstatfs *FSData)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004965{
4966/* level 0x103 SMB_QUERY_FILE_SYSTEM_INFO */
4967 TRANSACTION2_QFSI_REQ *pSMB = NULL;
4968 TRANSACTION2_QFSI_RSP *pSMBr = NULL;
4969 FILE_SYSTEM_INFO *response_data;
4970 int rc = 0;
4971 int bytes_returned = 0;
4972 __u16 params, byte_count;
4973
Joe Perchesb6b38f72010-04-21 03:50:45 +00004974 cFYI(1, "In QFSInfo");
Linus Torvalds1da177e2005-04-16 15:20:36 -07004975QFSInfoRetry:
4976 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
4977 (void **) &pSMBr);
4978 if (rc)
4979 return rc;
4980
4981 params = 2; /* level */
4982 pSMB->TotalDataCount = 0;
4983 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French20962432005-09-21 22:05:57 -07004984 pSMB->MaxDataCount = cpu_to_le16(1000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004985 pSMB->MaxSetupCount = 0;
4986 pSMB->Reserved = 0;
4987 pSMB->Flags = 0;
4988 pSMB->Timeout = 0;
4989 pSMB->Reserved2 = 0;
4990 byte_count = params + 1 /* pad */ ;
4991 pSMB->TotalParameterCount = cpu_to_le16(params);
4992 pSMB->ParameterCount = pSMB->TotalParameterCount;
4993 pSMB->ParameterOffset = cpu_to_le16(offsetof(
Steve French50c2f752007-07-13 00:33:32 +00004994 struct smb_com_transaction2_qfsi_req, InformationLevel) - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004995 pSMB->DataCount = 0;
4996 pSMB->DataOffset = 0;
4997 pSMB->SetupCount = 1;
4998 pSMB->Reserved3 = 0;
4999 pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FS_INFORMATION);
5000 pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FS_SIZE_INFO);
Steve Frenchbe8e3b02011-04-29 05:40:20 +00005001 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005002 pSMB->ByteCount = cpu_to_le16(byte_count);
5003
5004 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
5005 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
5006 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00005007 cFYI(1, "Send error in QFSInfo = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005008 } else { /* decode response */
Steve French50c2f752007-07-13 00:33:32 +00005009 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005010
Jeff Layton820a8032011-05-04 08:05:26 -04005011 if (rc || get_bcc(&pSMBr->hdr) < 24)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005012 rc = -EIO; /* bad smb */
5013 else {
5014 __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005015
5016 response_data =
5017 (FILE_SYSTEM_INFO
5018 *) (((char *) &pSMBr->hdr.Protocol) +
5019 data_offset);
5020 FSData->f_bsize =
5021 le32_to_cpu(response_data->BytesPerSector) *
5022 le32_to_cpu(response_data->
5023 SectorsPerAllocationUnit);
5024 FSData->f_blocks =
5025 le64_to_cpu(response_data->TotalAllocationUnits);
5026 FSData->f_bfree = FSData->f_bavail =
5027 le64_to_cpu(response_data->FreeAllocationUnits);
Joe Perchesb6b38f72010-04-21 03:50:45 +00005028 cFYI(1, "Blocks: %lld Free: %lld Block size %ld",
5029 (unsigned long long)FSData->f_blocks,
5030 (unsigned long long)FSData->f_bfree,
5031 FSData->f_bsize);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005032 }
5033 }
5034 cifs_buf_release(pSMB);
5035
5036 if (rc == -EAGAIN)
5037 goto QFSInfoRetry;
5038
5039 return rc;
5040}
5041
5042int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04005043CIFSSMBQFSAttributeInfo(const unsigned int xid, struct cifs_tcon *tcon)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005044{
5045/* level 0x105 SMB_QUERY_FILE_SYSTEM_INFO */
5046 TRANSACTION2_QFSI_REQ *pSMB = NULL;
5047 TRANSACTION2_QFSI_RSP *pSMBr = NULL;
5048 FILE_SYSTEM_ATTRIBUTE_INFO *response_data;
5049 int rc = 0;
5050 int bytes_returned = 0;
5051 __u16 params, byte_count;
5052
Joe Perchesb6b38f72010-04-21 03:50:45 +00005053 cFYI(1, "In QFSAttributeInfo");
Linus Torvalds1da177e2005-04-16 15:20:36 -07005054QFSAttributeRetry:
5055 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
5056 (void **) &pSMBr);
5057 if (rc)
5058 return rc;
5059
5060 params = 2; /* level */
5061 pSMB->TotalDataCount = 0;
5062 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French582d21e2008-05-13 04:54:12 +00005063 /* BB find exact max SMB PDU from sess structure BB */
5064 pSMB->MaxDataCount = cpu_to_le16(1000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005065 pSMB->MaxSetupCount = 0;
5066 pSMB->Reserved = 0;
5067 pSMB->Flags = 0;
5068 pSMB->Timeout = 0;
5069 pSMB->Reserved2 = 0;
5070 byte_count = params + 1 /* pad */ ;
5071 pSMB->TotalParameterCount = cpu_to_le16(params);
5072 pSMB->ParameterCount = pSMB->TotalParameterCount;
5073 pSMB->ParameterOffset = cpu_to_le16(offsetof(
Steve French50c2f752007-07-13 00:33:32 +00005074 struct smb_com_transaction2_qfsi_req, InformationLevel) - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005075 pSMB->DataCount = 0;
5076 pSMB->DataOffset = 0;
5077 pSMB->SetupCount = 1;
5078 pSMB->Reserved3 = 0;
5079 pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FS_INFORMATION);
5080 pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FS_ATTRIBUTE_INFO);
Steve Frenchbe8e3b02011-04-29 05:40:20 +00005081 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005082 pSMB->ByteCount = cpu_to_le16(byte_count);
5083
5084 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
5085 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
5086 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00005087 cERROR(1, "Send error in QFSAttributeInfo = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005088 } else { /* decode response */
5089 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
5090
Jeff Layton820a8032011-05-04 08:05:26 -04005091 if (rc || get_bcc(&pSMBr->hdr) < 13) {
Steve French50c2f752007-07-13 00:33:32 +00005092 /* BB also check if enough bytes returned */
Linus Torvalds1da177e2005-04-16 15:20:36 -07005093 rc = -EIO; /* bad smb */
5094 } else {
5095 __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
5096 response_data =
5097 (FILE_SYSTEM_ATTRIBUTE_INFO
5098 *) (((char *) &pSMBr->hdr.Protocol) +
5099 data_offset);
5100 memcpy(&tcon->fsAttrInfo, response_data,
Steve French26f57362007-08-30 22:09:15 +00005101 sizeof(FILE_SYSTEM_ATTRIBUTE_INFO));
Linus Torvalds1da177e2005-04-16 15:20:36 -07005102 }
5103 }
5104 cifs_buf_release(pSMB);
5105
5106 if (rc == -EAGAIN)
5107 goto QFSAttributeRetry;
5108
5109 return rc;
5110}
5111
5112int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04005113CIFSSMBQFSDeviceInfo(const unsigned int xid, struct cifs_tcon *tcon)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005114{
5115/* level 0x104 SMB_QUERY_FILE_SYSTEM_INFO */
5116 TRANSACTION2_QFSI_REQ *pSMB = NULL;
5117 TRANSACTION2_QFSI_RSP *pSMBr = NULL;
5118 FILE_SYSTEM_DEVICE_INFO *response_data;
5119 int rc = 0;
5120 int bytes_returned = 0;
5121 __u16 params, byte_count;
5122
Joe Perchesb6b38f72010-04-21 03:50:45 +00005123 cFYI(1, "In QFSDeviceInfo");
Linus Torvalds1da177e2005-04-16 15:20:36 -07005124QFSDeviceRetry:
5125 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
5126 (void **) &pSMBr);
5127 if (rc)
5128 return rc;
5129
5130 params = 2; /* level */
5131 pSMB->TotalDataCount = 0;
5132 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French582d21e2008-05-13 04:54:12 +00005133 /* BB find exact max SMB PDU from sess structure BB */
5134 pSMB->MaxDataCount = cpu_to_le16(1000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005135 pSMB->MaxSetupCount = 0;
5136 pSMB->Reserved = 0;
5137 pSMB->Flags = 0;
5138 pSMB->Timeout = 0;
5139 pSMB->Reserved2 = 0;
5140 byte_count = params + 1 /* pad */ ;
5141 pSMB->TotalParameterCount = cpu_to_le16(params);
5142 pSMB->ParameterCount = pSMB->TotalParameterCount;
5143 pSMB->ParameterOffset = cpu_to_le16(offsetof(
Steve French50c2f752007-07-13 00:33:32 +00005144 struct smb_com_transaction2_qfsi_req, InformationLevel) - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005145
5146 pSMB->DataCount = 0;
5147 pSMB->DataOffset = 0;
5148 pSMB->SetupCount = 1;
5149 pSMB->Reserved3 = 0;
5150 pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FS_INFORMATION);
5151 pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_FS_DEVICE_INFO);
Steve Frenchbe8e3b02011-04-29 05:40:20 +00005152 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005153 pSMB->ByteCount = cpu_to_le16(byte_count);
5154
5155 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
5156 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
5157 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00005158 cFYI(1, "Send error in QFSDeviceInfo = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005159 } else { /* decode response */
5160 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
5161
Jeff Layton820a8032011-05-04 08:05:26 -04005162 if (rc || get_bcc(&pSMBr->hdr) <
5163 sizeof(FILE_SYSTEM_DEVICE_INFO))
Linus Torvalds1da177e2005-04-16 15:20:36 -07005164 rc = -EIO; /* bad smb */
5165 else {
5166 __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
5167 response_data =
Steve French737b7582005-04-28 22:41:06 -07005168 (FILE_SYSTEM_DEVICE_INFO *)
5169 (((char *) &pSMBr->hdr.Protocol) +
Linus Torvalds1da177e2005-04-16 15:20:36 -07005170 data_offset);
5171 memcpy(&tcon->fsDevInfo, response_data,
Steve French26f57362007-08-30 22:09:15 +00005172 sizeof(FILE_SYSTEM_DEVICE_INFO));
Linus Torvalds1da177e2005-04-16 15:20:36 -07005173 }
5174 }
5175 cifs_buf_release(pSMB);
5176
5177 if (rc == -EAGAIN)
5178 goto QFSDeviceRetry;
5179
5180 return rc;
5181}
5182
5183int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04005184CIFSSMBQFSUnixInfo(const unsigned int xid, struct cifs_tcon *tcon)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005185{
5186/* level 0x200 SMB_QUERY_CIFS_UNIX_INFO */
5187 TRANSACTION2_QFSI_REQ *pSMB = NULL;
5188 TRANSACTION2_QFSI_RSP *pSMBr = NULL;
5189 FILE_SYSTEM_UNIX_INFO *response_data;
5190 int rc = 0;
5191 int bytes_returned = 0;
5192 __u16 params, byte_count;
5193
Joe Perchesb6b38f72010-04-21 03:50:45 +00005194 cFYI(1, "In QFSUnixInfo");
Linus Torvalds1da177e2005-04-16 15:20:36 -07005195QFSUnixRetry:
Jeff Laytonf5695992010-09-29 15:27:08 -04005196 rc = smb_init_no_reconnect(SMB_COM_TRANSACTION2, 15, tcon,
5197 (void **) &pSMB, (void **) &pSMBr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005198 if (rc)
5199 return rc;
5200
5201 params = 2; /* level */
5202 pSMB->TotalDataCount = 0;
5203 pSMB->DataCount = 0;
5204 pSMB->DataOffset = 0;
5205 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French582d21e2008-05-13 04:54:12 +00005206 /* BB find exact max SMB PDU from sess structure BB */
5207 pSMB->MaxDataCount = cpu_to_le16(100);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005208 pSMB->MaxSetupCount = 0;
5209 pSMB->Reserved = 0;
5210 pSMB->Flags = 0;
5211 pSMB->Timeout = 0;
5212 pSMB->Reserved2 = 0;
5213 byte_count = params + 1 /* pad */ ;
5214 pSMB->ParameterCount = cpu_to_le16(params);
5215 pSMB->TotalParameterCount = pSMB->ParameterCount;
Steve French50c2f752007-07-13 00:33:32 +00005216 pSMB->ParameterOffset = cpu_to_le16(offsetof(struct
5217 smb_com_transaction2_qfsi_req, InformationLevel) - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005218 pSMB->SetupCount = 1;
5219 pSMB->Reserved3 = 0;
5220 pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FS_INFORMATION);
5221 pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_CIFS_UNIX_INFO);
Steve Frenchbe8e3b02011-04-29 05:40:20 +00005222 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005223 pSMB->ByteCount = cpu_to_le16(byte_count);
5224
5225 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
5226 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
5227 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00005228 cERROR(1, "Send error in QFSUnixInfo = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005229 } else { /* decode response */
5230 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
5231
Jeff Layton820a8032011-05-04 08:05:26 -04005232 if (rc || get_bcc(&pSMBr->hdr) < 13) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07005233 rc = -EIO; /* bad smb */
5234 } else {
5235 __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
5236 response_data =
5237 (FILE_SYSTEM_UNIX_INFO
5238 *) (((char *) &pSMBr->hdr.Protocol) +
5239 data_offset);
5240 memcpy(&tcon->fsUnixInfo, response_data,
Steve French26f57362007-08-30 22:09:15 +00005241 sizeof(FILE_SYSTEM_UNIX_INFO));
Linus Torvalds1da177e2005-04-16 15:20:36 -07005242 }
5243 }
5244 cifs_buf_release(pSMB);
5245
5246 if (rc == -EAGAIN)
5247 goto QFSUnixRetry;
5248
5249
5250 return rc;
5251}
5252
Jeremy Allisonac670552005-06-22 17:26:35 -07005253int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04005254CIFSSMBSetFSUnixInfo(const unsigned int xid, struct cifs_tcon *tcon, __u64 cap)
Jeremy Allisonac670552005-06-22 17:26:35 -07005255{
5256/* level 0x200 SMB_SET_CIFS_UNIX_INFO */
5257 TRANSACTION2_SETFSI_REQ *pSMB = NULL;
5258 TRANSACTION2_SETFSI_RSP *pSMBr = NULL;
5259 int rc = 0;
5260 int bytes_returned = 0;
5261 __u16 params, param_offset, offset, byte_count;
5262
Joe Perchesb6b38f72010-04-21 03:50:45 +00005263 cFYI(1, "In SETFSUnixInfo");
Jeremy Allisonac670552005-06-22 17:26:35 -07005264SETFSUnixRetry:
Steve Frenchf26282c2006-03-01 09:17:37 +00005265 /* BB switch to small buf init to save memory */
Jeff Laytonf5695992010-09-29 15:27:08 -04005266 rc = smb_init_no_reconnect(SMB_COM_TRANSACTION2, 15, tcon,
5267 (void **) &pSMB, (void **) &pSMBr);
Jeremy Allisonac670552005-06-22 17:26:35 -07005268 if (rc)
5269 return rc;
5270
5271 params = 4; /* 2 bytes zero followed by info level. */
5272 pSMB->MaxSetupCount = 0;
5273 pSMB->Reserved = 0;
5274 pSMB->Flags = 0;
5275 pSMB->Timeout = 0;
5276 pSMB->Reserved2 = 0;
Steve French50c2f752007-07-13 00:33:32 +00005277 param_offset = offsetof(struct smb_com_transaction2_setfsi_req, FileNum)
5278 - 4;
Jeremy Allisonac670552005-06-22 17:26:35 -07005279 offset = param_offset + params;
5280
5281 pSMB->MaxParameterCount = cpu_to_le16(4);
Steve French582d21e2008-05-13 04:54:12 +00005282 /* BB find exact max SMB PDU from sess structure BB */
5283 pSMB->MaxDataCount = cpu_to_le16(100);
Jeremy Allisonac670552005-06-22 17:26:35 -07005284 pSMB->SetupCount = 1;
5285 pSMB->Reserved3 = 0;
5286 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FS_INFORMATION);
5287 byte_count = 1 /* pad */ + params + 12;
5288
5289 pSMB->DataCount = cpu_to_le16(12);
5290 pSMB->ParameterCount = cpu_to_le16(params);
5291 pSMB->TotalDataCount = pSMB->DataCount;
5292 pSMB->TotalParameterCount = pSMB->ParameterCount;
5293 pSMB->ParameterOffset = cpu_to_le16(param_offset);
5294 pSMB->DataOffset = cpu_to_le16(offset);
5295
5296 /* Params. */
5297 pSMB->FileNum = 0;
5298 pSMB->InformationLevel = cpu_to_le16(SMB_SET_CIFS_UNIX_INFO);
5299
5300 /* Data. */
5301 pSMB->ClientUnixMajor = cpu_to_le16(CIFS_UNIX_MAJOR_VERSION);
5302 pSMB->ClientUnixMinor = cpu_to_le16(CIFS_UNIX_MINOR_VERSION);
5303 pSMB->ClientUnixCap = cpu_to_le64(cap);
5304
Steve Frenchbe8e3b02011-04-29 05:40:20 +00005305 inc_rfc1001_len(pSMB, byte_count);
Jeremy Allisonac670552005-06-22 17:26:35 -07005306 pSMB->ByteCount = cpu_to_le16(byte_count);
5307
5308 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
5309 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
5310 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00005311 cERROR(1, "Send error in SETFSUnixInfo = %d", rc);
Jeremy Allisonac670552005-06-22 17:26:35 -07005312 } else { /* decode response */
5313 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
Steve Frenchad7a2922008-02-07 23:25:02 +00005314 if (rc)
Jeremy Allisonac670552005-06-22 17:26:35 -07005315 rc = -EIO; /* bad smb */
Jeremy Allisonac670552005-06-22 17:26:35 -07005316 }
5317 cifs_buf_release(pSMB);
5318
5319 if (rc == -EAGAIN)
5320 goto SETFSUnixRetry;
5321
5322 return rc;
5323}
5324
5325
Linus Torvalds1da177e2005-04-16 15:20:36 -07005326
5327int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04005328CIFSSMBQFSPosixInfo(const unsigned int xid, struct cifs_tcon *tcon,
Steve French737b7582005-04-28 22:41:06 -07005329 struct kstatfs *FSData)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005330{
5331/* level 0x201 SMB_QUERY_CIFS_POSIX_INFO */
5332 TRANSACTION2_QFSI_REQ *pSMB = NULL;
5333 TRANSACTION2_QFSI_RSP *pSMBr = NULL;
5334 FILE_SYSTEM_POSIX_INFO *response_data;
5335 int rc = 0;
5336 int bytes_returned = 0;
5337 __u16 params, byte_count;
5338
Joe Perchesb6b38f72010-04-21 03:50:45 +00005339 cFYI(1, "In QFSPosixInfo");
Linus Torvalds1da177e2005-04-16 15:20:36 -07005340QFSPosixRetry:
5341 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
5342 (void **) &pSMBr);
5343 if (rc)
5344 return rc;
5345
5346 params = 2; /* level */
5347 pSMB->TotalDataCount = 0;
5348 pSMB->DataCount = 0;
5349 pSMB->DataOffset = 0;
5350 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French582d21e2008-05-13 04:54:12 +00005351 /* BB find exact max SMB PDU from sess structure BB */
5352 pSMB->MaxDataCount = cpu_to_le16(100);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005353 pSMB->MaxSetupCount = 0;
5354 pSMB->Reserved = 0;
5355 pSMB->Flags = 0;
5356 pSMB->Timeout = 0;
5357 pSMB->Reserved2 = 0;
5358 byte_count = params + 1 /* pad */ ;
5359 pSMB->ParameterCount = cpu_to_le16(params);
5360 pSMB->TotalParameterCount = pSMB->ParameterCount;
Steve French50c2f752007-07-13 00:33:32 +00005361 pSMB->ParameterOffset = cpu_to_le16(offsetof(struct
5362 smb_com_transaction2_qfsi_req, InformationLevel) - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005363 pSMB->SetupCount = 1;
5364 pSMB->Reserved3 = 0;
5365 pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_FS_INFORMATION);
5366 pSMB->InformationLevel = cpu_to_le16(SMB_QUERY_POSIX_FS_INFO);
Steve Frenchbe8e3b02011-04-29 05:40:20 +00005367 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005368 pSMB->ByteCount = cpu_to_le16(byte_count);
5369
5370 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
5371 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
5372 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00005373 cFYI(1, "Send error in QFSUnixInfo = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005374 } else { /* decode response */
5375 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
5376
Jeff Layton820a8032011-05-04 08:05:26 -04005377 if (rc || get_bcc(&pSMBr->hdr) < 13) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07005378 rc = -EIO; /* bad smb */
5379 } else {
5380 __u16 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
5381 response_data =
5382 (FILE_SYSTEM_POSIX_INFO
5383 *) (((char *) &pSMBr->hdr.Protocol) +
5384 data_offset);
5385 FSData->f_bsize =
5386 le32_to_cpu(response_data->BlockSize);
5387 FSData->f_blocks =
5388 le64_to_cpu(response_data->TotalBlocks);
5389 FSData->f_bfree =
5390 le64_to_cpu(response_data->BlocksAvail);
Steve French790fe572007-07-07 19:25:05 +00005391 if (response_data->UserBlocksAvail == cpu_to_le64(-1)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07005392 FSData->f_bavail = FSData->f_bfree;
5393 } else {
5394 FSData->f_bavail =
Steve French50c2f752007-07-13 00:33:32 +00005395 le64_to_cpu(response_data->UserBlocksAvail);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005396 }
Steve French790fe572007-07-07 19:25:05 +00005397 if (response_data->TotalFileNodes != cpu_to_le64(-1))
Linus Torvalds1da177e2005-04-16 15:20:36 -07005398 FSData->f_files =
Steve French50c2f752007-07-13 00:33:32 +00005399 le64_to_cpu(response_data->TotalFileNodes);
Steve French790fe572007-07-07 19:25:05 +00005400 if (response_data->FreeFileNodes != cpu_to_le64(-1))
Linus Torvalds1da177e2005-04-16 15:20:36 -07005401 FSData->f_ffree =
Steve French50c2f752007-07-13 00:33:32 +00005402 le64_to_cpu(response_data->FreeFileNodes);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005403 }
5404 }
5405 cifs_buf_release(pSMB);
5406
5407 if (rc == -EAGAIN)
5408 goto QFSPosixRetry;
5409
5410 return rc;
5411}
5412
5413
Steve French50c2f752007-07-13 00:33:32 +00005414/* We can not use write of zero bytes trick to
5415 set file size due to need for large file support. Also note that
5416 this SetPathInfo is preferred to SetFileInfo based method in next
Linus Torvalds1da177e2005-04-16 15:20:36 -07005417 routine which is only needed to work around a sharing violation bug
5418 in Samba which this routine can run into */
5419
5420int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04005421CIFSSMBSetEOF(const unsigned int xid, struct cifs_tcon *tcon,
5422 const char *fileName, __u64 size, bool SetAllocation,
Steve French737b7582005-04-28 22:41:06 -07005423 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005424{
5425 struct smb_com_transaction2_spi_req *pSMB = NULL;
5426 struct smb_com_transaction2_spi_rsp *pSMBr = NULL;
5427 struct file_end_of_file_info *parm_data;
5428 int name_len;
5429 int rc = 0;
5430 int bytes_returned = 0;
5431 __u16 params, byte_count, data_count, param_offset, offset;
5432
Joe Perchesb6b38f72010-04-21 03:50:45 +00005433 cFYI(1, "In SetEOF");
Linus Torvalds1da177e2005-04-16 15:20:36 -07005434SetEOFRetry:
5435 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
5436 (void **) &pSMBr);
5437 if (rc)
5438 return rc;
5439
5440 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
5441 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06005442 cifsConvertToUTF16((__le16 *) pSMB->FileName, fileName,
5443 PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005444 name_len++; /* trailing null */
5445 name_len *= 2;
Steve French3e87d802005-09-18 20:49:21 -07005446 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07005447 name_len = strnlen(fileName, PATH_MAX);
5448 name_len++; /* trailing null */
5449 strncpy(pSMB->FileName, fileName, name_len);
5450 }
5451 params = 6 + name_len;
Steve French26f57362007-08-30 22:09:15 +00005452 data_count = sizeof(struct file_end_of_file_info);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005453 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French3e87d802005-09-18 20:49:21 -07005454 pSMB->MaxDataCount = cpu_to_le16(4100);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005455 pSMB->MaxSetupCount = 0;
5456 pSMB->Reserved = 0;
5457 pSMB->Flags = 0;
5458 pSMB->Timeout = 0;
5459 pSMB->Reserved2 = 0;
5460 param_offset = offsetof(struct smb_com_transaction2_spi_req,
Steve French50c2f752007-07-13 00:33:32 +00005461 InformationLevel) - 4;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005462 offset = param_offset + params;
Steve French790fe572007-07-07 19:25:05 +00005463 if (SetAllocation) {
Steve French50c2f752007-07-13 00:33:32 +00005464 if (tcon->ses->capabilities & CAP_INFOLEVEL_PASSTHRU)
5465 pSMB->InformationLevel =
5466 cpu_to_le16(SMB_SET_FILE_ALLOCATION_INFO2);
5467 else
5468 pSMB->InformationLevel =
5469 cpu_to_le16(SMB_SET_FILE_ALLOCATION_INFO);
5470 } else /* Set File Size */ {
Linus Torvalds1da177e2005-04-16 15:20:36 -07005471 if (tcon->ses->capabilities & CAP_INFOLEVEL_PASSTHRU)
5472 pSMB->InformationLevel =
Steve French50c2f752007-07-13 00:33:32 +00005473 cpu_to_le16(SMB_SET_FILE_END_OF_FILE_INFO2);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005474 else
5475 pSMB->InformationLevel =
Steve French50c2f752007-07-13 00:33:32 +00005476 cpu_to_le16(SMB_SET_FILE_END_OF_FILE_INFO);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005477 }
5478
5479 parm_data =
5480 (struct file_end_of_file_info *) (((char *) &pSMB->hdr.Protocol) +
5481 offset);
5482 pSMB->ParameterOffset = cpu_to_le16(param_offset);
5483 pSMB->DataOffset = cpu_to_le16(offset);
5484 pSMB->SetupCount = 1;
5485 pSMB->Reserved3 = 0;
5486 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
5487 byte_count = 3 /* pad */ + params + data_count;
5488 pSMB->DataCount = cpu_to_le16(data_count);
5489 pSMB->TotalDataCount = pSMB->DataCount;
5490 pSMB->ParameterCount = cpu_to_le16(params);
5491 pSMB->TotalParameterCount = pSMB->ParameterCount;
5492 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00005493 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005494 parm_data->FileSize = cpu_to_le64(size);
5495 pSMB->ByteCount = cpu_to_le16(byte_count);
5496 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
5497 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Steve Frenchad7a2922008-02-07 23:25:02 +00005498 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00005499 cFYI(1, "SetPathInfo (file size) returned %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005500
5501 cifs_buf_release(pSMB);
5502
5503 if (rc == -EAGAIN)
5504 goto SetEOFRetry;
5505
5506 return rc;
5507}
5508
5509int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04005510CIFSSMBSetFileSize(const unsigned int xid, struct cifs_tcon *tcon, __u64 size,
Steve French4b18f2a2008-04-29 00:06:05 +00005511 __u16 fid, __u32 pid_of_opener, bool SetAllocation)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005512{
5513 struct smb_com_transaction2_sfi_req *pSMB = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005514 struct file_end_of_file_info *parm_data;
5515 int rc = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005516 __u16 params, param_offset, offset, byte_count, count;
5517
Joe Perchesb6b38f72010-04-21 03:50:45 +00005518 cFYI(1, "SetFileSize (via SetFileInfo) %lld",
5519 (long long)size);
Steve Frenchcd634992005-04-28 22:41:10 -07005520 rc = small_smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB);
5521
Linus Torvalds1da177e2005-04-16 15:20:36 -07005522 if (rc)
5523 return rc;
5524
5525 pSMB->hdr.Pid = cpu_to_le16((__u16)pid_of_opener);
5526 pSMB->hdr.PidHigh = cpu_to_le16((__u16)(pid_of_opener >> 16));
Steve French50c2f752007-07-13 00:33:32 +00005527
Linus Torvalds1da177e2005-04-16 15:20:36 -07005528 params = 6;
5529 pSMB->MaxSetupCount = 0;
5530 pSMB->Reserved = 0;
5531 pSMB->Flags = 0;
5532 pSMB->Timeout = 0;
5533 pSMB->Reserved2 = 0;
5534 param_offset = offsetof(struct smb_com_transaction2_sfi_req, Fid) - 4;
5535 offset = param_offset + params;
5536
Linus Torvalds1da177e2005-04-16 15:20:36 -07005537 count = sizeof(struct file_end_of_file_info);
5538 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French582d21e2008-05-13 04:54:12 +00005539 /* BB find exact max SMB PDU from sess structure BB */
5540 pSMB->MaxDataCount = cpu_to_le16(1000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005541 pSMB->SetupCount = 1;
5542 pSMB->Reserved3 = 0;
5543 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FILE_INFORMATION);
5544 byte_count = 3 /* pad */ + params + count;
5545 pSMB->DataCount = cpu_to_le16(count);
5546 pSMB->ParameterCount = cpu_to_le16(params);
5547 pSMB->TotalDataCount = pSMB->DataCount;
5548 pSMB->TotalParameterCount = pSMB->ParameterCount;
5549 pSMB->ParameterOffset = cpu_to_le16(param_offset);
5550 parm_data =
Steve French50c2f752007-07-13 00:33:32 +00005551 (struct file_end_of_file_info *) (((char *) &pSMB->hdr.Protocol)
5552 + offset);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005553 pSMB->DataOffset = cpu_to_le16(offset);
5554 parm_data->FileSize = cpu_to_le64(size);
5555 pSMB->Fid = fid;
Steve French790fe572007-07-07 19:25:05 +00005556 if (SetAllocation) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07005557 if (tcon->ses->capabilities & CAP_INFOLEVEL_PASSTHRU)
5558 pSMB->InformationLevel =
5559 cpu_to_le16(SMB_SET_FILE_ALLOCATION_INFO2);
5560 else
5561 pSMB->InformationLevel =
5562 cpu_to_le16(SMB_SET_FILE_ALLOCATION_INFO);
Steve French50c2f752007-07-13 00:33:32 +00005563 } else /* Set File Size */ {
Linus Torvalds1da177e2005-04-16 15:20:36 -07005564 if (tcon->ses->capabilities & CAP_INFOLEVEL_PASSTHRU)
5565 pSMB->InformationLevel =
Steve French50c2f752007-07-13 00:33:32 +00005566 cpu_to_le16(SMB_SET_FILE_END_OF_FILE_INFO2);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005567 else
5568 pSMB->InformationLevel =
Steve French50c2f752007-07-13 00:33:32 +00005569 cpu_to_le16(SMB_SET_FILE_END_OF_FILE_INFO);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005570 }
5571 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00005572 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005573 pSMB->ByteCount = cpu_to_le16(byte_count);
Pavel Shilovsky792af7b2012-03-23 14:28:02 -04005574 rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005575 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00005576 cFYI(1, "Send error in SetFileInfo (SetFileSize) = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005577 }
5578
Steve French50c2f752007-07-13 00:33:32 +00005579 /* Note: On -EAGAIN error only caller can retry on handle based calls
Linus Torvalds1da177e2005-04-16 15:20:36 -07005580 since file handle passed in no longer valid */
5581
5582 return rc;
5583}
5584
Steve French50c2f752007-07-13 00:33:32 +00005585/* Some legacy servers such as NT4 require that the file times be set on
Linus Torvalds1da177e2005-04-16 15:20:36 -07005586 an open handle, rather than by pathname - this is awkward due to
5587 potential access conflicts on the open, but it is unavoidable for these
5588 old servers since the only other choice is to go from 100 nanosecond DCE
5589 time and resort to the original setpathinfo level which takes the ancient
5590 DOS time format with 2 second granularity */
5591int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04005592CIFSSMBSetFileInfo(const unsigned int xid, struct cifs_tcon *tcon,
Jeff Layton2dd2dfa2008-08-02 07:26:12 -04005593 const FILE_BASIC_INFO *data, __u16 fid, __u32 pid_of_opener)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005594{
5595 struct smb_com_transaction2_sfi_req *pSMB = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005596 char *data_offset;
5597 int rc = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005598 __u16 params, param_offset, offset, byte_count, count;
5599
Joe Perchesb6b38f72010-04-21 03:50:45 +00005600 cFYI(1, "Set Times (via SetFileInfo)");
Steve Frenchcd634992005-04-28 22:41:10 -07005601 rc = small_smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB);
5602
Linus Torvalds1da177e2005-04-16 15:20:36 -07005603 if (rc)
5604 return rc;
5605
Jeff Layton2dd2dfa2008-08-02 07:26:12 -04005606 pSMB->hdr.Pid = cpu_to_le16((__u16)pid_of_opener);
5607 pSMB->hdr.PidHigh = cpu_to_le16((__u16)(pid_of_opener >> 16));
Steve French50c2f752007-07-13 00:33:32 +00005608
Linus Torvalds1da177e2005-04-16 15:20:36 -07005609 params = 6;
5610 pSMB->MaxSetupCount = 0;
5611 pSMB->Reserved = 0;
5612 pSMB->Flags = 0;
5613 pSMB->Timeout = 0;
5614 pSMB->Reserved2 = 0;
5615 param_offset = offsetof(struct smb_com_transaction2_sfi_req, Fid) - 4;
5616 offset = param_offset + params;
5617
Jeff Laytonb2a3ad92012-03-26 09:55:29 -04005618 data_offset = (char *)pSMB +
5619 offsetof(struct smb_hdr, Protocol) + offset;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005620
Steve French26f57362007-08-30 22:09:15 +00005621 count = sizeof(FILE_BASIC_INFO);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005622 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French582d21e2008-05-13 04:54:12 +00005623 /* BB find max SMB PDU from sess */
5624 pSMB->MaxDataCount = cpu_to_le16(1000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005625 pSMB->SetupCount = 1;
5626 pSMB->Reserved3 = 0;
5627 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FILE_INFORMATION);
5628 byte_count = 3 /* pad */ + params + count;
5629 pSMB->DataCount = cpu_to_le16(count);
5630 pSMB->ParameterCount = cpu_to_le16(params);
5631 pSMB->TotalDataCount = pSMB->DataCount;
5632 pSMB->TotalParameterCount = pSMB->ParameterCount;
5633 pSMB->ParameterOffset = cpu_to_le16(param_offset);
5634 pSMB->DataOffset = cpu_to_le16(offset);
5635 pSMB->Fid = fid;
5636 if (tcon->ses->capabilities & CAP_INFOLEVEL_PASSTHRU)
5637 pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_BASIC_INFO2);
5638 else
5639 pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_BASIC_INFO);
5640 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00005641 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005642 pSMB->ByteCount = cpu_to_le16(byte_count);
Steve French50c2f752007-07-13 00:33:32 +00005643 memcpy(data_offset, data, sizeof(FILE_BASIC_INFO));
Pavel Shilovsky792af7b2012-03-23 14:28:02 -04005644 rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
Steve Frenchad7a2922008-02-07 23:25:02 +00005645 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00005646 cFYI(1, "Send error in Set Time (SetFileInfo) = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005647
Steve French50c2f752007-07-13 00:33:32 +00005648 /* Note: On -EAGAIN error only caller can retry on handle based calls
Linus Torvalds1da177e2005-04-16 15:20:36 -07005649 since file handle passed in no longer valid */
5650
5651 return rc;
5652}
5653
Jeff Layton6d22f092008-09-23 11:48:35 -04005654int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04005655CIFSSMBSetFileDisposition(const unsigned int xid, struct cifs_tcon *tcon,
Jeff Layton6d22f092008-09-23 11:48:35 -04005656 bool delete_file, __u16 fid, __u32 pid_of_opener)
5657{
5658 struct smb_com_transaction2_sfi_req *pSMB = NULL;
5659 char *data_offset;
5660 int rc = 0;
5661 __u16 params, param_offset, offset, byte_count, count;
5662
Joe Perchesb6b38f72010-04-21 03:50:45 +00005663 cFYI(1, "Set File Disposition (via SetFileInfo)");
Jeff Layton6d22f092008-09-23 11:48:35 -04005664 rc = small_smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB);
5665
5666 if (rc)
5667 return rc;
5668
5669 pSMB->hdr.Pid = cpu_to_le16((__u16)pid_of_opener);
5670 pSMB->hdr.PidHigh = cpu_to_le16((__u16)(pid_of_opener >> 16));
5671
5672 params = 6;
5673 pSMB->MaxSetupCount = 0;
5674 pSMB->Reserved = 0;
5675 pSMB->Flags = 0;
5676 pSMB->Timeout = 0;
5677 pSMB->Reserved2 = 0;
5678 param_offset = offsetof(struct smb_com_transaction2_sfi_req, Fid) - 4;
5679 offset = param_offset + params;
5680
5681 data_offset = (char *) (&pSMB->hdr.Protocol) + offset;
5682
5683 count = 1;
5684 pSMB->MaxParameterCount = cpu_to_le16(2);
5685 /* BB find max SMB PDU from sess */
5686 pSMB->MaxDataCount = cpu_to_le16(1000);
5687 pSMB->SetupCount = 1;
5688 pSMB->Reserved3 = 0;
5689 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FILE_INFORMATION);
5690 byte_count = 3 /* pad */ + params + count;
5691 pSMB->DataCount = cpu_to_le16(count);
5692 pSMB->ParameterCount = cpu_to_le16(params);
5693 pSMB->TotalDataCount = pSMB->DataCount;
5694 pSMB->TotalParameterCount = pSMB->ParameterCount;
5695 pSMB->ParameterOffset = cpu_to_le16(param_offset);
5696 pSMB->DataOffset = cpu_to_le16(offset);
5697 pSMB->Fid = fid;
5698 pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_DISPOSITION_INFO);
5699 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00005700 inc_rfc1001_len(pSMB, byte_count);
Jeff Layton6d22f092008-09-23 11:48:35 -04005701 pSMB->ByteCount = cpu_to_le16(byte_count);
5702 *data_offset = delete_file ? 1 : 0;
Pavel Shilovsky792af7b2012-03-23 14:28:02 -04005703 rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
Jeff Layton6d22f092008-09-23 11:48:35 -04005704 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00005705 cFYI(1, "Send error in SetFileDisposition = %d", rc);
Jeff Layton6d22f092008-09-23 11:48:35 -04005706
5707 return rc;
5708}
Linus Torvalds1da177e2005-04-16 15:20:36 -07005709
5710int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04005711CIFSSMBSetPathInfo(const unsigned int xid, struct cifs_tcon *tcon,
Jeff Layton6fc000e2008-08-02 07:26:12 -04005712 const char *fileName, const FILE_BASIC_INFO *data,
5713 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005714{
5715 TRANSACTION2_SPI_REQ *pSMB = NULL;
5716 TRANSACTION2_SPI_RSP *pSMBr = NULL;
5717 int name_len;
5718 int rc = 0;
5719 int bytes_returned = 0;
5720 char *data_offset;
5721 __u16 params, param_offset, offset, byte_count, count;
5722
Joe Perchesb6b38f72010-04-21 03:50:45 +00005723 cFYI(1, "In SetTimes");
Linus Torvalds1da177e2005-04-16 15:20:36 -07005724
5725SetTimesRetry:
5726 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
5727 (void **) &pSMBr);
5728 if (rc)
5729 return rc;
5730
5731 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
5732 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06005733 cifsConvertToUTF16((__le16 *) pSMB->FileName, fileName,
5734 PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005735 name_len++; /* trailing null */
5736 name_len *= 2;
Steve French50c2f752007-07-13 00:33:32 +00005737 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07005738 name_len = strnlen(fileName, PATH_MAX);
5739 name_len++; /* trailing null */
5740 strncpy(pSMB->FileName, fileName, name_len);
5741 }
5742
5743 params = 6 + name_len;
Steve French26f57362007-08-30 22:09:15 +00005744 count = sizeof(FILE_BASIC_INFO);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005745 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French582d21e2008-05-13 04:54:12 +00005746 /* BB find max SMB PDU from sess structure BB */
5747 pSMB->MaxDataCount = cpu_to_le16(1000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005748 pSMB->MaxSetupCount = 0;
5749 pSMB->Reserved = 0;
5750 pSMB->Flags = 0;
5751 pSMB->Timeout = 0;
5752 pSMB->Reserved2 = 0;
5753 param_offset = offsetof(struct smb_com_transaction2_spi_req,
Steve French50c2f752007-07-13 00:33:32 +00005754 InformationLevel) - 4;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005755 offset = param_offset + params;
5756 data_offset = (char *) (&pSMB->hdr.Protocol) + offset;
5757 pSMB->ParameterOffset = cpu_to_le16(param_offset);
5758 pSMB->DataOffset = cpu_to_le16(offset);
5759 pSMB->SetupCount = 1;
5760 pSMB->Reserved3 = 0;
5761 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
5762 byte_count = 3 /* pad */ + params + count;
5763
5764 pSMB->DataCount = cpu_to_le16(count);
5765 pSMB->ParameterCount = cpu_to_le16(params);
5766 pSMB->TotalDataCount = pSMB->DataCount;
5767 pSMB->TotalParameterCount = pSMB->ParameterCount;
5768 if (tcon->ses->capabilities & CAP_INFOLEVEL_PASSTHRU)
5769 pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_BASIC_INFO2);
5770 else
5771 pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_BASIC_INFO);
5772 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00005773 inc_rfc1001_len(pSMB, byte_count);
Steve French26f57362007-08-30 22:09:15 +00005774 memcpy(data_offset, data, sizeof(FILE_BASIC_INFO));
Linus Torvalds1da177e2005-04-16 15:20:36 -07005775 pSMB->ByteCount = cpu_to_le16(byte_count);
5776 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
5777 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Steve Frenchad7a2922008-02-07 23:25:02 +00005778 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00005779 cFYI(1, "SetPathInfo (times) returned %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005780
5781 cifs_buf_release(pSMB);
5782
5783 if (rc == -EAGAIN)
5784 goto SetTimesRetry;
5785
5786 return rc;
5787}
5788
5789/* Can not be used to set time stamps yet (due to old DOS time format) */
5790/* Can be used to set attributes */
5791#if 0 /* Possibly not needed - since it turns out that strangely NT4 has a bug
5792 handling it anyway and NT4 was what we thought it would be needed for
5793 Do not delete it until we prove whether needed for Win9x though */
5794int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04005795CIFSSMBSetAttrLegacy(unsigned int xid, struct cifs_tcon *tcon, char *fileName,
Linus Torvalds1da177e2005-04-16 15:20:36 -07005796 __u16 dos_attrs, const struct nls_table *nls_codepage)
5797{
5798 SETATTR_REQ *pSMB = NULL;
5799 SETATTR_RSP *pSMBr = NULL;
5800 int rc = 0;
5801 int bytes_returned;
5802 int name_len;
5803
Joe Perchesb6b38f72010-04-21 03:50:45 +00005804 cFYI(1, "In SetAttrLegacy");
Linus Torvalds1da177e2005-04-16 15:20:36 -07005805
5806SetAttrLgcyRetry:
5807 rc = smb_init(SMB_COM_SETATTR, 8, tcon, (void **) &pSMB,
5808 (void **) &pSMBr);
5809 if (rc)
5810 return rc;
5811
5812 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
5813 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06005814 ConvertToUTF16((__le16 *) pSMB->fileName, fileName,
5815 PATH_MAX, nls_codepage);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005816 name_len++; /* trailing null */
5817 name_len *= 2;
Steve French50c2f752007-07-13 00:33:32 +00005818 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07005819 name_len = strnlen(fileName, PATH_MAX);
5820 name_len++; /* trailing null */
5821 strncpy(pSMB->fileName, fileName, name_len);
5822 }
5823 pSMB->attr = cpu_to_le16(dos_attrs);
5824 pSMB->BufferFormat = 0x04;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00005825 inc_rfc1001_len(pSMB, name_len + 1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005826 pSMB->ByteCount = cpu_to_le16(name_len + 1);
5827 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
5828 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Steve Frenchad7a2922008-02-07 23:25:02 +00005829 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00005830 cFYI(1, "Error in LegacySetAttr = %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005831
5832 cifs_buf_release(pSMB);
5833
5834 if (rc == -EAGAIN)
5835 goto SetAttrLgcyRetry;
5836
5837 return rc;
5838}
5839#endif /* temporarily unneeded SetAttr legacy function */
5840
Jeff Layton654cf142009-07-09 20:02:49 -04005841static void
5842cifs_fill_unix_set_info(FILE_UNIX_BASIC_INFO *data_offset,
5843 const struct cifs_unix_set_info_args *args)
5844{
5845 u64 mode = args->mode;
5846
5847 /*
5848 * Samba server ignores set of file size to zero due to bugs in some
5849 * older clients, but we should be precise - we use SetFileSize to
5850 * set file size and do not want to truncate file size to zero
Lucas De Marchi25985ed2011-03-30 22:57:33 -03005851 * accidentally as happened on one Samba server beta by putting
Jeff Layton654cf142009-07-09 20:02:49 -04005852 * zero instead of -1 here
5853 */
5854 data_offset->EndOfFile = cpu_to_le64(NO_CHANGE_64);
5855 data_offset->NumOfBytes = cpu_to_le64(NO_CHANGE_64);
5856 data_offset->LastStatusChange = cpu_to_le64(args->ctime);
5857 data_offset->LastAccessTime = cpu_to_le64(args->atime);
5858 data_offset->LastModificationTime = cpu_to_le64(args->mtime);
5859 data_offset->Uid = cpu_to_le64(args->uid);
5860 data_offset->Gid = cpu_to_le64(args->gid);
5861 /* better to leave device as zero when it is */
5862 data_offset->DevMajor = cpu_to_le64(MAJOR(args->device));
5863 data_offset->DevMinor = cpu_to_le64(MINOR(args->device));
5864 data_offset->Permissions = cpu_to_le64(mode);
5865
5866 if (S_ISREG(mode))
5867 data_offset->Type = cpu_to_le32(UNIX_FILE);
5868 else if (S_ISDIR(mode))
5869 data_offset->Type = cpu_to_le32(UNIX_DIR);
5870 else if (S_ISLNK(mode))
5871 data_offset->Type = cpu_to_le32(UNIX_SYMLINK);
5872 else if (S_ISCHR(mode))
5873 data_offset->Type = cpu_to_le32(UNIX_CHARDEV);
5874 else if (S_ISBLK(mode))
5875 data_offset->Type = cpu_to_le32(UNIX_BLOCKDEV);
5876 else if (S_ISFIFO(mode))
5877 data_offset->Type = cpu_to_le32(UNIX_FIFO);
5878 else if (S_ISSOCK(mode))
5879 data_offset->Type = cpu_to_le32(UNIX_SOCKET);
5880}
5881
Linus Torvalds1da177e2005-04-16 15:20:36 -07005882int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04005883CIFSSMBUnixSetFileInfo(const unsigned int xid, struct cifs_tcon *tcon,
Jeff Layton3bbeeb32009-07-09 20:02:50 -04005884 const struct cifs_unix_set_info_args *args,
5885 u16 fid, u32 pid_of_opener)
5886{
5887 struct smb_com_transaction2_sfi_req *pSMB = NULL;
Jeff Laytonb2a3ad92012-03-26 09:55:29 -04005888 char *data_offset;
Jeff Layton3bbeeb32009-07-09 20:02:50 -04005889 int rc = 0;
5890 u16 params, param_offset, offset, byte_count, count;
5891
Joe Perchesb6b38f72010-04-21 03:50:45 +00005892 cFYI(1, "Set Unix Info (via SetFileInfo)");
Jeff Layton3bbeeb32009-07-09 20:02:50 -04005893 rc = small_smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB);
5894
5895 if (rc)
5896 return rc;
5897
5898 pSMB->hdr.Pid = cpu_to_le16((__u16)pid_of_opener);
5899 pSMB->hdr.PidHigh = cpu_to_le16((__u16)(pid_of_opener >> 16));
5900
5901 params = 6;
5902 pSMB->MaxSetupCount = 0;
5903 pSMB->Reserved = 0;
5904 pSMB->Flags = 0;
5905 pSMB->Timeout = 0;
5906 pSMB->Reserved2 = 0;
5907 param_offset = offsetof(struct smb_com_transaction2_sfi_req, Fid) - 4;
5908 offset = param_offset + params;
5909
Jeff Laytonb2a3ad92012-03-26 09:55:29 -04005910 data_offset = (char *)pSMB +
5911 offsetof(struct smb_hdr, Protocol) + offset;
5912
Jeff Layton3bbeeb32009-07-09 20:02:50 -04005913 count = sizeof(FILE_UNIX_BASIC_INFO);
5914
5915 pSMB->MaxParameterCount = cpu_to_le16(2);
5916 /* BB find max SMB PDU from sess */
5917 pSMB->MaxDataCount = cpu_to_le16(1000);
5918 pSMB->SetupCount = 1;
5919 pSMB->Reserved3 = 0;
5920 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_FILE_INFORMATION);
5921 byte_count = 3 /* pad */ + params + count;
5922 pSMB->DataCount = cpu_to_le16(count);
5923 pSMB->ParameterCount = cpu_to_le16(params);
5924 pSMB->TotalDataCount = pSMB->DataCount;
5925 pSMB->TotalParameterCount = pSMB->ParameterCount;
5926 pSMB->ParameterOffset = cpu_to_le16(param_offset);
5927 pSMB->DataOffset = cpu_to_le16(offset);
5928 pSMB->Fid = fid;
5929 pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_UNIX_BASIC);
5930 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00005931 inc_rfc1001_len(pSMB, byte_count);
Jeff Layton3bbeeb32009-07-09 20:02:50 -04005932 pSMB->ByteCount = cpu_to_le16(byte_count);
5933
Jeff Laytonb2a3ad92012-03-26 09:55:29 -04005934 cifs_fill_unix_set_info((FILE_UNIX_BASIC_INFO *)data_offset, args);
Jeff Layton3bbeeb32009-07-09 20:02:50 -04005935
Pavel Shilovsky792af7b2012-03-23 14:28:02 -04005936 rc = SendReceiveNoRsp(xid, tcon->ses, (char *) pSMB, 0);
Jeff Layton3bbeeb32009-07-09 20:02:50 -04005937 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00005938 cFYI(1, "Send error in Set Time (SetFileInfo) = %d", rc);
Jeff Layton3bbeeb32009-07-09 20:02:50 -04005939
5940 /* Note: On -EAGAIN error only caller can retry on handle based calls
5941 since file handle passed in no longer valid */
5942
5943 return rc;
5944}
5945
5946int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04005947CIFSSMBUnixSetPathInfo(const unsigned int xid, struct cifs_tcon *tcon,
Pavel Shilovskyff691e92012-07-13 14:04:46 +04005948 const char *file_name,
Jeff Layton01ea95e2009-07-09 20:02:49 -04005949 const struct cifs_unix_set_info_args *args,
5950 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005951{
5952 TRANSACTION2_SPI_REQ *pSMB = NULL;
5953 TRANSACTION2_SPI_RSP *pSMBr = NULL;
5954 int name_len;
5955 int rc = 0;
5956 int bytes_returned = 0;
5957 FILE_UNIX_BASIC_INFO *data_offset;
5958 __u16 params, param_offset, offset, count, byte_count;
5959
Joe Perchesb6b38f72010-04-21 03:50:45 +00005960 cFYI(1, "In SetUID/GID/Mode");
Linus Torvalds1da177e2005-04-16 15:20:36 -07005961setPermsRetry:
5962 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
5963 (void **) &pSMBr);
5964 if (rc)
5965 return rc;
5966
5967 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
5968 name_len =
Pavel Shilovskyff691e92012-07-13 14:04:46 +04005969 cifsConvertToUTF16((__le16 *) pSMB->FileName, file_name,
Steve Frenchacbbb762012-01-18 22:32:33 -06005970 PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005971 name_len++; /* trailing null */
5972 name_len *= 2;
Steve French3e87d802005-09-18 20:49:21 -07005973 } else { /* BB improve the check for buffer overruns BB */
Pavel Shilovskyff691e92012-07-13 14:04:46 +04005974 name_len = strnlen(file_name, PATH_MAX);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005975 name_len++; /* trailing null */
Pavel Shilovskyff691e92012-07-13 14:04:46 +04005976 strncpy(pSMB->FileName, file_name, name_len);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005977 }
5978
5979 params = 6 + name_len;
Steve French26f57362007-08-30 22:09:15 +00005980 count = sizeof(FILE_UNIX_BASIC_INFO);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005981 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French582d21e2008-05-13 04:54:12 +00005982 /* BB find max SMB PDU from sess structure BB */
5983 pSMB->MaxDataCount = cpu_to_le16(1000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005984 pSMB->MaxSetupCount = 0;
5985 pSMB->Reserved = 0;
5986 pSMB->Flags = 0;
5987 pSMB->Timeout = 0;
5988 pSMB->Reserved2 = 0;
5989 param_offset = offsetof(struct smb_com_transaction2_spi_req,
Steve French50c2f752007-07-13 00:33:32 +00005990 InformationLevel) - 4;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005991 offset = param_offset + params;
5992 data_offset =
5993 (FILE_UNIX_BASIC_INFO *) ((char *) &pSMB->hdr.Protocol +
5994 offset);
5995 memset(data_offset, 0, count);
5996 pSMB->DataOffset = cpu_to_le16(offset);
5997 pSMB->ParameterOffset = cpu_to_le16(param_offset);
5998 pSMB->SetupCount = 1;
5999 pSMB->Reserved3 = 0;
6000 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
6001 byte_count = 3 /* pad */ + params + count;
6002 pSMB->ParameterCount = cpu_to_le16(params);
6003 pSMB->DataCount = cpu_to_le16(count);
6004 pSMB->TotalParameterCount = pSMB->ParameterCount;
6005 pSMB->TotalDataCount = pSMB->DataCount;
6006 pSMB->InformationLevel = cpu_to_le16(SMB_SET_FILE_UNIX_BASIC);
6007 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00006008 inc_rfc1001_len(pSMB, byte_count);
Steve French50c2f752007-07-13 00:33:32 +00006009
Jeff Layton654cf142009-07-09 20:02:49 -04006010 cifs_fill_unix_set_info(data_offset, args);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006011
6012 pSMB->ByteCount = cpu_to_le16(byte_count);
6013 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
6014 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Steve Frenchad7a2922008-02-07 23:25:02 +00006015 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00006016 cFYI(1, "SetPathInfo (perms) returned %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006017
Steve French0d817bc2008-05-22 02:02:03 +00006018 cifs_buf_release(pSMB);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006019 if (rc == -EAGAIN)
6020 goto setPermsRetry;
6021 return rc;
6022}
6023
Linus Torvalds1da177e2005-04-16 15:20:36 -07006024#ifdef CONFIG_CIFS_XATTR
Jeff Layton31c05192010-02-10 16:18:26 -05006025/*
6026 * Do a path-based QUERY_ALL_EAS call and parse the result. This is a common
6027 * function used by listxattr and getxattr type calls. When ea_name is set,
6028 * it looks for that attribute name and stuffs that value into the EAData
6029 * buffer. When ea_name is NULL, it stuffs a list of attribute names into the
6030 * buffer. In both cases, the return value is either the length of the
6031 * resulting data or a negative error code. If EAData is a NULL pointer then
6032 * the data isn't copied to it, but the length is returned.
6033 */
Linus Torvalds1da177e2005-04-16 15:20:36 -07006034ssize_t
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04006035CIFSSMBQAllEAs(const unsigned int xid, struct cifs_tcon *tcon,
Jeff Layton31c05192010-02-10 16:18:26 -05006036 const unsigned char *searchName, const unsigned char *ea_name,
6037 char *EAData, size_t buf_size,
6038 const struct nls_table *nls_codepage, int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07006039{
6040 /* BB assumes one setup word */
6041 TRANSACTION2_QPI_REQ *pSMB = NULL;
6042 TRANSACTION2_QPI_RSP *pSMBr = NULL;
6043 int rc = 0;
6044 int bytes_returned;
Jeff Layton6e462b92010-02-10 16:18:26 -05006045 int list_len;
Jeff Laytonf0d38682010-02-10 16:18:26 -05006046 struct fealist *ea_response_data;
Steve French50c2f752007-07-13 00:33:32 +00006047 struct fea *temp_fea;
6048 char *temp_ptr;
Jeff Layton0cd126b2010-02-10 16:18:26 -05006049 char *end_of_smb;
Jeff Laytonf0d38682010-02-10 16:18:26 -05006050 __u16 params, byte_count, data_offset;
Jeff Layton5980fc92011-07-28 12:48:26 -04006051 unsigned int ea_name_len = ea_name ? strlen(ea_name) : 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006052
Joe Perchesb6b38f72010-04-21 03:50:45 +00006053 cFYI(1, "In Query All EAs path %s", searchName);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006054QAllEAsRetry:
6055 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
6056 (void **) &pSMBr);
6057 if (rc)
6058 return rc;
6059
6060 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
Jeff Layton6e462b92010-02-10 16:18:26 -05006061 list_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06006062 cifsConvertToUTF16((__le16 *) pSMB->FileName, searchName,
6063 PATH_MAX, nls_codepage, remap);
Jeff Layton6e462b92010-02-10 16:18:26 -05006064 list_len++; /* trailing null */
6065 list_len *= 2;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006066 } else { /* BB improve the check for buffer overruns BB */
Jeff Layton6e462b92010-02-10 16:18:26 -05006067 list_len = strnlen(searchName, PATH_MAX);
6068 list_len++; /* trailing null */
6069 strncpy(pSMB->FileName, searchName, list_len);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006070 }
6071
Jeff Layton6e462b92010-02-10 16:18:26 -05006072 params = 2 /* level */ + 4 /* reserved */ + list_len /* includes NUL */;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006073 pSMB->TotalDataCount = 0;
6074 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French582d21e2008-05-13 04:54:12 +00006075 /* BB find exact max SMB PDU from sess structure BB */
Jeff Laytone5296142010-02-10 16:18:26 -05006076 pSMB->MaxDataCount = cpu_to_le16(CIFSMaxBufSize);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006077 pSMB->MaxSetupCount = 0;
6078 pSMB->Reserved = 0;
6079 pSMB->Flags = 0;
6080 pSMB->Timeout = 0;
6081 pSMB->Reserved2 = 0;
6082 pSMB->ParameterOffset = cpu_to_le16(offsetof(
Steve French50c2f752007-07-13 00:33:32 +00006083 struct smb_com_transaction2_qpi_req, InformationLevel) - 4);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006084 pSMB->DataCount = 0;
6085 pSMB->DataOffset = 0;
6086 pSMB->SetupCount = 1;
6087 pSMB->Reserved3 = 0;
6088 pSMB->SubCommand = cpu_to_le16(TRANS2_QUERY_PATH_INFORMATION);
6089 byte_count = params + 1 /* pad */ ;
6090 pSMB->TotalParameterCount = cpu_to_le16(params);
6091 pSMB->ParameterCount = pSMB->TotalParameterCount;
6092 pSMB->InformationLevel = cpu_to_le16(SMB_INFO_QUERY_ALL_EAS);
6093 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00006094 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006095 pSMB->ByteCount = cpu_to_le16(byte_count);
6096
6097 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
6098 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
6099 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00006100 cFYI(1, "Send error in QueryAllEAs = %d", rc);
Jeff Laytonf0d38682010-02-10 16:18:26 -05006101 goto QAllEAsOut;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006102 }
Jeff Laytonf0d38682010-02-10 16:18:26 -05006103
6104
6105 /* BB also check enough total bytes returned */
6106 /* BB we need to improve the validity checking
6107 of these trans2 responses */
6108
6109 rc = validate_t2((struct smb_t2_rsp *)pSMBr);
Jeff Layton820a8032011-05-04 08:05:26 -04006110 if (rc || get_bcc(&pSMBr->hdr) < 4) {
Jeff Laytonf0d38682010-02-10 16:18:26 -05006111 rc = -EIO; /* bad smb */
6112 goto QAllEAsOut;
6113 }
6114
6115 /* check that length of list is not more than bcc */
6116 /* check that each entry does not go beyond length
6117 of list */
6118 /* check that each element of each entry does not
6119 go beyond end of list */
6120 /* validate_trans2_offsets() */
6121 /* BB check if start of smb + data_offset > &bcc+ bcc */
6122
6123 data_offset = le16_to_cpu(pSMBr->t2.DataOffset);
6124 ea_response_data = (struct fealist *)
6125 (((char *) &pSMBr->hdr.Protocol) + data_offset);
6126
Jeff Layton6e462b92010-02-10 16:18:26 -05006127 list_len = le32_to_cpu(ea_response_data->list_len);
Joe Perchesb6b38f72010-04-21 03:50:45 +00006128 cFYI(1, "ea length %d", list_len);
Jeff Layton6e462b92010-02-10 16:18:26 -05006129 if (list_len <= 8) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00006130 cFYI(1, "empty EA list returned from server");
Jeff Laytonf0d38682010-02-10 16:18:26 -05006131 goto QAllEAsOut;
6132 }
6133
Jeff Layton0cd126b2010-02-10 16:18:26 -05006134 /* make sure list_len doesn't go past end of SMB */
Jeff Layton690c5222011-01-20 13:36:51 -05006135 end_of_smb = (char *)pByteArea(&pSMBr->hdr) + get_bcc(&pSMBr->hdr);
Jeff Layton0cd126b2010-02-10 16:18:26 -05006136 if ((char *)ea_response_data + list_len > end_of_smb) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00006137 cFYI(1, "EA list appears to go beyond SMB");
Jeff Layton0cd126b2010-02-10 16:18:26 -05006138 rc = -EIO;
6139 goto QAllEAsOut;
6140 }
6141
Jeff Laytonf0d38682010-02-10 16:18:26 -05006142 /* account for ea list len */
Jeff Layton6e462b92010-02-10 16:18:26 -05006143 list_len -= 4;
Jeff Laytonf0d38682010-02-10 16:18:26 -05006144 temp_fea = ea_response_data->list;
6145 temp_ptr = (char *)temp_fea;
Jeff Layton6e462b92010-02-10 16:18:26 -05006146 while (list_len > 0) {
Steve French122ca002010-02-24 21:56:48 +00006147 unsigned int name_len;
Jeff Laytonf0d38682010-02-10 16:18:26 -05006148 __u16 value_len;
Jeff Layton0cd126b2010-02-10 16:18:26 -05006149
Jeff Layton6e462b92010-02-10 16:18:26 -05006150 list_len -= 4;
Jeff Laytonf0d38682010-02-10 16:18:26 -05006151 temp_ptr += 4;
Jeff Layton0cd126b2010-02-10 16:18:26 -05006152 /* make sure we can read name_len and value_len */
6153 if (list_len < 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00006154 cFYI(1, "EA entry goes beyond length of list");
Jeff Layton0cd126b2010-02-10 16:18:26 -05006155 rc = -EIO;
6156 goto QAllEAsOut;
6157 }
6158
6159 name_len = temp_fea->name_len;
6160 value_len = le16_to_cpu(temp_fea->value_len);
6161 list_len -= name_len + 1 + value_len;
6162 if (list_len < 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00006163 cFYI(1, "EA entry goes beyond length of list");
Jeff Layton0cd126b2010-02-10 16:18:26 -05006164 rc = -EIO;
6165 goto QAllEAsOut;
6166 }
6167
Jeff Layton31c05192010-02-10 16:18:26 -05006168 if (ea_name) {
Jeff Layton91d065c2011-07-26 18:23:47 -04006169 if (ea_name_len == name_len &&
Jeff Laytonac423442011-10-11 06:41:32 -04006170 memcmp(ea_name, temp_ptr, name_len) == 0) {
Jeff Layton31c05192010-02-10 16:18:26 -05006171 temp_ptr += name_len + 1;
6172 rc = value_len;
6173 if (buf_size == 0)
6174 goto QAllEAsOut;
6175 if ((size_t)value_len > buf_size) {
6176 rc = -ERANGE;
6177 goto QAllEAsOut;
6178 }
6179 memcpy(EAData, temp_ptr, value_len);
6180 goto QAllEAsOut;
6181 }
Jeff Laytonf0d38682010-02-10 16:18:26 -05006182 } else {
Jeff Layton31c05192010-02-10 16:18:26 -05006183 /* account for prefix user. and trailing null */
6184 rc += (5 + 1 + name_len);
6185 if (rc < (int) buf_size) {
6186 memcpy(EAData, "user.", 5);
6187 EAData += 5;
6188 memcpy(EAData, temp_ptr, name_len);
6189 EAData += name_len;
6190 /* null terminate name */
6191 *EAData = 0;
6192 ++EAData;
6193 } else if (buf_size == 0) {
6194 /* skip copy - calc size only */
6195 } else {
6196 /* stop before overrun buffer */
6197 rc = -ERANGE;
6198 break;
6199 }
Jeff Laytonf0d38682010-02-10 16:18:26 -05006200 }
Jeff Layton0cd126b2010-02-10 16:18:26 -05006201 temp_ptr += name_len + 1 + value_len;
Jeff Laytonf0d38682010-02-10 16:18:26 -05006202 temp_fea = (struct fea *)temp_ptr;
6203 }
6204
Jeff Layton31c05192010-02-10 16:18:26 -05006205 /* didn't find the named attribute */
6206 if (ea_name)
6207 rc = -ENODATA;
6208
Jeff Laytonf0d38682010-02-10 16:18:26 -05006209QAllEAsOut:
Steve French0d817bc2008-05-22 02:02:03 +00006210 cifs_buf_release(pSMB);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006211 if (rc == -EAGAIN)
6212 goto QAllEAsRetry;
6213
6214 return (ssize_t)rc;
6215}
6216
Linus Torvalds1da177e2005-04-16 15:20:36 -07006217int
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04006218CIFSSMBSetEA(const unsigned int xid, struct cifs_tcon *tcon,
6219 const char *fileName, const char *ea_name, const void *ea_value,
Steve French50c2f752007-07-13 00:33:32 +00006220 const __u16 ea_value_len, const struct nls_table *nls_codepage,
6221 int remap)
Linus Torvalds1da177e2005-04-16 15:20:36 -07006222{
6223 struct smb_com_transaction2_spi_req *pSMB = NULL;
6224 struct smb_com_transaction2_spi_rsp *pSMBr = NULL;
6225 struct fealist *parm_data;
6226 int name_len;
6227 int rc = 0;
6228 int bytes_returned = 0;
6229 __u16 params, param_offset, byte_count, offset, count;
6230
Joe Perchesb6b38f72010-04-21 03:50:45 +00006231 cFYI(1, "In SetEA");
Linus Torvalds1da177e2005-04-16 15:20:36 -07006232SetEARetry:
6233 rc = smb_init(SMB_COM_TRANSACTION2, 15, tcon, (void **) &pSMB,
6234 (void **) &pSMBr);
6235 if (rc)
6236 return rc;
6237
6238 if (pSMB->hdr.Flags2 & SMBFLG2_UNICODE) {
6239 name_len =
Steve Frenchacbbb762012-01-18 22:32:33 -06006240 cifsConvertToUTF16((__le16 *) pSMB->FileName, fileName,
6241 PATH_MAX, nls_codepage, remap);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006242 name_len++; /* trailing null */
6243 name_len *= 2;
Steve French50c2f752007-07-13 00:33:32 +00006244 } else { /* BB improve the check for buffer overruns BB */
Linus Torvalds1da177e2005-04-16 15:20:36 -07006245 name_len = strnlen(fileName, PATH_MAX);
6246 name_len++; /* trailing null */
6247 strncpy(pSMB->FileName, fileName, name_len);
6248 }
6249
6250 params = 6 + name_len;
6251
6252 /* done calculating parms using name_len of file name,
6253 now use name_len to calculate length of ea name
6254 we are going to create in the inode xattrs */
Steve French790fe572007-07-07 19:25:05 +00006255 if (ea_name == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -07006256 name_len = 0;
6257 else
Steve French50c2f752007-07-13 00:33:32 +00006258 name_len = strnlen(ea_name, 255);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006259
Steve Frenchdae5dbd2007-12-30 23:49:57 +00006260 count = sizeof(*parm_data) + ea_value_len + name_len;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006261 pSMB->MaxParameterCount = cpu_to_le16(2);
Steve French582d21e2008-05-13 04:54:12 +00006262 /* BB find max SMB PDU from sess */
6263 pSMB->MaxDataCount = cpu_to_le16(1000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006264 pSMB->MaxSetupCount = 0;
6265 pSMB->Reserved = 0;
6266 pSMB->Flags = 0;
6267 pSMB->Timeout = 0;
6268 pSMB->Reserved2 = 0;
6269 param_offset = offsetof(struct smb_com_transaction2_spi_req,
Steve French50c2f752007-07-13 00:33:32 +00006270 InformationLevel) - 4;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006271 offset = param_offset + params;
6272 pSMB->InformationLevel =
6273 cpu_to_le16(SMB_SET_FILE_EA);
6274
6275 parm_data =
6276 (struct fealist *) (((char *) &pSMB->hdr.Protocol) +
6277 offset);
6278 pSMB->ParameterOffset = cpu_to_le16(param_offset);
6279 pSMB->DataOffset = cpu_to_le16(offset);
6280 pSMB->SetupCount = 1;
6281 pSMB->Reserved3 = 0;
6282 pSMB->SubCommand = cpu_to_le16(TRANS2_SET_PATH_INFORMATION);
6283 byte_count = 3 /* pad */ + params + count;
6284 pSMB->DataCount = cpu_to_le16(count);
6285 parm_data->list_len = cpu_to_le32(count);
6286 parm_data->list[0].EA_flags = 0;
6287 /* we checked above that name len is less than 255 */
Alexey Dobriyan53b35312006-03-24 03:16:13 -08006288 parm_data->list[0].name_len = (__u8)name_len;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006289 /* EA names are always ASCII */
Steve French790fe572007-07-07 19:25:05 +00006290 if (ea_name)
Steve French50c2f752007-07-13 00:33:32 +00006291 strncpy(parm_data->list[0].name, ea_name, name_len);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006292 parm_data->list[0].name[name_len] = 0;
6293 parm_data->list[0].value_len = cpu_to_le16(ea_value_len);
6294 /* caller ensures that ea_value_len is less than 64K but
6295 we need to ensure that it fits within the smb */
6296
Steve French50c2f752007-07-13 00:33:32 +00006297 /*BB add length check to see if it would fit in
6298 negotiated SMB buffer size BB */
Steve French790fe572007-07-07 19:25:05 +00006299 /* if (ea_value_len > buffer_size - 512 (enough for header)) */
6300 if (ea_value_len)
Steve French50c2f752007-07-13 00:33:32 +00006301 memcpy(parm_data->list[0].name+name_len+1,
6302 ea_value, ea_value_len);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006303
6304 pSMB->TotalDataCount = pSMB->DataCount;
6305 pSMB->ParameterCount = cpu_to_le16(params);
6306 pSMB->TotalParameterCount = pSMB->ParameterCount;
6307 pSMB->Reserved4 = 0;
Steve Frenchbe8e3b02011-04-29 05:40:20 +00006308 inc_rfc1001_len(pSMB, byte_count);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006309 pSMB->ByteCount = cpu_to_le16(byte_count);
6310 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
6311 (struct smb_hdr *) pSMBr, &bytes_returned, 0);
Steve Frenchad7a2922008-02-07 23:25:02 +00006312 if (rc)
Joe Perchesb6b38f72010-04-21 03:50:45 +00006313 cFYI(1, "SetPathInfo (EA) returned %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006314
6315 cifs_buf_release(pSMB);
6316
6317 if (rc == -EAGAIN)
6318 goto SetEARetry;
6319
6320 return rc;
6321}
Linus Torvalds1da177e2005-04-16 15:20:36 -07006322#endif
Steve French0eff0e22011-02-24 05:39:23 +00006323
6324#ifdef CONFIG_CIFS_DNOTIFY_EXPERIMENTAL /* BB unused temporarily */
6325/*
6326 * Years ago the kernel added a "dnotify" function for Samba server,
6327 * to allow network clients (such as Windows) to display updated
6328 * lists of files in directory listings automatically when
6329 * files are added by one user when another user has the
6330 * same directory open on their desktop. The Linux cifs kernel
6331 * client hooked into the kernel side of this interface for
6332 * the same reason, but ironically when the VFS moved from
6333 * "dnotify" to "inotify" it became harder to plug in Linux
6334 * network file system clients (the most obvious use case
6335 * for notify interfaces is when multiple users can update
6336 * the contents of the same directory - exactly what network
6337 * file systems can do) although the server (Samba) could
6338 * still use it. For the short term we leave the worker
6339 * function ifdeffed out (below) until inotify is fixed
6340 * in the VFS to make it easier to plug in network file
6341 * system clients. If inotify turns out to be permanently
6342 * incompatible for network fs clients, we could instead simply
6343 * expose this config flag by adding a future cifs (and smb2) notify ioctl.
6344 */
Pavel Shilovsky6d5786a2012-06-20 11:21:16 +04006345int CIFSSMBNotify(const unsigned int xid, struct cifs_tcon *tcon,
Steve French0eff0e22011-02-24 05:39:23 +00006346 const int notify_subdirs, const __u16 netfid,
6347 __u32 filter, struct file *pfile, int multishot,
6348 const struct nls_table *nls_codepage)
6349{
6350 int rc = 0;
6351 struct smb_com_transaction_change_notify_req *pSMB = NULL;
6352 struct smb_com_ntransaction_change_notify_rsp *pSMBr = NULL;
6353 struct dir_notify_req *dnotify_req;
6354 int bytes_returned;
6355
6356 cFYI(1, "In CIFSSMBNotify for file handle %d", (int)netfid);
6357 rc = smb_init(SMB_COM_NT_TRANSACT, 23, tcon, (void **) &pSMB,
6358 (void **) &pSMBr);
6359 if (rc)
6360 return rc;
6361
6362 pSMB->TotalParameterCount = 0 ;
6363 pSMB->TotalDataCount = 0;
6364 pSMB->MaxParameterCount = cpu_to_le32(2);
Jeff Laytonc974bef2011-10-11 06:41:32 -04006365 pSMB->MaxDataCount = cpu_to_le32(CIFSMaxBufSize & 0xFFFFFF00);
Steve French0eff0e22011-02-24 05:39:23 +00006366 pSMB->MaxSetupCount = 4;
6367 pSMB->Reserved = 0;
6368 pSMB->ParameterOffset = 0;
6369 pSMB->DataCount = 0;
6370 pSMB->DataOffset = 0;
6371 pSMB->SetupCount = 4; /* single byte does not need le conversion */
6372 pSMB->SubCommand = cpu_to_le16(NT_TRANSACT_NOTIFY_CHANGE);
6373 pSMB->ParameterCount = pSMB->TotalParameterCount;
6374 if (notify_subdirs)
6375 pSMB->WatchTree = 1; /* one byte - no le conversion needed */
6376 pSMB->Reserved2 = 0;
6377 pSMB->CompletionFilter = cpu_to_le32(filter);
6378 pSMB->Fid = netfid; /* file handle always le */
6379 pSMB->ByteCount = 0;
6380
6381 rc = SendReceive(xid, tcon->ses, (struct smb_hdr *) pSMB,
6382 (struct smb_hdr *)pSMBr, &bytes_returned,
6383 CIFS_ASYNC_OP);
6384 if (rc) {
6385 cFYI(1, "Error in Notify = %d", rc);
6386 } else {
6387 /* Add file to outstanding requests */
6388 /* BB change to kmem cache alloc */
6389 dnotify_req = kmalloc(
6390 sizeof(struct dir_notify_req),
6391 GFP_KERNEL);
6392 if (dnotify_req) {
6393 dnotify_req->Pid = pSMB->hdr.Pid;
6394 dnotify_req->PidHigh = pSMB->hdr.PidHigh;
6395 dnotify_req->Mid = pSMB->hdr.Mid;
6396 dnotify_req->Tid = pSMB->hdr.Tid;
6397 dnotify_req->Uid = pSMB->hdr.Uid;
6398 dnotify_req->netfid = netfid;
6399 dnotify_req->pfile = pfile;
6400 dnotify_req->filter = filter;
6401 dnotify_req->multishot = multishot;
6402 spin_lock(&GlobalMid_Lock);
6403 list_add_tail(&dnotify_req->lhead,
6404 &GlobalDnotifyReqList);
6405 spin_unlock(&GlobalMid_Lock);
6406 } else
6407 rc = -ENOMEM;
6408 }
6409 cifs_buf_release(pSMB);
6410 return rc;
6411}
6412#endif /* was needed for dnotify, and will be needed for inotify when VFS fix */