blob: 5b042fc4645029cfce2185dbcc61d13668d4300b [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"
32
Christoph Hellwig70eff552008-02-15 20:55:05 +000033
Igor Mammedov79626702008-03-09 03:44:18 +000034static void cifs_set_ops(struct inode *inode, const bool is_dfs_referral)
Christoph Hellwig70eff552008-02-15 20:55:05 +000035{
36 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
37
38 switch (inode->i_mode & S_IFMT) {
39 case S_IFREG:
40 inode->i_op = &cifs_file_inode_ops;
41 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DIRECT_IO) {
42 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_BRL)
43 inode->i_fop = &cifs_file_direct_nobrl_ops;
44 else
45 inode->i_fop = &cifs_file_direct_ops;
46 } else if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_BRL)
47 inode->i_fop = &cifs_file_nobrl_ops;
48 else { /* not direct, send byte range locks */
49 inode->i_fop = &cifs_file_ops;
50 }
51
52
53 /* check if server can support readpages */
54 if (cifs_sb->tcon->ses->server->maxBuf <
55 PAGE_CACHE_SIZE + MAX_CIFS_HDR_SIZE)
56 inode->i_data.a_ops = &cifs_addr_ops_smallbuf;
57 else
58 inode->i_data.a_ops = &cifs_addr_ops;
59 break;
60 case S_IFDIR:
Steve Frenchbc5b6e22008-03-11 21:07:48 +000061#ifdef CONFIG_CIFS_DFS_UPCALL
Igor Mammedov79626702008-03-09 03:44:18 +000062 if (is_dfs_referral) {
63 inode->i_op = &cifs_dfs_referral_inode_operations;
64 } else {
Steve Frenchbc5b6e22008-03-11 21:07:48 +000065#else /* NO DFS support, treat as a directory */
66 {
67#endif
Igor Mammedov79626702008-03-09 03:44:18 +000068 inode->i_op = &cifs_dir_inode_ops;
69 inode->i_fop = &cifs_dir_ops;
70 }
Christoph Hellwig70eff552008-02-15 20:55:05 +000071 break;
72 case S_IFLNK:
73 inode->i_op = &cifs_symlink_inode_ops;
74 break;
75 default:
76 init_special_inode(inode, inode->i_mode, inode->i_rdev);
77 break;
78 }
79}
80
Jeff Laytondf2cf172010-02-12 07:44:16 -050081/* check inode attributes against fattr. If they don't match, tag the
82 * inode for cache invalidation
83 */
84static void
85cifs_revalidate_cache(struct inode *inode, struct cifs_fattr *fattr)
86{
87 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
88
Steve Frenchf19159d2010-04-21 04:12:10 +000089 cFYI(1, "%s: revalidating inode %llu", __func__, cifs_i->uniqueid);
Jeff Laytondf2cf172010-02-12 07:44:16 -050090
91 if (inode->i_state & I_NEW) {
Steve Frenchf19159d2010-04-21 04:12:10 +000092 cFYI(1, "%s: inode %llu is new", __func__, cifs_i->uniqueid);
Jeff Laytondf2cf172010-02-12 07:44:16 -050093 return;
94 }
95
96 /* don't bother with revalidation if we have an oplock */
97 if (cifs_i->clientCanCacheRead) {
Steve Frenchf19159d2010-04-21 04:12:10 +000098 cFYI(1, "%s: inode %llu is oplocked", __func__,
99 cifs_i->uniqueid);
Jeff Laytondf2cf172010-02-12 07:44:16 -0500100 return;
101 }
102
103 /* revalidate if mtime or size have changed */
104 if (timespec_equal(&inode->i_mtime, &fattr->cf_mtime) &&
105 cifs_i->server_eof == fattr->cf_eof) {
Steve Frenchf19159d2010-04-21 04:12:10 +0000106 cFYI(1, "%s: inode %llu is unchanged", __func__,
107 cifs_i->uniqueid);
Jeff Laytondf2cf172010-02-12 07:44:16 -0500108 return;
109 }
110
Steve Frenchf19159d2010-04-21 04:12:10 +0000111 cFYI(1, "%s: invalidating inode %llu mapping", __func__,
112 cifs_i->uniqueid);
Jeff Laytondf2cf172010-02-12 07:44:16 -0500113 cifs_i->invalid_mapping = true;
114}
115
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400116/* populate an inode with info from a cifs_fattr struct */
117void
118cifs_fattr_to_inode(struct inode *inode, struct cifs_fattr *fattr)
Christoph Hellwig75f12982008-02-25 20:25:21 +0000119{
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400120 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400121 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
122 unsigned long oldtime = cifs_i->time;
Christoph Hellwig75f12982008-02-25 20:25:21 +0000123
Jeff Laytondf2cf172010-02-12 07:44:16 -0500124 cifs_revalidate_cache(inode, fattr);
125
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400126 inode->i_atime = fattr->cf_atime;
127 inode->i_mtime = fattr->cf_mtime;
128 inode->i_ctime = fattr->cf_ctime;
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400129 inode->i_rdev = fattr->cf_rdev;
130 inode->i_nlink = fattr->cf_nlink;
131 inode->i_uid = fattr->cf_uid;
132 inode->i_gid = fattr->cf_gid;
133
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400134 /* if dynperm is set, don't clobber existing mode */
135 if (inode->i_state & I_NEW ||
136 !(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM))
137 inode->i_mode = fattr->cf_mode;
138
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400139 cifs_i->cifsAttrs = fattr->cf_cifsattrs;
140 cifs_i->uniqueid = fattr->cf_uniqueid;
141
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 Laytoncc0bad72009-06-25 00:56:52 -0400173/* Fill a cifs_fattr struct with info from FILE_UNIX_BASIC_INFO. */
174void
175cifs_unix_basic_to_fattr(struct cifs_fattr *fattr, FILE_UNIX_BASIC_INFO *info,
176 struct cifs_sb_info *cifs_sb)
177{
178 memset(fattr, 0, sizeof(*fattr));
179 fattr->cf_uniqueid = le64_to_cpu(info->UniqueId);
180 fattr->cf_bytes = le64_to_cpu(info->NumOfBytes);
181 fattr->cf_eof = le64_to_cpu(info->EndOfFile);
182
183 fattr->cf_atime = cifs_NTtimeToUnix(info->LastAccessTime);
184 fattr->cf_mtime = cifs_NTtimeToUnix(info->LastModificationTime);
185 fattr->cf_ctime = cifs_NTtimeToUnix(info->LastStatusChange);
186 fattr->cf_mode = le64_to_cpu(info->Permissions);
187
188 /*
189 * Since we set the inode type below we need to mask off
190 * to avoid strange results if bits set above.
191 */
192 fattr->cf_mode &= ~S_IFMT;
193 switch (le32_to_cpu(info->Type)) {
194 case UNIX_FILE:
195 fattr->cf_mode |= S_IFREG;
196 fattr->cf_dtype = DT_REG;
197 break;
198 case UNIX_SYMLINK:
199 fattr->cf_mode |= S_IFLNK;
200 fattr->cf_dtype = DT_LNK;
201 break;
202 case UNIX_DIR:
203 fattr->cf_mode |= S_IFDIR;
204 fattr->cf_dtype = DT_DIR;
205 break;
206 case UNIX_CHARDEV:
207 fattr->cf_mode |= S_IFCHR;
208 fattr->cf_dtype = DT_CHR;
209 fattr->cf_rdev = MKDEV(le64_to_cpu(info->DevMajor),
210 le64_to_cpu(info->DevMinor) & MINORMASK);
211 break;
212 case UNIX_BLOCKDEV:
213 fattr->cf_mode |= S_IFBLK;
214 fattr->cf_dtype = DT_BLK;
215 fattr->cf_rdev = MKDEV(le64_to_cpu(info->DevMajor),
216 le64_to_cpu(info->DevMinor) & MINORMASK);
217 break;
218 case UNIX_FIFO:
219 fattr->cf_mode |= S_IFIFO;
220 fattr->cf_dtype = DT_FIFO;
221 break;
222 case UNIX_SOCKET:
223 fattr->cf_mode |= S_IFSOCK;
224 fattr->cf_dtype = DT_SOCK;
225 break;
226 default:
227 /* safest to call it a file if we do not know */
228 fattr->cf_mode |= S_IFREG;
229 fattr->cf_dtype = DT_REG;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000230 cFYI(1, "unknown type %d", le32_to_cpu(info->Type));
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400231 break;
232 }
233
234 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_OVERR_UID)
235 fattr->cf_uid = cifs_sb->mnt_uid;
236 else
237 fattr->cf_uid = le64_to_cpu(info->Uid);
238
239 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_OVERR_GID)
240 fattr->cf_gid = cifs_sb->mnt_gid;
241 else
242 fattr->cf_gid = le64_to_cpu(info->Gid);
243
244 fattr->cf_nlink = le64_to_cpu(info->Nlinks);
245}
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000246
247/*
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400248 * Fill a cifs_fattr struct with fake inode info.
249 *
250 * Needed to setup cifs_fattr data for the directory which is the
251 * junction to the new submount (ie to setup the fake directory
252 * which represents a DFS referral).
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000253 */
Steve Frenchf1230c92009-07-22 23:13:01 +0000254static void
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400255cifs_create_dfs_fattr(struct cifs_fattr *fattr, struct super_block *sb)
Steve French0e4bbde2008-05-20 19:50:46 +0000256{
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400257 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
Steve French0e4bbde2008-05-20 19:50:46 +0000258
Joe Perchesb6b38f72010-04-21 03:50:45 +0000259 cFYI(1, "creating fake fattr for DFS referral");
Steve French0e4bbde2008-05-20 19:50:46 +0000260
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400261 memset(fattr, 0, sizeof(*fattr));
262 fattr->cf_mode = S_IFDIR | S_IXUGO | S_IRWXU;
263 fattr->cf_uid = cifs_sb->mnt_uid;
264 fattr->cf_gid = cifs_sb->mnt_gid;
265 fattr->cf_atime = CURRENT_TIME;
266 fattr->cf_ctime = CURRENT_TIME;
267 fattr->cf_mtime = CURRENT_TIME;
268 fattr->cf_nlink = 2;
269 fattr->cf_flags |= CIFS_FATTR_DFS_REFERRAL;
Steve French0e4bbde2008-05-20 19:50:46 +0000270}
271
Jeff Laytonabab0952010-02-12 07:44:18 -0500272int cifs_get_file_info_unix(struct file *filp)
273{
274 int rc;
275 int xid;
276 FILE_UNIX_BASIC_INFO find_data;
277 struct cifs_fattr fattr;
278 struct inode *inode = filp->f_path.dentry->d_inode;
279 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
280 struct cifsTconInfo *tcon = cifs_sb->tcon;
281 struct cifsFileInfo *cfile = (struct cifsFileInfo *) filp->private_data;
282
283 xid = GetXid();
284 rc = CIFSSMBUnixQFileInfo(xid, tcon, cfile->netfid, &find_data);
285 if (!rc) {
286 cifs_unix_basic_to_fattr(&fattr, &find_data, cifs_sb);
287 } else if (rc == -EREMOTE) {
288 cifs_create_dfs_fattr(&fattr, inode->i_sb);
289 rc = 0;
290 }
291
292 cifs_fattr_to_inode(inode, &fattr);
293 FreeXid(xid);
294 return rc;
295}
296
Linus Torvalds1da177e2005-04-16 15:20:36 -0700297int cifs_get_inode_info_unix(struct inode **pinode,
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400298 const unsigned char *full_path,
299 struct super_block *sb, int xid)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700300{
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400301 int rc;
Steve French0e4bbde2008-05-20 19:50:46 +0000302 FILE_UNIX_BASIC_INFO find_data;
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400303 struct cifs_fattr fattr;
304 struct cifsTconInfo *tcon;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700305 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700306
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400307 tcon = cifs_sb->tcon;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000308 cFYI(1, "Getting info on %s", full_path);
Igor Mammedov79626702008-03-09 03:44:18 +0000309
Linus Torvalds1da177e2005-04-16 15:20:36 -0700310 /* could have done a find first instead but this returns more info */
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400311 rc = CIFSSMBUnixQPathInfo(xid, tcon, full_path, &find_data,
Steve French737b7582005-04-28 22:41:06 -0700312 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
313 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400314
315 if (!rc) {
316 cifs_unix_basic_to_fattr(&fattr, &find_data, cifs_sb);
317 } else if (rc == -EREMOTE) {
318 cifs_create_dfs_fattr(&fattr, sb);
Jeff Laytone911d0c2008-07-12 13:47:59 -0700319 rc = 0;
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400320 } else {
321 return rc;
Steve French0e4bbde2008-05-20 19:50:46 +0000322 }
323
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400324 if (*pinode == NULL) {
325 /* get new inode */
326 *pinode = cifs_iget(sb, &fattr);
327 if (!*pinode)
328 rc = -ENOMEM;
329 } else {
330 /* we already have inode, update it */
331 cifs_fattr_to_inode(*pinode, &fattr);
332 }
Steve French0e4bbde2008-05-20 19:50:46 +0000333
Linus Torvalds1da177e2005-04-16 15:20:36 -0700334 return rc;
335}
336
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400337static int
338cifs_sfu_type(struct cifs_fattr *fattr, const unsigned char *path,
339 struct cifs_sb_info *cifs_sb, int xid)
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800340{
341 int rc;
Steve French4b18f2a2008-04-29 00:06:05 +0000342 int oplock = 0;
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800343 __u16 netfid;
344 struct cifsTconInfo *pTcon = cifs_sb->tcon;
Steve French86c96b42005-11-18 20:25:31 -0800345 char buf[24];
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800346 unsigned int bytes_read;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000347 char *pbuf;
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800348
349 pbuf = buf;
350
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400351 fattr->cf_mode &= ~S_IFMT;
352
353 if (fattr->cf_eof == 0) {
354 fattr->cf_mode |= S_IFIFO;
355 fattr->cf_dtype = DT_FIFO;
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800356 return 0;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400357 } else if (fattr->cf_eof < 8) {
358 fattr->cf_mode |= S_IFREG;
359 fattr->cf_dtype = DT_REG;
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800360 return -EINVAL; /* EOPNOTSUPP? */
361 }
Steve French50c2f752007-07-13 00:33:32 +0000362
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800363 rc = CIFSSMBOpen(xid, pTcon, path, FILE_OPEN, GENERIC_READ,
364 CREATE_NOT_DIR, &netfid, &oplock, NULL,
365 cifs_sb->local_nls,
366 cifs_sb->mnt_cifs_flags &
367 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000368 if (rc == 0) {
Steve Frenchec637e32005-12-12 20:53:18 -0800369 int buf_type = CIFS_NO_BUFFER;
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800370 /* Read header */
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400371 rc = CIFSSMBRead(xid, pTcon, netfid,
Steve French86c96b42005-11-18 20:25:31 -0800372 24 /* length */, 0 /* offset */,
Steve Frenchec637e32005-12-12 20:53:18 -0800373 &bytes_read, &pbuf, &buf_type);
Steve French4523cc32007-04-30 20:13:06 +0000374 if ((rc == 0) && (bytes_read >= 8)) {
375 if (memcmp("IntxBLK", pbuf, 8) == 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000376 cFYI(1, "Block device");
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400377 fattr->cf_mode |= S_IFBLK;
378 fattr->cf_dtype = DT_BLK;
Steve French4523cc32007-04-30 20:13:06 +0000379 if (bytes_read == 24) {
Steve French86c96b42005-11-18 20:25:31 -0800380 /* we have enough to decode dev num */
381 __u64 mjr; /* major */
382 __u64 mnr; /* minor */
383 mjr = le64_to_cpu(*(__le64 *)(pbuf+8));
384 mnr = le64_to_cpu(*(__le64 *)(pbuf+16));
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400385 fattr->cf_rdev = MKDEV(mjr, mnr);
Steve French86c96b42005-11-18 20:25:31 -0800386 }
Steve French4523cc32007-04-30 20:13:06 +0000387 } else if (memcmp("IntxCHR", pbuf, 8) == 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000388 cFYI(1, "Char device");
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400389 fattr->cf_mode |= S_IFCHR;
390 fattr->cf_dtype = DT_CHR;
Steve French4523cc32007-04-30 20:13:06 +0000391 if (bytes_read == 24) {
Steve French86c96b42005-11-18 20:25:31 -0800392 /* we have enough to decode dev num */
393 __u64 mjr; /* major */
394 __u64 mnr; /* minor */
395 mjr = le64_to_cpu(*(__le64 *)(pbuf+8));
396 mnr = le64_to_cpu(*(__le64 *)(pbuf+16));
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400397 fattr->cf_rdev = MKDEV(mjr, mnr);
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000398 }
Steve French4523cc32007-04-30 20:13:06 +0000399 } else if (memcmp("IntxLNK", pbuf, 7) == 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000400 cFYI(1, "Symlink");
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400401 fattr->cf_mode |= S_IFLNK;
402 fattr->cf_dtype = DT_LNK;
Steve French86c96b42005-11-18 20:25:31 -0800403 } else {
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400404 fattr->cf_mode |= S_IFREG; /* file? */
405 fattr->cf_dtype = DT_REG;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000406 rc = -EOPNOTSUPP;
Steve French86c96b42005-11-18 20:25:31 -0800407 }
Steve French3020a1f2005-11-18 11:31:10 -0800408 } else {
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400409 fattr->cf_mode |= S_IFREG; /* then it is a file */
410 fattr->cf_dtype = DT_REG;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000411 rc = -EOPNOTSUPP; /* or some unknown SFU type */
412 }
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800413 CIFSSMBClose(xid, pTcon, netfid);
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800414 }
415 return rc;
Steve Frenchd6e2f2a42005-11-15 16:43:39 -0800416}
417
Steve French9e294f12005-11-17 16:59:21 -0800418#define SFBITS_MASK (S_ISVTX | S_ISGID | S_ISUID) /* SETFILEBITS valid bits */
419
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400420/*
421 * Fetch mode bits as provided by SFU.
422 *
423 * FIXME: Doesn't this clobber the type bit we got from cifs_sfu_type ?
424 */
425static int cifs_sfu_mode(struct cifs_fattr *fattr, const unsigned char *path,
426 struct cifs_sb_info *cifs_sb, int xid)
Steve French9e294f12005-11-17 16:59:21 -0800427{
Steve French3020a1f2005-11-18 11:31:10 -0800428#ifdef CONFIG_CIFS_XATTR
Steve French9e294f12005-11-17 16:59:21 -0800429 ssize_t rc;
430 char ea_value[4];
431 __u32 mode;
432
Jeff Layton31c05192010-02-10 16:18:26 -0500433 rc = CIFSSMBQAllEAs(xid, cifs_sb->tcon, path, "SETFILEBITS",
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400434 ea_value, 4 /* size of buf */, cifs_sb->local_nls,
435 cifs_sb->mnt_cifs_flags &
436 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French4523cc32007-04-30 20:13:06 +0000437 if (rc < 0)
Steve French9e294f12005-11-17 16:59:21 -0800438 return (int)rc;
439 else if (rc > 3) {
440 mode = le32_to_cpu(*((__le32 *)ea_value));
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400441 fattr->cf_mode &= ~SFBITS_MASK;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000442 cFYI(1, "special bits 0%o org mode 0%o", mode,
443 fattr->cf_mode);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400444 fattr->cf_mode = (mode & SFBITS_MASK) | fattr->cf_mode;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000445 cFYI(1, "special mode bits 0%o", mode);
Steve French9e294f12005-11-17 16:59:21 -0800446 }
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400447
448 return 0;
Steve French3020a1f2005-11-18 11:31:10 -0800449#else
450 return -EOPNOTSUPP;
451#endif
Steve French9e294f12005-11-17 16:59:21 -0800452}
453
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400454/* Fill a cifs_fattr struct with info from FILE_ALL_INFO */
Steve Frenchf1230c92009-07-22 23:13:01 +0000455static void
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400456cifs_all_info_to_fattr(struct cifs_fattr *fattr, FILE_ALL_INFO *info,
457 struct cifs_sb_info *cifs_sb, bool adjust_tz)
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000458{
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400459 memset(fattr, 0, sizeof(*fattr));
460 fattr->cf_cifsattrs = le32_to_cpu(info->Attributes);
461 if (info->DeletePending)
462 fattr->cf_flags |= CIFS_FATTR_DELETE_PENDING;
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000463
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400464 if (info->LastAccessTime)
465 fattr->cf_atime = cifs_NTtimeToUnix(info->LastAccessTime);
466 else
467 fattr->cf_atime = CURRENT_TIME;
468
469 fattr->cf_ctime = cifs_NTtimeToUnix(info->ChangeTime);
470 fattr->cf_mtime = cifs_NTtimeToUnix(info->LastWriteTime);
471
472 if (adjust_tz) {
473 fattr->cf_ctime.tv_sec += cifs_sb->tcon->ses->server->timeAdj;
474 fattr->cf_mtime.tv_sec += cifs_sb->tcon->ses->server->timeAdj;
475 }
476
477 fattr->cf_eof = le64_to_cpu(info->EndOfFile);
478 fattr->cf_bytes = le64_to_cpu(info->AllocationSize);
479
480 if (fattr->cf_cifsattrs & ATTR_DIRECTORY) {
481 fattr->cf_mode = S_IFDIR | cifs_sb->mnt_dir_mode;
482 fattr->cf_dtype = DT_DIR;
483 } else {
484 fattr->cf_mode = S_IFREG | cifs_sb->mnt_file_mode;
485 fattr->cf_dtype = DT_REG;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400486
Jeff Laytond0c280d2009-07-09 01:46:44 -0400487 /* clear write bits if ATTR_READONLY is set */
488 if (fattr->cf_cifsattrs & ATTR_READONLY)
489 fattr->cf_mode &= ~(S_IWUGO);
490 }
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400491
492 fattr->cf_nlink = le32_to_cpu(info->NumberOfLinks);
493
494 fattr->cf_uid = cifs_sb->mnt_uid;
495 fattr->cf_gid = cifs_sb->mnt_gid;
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000496}
497
Jeff Laytonabab0952010-02-12 07:44:18 -0500498int cifs_get_file_info(struct file *filp)
499{
500 int rc;
501 int xid;
502 FILE_ALL_INFO find_data;
503 struct cifs_fattr fattr;
504 struct inode *inode = filp->f_path.dentry->d_inode;
505 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
506 struct cifsTconInfo *tcon = cifs_sb->tcon;
507 struct cifsFileInfo *cfile = (struct cifsFileInfo *) filp->private_data;
508
509 xid = GetXid();
510 rc = CIFSSMBQFileInfo(xid, tcon, cfile->netfid, &find_data);
511 if (rc == -EOPNOTSUPP || rc == -EINVAL) {
512 /*
513 * FIXME: legacy server -- fall back to path-based call?
Steve Frenchff2157132010-03-09 20:30:42 +0000514 * for now, just skip revalidating and mark inode for
515 * immediate reval.
516 */
Jeff Laytonabab0952010-02-12 07:44:18 -0500517 rc = 0;
518 CIFS_I(inode)->time = 0;
519 goto cgfi_exit;
520 } else if (rc == -EREMOTE) {
521 cifs_create_dfs_fattr(&fattr, inode->i_sb);
522 rc = 0;
523 } else if (rc)
524 goto cgfi_exit;
525
526 /*
527 * don't bother with SFU junk here -- just mark inode as needing
528 * revalidation.
529 */
530 cifs_all_info_to_fattr(&fattr, &find_data, cifs_sb, false);
531 fattr.cf_uniqueid = CIFS_I(inode)->uniqueid;
532 fattr.cf_flags |= CIFS_FATTR_NEED_REVAL;
533 cifs_fattr_to_inode(inode, &fattr);
534cgfi_exit:
535 FreeXid(xid);
536 return rc;
537}
538
Linus Torvalds1da177e2005-04-16 15:20:36 -0700539int cifs_get_inode_info(struct inode **pinode,
Steve French646dd532008-05-15 01:50:56 +0000540 const unsigned char *full_path, FILE_ALL_INFO *pfindData,
Steve French8b1327f2008-03-14 22:37:16 +0000541 struct super_block *sb, int xid, const __u16 *pfid)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700542{
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400543 int rc = 0, tmprc;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700544 struct cifsTconInfo *pTcon;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700545 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700546 char *buf = NULL;
Steve French5ade9de2008-05-02 20:56:23 +0000547 bool adjustTZ = false;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400548 struct cifs_fattr fattr;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700549
550 pTcon = cifs_sb->tcon;
Joe Perchesb6b38f72010-04-21 03:50:45 +0000551 cFYI(1, "Getting info on %s", full_path);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700552
Steve Frenchd0d2f2d2005-06-02 15:12:36 -0700553 if ((pfindData == NULL) && (*pinode != NULL)) {
554 if (CIFS_I(*pinode)->clientCanCacheRead) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000555 cFYI(1, "No need to revalidate cached inode sizes");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700556 return rc;
557 }
558 }
559
560 /* if file info not passed in then get it from server */
Steve Frenchd0d2f2d2005-06-02 15:12:36 -0700561 if (pfindData == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700562 buf = kmalloc(sizeof(FILE_ALL_INFO), GFP_KERNEL);
Steve Frenchd0d2f2d2005-06-02 15:12:36 -0700563 if (buf == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700564 return -ENOMEM;
565 pfindData = (FILE_ALL_INFO *)buf;
Igor Mammedov79626702008-03-09 03:44:18 +0000566
Linus Torvalds1da177e2005-04-16 15:20:36 -0700567 /* could do find first instead but this returns more info */
Igor Mammedov79626702008-03-09 03:44:18 +0000568 rc = CIFSSMBQPathInfo(xid, pTcon, full_path, pfindData,
Steve Frenchacf1a1b2006-10-12 03:28:28 +0000569 0 /* not legacy */,
Steve French6b8edfe2005-08-23 20:26:03 -0700570 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
Steve French737b7582005-04-28 22:41:06 -0700571 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French6b8edfe2005-08-23 20:26:03 -0700572 /* BB optimize code so we do not make the above call
573 when server claims no NT SMB support and the above call
574 failed at least once - set flag in tcon or mount */
Steve French4523cc32007-04-30 20:13:06 +0000575 if ((rc == -EOPNOTSUPP) || (rc == -EINVAL)) {
Igor Mammedov79626702008-03-09 03:44:18 +0000576 rc = SMBQueryInformation(xid, pTcon, full_path,
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000577 pfindData, cifs_sb->local_nls,
Steve French6b8edfe2005-08-23 20:26:03 -0700578 cifs_sb->mnt_cifs_flags &
579 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French4b18f2a2008-04-29 00:06:05 +0000580 adjustTZ = true;
Steve French6b8edfe2005-08-23 20:26:03 -0700581 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700582 }
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400583
584 if (!rc) {
585 cifs_all_info_to_fattr(&fattr, (FILE_ALL_INFO *) pfindData,
586 cifs_sb, adjustTZ);
587 } else if (rc == -EREMOTE) {
588 cifs_create_dfs_fattr(&fattr, sb);
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000589 rc = 0;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400590 } else {
Igor Mammedov79626702008-03-09 03:44:18 +0000591 goto cgii_exit;
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400592 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700593
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400594 /*
595 * If an inode wasn't passed in, then get the inode number
596 *
597 * Is an i_ino of zero legal? Can we use that to check if the server
598 * supports returning inode numbers? Are there other sanity checks we
599 * can use to ensure that the server is really filling in that field?
600 *
601 * We can not use the IndexNumber field by default from Windows or
602 * Samba (in ALL_INFO buf) but we can request it explicitly. The SNIA
603 * CIFS spec claims that this value is unique within the scope of a
604 * share, and the windows docs hint that it's actually unique
605 * per-machine.
606 *
607 * There may be higher info levels that work but are there Windows
608 * server or network appliances for which IndexNumber field is not
609 * guaranteed unique?
610 */
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000611 if (*pinode == NULL) {
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000612 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM) {
613 int rc1 = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700614
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000615 rc1 = CIFSGetSrvInodeNumber(xid, pTcon,
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400616 full_path, &fattr.cf_uniqueid,
Steve French737b7582005-04-28 22:41:06 -0700617 cifs_sb->local_nls,
618 cifs_sb->mnt_cifs_flags &
619 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Laytonec06aed2009-11-06 14:18:29 -0500620 if (rc1 || !fattr.cf_uniqueid) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000621 cFYI(1, "GetSrvInodeNum rc %d", rc1);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400622 fattr.cf_uniqueid = iunique(sb, ROOT_I);
Jeff Laytonec06aed2009-11-06 14:18:29 -0500623 cifs_autodisable_serverino(cifs_sb);
Jeff Layton132ac7b2009-02-10 07:33:57 -0500624 }
Jeff Layton132ac7b2009-02-10 07:33:57 -0500625 } else {
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400626 fattr.cf_uniqueid = iunique(sb, ROOT_I);
Jeff Layton132ac7b2009-02-10 07:33:57 -0500627 }
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000628 } else {
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400629 fattr.cf_uniqueid = CIFS_I(*pinode)->uniqueid;
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000630 }
631
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400632 /* query for SFU type info if supported and needed */
633 if (fattr.cf_cifsattrs & ATTR_SYSTEM &&
634 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_UNX_EMUL) {
635 tmprc = cifs_sfu_type(&fattr, full_path, cifs_sb, xid);
636 if (tmprc)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000637 cFYI(1, "cifs_sfu_type failed: %d", tmprc);
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000638 }
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000639
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000640#ifdef CONFIG_CIFS_EXPERIMENTAL
641 /* fill in 0777 bits from ACL */
642 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000643 cFYI(1, "Getting mode bits from ACL");
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400644 cifs_acl_to_fattr(cifs_sb, &fattr, *pinode, full_path, pfid);
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000645 }
646#endif
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400647
648 /* fill in remaining high mode bits e.g. SUID, VTX */
649 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_UNX_EMUL)
650 cifs_sfu_mode(&fattr, full_path, cifs_sb, xid);
651
652 if (!*pinode) {
653 *pinode = cifs_iget(sb, &fattr);
654 if (!*pinode)
655 rc = -ENOMEM;
656 } else {
657 cifs_fattr_to_inode(*pinode, &fattr);
Steve Frenchb9a3260f2008-05-20 21:52:32 +0000658 }
659
Igor Mammedov79626702008-03-09 03:44:18 +0000660cgii_exit:
Linus Torvalds1da177e2005-04-16 15:20:36 -0700661 kfree(buf);
662 return rc;
663}
664
Steve French7f8ed422007-09-28 22:28:55 +0000665static const struct inode_operations cifs_ipc_inode_ops = {
666 .lookup = cifs_lookup,
667};
668
Igor Mammedove4cce942009-02-10 14:10:26 +0300669char *cifs_build_path_to_root(struct cifs_sb_info *cifs_sb)
Steve French8be0ed42008-12-05 19:14:12 +0000670{
671 int pplen = cifs_sb->prepathlen;
672 int dfsplen;
673 char *full_path = NULL;
674
675 /* if no prefix path, simply set path to the root of share to "" */
676 if (pplen == 0) {
677 full_path = kmalloc(1, GFP_KERNEL);
678 if (full_path)
679 full_path[0] = 0;
680 return full_path;
681 }
682
683 if (cifs_sb->tcon && (cifs_sb->tcon->Flags & SMB_SHARE_IS_IN_DFS))
684 dfsplen = strnlen(cifs_sb->tcon->treeName, MAX_TREE_SIZE + 1);
685 else
686 dfsplen = 0;
687
688 full_path = kmalloc(dfsplen + pplen + 1, GFP_KERNEL);
689 if (full_path == NULL)
690 return full_path;
691
692 if (dfsplen) {
693 strncpy(full_path, cifs_sb->tcon->treeName, dfsplen);
694 /* switch slash direction in prepath depending on whether
695 * windows or posix style path names
696 */
697 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_POSIX_PATHS) {
698 int i;
699 for (i = 0; i < dfsplen; i++) {
700 if (full_path[i] == '\\')
701 full_path[i] = '/';
702 }
703 }
704 }
705 strncpy(full_path + dfsplen, cifs_sb->prepath, pplen);
706 full_path[dfsplen + pplen] = 0; /* add trailing null */
707 return full_path;
708}
709
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400710static int
711cifs_find_inode(struct inode *inode, void *opaque)
712{
713 struct cifs_fattr *fattr = (struct cifs_fattr *) opaque;
714
715 if (CIFS_I(inode)->uniqueid != fattr->cf_uniqueid)
716 return 0;
717
Jeff Layton3d694382010-05-11 14:59:55 -0400718 /*
719 * uh oh -- it's a directory. We can't use it since hardlinked dirs are
720 * verboten. Disable serverino and return it as if it were found, the
721 * caller can discard it, generate a uniqueid and retry the find
722 */
723 if (S_ISDIR(inode->i_mode) && !list_empty(&inode->i_dentry)) {
724 fattr->cf_flags |= CIFS_FATTR_INO_COLLISION;
725 cifs_autodisable_serverino(CIFS_SB(inode->i_sb));
726 }
727
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400728 return 1;
729}
730
731static int
732cifs_init_inode(struct inode *inode, void *opaque)
733{
734 struct cifs_fattr *fattr = (struct cifs_fattr *) opaque;
735
736 CIFS_I(inode)->uniqueid = fattr->cf_uniqueid;
737 return 0;
738}
739
740/* Given fattrs, get a corresponding inode */
741struct inode *
742cifs_iget(struct super_block *sb, struct cifs_fattr *fattr)
743{
744 unsigned long hash;
745 struct inode *inode;
746
Jeff Layton3d694382010-05-11 14:59:55 -0400747retry_iget5_locked:
Joe Perchesb6b38f72010-04-21 03:50:45 +0000748 cFYI(1, "looking for uniqueid=%llu", fattr->cf_uniqueid);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400749
750 /* hash down to 32-bits on 32-bit arch */
751 hash = cifs_uniqueid_to_ino_t(fattr->cf_uniqueid);
752
753 inode = iget5_locked(sb, hash, cifs_find_inode, cifs_init_inode, fattr);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400754 if (inode) {
Jeff Layton3d694382010-05-11 14:59:55 -0400755 /* was there a problematic inode number collision? */
756 if (fattr->cf_flags & CIFS_FATTR_INO_COLLISION) {
757 iput(inode);
758 fattr->cf_uniqueid = iunique(sb, ROOT_I);
759 fattr->cf_flags &= ~CIFS_FATTR_INO_COLLISION;
760 goto retry_iget5_locked;
761 }
762
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400763 cifs_fattr_to_inode(inode, fattr);
764 if (sb->s_flags & MS_NOATIME)
765 inode->i_flags |= S_NOATIME | S_NOCMTIME;
766 if (inode->i_state & I_NEW) {
767 inode->i_ino = hash;
768 unlock_new_inode(inode);
769 }
770 }
771
772 return inode;
773}
774
Linus Torvalds1da177e2005-04-16 15:20:36 -0700775/* gets root inode */
Jeff Laytonbd433d42009-05-27 09:37:34 -0400776struct inode *cifs_root_iget(struct super_block *sb, unsigned long ino)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700777{
David Howellsce634ab2008-02-07 00:15:33 -0800778 int xid;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700779 struct cifs_sb_info *cifs_sb;
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400780 struct inode *inode = NULL;
David Howellsce634ab2008-02-07 00:15:33 -0800781 long rc;
Steve French8be0ed42008-12-05 19:14:12 +0000782 char *full_path;
David Howellsce634ab2008-02-07 00:15:33 -0800783
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400784 cifs_sb = CIFS_SB(sb);
Igor Mammedove4cce942009-02-10 14:10:26 +0300785 full_path = cifs_build_path_to_root(cifs_sb);
Steve French8be0ed42008-12-05 19:14:12 +0000786 if (full_path == NULL)
787 return ERR_PTR(-ENOMEM);
Steve Frenchc18c8422007-07-18 23:21:09 +0000788
Steve French8be0ed42008-12-05 19:14:12 +0000789 xid = GetXid();
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400790 if (cifs_sb->tcon->unix_ext)
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400791 rc = cifs_get_inode_info_unix(&inode, full_path, sb, xid);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400792 else
793 rc = cifs_get_inode_info(&inode, full_path, NULL, sb,
Steve French8be0ed42008-12-05 19:14:12 +0000794 xid, NULL);
Jeff Layton0b8f18e2009-07-09 01:46:37 -0400795
796 if (!inode)
797 return ERR_PTR(-ENOMEM);
Jeff Laytoncc0bad72009-06-25 00:56:52 -0400798
Steve French7f8ed422007-09-28 22:28:55 +0000799 if (rc && cifs_sb->tcon->ipc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000800 cFYI(1, "ipc connection - fake read inode");
Steve French7f8ed422007-09-28 22:28:55 +0000801 inode->i_mode |= S_IFDIR;
802 inode->i_nlink = 2;
803 inode->i_op = &cifs_ipc_inode_ops;
804 inode->i_fop = &simple_dir_operations;
805 inode->i_uid = cifs_sb->mnt_uid;
806 inode->i_gid = cifs_sb->mnt_gid;
Steve Frenchad661332008-08-12 14:14:40 +0000807 } else if (rc) {
Steve French8be0ed42008-12-05 19:14:12 +0000808 kfree(full_path);
David Howellsce634ab2008-02-07 00:15:33 -0800809 _FreeXid(xid);
810 iget_failed(inode);
811 return ERR_PTR(rc);
Steve French7f8ed422007-09-28 22:28:55 +0000812 }
813
David Howellsce634ab2008-02-07 00:15:33 -0800814
Steve French8be0ed42008-12-05 19:14:12 +0000815 kfree(full_path);
David Howellsce634ab2008-02-07 00:15:33 -0800816 /* can not call macro FreeXid here since in a void func
817 * TODO: This is no longer true
818 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700819 _FreeXid(xid);
David Howellsce634ab2008-02-07 00:15:33 -0800820 return inode;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700821}
822
Steve French388e57b2008-09-16 23:50:58 +0000823static int
824cifs_set_file_info(struct inode *inode, struct iattr *attrs, int xid,
825 char *full_path, __u32 dosattr)
826{
827 int rc;
828 int oplock = 0;
829 __u16 netfid;
830 __u32 netpid;
831 bool set_time = false;
832 struct cifsFileInfo *open_file;
833 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
834 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
835 struct cifsTconInfo *pTcon = cifs_sb->tcon;
836 FILE_BASIC_INFO info_buf;
837
Steve French1adcb712009-02-25 14:19:56 +0000838 if (attrs == NULL)
839 return -EINVAL;
840
Steve French388e57b2008-09-16 23:50:58 +0000841 if (attrs->ia_valid & ATTR_ATIME) {
842 set_time = true;
843 info_buf.LastAccessTime =
844 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_atime));
845 } else
846 info_buf.LastAccessTime = 0;
847
848 if (attrs->ia_valid & ATTR_MTIME) {
849 set_time = true;
850 info_buf.LastWriteTime =
851 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_mtime));
852 } else
853 info_buf.LastWriteTime = 0;
854
855 /*
856 * Samba throws this field away, but windows may actually use it.
857 * Do not set ctime unless other time stamps are changed explicitly
858 * (i.e. by utimes()) since we would then have a mix of client and
859 * server times.
860 */
861 if (set_time && (attrs->ia_valid & ATTR_CTIME)) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000862 cFYI(1, "CIFS - CTIME changed");
Steve French388e57b2008-09-16 23:50:58 +0000863 info_buf.ChangeTime =
864 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_ctime));
865 } else
866 info_buf.ChangeTime = 0;
867
868 info_buf.CreationTime = 0; /* don't change */
869 info_buf.Attributes = cpu_to_le32(dosattr);
870
871 /*
872 * If the file is already open for write, just use that fileid
873 */
874 open_file = find_writable_file(cifsInode);
875 if (open_file) {
876 netfid = open_file->netfid;
877 netpid = open_file->pid;
878 goto set_via_filehandle;
879 }
880
881 /*
882 * NT4 apparently returns success on this call, but it doesn't
883 * really work.
884 */
885 if (!(pTcon->ses->flags & CIFS_SES_NT4)) {
886 rc = CIFSSMBSetPathInfo(xid, pTcon, full_path,
887 &info_buf, cifs_sb->local_nls,
888 cifs_sb->mnt_cifs_flags &
889 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Layton6b37faa2008-10-06 21:54:41 +0000890 if (rc == 0) {
891 cifsInode->cifsAttrs = dosattr;
892 goto out;
893 } else if (rc != -EOPNOTSUPP && rc != -EINVAL)
Steve French388e57b2008-09-16 23:50:58 +0000894 goto out;
895 }
896
Joe Perchesb6b38f72010-04-21 03:50:45 +0000897 cFYI(1, "calling SetFileInfo since SetPathInfo for "
898 "times not supported by this server");
Steve French388e57b2008-09-16 23:50:58 +0000899 rc = CIFSSMBOpen(xid, pTcon, full_path, FILE_OPEN,
900 SYNCHRONIZE | FILE_WRITE_ATTRIBUTES,
901 CREATE_NOT_DIR, &netfid, &oplock,
902 NULL, cifs_sb->local_nls,
903 cifs_sb->mnt_cifs_flags &
904 CIFS_MOUNT_MAP_SPECIAL_CHR);
905
906 if (rc != 0) {
907 if (rc == -EIO)
908 rc = -EINVAL;
909 goto out;
910 }
911
912 netpid = current->tgid;
913
914set_via_filehandle:
915 rc = CIFSSMBSetFileInfo(xid, pTcon, &info_buf, netfid, netpid);
Steve Frenchd3889082008-09-24 19:22:52 +0000916 if (!rc)
917 cifsInode->cifsAttrs = dosattr;
918
Steve French388e57b2008-09-16 23:50:58 +0000919 if (open_file == NULL)
920 CIFSSMBClose(xid, pTcon, netfid);
921 else
Dave Kleikamp6ab409b2009-08-31 11:07:12 -0400922 cifsFileInfo_put(open_file);
Steve French388e57b2008-09-16 23:50:58 +0000923out:
924 return rc;
925}
926
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400927/*
928 * open the given file (if it isn't already), set the DELETE_ON_CLOSE bit
929 * and rename it to a random name that hopefully won't conflict with
930 * anything else.
931 */
932static int
Steve French32709582008-10-20 00:44:19 +0000933cifs_rename_pending_delete(char *full_path, struct dentry *dentry, int xid)
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400934{
935 int oplock = 0;
936 int rc;
937 __u16 netfid;
Steve French32709582008-10-20 00:44:19 +0000938 struct inode *inode = dentry->d_inode;
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400939 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
940 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
941 struct cifsTconInfo *tcon = cifs_sb->tcon;
Steve French32709582008-10-20 00:44:19 +0000942 __u32 dosattr, origattr;
943 FILE_BASIC_INFO *info_buf = NULL;
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400944
945 rc = CIFSSMBOpen(xid, tcon, full_path, FILE_OPEN,
Jeff Laytondd1db2d2008-10-16 19:27:12 -0400946 DELETE|FILE_WRITE_ATTRIBUTES, CREATE_NOT_DIR,
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400947 &netfid, &oplock, NULL, cifs_sb->local_nls,
948 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
949 if (rc != 0)
950 goto out;
951
Steve French32709582008-10-20 00:44:19 +0000952 origattr = cifsInode->cifsAttrs;
953 if (origattr == 0)
954 origattr |= ATTR_NORMAL;
955
956 dosattr = origattr & ~ATTR_READONLY;
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400957 if (dosattr == 0)
958 dosattr |= ATTR_NORMAL;
959 dosattr |= ATTR_HIDDEN;
960
Steve French32709582008-10-20 00:44:19 +0000961 /* set ATTR_HIDDEN and clear ATTR_READONLY, but only if needed */
962 if (dosattr != origattr) {
963 info_buf = kzalloc(sizeof(*info_buf), GFP_KERNEL);
964 if (info_buf == NULL) {
965 rc = -ENOMEM;
966 goto out_close;
967 }
968 info_buf->Attributes = cpu_to_le32(dosattr);
969 rc = CIFSSMBSetFileInfo(xid, tcon, info_buf, netfid,
970 current->tgid);
971 /* although we would like to mark the file hidden
972 if that fails we will still try to rename it */
Steve French41346092008-10-20 18:24:42 +0000973 if (rc != 0)
Steve French32709582008-10-20 00:44:19 +0000974 cifsInode->cifsAttrs = dosattr;
975 else
976 dosattr = origattr; /* since not able to change them */
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400977 }
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400978
Jeff Laytondd1db2d2008-10-16 19:27:12 -0400979 /* rename the file */
980 rc = CIFSSMBRenameOpenFile(xid, tcon, netfid, NULL, cifs_sb->local_nls,
Jeff Laytona12a1ac2008-09-23 11:48:35 -0400981 cifs_sb->mnt_cifs_flags &
982 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French32709582008-10-20 00:44:19 +0000983 if (rc != 0) {
984 rc = -ETXTBSY;
985 goto undo_setattr;
986 }
Jeff Layton6d22f092008-09-23 11:48:35 -0400987
Steve French32709582008-10-20 00:44:19 +0000988 /* try to set DELETE_ON_CLOSE */
989 if (!cifsInode->delete_pending) {
990 rc = CIFSSMBSetFileDisposition(xid, tcon, true, netfid,
991 current->tgid);
992 /*
993 * some samba versions return -ENOENT when we try to set the
994 * file disposition here. Likely a samba bug, but work around
995 * it for now. This means that some cifsXXX files may hang
996 * around after they shouldn't.
997 *
998 * BB: remove this hack after more servers have the fix
999 */
1000 if (rc == -ENOENT)
1001 rc = 0;
1002 else if (rc != 0) {
1003 rc = -ETXTBSY;
1004 goto undo_rename;
1005 }
1006 cifsInode->delete_pending = true;
1007 }
Jeff Layton7ce86d52008-09-24 11:32:59 -04001008
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001009out_close:
1010 CIFSSMBClose(xid, tcon, netfid);
1011out:
Steve French32709582008-10-20 00:44:19 +00001012 kfree(info_buf);
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001013 return rc;
Steve French32709582008-10-20 00:44:19 +00001014
1015 /*
1016 * reset everything back to the original state. Don't bother
1017 * dealing with errors here since we can't do anything about
1018 * them anyway.
1019 */
1020undo_rename:
1021 CIFSSMBRenameOpenFile(xid, tcon, netfid, dentry->d_name.name,
1022 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1023 CIFS_MOUNT_MAP_SPECIAL_CHR);
1024undo_setattr:
1025 if (dosattr != origattr) {
1026 info_buf->Attributes = cpu_to_le32(origattr);
1027 if (!CIFSSMBSetFileInfo(xid, tcon, info_buf, netfid,
1028 current->tgid))
1029 cifsInode->cifsAttrs = origattr;
1030 }
1031
1032 goto out_close;
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001033}
1034
Steve Frenchff694522009-04-20 19:45:13 +00001035
1036/*
1037 * If dentry->d_inode is null (usually meaning the cached dentry
1038 * is a negative dentry) then we would attempt a standard SMB delete, but
André Goddard Rosaaf901ca2009-11-14 13:09:05 -02001039 * if that fails we can not attempt the fall back mechanisms on EACCESS
1040 * but will return the EACCESS to the caller. Note that the VFS does not call
Steve Frenchff694522009-04-20 19:45:13 +00001041 * unlink on negative dentries currently.
1042 */
Jeff Layton5f0319a2008-09-16 14:05:16 -04001043int cifs_unlink(struct inode *dir, struct dentry *dentry)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001044{
1045 int rc = 0;
1046 int xid;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001047 char *full_path = NULL;
Jeff Layton5f0319a2008-09-16 14:05:16 -04001048 struct inode *inode = dentry->d_inode;
Steve Frenchff694522009-04-20 19:45:13 +00001049 struct cifsInodeInfo *cifs_inode;
Jeff Layton5f0319a2008-09-16 14:05:16 -04001050 struct super_block *sb = dir->i_sb;
1051 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
1052 struct cifsTconInfo *tcon = cifs_sb->tcon;
Steve French60502472008-10-07 18:42:52 +00001053 struct iattr *attrs = NULL;
1054 __u32 dosattr = 0, origattr = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001055
Joe Perchesb6b38f72010-04-21 03:50:45 +00001056 cFYI(1, "cifs_unlink, dir=0x%p, dentry=0x%p", dir, dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001057
1058 xid = GetXid();
1059
Jeff Layton5f0319a2008-09-16 14:05:16 -04001060 /* Unlink can be called from rename so we can not take the
1061 * sb->s_vfs_rename_mutex here */
1062 full_path = build_path_from_dentry(dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001063 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301064 rc = -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001065 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301066 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001067 }
Steve French2d785a52007-07-15 01:48:57 +00001068
Jeff Layton5f0319a2008-09-16 14:05:16 -04001069 if ((tcon->ses->capabilities & CAP_UNIX) &&
Steve French2d785a52007-07-15 01:48:57 +00001070 (CIFS_UNIX_POSIX_PATH_OPS_CAP &
Jeff Layton5f0319a2008-09-16 14:05:16 -04001071 le64_to_cpu(tcon->fsUnixInfo.Capability))) {
1072 rc = CIFSPOSIXDelFile(xid, tcon, full_path,
Steve French2d785a52007-07-15 01:48:57 +00001073 SMB_POSIX_UNLINK_FILE_TARGET, cifs_sb->local_nls,
1074 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001075 cFYI(1, "posix del rc %d", rc);
Steve French2d785a52007-07-15 01:48:57 +00001076 if ((rc == 0) || (rc == -ENOENT))
1077 goto psx_del_no_retry;
1078 }
1079
Steve French60502472008-10-07 18:42:52 +00001080retry_std_delete:
Jeff Layton5f0319a2008-09-16 14:05:16 -04001081 rc = CIFSSMBDelFile(xid, tcon, full_path, cifs_sb->local_nls,
Steve French737b7582005-04-28 22:41:06 -07001082 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French60502472008-10-07 18:42:52 +00001083
Steve French2d785a52007-07-15 01:48:57 +00001084psx_del_no_retry:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001085 if (!rc) {
Jeff Layton5f0319a2008-09-16 14:05:16 -04001086 if (inode)
1087 drop_nlink(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001088 } else if (rc == -ENOENT) {
Jeff Layton5f0319a2008-09-16 14:05:16 -04001089 d_drop(dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001090 } else if (rc == -ETXTBSY) {
Steve French32709582008-10-20 00:44:19 +00001091 rc = cifs_rename_pending_delete(full_path, dentry, xid);
Jeff Laytona12a1ac2008-09-23 11:48:35 -04001092 if (rc == 0)
1093 drop_nlink(inode);
Steve Frenchff694522009-04-20 19:45:13 +00001094 } else if ((rc == -EACCES) && (dosattr == 0) && inode) {
Steve French388e57b2008-09-16 23:50:58 +00001095 attrs = kzalloc(sizeof(*attrs), GFP_KERNEL);
1096 if (attrs == NULL) {
1097 rc = -ENOMEM;
1098 goto out_reval;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001099 }
Steve French388e57b2008-09-16 23:50:58 +00001100
1101 /* try to reset dos attributes */
Steve Frenchff694522009-04-20 19:45:13 +00001102 cifs_inode = CIFS_I(inode);
1103 origattr = cifs_inode->cifsAttrs;
Steve French60502472008-10-07 18:42:52 +00001104 if (origattr == 0)
1105 origattr |= ATTR_NORMAL;
1106 dosattr = origattr & ~ATTR_READONLY;
Steve French388e57b2008-09-16 23:50:58 +00001107 if (dosattr == 0)
1108 dosattr |= ATTR_NORMAL;
1109 dosattr |= ATTR_HIDDEN;
1110
1111 rc = cifs_set_file_info(inode, attrs, xid, full_path, dosattr);
Steve French388e57b2008-09-16 23:50:58 +00001112 if (rc != 0)
1113 goto out_reval;
Steve French60502472008-10-07 18:42:52 +00001114
1115 goto retry_std_delete;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001116 }
Steve French60502472008-10-07 18:42:52 +00001117
1118 /* undo the setattr if we errored out and it's needed */
1119 if (rc != 0 && dosattr != 0)
1120 cifs_set_file_info(inode, attrs, xid, full_path, origattr);
1121
Steve French388e57b2008-09-16 23:50:58 +00001122out_reval:
Jeff Layton5f0319a2008-09-16 14:05:16 -04001123 if (inode) {
Steve Frenchff694522009-04-20 19:45:13 +00001124 cifs_inode = CIFS_I(inode);
1125 cifs_inode->time = 0; /* will force revalidate to get info
Steve Frenchb2aeb9d2005-05-17 13:16:18 -05001126 when needed */
Jeff Layton5f0319a2008-09-16 14:05:16 -04001127 inode->i_ctime = current_fs_time(sb);
Steve Frenchb2aeb9d2005-05-17 13:16:18 -05001128 }
Jeff Layton5f0319a2008-09-16 14:05:16 -04001129 dir->i_ctime = dir->i_mtime = current_fs_time(sb);
Steve Frenchff694522009-04-20 19:45:13 +00001130 cifs_inode = CIFS_I(dir);
Steve French60502472008-10-07 18:42:52 +00001131 CIFS_I(dir)->time = 0; /* force revalidate of dir as well */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001132
1133 kfree(full_path);
Steve French60502472008-10-07 18:42:52 +00001134 kfree(attrs);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001135 FreeXid(xid);
1136 return rc;
1137}
1138
1139int cifs_mkdir(struct inode *inode, struct dentry *direntry, int mode)
1140{
Jeff Layton6b37faa2008-10-06 21:54:41 +00001141 int rc = 0, tmprc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001142 int xid;
1143 struct cifs_sb_info *cifs_sb;
1144 struct cifsTconInfo *pTcon;
1145 char *full_path = NULL;
1146 struct inode *newinode = NULL;
Jeff Laytoncc0bad72009-06-25 00:56:52 -04001147 struct cifs_fattr fattr;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001148
Joe Perchesb6b38f72010-04-21 03:50:45 +00001149 cFYI(1, "In cifs_mkdir, mode = 0x%x inode = 0x%p", mode, inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001150
1151 xid = GetXid();
1152
1153 cifs_sb = CIFS_SB(inode->i_sb);
1154 pTcon = cifs_sb->tcon;
1155
Steve French7f573562005-08-30 11:32:14 -07001156 full_path = build_path_from_dentry(direntry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001157 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301158 rc = -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001159 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301160 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001161 }
Steve French50c2f752007-07-13 00:33:32 +00001162
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001163 if ((pTcon->ses->capabilities & CAP_UNIX) &&
1164 (CIFS_UNIX_POSIX_PATH_OPS_CAP &
Steve French2dd29d32007-04-23 22:07:35 +00001165 le64_to_cpu(pTcon->fsUnixInfo.Capability))) {
1166 u32 oplock = 0;
Steve Frenchf6d09982008-01-08 23:18:22 +00001167 FILE_UNIX_BASIC_INFO *pInfo =
Steve French2dd29d32007-04-23 22:07:35 +00001168 kzalloc(sizeof(FILE_UNIX_BASIC_INFO), GFP_KERNEL);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001169 if (pInfo == NULL) {
Steve French2dd29d32007-04-23 22:07:35 +00001170 rc = -ENOMEM;
1171 goto mkdir_out;
1172 }
Steve French50c2f752007-07-13 00:33:32 +00001173
Al Viroce3b0f82009-03-29 19:08:22 -04001174 mode &= ~current_umask();
Steve French2dd29d32007-04-23 22:07:35 +00001175 rc = CIFSPOSIXCreate(xid, pTcon, SMB_O_DIRECTORY | SMB_O_CREAT,
1176 mode, NULL /* netfid */, pInfo, &oplock,
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001177 full_path, cifs_sb->local_nls,
1178 cifs_sb->mnt_cifs_flags &
Steve French2dd29d32007-04-23 22:07:35 +00001179 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve Frenchc45d7072007-09-17 02:04:21 +00001180 if (rc == -EOPNOTSUPP) {
1181 kfree(pInfo);
1182 goto mkdir_retry_old;
1183 } else if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001184 cFYI(1, "posix mkdir returned 0x%x", rc);
Steve French2dd29d32007-04-23 22:07:35 +00001185 d_drop(direntry);
1186 } else {
Cyril Gorcunov8f2376a2007-10-14 17:58:43 +00001187 if (pInfo->Type == cpu_to_le32(-1)) {
1188 /* no return info, go query for it */
Steve French5a07cdf2007-09-16 23:12:47 +00001189 kfree(pInfo);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001190 goto mkdir_get_info;
Steve French5a07cdf2007-09-16 23:12:47 +00001191 }
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001192/*BB check (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID ) to see if need
1193 to set uid/gid */
Steve French2dd29d32007-04-23 22:07:35 +00001194 inc_nlink(inode);
1195 if (pTcon->nocase)
1196 direntry->d_op = &cifs_ci_dentry_ops;
1197 else
1198 direntry->d_op = &cifs_dentry_ops;
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001199
Jeff Laytoncc0bad72009-06-25 00:56:52 -04001200 cifs_unix_basic_to_fattr(&fattr, pInfo, cifs_sb);
1201 newinode = cifs_iget(inode->i_sb, &fattr);
1202 if (!newinode) {
Steve French5a07cdf2007-09-16 23:12:47 +00001203 kfree(pInfo);
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001204 goto mkdir_get_info;
Steve French5a07cdf2007-09-16 23:12:47 +00001205 }
Jeff Layton6b37faa2008-10-06 21:54:41 +00001206
Steve French2dd29d32007-04-23 22:07:35 +00001207 d_instantiate(direntry, newinode);
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001208
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001209#ifdef CONFIG_CIFS_DEBUG2
Joe Perchesb6b38f72010-04-21 03:50:45 +00001210 cFYI(1, "instantiated dentry %p %s to inode %p",
1211 direntry, direntry->d_name.name, newinode);
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001212
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001213 if (newinode->i_nlink != 2)
Joe Perchesb6b38f72010-04-21 03:50:45 +00001214 cFYI(1, "unexpected number of links %d",
1215 newinode->i_nlink);
Steve Frenchcbac3cb2007-04-25 11:46:06 +00001216#endif
Steve French2dd29d32007-04-23 22:07:35 +00001217 }
1218 kfree(pInfo);
1219 goto mkdir_out;
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001220 }
Steve Frenchc45d7072007-09-17 02:04:21 +00001221mkdir_retry_old:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001222 /* BB add setting the equivalent of mode via CreateX w/ACLs */
Steve French737b7582005-04-28 22:41:06 -07001223 rc = CIFSSMBMkDir(xid, pTcon, full_path, cifs_sb->local_nls,
1224 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001225 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001226 cFYI(1, "cifs_mkdir returned 0x%x", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001227 d_drop(direntry);
1228 } else {
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001229mkdir_get_info:
Dave Hansend8c76e62006-09-30 23:29:04 -07001230 inc_nlink(inode);
Steve Frenchc18c8422007-07-18 23:21:09 +00001231 if (pTcon->unix_ext)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001232 rc = cifs_get_inode_info_unix(&newinode, full_path,
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001233 inode->i_sb, xid);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001234 else
1235 rc = cifs_get_inode_info(&newinode, full_path, NULL,
Steve French8b1327f2008-03-14 22:37:16 +00001236 inode->i_sb, xid, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001237
Steve Frenchb92327f2005-08-22 20:09:43 -07001238 if (pTcon->nocase)
1239 direntry->d_op = &cifs_ci_dentry_ops;
1240 else
1241 direntry->d_op = &cifs_dentry_ops;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001242 d_instantiate(direntry, newinode);
Steve French2dd29d32007-04-23 22:07:35 +00001243 /* setting nlink not necessary except in cases where we
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001244 * failed to get it from the server or was set bogus */
Steve French2dd29d32007-04-23 22:07:35 +00001245 if ((direntry->d_inode) && (direntry->d_inode->i_nlink < 2))
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001246 direntry->d_inode->i_nlink = 2;
Jeff Layton95089912008-08-06 04:39:02 +00001247
Al Viroce3b0f82009-03-29 19:08:22 -04001248 mode &= ~current_umask();
Jeff Layton95089912008-08-06 04:39:02 +00001249 /* must turn on setgid bit if parent dir has it */
1250 if (inode->i_mode & S_ISGID)
1251 mode |= S_ISGID;
1252
Steve Frenchc18c8422007-07-18 23:21:09 +00001253 if (pTcon->unix_ext) {
Jeff Layton4e1e7fb2008-08-02 07:26:12 -04001254 struct cifs_unix_set_info_args args = {
1255 .mode = mode,
1256 .ctime = NO_CHANGE_64,
1257 .atime = NO_CHANGE_64,
1258 .mtime = NO_CHANGE_64,
1259 .device = 0,
1260 };
Steve Frenchd0d2f2d2005-06-02 15:12:36 -07001261 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID) {
David Howellsa001e5b2008-11-14 10:38:47 +11001262 args.uid = (__u64)current_fsuid();
Jeff Layton95089912008-08-06 04:39:02 +00001263 if (inode->i_mode & S_ISGID)
1264 args.gid = (__u64)inode->i_gid;
1265 else
David Howellsa001e5b2008-11-14 10:38:47 +11001266 args.gid = (__u64)current_fsgid();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001267 } else {
Jeff Layton4e1e7fb2008-08-02 07:26:12 -04001268 args.uid = NO_CHANGE_64;
1269 args.gid = NO_CHANGE_64;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001270 }
Jeff Layton01ea95e2009-07-09 20:02:49 -04001271 CIFSSMBUnixSetPathInfo(xid, pTcon, full_path, &args,
1272 cifs_sb->local_nls,
1273 cifs_sb->mnt_cifs_flags &
1274 CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French3ce53fc2007-06-08 14:55:14 +00001275 } else {
Jeff Layton67750fb2008-05-09 22:28:02 +00001276 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL) &&
1277 (mode & S_IWUGO) == 0) {
1278 FILE_BASIC_INFO pInfo;
Jeff Layton6b37faa2008-10-06 21:54:41 +00001279 struct cifsInodeInfo *cifsInode;
1280 u32 dosattrs;
1281
Jeff Layton67750fb2008-05-09 22:28:02 +00001282 memset(&pInfo, 0, sizeof(pInfo));
Jeff Layton6b37faa2008-10-06 21:54:41 +00001283 cifsInode = CIFS_I(newinode);
1284 dosattrs = cifsInode->cifsAttrs|ATTR_READONLY;
1285 pInfo.Attributes = cpu_to_le32(dosattrs);
1286 tmprc = CIFSSMBSetPathInfo(xid, pTcon,
1287 full_path, &pInfo,
1288 cifs_sb->local_nls,
Jeff Layton67750fb2008-05-09 22:28:02 +00001289 cifs_sb->mnt_cifs_flags &
1290 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Layton6b37faa2008-10-06 21:54:41 +00001291 if (tmprc == 0)
1292 cifsInode->cifsAttrs = dosattrs;
Jeff Layton67750fb2008-05-09 22:28:02 +00001293 }
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001294 if (direntry->d_inode) {
Jeff Laytonb0fd30d2008-05-22 09:33:34 -04001295 if (cifs_sb->mnt_cifs_flags &
1296 CIFS_MOUNT_DYNPERM)
1297 direntry->d_inode->i_mode =
1298 (mode | S_IFDIR);
Steve French4e94a102008-05-23 18:22:46 +00001299
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001300 if (cifs_sb->mnt_cifs_flags &
Steve French6473a552005-11-29 20:20:10 -08001301 CIFS_MOUNT_SET_UID) {
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001302 direntry->d_inode->i_uid =
David Howellsa001e5b2008-11-14 10:38:47 +11001303 current_fsuid();
Jeff Layton95089912008-08-06 04:39:02 +00001304 if (inode->i_mode & S_ISGID)
1305 direntry->d_inode->i_gid =
1306 inode->i_gid;
1307 else
1308 direntry->d_inode->i_gid =
David Howellsa001e5b2008-11-14 10:38:47 +11001309 current_fsgid();
Steve French6473a552005-11-29 20:20:10 -08001310 }
1311 }
Steve French2a138ebb2005-11-29 21:22:19 -08001312 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001313 }
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001314mkdir_out:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001315 kfree(full_path);
1316 FreeXid(xid);
1317 return rc;
1318}
1319
1320int cifs_rmdir(struct inode *inode, struct dentry *direntry)
1321{
1322 int rc = 0;
1323 int xid;
1324 struct cifs_sb_info *cifs_sb;
1325 struct cifsTconInfo *pTcon;
1326 char *full_path = NULL;
1327 struct cifsInodeInfo *cifsInode;
1328
Joe Perchesb6b38f72010-04-21 03:50:45 +00001329 cFYI(1, "cifs_rmdir, inode = 0x%p", inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001330
1331 xid = GetXid();
1332
1333 cifs_sb = CIFS_SB(inode->i_sb);
1334 pTcon = cifs_sb->tcon;
1335
Steve French7f573562005-08-30 11:32:14 -07001336 full_path = build_path_from_dentry(direntry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001337 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301338 rc = -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001339 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301340 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001341 }
1342
Steve French737b7582005-04-28 22:41:06 -07001343 rc = CIFSSMBRmDir(xid, pTcon, full_path, cifs_sb->local_nls,
1344 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001345
1346 if (!rc) {
Dave Hansen9a53c3a2006-09-30 23:29:03 -07001347 drop_nlink(inode);
Steve French3677db12007-02-26 16:46:11 +00001348 spin_lock(&direntry->d_inode->i_lock);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001349 i_size_write(direntry->d_inode, 0);
Dave Hansence71ec32006-09-30 23:29:06 -07001350 clear_nlink(direntry->d_inode);
Steve French3677db12007-02-26 16:46:11 +00001351 spin_unlock(&direntry->d_inode->i_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001352 }
1353
1354 cifsInode = CIFS_I(direntry->d_inode);
1355 cifsInode->time = 0; /* force revalidate to go get info when
1356 needed */
Steve French42c245442009-01-13 22:03:55 +00001357
1358 cifsInode = CIFS_I(inode);
1359 cifsInode->time = 0; /* force revalidate to get parent dir info
1360 since cached search results now invalid */
1361
Linus Torvalds1da177e2005-04-16 15:20:36 -07001362 direntry->d_inode->i_ctime = inode->i_ctime = inode->i_mtime =
1363 current_fs_time(inode->i_sb);
1364
1365 kfree(full_path);
1366 FreeXid(xid);
1367 return rc;
1368}
1369
Steve Frenchee2fd962008-09-23 18:23:33 +00001370static int
1371cifs_do_rename(int xid, struct dentry *from_dentry, const char *fromPath,
1372 struct dentry *to_dentry, const char *toPath)
1373{
1374 struct cifs_sb_info *cifs_sb = CIFS_SB(from_dentry->d_sb);
1375 struct cifsTconInfo *pTcon = cifs_sb->tcon;
1376 __u16 srcfid;
1377 int oplock, rc;
1378
1379 /* try path-based rename first */
1380 rc = CIFSSMBRename(xid, pTcon, fromPath, toPath, cifs_sb->local_nls,
1381 cifs_sb->mnt_cifs_flags &
1382 CIFS_MOUNT_MAP_SPECIAL_CHR);
1383
1384 /*
1385 * don't bother with rename by filehandle unless file is busy and
1386 * source Note that cross directory moves do not work with
1387 * rename by filehandle to various Windows servers.
1388 */
1389 if (rc == 0 || rc != -ETXTBSY)
1390 return rc;
1391
1392 /* open the file to be renamed -- we need DELETE perms */
1393 rc = CIFSSMBOpen(xid, pTcon, fromPath, FILE_OPEN, DELETE,
1394 CREATE_NOT_DIR, &srcfid, &oplock, NULL,
1395 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1396 CIFS_MOUNT_MAP_SPECIAL_CHR);
1397
1398 if (rc == 0) {
1399 rc = CIFSSMBRenameOpenFile(xid, pTcon, srcfid,
1400 (const char *) to_dentry->d_name.name,
1401 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1402 CIFS_MOUNT_MAP_SPECIAL_CHR);
1403
1404 CIFSSMBClose(xid, pTcon, srcfid);
1405 }
1406
1407 return rc;
1408}
1409
Jeff Layton14121bd2008-10-20 14:45:22 -04001410int cifs_rename(struct inode *source_dir, struct dentry *source_dentry,
1411 struct inode *target_dir, struct dentry *target_dentry)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001412{
Steve Frenchee2fd962008-09-23 18:23:33 +00001413 char *fromName = NULL;
1414 char *toName = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001415 struct cifs_sb_info *cifs_sb_source;
1416 struct cifs_sb_info *cifs_sb_target;
Jeff Layton14121bd2008-10-20 14:45:22 -04001417 struct cifsTconInfo *tcon;
Steve Frenchee2fd962008-09-23 18:23:33 +00001418 FILE_UNIX_BASIC_INFO *info_buf_source = NULL;
1419 FILE_UNIX_BASIC_INFO *info_buf_target;
Jeff Layton8d281ef2008-10-22 13:57:01 -04001420 int xid, rc, tmprc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001421
Jeff Layton14121bd2008-10-20 14:45:22 -04001422 cifs_sb_target = CIFS_SB(target_dir->i_sb);
1423 cifs_sb_source = CIFS_SB(source_dir->i_sb);
1424 tcon = cifs_sb_source->tcon;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001425
Steve Frenchee2fd962008-09-23 18:23:33 +00001426 xid = GetXid();
1427
1428 /*
1429 * BB: this might be allowed if same server, but different share.
1430 * Consider adding support for this
1431 */
Jeff Layton14121bd2008-10-20 14:45:22 -04001432 if (tcon != cifs_sb_target->tcon) {
Steve Frenchee2fd962008-09-23 18:23:33 +00001433 rc = -EXDEV;
1434 goto cifs_rename_exit;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001435 }
1436
Steve Frenchee2fd962008-09-23 18:23:33 +00001437 /*
1438 * we already have the rename sem so we do not need to
1439 * grab it again here to protect the path integrity
1440 */
Jeff Layton14121bd2008-10-20 14:45:22 -04001441 fromName = build_path_from_dentry(source_dentry);
Steve Frenchee2fd962008-09-23 18:23:33 +00001442 if (fromName == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001443 rc = -ENOMEM;
1444 goto cifs_rename_exit;
1445 }
1446
Jeff Layton14121bd2008-10-20 14:45:22 -04001447 toName = build_path_from_dentry(target_dentry);
Steve Frenchee2fd962008-09-23 18:23:33 +00001448 if (toName == NULL) {
1449 rc = -ENOMEM;
1450 goto cifs_rename_exit;
1451 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001452
Jeff Layton14121bd2008-10-20 14:45:22 -04001453 rc = cifs_do_rename(xid, source_dentry, fromName,
1454 target_dentry, toName);
Steve Frenchee2fd962008-09-23 18:23:33 +00001455
Jeff Layton14121bd2008-10-20 14:45:22 -04001456 if (rc == -EEXIST && tcon->unix_ext) {
Steve Frenchee2fd962008-09-23 18:23:33 +00001457 /*
Jeff Layton14121bd2008-10-20 14:45:22 -04001458 * Are src and dst hardlinks of same inode? We can
1459 * only tell with unix extensions enabled
Steve Frenchee2fd962008-09-23 18:23:33 +00001460 */
Jeff Layton14121bd2008-10-20 14:45:22 -04001461 info_buf_source =
1462 kmalloc(2 * sizeof(FILE_UNIX_BASIC_INFO),
1463 GFP_KERNEL);
1464 if (info_buf_source == NULL) {
1465 rc = -ENOMEM;
1466 goto cifs_rename_exit;
1467 }
1468
1469 info_buf_target = info_buf_source + 1;
Jeff Layton8d281ef2008-10-22 13:57:01 -04001470 tmprc = CIFSSMBUnixQPathInfo(xid, tcon, fromName,
Jeff Layton14121bd2008-10-20 14:45:22 -04001471 info_buf_source,
1472 cifs_sb_source->local_nls,
1473 cifs_sb_source->mnt_cifs_flags &
1474 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Layton8d281ef2008-10-22 13:57:01 -04001475 if (tmprc != 0)
Jeff Layton14121bd2008-10-20 14:45:22 -04001476 goto unlink_target;
1477
Jeff Layton8d281ef2008-10-22 13:57:01 -04001478 tmprc = CIFSSMBUnixQPathInfo(xid, tcon,
Jeff Layton14121bd2008-10-20 14:45:22 -04001479 toName, info_buf_target,
1480 cifs_sb_target->local_nls,
1481 /* remap based on source sb */
1482 cifs_sb_source->mnt_cifs_flags &
1483 CIFS_MOUNT_MAP_SPECIAL_CHR);
1484
Jeff Layton8d281ef2008-10-22 13:57:01 -04001485 if (tmprc == 0 && (info_buf_source->UniqueId ==
Jeff Laytonae6884a2008-11-03 14:05:08 -05001486 info_buf_target->UniqueId)) {
Jeff Layton14121bd2008-10-20 14:45:22 -04001487 /* same file, POSIX says that this is a noop */
Jeff Laytonae6884a2008-11-03 14:05:08 -05001488 rc = 0;
Jeff Layton14121bd2008-10-20 14:45:22 -04001489 goto cifs_rename_exit;
Jeff Laytonae6884a2008-11-03 14:05:08 -05001490 }
Jeff Layton14121bd2008-10-20 14:45:22 -04001491 } /* else ... BB we could add the same check for Windows by
1492 checking the UniqueId via FILE_INTERNAL_INFO */
1493
Jeff Layton14121bd2008-10-20 14:45:22 -04001494unlink_target:
Jeff Laytonfc6f3942009-04-17 11:45:30 -04001495 /* Try unlinking the target dentry if it's not negative */
1496 if (target_dentry->d_inode && (rc == -EACCES || rc == -EEXIST)) {
Jeff Layton8d281ef2008-10-22 13:57:01 -04001497 tmprc = cifs_unlink(target_dir, target_dentry);
Jeff Layton14121bd2008-10-20 14:45:22 -04001498 if (tmprc)
1499 goto cifs_rename_exit;
1500
Jeff Layton14121bd2008-10-20 14:45:22 -04001501 rc = cifs_do_rename(xid, source_dentry, fromName,
1502 target_dentry, toName);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001503 }
1504
1505cifs_rename_exit:
Steve Frenchee2fd962008-09-23 18:23:33 +00001506 kfree(info_buf_source);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001507 kfree(fromName);
1508 kfree(toName);
1509 FreeXid(xid);
1510 return rc;
1511}
1512
Jeff Laytondf2cf172010-02-12 07:44:16 -05001513static bool
1514cifs_inode_needs_reval(struct inode *inode)
1515{
1516 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
1517
1518 if (cifs_i->clientCanCacheRead)
1519 return false;
1520
1521 if (!lookupCacheEnabled)
1522 return true;
1523
1524 if (cifs_i->time == 0)
1525 return true;
1526
1527 /* FIXME: the actimeo should be tunable */
1528 if (time_after_eq(jiffies, cifs_i->time + HZ))
1529 return true;
1530
1531 return false;
1532}
1533
1534/* check invalid_mapping flag and zap the cache if it's set */
1535static void
1536cifs_invalidate_mapping(struct inode *inode)
1537{
1538 int rc;
1539 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
1540
1541 cifs_i->invalid_mapping = false;
1542
1543 /* write back any cached data */
1544 if (inode->i_mapping && inode->i_mapping->nrpages != 0) {
1545 rc = filemap_write_and_wait(inode->i_mapping);
1546 if (rc)
1547 cifs_i->write_behind_rc = rc;
1548 }
1549 invalidate_remote_inode(inode);
1550}
1551
Jeff Laytonabab0952010-02-12 07:44:18 -05001552int cifs_revalidate_file(struct file *filp)
1553{
1554 int rc = 0;
1555 struct inode *inode = filp->f_path.dentry->d_inode;
1556
1557 if (!cifs_inode_needs_reval(inode))
1558 goto check_inval;
1559
1560 if (CIFS_SB(inode->i_sb)->tcon->unix_ext)
1561 rc = cifs_get_file_info_unix(filp);
1562 else
1563 rc = cifs_get_file_info(filp);
1564
1565check_inval:
1566 if (CIFS_I(inode)->invalid_mapping)
1567 cifs_invalidate_mapping(inode);
1568
1569 return rc;
1570}
1571
Jeff Laytondf2cf172010-02-12 07:44:16 -05001572/* revalidate a dentry's inode attributes */
1573int cifs_revalidate_dentry(struct dentry *dentry)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001574{
1575 int xid;
Jeff Laytondf2cf172010-02-12 07:44:16 -05001576 int rc = 0;
1577 char *full_path = NULL;
1578 struct inode *inode = dentry->d_inode;
1579 struct super_block *sb = dentry->d_sb;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001580
Jeff Laytondf2cf172010-02-12 07:44:16 -05001581 if (inode == NULL)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001582 return -ENOENT;
1583
Linus Torvalds1da177e2005-04-16 15:20:36 -07001584 xid = GetXid();
1585
Jeff Laytondf2cf172010-02-12 07:44:16 -05001586 if (!cifs_inode_needs_reval(inode))
1587 goto check_inval;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001588
1589 /* can not safely grab the rename sem here if rename calls revalidate
1590 since that would deadlock */
Jeff Laytondf2cf172010-02-12 07:44:16 -05001591 full_path = build_path_from_dentry(dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001592 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301593 rc = -ENOMEM;
Jeff Laytondf2cf172010-02-12 07:44:16 -05001594 goto check_inval;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001595 }
Jeff Laytondf2cf172010-02-12 07:44:16 -05001596
Steve Frenchf19159d2010-04-21 04:12:10 +00001597 cFYI(1, "Revalidate: %s inode 0x%p count %d dentry: 0x%p d_time %ld "
Jeff Laytondf2cf172010-02-12 07:44:16 -05001598 "jiffies %ld", full_path, inode, inode->i_count.counter,
Steve Frenchf19159d2010-04-21 04:12:10 +00001599 dentry, dentry->d_time, jiffies);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001600
Jeff Laytondf2cf172010-02-12 07:44:16 -05001601 if (CIFS_SB(sb)->tcon->unix_ext)
1602 rc = cifs_get_inode_info_unix(&inode, full_path, sb, xid);
1603 else
1604 rc = cifs_get_inode_info(&inode, full_path, NULL, sb,
1605 xid, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001606
Jeff Laytondf2cf172010-02-12 07:44:16 -05001607check_inval:
1608 if (CIFS_I(inode)->invalid_mapping)
1609 cifs_invalidate_mapping(inode);
Steve French50c2f752007-07-13 00:33:32 +00001610
Linus Torvalds1da177e2005-04-16 15:20:36 -07001611 kfree(full_path);
1612 FreeXid(xid);
1613 return rc;
1614}
1615
1616int cifs_getattr(struct vfsmount *mnt, struct dentry *dentry,
1617 struct kstat *stat)
1618{
Jeff Laytondf2cf172010-02-12 07:44:16 -05001619 int err = cifs_revalidate_dentry(dentry);
Steve French5fe14c82006-11-07 19:26:33 +00001620 if (!err) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001621 generic_fillattr(dentry->d_inode, stat);
Steve French5fe14c82006-11-07 19:26:33 +00001622 stat->blksize = CIFS_MAX_MSGSIZE;
Jeff Laytoncc0bad72009-06-25 00:56:52 -04001623 stat->ino = CIFS_I(dentry->d_inode)->uniqueid;
Steve French5fe14c82006-11-07 19:26:33 +00001624 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001625 return err;
1626}
1627
1628static int cifs_truncate_page(struct address_space *mapping, loff_t from)
1629{
1630 pgoff_t index = from >> PAGE_CACHE_SHIFT;
1631 unsigned offset = from & (PAGE_CACHE_SIZE - 1);
1632 struct page *page;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001633 int rc = 0;
1634
1635 page = grab_cache_page(mapping, index);
1636 if (!page)
1637 return -ENOMEM;
1638
Christoph Lametereebd2aa2008-02-04 22:28:29 -08001639 zero_user_segment(page, offset, PAGE_CACHE_SIZE);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001640 unlock_page(page);
1641 page_cache_release(page);
1642 return rc;
1643}
1644
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001645static int cifs_vmtruncate(struct inode *inode, loff_t offset)
Steve French3677db12007-02-26 16:46:11 +00001646{
npiggin@suse.dec08d3b02009-08-21 02:35:06 +10001647 loff_t oldsize;
1648 int err;
Steve French3677db12007-02-26 16:46:11 +00001649
Steve Frenchba6a46a2007-02-26 20:06:29 +00001650 spin_lock(&inode->i_lock);
npiggin@suse.dec08d3b02009-08-21 02:35:06 +10001651 err = inode_newsize_ok(inode, offset);
1652 if (err) {
Steve Frenchba6a46a2007-02-26 20:06:29 +00001653 spin_unlock(&inode->i_lock);
npiggin@suse.dec08d3b02009-08-21 02:35:06 +10001654 goto out;
Steve Frenchba6a46a2007-02-26 20:06:29 +00001655 }
Steve French3677db12007-02-26 16:46:11 +00001656
npiggin@suse.dec08d3b02009-08-21 02:35:06 +10001657 oldsize = inode->i_size;
Steve French3677db12007-02-26 16:46:11 +00001658 i_size_write(inode, offset);
Steve Frenchba6a46a2007-02-26 20:06:29 +00001659 spin_unlock(&inode->i_lock);
npiggin@suse.dec08d3b02009-08-21 02:35:06 +10001660 truncate_pagecache(inode, oldsize, offset);
Al Viroacfa4382008-12-04 10:06:33 -05001661 if (inode->i_op->truncate)
Steve French3677db12007-02-26 16:46:11 +00001662 inode->i_op->truncate(inode);
npiggin@suse.dec08d3b02009-08-21 02:35:06 +10001663out:
1664 return err;
Steve French3677db12007-02-26 16:46:11 +00001665}
1666
Jeff Layton8efdbde2008-07-23 21:28:12 +00001667static int
1668cifs_set_file_size(struct inode *inode, struct iattr *attrs,
1669 int xid, char *full_path)
1670{
1671 int rc;
1672 struct cifsFileInfo *open_file;
1673 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
1674 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
1675 struct cifsTconInfo *pTcon = cifs_sb->tcon;
1676
1677 /*
1678 * To avoid spurious oplock breaks from server, in the case of
1679 * inodes that we already have open, avoid doing path based
1680 * setting of file size if we can do it by handle.
1681 * This keeps our caching token (oplock) and avoids timeouts
1682 * when the local oplock break takes longer to flush
1683 * writebehind data than the SMB timeout for the SetPathInfo
1684 * request would allow
1685 */
1686 open_file = find_writable_file(cifsInode);
1687 if (open_file) {
1688 __u16 nfid = open_file->netfid;
1689 __u32 npid = open_file->pid;
1690 rc = CIFSSMBSetFileSize(xid, pTcon, attrs->ia_size, nfid,
1691 npid, false);
Dave Kleikamp6ab409b2009-08-31 11:07:12 -04001692 cifsFileInfo_put(open_file);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001693 cFYI(1, "SetFSize for attrs rc = %d", rc);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001694 if ((rc == -EINVAL) || (rc == -EOPNOTSUPP)) {
1695 unsigned int bytes_written;
1696 rc = CIFSSMBWrite(xid, pTcon, nfid, 0, attrs->ia_size,
1697 &bytes_written, NULL, NULL, 1);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001698 cFYI(1, "Wrt seteof rc %d", rc);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001699 }
1700 } else
1701 rc = -EINVAL;
1702
1703 if (rc != 0) {
1704 /* Set file size by pathname rather than by handle
1705 either because no valid, writeable file handle for
1706 it was found or because there was an error setting
1707 it by handle */
1708 rc = CIFSSMBSetEOF(xid, pTcon, full_path, attrs->ia_size,
1709 false, cifs_sb->local_nls,
1710 cifs_sb->mnt_cifs_flags &
1711 CIFS_MOUNT_MAP_SPECIAL_CHR);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001712 cFYI(1, "SetEOF by path (setattrs) rc = %d", rc);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001713 if ((rc == -EINVAL) || (rc == -EOPNOTSUPP)) {
1714 __u16 netfid;
1715 int oplock = 0;
1716
1717 rc = SMBLegacyOpen(xid, pTcon, full_path,
1718 FILE_OPEN, GENERIC_WRITE,
1719 CREATE_NOT_DIR, &netfid, &oplock, NULL,
1720 cifs_sb->local_nls,
1721 cifs_sb->mnt_cifs_flags &
1722 CIFS_MOUNT_MAP_SPECIAL_CHR);
1723 if (rc == 0) {
1724 unsigned int bytes_written;
1725 rc = CIFSSMBWrite(xid, pTcon, netfid, 0,
1726 attrs->ia_size,
1727 &bytes_written, NULL,
1728 NULL, 1);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001729 cFYI(1, "wrt seteof rc %d", rc);
Jeff Layton8efdbde2008-07-23 21:28:12 +00001730 CIFSSMBClose(xid, pTcon, netfid);
1731 }
1732 }
1733 }
1734
1735 if (rc == 0) {
Jeff Laytonfbec9ab2009-04-03 13:44:00 -04001736 cifsInode->server_eof = attrs->ia_size;
Jeff Layton8efdbde2008-07-23 21:28:12 +00001737 rc = cifs_vmtruncate(inode, attrs->ia_size);
1738 cifs_truncate_page(inode->i_mapping, inode->i_size);
1739 }
1740
1741 return rc;
1742}
1743
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001744static int
1745cifs_setattr_unix(struct dentry *direntry, struct iattr *attrs)
1746{
1747 int rc;
1748 int xid;
1749 char *full_path = NULL;
1750 struct inode *inode = direntry->d_inode;
1751 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
1752 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
1753 struct cifsTconInfo *pTcon = cifs_sb->tcon;
1754 struct cifs_unix_set_info_args *args = NULL;
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001755 struct cifsFileInfo *open_file;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001756
Joe Perchesb6b38f72010-04-21 03:50:45 +00001757 cFYI(1, "setattr_unix on file %s attrs->ia_valid=0x%x",
1758 direntry->d_name.name, attrs->ia_valid);
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001759
1760 xid = GetXid();
1761
1762 if ((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_PERM) == 0) {
1763 /* check if we have permission to change attrs */
1764 rc = inode_change_ok(inode, attrs);
1765 if (rc < 0)
1766 goto out;
1767 else
1768 rc = 0;
1769 }
1770
1771 full_path = build_path_from_dentry(direntry);
1772 if (full_path == NULL) {
1773 rc = -ENOMEM;
1774 goto out;
1775 }
1776
Jeff Layton0f4d6342009-03-26 13:35:37 -04001777 /*
1778 * Attempt to flush data before changing attributes. We need to do
1779 * this for ATTR_SIZE and ATTR_MTIME for sure, and if we change the
1780 * ownership or mode then we may also need to do this. Here, we take
1781 * the safe way out and just do the flush on all setattr requests. If
1782 * the flush returns error, store it to report later and continue.
1783 *
1784 * BB: This should be smarter. Why bother flushing pages that
1785 * will be truncated anyway? Also, should we error out here if
1786 * the flush returns error?
1787 */
1788 rc = filemap_write_and_wait(inode->i_mapping);
1789 if (rc != 0) {
1790 cifsInode->write_behind_rc = rc;
1791 rc = 0;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001792 }
1793
1794 if (attrs->ia_valid & ATTR_SIZE) {
1795 rc = cifs_set_file_size(inode, attrs, xid, full_path);
1796 if (rc != 0)
1797 goto out;
1798 }
1799
1800 /* skip mode change if it's just for clearing setuid/setgid */
1801 if (attrs->ia_valid & (ATTR_KILL_SUID|ATTR_KILL_SGID))
1802 attrs->ia_valid &= ~ATTR_MODE;
1803
1804 args = kmalloc(sizeof(*args), GFP_KERNEL);
1805 if (args == NULL) {
1806 rc = -ENOMEM;
1807 goto out;
1808 }
1809
1810 /* set up the struct */
1811 if (attrs->ia_valid & ATTR_MODE)
1812 args->mode = attrs->ia_mode;
1813 else
1814 args->mode = NO_CHANGE_64;
1815
1816 if (attrs->ia_valid & ATTR_UID)
1817 args->uid = attrs->ia_uid;
1818 else
1819 args->uid = NO_CHANGE_64;
1820
1821 if (attrs->ia_valid & ATTR_GID)
1822 args->gid = attrs->ia_gid;
1823 else
1824 args->gid = NO_CHANGE_64;
1825
1826 if (attrs->ia_valid & ATTR_ATIME)
1827 args->atime = cifs_UnixTimeToNT(attrs->ia_atime);
1828 else
1829 args->atime = NO_CHANGE_64;
1830
1831 if (attrs->ia_valid & ATTR_MTIME)
1832 args->mtime = cifs_UnixTimeToNT(attrs->ia_mtime);
1833 else
1834 args->mtime = NO_CHANGE_64;
1835
1836 if (attrs->ia_valid & ATTR_CTIME)
1837 args->ctime = cifs_UnixTimeToNT(attrs->ia_ctime);
1838 else
1839 args->ctime = NO_CHANGE_64;
1840
1841 args->device = 0;
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001842 open_file = find_writable_file(cifsInode);
1843 if (open_file) {
1844 u16 nfid = open_file->netfid;
1845 u32 npid = open_file->pid;
1846 rc = CIFSSMBUnixSetFileInfo(xid, pTcon, args, nfid, npid);
Dave Kleikamp6ab409b2009-08-31 11:07:12 -04001847 cifsFileInfo_put(open_file);
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001848 } else {
1849 rc = CIFSSMBUnixSetPathInfo(xid, pTcon, full_path, args,
Jeff Layton01ea95e2009-07-09 20:02:49 -04001850 cifs_sb->local_nls,
1851 cifs_sb->mnt_cifs_flags &
1852 CIFS_MOUNT_MAP_SPECIAL_CHR);
Jeff Layton3bbeeb32009-07-09 20:02:50 -04001853 }
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001854
Steve Frenchccd4bb12010-02-08 17:39:58 +00001855 if (!rc) {
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001856 rc = inode_setattr(inode, attrs);
Steve Frenchccd4bb12010-02-08 17:39:58 +00001857
1858 /* force revalidate when any of these times are set since some
1859 of the fs types (eg ext3, fat) do not have fine enough
1860 time granularity to match protocol, and we do not have a
1861 a way (yet) to query the server fs's time granularity (and
1862 whether it rounds times down).
1863 */
1864 if (!rc && (attrs->ia_valid & (ATTR_MTIME | ATTR_CTIME)))
1865 cifsInode->time = 0;
1866 }
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001867out:
1868 kfree(args);
1869 kfree(full_path);
1870 FreeXid(xid);
1871 return rc;
1872}
1873
Jeff Layton0510eeb2008-08-02 07:26:12 -04001874static int
1875cifs_setattr_nounix(struct dentry *direntry, struct iattr *attrs)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001876{
1877 int xid;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001878 struct inode *inode = direntry->d_inode;
1879 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001880 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001881 char *full_path = NULL;
1882 int rc = -EACCES;
Jeff Laytonfeb3e202008-08-02 07:26:12 -04001883 __u32 dosattr = 0;
Jeff Layton4e1e7fb2008-08-02 07:26:12 -04001884 __u64 mode = NO_CHANGE_64;
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001885
Linus Torvalds1da177e2005-04-16 15:20:36 -07001886 xid = GetXid();
1887
Joe Perchesb6b38f72010-04-21 03:50:45 +00001888 cFYI(1, "setattr on file %s attrs->iavalid 0x%x",
1889 direntry->d_name.name, attrs->ia_valid);
Steve French6473a552005-11-29 20:20:10 -08001890
Steve French2a138ebb2005-11-29 21:22:19 -08001891 if ((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_PERM) == 0) {
Steve French6473a552005-11-29 20:20:10 -08001892 /* check if we have permission to change attrs */
Jeff Layton02eadef2008-05-09 21:26:11 +00001893 rc = inode_change_ok(inode, attrs);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001894 if (rc < 0) {
Steve French6473a552005-11-29 20:20:10 -08001895 FreeXid(xid);
1896 return rc;
1897 } else
1898 rc = 0;
1899 }
Steve French50c2f752007-07-13 00:33:32 +00001900
Steve French7f573562005-08-30 11:32:14 -07001901 full_path = build_path_from_dentry(direntry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001902 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301903 rc = -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001904 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301905 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001906 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001907
Jeff Layton0f4d6342009-03-26 13:35:37 -04001908 /*
1909 * Attempt to flush data before changing attributes. We need to do
1910 * this for ATTR_SIZE and ATTR_MTIME for sure, and if we change the
1911 * ownership or mode then we may also need to do this. Here, we take
1912 * the safe way out and just do the flush on all setattr requests. If
1913 * the flush returns error, store it to report later and continue.
1914 *
1915 * BB: This should be smarter. Why bother flushing pages that
1916 * will be truncated anyway? Also, should we error out here if
1917 * the flush returns error?
1918 */
1919 rc = filemap_write_and_wait(inode->i_mapping);
1920 if (rc != 0) {
1921 cifsInode->write_behind_rc = rc;
1922 rc = 0;
Steve French50531442008-03-14 19:21:31 +00001923 }
Jeff Laytoncea21802007-11-20 23:19:03 +00001924
Steve French50531442008-03-14 19:21:31 +00001925 if (attrs->ia_valid & ATTR_SIZE) {
Jeff Layton8efdbde2008-07-23 21:28:12 +00001926 rc = cifs_set_file_size(inode, attrs, xid, full_path);
1927 if (rc != 0)
Steve Frenche30dcf32005-09-20 20:49:16 -07001928 goto cifs_setattr_exit;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001929 }
Jeff Layton4ca691a2008-05-22 09:33:34 -04001930
1931 /*
1932 * Without unix extensions we can't send ownership changes to the
1933 * server, so silently ignore them. This is consistent with how
1934 * local DOS/Windows filesystems behave (VFAT, NTFS, etc). With
1935 * CIFSACL support + proper Windows to Unix idmapping, we may be
1936 * able to support this in the future.
1937 */
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001938 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID))
Jeff Layton4ca691a2008-05-22 09:33:34 -04001939 attrs->ia_valid &= ~(ATTR_UID | ATTR_GID);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001940
Jeff Laytond32c4f22007-10-18 03:05:22 -07001941 /* skip mode change if it's just for clearing setuid/setgid */
1942 if (attrs->ia_valid & (ATTR_KILL_SUID|ATTR_KILL_SGID))
1943 attrs->ia_valid &= ~ATTR_MODE;
1944
Linus Torvalds1da177e2005-04-16 15:20:36 -07001945 if (attrs->ia_valid & ATTR_MODE) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001946 cFYI(1, "Mode changed to 0%o", attrs->ia_mode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001947 mode = attrs->ia_mode;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001948 }
1949
Jeff Layton3fe5c1d2008-08-02 07:26:12 -04001950 if (attrs->ia_valid & ATTR_MODE) {
Steve Frenchcdbce9c82005-11-19 21:04:52 -08001951 rc = 0;
Steve French97837582007-12-31 07:47:21 +00001952#ifdef CONFIG_CIFS_EXPERIMENTAL
1953 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL)
Jeff Layton02eadef2008-05-09 21:26:11 +00001954 rc = mode_to_acl(inode, full_path, mode);
Jeff Layton51328612008-05-22 09:33:34 -04001955 else
Steve French97837582007-12-31 07:47:21 +00001956#endif
Jeff Layton51328612008-05-22 09:33:34 -04001957 if (((mode & S_IWUGO) == 0) &&
1958 (cifsInode->cifsAttrs & ATTR_READONLY) == 0) {
Jeff Laytonfeb3e202008-08-02 07:26:12 -04001959
1960 dosattr = cifsInode->cifsAttrs | ATTR_READONLY;
1961
Jeff Layton51328612008-05-22 09:33:34 -04001962 /* fix up mode if we're not using dynperm */
1963 if ((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM) == 0)
1964 attrs->ia_mode = inode->i_mode & ~S_IWUGO;
1965 } else if ((mode & S_IWUGO) &&
1966 (cifsInode->cifsAttrs & ATTR_READONLY)) {
Jeff Laytonfeb3e202008-08-02 07:26:12 -04001967
1968 dosattr = cifsInode->cifsAttrs & ~ATTR_READONLY;
1969 /* Attributes of 0 are ignored */
1970 if (dosattr == 0)
1971 dosattr |= ATTR_NORMAL;
Jeff Layton51328612008-05-22 09:33:34 -04001972
1973 /* reset local inode permissions to normal */
1974 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM)) {
1975 attrs->ia_mode &= ~(S_IALLUGO);
1976 if (S_ISDIR(inode->i_mode))
1977 attrs->ia_mode |=
1978 cifs_sb->mnt_dir_mode;
1979 else
1980 attrs->ia_mode |=
1981 cifs_sb->mnt_file_mode;
1982 }
1983 } else if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM)) {
1984 /* ignore mode change - ATTR_READONLY hasn't changed */
1985 attrs->ia_valid &= ~ATTR_MODE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001986 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001987 }
1988
Jeff Laytonfeb3e202008-08-02 07:26:12 -04001989 if (attrs->ia_valid & (ATTR_MTIME|ATTR_ATIME|ATTR_CTIME) ||
1990 ((attrs->ia_valid & ATTR_MODE) && dosattr)) {
1991 rc = cifs_set_file_info(inode, attrs, xid, full_path, dosattr);
1992 /* BB: check for rc = -EOPNOTSUPP and switch to legacy mode */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001993
Steve Frenche30dcf32005-09-20 20:49:16 -07001994 /* Even if error on time set, no sense failing the call if
1995 the server would set the time to a reasonable value anyway,
1996 and this check ensures that we are not being called from
1997 sys_utimes in which case we ought to fail the call back to
1998 the user when the server rejects the call */
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001999 if ((rc) && (attrs->ia_valid &
Jeff Laytonfeb3e202008-08-02 07:26:12 -04002000 (ATTR_MODE | ATTR_GID | ATTR_UID | ATTR_SIZE)))
Steve Frenche30dcf32005-09-20 20:49:16 -07002001 rc = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002002 }
2003
2004 /* do not need local check to inode_check_ok since the server does
2005 that */
2006 if (!rc)
Jeff Layton02eadef2008-05-09 21:26:11 +00002007 rc = inode_setattr(inode, attrs);
Steve Frenche30dcf32005-09-20 20:49:16 -07002008cifs_setattr_exit:
Linus Torvalds1da177e2005-04-16 15:20:36 -07002009 kfree(full_path);
2010 FreeXid(xid);
2011 return rc;
2012}
2013
Jeff Layton0510eeb2008-08-02 07:26:12 -04002014int
2015cifs_setattr(struct dentry *direntry, struct iattr *attrs)
2016{
2017 struct inode *inode = direntry->d_inode;
2018 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
2019 struct cifsTconInfo *pTcon = cifs_sb->tcon;
2020
2021 if (pTcon->unix_ext)
2022 return cifs_setattr_unix(direntry, attrs);
2023
2024 return cifs_setattr_nounix(direntry, attrs);
2025
2026 /* BB: add cifs_setattr_legacy for really old servers */
2027}
2028
Steve French99ee4db2007-02-27 05:35:17 +00002029#if 0
Linus Torvalds1da177e2005-04-16 15:20:36 -07002030void cifs_delete_inode(struct inode *inode)
2031{
Joe Perchesb6b38f72010-04-21 03:50:45 +00002032 cFYI(1, "In cifs_delete_inode, inode = 0x%p", inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002033 /* may have to add back in if and when safe distributed caching of
2034 directories added e.g. via FindNotify */
2035}
Steve French99ee4db2007-02-27 05:35:17 +00002036#endif