blob: 0fa145596fcf7c1ef2425f4babd09f713ce8dd11 [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);
Joe Perchesc21dfb62010-07-12 13:50:14 -0700291 struct cifsFileInfo *cfile = filp->private_data;
Jeff Laytonba00ba62010-09-20 16:01:31 -0700292 struct cifsTconInfo *tcon = cfile->tcon;
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
Stefan Metzmacher1b12b9c2010-08-05 21:19:56 +0200335 /* check for Minshall+French symlinks */
336 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MF_SYMLINKS) {
337 int tmprc = CIFSCheckMFSymlink(&fattr, full_path, cifs_sb, xid);
338 if (tmprc)
339 cFYI(1, "CIFSCheckMFSymlink: %d", tmprc);
340 }
341
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400342 if (*pinode == NULL) {
343 /* get new inode */
Jeff Layton4065c802010-05-17 07:18:58 -0400344 cifs_fill_uniqueid(sb, &fattr);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400345 *pinode = cifs_iget(sb, &fattr);
346 if (!*pinode)
347 rc = -ENOMEM;
348 } else {
349 /* we already have inode, update it */
350 cifs_fattr_to_inode(*pinode, &fattr);
351 }
Steve French0e4bbde2008-05-20 19:50:46 +0000352
Linus Torvalds1da177e2005-04-16 15:20:36 -0700353 return rc;
354}
355
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400356static int
357cifs_sfu_type(struct cifs_fattr *fattr, const unsigned char *path,
358 struct cifs_sb_info *cifs_sb, int xid)
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800359{
360 int rc;
Steve French4b18f2a2008-04-29 00:06:05 +0000361 int oplock = 0;
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800362 __u16 netfid;
363 struct cifsTconInfo *pTcon = cifs_sb->tcon;
Steve French86c96b42005-11-18 20:25:31 -0800364 char buf[24];
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800365 unsigned int bytes_read;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000366 char *pbuf;
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800367
368 pbuf = buf;
369
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400370 fattr->cf_mode &= ~S_IFMT;
371
372 if (fattr->cf_eof == 0) {
373 fattr->cf_mode |= S_IFIFO;
374 fattr->cf_dtype = DT_FIFO;
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800375 return 0;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400376 } else if (fattr->cf_eof < 8) {
377 fattr->cf_mode |= S_IFREG;
378 fattr->cf_dtype = DT_REG;
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800379 return -EINVAL; /* EOPNOTSUPP? */
380 }
Steve French50c2f752007-07-13 00:33:32 +0000381
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800382 rc = CIFSSMBOpen(xid, pTcon, path, FILE_OPEN, GENERIC_READ,
383 CREATE_NOT_DIR, &netfid, &oplock, NULL,
384 cifs_sb->local_nls,
385 cifs_sb->mnt_cifs_flags &
386 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000387 if (rc == 0) {
Steve Frenchec637e32005-12-12 20:53:18 -0800388 int buf_type = CIFS_NO_BUFFER;
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800389 /* Read header */
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400390 rc = CIFSSMBRead(xid, pTcon, netfid,
Steve French86c96b42005-11-18 20:25:31 -0800391 24 /* length */, 0 /* offset */,
Steve Frenchec637e32005-12-12 20:53:18 -0800392 &bytes_read, &pbuf, &buf_type);
Steve French4523cc32007-04-30 20:13:06 +0000393 if ((rc == 0) && (bytes_read >= 8)) {
394 if (memcmp("IntxBLK", pbuf, 8) == 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000395 cFYI(1, "Block device");
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400396 fattr->cf_mode |= S_IFBLK;
397 fattr->cf_dtype = DT_BLK;
Steve French4523cc32007-04-30 20:13:06 +0000398 if (bytes_read == 24) {
Steve French86c96b42005-11-18 20:25:31 -0800399 /* we have enough to decode dev num */
400 __u64 mjr; /* major */
401 __u64 mnr; /* minor */
402 mjr = le64_to_cpu(*(__le64 *)(pbuf+8));
403 mnr = le64_to_cpu(*(__le64 *)(pbuf+16));
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400404 fattr->cf_rdev = MKDEV(mjr, mnr);
Steve French86c96b42005-11-18 20:25:31 -0800405 }
Steve French4523cc32007-04-30 20:13:06 +0000406 } else if (memcmp("IntxCHR", pbuf, 8) == 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000407 cFYI(1, "Char device");
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400408 fattr->cf_mode |= S_IFCHR;
409 fattr->cf_dtype = DT_CHR;
Steve French4523cc32007-04-30 20:13:06 +0000410 if (bytes_read == 24) {
Steve French86c96b42005-11-18 20:25:31 -0800411 /* we have enough to decode dev num */
412 __u64 mjr; /* major */
413 __u64 mnr; /* minor */
414 mjr = le64_to_cpu(*(__le64 *)(pbuf+8));
415 mnr = le64_to_cpu(*(__le64 *)(pbuf+16));
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400416 fattr->cf_rdev = MKDEV(mjr, mnr);
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000417 }
Steve French4523cc32007-04-30 20:13:06 +0000418 } else if (memcmp("IntxLNK", pbuf, 7) == 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000419 cFYI(1, "Symlink");
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400420 fattr->cf_mode |= S_IFLNK;
421 fattr->cf_dtype = DT_LNK;
Steve French86c96b42005-11-18 20:25:31 -0800422 } else {
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400423 fattr->cf_mode |= S_IFREG; /* file? */
424 fattr->cf_dtype = DT_REG;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000425 rc = -EOPNOTSUPP;
Steve French86c96b42005-11-18 20:25:31 -0800426 }
Steve French3020a1f2005-11-18 11:31:10 -0800427 } else {
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400428 fattr->cf_mode |= S_IFREG; /* then it is a file */
429 fattr->cf_dtype = DT_REG;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000430 rc = -EOPNOTSUPP; /* or some unknown SFU type */
431 }
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800432 CIFSSMBClose(xid, pTcon, netfid);
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800433 }
434 return rc;
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800435}
436
Steve French9e294f12005-11-17 16:59:21 -0800437#define SFBITS_MASK (S_ISVTX | S_ISGID | S_ISUID) /* SETFILEBITS valid bits */
438
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400439/*
440 * Fetch mode bits as provided by SFU.
441 *
442 * FIXME: Doesn't this clobber the type bit we got from cifs_sfu_type ?
443 */
444static int cifs_sfu_mode(struct cifs_fattr *fattr, const unsigned char *path,
445 struct cifs_sb_info *cifs_sb, int xid)
Steve French9e294f12005-11-17 16:59:21 -0800446{
Steve French3020a1f2005-11-18 11:31:10 -0800447#ifdef CONFIG_CIFS_XATTR
Steve French9e294f12005-11-17 16:59:21 -0800448 ssize_t rc;
449 char ea_value[4];
450 __u32 mode;
451
Jeff Layton31c05192010-02-10 16:18:26 -0500452 rc = CIFSSMBQAllEAs(xid, cifs_sb->tcon, path, "SETFILEBITS",
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400453 ea_value, 4 /* size of buf */, cifs_sb->local_nls,
454 cifs_sb->mnt_cifs_flags &
455 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French4523cc32007-04-30 20:13:06 +0000456 if (rc < 0)
Steve French9e294f12005-11-17 16:59:21 -0800457 return (int)rc;
458 else if (rc > 3) {
459 mode = le32_to_cpu(*((__le32 *)ea_value));
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400460 fattr->cf_mode &= ~SFBITS_MASK;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000461 cFYI(1, "special bits 0%o org mode 0%o", mode,
462 fattr->cf_mode);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400463 fattr->cf_mode = (mode & SFBITS_MASK) | fattr->cf_mode;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000464 cFYI(1, "special mode bits 0%o", mode);
Steve French9e294f12005-11-17 16:59:21 -0800465 }
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400466
467 return 0;
Steve French3020a1f2005-11-18 11:31:10 -0800468#else
469 return -EOPNOTSUPP;
470#endif
Steve French9e294f12005-11-17 16:59:21 -0800471}
472
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400473/* Fill a cifs_fattr struct with info from FILE_ALL_INFO */
Steve Frenchf1230c92009-07-22 23:13:01 +0000474static void
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400475cifs_all_info_to_fattr(struct cifs_fattr *fattr, FILE_ALL_INFO *info,
476 struct cifs_sb_info *cifs_sb, bool adjust_tz)
Steve Frenchb9a32602008-05-20 21:52:32 +0000477{
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400478 memset(fattr, 0, sizeof(*fattr));
479 fattr->cf_cifsattrs = le32_to_cpu(info->Attributes);
480 if (info->DeletePending)
481 fattr->cf_flags |= CIFS_FATTR_DELETE_PENDING;
Steve Frenchb9a32602008-05-20 21:52:32 +0000482
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400483 if (info->LastAccessTime)
484 fattr->cf_atime = cifs_NTtimeToUnix(info->LastAccessTime);
485 else
486 fattr->cf_atime = CURRENT_TIME;
487
488 fattr->cf_ctime = cifs_NTtimeToUnix(info->ChangeTime);
489 fattr->cf_mtime = cifs_NTtimeToUnix(info->LastWriteTime);
490
491 if (adjust_tz) {
492 fattr->cf_ctime.tv_sec += cifs_sb->tcon->ses->server->timeAdj;
493 fattr->cf_mtime.tv_sec += cifs_sb->tcon->ses->server->timeAdj;
494 }
495
496 fattr->cf_eof = le64_to_cpu(info->EndOfFile);
497 fattr->cf_bytes = le64_to_cpu(info->AllocationSize);
498
499 if (fattr->cf_cifsattrs & ATTR_DIRECTORY) {
500 fattr->cf_mode = S_IFDIR | cifs_sb->mnt_dir_mode;
501 fattr->cf_dtype = DT_DIR;
502 } else {
503 fattr->cf_mode = S_IFREG | cifs_sb->mnt_file_mode;
504 fattr->cf_dtype = DT_REG;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400505
Jeff Laytond0c280d2009-07-09 01:46:44 -0400506 /* clear write bits if ATTR_READONLY is set */
507 if (fattr->cf_cifsattrs & ATTR_READONLY)
508 fattr->cf_mode &= ~(S_IWUGO);
509 }
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400510
511 fattr->cf_nlink = le32_to_cpu(info->NumberOfLinks);
512
513 fattr->cf_uid = cifs_sb->mnt_uid;
514 fattr->cf_gid = cifs_sb->mnt_gid;
Steve Frenchb9a32602008-05-20 21:52:32 +0000515}
516
Jeff Laytonabab0952010-02-12 07:44:18 -0500517int cifs_get_file_info(struct file *filp)
518{
519 int rc;
520 int xid;
521 FILE_ALL_INFO find_data;
522 struct cifs_fattr fattr;
523 struct inode *inode = filp->f_path.dentry->d_inode;
524 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
Joe Perchesc21dfb62010-07-12 13:50:14 -0700525 struct cifsFileInfo *cfile = filp->private_data;
Jeff Laytonba00ba62010-09-20 16:01:31 -0700526 struct cifsTconInfo *tcon = cfile->tcon;
Jeff Laytonabab0952010-02-12 07:44:18 -0500527
528 xid = GetXid();
529 rc = CIFSSMBQFileInfo(xid, tcon, cfile->netfid, &find_data);
530 if (rc == -EOPNOTSUPP || rc == -EINVAL) {
531 /*
532 * FIXME: legacy server -- fall back to path-based call?
Steve Frenchff2157132010-03-09 20:30:42 +0000533 * for now, just skip revalidating and mark inode for
534 * immediate reval.
535 */
Jeff Laytonabab0952010-02-12 07:44:18 -0500536 rc = 0;
537 CIFS_I(inode)->time = 0;
538 goto cgfi_exit;
539 } else if (rc == -EREMOTE) {
540 cifs_create_dfs_fattr(&fattr, inode->i_sb);
541 rc = 0;
542 } else if (rc)
543 goto cgfi_exit;
544
545 /*
546 * don't bother with SFU junk here -- just mark inode as needing
547 * revalidation.
548 */
549 cifs_all_info_to_fattr(&fattr, &find_data, cifs_sb, false);
550 fattr.cf_uniqueid = CIFS_I(inode)->uniqueid;
551 fattr.cf_flags |= CIFS_FATTR_NEED_REVAL;
552 cifs_fattr_to_inode(inode, &fattr);
553cgfi_exit:
554 FreeXid(xid);
555 return rc;
556}
557
Linus Torvalds1da177e2005-04-16 15:20:36 -0700558int cifs_get_inode_info(struct inode **pinode,
Steve French646dd532008-05-15 01:50:56 +0000559 const unsigned char *full_path, FILE_ALL_INFO *pfindData,
Steve French8b1327f2008-03-14 22:37:16 +0000560 struct super_block *sb, int xid, const __u16 *pfid)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700561{
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400562 int rc = 0, tmprc;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700563 struct cifsTconInfo *pTcon;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700564 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700565 char *buf = NULL;
Steve French5ade9de2008-05-02 20:56:23 +0000566 bool adjustTZ = false;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400567 struct cifs_fattr fattr;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700568
569 pTcon = cifs_sb->tcon;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000570 cFYI(1, "Getting info on %s", full_path);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700571
Steve Frenchd0d2f2d2005-06-02 15:12:36 -0700572 if ((pfindData == NULL) && (*pinode != NULL)) {
573 if (CIFS_I(*pinode)->clientCanCacheRead) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000574 cFYI(1, "No need to revalidate cached inode sizes");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700575 return rc;
576 }
577 }
578
579 /* if file info not passed in then get it from server */
Steve Frenchd0d2f2d2005-06-02 15:12:36 -0700580 if (pfindData == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700581 buf = kmalloc(sizeof(FILE_ALL_INFO), GFP_KERNEL);
Steve Frenchd0d2f2d2005-06-02 15:12:36 -0700582 if (buf == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700583 return -ENOMEM;
584 pfindData = (FILE_ALL_INFO *)buf;
Igor Mammedov79626702008-03-09 03:44:18 +0000585
Linus Torvalds1da177e2005-04-16 15:20:36 -0700586 /* could do find first instead but this returns more info */
Igor Mammedov79626702008-03-09 03:44:18 +0000587 rc = CIFSSMBQPathInfo(xid, pTcon, full_path, pfindData,
Steve Frenchacf1a1b2006-10-12 03:28:28 +0000588 0 /* not legacy */,
Steve French6b8edfe2005-08-23 20:26:03 -0700589 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
Steve French737b7582005-04-28 22:41:06 -0700590 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French6b8edfe2005-08-23 20:26:03 -0700591 /* BB optimize code so we do not make the above call
592 when server claims no NT SMB support and the above call
593 failed at least once - set flag in tcon or mount */
Steve French4523cc32007-04-30 20:13:06 +0000594 if ((rc == -EOPNOTSUPP) || (rc == -EINVAL)) {
Igor Mammedov79626702008-03-09 03:44:18 +0000595 rc = SMBQueryInformation(xid, pTcon, full_path,
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000596 pfindData, cifs_sb->local_nls,
Steve French6b8edfe2005-08-23 20:26:03 -0700597 cifs_sb->mnt_cifs_flags &
598 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French4b18f2a2008-04-29 00:06:05 +0000599 adjustTZ = true;
Steve French6b8edfe2005-08-23 20:26:03 -0700600 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700601 }
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400602
603 if (!rc) {
604 cifs_all_info_to_fattr(&fattr, (FILE_ALL_INFO *) pfindData,
605 cifs_sb, adjustTZ);
606 } else if (rc == -EREMOTE) {
607 cifs_create_dfs_fattr(&fattr, sb);
Steve Frenchb9a32602008-05-20 21:52:32 +0000608 rc = 0;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400609 } else {
Igor Mammedov79626702008-03-09 03:44:18 +0000610 goto cgii_exit;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400611 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700612
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400613 /*
614 * If an inode wasn't passed in, then get the inode number
615 *
616 * Is an i_ino of zero legal? Can we use that to check if the server
617 * supports returning inode numbers? Are there other sanity checks we
618 * can use to ensure that the server is really filling in that field?
619 *
620 * We can not use the IndexNumber field by default from Windows or
621 * Samba (in ALL_INFO buf) but we can request it explicitly. The SNIA
622 * CIFS spec claims that this value is unique within the scope of a
623 * share, and the windows docs hint that it's actually unique
624 * per-machine.
625 *
626 * There may be higher info levels that work but are there Windows
627 * server or network appliances for which IndexNumber field is not
628 * guaranteed unique?
629 */
Steve Frenchb9a32602008-05-20 21:52:32 +0000630 if (*pinode == NULL) {
Steve Frenchb9a32602008-05-20 21:52:32 +0000631 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM) {
632 int rc1 = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700633
Steve Frenchb9a32602008-05-20 21:52:32 +0000634 rc1 = CIFSGetSrvInodeNumber(xid, pTcon,
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400635 full_path, &fattr.cf_uniqueid,
Steve French737b7582005-04-28 22:41:06 -0700636 cifs_sb->local_nls,
637 cifs_sb->mnt_cifs_flags &
638 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Laytonec06aed2009-11-06 14:18:29 -0500639 if (rc1 || !fattr.cf_uniqueid) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000640 cFYI(1, "GetSrvInodeNum rc %d", rc1);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400641 fattr.cf_uniqueid = iunique(sb, ROOT_I);
Jeff Laytonec06aed2009-11-06 14:18:29 -0500642 cifs_autodisable_serverino(cifs_sb);
Jeff Layton132ac7b2009-02-10 07:33:57 -0500643 }
Jeff Layton132ac7b2009-02-10 07:33:57 -0500644 } else {
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400645 fattr.cf_uniqueid = iunique(sb, ROOT_I);
Jeff Layton132ac7b2009-02-10 07:33:57 -0500646 }
Steve Frenchb9a32602008-05-20 21:52:32 +0000647 } else {
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400648 fattr.cf_uniqueid = CIFS_I(*pinode)->uniqueid;
Steve Frenchb9a32602008-05-20 21:52:32 +0000649 }
650
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400651 /* query for SFU type info if supported and needed */
652 if (fattr.cf_cifsattrs & ATTR_SYSTEM &&
653 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_UNX_EMUL) {
654 tmprc = cifs_sfu_type(&fattr, full_path, cifs_sb, xid);
655 if (tmprc)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000656 cFYI(1, "cifs_sfu_type failed: %d", tmprc);
Steve Frenchb9a32602008-05-20 21:52:32 +0000657 }
Steve Frenchb9a32602008-05-20 21:52:32 +0000658
Steve Frenchb9a32602008-05-20 21:52:32 +0000659#ifdef CONFIG_CIFS_EXPERIMENTAL
660 /* fill in 0777 bits from ACL */
661 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000662 cFYI(1, "Getting mode bits from ACL");
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400663 cifs_acl_to_fattr(cifs_sb, &fattr, *pinode, full_path, pfid);
Steve Frenchb9a32602008-05-20 21:52:32 +0000664 }
665#endif
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400666
667 /* fill in remaining high mode bits e.g. SUID, VTX */
668 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_UNX_EMUL)
669 cifs_sfu_mode(&fattr, full_path, cifs_sb, xid);
670
Stefan Metzmacher1b12b9c2010-08-05 21:19:56 +0200671 /* check for Minshall+French symlinks */
672 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MF_SYMLINKS) {
673 tmprc = CIFSCheckMFSymlink(&fattr, full_path, cifs_sb, xid);
674 if (tmprc)
675 cFYI(1, "CIFSCheckMFSymlink: %d", tmprc);
676 }
677
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400678 if (!*pinode) {
679 *pinode = cifs_iget(sb, &fattr);
680 if (!*pinode)
681 rc = -ENOMEM;
682 } else {
683 cifs_fattr_to_inode(*pinode, &fattr);
Steve Frenchb9a32602008-05-20 21:52:32 +0000684 }
685
Igor Mammedov79626702008-03-09 03:44:18 +0000686cgii_exit:
Linus Torvalds1da177e2005-04-16 15:20:36 -0700687 kfree(buf);
688 return rc;
689}
690
Steve French7f8ed422007-09-28 22:28:55 +0000691static const struct inode_operations cifs_ipc_inode_ops = {
692 .lookup = cifs_lookup,
693};
694
Igor Mammedove4cce942009-02-10 14:10:26 +0300695char *cifs_build_path_to_root(struct cifs_sb_info *cifs_sb)
Steve French8be0ed42008-12-05 19:14:12 +0000696{
697 int pplen = cifs_sb->prepathlen;
698 int dfsplen;
699 char *full_path = NULL;
700
701 /* if no prefix path, simply set path to the root of share to "" */
702 if (pplen == 0) {
703 full_path = kmalloc(1, GFP_KERNEL);
704 if (full_path)
705 full_path[0] = 0;
706 return full_path;
707 }
708
709 if (cifs_sb->tcon && (cifs_sb->tcon->Flags & SMB_SHARE_IS_IN_DFS))
710 dfsplen = strnlen(cifs_sb->tcon->treeName, MAX_TREE_SIZE + 1);
711 else
712 dfsplen = 0;
713
714 full_path = kmalloc(dfsplen + pplen + 1, GFP_KERNEL);
715 if (full_path == NULL)
716 return full_path;
717
718 if (dfsplen) {
719 strncpy(full_path, cifs_sb->tcon->treeName, dfsplen);
720 /* switch slash direction in prepath depending on whether
721 * windows or posix style path names
722 */
723 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_POSIX_PATHS) {
724 int i;
725 for (i = 0; i < dfsplen; i++) {
726 if (full_path[i] == '\\')
727 full_path[i] = '/';
728 }
729 }
730 }
731 strncpy(full_path + dfsplen, cifs_sb->prepath, pplen);
732 full_path[dfsplen + pplen] = 0; /* add trailing null */
733 return full_path;
734}
735
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400736static int
737cifs_find_inode(struct inode *inode, void *opaque)
738{
739 struct cifs_fattr *fattr = (struct cifs_fattr *) opaque;
740
Jeff Laytonf30b9c12010-07-19 18:00:17 -0400741 /* don't match inode with different uniqueid */
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400742 if (CIFS_I(inode)->uniqueid != fattr->cf_uniqueid)
743 return 0;
744
Jeff Laytonf30b9c12010-07-19 18:00:17 -0400745 /* don't match inode of different type */
746 if ((inode->i_mode & S_IFMT) != (fattr->cf_mode & S_IFMT))
747 return 0;
748
Jeff Layton5acfec22010-08-02 17:43:54 -0400749 /* if it's not a directory or has no dentries, then flag it */
750 if (S_ISDIR(inode->i_mode) && !list_empty(&inode->i_dentry))
Jeff Layton3d694382010-05-11 14:59:55 -0400751 fattr->cf_flags |= CIFS_FATTR_INO_COLLISION;
Jeff Layton3d694382010-05-11 14:59:55 -0400752
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400753 return 1;
754}
755
756static int
757cifs_init_inode(struct inode *inode, void *opaque)
758{
759 struct cifs_fattr *fattr = (struct cifs_fattr *) opaque;
760
761 CIFS_I(inode)->uniqueid = fattr->cf_uniqueid;
762 return 0;
763}
764
Jeff Layton5acfec22010-08-02 17:43:54 -0400765/*
766 * walk dentry list for an inode and report whether it has aliases that
767 * are hashed. We use this to determine if a directory inode can actually
768 * be used.
769 */
770static bool
771inode_has_hashed_dentries(struct inode *inode)
772{
773 struct dentry *dentry;
774
775 spin_lock(&dcache_lock);
776 list_for_each_entry(dentry, &inode->i_dentry, d_alias) {
777 if (!d_unhashed(dentry) || IS_ROOT(dentry)) {
778 spin_unlock(&dcache_lock);
779 return true;
780 }
781 }
782 spin_unlock(&dcache_lock);
783 return false;
784}
785
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400786/* Given fattrs, get a corresponding inode */
787struct inode *
788cifs_iget(struct super_block *sb, struct cifs_fattr *fattr)
789{
790 unsigned long hash;
791 struct inode *inode;
792
Jeff Layton3d694382010-05-11 14:59:55 -0400793retry_iget5_locked:
Joe Perchesb6b38f72010-04-21 03:50:45 +0000794 cFYI(1, "looking for uniqueid=%llu", fattr->cf_uniqueid);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400795
796 /* hash down to 32-bits on 32-bit arch */
797 hash = cifs_uniqueid_to_ino_t(fattr->cf_uniqueid);
798
799 inode = iget5_locked(sb, hash, cifs_find_inode, cifs_init_inode, fattr);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400800 if (inode) {
Jeff Layton5acfec22010-08-02 17:43:54 -0400801 /* was there a potentially problematic inode collision? */
Jeff Layton3d694382010-05-11 14:59:55 -0400802 if (fattr->cf_flags & CIFS_FATTR_INO_COLLISION) {
Jeff Layton3d694382010-05-11 14:59:55 -0400803 fattr->cf_flags &= ~CIFS_FATTR_INO_COLLISION;
Jeff Layton5acfec22010-08-02 17:43:54 -0400804
805 if (inode_has_hashed_dentries(inode)) {
806 cifs_autodisable_serverino(CIFS_SB(sb));
807 iput(inode);
808 fattr->cf_uniqueid = iunique(sb, ROOT_I);
809 goto retry_iget5_locked;
810 }
Jeff Layton3d694382010-05-11 14:59:55 -0400811 }
812
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400813 cifs_fattr_to_inode(inode, fattr);
814 if (sb->s_flags & MS_NOATIME)
815 inode->i_flags |= S_NOATIME | S_NOCMTIME;
816 if (inode->i_state & I_NEW) {
817 inode->i_ino = hash;
Steve French0ccd4802010-07-16 04:31:02 +0000818#ifdef CONFIG_CIFS_FSCACHE
Suresh Jayaraman9451a9a2010-07-05 18:12:45 +0530819 /* initialize per-inode cache cookie pointer */
820 CIFS_I(inode)->fscache = NULL;
Steve French0ccd4802010-07-16 04:31:02 +0000821#endif
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400822 unlock_new_inode(inode);
823 }
824 }
825
826 return inode;
827}
828
Linus Torvalds1da177e2005-04-16 15:20:36 -0700829/* gets root inode */
Jeff Laytonbd433d42009-05-27 09:37:34 -0400830struct inode *cifs_root_iget(struct super_block *sb, unsigned long ino)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700831{
David Howellsce634ab2008-02-07 00:15:33 -0800832 int xid;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700833 struct cifs_sb_info *cifs_sb;
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400834 struct inode *inode = NULL;
David Howellsce634ab2008-02-07 00:15:33 -0800835 long rc;
Steve French8be0ed42008-12-05 19:14:12 +0000836 char *full_path;
David Howellsce634ab2008-02-07 00:15:33 -0800837
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400838 cifs_sb = CIFS_SB(sb);
Igor Mammedove4cce942009-02-10 14:10:26 +0300839 full_path = cifs_build_path_to_root(cifs_sb);
Steve French8be0ed42008-12-05 19:14:12 +0000840 if (full_path == NULL)
841 return ERR_PTR(-ENOMEM);
Steve Frenchc18c8422007-07-18 23:21:09 +0000842
Steve French8be0ed42008-12-05 19:14:12 +0000843 xid = GetXid();
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400844 if (cifs_sb->tcon->unix_ext)
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400845 rc = cifs_get_inode_info_unix(&inode, full_path, sb, xid);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400846 else
847 rc = cifs_get_inode_info(&inode, full_path, NULL, sb,
Steve French8be0ed42008-12-05 19:14:12 +0000848 xid, NULL);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400849
850 if (!inode)
Suresh Jayaramanf0138a72010-08-26 14:46:09 +0530851 return ERR_PTR(rc);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400852
Steve French0ccd4802010-07-16 04:31:02 +0000853#ifdef CONFIG_CIFS_FSCACHE
Suresh Jayaramand03382c2010-07-05 18:12:27 +0530854 /* populate tcon->resource_id */
855 cifs_sb->tcon->resource_id = CIFS_I(inode)->uniqueid;
Steve French0ccd4802010-07-16 04:31:02 +0000856#endif
Suresh Jayaramand03382c2010-07-05 18:12:27 +0530857
Steve French7f8ed422007-09-28 22:28:55 +0000858 if (rc && cifs_sb->tcon->ipc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000859 cFYI(1, "ipc connection - fake read inode");
Steve French7f8ed422007-09-28 22:28:55 +0000860 inode->i_mode |= S_IFDIR;
861 inode->i_nlink = 2;
862 inode->i_op = &cifs_ipc_inode_ops;
863 inode->i_fop = &simple_dir_operations;
864 inode->i_uid = cifs_sb->mnt_uid;
865 inode->i_gid = cifs_sb->mnt_gid;
Steve Frenchad661332008-08-12 14:14:40 +0000866 } else if (rc) {
Steve French8be0ed42008-12-05 19:14:12 +0000867 kfree(full_path);
David Howellsce634ab2008-02-07 00:15:33 -0800868 _FreeXid(xid);
869 iget_failed(inode);
870 return ERR_PTR(rc);
Steve French7f8ed422007-09-28 22:28:55 +0000871 }
872
David Howellsce634ab2008-02-07 00:15:33 -0800873
Steve French8be0ed42008-12-05 19:14:12 +0000874 kfree(full_path);
David Howellsce634ab2008-02-07 00:15:33 -0800875 /* can not call macro FreeXid here since in a void func
876 * TODO: This is no longer true
877 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700878 _FreeXid(xid);
David Howellsce634ab2008-02-07 00:15:33 -0800879 return inode;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700880}
881
Steve French388e57b2008-09-16 23:50:58 +0000882static int
883cifs_set_file_info(struct inode *inode, struct iattr *attrs, int xid,
884 char *full_path, __u32 dosattr)
885{
886 int rc;
887 int oplock = 0;
888 __u16 netfid;
889 __u32 netpid;
890 bool set_time = false;
891 struct cifsFileInfo *open_file;
892 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
893 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
Jeff Laytonba00ba62010-09-20 16:01:31 -0700894 struct cifsTconInfo *pTcon;
Steve French388e57b2008-09-16 23:50:58 +0000895 FILE_BASIC_INFO info_buf;
896
Steve French1adcb712009-02-25 14:19:56 +0000897 if (attrs == NULL)
898 return -EINVAL;
899
Steve French388e57b2008-09-16 23:50:58 +0000900 if (attrs->ia_valid & ATTR_ATIME) {
901 set_time = true;
902 info_buf.LastAccessTime =
903 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_atime));
904 } else
905 info_buf.LastAccessTime = 0;
906
907 if (attrs->ia_valid & ATTR_MTIME) {
908 set_time = true;
909 info_buf.LastWriteTime =
910 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_mtime));
911 } else
912 info_buf.LastWriteTime = 0;
913
914 /*
915 * Samba throws this field away, but windows may actually use it.
916 * Do not set ctime unless other time stamps are changed explicitly
917 * (i.e. by utimes()) since we would then have a mix of client and
918 * server times.
919 */
920 if (set_time && (attrs->ia_valid & ATTR_CTIME)) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000921 cFYI(1, "CIFS - CTIME changed");
Steve French388e57b2008-09-16 23:50:58 +0000922 info_buf.ChangeTime =
923 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_ctime));
924 } else
925 info_buf.ChangeTime = 0;
926
927 info_buf.CreationTime = 0; /* don't change */
928 info_buf.Attributes = cpu_to_le32(dosattr);
929
930 /*
931 * If the file is already open for write, just use that fileid
932 */
933 open_file = find_writable_file(cifsInode);
934 if (open_file) {
935 netfid = open_file->netfid;
936 netpid = open_file->pid;
Jeff Laytonba00ba62010-09-20 16:01:31 -0700937 pTcon = open_file->tcon;
Steve French388e57b2008-09-16 23:50:58 +0000938 goto set_via_filehandle;
939 }
940
Jeff Laytonba00ba62010-09-20 16:01:31 -0700941 pTcon = cifs_sb->tcon;
942
Steve French388e57b2008-09-16 23:50:58 +0000943 /*
944 * NT4 apparently returns success on this call, but it doesn't
945 * really work.
946 */
947 if (!(pTcon->ses->flags & CIFS_SES_NT4)) {
948 rc = CIFSSMBSetPathInfo(xid, pTcon, full_path,
949 &info_buf, cifs_sb->local_nls,
950 cifs_sb->mnt_cifs_flags &
951 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Layton6b37faa2008-10-06 21:54:41 +0000952 if (rc == 0) {
953 cifsInode->cifsAttrs = dosattr;
954 goto out;
955 } else if (rc != -EOPNOTSUPP && rc != -EINVAL)
Steve French388e57b2008-09-16 23:50:58 +0000956 goto out;
957 }
958
Joe Perchesb6b38f72010-04-21 03:50:45 +0000959 cFYI(1, "calling SetFileInfo since SetPathInfo for "
960 "times not supported by this server");
Steve French388e57b2008-09-16 23:50:58 +0000961 rc = CIFSSMBOpen(xid, pTcon, full_path, FILE_OPEN,
962 SYNCHRONIZE | FILE_WRITE_ATTRIBUTES,
963 CREATE_NOT_DIR, &netfid, &oplock,
964 NULL, cifs_sb->local_nls,
965 cifs_sb->mnt_cifs_flags &
966 CIFS_MOUNT_MAP_SPECIAL_CHR);
967
968 if (rc != 0) {
969 if (rc == -EIO)
970 rc = -EINVAL;
971 goto out;
972 }
973
974 netpid = current->tgid;
975
976set_via_filehandle:
977 rc = CIFSSMBSetFileInfo(xid, pTcon, &info_buf, netfid, netpid);
Steve Frenchd3889082008-09-24 19:22:52 +0000978 if (!rc)
979 cifsInode->cifsAttrs = dosattr;
980
Steve French388e57b2008-09-16 23:50:58 +0000981 if (open_file == NULL)
982 CIFSSMBClose(xid, pTcon, netfid);
983 else
Dave Kleikamp6ab409b2009-08-31 11:07:12 -0400984 cifsFileInfo_put(open_file);
Steve French388e57b2008-09-16 23:50:58 +0000985out:
986 return rc;
987}
988
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400989/*
990 * open the given file (if it isn't already), set the DELETE_ON_CLOSE bit
991 * and rename it to a random name that hopefully won't conflict with
992 * anything else.
993 */
994static int
Steve French32709582008-10-20 00:44:19 +0000995cifs_rename_pending_delete(char *full_path, struct dentry *dentry, int xid)
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400996{
997 int oplock = 0;
998 int rc;
999 __u16 netfid;
Steve French32709582008-10-20 00:44:19 +00001000 struct inode *inode = dentry->d_inode;
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001001 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
1002 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
1003 struct cifsTconInfo *tcon = cifs_sb->tcon;
Steve French32709582008-10-20 00:44:19 +00001004 __u32 dosattr, origattr;
1005 FILE_BASIC_INFO *info_buf = NULL;
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001006
1007 rc = CIFSSMBOpen(xid, tcon, full_path, FILE_OPEN,
Jeff Laytondd1db2d2008-10-16 19:27:12 -04001008 DELETE|FILE_WRITE_ATTRIBUTES, CREATE_NOT_DIR,
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001009 &netfid, &oplock, NULL, cifs_sb->local_nls,
1010 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
1011 if (rc != 0)
1012 goto out;
1013
Steve French32709582008-10-20 00:44:19 +00001014 origattr = cifsInode->cifsAttrs;
1015 if (origattr == 0)
1016 origattr |= ATTR_NORMAL;
1017
1018 dosattr = origattr & ~ATTR_READONLY;
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001019 if (dosattr == 0)
1020 dosattr |= ATTR_NORMAL;
1021 dosattr |= ATTR_HIDDEN;
1022
Steve French32709582008-10-20 00:44:19 +00001023 /* set ATTR_HIDDEN and clear ATTR_READONLY, but only if needed */
1024 if (dosattr != origattr) {
1025 info_buf = kzalloc(sizeof(*info_buf), GFP_KERNEL);
1026 if (info_buf == NULL) {
1027 rc = -ENOMEM;
1028 goto out_close;
1029 }
1030 info_buf->Attributes = cpu_to_le32(dosattr);
1031 rc = CIFSSMBSetFileInfo(xid, tcon, info_buf, netfid,
1032 current->tgid);
1033 /* although we would like to mark the file hidden
1034 if that fails we will still try to rename it */
Steve French41346092008-10-20 18:24:42 +00001035 if (rc != 0)
Steve French32709582008-10-20 00:44:19 +00001036 cifsInode->cifsAttrs = dosattr;
1037 else
1038 dosattr = origattr; /* since not able to change them */
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001039 }
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001040
Jeff Laytondd1db2d2008-10-16 19:27:12 -04001041 /* rename the file */
1042 rc = CIFSSMBRenameOpenFile(xid, tcon, netfid, NULL, cifs_sb->local_nls,
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001043 cifs_sb->mnt_cifs_flags &
1044 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French32709582008-10-20 00:44:19 +00001045 if (rc != 0) {
1046 rc = -ETXTBSY;
1047 goto undo_setattr;
1048 }
Jeff Layton6d22f092008-09-23 11:48:35 -04001049
Steve French32709582008-10-20 00:44:19 +00001050 /* try to set DELETE_ON_CLOSE */
1051 if (!cifsInode->delete_pending) {
1052 rc = CIFSSMBSetFileDisposition(xid, tcon, true, netfid,
1053 current->tgid);
1054 /*
1055 * some samba versions return -ENOENT when we try to set the
1056 * file disposition here. Likely a samba bug, but work around
1057 * it for now. This means that some cifsXXX files may hang
1058 * around after they shouldn't.
1059 *
1060 * BB: remove this hack after more servers have the fix
1061 */
1062 if (rc == -ENOENT)
1063 rc = 0;
1064 else if (rc != 0) {
1065 rc = -ETXTBSY;
1066 goto undo_rename;
1067 }
1068 cifsInode->delete_pending = true;
1069 }
Jeff Layton7ce86d52008-09-24 11:32:59 -04001070
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001071out_close:
1072 CIFSSMBClose(xid, tcon, netfid);
1073out:
Steve French32709582008-10-20 00:44:19 +00001074 kfree(info_buf);
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001075 return rc;
Steve French32709582008-10-20 00:44:19 +00001076
1077 /*
1078 * reset everything back to the original state. Don't bother
1079 * dealing with errors here since we can't do anything about
1080 * them anyway.
1081 */
1082undo_rename:
1083 CIFSSMBRenameOpenFile(xid, tcon, netfid, dentry->d_name.name,
1084 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1085 CIFS_MOUNT_MAP_SPECIAL_CHR);
1086undo_setattr:
1087 if (dosattr != origattr) {
1088 info_buf->Attributes = cpu_to_le32(origattr);
1089 if (!CIFSSMBSetFileInfo(xid, tcon, info_buf, netfid,
1090 current->tgid))
1091 cifsInode->cifsAttrs = origattr;
1092 }
1093
1094 goto out_close;
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001095}
1096
Steve Frenchff694522009-04-20 19:45:13 +00001097
1098/*
1099 * If dentry->d_inode is null (usually meaning the cached dentry
1100 * is a negative dentry) then we would attempt a standard SMB delete, but
André Goddard Rosaaf901ca2009-11-14 13:09:05 -02001101 * if that fails we can not attempt the fall back mechanisms on EACCESS
1102 * but will return the EACCESS to the caller. Note that the VFS does not call
Steve Frenchff694522009-04-20 19:45:13 +00001103 * unlink on negative dentries currently.
1104 */
Jeff Layton5f0319a2008-09-16 14:05:16 -04001105int cifs_unlink(struct inode *dir, struct dentry *dentry)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001106{
1107 int rc = 0;
1108 int xid;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001109 char *full_path = NULL;
Jeff Layton5f0319a2008-09-16 14:05:16 -04001110 struct inode *inode = dentry->d_inode;
Steve Frenchff694522009-04-20 19:45:13 +00001111 struct cifsInodeInfo *cifs_inode;
Jeff Layton5f0319a2008-09-16 14:05:16 -04001112 struct super_block *sb = dir->i_sb;
1113 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
1114 struct cifsTconInfo *tcon = cifs_sb->tcon;
Steve French60502472008-10-07 18:42:52 +00001115 struct iattr *attrs = NULL;
1116 __u32 dosattr = 0, origattr = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001117
Joe Perchesb6b38f72010-04-21 03:50:45 +00001118 cFYI(1, "cifs_unlink, dir=0x%p, dentry=0x%p", dir, dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001119
1120 xid = GetXid();
1121
Jeff Layton5f0319a2008-09-16 14:05:16 -04001122 /* Unlink can be called from rename so we can not take the
1123 * sb->s_vfs_rename_mutex here */
1124 full_path = build_path_from_dentry(dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001125 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301126 rc = -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001127 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301128 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001129 }
Steve French2d785a52007-07-15 01:48:57 +00001130
Jeff Layton5f0319a2008-09-16 14:05:16 -04001131 if ((tcon->ses->capabilities & CAP_UNIX) &&
Steve French2d785a52007-07-15 01:48:57 +00001132 (CIFS_UNIX_POSIX_PATH_OPS_CAP &
Jeff Layton5f0319a2008-09-16 14:05:16 -04001133 le64_to_cpu(tcon->fsUnixInfo.Capability))) {
1134 rc = CIFSPOSIXDelFile(xid, tcon, full_path,
Steve French2d785a52007-07-15 01:48:57 +00001135 SMB_POSIX_UNLINK_FILE_TARGET, cifs_sb->local_nls,
1136 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001137 cFYI(1, "posix del rc %d", rc);
Steve French2d785a52007-07-15 01:48:57 +00001138 if ((rc == 0) || (rc == -ENOENT))
1139 goto psx_del_no_retry;
1140 }
1141
Steve French60502472008-10-07 18:42:52 +00001142retry_std_delete:
Jeff Layton5f0319a2008-09-16 14:05:16 -04001143 rc = CIFSSMBDelFile(xid, tcon, full_path, cifs_sb->local_nls,
Steve French737b7582005-04-28 22:41:06 -07001144 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French60502472008-10-07 18:42:52 +00001145
Steve French2d785a52007-07-15 01:48:57 +00001146psx_del_no_retry:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001147 if (!rc) {
Jeff Layton5f0319a2008-09-16 14:05:16 -04001148 if (inode)
1149 drop_nlink(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001150 } else if (rc == -ENOENT) {
Jeff Layton5f0319a2008-09-16 14:05:16 -04001151 d_drop(dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001152 } else if (rc == -ETXTBSY) {
Steve French32709582008-10-20 00:44:19 +00001153 rc = cifs_rename_pending_delete(full_path, dentry, xid);
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001154 if (rc == 0)
1155 drop_nlink(inode);
Steve Frenchff694522009-04-20 19:45:13 +00001156 } else if ((rc == -EACCES) && (dosattr == 0) && inode) {
Steve French388e57b2008-09-16 23:50:58 +00001157 attrs = kzalloc(sizeof(*attrs), GFP_KERNEL);
1158 if (attrs == NULL) {
1159 rc = -ENOMEM;
1160 goto out_reval;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001161 }
Steve French388e57b2008-09-16 23:50:58 +00001162
1163 /* try to reset dos attributes */
Steve Frenchff694522009-04-20 19:45:13 +00001164 cifs_inode = CIFS_I(inode);
1165 origattr = cifs_inode->cifsAttrs;
Steve French60502472008-10-07 18:42:52 +00001166 if (origattr == 0)
1167 origattr |= ATTR_NORMAL;
1168 dosattr = origattr & ~ATTR_READONLY;
Steve French388e57b2008-09-16 23:50:58 +00001169 if (dosattr == 0)
1170 dosattr |= ATTR_NORMAL;
1171 dosattr |= ATTR_HIDDEN;
1172
1173 rc = cifs_set_file_info(inode, attrs, xid, full_path, dosattr);
Steve French388e57b2008-09-16 23:50:58 +00001174 if (rc != 0)
1175 goto out_reval;
Steve French60502472008-10-07 18:42:52 +00001176
1177 goto retry_std_delete;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001178 }
Steve French60502472008-10-07 18:42:52 +00001179
1180 /* undo the setattr if we errored out and it's needed */
1181 if (rc != 0 && dosattr != 0)
1182 cifs_set_file_info(inode, attrs, xid, full_path, origattr);
1183
Steve French388e57b2008-09-16 23:50:58 +00001184out_reval:
Jeff Layton5f0319a2008-09-16 14:05:16 -04001185 if (inode) {
Steve Frenchff694522009-04-20 19:45:13 +00001186 cifs_inode = CIFS_I(inode);
1187 cifs_inode->time = 0; /* will force revalidate to get info
Steve Frenchb2aeb9d2005-05-17 13:16:18 -05001188 when needed */
Jeff Layton5f0319a2008-09-16 14:05:16 -04001189 inode->i_ctime = current_fs_time(sb);
Steve Frenchb2aeb9d2005-05-17 13:16:18 -05001190 }
Jeff Layton5f0319a2008-09-16 14:05:16 -04001191 dir->i_ctime = dir->i_mtime = current_fs_time(sb);
Steve Frenchff694522009-04-20 19:45:13 +00001192 cifs_inode = CIFS_I(dir);
Steve French60502472008-10-07 18:42:52 +00001193 CIFS_I(dir)->time = 0; /* force revalidate of dir as well */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001194
1195 kfree(full_path);
Steve French60502472008-10-07 18:42:52 +00001196 kfree(attrs);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001197 FreeXid(xid);
1198 return rc;
1199}
1200
1201int cifs_mkdir(struct inode *inode, struct dentry *direntry, int mode)
1202{
Jeff Layton6b37faa2008-10-06 21:54:41 +00001203 int rc = 0, tmprc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001204 int xid;
1205 struct cifs_sb_info *cifs_sb;
1206 struct cifsTconInfo *pTcon;
1207 char *full_path = NULL;
1208 struct inode *newinode = NULL;
Jeff Laytoncc0bad72009-06-25 00:56:52 -04001209 struct cifs_fattr fattr;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001210
Joe Perchesb6b38f72010-04-21 03:50:45 +00001211 cFYI(1, "In cifs_mkdir, mode = 0x%x inode = 0x%p", mode, inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001212
1213 xid = GetXid();
1214
1215 cifs_sb = CIFS_SB(inode->i_sb);
1216 pTcon = cifs_sb->tcon;
1217
Steve French7f573562005-08-30 11:32:14 -07001218 full_path = build_path_from_dentry(direntry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001219 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301220 rc = -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001221 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301222 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001223 }
Steve French50c2f752007-07-13 00:33:32 +00001224
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001225 if ((pTcon->ses->capabilities & CAP_UNIX) &&
1226 (CIFS_UNIX_POSIX_PATH_OPS_CAP &
Steve French2dd29d32007-04-23 22:07:35 +00001227 le64_to_cpu(pTcon->fsUnixInfo.Capability))) {
1228 u32 oplock = 0;
Steve Frenchf6d09982008-01-08 23:18:22 +00001229 FILE_UNIX_BASIC_INFO *pInfo =
Steve French2dd29d32007-04-23 22:07:35 +00001230 kzalloc(sizeof(FILE_UNIX_BASIC_INFO), GFP_KERNEL);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001231 if (pInfo == NULL) {
Steve French2dd29d32007-04-23 22:07:35 +00001232 rc = -ENOMEM;
1233 goto mkdir_out;
1234 }
Steve French50c2f752007-07-13 00:33:32 +00001235
Al Viroce3b0f82009-03-29 19:08:22 -04001236 mode &= ~current_umask();
Steve French2dd29d32007-04-23 22:07:35 +00001237 rc = CIFSPOSIXCreate(xid, pTcon, SMB_O_DIRECTORY | SMB_O_CREAT,
1238 mode, NULL /* netfid */, pInfo, &oplock,
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001239 full_path, cifs_sb->local_nls,
1240 cifs_sb->mnt_cifs_flags &
Steve French2dd29d32007-04-23 22:07:35 +00001241 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve Frenchc45d7072007-09-17 02:04:21 +00001242 if (rc == -EOPNOTSUPP) {
1243 kfree(pInfo);
1244 goto mkdir_retry_old;
1245 } else if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001246 cFYI(1, "posix mkdir returned 0x%x", rc);
Steve French2dd29d32007-04-23 22:07:35 +00001247 d_drop(direntry);
1248 } else {
Cyril Gorcunov8f2376a2007-10-14 17:58:43 +00001249 if (pInfo->Type == cpu_to_le32(-1)) {
1250 /* no return info, go query for it */
Steve French5a07cdf2007-09-16 23:12:47 +00001251 kfree(pInfo);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001252 goto mkdir_get_info;
Steve French5a07cdf2007-09-16 23:12:47 +00001253 }
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001254/*BB check (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID ) to see if need
1255 to set uid/gid */
Steve French2dd29d32007-04-23 22:07:35 +00001256 inc_nlink(inode);
1257 if (pTcon->nocase)
1258 direntry->d_op = &cifs_ci_dentry_ops;
1259 else
1260 direntry->d_op = &cifs_dentry_ops;
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001261
Jeff Laytoncc0bad72009-06-25 00:56:52 -04001262 cifs_unix_basic_to_fattr(&fattr, pInfo, cifs_sb);
Jeff Layton4065c802010-05-17 07:18:58 -04001263 cifs_fill_uniqueid(inode->i_sb, &fattr);
Jeff Laytoncc0bad72009-06-25 00:56:52 -04001264 newinode = cifs_iget(inode->i_sb, &fattr);
1265 if (!newinode) {
Steve French5a07cdf2007-09-16 23:12:47 +00001266 kfree(pInfo);
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001267 goto mkdir_get_info;
Steve French5a07cdf2007-09-16 23:12:47 +00001268 }
Jeff Layton6b37faa2008-10-06 21:54:41 +00001269
Steve French2dd29d32007-04-23 22:07:35 +00001270 d_instantiate(direntry, newinode);
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001271
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001272#ifdef CONFIG_CIFS_DEBUG2
Joe Perchesb6b38f72010-04-21 03:50:45 +00001273 cFYI(1, "instantiated dentry %p %s to inode %p",
1274 direntry, direntry->d_name.name, newinode);
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001275
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001276 if (newinode->i_nlink != 2)
Joe Perchesb6b38f72010-04-21 03:50:45 +00001277 cFYI(1, "unexpected number of links %d",
1278 newinode->i_nlink);
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001279#endif
Steve French2dd29d32007-04-23 22:07:35 +00001280 }
1281 kfree(pInfo);
1282 goto mkdir_out;
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001283 }
Steve Frenchc45d7072007-09-17 02:04:21 +00001284mkdir_retry_old:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001285 /* BB add setting the equivalent of mode via CreateX w/ACLs */
Steve French737b7582005-04-28 22:41:06 -07001286 rc = CIFSSMBMkDir(xid, pTcon, full_path, cifs_sb->local_nls,
1287 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001288 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001289 cFYI(1, "cifs_mkdir returned 0x%x", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001290 d_drop(direntry);
1291 } else {
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001292mkdir_get_info:
Dave Hansend8c76e62006-09-30 23:29:04 -07001293 inc_nlink(inode);
Steve Frenchc18c8422007-07-18 23:21:09 +00001294 if (pTcon->unix_ext)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001295 rc = cifs_get_inode_info_unix(&newinode, full_path,
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001296 inode->i_sb, xid);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001297 else
1298 rc = cifs_get_inode_info(&newinode, full_path, NULL,
Steve French8b1327f2008-03-14 22:37:16 +00001299 inode->i_sb, xid, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001300
Steve Frenchb92327f2005-08-22 20:09:43 -07001301 if (pTcon->nocase)
1302 direntry->d_op = &cifs_ci_dentry_ops;
1303 else
1304 direntry->d_op = &cifs_dentry_ops;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001305 d_instantiate(direntry, newinode);
Steve French2dd29d32007-04-23 22:07:35 +00001306 /* setting nlink not necessary except in cases where we
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001307 * failed to get it from the server or was set bogus */
Steve French2dd29d32007-04-23 22:07:35 +00001308 if ((direntry->d_inode) && (direntry->d_inode->i_nlink < 2))
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001309 direntry->d_inode->i_nlink = 2;
Jeff Layton95089912008-08-06 04:39:02 +00001310
Al Viroce3b0f82009-03-29 19:08:22 -04001311 mode &= ~current_umask();
Jeff Layton95089912008-08-06 04:39:02 +00001312 /* must turn on setgid bit if parent dir has it */
1313 if (inode->i_mode & S_ISGID)
1314 mode |= S_ISGID;
1315
Steve Frenchc18c8422007-07-18 23:21:09 +00001316 if (pTcon->unix_ext) {
Jeff Layton4e1e7fb2008-08-02 07:26:12 -04001317 struct cifs_unix_set_info_args args = {
1318 .mode = mode,
1319 .ctime = NO_CHANGE_64,
1320 .atime = NO_CHANGE_64,
1321 .mtime = NO_CHANGE_64,
1322 .device = 0,
1323 };
Steve Frenchd0d2f2d2005-06-02 15:12:36 -07001324 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID) {
David Howellsa001e5b2008-11-14 10:38:47 +11001325 args.uid = (__u64)current_fsuid();
Jeff Layton95089912008-08-06 04:39:02 +00001326 if (inode->i_mode & S_ISGID)
1327 args.gid = (__u64)inode->i_gid;
1328 else
David Howellsa001e5b2008-11-14 10:38:47 +11001329 args.gid = (__u64)current_fsgid();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001330 } else {
Jeff Layton4e1e7fb2008-08-02 07:26:12 -04001331 args.uid = NO_CHANGE_64;
1332 args.gid = NO_CHANGE_64;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001333 }
Jeff Layton01ea95e2009-07-09 20:02:49 -04001334 CIFSSMBUnixSetPathInfo(xid, pTcon, full_path, &args,
1335 cifs_sb->local_nls,
1336 cifs_sb->mnt_cifs_flags &
1337 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French3ce53fc2007-06-08 14:55:14 +00001338 } else {
Jeff Layton67750fb2008-05-09 22:28:02 +00001339 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL) &&
1340 (mode & S_IWUGO) == 0) {
1341 FILE_BASIC_INFO pInfo;
Jeff Layton6b37faa2008-10-06 21:54:41 +00001342 struct cifsInodeInfo *cifsInode;
1343 u32 dosattrs;
1344
Jeff Layton67750fb2008-05-09 22:28:02 +00001345 memset(&pInfo, 0, sizeof(pInfo));
Jeff Layton6b37faa2008-10-06 21:54:41 +00001346 cifsInode = CIFS_I(newinode);
1347 dosattrs = cifsInode->cifsAttrs|ATTR_READONLY;
1348 pInfo.Attributes = cpu_to_le32(dosattrs);
1349 tmprc = CIFSSMBSetPathInfo(xid, pTcon,
1350 full_path, &pInfo,
1351 cifs_sb->local_nls,
Jeff Layton67750fb2008-05-09 22:28:02 +00001352 cifs_sb->mnt_cifs_flags &
1353 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Layton6b37faa2008-10-06 21:54:41 +00001354 if (tmprc == 0)
1355 cifsInode->cifsAttrs = dosattrs;
Jeff Layton67750fb2008-05-09 22:28:02 +00001356 }
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001357 if (direntry->d_inode) {
Jeff Laytonb0fd30d2008-05-22 09:33:34 -04001358 if (cifs_sb->mnt_cifs_flags &
1359 CIFS_MOUNT_DYNPERM)
1360 direntry->d_inode->i_mode =
1361 (mode | S_IFDIR);
Steve French4e94a102008-05-23 18:22:46 +00001362
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001363 if (cifs_sb->mnt_cifs_flags &
Steve French6473a552005-11-29 20:20:10 -08001364 CIFS_MOUNT_SET_UID) {
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001365 direntry->d_inode->i_uid =
David Howellsa001e5b2008-11-14 10:38:47 +11001366 current_fsuid();
Jeff Layton95089912008-08-06 04:39:02 +00001367 if (inode->i_mode & S_ISGID)
1368 direntry->d_inode->i_gid =
1369 inode->i_gid;
1370 else
1371 direntry->d_inode->i_gid =
David Howellsa001e5b2008-11-14 10:38:47 +11001372 current_fsgid();
Steve French6473a552005-11-29 20:20:10 -08001373 }
1374 }
Steve French2a138ebb2005-11-29 21:22:19 -08001375 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001376 }
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001377mkdir_out:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001378 kfree(full_path);
1379 FreeXid(xid);
1380 return rc;
1381}
1382
1383int cifs_rmdir(struct inode *inode, struct dentry *direntry)
1384{
1385 int rc = 0;
1386 int xid;
1387 struct cifs_sb_info *cifs_sb;
1388 struct cifsTconInfo *pTcon;
1389 char *full_path = NULL;
1390 struct cifsInodeInfo *cifsInode;
1391
Joe Perchesb6b38f72010-04-21 03:50:45 +00001392 cFYI(1, "cifs_rmdir, inode = 0x%p", inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001393
1394 xid = GetXid();
1395
1396 cifs_sb = CIFS_SB(inode->i_sb);
1397 pTcon = cifs_sb->tcon;
1398
Steve French7f573562005-08-30 11:32:14 -07001399 full_path = build_path_from_dentry(direntry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001400 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301401 rc = -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001402 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301403 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001404 }
1405
Steve French737b7582005-04-28 22:41:06 -07001406 rc = CIFSSMBRmDir(xid, pTcon, full_path, cifs_sb->local_nls,
1407 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001408
1409 if (!rc) {
Dave Hansen9a53c3a2006-09-30 23:29:03 -07001410 drop_nlink(inode);
Steve French3677db12007-02-26 16:46:11 +00001411 spin_lock(&direntry->d_inode->i_lock);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001412 i_size_write(direntry->d_inode, 0);
Dave Hansence71ec32006-09-30 23:29:06 -07001413 clear_nlink(direntry->d_inode);
Steve French3677db12007-02-26 16:46:11 +00001414 spin_unlock(&direntry->d_inode->i_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001415 }
1416
1417 cifsInode = CIFS_I(direntry->d_inode);
1418 cifsInode->time = 0; /* force revalidate to go get info when
1419 needed */
Steve French42c245442009-01-13 22:03:55 +00001420
1421 cifsInode = CIFS_I(inode);
1422 cifsInode->time = 0; /* force revalidate to get parent dir info
1423 since cached search results now invalid */
1424
Linus Torvalds1da177e2005-04-16 15:20:36 -07001425 direntry->d_inode->i_ctime = inode->i_ctime = inode->i_mtime =
1426 current_fs_time(inode->i_sb);
1427
1428 kfree(full_path);
1429 FreeXid(xid);
1430 return rc;
1431}
1432
Steve Frenchee2fd962008-09-23 18:23:33 +00001433static int
1434cifs_do_rename(int xid, struct dentry *from_dentry, const char *fromPath,
1435 struct dentry *to_dentry, const char *toPath)
1436{
1437 struct cifs_sb_info *cifs_sb = CIFS_SB(from_dentry->d_sb);
1438 struct cifsTconInfo *pTcon = cifs_sb->tcon;
1439 __u16 srcfid;
1440 int oplock, rc;
1441
1442 /* try path-based rename first */
1443 rc = CIFSSMBRename(xid, pTcon, fromPath, toPath, cifs_sb->local_nls,
1444 cifs_sb->mnt_cifs_flags &
1445 CIFS_MOUNT_MAP_SPECIAL_CHR);
1446
1447 /*
1448 * don't bother with rename by filehandle unless file is busy and
1449 * source Note that cross directory moves do not work with
1450 * rename by filehandle to various Windows servers.
1451 */
1452 if (rc == 0 || rc != -ETXTBSY)
1453 return rc;
1454
Jeff Laytoned0e3ac2010-06-01 16:21:01 -04001455 /* open-file renames don't work across directories */
1456 if (to_dentry->d_parent != from_dentry->d_parent)
1457 return rc;
1458
Steve Frenchee2fd962008-09-23 18:23:33 +00001459 /* open the file to be renamed -- we need DELETE perms */
1460 rc = CIFSSMBOpen(xid, pTcon, fromPath, FILE_OPEN, DELETE,
1461 CREATE_NOT_DIR, &srcfid, &oplock, NULL,
1462 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1463 CIFS_MOUNT_MAP_SPECIAL_CHR);
1464
1465 if (rc == 0) {
1466 rc = CIFSSMBRenameOpenFile(xid, pTcon, srcfid,
1467 (const char *) to_dentry->d_name.name,
1468 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1469 CIFS_MOUNT_MAP_SPECIAL_CHR);
1470
1471 CIFSSMBClose(xid, pTcon, srcfid);
1472 }
1473
1474 return rc;
1475}
1476
Jeff Layton14121bd2008-10-20 14:45:22 -04001477int cifs_rename(struct inode *source_dir, struct dentry *source_dentry,
1478 struct inode *target_dir, struct dentry *target_dentry)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001479{
Steve Frenchee2fd962008-09-23 18:23:33 +00001480 char *fromName = NULL;
1481 char *toName = NULL;
Jeff Layton639e7a92010-09-03 11:50:09 -04001482 struct cifs_sb_info *cifs_sb;
Jeff Layton14121bd2008-10-20 14:45:22 -04001483 struct cifsTconInfo *tcon;
Steve Frenchee2fd962008-09-23 18:23:33 +00001484 FILE_UNIX_BASIC_INFO *info_buf_source = NULL;
1485 FILE_UNIX_BASIC_INFO *info_buf_target;
Jeff Layton8d281ef2008-10-22 13:57:01 -04001486 int xid, rc, tmprc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001487
Jeff Layton639e7a92010-09-03 11:50:09 -04001488 cifs_sb = CIFS_SB(source_dir->i_sb);
1489 tcon = cifs_sb->tcon;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001490
Steve Frenchee2fd962008-09-23 18:23:33 +00001491 xid = GetXid();
1492
1493 /*
Steve Frenchee2fd962008-09-23 18:23:33 +00001494 * we already have the rename sem so we do not need to
1495 * grab it again here to protect the path integrity
1496 */
Jeff Layton14121bd2008-10-20 14:45:22 -04001497 fromName = build_path_from_dentry(source_dentry);
Steve Frenchee2fd962008-09-23 18:23:33 +00001498 if (fromName == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001499 rc = -ENOMEM;
1500 goto cifs_rename_exit;
1501 }
1502
Jeff Layton14121bd2008-10-20 14:45:22 -04001503 toName = build_path_from_dentry(target_dentry);
Steve Frenchee2fd962008-09-23 18:23:33 +00001504 if (toName == NULL) {
1505 rc = -ENOMEM;
1506 goto cifs_rename_exit;
1507 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001508
Jeff Layton14121bd2008-10-20 14:45:22 -04001509 rc = cifs_do_rename(xid, source_dentry, fromName,
1510 target_dentry, toName);
Steve Frenchee2fd962008-09-23 18:23:33 +00001511
Jeff Layton14121bd2008-10-20 14:45:22 -04001512 if (rc == -EEXIST && tcon->unix_ext) {
Steve Frenchee2fd962008-09-23 18:23:33 +00001513 /*
Jeff Layton14121bd2008-10-20 14:45:22 -04001514 * Are src and dst hardlinks of same inode? We can
1515 * only tell with unix extensions enabled
Steve Frenchee2fd962008-09-23 18:23:33 +00001516 */
Jeff Layton14121bd2008-10-20 14:45:22 -04001517 info_buf_source =
1518 kmalloc(2 * sizeof(FILE_UNIX_BASIC_INFO),
1519 GFP_KERNEL);
1520 if (info_buf_source == NULL) {
1521 rc = -ENOMEM;
1522 goto cifs_rename_exit;
1523 }
1524
1525 info_buf_target = info_buf_source + 1;
Jeff Layton8d281ef2008-10-22 13:57:01 -04001526 tmprc = CIFSSMBUnixQPathInfo(xid, tcon, fromName,
Jeff Layton14121bd2008-10-20 14:45:22 -04001527 info_buf_source,
Jeff Layton639e7a92010-09-03 11:50:09 -04001528 cifs_sb->local_nls,
1529 cifs_sb->mnt_cifs_flags &
Jeff Layton14121bd2008-10-20 14:45:22 -04001530 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Layton8d281ef2008-10-22 13:57:01 -04001531 if (tmprc != 0)
Jeff Layton14121bd2008-10-20 14:45:22 -04001532 goto unlink_target;
1533
Jeff Layton639e7a92010-09-03 11:50:09 -04001534 tmprc = CIFSSMBUnixQPathInfo(xid, tcon, toName,
1535 info_buf_target,
1536 cifs_sb->local_nls,
1537 cifs_sb->mnt_cifs_flags &
Jeff Layton14121bd2008-10-20 14:45:22 -04001538 CIFS_MOUNT_MAP_SPECIAL_CHR);
1539
Jeff Layton8d281ef2008-10-22 13:57:01 -04001540 if (tmprc == 0 && (info_buf_source->UniqueId ==
Jeff Laytonae6884a2008-11-03 14:05:08 -05001541 info_buf_target->UniqueId)) {
Jeff Layton14121bd2008-10-20 14:45:22 -04001542 /* same file, POSIX says that this is a noop */
Jeff Laytonae6884a2008-11-03 14:05:08 -05001543 rc = 0;
Jeff Layton14121bd2008-10-20 14:45:22 -04001544 goto cifs_rename_exit;
Jeff Laytonae6884a2008-11-03 14:05:08 -05001545 }
Jeff Layton14121bd2008-10-20 14:45:22 -04001546 } /* else ... BB we could add the same check for Windows by
1547 checking the UniqueId via FILE_INTERNAL_INFO */
1548
Jeff Layton14121bd2008-10-20 14:45:22 -04001549unlink_target:
Jeff Laytonfc6f3942009-04-17 11:45:30 -04001550 /* Try unlinking the target dentry if it's not negative */
1551 if (target_dentry->d_inode && (rc == -EACCES || rc == -EEXIST)) {
Jeff Layton8d281ef2008-10-22 13:57:01 -04001552 tmprc = cifs_unlink(target_dir, target_dentry);
Jeff Layton14121bd2008-10-20 14:45:22 -04001553 if (tmprc)
1554 goto cifs_rename_exit;
1555
Jeff Layton14121bd2008-10-20 14:45:22 -04001556 rc = cifs_do_rename(xid, source_dentry, fromName,
1557 target_dentry, toName);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001558 }
1559
1560cifs_rename_exit:
Steve Frenchee2fd962008-09-23 18:23:33 +00001561 kfree(info_buf_source);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001562 kfree(fromName);
1563 kfree(toName);
1564 FreeXid(xid);
1565 return rc;
1566}
1567
Jeff Laytondf2cf172010-02-12 07:44:16 -05001568static bool
1569cifs_inode_needs_reval(struct inode *inode)
1570{
1571 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
1572
1573 if (cifs_i->clientCanCacheRead)
1574 return false;
1575
1576 if (!lookupCacheEnabled)
1577 return true;
1578
1579 if (cifs_i->time == 0)
1580 return true;
1581
1582 /* FIXME: the actimeo should be tunable */
1583 if (time_after_eq(jiffies, cifs_i->time + HZ))
1584 return true;
1585
Jeff Laytondb192722010-05-17 14:51:49 -04001586 /* hardlinked files w/ noserverino get "special" treatment */
1587 if (!(CIFS_SB(inode->i_sb)->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM) &&
1588 S_ISREG(inode->i_mode) && inode->i_nlink != 1)
1589 return true;
1590
Jeff Laytondf2cf172010-02-12 07:44:16 -05001591 return false;
1592}
1593
1594/* check invalid_mapping flag and zap the cache if it's set */
1595static void
1596cifs_invalidate_mapping(struct inode *inode)
1597{
1598 int rc;
1599 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
1600
1601 cifs_i->invalid_mapping = false;
1602
1603 /* write back any cached data */
1604 if (inode->i_mapping && inode->i_mapping->nrpages != 0) {
1605 rc = filemap_write_and_wait(inode->i_mapping);
1606 if (rc)
1607 cifs_i->write_behind_rc = rc;
1608 }
1609 invalidate_remote_inode(inode);
Suresh Jayaraman9451a9a2010-07-05 18:12:45 +05301610 cifs_fscache_reset_inode_cookie(inode);
Jeff Laytondf2cf172010-02-12 07:44:16 -05001611}
1612
Jeff Laytonabab0952010-02-12 07:44:18 -05001613int cifs_revalidate_file(struct file *filp)
1614{
1615 int rc = 0;
1616 struct inode *inode = filp->f_path.dentry->d_inode;
Jeff Laytonba00ba62010-09-20 16:01:31 -07001617 struct cifsFileInfo *cfile = (struct cifsFileInfo *) filp->private_data;
Jeff Laytonabab0952010-02-12 07:44:18 -05001618
1619 if (!cifs_inode_needs_reval(inode))
1620 goto check_inval;
1621
Jeff Laytonba00ba62010-09-20 16:01:31 -07001622 if (cfile->tcon->unix_ext)
Jeff Laytonabab0952010-02-12 07:44:18 -05001623 rc = cifs_get_file_info_unix(filp);
1624 else
1625 rc = cifs_get_file_info(filp);
1626
1627check_inval:
1628 if (CIFS_I(inode)->invalid_mapping)
1629 cifs_invalidate_mapping(inode);
1630
1631 return rc;
1632}
1633
Jeff Laytondf2cf172010-02-12 07:44:16 -05001634/* revalidate a dentry's inode attributes */
1635int cifs_revalidate_dentry(struct dentry *dentry)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001636{
1637 int xid;
Jeff Laytondf2cf172010-02-12 07:44:16 -05001638 int rc = 0;
1639 char *full_path = NULL;
1640 struct inode *inode = dentry->d_inode;
1641 struct super_block *sb = dentry->d_sb;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001642
Jeff Laytondf2cf172010-02-12 07:44:16 -05001643 if (inode == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001644 return -ENOENT;
1645
Linus Torvalds1da177e2005-04-16 15:20:36 -07001646 xid = GetXid();
1647
Jeff Laytondf2cf172010-02-12 07:44:16 -05001648 if (!cifs_inode_needs_reval(inode))
1649 goto check_inval;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001650
1651 /* can not safely grab the rename sem here if rename calls revalidate
1652 since that would deadlock */
Jeff Laytondf2cf172010-02-12 07:44:16 -05001653 full_path = build_path_from_dentry(dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001654 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301655 rc = -ENOMEM;
Jeff Laytondf2cf172010-02-12 07:44:16 -05001656 goto check_inval;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001657 }
Jeff Laytondf2cf172010-02-12 07:44:16 -05001658
Steve Frenchf19159d2010-04-21 04:12:10 +00001659 cFYI(1, "Revalidate: %s inode 0x%p count %d dentry: 0x%p d_time %ld "
Jeff Laytondf2cf172010-02-12 07:44:16 -05001660 "jiffies %ld", full_path, inode, inode->i_count.counter,
Steve Frenchf19159d2010-04-21 04:12:10 +00001661 dentry, dentry->d_time, jiffies);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001662
Jeff Laytondf2cf172010-02-12 07:44:16 -05001663 if (CIFS_SB(sb)->tcon->unix_ext)
1664 rc = cifs_get_inode_info_unix(&inode, full_path, sb, xid);
1665 else
1666 rc = cifs_get_inode_info(&inode, full_path, NULL, sb,
1667 xid, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001668
Jeff Laytondf2cf172010-02-12 07:44:16 -05001669check_inval:
1670 if (CIFS_I(inode)->invalid_mapping)
1671 cifs_invalidate_mapping(inode);
Steve French50c2f752007-07-13 00:33:32 +00001672
Linus Torvalds1da177e2005-04-16 15:20:36 -07001673 kfree(full_path);
1674 FreeXid(xid);
1675 return rc;
1676}
1677
1678int cifs_getattr(struct vfsmount *mnt, struct dentry *dentry,
1679 struct kstat *stat)
1680{
Jeff Laytondf2cf172010-02-12 07:44:16 -05001681 int err = cifs_revalidate_dentry(dentry);
Steve French5fe14c82006-11-07 19:26:33 +00001682 if (!err) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001683 generic_fillattr(dentry->d_inode, stat);
Steve French5fe14c82006-11-07 19:26:33 +00001684 stat->blksize = CIFS_MAX_MSGSIZE;
Jeff Laytoncc0bad72009-06-25 00:56:52 -04001685 stat->ino = CIFS_I(dentry->d_inode)->uniqueid;
Steve French5fe14c82006-11-07 19:26:33 +00001686 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001687 return err;
1688}
1689
1690static int cifs_truncate_page(struct address_space *mapping, loff_t from)
1691{
1692 pgoff_t index = from >> PAGE_CACHE_SHIFT;
1693 unsigned offset = from & (PAGE_CACHE_SIZE - 1);
1694 struct page *page;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001695 int rc = 0;
1696
1697 page = grab_cache_page(mapping, index);
1698 if (!page)
1699 return -ENOMEM;
1700
Christoph Lametereebd2aa2008-02-04 22:28:29 -08001701 zero_user_segment(page, offset, PAGE_CACHE_SIZE);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001702 unlock_page(page);
1703 page_cache_release(page);
1704 return rc;
1705}
1706
Christoph Hellwig1b947462010-07-18 17:51:21 -04001707static void cifs_setsize(struct inode *inode, loff_t offset)
Steve French3677db12007-02-26 16:46:11 +00001708{
npiggin@suse.dec08d3b02009-08-21 02:35:06 +10001709 loff_t oldsize;
Steve French3677db12007-02-26 16:46:11 +00001710
Steve Frenchba6a46a2007-02-26 20:06:29 +00001711 spin_lock(&inode->i_lock);
npiggin@suse.dec08d3b02009-08-21 02:35:06 +10001712 oldsize = inode->i_size;
Steve French3677db12007-02-26 16:46:11 +00001713 i_size_write(inode, offset);
Steve Frenchba6a46a2007-02-26 20:06:29 +00001714 spin_unlock(&inode->i_lock);
Christoph Hellwig1b947462010-07-18 17:51:21 -04001715
npiggin@suse.dec08d3b02009-08-21 02:35:06 +10001716 truncate_pagecache(inode, oldsize, offset);
Steve French3677db12007-02-26 16:46:11 +00001717}
1718
Jeff Layton8efdbde2008-07-23 21:28:12 +00001719static int
1720cifs_set_file_size(struct inode *inode, struct iattr *attrs,
1721 int xid, char *full_path)
1722{
1723 int rc;
1724 struct cifsFileInfo *open_file;
1725 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
1726 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
Jeff Laytonba00ba62010-09-20 16:01:31 -07001727 struct cifsTconInfo *pTcon = NULL;
Jeff Layton8efdbde2008-07-23 21:28:12 +00001728
1729 /*
1730 * To avoid spurious oplock breaks from server, in the case of
1731 * inodes that we already have open, avoid doing path based
1732 * setting of file size if we can do it by handle.
1733 * This keeps our caching token (oplock) and avoids timeouts
1734 * when the local oplock break takes longer to flush
1735 * writebehind data than the SMB timeout for the SetPathInfo
1736 * request would allow
1737 */
1738 open_file = find_writable_file(cifsInode);
1739 if (open_file) {
1740 __u16 nfid = open_file->netfid;
1741 __u32 npid = open_file->pid;
Jeff Laytonba00ba62010-09-20 16:01:31 -07001742 pTcon = open_file->tcon;
Jeff Layton8efdbde2008-07-23 21:28:12 +00001743 rc = CIFSSMBSetFileSize(xid, pTcon, attrs->ia_size, nfid,
1744 npid, false);
Dave Kleikamp6ab409b2009-08-31 11:07:12 -04001745 cifsFileInfo_put(open_file);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001746 cFYI(1, "SetFSize for attrs rc = %d", rc);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001747 if ((rc == -EINVAL) || (rc == -EOPNOTSUPP)) {
1748 unsigned int bytes_written;
1749 rc = CIFSSMBWrite(xid, pTcon, nfid, 0, attrs->ia_size,
1750 &bytes_written, NULL, NULL, 1);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001751 cFYI(1, "Wrt seteof rc %d", rc);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001752 }
1753 } else
1754 rc = -EINVAL;
1755
1756 if (rc != 0) {
Jeff Laytonba00ba62010-09-20 16:01:31 -07001757 if (pTcon == NULL)
1758 pTcon = cifs_sb->tcon;
1759
Jeff Layton8efdbde2008-07-23 21:28:12 +00001760 /* Set file size by pathname rather than by handle
1761 either because no valid, writeable file handle for
1762 it was found or because there was an error setting
1763 it by handle */
1764 rc = CIFSSMBSetEOF(xid, pTcon, full_path, attrs->ia_size,
1765 false, cifs_sb->local_nls,
1766 cifs_sb->mnt_cifs_flags &
1767 CIFS_MOUNT_MAP_SPECIAL_CHR);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001768 cFYI(1, "SetEOF by path (setattrs) rc = %d", rc);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001769 if ((rc == -EINVAL) || (rc == -EOPNOTSUPP)) {
1770 __u16 netfid;
1771 int oplock = 0;
1772
1773 rc = SMBLegacyOpen(xid, pTcon, full_path,
1774 FILE_OPEN, GENERIC_WRITE,
1775 CREATE_NOT_DIR, &netfid, &oplock, NULL,
1776 cifs_sb->local_nls,
1777 cifs_sb->mnt_cifs_flags &
1778 CIFS_MOUNT_MAP_SPECIAL_CHR);
1779 if (rc == 0) {
1780 unsigned int bytes_written;
1781 rc = CIFSSMBWrite(xid, pTcon, netfid, 0,
1782 attrs->ia_size,
1783 &bytes_written, NULL,
1784 NULL, 1);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001785 cFYI(1, "wrt seteof rc %d", rc);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001786 CIFSSMBClose(xid, pTcon, netfid);
1787 }
1788 }
1789 }
1790
1791 if (rc == 0) {
Jeff Laytonfbec9ab2009-04-03 13:44:00 -04001792 cifsInode->server_eof = attrs->ia_size;
Christoph Hellwig1b947462010-07-18 17:51:21 -04001793 cifs_setsize(inode, attrs->ia_size);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001794 cifs_truncate_page(inode->i_mapping, inode->i_size);
1795 }
1796
1797 return rc;
1798}
1799
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001800static int
1801cifs_setattr_unix(struct dentry *direntry, struct iattr *attrs)
1802{
1803 int rc;
1804 int xid;
1805 char *full_path = NULL;
1806 struct inode *inode = direntry->d_inode;
1807 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
1808 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
Jeff Laytonba00ba62010-09-20 16:01:31 -07001809 struct cifsTconInfo *pTcon;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001810 struct cifs_unix_set_info_args *args = NULL;
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001811 struct cifsFileInfo *open_file;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001812
Joe Perchesb6b38f72010-04-21 03:50:45 +00001813 cFYI(1, "setattr_unix on file %s attrs->ia_valid=0x%x",
1814 direntry->d_name.name, attrs->ia_valid);
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001815
1816 xid = GetXid();
1817
Christoph Hellwigdb78b872010-06-04 11:30:03 +02001818 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_PERM)
1819 attrs->ia_valid |= ATTR_FORCE;
1820
1821 rc = inode_change_ok(inode, attrs);
1822 if (rc < 0)
1823 goto out;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001824
1825 full_path = build_path_from_dentry(direntry);
1826 if (full_path == NULL) {
1827 rc = -ENOMEM;
1828 goto out;
1829 }
1830
Jeff Layton0f4d6342009-03-26 13:35:37 -04001831 /*
1832 * Attempt to flush data before changing attributes. We need to do
1833 * this for ATTR_SIZE and ATTR_MTIME for sure, and if we change the
1834 * ownership or mode then we may also need to do this. Here, we take
1835 * the safe way out and just do the flush on all setattr requests. If
1836 * the flush returns error, store it to report later and continue.
1837 *
1838 * BB: This should be smarter. Why bother flushing pages that
1839 * will be truncated anyway? Also, should we error out here if
1840 * the flush returns error?
1841 */
1842 rc = filemap_write_and_wait(inode->i_mapping);
1843 if (rc != 0) {
1844 cifsInode->write_behind_rc = rc;
1845 rc = 0;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001846 }
1847
1848 if (attrs->ia_valid & ATTR_SIZE) {
1849 rc = cifs_set_file_size(inode, attrs, xid, full_path);
1850 if (rc != 0)
1851 goto out;
1852 }
1853
1854 /* skip mode change if it's just for clearing setuid/setgid */
1855 if (attrs->ia_valid & (ATTR_KILL_SUID|ATTR_KILL_SGID))
1856 attrs->ia_valid &= ~ATTR_MODE;
1857
1858 args = kmalloc(sizeof(*args), GFP_KERNEL);
1859 if (args == NULL) {
1860 rc = -ENOMEM;
1861 goto out;
1862 }
1863
1864 /* set up the struct */
1865 if (attrs->ia_valid & ATTR_MODE)
1866 args->mode = attrs->ia_mode;
1867 else
1868 args->mode = NO_CHANGE_64;
1869
1870 if (attrs->ia_valid & ATTR_UID)
1871 args->uid = attrs->ia_uid;
1872 else
1873 args->uid = NO_CHANGE_64;
1874
1875 if (attrs->ia_valid & ATTR_GID)
1876 args->gid = attrs->ia_gid;
1877 else
1878 args->gid = NO_CHANGE_64;
1879
1880 if (attrs->ia_valid & ATTR_ATIME)
1881 args->atime = cifs_UnixTimeToNT(attrs->ia_atime);
1882 else
1883 args->atime = NO_CHANGE_64;
1884
1885 if (attrs->ia_valid & ATTR_MTIME)
1886 args->mtime = cifs_UnixTimeToNT(attrs->ia_mtime);
1887 else
1888 args->mtime = NO_CHANGE_64;
1889
1890 if (attrs->ia_valid & ATTR_CTIME)
1891 args->ctime = cifs_UnixTimeToNT(attrs->ia_ctime);
1892 else
1893 args->ctime = NO_CHANGE_64;
1894
1895 args->device = 0;
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001896 open_file = find_writable_file(cifsInode);
1897 if (open_file) {
1898 u16 nfid = open_file->netfid;
1899 u32 npid = open_file->pid;
Jeff Laytonba00ba62010-09-20 16:01:31 -07001900 pTcon = open_file->tcon;
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001901 rc = CIFSSMBUnixSetFileInfo(xid, pTcon, args, nfid, npid);
Dave Kleikamp6ab409b2009-08-31 11:07:12 -04001902 cifsFileInfo_put(open_file);
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001903 } else {
Jeff Laytonba00ba62010-09-20 16:01:31 -07001904 pTcon = cifs_sb->tcon;
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001905 rc = CIFSSMBUnixSetPathInfo(xid, pTcon, full_path, args,
Jeff Layton01ea95e2009-07-09 20:02:49 -04001906 cifs_sb->local_nls,
1907 cifs_sb->mnt_cifs_flags &
1908 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001909 }
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001910
Christoph Hellwig10257742010-06-04 11:30:02 +02001911 if (rc)
1912 goto out;
Steve Frenchccd4bb12010-02-08 17:39:58 +00001913
Christoph Hellwig10257742010-06-04 11:30:02 +02001914 if ((attrs->ia_valid & ATTR_SIZE) &&
Christoph Hellwig1b947462010-07-18 17:51:21 -04001915 attrs->ia_size != i_size_read(inode))
1916 truncate_setsize(inode, attrs->ia_size);
Christoph Hellwig10257742010-06-04 11:30:02 +02001917
1918 setattr_copy(inode, attrs);
1919 mark_inode_dirty(inode);
1920
1921 /* force revalidate when any of these times are set since some
1922 of the fs types (eg ext3, fat) do not have fine enough
1923 time granularity to match protocol, and we do not have a
1924 a way (yet) to query the server fs's time granularity (and
1925 whether it rounds times down).
1926 */
1927 if (attrs->ia_valid & (ATTR_MTIME | ATTR_CTIME))
1928 cifsInode->time = 0;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001929out:
1930 kfree(args);
1931 kfree(full_path);
1932 FreeXid(xid);
1933 return rc;
1934}
1935
Jeff Layton0510eeb2008-08-02 07:26:12 -04001936static int
1937cifs_setattr_nounix(struct dentry *direntry, struct iattr *attrs)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001938{
1939 int xid;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001940 struct inode *inode = direntry->d_inode;
1941 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001942 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001943 char *full_path = NULL;
1944 int rc = -EACCES;
Jeff Laytonfeb3e202008-08-02 07:26:12 -04001945 __u32 dosattr = 0;
Jeff Layton4e1e7fb2008-08-02 07:26:12 -04001946 __u64 mode = NO_CHANGE_64;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001947
Linus Torvalds1da177e2005-04-16 15:20:36 -07001948 xid = GetXid();
1949
Joe Perchesb6b38f72010-04-21 03:50:45 +00001950 cFYI(1, "setattr on file %s attrs->iavalid 0x%x",
1951 direntry->d_name.name, attrs->ia_valid);
Steve French6473a552005-11-29 20:20:10 -08001952
Christoph Hellwigdb78b872010-06-04 11:30:03 +02001953 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_PERM)
1954 attrs->ia_valid |= ATTR_FORCE;
1955
1956 rc = inode_change_ok(inode, attrs);
1957 if (rc < 0) {
1958 FreeXid(xid);
1959 return rc;
Steve French6473a552005-11-29 20:20:10 -08001960 }
Steve French50c2f752007-07-13 00:33:32 +00001961
Steve French7f573562005-08-30 11:32:14 -07001962 full_path = build_path_from_dentry(direntry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001963 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301964 rc = -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001965 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301966 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001967 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001968
Jeff Layton0f4d6342009-03-26 13:35:37 -04001969 /*
1970 * Attempt to flush data before changing attributes. We need to do
1971 * this for ATTR_SIZE and ATTR_MTIME for sure, and if we change the
1972 * ownership or mode then we may also need to do this. Here, we take
1973 * the safe way out and just do the flush on all setattr requests. If
1974 * the flush returns error, store it to report later and continue.
1975 *
1976 * BB: This should be smarter. Why bother flushing pages that
1977 * will be truncated anyway? Also, should we error out here if
1978 * the flush returns error?
1979 */
1980 rc = filemap_write_and_wait(inode->i_mapping);
1981 if (rc != 0) {
1982 cifsInode->write_behind_rc = rc;
1983 rc = 0;
Steve French50531442008-03-14 19:21:31 +00001984 }
Jeff Laytoncea21802007-11-20 23:19:03 +00001985
Steve French50531442008-03-14 19:21:31 +00001986 if (attrs->ia_valid & ATTR_SIZE) {
Jeff Layton8efdbde2008-07-23 21:28:12 +00001987 rc = cifs_set_file_size(inode, attrs, xid, full_path);
1988 if (rc != 0)
Steve Frenche30dcf32005-09-20 20:49:16 -07001989 goto cifs_setattr_exit;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001990 }
Jeff Layton4ca691a2008-05-22 09:33:34 -04001991
1992 /*
1993 * Without unix extensions we can't send ownership changes to the
1994 * server, so silently ignore them. This is consistent with how
1995 * local DOS/Windows filesystems behave (VFAT, NTFS, etc). With
1996 * CIFSACL support + proper Windows to Unix idmapping, we may be
1997 * able to support this in the future.
1998 */
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001999 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID))
Jeff Layton4ca691a2008-05-22 09:33:34 -04002000 attrs->ia_valid &= ~(ATTR_UID | ATTR_GID);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002001
Jeff Laytond32c4f22007-10-18 03:05:22 -07002002 /* skip mode change if it's just for clearing setuid/setgid */
2003 if (attrs->ia_valid & (ATTR_KILL_SUID|ATTR_KILL_SGID))
2004 attrs->ia_valid &= ~ATTR_MODE;
2005
Linus Torvalds1da177e2005-04-16 15:20:36 -07002006 if (attrs->ia_valid & ATTR_MODE) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00002007 cFYI(1, "Mode changed to 0%o", attrs->ia_mode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002008 mode = attrs->ia_mode;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002009 }
2010
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04002011 if (attrs->ia_valid & ATTR_MODE) {
Steve Frenchcdbce9c2005-11-19 21:04:52 -08002012 rc = 0;
Steve French97837582007-12-31 07:47:21 +00002013#ifdef CONFIG_CIFS_EXPERIMENTAL
2014 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL)
Jeff Layton02eadef2008-05-09 21:26:11 +00002015 rc = mode_to_acl(inode, full_path, mode);
Jeff Layton51328612008-05-22 09:33:34 -04002016 else
Steve French97837582007-12-31 07:47:21 +00002017#endif
Jeff Layton51328612008-05-22 09:33:34 -04002018 if (((mode & S_IWUGO) == 0) &&
2019 (cifsInode->cifsAttrs & ATTR_READONLY) == 0) {
Jeff Laytonfeb3e202008-08-02 07:26:12 -04002020
2021 dosattr = cifsInode->cifsAttrs | ATTR_READONLY;
2022
Jeff Layton51328612008-05-22 09:33:34 -04002023 /* fix up mode if we're not using dynperm */
2024 if ((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM) == 0)
2025 attrs->ia_mode = inode->i_mode & ~S_IWUGO;
2026 } else if ((mode & S_IWUGO) &&
2027 (cifsInode->cifsAttrs & ATTR_READONLY)) {
Jeff Laytonfeb3e202008-08-02 07:26:12 -04002028
2029 dosattr = cifsInode->cifsAttrs & ~ATTR_READONLY;
2030 /* Attributes of 0 are ignored */
2031 if (dosattr == 0)
2032 dosattr |= ATTR_NORMAL;
Jeff Layton51328612008-05-22 09:33:34 -04002033
2034 /* reset local inode permissions to normal */
2035 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM)) {
2036 attrs->ia_mode &= ~(S_IALLUGO);
2037 if (S_ISDIR(inode->i_mode))
2038 attrs->ia_mode |=
2039 cifs_sb->mnt_dir_mode;
2040 else
2041 attrs->ia_mode |=
2042 cifs_sb->mnt_file_mode;
2043 }
2044 } else if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM)) {
2045 /* ignore mode change - ATTR_READONLY hasn't changed */
2046 attrs->ia_valid &= ~ATTR_MODE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002047 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002048 }
2049
Jeff Laytonfeb3e202008-08-02 07:26:12 -04002050 if (attrs->ia_valid & (ATTR_MTIME|ATTR_ATIME|ATTR_CTIME) ||
2051 ((attrs->ia_valid & ATTR_MODE) && dosattr)) {
2052 rc = cifs_set_file_info(inode, attrs, xid, full_path, dosattr);
2053 /* BB: check for rc = -EOPNOTSUPP and switch to legacy mode */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002054
Steve Frenche30dcf32005-09-20 20:49:16 -07002055 /* Even if error on time set, no sense failing the call if
2056 the server would set the time to a reasonable value anyway,
2057 and this check ensures that we are not being called from
2058 sys_utimes in which case we ought to fail the call back to
2059 the user when the server rejects the call */
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002060 if ((rc) && (attrs->ia_valid &
Jeff Laytonfeb3e202008-08-02 07:26:12 -04002061 (ATTR_MODE | ATTR_GID | ATTR_UID | ATTR_SIZE)))
Steve Frenche30dcf32005-09-20 20:49:16 -07002062 rc = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002063 }
2064
2065 /* do not need local check to inode_check_ok since the server does
2066 that */
Christoph Hellwig10257742010-06-04 11:30:02 +02002067 if (rc)
2068 goto cifs_setattr_exit;
2069
2070 if ((attrs->ia_valid & ATTR_SIZE) &&
Christoph Hellwig1b947462010-07-18 17:51:21 -04002071 attrs->ia_size != i_size_read(inode))
2072 truncate_setsize(inode, attrs->ia_size);
Christoph Hellwig10257742010-06-04 11:30:02 +02002073
2074 setattr_copy(inode, attrs);
2075 mark_inode_dirty(inode);
2076 return 0;
2077
Steve Frenche30dcf32005-09-20 20:49:16 -07002078cifs_setattr_exit:
Linus Torvalds1da177e2005-04-16 15:20:36 -07002079 kfree(full_path);
2080 FreeXid(xid);
2081 return rc;
2082}
2083
Jeff Layton0510eeb2008-08-02 07:26:12 -04002084int
2085cifs_setattr(struct dentry *direntry, struct iattr *attrs)
2086{
2087 struct inode *inode = direntry->d_inode;
2088 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
2089 struct cifsTconInfo *pTcon = cifs_sb->tcon;
2090
2091 if (pTcon->unix_ext)
2092 return cifs_setattr_unix(direntry, attrs);
2093
2094 return cifs_setattr_nounix(direntry, attrs);
2095
2096 /* BB: add cifs_setattr_legacy for really old servers */
2097}
2098
Steve French99ee4db2007-02-27 05:35:17 +00002099#if 0
Linus Torvalds1da177e2005-04-16 15:20:36 -07002100void cifs_delete_inode(struct inode *inode)
2101{
Joe Perchesb6b38f72010-04-21 03:50:45 +00002102 cFYI(1, "In cifs_delete_inode, inode = 0x%p", inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002103 /* may have to add back in if and when safe distributed caching of
2104 directories added e.g. via FindNotify */
2105}
Steve French99ee4db2007-02-27 05:35:17 +00002106#endif