blob: 53cce8cc2224f4abe4754d2cc320a05a6f36d215 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * fs/cifs/inode.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 * This library is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU Lesser General Public License as published
9 * by the Free Software Foundation; either version 2.1 of the License, or
10 * (at your option) any later version.
11 *
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
15 * the GNU Lesser General Public License for more details.
16 *
17 * You should have received a copy of the GNU Lesser General Public License
18 * along with this library; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20 */
21#include <linux/fs.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070022#include <linux/stat.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090023#include <linux/slab.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070024#include <linux/pagemap.h>
25#include <asm/div64.h>
26#include "cifsfs.h"
27#include "cifspdu.h"
28#include "cifsglob.h"
29#include "cifsproto.h"
30#include "cifs_debug.h"
31#include "cifs_fs_sb.h"
Suresh Jayaraman9451a9a2010-07-05 18:12:45 +053032#include "fscache.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070033
Christoph Hellwig70eff552008-02-15 20:55:05 +000034
Igor Mammedov79626702008-03-09 03:44:18 +000035static void cifs_set_ops(struct inode *inode, const bool is_dfs_referral)
Christoph Hellwig70eff552008-02-15 20:55:05 +000036{
37 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
38
39 switch (inode->i_mode & S_IFMT) {
40 case S_IFREG:
41 inode->i_op = &cifs_file_inode_ops;
42 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DIRECT_IO) {
43 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_BRL)
44 inode->i_fop = &cifs_file_direct_nobrl_ops;
45 else
46 inode->i_fop = &cifs_file_direct_ops;
47 } else if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_BRL)
48 inode->i_fop = &cifs_file_nobrl_ops;
49 else { /* not direct, send byte range locks */
50 inode->i_fop = &cifs_file_ops;
51 }
52
53
54 /* check if server can support readpages */
55 if (cifs_sb->tcon->ses->server->maxBuf <
56 PAGE_CACHE_SIZE + MAX_CIFS_HDR_SIZE)
57 inode->i_data.a_ops = &cifs_addr_ops_smallbuf;
58 else
59 inode->i_data.a_ops = &cifs_addr_ops;
60 break;
61 case S_IFDIR:
Steve Frenchbc5b6e22008-03-11 21:07:48 +000062#ifdef CONFIG_CIFS_DFS_UPCALL
Igor Mammedov79626702008-03-09 03:44:18 +000063 if (is_dfs_referral) {
64 inode->i_op = &cifs_dfs_referral_inode_operations;
65 } else {
Steve Frenchbc5b6e22008-03-11 21:07:48 +000066#else /* NO DFS support, treat as a directory */
67 {
68#endif
Igor Mammedov79626702008-03-09 03:44:18 +000069 inode->i_op = &cifs_dir_inode_ops;
70 inode->i_fop = &cifs_dir_ops;
71 }
Christoph Hellwig70eff552008-02-15 20:55:05 +000072 break;
73 case S_IFLNK:
74 inode->i_op = &cifs_symlink_inode_ops;
75 break;
76 default:
77 init_special_inode(inode, inode->i_mode, inode->i_rdev);
78 break;
79 }
80}
81
Jeff Laytondf2cf172010-02-12 07:44:16 -050082/* check inode attributes against fattr. If they don't match, tag the
83 * inode for cache invalidation
84 */
85static void
86cifs_revalidate_cache(struct inode *inode, struct cifs_fattr *fattr)
87{
88 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
89
Steve Frenchf19159d2010-04-21 04:12:10 +000090 cFYI(1, "%s: revalidating inode %llu", __func__, cifs_i->uniqueid);
Jeff Laytondf2cf172010-02-12 07:44:16 -050091
92 if (inode->i_state & I_NEW) {
Steve Frenchf19159d2010-04-21 04:12:10 +000093 cFYI(1, "%s: inode %llu is new", __func__, cifs_i->uniqueid);
Jeff Laytondf2cf172010-02-12 07:44:16 -050094 return;
95 }
96
97 /* don't bother with revalidation if we have an oplock */
98 if (cifs_i->clientCanCacheRead) {
Steve Frenchf19159d2010-04-21 04:12:10 +000099 cFYI(1, "%s: inode %llu is oplocked", __func__,
100 cifs_i->uniqueid);
Jeff Laytondf2cf172010-02-12 07:44:16 -0500101 return;
102 }
103
104 /* revalidate if mtime or size have changed */
105 if (timespec_equal(&inode->i_mtime, &fattr->cf_mtime) &&
106 cifs_i->server_eof == fattr->cf_eof) {
Steve Frenchf19159d2010-04-21 04:12:10 +0000107 cFYI(1, "%s: inode %llu is unchanged", __func__,
108 cifs_i->uniqueid);
Jeff Laytondf2cf172010-02-12 07:44:16 -0500109 return;
110 }
111
Steve Frenchf19159d2010-04-21 04:12:10 +0000112 cFYI(1, "%s: invalidating inode %llu mapping", __func__,
113 cifs_i->uniqueid);
Jeff Laytondf2cf172010-02-12 07:44:16 -0500114 cifs_i->invalid_mapping = true;
115}
116
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400117/* populate an inode with info from a cifs_fattr struct */
118void
119cifs_fattr_to_inode(struct inode *inode, struct cifs_fattr *fattr)
Christoph Hellwig75f12982008-02-25 20:25:21 +0000120{
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400121 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400122 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
123 unsigned long oldtime = cifs_i->time;
Christoph Hellwig75f12982008-02-25 20:25:21 +0000124
Jeff Laytondf2cf172010-02-12 07:44:16 -0500125 cifs_revalidate_cache(inode, fattr);
126
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400127 inode->i_atime = fattr->cf_atime;
128 inode->i_mtime = fattr->cf_mtime;
129 inode->i_ctime = fattr->cf_ctime;
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400130 inode->i_rdev = fattr->cf_rdev;
131 inode->i_nlink = fattr->cf_nlink;
132 inode->i_uid = fattr->cf_uid;
133 inode->i_gid = fattr->cf_gid;
134
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400135 /* if dynperm is set, don't clobber existing mode */
136 if (inode->i_state & I_NEW ||
137 !(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM))
138 inode->i_mode = fattr->cf_mode;
139
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400140 cifs_i->cifsAttrs = fattr->cf_cifsattrs;
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400141
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400142 if (fattr->cf_flags & CIFS_FATTR_NEED_REVAL)
143 cifs_i->time = 0;
144 else
145 cifs_i->time = jiffies;
146
Joe Perchesb6b38f72010-04-21 03:50:45 +0000147 cFYI(1, "inode 0x%p old_time=%ld new_time=%ld", inode,
148 oldtime, cifs_i->time);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400149
150 cifs_i->delete_pending = fattr->cf_flags & CIFS_FATTR_DELETE_PENDING;
Christoph Hellwig75f12982008-02-25 20:25:21 +0000151
Jeff Layton835a36c2010-02-10 16:21:33 -0500152 cifs_i->server_eof = fattr->cf_eof;
Christoph Hellwig75f12982008-02-25 20:25:21 +0000153 /*
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400154 * Can't safely change the file size here if the client is writing to
155 * it due to potential races.
Christoph Hellwig75f12982008-02-25 20:25:21 +0000156 */
Christoph Hellwig75f12982008-02-25 20:25:21 +0000157 spin_lock(&inode->i_lock);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400158 if (is_size_safe_to_change(cifs_i, fattr->cf_eof)) {
159 i_size_write(inode, fattr->cf_eof);
Christoph Hellwig75f12982008-02-25 20:25:21 +0000160
161 /*
162 * i_blocks is not related to (i_size / i_blksize),
163 * but instead 512 byte (2**9) size is required for
164 * calculating num blocks.
165 */
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400166 inode->i_blocks = (512 - 1 + fattr->cf_bytes) >> 9;
Christoph Hellwig75f12982008-02-25 20:25:21 +0000167 }
168 spin_unlock(&inode->i_lock);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400169
170 cifs_set_ops(inode, fattr->cf_flags & CIFS_FATTR_DFS_REFERRAL);
Christoph Hellwig75f12982008-02-25 20:25:21 +0000171}
172
Jeff Layton4065c802010-05-17 07:18:58 -0400173void
174cifs_fill_uniqueid(struct super_block *sb, struct cifs_fattr *fattr)
175{
176 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
177
178 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM)
179 return;
180
181 fattr->cf_uniqueid = iunique(sb, ROOT_I);
182}
183
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400184/* Fill a cifs_fattr struct with info from FILE_UNIX_BASIC_INFO. */
185void
186cifs_unix_basic_to_fattr(struct cifs_fattr *fattr, FILE_UNIX_BASIC_INFO *info,
187 struct cifs_sb_info *cifs_sb)
188{
189 memset(fattr, 0, sizeof(*fattr));
190 fattr->cf_uniqueid = le64_to_cpu(info->UniqueId);
191 fattr->cf_bytes = le64_to_cpu(info->NumOfBytes);
192 fattr->cf_eof = le64_to_cpu(info->EndOfFile);
193
194 fattr->cf_atime = cifs_NTtimeToUnix(info->LastAccessTime);
195 fattr->cf_mtime = cifs_NTtimeToUnix(info->LastModificationTime);
196 fattr->cf_ctime = cifs_NTtimeToUnix(info->LastStatusChange);
197 fattr->cf_mode = le64_to_cpu(info->Permissions);
198
199 /*
200 * Since we set the inode type below we need to mask off
201 * to avoid strange results if bits set above.
202 */
203 fattr->cf_mode &= ~S_IFMT;
204 switch (le32_to_cpu(info->Type)) {
205 case UNIX_FILE:
206 fattr->cf_mode |= S_IFREG;
207 fattr->cf_dtype = DT_REG;
208 break;
209 case UNIX_SYMLINK:
210 fattr->cf_mode |= S_IFLNK;
211 fattr->cf_dtype = DT_LNK;
212 break;
213 case UNIX_DIR:
214 fattr->cf_mode |= S_IFDIR;
215 fattr->cf_dtype = DT_DIR;
216 break;
217 case UNIX_CHARDEV:
218 fattr->cf_mode |= S_IFCHR;
219 fattr->cf_dtype = DT_CHR;
220 fattr->cf_rdev = MKDEV(le64_to_cpu(info->DevMajor),
221 le64_to_cpu(info->DevMinor) & MINORMASK);
222 break;
223 case UNIX_BLOCKDEV:
224 fattr->cf_mode |= S_IFBLK;
225 fattr->cf_dtype = DT_BLK;
226 fattr->cf_rdev = MKDEV(le64_to_cpu(info->DevMajor),
227 le64_to_cpu(info->DevMinor) & MINORMASK);
228 break;
229 case UNIX_FIFO:
230 fattr->cf_mode |= S_IFIFO;
231 fattr->cf_dtype = DT_FIFO;
232 break;
233 case UNIX_SOCKET:
234 fattr->cf_mode |= S_IFSOCK;
235 fattr->cf_dtype = DT_SOCK;
236 break;
237 default:
238 /* safest to call it a file if we do not know */
239 fattr->cf_mode |= S_IFREG;
240 fattr->cf_dtype = DT_REG;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000241 cFYI(1, "unknown type %d", le32_to_cpu(info->Type));
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400242 break;
243 }
244
245 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_OVERR_UID)
246 fattr->cf_uid = cifs_sb->mnt_uid;
247 else
248 fattr->cf_uid = le64_to_cpu(info->Uid);
249
250 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_OVERR_GID)
251 fattr->cf_gid = cifs_sb->mnt_gid;
252 else
253 fattr->cf_gid = le64_to_cpu(info->Gid);
254
255 fattr->cf_nlink = le64_to_cpu(info->Nlinks);
256}
Steve Frenchb9a32602008-05-20 21:52:32 +0000257
258/*
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400259 * Fill a cifs_fattr struct with fake inode info.
260 *
261 * Needed to setup cifs_fattr data for the directory which is the
262 * junction to the new submount (ie to setup the fake directory
263 * which represents a DFS referral).
Steve Frenchb9a32602008-05-20 21:52:32 +0000264 */
Steve Frenchf1230c92009-07-22 23:13:01 +0000265static void
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400266cifs_create_dfs_fattr(struct cifs_fattr *fattr, struct super_block *sb)
Steve French0e4bbde2008-05-20 19:50:46 +0000267{
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400268 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
Steve French0e4bbde2008-05-20 19:50:46 +0000269
Joe Perchesb6b38f72010-04-21 03:50:45 +0000270 cFYI(1, "creating fake fattr for DFS referral");
Steve French0e4bbde2008-05-20 19:50:46 +0000271
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400272 memset(fattr, 0, sizeof(*fattr));
273 fattr->cf_mode = S_IFDIR | S_IXUGO | S_IRWXU;
274 fattr->cf_uid = cifs_sb->mnt_uid;
275 fattr->cf_gid = cifs_sb->mnt_gid;
276 fattr->cf_atime = CURRENT_TIME;
277 fattr->cf_ctime = CURRENT_TIME;
278 fattr->cf_mtime = CURRENT_TIME;
279 fattr->cf_nlink = 2;
280 fattr->cf_flags |= CIFS_FATTR_DFS_REFERRAL;
Steve French0e4bbde2008-05-20 19:50:46 +0000281}
282
Jeff Laytonabab0952010-02-12 07:44:18 -0500283int cifs_get_file_info_unix(struct file *filp)
284{
285 int rc;
286 int xid;
287 FILE_UNIX_BASIC_INFO find_data;
288 struct cifs_fattr fattr;
289 struct inode *inode = filp->f_path.dentry->d_inode;
290 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
291 struct cifsTconInfo *tcon = cifs_sb->tcon;
Joe Perchesc21dfb62010-07-12 13:50:14 -0700292 struct cifsFileInfo *cfile = filp->private_data;
Jeff Laytonabab0952010-02-12 07:44:18 -0500293
294 xid = GetXid();
295 rc = CIFSSMBUnixQFileInfo(xid, tcon, cfile->netfid, &find_data);
296 if (!rc) {
297 cifs_unix_basic_to_fattr(&fattr, &find_data, cifs_sb);
298 } else if (rc == -EREMOTE) {
299 cifs_create_dfs_fattr(&fattr, inode->i_sb);
300 rc = 0;
301 }
302
303 cifs_fattr_to_inode(inode, &fattr);
304 FreeXid(xid);
305 return rc;
306}
307
Linus Torvalds1da177e2005-04-16 15:20:36 -0700308int cifs_get_inode_info_unix(struct inode **pinode,
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400309 const unsigned char *full_path,
310 struct super_block *sb, int xid)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700311{
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400312 int rc;
Steve French0e4bbde2008-05-20 19:50:46 +0000313 FILE_UNIX_BASIC_INFO find_data;
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400314 struct cifs_fattr fattr;
315 struct cifsTconInfo *tcon;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700316 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700317
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400318 tcon = cifs_sb->tcon;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000319 cFYI(1, "Getting info on %s", full_path);
Igor Mammedov79626702008-03-09 03:44:18 +0000320
Linus Torvalds1da177e2005-04-16 15:20:36 -0700321 /* could have done a find first instead but this returns more info */
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400322 rc = CIFSSMBUnixQPathInfo(xid, tcon, full_path, &find_data,
Steve French737b7582005-04-28 22:41:06 -0700323 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
324 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400325
326 if (!rc) {
327 cifs_unix_basic_to_fattr(&fattr, &find_data, cifs_sb);
328 } else if (rc == -EREMOTE) {
329 cifs_create_dfs_fattr(&fattr, sb);
Jeff Laytone911d0c2008-07-12 13:47:59 -0700330 rc = 0;
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400331 } else {
332 return rc;
Steve French0e4bbde2008-05-20 19:50:46 +0000333 }
334
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400335 if (*pinode == NULL) {
336 /* get new inode */
Jeff Layton4065c802010-05-17 07:18:58 -0400337 cifs_fill_uniqueid(sb, &fattr);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400338 *pinode = cifs_iget(sb, &fattr);
339 if (!*pinode)
340 rc = -ENOMEM;
341 } else {
342 /* we already have inode, update it */
343 cifs_fattr_to_inode(*pinode, &fattr);
344 }
Steve French0e4bbde2008-05-20 19:50:46 +0000345
Linus Torvalds1da177e2005-04-16 15:20:36 -0700346 return rc;
347}
348
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400349static int
350cifs_sfu_type(struct cifs_fattr *fattr, const unsigned char *path,
351 struct cifs_sb_info *cifs_sb, int xid)
Steve Frenchd6e2f2a2005-11-15 16:43:39 -0800352{
353 int rc;
Steve French4b18f2a2008-04-29 00:06:05 +0000354 int oplock = 0;
Steve Frenchd6e2f2a2005-11-15 16:43:39 -0800355 __u16 netfid;
356 struct cifsTconInfo *pTcon = cifs_sb->tcon;
Steve French86c96b42005-11-18 20:25:31 -0800357 char buf[24];
Steve Frenchd6e2f2a2005-11-15 16:43:39 -0800358 unsigned int bytes_read;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000359 char *pbuf;
Steve Frenchd6e2f2a2005-11-15 16:43:39 -0800360
361 pbuf = buf;
362
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400363 fattr->cf_mode &= ~S_IFMT;
364
365 if (fattr->cf_eof == 0) {
366 fattr->cf_mode |= S_IFIFO;
367 fattr->cf_dtype = DT_FIFO;
Steve Frenchd6e2f2a2005-11-15 16:43:39 -0800368 return 0;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400369 } else if (fattr->cf_eof < 8) {
370 fattr->cf_mode |= S_IFREG;
371 fattr->cf_dtype = DT_REG;
Steve Frenchd6e2f2a2005-11-15 16:43:39 -0800372 return -EINVAL; /* EOPNOTSUPP? */
373 }
Steve French50c2f752007-07-13 00:33:32 +0000374
Steve Frenchd6e2f2a2005-11-15 16:43:39 -0800375 rc = CIFSSMBOpen(xid, pTcon, path, FILE_OPEN, GENERIC_READ,
376 CREATE_NOT_DIR, &netfid, &oplock, NULL,
377 cifs_sb->local_nls,
378 cifs_sb->mnt_cifs_flags &
379 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000380 if (rc == 0) {
Steve Frenchec637e32005-12-12 20:53:18 -0800381 int buf_type = CIFS_NO_BUFFER;
Steve Frenchd6e2f2a2005-11-15 16:43:39 -0800382 /* Read header */
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400383 rc = CIFSSMBRead(xid, pTcon, netfid,
Steve French86c96b42005-11-18 20:25:31 -0800384 24 /* length */, 0 /* offset */,
Steve Frenchec637e32005-12-12 20:53:18 -0800385 &bytes_read, &pbuf, &buf_type);
Steve French4523cc32007-04-30 20:13:06 +0000386 if ((rc == 0) && (bytes_read >= 8)) {
387 if (memcmp("IntxBLK", pbuf, 8) == 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000388 cFYI(1, "Block device");
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400389 fattr->cf_mode |= S_IFBLK;
390 fattr->cf_dtype = DT_BLK;
Steve French4523cc32007-04-30 20:13:06 +0000391 if (bytes_read == 24) {
Steve French86c96b42005-11-18 20:25:31 -0800392 /* we have enough to decode dev num */
393 __u64 mjr; /* major */
394 __u64 mnr; /* minor */
395 mjr = le64_to_cpu(*(__le64 *)(pbuf+8));
396 mnr = le64_to_cpu(*(__le64 *)(pbuf+16));
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400397 fattr->cf_rdev = MKDEV(mjr, mnr);
Steve French86c96b42005-11-18 20:25:31 -0800398 }
Steve French4523cc32007-04-30 20:13:06 +0000399 } else if (memcmp("IntxCHR", pbuf, 8) == 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000400 cFYI(1, "Char device");
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400401 fattr->cf_mode |= S_IFCHR;
402 fattr->cf_dtype = DT_CHR;
Steve French4523cc32007-04-30 20:13:06 +0000403 if (bytes_read == 24) {
Steve French86c96b42005-11-18 20:25:31 -0800404 /* we have enough to decode dev num */
405 __u64 mjr; /* major */
406 __u64 mnr; /* minor */
407 mjr = le64_to_cpu(*(__le64 *)(pbuf+8));
408 mnr = le64_to_cpu(*(__le64 *)(pbuf+16));
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400409 fattr->cf_rdev = MKDEV(mjr, mnr);
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000410 }
Steve French4523cc32007-04-30 20:13:06 +0000411 } else if (memcmp("IntxLNK", pbuf, 7) == 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000412 cFYI(1, "Symlink");
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400413 fattr->cf_mode |= S_IFLNK;
414 fattr->cf_dtype = DT_LNK;
Steve French86c96b42005-11-18 20:25:31 -0800415 } else {
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400416 fattr->cf_mode |= S_IFREG; /* file? */
417 fattr->cf_dtype = DT_REG;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000418 rc = -EOPNOTSUPP;
Steve French86c96b42005-11-18 20:25:31 -0800419 }
Steve French3020a1f2005-11-18 11:31:10 -0800420 } else {
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400421 fattr->cf_mode |= S_IFREG; /* then it is a file */
422 fattr->cf_dtype = DT_REG;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000423 rc = -EOPNOTSUPP; /* or some unknown SFU type */
424 }
Steve Frenchd6e2f2a2005-11-15 16:43:39 -0800425 CIFSSMBClose(xid, pTcon, netfid);
Steve Frenchd6e2f2a2005-11-15 16:43:39 -0800426 }
427 return rc;
Steve Frenchd6e2f2a2005-11-15 16:43:39 -0800428}
429
Steve French9e294f12005-11-17 16:59:21 -0800430#define SFBITS_MASK (S_ISVTX | S_ISGID | S_ISUID) /* SETFILEBITS valid bits */
431
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400432/*
433 * Fetch mode bits as provided by SFU.
434 *
435 * FIXME: Doesn't this clobber the type bit we got from cifs_sfu_type ?
436 */
437static int cifs_sfu_mode(struct cifs_fattr *fattr, const unsigned char *path,
438 struct cifs_sb_info *cifs_sb, int xid)
Steve French9e294f12005-11-17 16:59:21 -0800439{
Steve French3020a1f2005-11-18 11:31:10 -0800440#ifdef CONFIG_CIFS_XATTR
Steve French9e294f12005-11-17 16:59:21 -0800441 ssize_t rc;
442 char ea_value[4];
443 __u32 mode;
444
Jeff Layton31c05192010-02-10 16:18:26 -0500445 rc = CIFSSMBQAllEAs(xid, cifs_sb->tcon, path, "SETFILEBITS",
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400446 ea_value, 4 /* size of buf */, cifs_sb->local_nls,
447 cifs_sb->mnt_cifs_flags &
448 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French4523cc32007-04-30 20:13:06 +0000449 if (rc < 0)
Steve French9e294f12005-11-17 16:59:21 -0800450 return (int)rc;
451 else if (rc > 3) {
452 mode = le32_to_cpu(*((__le32 *)ea_value));
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400453 fattr->cf_mode &= ~SFBITS_MASK;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000454 cFYI(1, "special bits 0%o org mode 0%o", mode,
455 fattr->cf_mode);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400456 fattr->cf_mode = (mode & SFBITS_MASK) | fattr->cf_mode;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000457 cFYI(1, "special mode bits 0%o", mode);
Steve French9e294f12005-11-17 16:59:21 -0800458 }
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400459
460 return 0;
Steve French3020a1f2005-11-18 11:31:10 -0800461#else
462 return -EOPNOTSUPP;
463#endif
Steve French9e294f12005-11-17 16:59:21 -0800464}
465
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400466/* Fill a cifs_fattr struct with info from FILE_ALL_INFO */
Steve Frenchf1230c92009-07-22 23:13:01 +0000467static void
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400468cifs_all_info_to_fattr(struct cifs_fattr *fattr, FILE_ALL_INFO *info,
469 struct cifs_sb_info *cifs_sb, bool adjust_tz)
Steve Frenchb9a32602008-05-20 21:52:32 +0000470{
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400471 memset(fattr, 0, sizeof(*fattr));
472 fattr->cf_cifsattrs = le32_to_cpu(info->Attributes);
473 if (info->DeletePending)
474 fattr->cf_flags |= CIFS_FATTR_DELETE_PENDING;
Steve Frenchb9a32602008-05-20 21:52:32 +0000475
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400476 if (info->LastAccessTime)
477 fattr->cf_atime = cifs_NTtimeToUnix(info->LastAccessTime);
478 else
479 fattr->cf_atime = CURRENT_TIME;
480
481 fattr->cf_ctime = cifs_NTtimeToUnix(info->ChangeTime);
482 fattr->cf_mtime = cifs_NTtimeToUnix(info->LastWriteTime);
483
484 if (adjust_tz) {
485 fattr->cf_ctime.tv_sec += cifs_sb->tcon->ses->server->timeAdj;
486 fattr->cf_mtime.tv_sec += cifs_sb->tcon->ses->server->timeAdj;
487 }
488
489 fattr->cf_eof = le64_to_cpu(info->EndOfFile);
490 fattr->cf_bytes = le64_to_cpu(info->AllocationSize);
491
492 if (fattr->cf_cifsattrs & ATTR_DIRECTORY) {
493 fattr->cf_mode = S_IFDIR | cifs_sb->mnt_dir_mode;
494 fattr->cf_dtype = DT_DIR;
495 } else {
496 fattr->cf_mode = S_IFREG | cifs_sb->mnt_file_mode;
497 fattr->cf_dtype = DT_REG;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400498
Jeff Laytond0c280d2009-07-09 01:46:44 -0400499 /* clear write bits if ATTR_READONLY is set */
500 if (fattr->cf_cifsattrs & ATTR_READONLY)
501 fattr->cf_mode &= ~(S_IWUGO);
502 }
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400503
504 fattr->cf_nlink = le32_to_cpu(info->NumberOfLinks);
505
506 fattr->cf_uid = cifs_sb->mnt_uid;
507 fattr->cf_gid = cifs_sb->mnt_gid;
Steve Frenchb9a32602008-05-20 21:52:32 +0000508}
509
Jeff Laytonabab0952010-02-12 07:44:18 -0500510int cifs_get_file_info(struct file *filp)
511{
512 int rc;
513 int xid;
514 FILE_ALL_INFO find_data;
515 struct cifs_fattr fattr;
516 struct inode *inode = filp->f_path.dentry->d_inode;
517 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
518 struct cifsTconInfo *tcon = cifs_sb->tcon;
Joe Perchesc21dfb62010-07-12 13:50:14 -0700519 struct cifsFileInfo *cfile = filp->private_data;
Jeff Laytonabab0952010-02-12 07:44:18 -0500520
521 xid = GetXid();
522 rc = CIFSSMBQFileInfo(xid, tcon, cfile->netfid, &find_data);
523 if (rc == -EOPNOTSUPP || rc == -EINVAL) {
524 /*
525 * FIXME: legacy server -- fall back to path-based call?
Steve Frenchff2157132010-03-09 20:30:42 +0000526 * for now, just skip revalidating and mark inode for
527 * immediate reval.
528 */
Jeff Laytonabab0952010-02-12 07:44:18 -0500529 rc = 0;
530 CIFS_I(inode)->time = 0;
531 goto cgfi_exit;
532 } else if (rc == -EREMOTE) {
533 cifs_create_dfs_fattr(&fattr, inode->i_sb);
534 rc = 0;
535 } else if (rc)
536 goto cgfi_exit;
537
538 /*
539 * don't bother with SFU junk here -- just mark inode as needing
540 * revalidation.
541 */
542 cifs_all_info_to_fattr(&fattr, &find_data, cifs_sb, false);
543 fattr.cf_uniqueid = CIFS_I(inode)->uniqueid;
544 fattr.cf_flags |= CIFS_FATTR_NEED_REVAL;
545 cifs_fattr_to_inode(inode, &fattr);
546cgfi_exit:
547 FreeXid(xid);
548 return rc;
549}
550
Linus Torvalds1da177e2005-04-16 15:20:36 -0700551int cifs_get_inode_info(struct inode **pinode,
Steve French646dd532008-05-15 01:50:56 +0000552 const unsigned char *full_path, FILE_ALL_INFO *pfindData,
Steve French8b1327f2008-03-14 22:37:16 +0000553 struct super_block *sb, int xid, const __u16 *pfid)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700554{
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400555 int rc = 0, tmprc;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700556 struct cifsTconInfo *pTcon;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700557 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700558 char *buf = NULL;
Steve French5ade9de2008-05-02 20:56:23 +0000559 bool adjustTZ = false;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400560 struct cifs_fattr fattr;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700561
562 pTcon = cifs_sb->tcon;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000563 cFYI(1, "Getting info on %s", full_path);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700564
Steve Frenchd0d2f2d2005-06-02 15:12:36 -0700565 if ((pfindData == NULL) && (*pinode != NULL)) {
566 if (CIFS_I(*pinode)->clientCanCacheRead) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000567 cFYI(1, "No need to revalidate cached inode sizes");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700568 return rc;
569 }
570 }
571
572 /* if file info not passed in then get it from server */
Steve Frenchd0d2f2d2005-06-02 15:12:36 -0700573 if (pfindData == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700574 buf = kmalloc(sizeof(FILE_ALL_INFO), GFP_KERNEL);
Steve Frenchd0d2f2d2005-06-02 15:12:36 -0700575 if (buf == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700576 return -ENOMEM;
577 pfindData = (FILE_ALL_INFO *)buf;
Igor Mammedov79626702008-03-09 03:44:18 +0000578
Linus Torvalds1da177e2005-04-16 15:20:36 -0700579 /* could do find first instead but this returns more info */
Igor Mammedov79626702008-03-09 03:44:18 +0000580 rc = CIFSSMBQPathInfo(xid, pTcon, full_path, pfindData,
Steve Frenchacf1a1b2006-10-12 03:28:28 +0000581 0 /* not legacy */,
Steve French6b8edfe2005-08-23 20:26:03 -0700582 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
Steve French737b7582005-04-28 22:41:06 -0700583 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French6b8edfe2005-08-23 20:26:03 -0700584 /* BB optimize code so we do not make the above call
585 when server claims no NT SMB support and the above call
586 failed at least once - set flag in tcon or mount */
Steve French4523cc32007-04-30 20:13:06 +0000587 if ((rc == -EOPNOTSUPP) || (rc == -EINVAL)) {
Igor Mammedov79626702008-03-09 03:44:18 +0000588 rc = SMBQueryInformation(xid, pTcon, full_path,
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000589 pfindData, cifs_sb->local_nls,
Steve French6b8edfe2005-08-23 20:26:03 -0700590 cifs_sb->mnt_cifs_flags &
591 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French4b18f2a2008-04-29 00:06:05 +0000592 adjustTZ = true;
Steve French6b8edfe2005-08-23 20:26:03 -0700593 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700594 }
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400595
596 if (!rc) {
597 cifs_all_info_to_fattr(&fattr, (FILE_ALL_INFO *) pfindData,
598 cifs_sb, adjustTZ);
599 } else if (rc == -EREMOTE) {
600 cifs_create_dfs_fattr(&fattr, sb);
Steve Frenchb9a32602008-05-20 21:52:32 +0000601 rc = 0;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400602 } else {
Igor Mammedov79626702008-03-09 03:44:18 +0000603 goto cgii_exit;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400604 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700605
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400606 /*
607 * If an inode wasn't passed in, then get the inode number
608 *
609 * Is an i_ino of zero legal? Can we use that to check if the server
610 * supports returning inode numbers? Are there other sanity checks we
611 * can use to ensure that the server is really filling in that field?
612 *
613 * We can not use the IndexNumber field by default from Windows or
614 * Samba (in ALL_INFO buf) but we can request it explicitly. The SNIA
615 * CIFS spec claims that this value is unique within the scope of a
616 * share, and the windows docs hint that it's actually unique
617 * per-machine.
618 *
619 * There may be higher info levels that work but are there Windows
620 * server or network appliances for which IndexNumber field is not
621 * guaranteed unique?
622 */
Steve Frenchb9a32602008-05-20 21:52:32 +0000623 if (*pinode == NULL) {
Steve Frenchb9a32602008-05-20 21:52:32 +0000624 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM) {
625 int rc1 = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700626
Steve Frenchb9a32602008-05-20 21:52:32 +0000627 rc1 = CIFSGetSrvInodeNumber(xid, pTcon,
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400628 full_path, &fattr.cf_uniqueid,
Steve French737b7582005-04-28 22:41:06 -0700629 cifs_sb->local_nls,
630 cifs_sb->mnt_cifs_flags &
631 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Laytonec06aed2009-11-06 14:18:29 -0500632 if (rc1 || !fattr.cf_uniqueid) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000633 cFYI(1, "GetSrvInodeNum rc %d", rc1);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400634 fattr.cf_uniqueid = iunique(sb, ROOT_I);
Jeff Laytonec06aed2009-11-06 14:18:29 -0500635 cifs_autodisable_serverino(cifs_sb);
Jeff Layton132ac7b2009-02-10 07:33:57 -0500636 }
Jeff Layton132ac7b2009-02-10 07:33:57 -0500637 } else {
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400638 fattr.cf_uniqueid = iunique(sb, ROOT_I);
Jeff Layton132ac7b2009-02-10 07:33:57 -0500639 }
Steve Frenchb9a32602008-05-20 21:52:32 +0000640 } else {
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400641 fattr.cf_uniqueid = CIFS_I(*pinode)->uniqueid;
Steve Frenchb9a32602008-05-20 21:52:32 +0000642 }
643
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400644 /* query for SFU type info if supported and needed */
645 if (fattr.cf_cifsattrs & ATTR_SYSTEM &&
646 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_UNX_EMUL) {
647 tmprc = cifs_sfu_type(&fattr, full_path, cifs_sb, xid);
648 if (tmprc)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000649 cFYI(1, "cifs_sfu_type failed: %d", tmprc);
Steve Frenchb9a32602008-05-20 21:52:32 +0000650 }
Steve Frenchb9a32602008-05-20 21:52:32 +0000651
Steve Frenchb9a32602008-05-20 21:52:32 +0000652#ifdef CONFIG_CIFS_EXPERIMENTAL
653 /* fill in 0777 bits from ACL */
654 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000655 cFYI(1, "Getting mode bits from ACL");
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400656 cifs_acl_to_fattr(cifs_sb, &fattr, *pinode, full_path, pfid);
Steve Frenchb9a32602008-05-20 21:52:32 +0000657 }
658#endif
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400659
660 /* fill in remaining high mode bits e.g. SUID, VTX */
661 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_UNX_EMUL)
662 cifs_sfu_mode(&fattr, full_path, cifs_sb, xid);
663
664 if (!*pinode) {
665 *pinode = cifs_iget(sb, &fattr);
666 if (!*pinode)
667 rc = -ENOMEM;
668 } else {
669 cifs_fattr_to_inode(*pinode, &fattr);
Steve Frenchb9a32602008-05-20 21:52:32 +0000670 }
671
Igor Mammedov79626702008-03-09 03:44:18 +0000672cgii_exit:
Linus Torvalds1da177e2005-04-16 15:20:36 -0700673 kfree(buf);
674 return rc;
675}
676
Steve French7f8ed422007-09-28 22:28:55 +0000677static const struct inode_operations cifs_ipc_inode_ops = {
678 .lookup = cifs_lookup,
679};
680
Igor Mammedove4cce942009-02-10 14:10:26 +0300681char *cifs_build_path_to_root(struct cifs_sb_info *cifs_sb)
Steve French8be0ed42008-12-05 19:14:12 +0000682{
683 int pplen = cifs_sb->prepathlen;
684 int dfsplen;
685 char *full_path = NULL;
686
687 /* if no prefix path, simply set path to the root of share to "" */
688 if (pplen == 0) {
689 full_path = kmalloc(1, GFP_KERNEL);
690 if (full_path)
691 full_path[0] = 0;
692 return full_path;
693 }
694
695 if (cifs_sb->tcon && (cifs_sb->tcon->Flags & SMB_SHARE_IS_IN_DFS))
696 dfsplen = strnlen(cifs_sb->tcon->treeName, MAX_TREE_SIZE + 1);
697 else
698 dfsplen = 0;
699
700 full_path = kmalloc(dfsplen + pplen + 1, GFP_KERNEL);
701 if (full_path == NULL)
702 return full_path;
703
704 if (dfsplen) {
705 strncpy(full_path, cifs_sb->tcon->treeName, dfsplen);
706 /* switch slash direction in prepath depending on whether
707 * windows or posix style path names
708 */
709 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_POSIX_PATHS) {
710 int i;
711 for (i = 0; i < dfsplen; i++) {
712 if (full_path[i] == '\\')
713 full_path[i] = '/';
714 }
715 }
716 }
717 strncpy(full_path + dfsplen, cifs_sb->prepath, pplen);
718 full_path[dfsplen + pplen] = 0; /* add trailing null */
719 return full_path;
720}
721
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400722static int
723cifs_find_inode(struct inode *inode, void *opaque)
724{
725 struct cifs_fattr *fattr = (struct cifs_fattr *) opaque;
726
Jeff Laytonf30b9c12010-07-19 18:00:17 -0400727 /* don't match inode with different uniqueid */
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400728 if (CIFS_I(inode)->uniqueid != fattr->cf_uniqueid)
729 return 0;
730
Jeff Laytonf30b9c12010-07-19 18:00:17 -0400731 /* don't match inode of different type */
732 if ((inode->i_mode & S_IFMT) != (fattr->cf_mode & S_IFMT))
733 return 0;
734
Jeff Layton5acfec22010-08-02 17:43:54 -0400735 /* if it's not a directory or has no dentries, then flag it */
736 if (S_ISDIR(inode->i_mode) && !list_empty(&inode->i_dentry))
Jeff Layton3d694382010-05-11 14:59:55 -0400737 fattr->cf_flags |= CIFS_FATTR_INO_COLLISION;
Jeff Layton3d694382010-05-11 14:59:55 -0400738
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400739 return 1;
740}
741
742static int
743cifs_init_inode(struct inode *inode, void *opaque)
744{
745 struct cifs_fattr *fattr = (struct cifs_fattr *) opaque;
746
747 CIFS_I(inode)->uniqueid = fattr->cf_uniqueid;
748 return 0;
749}
750
Jeff Layton5acfec22010-08-02 17:43:54 -0400751/*
752 * walk dentry list for an inode and report whether it has aliases that
753 * are hashed. We use this to determine if a directory inode can actually
754 * be used.
755 */
756static bool
757inode_has_hashed_dentries(struct inode *inode)
758{
759 struct dentry *dentry;
760
761 spin_lock(&dcache_lock);
762 list_for_each_entry(dentry, &inode->i_dentry, d_alias) {
763 if (!d_unhashed(dentry) || IS_ROOT(dentry)) {
764 spin_unlock(&dcache_lock);
765 return true;
766 }
767 }
768 spin_unlock(&dcache_lock);
769 return false;
770}
771
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400772/* Given fattrs, get a corresponding inode */
773struct inode *
774cifs_iget(struct super_block *sb, struct cifs_fattr *fattr)
775{
776 unsigned long hash;
777 struct inode *inode;
778
Jeff Layton3d694382010-05-11 14:59:55 -0400779retry_iget5_locked:
Joe Perchesb6b38f72010-04-21 03:50:45 +0000780 cFYI(1, "looking for uniqueid=%llu", fattr->cf_uniqueid);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400781
782 /* hash down to 32-bits on 32-bit arch */
783 hash = cifs_uniqueid_to_ino_t(fattr->cf_uniqueid);
784
785 inode = iget5_locked(sb, hash, cifs_find_inode, cifs_init_inode, fattr);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400786 if (inode) {
Jeff Layton5acfec22010-08-02 17:43:54 -0400787 /* was there a potentially problematic inode collision? */
Jeff Layton3d694382010-05-11 14:59:55 -0400788 if (fattr->cf_flags & CIFS_FATTR_INO_COLLISION) {
Jeff Layton3d694382010-05-11 14:59:55 -0400789 fattr->cf_flags &= ~CIFS_FATTR_INO_COLLISION;
Jeff Layton5acfec22010-08-02 17:43:54 -0400790
791 if (inode_has_hashed_dentries(inode)) {
792 cifs_autodisable_serverino(CIFS_SB(sb));
793 iput(inode);
794 fattr->cf_uniqueid = iunique(sb, ROOT_I);
795 goto retry_iget5_locked;
796 }
Jeff Layton3d694382010-05-11 14:59:55 -0400797 }
798
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400799 cifs_fattr_to_inode(inode, fattr);
800 if (sb->s_flags & MS_NOATIME)
801 inode->i_flags |= S_NOATIME | S_NOCMTIME;
802 if (inode->i_state & I_NEW) {
803 inode->i_ino = hash;
Jeff Layton522440e2010-09-29 09:49:54 -0400804 if (S_ISREG(inode->i_mode))
805 inode->i_data.backing_dev_info = sb->s_bdi;
Steve French0ccd4802010-07-16 04:31:02 +0000806#ifdef CONFIG_CIFS_FSCACHE
Suresh Jayaraman9451a9a2010-07-05 18:12:45 +0530807 /* initialize per-inode cache cookie pointer */
808 CIFS_I(inode)->fscache = NULL;
Steve French0ccd4802010-07-16 04:31:02 +0000809#endif
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400810 unlock_new_inode(inode);
811 }
812 }
813
814 return inode;
815}
816
Linus Torvalds1da177e2005-04-16 15:20:36 -0700817/* gets root inode */
Jeff Laytonbd433d42009-05-27 09:37:34 -0400818struct inode *cifs_root_iget(struct super_block *sb, unsigned long ino)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700819{
David Howellsce634ab2008-02-07 00:15:33 -0800820 int xid;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700821 struct cifs_sb_info *cifs_sb;
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400822 struct inode *inode = NULL;
David Howellsce634ab2008-02-07 00:15:33 -0800823 long rc;
Steve French8be0ed42008-12-05 19:14:12 +0000824 char *full_path;
David Howellsce634ab2008-02-07 00:15:33 -0800825
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400826 cifs_sb = CIFS_SB(sb);
Igor Mammedove4cce942009-02-10 14:10:26 +0300827 full_path = cifs_build_path_to_root(cifs_sb);
Steve French8be0ed42008-12-05 19:14:12 +0000828 if (full_path == NULL)
829 return ERR_PTR(-ENOMEM);
Steve Frenchc18c8422007-07-18 23:21:09 +0000830
Steve French8be0ed42008-12-05 19:14:12 +0000831 xid = GetXid();
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400832 if (cifs_sb->tcon->unix_ext)
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400833 rc = cifs_get_inode_info_unix(&inode, full_path, sb, xid);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400834 else
835 rc = cifs_get_inode_info(&inode, full_path, NULL, sb,
Steve French8be0ed42008-12-05 19:14:12 +0000836 xid, NULL);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400837
838 if (!inode)
Suresh Jayaramanf0138a72010-08-26 14:46:09 +0530839 return ERR_PTR(rc);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400840
Steve French0ccd4802010-07-16 04:31:02 +0000841#ifdef CONFIG_CIFS_FSCACHE
Suresh Jayaramand03382c2010-07-05 18:12:27 +0530842 /* populate tcon->resource_id */
843 cifs_sb->tcon->resource_id = CIFS_I(inode)->uniqueid;
Steve French0ccd4802010-07-16 04:31:02 +0000844#endif
Suresh Jayaramand03382c2010-07-05 18:12:27 +0530845
Steve French7f8ed422007-09-28 22:28:55 +0000846 if (rc && cifs_sb->tcon->ipc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000847 cFYI(1, "ipc connection - fake read inode");
Steve French7f8ed422007-09-28 22:28:55 +0000848 inode->i_mode |= S_IFDIR;
849 inode->i_nlink = 2;
850 inode->i_op = &cifs_ipc_inode_ops;
851 inode->i_fop = &simple_dir_operations;
852 inode->i_uid = cifs_sb->mnt_uid;
853 inode->i_gid = cifs_sb->mnt_gid;
Steve Frenchad661332008-08-12 14:14:40 +0000854 } else if (rc) {
Steve French8be0ed42008-12-05 19:14:12 +0000855 kfree(full_path);
David Howellsce634ab2008-02-07 00:15:33 -0800856 _FreeXid(xid);
857 iget_failed(inode);
858 return ERR_PTR(rc);
Steve French7f8ed422007-09-28 22:28:55 +0000859 }
860
David Howellsce634ab2008-02-07 00:15:33 -0800861
Steve French8be0ed42008-12-05 19:14:12 +0000862 kfree(full_path);
David Howellsce634ab2008-02-07 00:15:33 -0800863 /* can not call macro FreeXid here since in a void func
864 * TODO: This is no longer true
865 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700866 _FreeXid(xid);
David Howellsce634ab2008-02-07 00:15:33 -0800867 return inode;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700868}
869
Steve French388e57b2008-09-16 23:50:58 +0000870static int
871cifs_set_file_info(struct inode *inode, struct iattr *attrs, int xid,
872 char *full_path, __u32 dosattr)
873{
874 int rc;
875 int oplock = 0;
876 __u16 netfid;
877 __u32 netpid;
878 bool set_time = false;
879 struct cifsFileInfo *open_file;
880 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
881 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
882 struct cifsTconInfo *pTcon = cifs_sb->tcon;
883 FILE_BASIC_INFO info_buf;
884
Steve French1adcb712009-02-25 14:19:56 +0000885 if (attrs == NULL)
886 return -EINVAL;
887
Steve French388e57b2008-09-16 23:50:58 +0000888 if (attrs->ia_valid & ATTR_ATIME) {
889 set_time = true;
890 info_buf.LastAccessTime =
891 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_atime));
892 } else
893 info_buf.LastAccessTime = 0;
894
895 if (attrs->ia_valid & ATTR_MTIME) {
896 set_time = true;
897 info_buf.LastWriteTime =
898 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_mtime));
899 } else
900 info_buf.LastWriteTime = 0;
901
902 /*
903 * Samba throws this field away, but windows may actually use it.
904 * Do not set ctime unless other time stamps are changed explicitly
905 * (i.e. by utimes()) since we would then have a mix of client and
906 * server times.
907 */
908 if (set_time && (attrs->ia_valid & ATTR_CTIME)) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000909 cFYI(1, "CIFS - CTIME changed");
Steve French388e57b2008-09-16 23:50:58 +0000910 info_buf.ChangeTime =
911 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_ctime));
912 } else
913 info_buf.ChangeTime = 0;
914
915 info_buf.CreationTime = 0; /* don't change */
916 info_buf.Attributes = cpu_to_le32(dosattr);
917
918 /*
919 * If the file is already open for write, just use that fileid
920 */
921 open_file = find_writable_file(cifsInode);
922 if (open_file) {
923 netfid = open_file->netfid;
924 netpid = open_file->pid;
925 goto set_via_filehandle;
926 }
927
928 /*
929 * NT4 apparently returns success on this call, but it doesn't
930 * really work.
931 */
932 if (!(pTcon->ses->flags & CIFS_SES_NT4)) {
933 rc = CIFSSMBSetPathInfo(xid, pTcon, full_path,
934 &info_buf, cifs_sb->local_nls,
935 cifs_sb->mnt_cifs_flags &
936 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Layton6b37faa2008-10-06 21:54:41 +0000937 if (rc == 0) {
938 cifsInode->cifsAttrs = dosattr;
939 goto out;
940 } else if (rc != -EOPNOTSUPP && rc != -EINVAL)
Steve French388e57b2008-09-16 23:50:58 +0000941 goto out;
942 }
943
Joe Perchesb6b38f72010-04-21 03:50:45 +0000944 cFYI(1, "calling SetFileInfo since SetPathInfo for "
945 "times not supported by this server");
Steve French388e57b2008-09-16 23:50:58 +0000946 rc = CIFSSMBOpen(xid, pTcon, full_path, FILE_OPEN,
947 SYNCHRONIZE | FILE_WRITE_ATTRIBUTES,
948 CREATE_NOT_DIR, &netfid, &oplock,
949 NULL, cifs_sb->local_nls,
950 cifs_sb->mnt_cifs_flags &
951 CIFS_MOUNT_MAP_SPECIAL_CHR);
952
953 if (rc != 0) {
954 if (rc == -EIO)
955 rc = -EINVAL;
956 goto out;
957 }
958
959 netpid = current->tgid;
960
961set_via_filehandle:
962 rc = CIFSSMBSetFileInfo(xid, pTcon, &info_buf, netfid, netpid);
Steve Frenchd3889082008-09-24 19:22:52 +0000963 if (!rc)
964 cifsInode->cifsAttrs = dosattr;
965
Steve French388e57b2008-09-16 23:50:58 +0000966 if (open_file == NULL)
967 CIFSSMBClose(xid, pTcon, netfid);
968 else
Dave Kleikamp6ab409b2009-08-31 11:07:12 -0400969 cifsFileInfo_put(open_file);
Steve French388e57b2008-09-16 23:50:58 +0000970out:
971 return rc;
972}
973
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400974/*
975 * open the given file (if it isn't already), set the DELETE_ON_CLOSE bit
976 * and rename it to a random name that hopefully won't conflict with
977 * anything else.
978 */
979static int
Steve French32709582008-10-20 00:44:19 +0000980cifs_rename_pending_delete(char *full_path, struct dentry *dentry, int xid)
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400981{
982 int oplock = 0;
983 int rc;
984 __u16 netfid;
Steve French32709582008-10-20 00:44:19 +0000985 struct inode *inode = dentry->d_inode;
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400986 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
987 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
988 struct cifsTconInfo *tcon = cifs_sb->tcon;
Steve French32709582008-10-20 00:44:19 +0000989 __u32 dosattr, origattr;
990 FILE_BASIC_INFO *info_buf = NULL;
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400991
992 rc = CIFSSMBOpen(xid, tcon, full_path, FILE_OPEN,
Jeff Laytondd1db2d2008-10-16 19:27:12 -0400993 DELETE|FILE_WRITE_ATTRIBUTES, CREATE_NOT_DIR,
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400994 &netfid, &oplock, NULL, cifs_sb->local_nls,
995 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
996 if (rc != 0)
997 goto out;
998
Steve French32709582008-10-20 00:44:19 +0000999 origattr = cifsInode->cifsAttrs;
1000 if (origattr == 0)
1001 origattr |= ATTR_NORMAL;
1002
1003 dosattr = origattr & ~ATTR_READONLY;
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001004 if (dosattr == 0)
1005 dosattr |= ATTR_NORMAL;
1006 dosattr |= ATTR_HIDDEN;
1007
Steve French32709582008-10-20 00:44:19 +00001008 /* set ATTR_HIDDEN and clear ATTR_READONLY, but only if needed */
1009 if (dosattr != origattr) {
1010 info_buf = kzalloc(sizeof(*info_buf), GFP_KERNEL);
1011 if (info_buf == NULL) {
1012 rc = -ENOMEM;
1013 goto out_close;
1014 }
1015 info_buf->Attributes = cpu_to_le32(dosattr);
1016 rc = CIFSSMBSetFileInfo(xid, tcon, info_buf, netfid,
1017 current->tgid);
1018 /* although we would like to mark the file hidden
1019 if that fails we will still try to rename it */
Steve French41346092008-10-20 18:24:42 +00001020 if (rc != 0)
Steve French32709582008-10-20 00:44:19 +00001021 cifsInode->cifsAttrs = dosattr;
1022 else
1023 dosattr = origattr; /* since not able to change them */
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001024 }
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001025
Jeff Laytondd1db2d2008-10-16 19:27:12 -04001026 /* rename the file */
1027 rc = CIFSSMBRenameOpenFile(xid, tcon, netfid, NULL, cifs_sb->local_nls,
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001028 cifs_sb->mnt_cifs_flags &
1029 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French32709582008-10-20 00:44:19 +00001030 if (rc != 0) {
1031 rc = -ETXTBSY;
1032 goto undo_setattr;
1033 }
Jeff Layton6d22f092008-09-23 11:48:35 -04001034
Steve French32709582008-10-20 00:44:19 +00001035 /* try to set DELETE_ON_CLOSE */
1036 if (!cifsInode->delete_pending) {
1037 rc = CIFSSMBSetFileDisposition(xid, tcon, true, netfid,
1038 current->tgid);
1039 /*
1040 * some samba versions return -ENOENT when we try to set the
1041 * file disposition here. Likely a samba bug, but work around
1042 * it for now. This means that some cifsXXX files may hang
1043 * around after they shouldn't.
1044 *
1045 * BB: remove this hack after more servers have the fix
1046 */
1047 if (rc == -ENOENT)
1048 rc = 0;
1049 else if (rc != 0) {
1050 rc = -ETXTBSY;
1051 goto undo_rename;
1052 }
1053 cifsInode->delete_pending = true;
1054 }
Jeff Layton7ce86d52008-09-24 11:32:59 -04001055
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001056out_close:
1057 CIFSSMBClose(xid, tcon, netfid);
1058out:
Steve French32709582008-10-20 00:44:19 +00001059 kfree(info_buf);
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001060 return rc;
Steve French32709582008-10-20 00:44:19 +00001061
1062 /*
1063 * reset everything back to the original state. Don't bother
1064 * dealing with errors here since we can't do anything about
1065 * them anyway.
1066 */
1067undo_rename:
1068 CIFSSMBRenameOpenFile(xid, tcon, netfid, dentry->d_name.name,
1069 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1070 CIFS_MOUNT_MAP_SPECIAL_CHR);
1071undo_setattr:
1072 if (dosattr != origattr) {
1073 info_buf->Attributes = cpu_to_le32(origattr);
1074 if (!CIFSSMBSetFileInfo(xid, tcon, info_buf, netfid,
1075 current->tgid))
1076 cifsInode->cifsAttrs = origattr;
1077 }
1078
1079 goto out_close;
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001080}
1081
Steve Frenchff694522009-04-20 19:45:13 +00001082
1083/*
1084 * If dentry->d_inode is null (usually meaning the cached dentry
1085 * is a negative dentry) then we would attempt a standard SMB delete, but
André Goddard Rosaaf901ca2009-11-14 13:09:05 -02001086 * if that fails we can not attempt the fall back mechanisms on EACCESS
1087 * but will return the EACCESS to the caller. Note that the VFS does not call
Steve Frenchff694522009-04-20 19:45:13 +00001088 * unlink on negative dentries currently.
1089 */
Jeff Layton5f0319a2008-09-16 14:05:16 -04001090int cifs_unlink(struct inode *dir, struct dentry *dentry)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001091{
1092 int rc = 0;
1093 int xid;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001094 char *full_path = NULL;
Jeff Layton5f0319a2008-09-16 14:05:16 -04001095 struct inode *inode = dentry->d_inode;
Steve Frenchff694522009-04-20 19:45:13 +00001096 struct cifsInodeInfo *cifs_inode;
Jeff Layton5f0319a2008-09-16 14:05:16 -04001097 struct super_block *sb = dir->i_sb;
1098 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
1099 struct cifsTconInfo *tcon = cifs_sb->tcon;
Steve French60502472008-10-07 18:42:52 +00001100 struct iattr *attrs = NULL;
1101 __u32 dosattr = 0, origattr = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001102
Joe Perchesb6b38f72010-04-21 03:50:45 +00001103 cFYI(1, "cifs_unlink, dir=0x%p, dentry=0x%p", dir, dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001104
1105 xid = GetXid();
1106
Jeff Layton5f0319a2008-09-16 14:05:16 -04001107 /* Unlink can be called from rename so we can not take the
1108 * sb->s_vfs_rename_mutex here */
1109 full_path = build_path_from_dentry(dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001110 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301111 rc = -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001112 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301113 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001114 }
Steve French2d785a52007-07-15 01:48:57 +00001115
Jeff Layton5f0319a2008-09-16 14:05:16 -04001116 if ((tcon->ses->capabilities & CAP_UNIX) &&
Steve French2d785a52007-07-15 01:48:57 +00001117 (CIFS_UNIX_POSIX_PATH_OPS_CAP &
Jeff Layton5f0319a2008-09-16 14:05:16 -04001118 le64_to_cpu(tcon->fsUnixInfo.Capability))) {
1119 rc = CIFSPOSIXDelFile(xid, tcon, full_path,
Steve French2d785a52007-07-15 01:48:57 +00001120 SMB_POSIX_UNLINK_FILE_TARGET, cifs_sb->local_nls,
1121 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001122 cFYI(1, "posix del rc %d", rc);
Steve French2d785a52007-07-15 01:48:57 +00001123 if ((rc == 0) || (rc == -ENOENT))
1124 goto psx_del_no_retry;
1125 }
1126
Steve French60502472008-10-07 18:42:52 +00001127retry_std_delete:
Jeff Layton5f0319a2008-09-16 14:05:16 -04001128 rc = CIFSSMBDelFile(xid, tcon, full_path, cifs_sb->local_nls,
Steve French737b7582005-04-28 22:41:06 -07001129 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French60502472008-10-07 18:42:52 +00001130
Steve French2d785a52007-07-15 01:48:57 +00001131psx_del_no_retry:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001132 if (!rc) {
Jeff Layton5f0319a2008-09-16 14:05:16 -04001133 if (inode)
1134 drop_nlink(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001135 } else if (rc == -ENOENT) {
Jeff Layton5f0319a2008-09-16 14:05:16 -04001136 d_drop(dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001137 } else if (rc == -ETXTBSY) {
Steve French32709582008-10-20 00:44:19 +00001138 rc = cifs_rename_pending_delete(full_path, dentry, xid);
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001139 if (rc == 0)
1140 drop_nlink(inode);
Steve Frenchff694522009-04-20 19:45:13 +00001141 } else if ((rc == -EACCES) && (dosattr == 0) && inode) {
Steve French388e57b2008-09-16 23:50:58 +00001142 attrs = kzalloc(sizeof(*attrs), GFP_KERNEL);
1143 if (attrs == NULL) {
1144 rc = -ENOMEM;
1145 goto out_reval;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001146 }
Steve French388e57b2008-09-16 23:50:58 +00001147
1148 /* try to reset dos attributes */
Steve Frenchff694522009-04-20 19:45:13 +00001149 cifs_inode = CIFS_I(inode);
1150 origattr = cifs_inode->cifsAttrs;
Steve French60502472008-10-07 18:42:52 +00001151 if (origattr == 0)
1152 origattr |= ATTR_NORMAL;
1153 dosattr = origattr & ~ATTR_READONLY;
Steve French388e57b2008-09-16 23:50:58 +00001154 if (dosattr == 0)
1155 dosattr |= ATTR_NORMAL;
1156 dosattr |= ATTR_HIDDEN;
1157
1158 rc = cifs_set_file_info(inode, attrs, xid, full_path, dosattr);
Steve French388e57b2008-09-16 23:50:58 +00001159 if (rc != 0)
1160 goto out_reval;
Steve French60502472008-10-07 18:42:52 +00001161
1162 goto retry_std_delete;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001163 }
Steve French60502472008-10-07 18:42:52 +00001164
1165 /* undo the setattr if we errored out and it's needed */
1166 if (rc != 0 && dosattr != 0)
1167 cifs_set_file_info(inode, attrs, xid, full_path, origattr);
1168
Steve French388e57b2008-09-16 23:50:58 +00001169out_reval:
Jeff Layton5f0319a2008-09-16 14:05:16 -04001170 if (inode) {
Steve Frenchff694522009-04-20 19:45:13 +00001171 cifs_inode = CIFS_I(inode);
1172 cifs_inode->time = 0; /* will force revalidate to get info
Steve Frenchb2aeb9d2005-05-17 13:16:18 -05001173 when needed */
Jeff Layton5f0319a2008-09-16 14:05:16 -04001174 inode->i_ctime = current_fs_time(sb);
Steve Frenchb2aeb9d2005-05-17 13:16:18 -05001175 }
Jeff Layton5f0319a2008-09-16 14:05:16 -04001176 dir->i_ctime = dir->i_mtime = current_fs_time(sb);
Steve Frenchff694522009-04-20 19:45:13 +00001177 cifs_inode = CIFS_I(dir);
Steve French60502472008-10-07 18:42:52 +00001178 CIFS_I(dir)->time = 0; /* force revalidate of dir as well */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001179
1180 kfree(full_path);
Steve French60502472008-10-07 18:42:52 +00001181 kfree(attrs);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001182 FreeXid(xid);
1183 return rc;
1184}
1185
1186int cifs_mkdir(struct inode *inode, struct dentry *direntry, int mode)
1187{
Jeff Layton6b37faa2008-10-06 21:54:41 +00001188 int rc = 0, tmprc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001189 int xid;
1190 struct cifs_sb_info *cifs_sb;
1191 struct cifsTconInfo *pTcon;
1192 char *full_path = NULL;
1193 struct inode *newinode = NULL;
Jeff Laytoncc0bad72009-06-25 00:56:52 -04001194 struct cifs_fattr fattr;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001195
Joe Perchesb6b38f72010-04-21 03:50:45 +00001196 cFYI(1, "In cifs_mkdir, mode = 0x%x inode = 0x%p", mode, inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001197
1198 xid = GetXid();
1199
1200 cifs_sb = CIFS_SB(inode->i_sb);
1201 pTcon = cifs_sb->tcon;
1202
Steve French7f573562005-08-30 11:32:14 -07001203 full_path = build_path_from_dentry(direntry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001204 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301205 rc = -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001206 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301207 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001208 }
Steve French50c2f752007-07-13 00:33:32 +00001209
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001210 if ((pTcon->ses->capabilities & CAP_UNIX) &&
1211 (CIFS_UNIX_POSIX_PATH_OPS_CAP &
Steve French2dd29d32007-04-23 22:07:35 +00001212 le64_to_cpu(pTcon->fsUnixInfo.Capability))) {
1213 u32 oplock = 0;
Steve Frenchf6d09982008-01-08 23:18:22 +00001214 FILE_UNIX_BASIC_INFO *pInfo =
Steve French2dd29d32007-04-23 22:07:35 +00001215 kzalloc(sizeof(FILE_UNIX_BASIC_INFO), GFP_KERNEL);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001216 if (pInfo == NULL) {
Steve French2dd29d32007-04-23 22:07:35 +00001217 rc = -ENOMEM;
1218 goto mkdir_out;
1219 }
Steve French50c2f752007-07-13 00:33:32 +00001220
Al Viroce3b0f82009-03-29 19:08:22 -04001221 mode &= ~current_umask();
Steve French2dd29d32007-04-23 22:07:35 +00001222 rc = CIFSPOSIXCreate(xid, pTcon, SMB_O_DIRECTORY | SMB_O_CREAT,
1223 mode, NULL /* netfid */, pInfo, &oplock,
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001224 full_path, cifs_sb->local_nls,
1225 cifs_sb->mnt_cifs_flags &
Steve French2dd29d32007-04-23 22:07:35 +00001226 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve Frenchc45d7072007-09-17 02:04:21 +00001227 if (rc == -EOPNOTSUPP) {
1228 kfree(pInfo);
1229 goto mkdir_retry_old;
1230 } else if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001231 cFYI(1, "posix mkdir returned 0x%x", rc);
Steve French2dd29d32007-04-23 22:07:35 +00001232 d_drop(direntry);
1233 } else {
Cyril Gorcunov8f2376a2007-10-14 17:58:43 +00001234 if (pInfo->Type == cpu_to_le32(-1)) {
1235 /* no return info, go query for it */
Steve French5a07cdf2007-09-16 23:12:47 +00001236 kfree(pInfo);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001237 goto mkdir_get_info;
Steve French5a07cdf2007-09-16 23:12:47 +00001238 }
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001239/*BB check (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID ) to see if need
1240 to set uid/gid */
Steve French2dd29d32007-04-23 22:07:35 +00001241 inc_nlink(inode);
1242 if (pTcon->nocase)
1243 direntry->d_op = &cifs_ci_dentry_ops;
1244 else
1245 direntry->d_op = &cifs_dentry_ops;
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001246
Jeff Laytoncc0bad72009-06-25 00:56:52 -04001247 cifs_unix_basic_to_fattr(&fattr, pInfo, cifs_sb);
Jeff Layton4065c802010-05-17 07:18:58 -04001248 cifs_fill_uniqueid(inode->i_sb, &fattr);
Jeff Laytoncc0bad72009-06-25 00:56:52 -04001249 newinode = cifs_iget(inode->i_sb, &fattr);
1250 if (!newinode) {
Steve French5a07cdf2007-09-16 23:12:47 +00001251 kfree(pInfo);
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001252 goto mkdir_get_info;
Steve French5a07cdf2007-09-16 23:12:47 +00001253 }
Jeff Layton6b37faa2008-10-06 21:54:41 +00001254
Steve French2dd29d32007-04-23 22:07:35 +00001255 d_instantiate(direntry, newinode);
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001256
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001257#ifdef CONFIG_CIFS_DEBUG2
Joe Perchesb6b38f72010-04-21 03:50:45 +00001258 cFYI(1, "instantiated dentry %p %s to inode %p",
1259 direntry, direntry->d_name.name, newinode);
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001260
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001261 if (newinode->i_nlink != 2)
Joe Perchesb6b38f72010-04-21 03:50:45 +00001262 cFYI(1, "unexpected number of links %d",
1263 newinode->i_nlink);
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001264#endif
Steve French2dd29d32007-04-23 22:07:35 +00001265 }
1266 kfree(pInfo);
1267 goto mkdir_out;
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001268 }
Steve Frenchc45d7072007-09-17 02:04:21 +00001269mkdir_retry_old:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001270 /* BB add setting the equivalent of mode via CreateX w/ACLs */
Steve French737b7582005-04-28 22:41:06 -07001271 rc = CIFSSMBMkDir(xid, pTcon, full_path, cifs_sb->local_nls,
1272 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001273 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001274 cFYI(1, "cifs_mkdir returned 0x%x", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001275 d_drop(direntry);
1276 } else {
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001277mkdir_get_info:
Dave Hansend8c76e62006-09-30 23:29:04 -07001278 inc_nlink(inode);
Steve Frenchc18c8422007-07-18 23:21:09 +00001279 if (pTcon->unix_ext)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001280 rc = cifs_get_inode_info_unix(&newinode, full_path,
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001281 inode->i_sb, xid);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001282 else
1283 rc = cifs_get_inode_info(&newinode, full_path, NULL,
Steve French8b1327f2008-03-14 22:37:16 +00001284 inode->i_sb, xid, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001285
Steve Frenchb92327f2005-08-22 20:09:43 -07001286 if (pTcon->nocase)
1287 direntry->d_op = &cifs_ci_dentry_ops;
1288 else
1289 direntry->d_op = &cifs_dentry_ops;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001290 d_instantiate(direntry, newinode);
Steve French2dd29d32007-04-23 22:07:35 +00001291 /* setting nlink not necessary except in cases where we
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001292 * failed to get it from the server or was set bogus */
Steve French2dd29d32007-04-23 22:07:35 +00001293 if ((direntry->d_inode) && (direntry->d_inode->i_nlink < 2))
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001294 direntry->d_inode->i_nlink = 2;
Jeff Layton95089912008-08-06 04:39:02 +00001295
Al Viroce3b0f82009-03-29 19:08:22 -04001296 mode &= ~current_umask();
Jeff Layton95089912008-08-06 04:39:02 +00001297 /* must turn on setgid bit if parent dir has it */
1298 if (inode->i_mode & S_ISGID)
1299 mode |= S_ISGID;
1300
Steve Frenchc18c8422007-07-18 23:21:09 +00001301 if (pTcon->unix_ext) {
Jeff Layton4e1e7fb2008-08-02 07:26:12 -04001302 struct cifs_unix_set_info_args args = {
1303 .mode = mode,
1304 .ctime = NO_CHANGE_64,
1305 .atime = NO_CHANGE_64,
1306 .mtime = NO_CHANGE_64,
1307 .device = 0,
1308 };
Steve Frenchd0d2f2d2005-06-02 15:12:36 -07001309 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID) {
David Howellsa001e5b2008-11-14 10:38:47 +11001310 args.uid = (__u64)current_fsuid();
Jeff Layton95089912008-08-06 04:39:02 +00001311 if (inode->i_mode & S_ISGID)
1312 args.gid = (__u64)inode->i_gid;
1313 else
David Howellsa001e5b2008-11-14 10:38:47 +11001314 args.gid = (__u64)current_fsgid();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001315 } else {
Jeff Layton4e1e7fb2008-08-02 07:26:12 -04001316 args.uid = NO_CHANGE_64;
1317 args.gid = NO_CHANGE_64;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001318 }
Jeff Layton01ea95e2009-07-09 20:02:49 -04001319 CIFSSMBUnixSetPathInfo(xid, pTcon, full_path, &args,
1320 cifs_sb->local_nls,
1321 cifs_sb->mnt_cifs_flags &
1322 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French3ce53fc2007-06-08 14:55:14 +00001323 } else {
Jeff Layton67750fb2008-05-09 22:28:02 +00001324 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL) &&
1325 (mode & S_IWUGO) == 0) {
1326 FILE_BASIC_INFO pInfo;
Jeff Layton6b37faa2008-10-06 21:54:41 +00001327 struct cifsInodeInfo *cifsInode;
1328 u32 dosattrs;
1329
Jeff Layton67750fb2008-05-09 22:28:02 +00001330 memset(&pInfo, 0, sizeof(pInfo));
Jeff Layton6b37faa2008-10-06 21:54:41 +00001331 cifsInode = CIFS_I(newinode);
1332 dosattrs = cifsInode->cifsAttrs|ATTR_READONLY;
1333 pInfo.Attributes = cpu_to_le32(dosattrs);
1334 tmprc = CIFSSMBSetPathInfo(xid, pTcon,
1335 full_path, &pInfo,
1336 cifs_sb->local_nls,
Jeff Layton67750fb2008-05-09 22:28:02 +00001337 cifs_sb->mnt_cifs_flags &
1338 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Layton6b37faa2008-10-06 21:54:41 +00001339 if (tmprc == 0)
1340 cifsInode->cifsAttrs = dosattrs;
Jeff Layton67750fb2008-05-09 22:28:02 +00001341 }
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001342 if (direntry->d_inode) {
Jeff Laytonb0fd30d2008-05-22 09:33:34 -04001343 if (cifs_sb->mnt_cifs_flags &
1344 CIFS_MOUNT_DYNPERM)
1345 direntry->d_inode->i_mode =
1346 (mode | S_IFDIR);
Steve French4e94a102008-05-23 18:22:46 +00001347
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001348 if (cifs_sb->mnt_cifs_flags &
Steve French6473a552005-11-29 20:20:10 -08001349 CIFS_MOUNT_SET_UID) {
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001350 direntry->d_inode->i_uid =
David Howellsa001e5b2008-11-14 10:38:47 +11001351 current_fsuid();
Jeff Layton95089912008-08-06 04:39:02 +00001352 if (inode->i_mode & S_ISGID)
1353 direntry->d_inode->i_gid =
1354 inode->i_gid;
1355 else
1356 direntry->d_inode->i_gid =
David Howellsa001e5b2008-11-14 10:38:47 +11001357 current_fsgid();
Steve French6473a552005-11-29 20:20:10 -08001358 }
1359 }
Steve French2a138ebb2005-11-29 21:22:19 -08001360 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001361 }
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001362mkdir_out:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001363 kfree(full_path);
1364 FreeXid(xid);
1365 return rc;
1366}
1367
1368int cifs_rmdir(struct inode *inode, struct dentry *direntry)
1369{
1370 int rc = 0;
1371 int xid;
1372 struct cifs_sb_info *cifs_sb;
1373 struct cifsTconInfo *pTcon;
1374 char *full_path = NULL;
1375 struct cifsInodeInfo *cifsInode;
1376
Joe Perchesb6b38f72010-04-21 03:50:45 +00001377 cFYI(1, "cifs_rmdir, inode = 0x%p", inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001378
1379 xid = GetXid();
1380
1381 cifs_sb = CIFS_SB(inode->i_sb);
1382 pTcon = cifs_sb->tcon;
1383
Steve French7f573562005-08-30 11:32:14 -07001384 full_path = build_path_from_dentry(direntry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001385 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301386 rc = -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001387 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301388 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001389 }
1390
Steve French737b7582005-04-28 22:41:06 -07001391 rc = CIFSSMBRmDir(xid, pTcon, full_path, cifs_sb->local_nls,
1392 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001393
1394 if (!rc) {
Dave Hansen9a53c3a2006-09-30 23:29:03 -07001395 drop_nlink(inode);
Steve French3677db12007-02-26 16:46:11 +00001396 spin_lock(&direntry->d_inode->i_lock);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001397 i_size_write(direntry->d_inode, 0);
Dave Hansence71ec32006-09-30 23:29:06 -07001398 clear_nlink(direntry->d_inode);
Steve French3677db12007-02-26 16:46:11 +00001399 spin_unlock(&direntry->d_inode->i_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001400 }
1401
1402 cifsInode = CIFS_I(direntry->d_inode);
1403 cifsInode->time = 0; /* force revalidate to go get info when
1404 needed */
Steve French42c24542009-01-13 22:03:55 +00001405
1406 cifsInode = CIFS_I(inode);
1407 cifsInode->time = 0; /* force revalidate to get parent dir info
1408 since cached search results now invalid */
1409
Linus Torvalds1da177e2005-04-16 15:20:36 -07001410 direntry->d_inode->i_ctime = inode->i_ctime = inode->i_mtime =
1411 current_fs_time(inode->i_sb);
1412
1413 kfree(full_path);
1414 FreeXid(xid);
1415 return rc;
1416}
1417
Steve Frenchee2fd962008-09-23 18:23:33 +00001418static int
1419cifs_do_rename(int xid, struct dentry *from_dentry, const char *fromPath,
1420 struct dentry *to_dentry, const char *toPath)
1421{
1422 struct cifs_sb_info *cifs_sb = CIFS_SB(from_dentry->d_sb);
1423 struct cifsTconInfo *pTcon = cifs_sb->tcon;
1424 __u16 srcfid;
1425 int oplock, rc;
1426
1427 /* try path-based rename first */
1428 rc = CIFSSMBRename(xid, pTcon, fromPath, toPath, cifs_sb->local_nls,
1429 cifs_sb->mnt_cifs_flags &
1430 CIFS_MOUNT_MAP_SPECIAL_CHR);
1431
1432 /*
1433 * don't bother with rename by filehandle unless file is busy and
1434 * source Note that cross directory moves do not work with
1435 * rename by filehandle to various Windows servers.
1436 */
1437 if (rc == 0 || rc != -ETXTBSY)
1438 return rc;
1439
Jeff Laytoned0e3ac2010-06-01 16:21:01 -04001440 /* open-file renames don't work across directories */
1441 if (to_dentry->d_parent != from_dentry->d_parent)
1442 return rc;
1443
Steve Frenchee2fd962008-09-23 18:23:33 +00001444 /* open the file to be renamed -- we need DELETE perms */
1445 rc = CIFSSMBOpen(xid, pTcon, fromPath, FILE_OPEN, DELETE,
1446 CREATE_NOT_DIR, &srcfid, &oplock, NULL,
1447 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1448 CIFS_MOUNT_MAP_SPECIAL_CHR);
1449
1450 if (rc == 0) {
1451 rc = CIFSSMBRenameOpenFile(xid, pTcon, srcfid,
1452 (const char *) to_dentry->d_name.name,
1453 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1454 CIFS_MOUNT_MAP_SPECIAL_CHR);
1455
1456 CIFSSMBClose(xid, pTcon, srcfid);
1457 }
1458
1459 return rc;
1460}
1461
Jeff Layton14121bd2008-10-20 14:45:22 -04001462int cifs_rename(struct inode *source_dir, struct dentry *source_dentry,
1463 struct inode *target_dir, struct dentry *target_dentry)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001464{
Steve Frenchee2fd962008-09-23 18:23:33 +00001465 char *fromName = NULL;
1466 char *toName = NULL;
Jeff Layton639e7a92010-09-03 11:50:09 -04001467 struct cifs_sb_info *cifs_sb;
Jeff Layton14121bd2008-10-20 14:45:22 -04001468 struct cifsTconInfo *tcon;
Steve Frenchee2fd962008-09-23 18:23:33 +00001469 FILE_UNIX_BASIC_INFO *info_buf_source = NULL;
1470 FILE_UNIX_BASIC_INFO *info_buf_target;
Jeff Layton8d281ef2008-10-22 13:57:01 -04001471 int xid, rc, tmprc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001472
Jeff Layton639e7a92010-09-03 11:50:09 -04001473 cifs_sb = CIFS_SB(source_dir->i_sb);
1474 tcon = cifs_sb->tcon;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001475
Steve Frenchee2fd962008-09-23 18:23:33 +00001476 xid = GetXid();
1477
1478 /*
Steve Frenchee2fd962008-09-23 18:23:33 +00001479 * we already have the rename sem so we do not need to
1480 * grab it again here to protect the path integrity
1481 */
Jeff Layton14121bd2008-10-20 14:45:22 -04001482 fromName = build_path_from_dentry(source_dentry);
Steve Frenchee2fd962008-09-23 18:23:33 +00001483 if (fromName == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001484 rc = -ENOMEM;
1485 goto cifs_rename_exit;
1486 }
1487
Jeff Layton14121bd2008-10-20 14:45:22 -04001488 toName = build_path_from_dentry(target_dentry);
Steve Frenchee2fd962008-09-23 18:23:33 +00001489 if (toName == NULL) {
1490 rc = -ENOMEM;
1491 goto cifs_rename_exit;
1492 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001493
Jeff Layton14121bd2008-10-20 14:45:22 -04001494 rc = cifs_do_rename(xid, source_dentry, fromName,
1495 target_dentry, toName);
Steve Frenchee2fd962008-09-23 18:23:33 +00001496
Jeff Layton14121bd2008-10-20 14:45:22 -04001497 if (rc == -EEXIST && tcon->unix_ext) {
Steve Frenchee2fd962008-09-23 18:23:33 +00001498 /*
Jeff Layton14121bd2008-10-20 14:45:22 -04001499 * Are src and dst hardlinks of same inode? We can
1500 * only tell with unix extensions enabled
Steve Frenchee2fd962008-09-23 18:23:33 +00001501 */
Jeff Layton14121bd2008-10-20 14:45:22 -04001502 info_buf_source =
1503 kmalloc(2 * sizeof(FILE_UNIX_BASIC_INFO),
1504 GFP_KERNEL);
1505 if (info_buf_source == NULL) {
1506 rc = -ENOMEM;
1507 goto cifs_rename_exit;
1508 }
1509
1510 info_buf_target = info_buf_source + 1;
Jeff Layton8d281ef2008-10-22 13:57:01 -04001511 tmprc = CIFSSMBUnixQPathInfo(xid, tcon, fromName,
Jeff Layton14121bd2008-10-20 14:45:22 -04001512 info_buf_source,
Jeff Layton639e7a92010-09-03 11:50:09 -04001513 cifs_sb->local_nls,
1514 cifs_sb->mnt_cifs_flags &
Jeff Layton14121bd2008-10-20 14:45:22 -04001515 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Layton8d281ef2008-10-22 13:57:01 -04001516 if (tmprc != 0)
Jeff Layton14121bd2008-10-20 14:45:22 -04001517 goto unlink_target;
1518
Jeff Layton639e7a92010-09-03 11:50:09 -04001519 tmprc = CIFSSMBUnixQPathInfo(xid, tcon, toName,
1520 info_buf_target,
1521 cifs_sb->local_nls,
1522 cifs_sb->mnt_cifs_flags &
Jeff Layton14121bd2008-10-20 14:45:22 -04001523 CIFS_MOUNT_MAP_SPECIAL_CHR);
1524
Jeff Layton8d281ef2008-10-22 13:57:01 -04001525 if (tmprc == 0 && (info_buf_source->UniqueId ==
Jeff Laytonae6884a2008-11-03 14:05:08 -05001526 info_buf_target->UniqueId)) {
Jeff Layton14121bd2008-10-20 14:45:22 -04001527 /* same file, POSIX says that this is a noop */
Jeff Laytonae6884a2008-11-03 14:05:08 -05001528 rc = 0;
Jeff Layton14121bd2008-10-20 14:45:22 -04001529 goto cifs_rename_exit;
Jeff Laytonae6884a2008-11-03 14:05:08 -05001530 }
Jeff Layton14121bd2008-10-20 14:45:22 -04001531 } /* else ... BB we could add the same check for Windows by
1532 checking the UniqueId via FILE_INTERNAL_INFO */
1533
Jeff Layton14121bd2008-10-20 14:45:22 -04001534unlink_target:
Jeff Laytonfc6f3942009-04-17 11:45:30 -04001535 /* Try unlinking the target dentry if it's not negative */
1536 if (target_dentry->d_inode && (rc == -EACCES || rc == -EEXIST)) {
Jeff Layton8d281ef2008-10-22 13:57:01 -04001537 tmprc = cifs_unlink(target_dir, target_dentry);
Jeff Layton14121bd2008-10-20 14:45:22 -04001538 if (tmprc)
1539 goto cifs_rename_exit;
1540
Jeff Layton14121bd2008-10-20 14:45:22 -04001541 rc = cifs_do_rename(xid, source_dentry, fromName,
1542 target_dentry, toName);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001543 }
1544
1545cifs_rename_exit:
Steve Frenchee2fd962008-09-23 18:23:33 +00001546 kfree(info_buf_source);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001547 kfree(fromName);
1548 kfree(toName);
1549 FreeXid(xid);
1550 return rc;
1551}
1552
Jeff Laytondf2cf172010-02-12 07:44:16 -05001553static bool
1554cifs_inode_needs_reval(struct inode *inode)
1555{
1556 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
1557
1558 if (cifs_i->clientCanCacheRead)
1559 return false;
1560
1561 if (!lookupCacheEnabled)
1562 return true;
1563
1564 if (cifs_i->time == 0)
1565 return true;
1566
1567 /* FIXME: the actimeo should be tunable */
1568 if (time_after_eq(jiffies, cifs_i->time + HZ))
1569 return true;
1570
Jeff Laytondb192722010-05-17 14:51:49 -04001571 /* hardlinked files w/ noserverino get "special" treatment */
1572 if (!(CIFS_SB(inode->i_sb)->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM) &&
1573 S_ISREG(inode->i_mode) && inode->i_nlink != 1)
1574 return true;
1575
Jeff Laytondf2cf172010-02-12 07:44:16 -05001576 return false;
1577}
1578
1579/* check invalid_mapping flag and zap the cache if it's set */
1580static void
1581cifs_invalidate_mapping(struct inode *inode)
1582{
1583 int rc;
1584 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
1585
1586 cifs_i->invalid_mapping = false;
1587
1588 /* write back any cached data */
1589 if (inode->i_mapping && inode->i_mapping->nrpages != 0) {
1590 rc = filemap_write_and_wait(inode->i_mapping);
1591 if (rc)
1592 cifs_i->write_behind_rc = rc;
1593 }
1594 invalidate_remote_inode(inode);
Suresh Jayaraman9451a9a2010-07-05 18:12:45 +05301595 cifs_fscache_reset_inode_cookie(inode);
Jeff Laytondf2cf172010-02-12 07:44:16 -05001596}
1597
Jeff Laytonabab0952010-02-12 07:44:18 -05001598int cifs_revalidate_file(struct file *filp)
1599{
1600 int rc = 0;
1601 struct inode *inode = filp->f_path.dentry->d_inode;
1602
1603 if (!cifs_inode_needs_reval(inode))
1604 goto check_inval;
1605
1606 if (CIFS_SB(inode->i_sb)->tcon->unix_ext)
1607 rc = cifs_get_file_info_unix(filp);
1608 else
1609 rc = cifs_get_file_info(filp);
1610
1611check_inval:
1612 if (CIFS_I(inode)->invalid_mapping)
1613 cifs_invalidate_mapping(inode);
1614
1615 return rc;
1616}
1617
Jeff Laytondf2cf172010-02-12 07:44:16 -05001618/* revalidate a dentry's inode attributes */
1619int cifs_revalidate_dentry(struct dentry *dentry)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001620{
1621 int xid;
Jeff Laytondf2cf172010-02-12 07:44:16 -05001622 int rc = 0;
1623 char *full_path = NULL;
1624 struct inode *inode = dentry->d_inode;
1625 struct super_block *sb = dentry->d_sb;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001626
Jeff Laytondf2cf172010-02-12 07:44:16 -05001627 if (inode == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001628 return -ENOENT;
1629
Linus Torvalds1da177e2005-04-16 15:20:36 -07001630 xid = GetXid();
1631
Jeff Laytondf2cf172010-02-12 07:44:16 -05001632 if (!cifs_inode_needs_reval(inode))
1633 goto check_inval;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001634
1635 /* can not safely grab the rename sem here if rename calls revalidate
1636 since that would deadlock */
Jeff Laytondf2cf172010-02-12 07:44:16 -05001637 full_path = build_path_from_dentry(dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001638 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301639 rc = -ENOMEM;
Jeff Laytondf2cf172010-02-12 07:44:16 -05001640 goto check_inval;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001641 }
Jeff Laytondf2cf172010-02-12 07:44:16 -05001642
Steve Frenchf19159d2010-04-21 04:12:10 +00001643 cFYI(1, "Revalidate: %s inode 0x%p count %d dentry: 0x%p d_time %ld "
Jeff Laytondf2cf172010-02-12 07:44:16 -05001644 "jiffies %ld", full_path, inode, inode->i_count.counter,
Steve Frenchf19159d2010-04-21 04:12:10 +00001645 dentry, dentry->d_time, jiffies);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001646
Jeff Laytondf2cf172010-02-12 07:44:16 -05001647 if (CIFS_SB(sb)->tcon->unix_ext)
1648 rc = cifs_get_inode_info_unix(&inode, full_path, sb, xid);
1649 else
1650 rc = cifs_get_inode_info(&inode, full_path, NULL, sb,
1651 xid, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001652
Jeff Laytondf2cf172010-02-12 07:44:16 -05001653check_inval:
1654 if (CIFS_I(inode)->invalid_mapping)
1655 cifs_invalidate_mapping(inode);
Steve French50c2f752007-07-13 00:33:32 +00001656
Linus Torvalds1da177e2005-04-16 15:20:36 -07001657 kfree(full_path);
1658 FreeXid(xid);
1659 return rc;
1660}
1661
1662int cifs_getattr(struct vfsmount *mnt, struct dentry *dentry,
1663 struct kstat *stat)
1664{
Jeff Laytondf2cf172010-02-12 07:44:16 -05001665 int err = cifs_revalidate_dentry(dentry);
Steve French5fe14c82006-11-07 19:26:33 +00001666 if (!err) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001667 generic_fillattr(dentry->d_inode, stat);
Steve French5fe14c82006-11-07 19:26:33 +00001668 stat->blksize = CIFS_MAX_MSGSIZE;
Jeff Laytoncc0bad72009-06-25 00:56:52 -04001669 stat->ino = CIFS_I(dentry->d_inode)->uniqueid;
Steve French5fe14c82006-11-07 19:26:33 +00001670 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001671 return err;
1672}
1673
1674static int cifs_truncate_page(struct address_space *mapping, loff_t from)
1675{
1676 pgoff_t index = from >> PAGE_CACHE_SHIFT;
1677 unsigned offset = from & (PAGE_CACHE_SIZE - 1);
1678 struct page *page;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001679 int rc = 0;
1680
1681 page = grab_cache_page(mapping, index);
1682 if (!page)
1683 return -ENOMEM;
1684
Christoph Lametereebd2aa2008-02-04 22:28:29 -08001685 zero_user_segment(page, offset, PAGE_CACHE_SIZE);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001686 unlock_page(page);
1687 page_cache_release(page);
1688 return rc;
1689}
1690
Christoph Hellwig1b947462010-07-18 17:51:21 -04001691static void cifs_setsize(struct inode *inode, loff_t offset)
Steve French3677db12007-02-26 16:46:11 +00001692{
npiggin@suse.dec08d3b02009-08-21 02:35:06 +10001693 loff_t oldsize;
Steve French3677db12007-02-26 16:46:11 +00001694
Steve Frenchba6a46a2007-02-26 20:06:29 +00001695 spin_lock(&inode->i_lock);
npiggin@suse.dec08d3b02009-08-21 02:35:06 +10001696 oldsize = inode->i_size;
Steve French3677db12007-02-26 16:46:11 +00001697 i_size_write(inode, offset);
Steve Frenchba6a46a2007-02-26 20:06:29 +00001698 spin_unlock(&inode->i_lock);
Christoph Hellwig1b947462010-07-18 17:51:21 -04001699
npiggin@suse.dec08d3b02009-08-21 02:35:06 +10001700 truncate_pagecache(inode, oldsize, offset);
Steve French3677db12007-02-26 16:46:11 +00001701}
1702
Jeff Layton8efdbde2008-07-23 21:28:12 +00001703static int
1704cifs_set_file_size(struct inode *inode, struct iattr *attrs,
1705 int xid, char *full_path)
1706{
1707 int rc;
1708 struct cifsFileInfo *open_file;
1709 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
1710 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
1711 struct cifsTconInfo *pTcon = cifs_sb->tcon;
1712
1713 /*
1714 * To avoid spurious oplock breaks from server, in the case of
1715 * inodes that we already have open, avoid doing path based
1716 * setting of file size if we can do it by handle.
1717 * This keeps our caching token (oplock) and avoids timeouts
1718 * when the local oplock break takes longer to flush
1719 * writebehind data than the SMB timeout for the SetPathInfo
1720 * request would allow
1721 */
1722 open_file = find_writable_file(cifsInode);
1723 if (open_file) {
1724 __u16 nfid = open_file->netfid;
1725 __u32 npid = open_file->pid;
1726 rc = CIFSSMBSetFileSize(xid, pTcon, attrs->ia_size, nfid,
1727 npid, false);
Dave Kleikamp6ab409b2009-08-31 11:07:12 -04001728 cifsFileInfo_put(open_file);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001729 cFYI(1, "SetFSize for attrs rc = %d", rc);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001730 if ((rc == -EINVAL) || (rc == -EOPNOTSUPP)) {
1731 unsigned int bytes_written;
1732 rc = CIFSSMBWrite(xid, pTcon, nfid, 0, attrs->ia_size,
1733 &bytes_written, NULL, NULL, 1);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001734 cFYI(1, "Wrt seteof rc %d", rc);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001735 }
1736 } else
1737 rc = -EINVAL;
1738
1739 if (rc != 0) {
1740 /* Set file size by pathname rather than by handle
1741 either because no valid, writeable file handle for
1742 it was found or because there was an error setting
1743 it by handle */
1744 rc = CIFSSMBSetEOF(xid, pTcon, full_path, attrs->ia_size,
1745 false, cifs_sb->local_nls,
1746 cifs_sb->mnt_cifs_flags &
1747 CIFS_MOUNT_MAP_SPECIAL_CHR);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001748 cFYI(1, "SetEOF by path (setattrs) rc = %d", rc);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001749 if ((rc == -EINVAL) || (rc == -EOPNOTSUPP)) {
1750 __u16 netfid;
1751 int oplock = 0;
1752
1753 rc = SMBLegacyOpen(xid, pTcon, full_path,
1754 FILE_OPEN, GENERIC_WRITE,
1755 CREATE_NOT_DIR, &netfid, &oplock, NULL,
1756 cifs_sb->local_nls,
1757 cifs_sb->mnt_cifs_flags &
1758 CIFS_MOUNT_MAP_SPECIAL_CHR);
1759 if (rc == 0) {
1760 unsigned int bytes_written;
1761 rc = CIFSSMBWrite(xid, pTcon, netfid, 0,
1762 attrs->ia_size,
1763 &bytes_written, NULL,
1764 NULL, 1);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001765 cFYI(1, "wrt seteof rc %d", rc);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001766 CIFSSMBClose(xid, pTcon, netfid);
1767 }
1768 }
1769 }
1770
1771 if (rc == 0) {
Jeff Laytonfbec9ab2009-04-03 13:44:00 -04001772 cifsInode->server_eof = attrs->ia_size;
Christoph Hellwig1b947462010-07-18 17:51:21 -04001773 cifs_setsize(inode, attrs->ia_size);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001774 cifs_truncate_page(inode->i_mapping, inode->i_size);
1775 }
1776
1777 return rc;
1778}
1779
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001780static int
1781cifs_setattr_unix(struct dentry *direntry, struct iattr *attrs)
1782{
1783 int rc;
1784 int xid;
1785 char *full_path = NULL;
1786 struct inode *inode = direntry->d_inode;
1787 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
1788 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
1789 struct cifsTconInfo *pTcon = cifs_sb->tcon;
1790 struct cifs_unix_set_info_args *args = NULL;
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001791 struct cifsFileInfo *open_file;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001792
Joe Perchesb6b38f72010-04-21 03:50:45 +00001793 cFYI(1, "setattr_unix on file %s attrs->ia_valid=0x%x",
1794 direntry->d_name.name, attrs->ia_valid);
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001795
1796 xid = GetXid();
1797
Christoph Hellwigdb78b872010-06-04 11:30:03 +02001798 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_PERM)
1799 attrs->ia_valid |= ATTR_FORCE;
1800
1801 rc = inode_change_ok(inode, attrs);
1802 if (rc < 0)
1803 goto out;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001804
1805 full_path = build_path_from_dentry(direntry);
1806 if (full_path == NULL) {
1807 rc = -ENOMEM;
1808 goto out;
1809 }
1810
Jeff Layton0f4d6342009-03-26 13:35:37 -04001811 /*
1812 * Attempt to flush data before changing attributes. We need to do
1813 * this for ATTR_SIZE and ATTR_MTIME for sure, and if we change the
1814 * ownership or mode then we may also need to do this. Here, we take
1815 * the safe way out and just do the flush on all setattr requests. If
1816 * the flush returns error, store it to report later and continue.
1817 *
1818 * BB: This should be smarter. Why bother flushing pages that
1819 * will be truncated anyway? Also, should we error out here if
1820 * the flush returns error?
1821 */
1822 rc = filemap_write_and_wait(inode->i_mapping);
1823 if (rc != 0) {
1824 cifsInode->write_behind_rc = rc;
1825 rc = 0;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001826 }
1827
1828 if (attrs->ia_valid & ATTR_SIZE) {
1829 rc = cifs_set_file_size(inode, attrs, xid, full_path);
1830 if (rc != 0)
1831 goto out;
1832 }
1833
1834 /* skip mode change if it's just for clearing setuid/setgid */
1835 if (attrs->ia_valid & (ATTR_KILL_SUID|ATTR_KILL_SGID))
1836 attrs->ia_valid &= ~ATTR_MODE;
1837
1838 args = kmalloc(sizeof(*args), GFP_KERNEL);
1839 if (args == NULL) {
1840 rc = -ENOMEM;
1841 goto out;
1842 }
1843
1844 /* set up the struct */
1845 if (attrs->ia_valid & ATTR_MODE)
1846 args->mode = attrs->ia_mode;
1847 else
1848 args->mode = NO_CHANGE_64;
1849
1850 if (attrs->ia_valid & ATTR_UID)
1851 args->uid = attrs->ia_uid;
1852 else
1853 args->uid = NO_CHANGE_64;
1854
1855 if (attrs->ia_valid & ATTR_GID)
1856 args->gid = attrs->ia_gid;
1857 else
1858 args->gid = NO_CHANGE_64;
1859
1860 if (attrs->ia_valid & ATTR_ATIME)
1861 args->atime = cifs_UnixTimeToNT(attrs->ia_atime);
1862 else
1863 args->atime = NO_CHANGE_64;
1864
1865 if (attrs->ia_valid & ATTR_MTIME)
1866 args->mtime = cifs_UnixTimeToNT(attrs->ia_mtime);
1867 else
1868 args->mtime = NO_CHANGE_64;
1869
1870 if (attrs->ia_valid & ATTR_CTIME)
1871 args->ctime = cifs_UnixTimeToNT(attrs->ia_ctime);
1872 else
1873 args->ctime = NO_CHANGE_64;
1874
1875 args->device = 0;
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001876 open_file = find_writable_file(cifsInode);
1877 if (open_file) {
1878 u16 nfid = open_file->netfid;
1879 u32 npid = open_file->pid;
1880 rc = CIFSSMBUnixSetFileInfo(xid, pTcon, args, nfid, npid);
Dave Kleikamp6ab409b2009-08-31 11:07:12 -04001881 cifsFileInfo_put(open_file);
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001882 } else {
1883 rc = CIFSSMBUnixSetPathInfo(xid, pTcon, full_path, args,
Jeff Layton01ea95e2009-07-09 20:02:49 -04001884 cifs_sb->local_nls,
1885 cifs_sb->mnt_cifs_flags &
1886 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001887 }
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001888
Christoph Hellwig10257742010-06-04 11:30:02 +02001889 if (rc)
1890 goto out;
Steve Frenchccd4bb12010-02-08 17:39:58 +00001891
Christoph Hellwig10257742010-06-04 11:30:02 +02001892 if ((attrs->ia_valid & ATTR_SIZE) &&
Christoph Hellwig1b947462010-07-18 17:51:21 -04001893 attrs->ia_size != i_size_read(inode))
1894 truncate_setsize(inode, attrs->ia_size);
Christoph Hellwig10257742010-06-04 11:30:02 +02001895
1896 setattr_copy(inode, attrs);
1897 mark_inode_dirty(inode);
1898
1899 /* force revalidate when any of these times are set since some
1900 of the fs types (eg ext3, fat) do not have fine enough
1901 time granularity to match protocol, and we do not have a
1902 a way (yet) to query the server fs's time granularity (and
1903 whether it rounds times down).
1904 */
1905 if (attrs->ia_valid & (ATTR_MTIME | ATTR_CTIME))
1906 cifsInode->time = 0;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001907out:
1908 kfree(args);
1909 kfree(full_path);
1910 FreeXid(xid);
1911 return rc;
1912}
1913
Jeff Layton0510eeb2008-08-02 07:26:12 -04001914static int
1915cifs_setattr_nounix(struct dentry *direntry, struct iattr *attrs)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001916{
1917 int xid;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001918 struct inode *inode = direntry->d_inode;
1919 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001920 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001921 char *full_path = NULL;
1922 int rc = -EACCES;
Jeff Laytonfeb3e202008-08-02 07:26:12 -04001923 __u32 dosattr = 0;
Jeff Layton4e1e7fb2008-08-02 07:26:12 -04001924 __u64 mode = NO_CHANGE_64;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001925
Linus Torvalds1da177e2005-04-16 15:20:36 -07001926 xid = GetXid();
1927
Joe Perchesb6b38f72010-04-21 03:50:45 +00001928 cFYI(1, "setattr on file %s attrs->iavalid 0x%x",
1929 direntry->d_name.name, attrs->ia_valid);
Steve French6473a552005-11-29 20:20:10 -08001930
Christoph Hellwigdb78b872010-06-04 11:30:03 +02001931 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_PERM)
1932 attrs->ia_valid |= ATTR_FORCE;
1933
1934 rc = inode_change_ok(inode, attrs);
1935 if (rc < 0) {
1936 FreeXid(xid);
1937 return rc;
Steve French6473a552005-11-29 20:20:10 -08001938 }
Steve French50c2f752007-07-13 00:33:32 +00001939
Steve French7f573562005-08-30 11:32:14 -07001940 full_path = build_path_from_dentry(direntry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001941 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301942 rc = -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001943 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301944 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001945 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001946
Jeff Layton0f4d6342009-03-26 13:35:37 -04001947 /*
1948 * Attempt to flush data before changing attributes. We need to do
1949 * this for ATTR_SIZE and ATTR_MTIME for sure, and if we change the
1950 * ownership or mode then we may also need to do this. Here, we take
1951 * the safe way out and just do the flush on all setattr requests. If
1952 * the flush returns error, store it to report later and continue.
1953 *
1954 * BB: This should be smarter. Why bother flushing pages that
1955 * will be truncated anyway? Also, should we error out here if
1956 * the flush returns error?
1957 */
1958 rc = filemap_write_and_wait(inode->i_mapping);
1959 if (rc != 0) {
1960 cifsInode->write_behind_rc = rc;
1961 rc = 0;
Steve French50531442008-03-14 19:21:31 +00001962 }
Jeff Laytoncea21802007-11-20 23:19:03 +00001963
Steve French50531442008-03-14 19:21:31 +00001964 if (attrs->ia_valid & ATTR_SIZE) {
Jeff Layton8efdbde2008-07-23 21:28:12 +00001965 rc = cifs_set_file_size(inode, attrs, xid, full_path);
1966 if (rc != 0)
Steve Frenche30dcf32005-09-20 20:49:16 -07001967 goto cifs_setattr_exit;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001968 }
Jeff Layton4ca691a2008-05-22 09:33:34 -04001969
1970 /*
1971 * Without unix extensions we can't send ownership changes to the
1972 * server, so silently ignore them. This is consistent with how
1973 * local DOS/Windows filesystems behave (VFAT, NTFS, etc). With
1974 * CIFSACL support + proper Windows to Unix idmapping, we may be
1975 * able to support this in the future.
1976 */
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001977 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID))
Jeff Layton4ca691a2008-05-22 09:33:34 -04001978 attrs->ia_valid &= ~(ATTR_UID | ATTR_GID);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001979
Jeff Laytond32c4f22007-10-18 03:05:22 -07001980 /* skip mode change if it's just for clearing setuid/setgid */
1981 if (attrs->ia_valid & (ATTR_KILL_SUID|ATTR_KILL_SGID))
1982 attrs->ia_valid &= ~ATTR_MODE;
1983
Linus Torvalds1da177e2005-04-16 15:20:36 -07001984 if (attrs->ia_valid & ATTR_MODE) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001985 cFYI(1, "Mode changed to 0%o", attrs->ia_mode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001986 mode = attrs->ia_mode;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001987 }
1988
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001989 if (attrs->ia_valid & ATTR_MODE) {
Steve Frenchcdbce9c2005-11-19 21:04:52 -08001990 rc = 0;
Steve French97837582007-12-31 07:47:21 +00001991#ifdef CONFIG_CIFS_EXPERIMENTAL
1992 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL)
Jeff Layton02eadef2008-05-09 21:26:11 +00001993 rc = mode_to_acl(inode, full_path, mode);
Jeff Layton51328612008-05-22 09:33:34 -04001994 else
Steve French97837582007-12-31 07:47:21 +00001995#endif
Jeff Layton51328612008-05-22 09:33:34 -04001996 if (((mode & S_IWUGO) == 0) &&
1997 (cifsInode->cifsAttrs & ATTR_READONLY) == 0) {
Jeff Laytonfeb3e202008-08-02 07:26:12 -04001998
1999 dosattr = cifsInode->cifsAttrs | ATTR_READONLY;
2000
Jeff Layton51328612008-05-22 09:33:34 -04002001 /* fix up mode if we're not using dynperm */
2002 if ((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM) == 0)
2003 attrs->ia_mode = inode->i_mode & ~S_IWUGO;
2004 } else if ((mode & S_IWUGO) &&
2005 (cifsInode->cifsAttrs & ATTR_READONLY)) {
Jeff Laytonfeb3e202008-08-02 07:26:12 -04002006
2007 dosattr = cifsInode->cifsAttrs & ~ATTR_READONLY;
2008 /* Attributes of 0 are ignored */
2009 if (dosattr == 0)
2010 dosattr |= ATTR_NORMAL;
Jeff Layton51328612008-05-22 09:33:34 -04002011
2012 /* reset local inode permissions to normal */
2013 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM)) {
2014 attrs->ia_mode &= ~(S_IALLUGO);
2015 if (S_ISDIR(inode->i_mode))
2016 attrs->ia_mode |=
2017 cifs_sb->mnt_dir_mode;
2018 else
2019 attrs->ia_mode |=
2020 cifs_sb->mnt_file_mode;
2021 }
2022 } else if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM)) {
2023 /* ignore mode change - ATTR_READONLY hasn't changed */
2024 attrs->ia_valid &= ~ATTR_MODE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002025 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002026 }
2027
Jeff Laytonfeb3e202008-08-02 07:26:12 -04002028 if (attrs->ia_valid & (ATTR_MTIME|ATTR_ATIME|ATTR_CTIME) ||
2029 ((attrs->ia_valid & ATTR_MODE) && dosattr)) {
2030 rc = cifs_set_file_info(inode, attrs, xid, full_path, dosattr);
2031 /* BB: check for rc = -EOPNOTSUPP and switch to legacy mode */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002032
Steve Frenche30dcf32005-09-20 20:49:16 -07002033 /* Even if error on time set, no sense failing the call if
2034 the server would set the time to a reasonable value anyway,
2035 and this check ensures that we are not being called from
2036 sys_utimes in which case we ought to fail the call back to
2037 the user when the server rejects the call */
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002038 if ((rc) && (attrs->ia_valid &
Jeff Laytonfeb3e202008-08-02 07:26:12 -04002039 (ATTR_MODE | ATTR_GID | ATTR_UID | ATTR_SIZE)))
Steve Frenche30dcf32005-09-20 20:49:16 -07002040 rc = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002041 }
2042
2043 /* do not need local check to inode_check_ok since the server does
2044 that */
Christoph Hellwig10257742010-06-04 11:30:02 +02002045 if (rc)
2046 goto cifs_setattr_exit;
2047
2048 if ((attrs->ia_valid & ATTR_SIZE) &&
Christoph Hellwig1b947462010-07-18 17:51:21 -04002049 attrs->ia_size != i_size_read(inode))
2050 truncate_setsize(inode, attrs->ia_size);
Christoph Hellwig10257742010-06-04 11:30:02 +02002051
2052 setattr_copy(inode, attrs);
2053 mark_inode_dirty(inode);
2054 return 0;
2055
Steve Frenche30dcf32005-09-20 20:49:16 -07002056cifs_setattr_exit:
Linus Torvalds1da177e2005-04-16 15:20:36 -07002057 kfree(full_path);
2058 FreeXid(xid);
2059 return rc;
2060}
2061
Jeff Layton0510eeb2008-08-02 07:26:12 -04002062int
2063cifs_setattr(struct dentry *direntry, struct iattr *attrs)
2064{
2065 struct inode *inode = direntry->d_inode;
2066 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
2067 struct cifsTconInfo *pTcon = cifs_sb->tcon;
2068
2069 if (pTcon->unix_ext)
2070 return cifs_setattr_unix(direntry, attrs);
2071
2072 return cifs_setattr_nounix(direntry, attrs);
2073
2074 /* BB: add cifs_setattr_legacy for really old servers */
2075}
2076
Steve French99ee4db2007-02-27 05:35:17 +00002077#if 0
Linus Torvalds1da177e2005-04-16 15:20:36 -07002078void cifs_delete_inode(struct inode *inode)
2079{
Joe Perchesb6b38f72010-04-21 03:50:45 +00002080 cFYI(1, "In cifs_delete_inode, inode = 0x%p", inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002081 /* may have to add back in if and when safe distributed caching of
2082 directories added e.g. via FindNotify */
2083}
Steve French99ee4db2007-02-27 05:35:17 +00002084#endif