]> bbs.cooldavid.org Git - net-next-2.6.git/blame - fs/cifs/inode.c
Merge branch 'urgent' of git://git.kernel.org/pub/scm/linux/kernel/git/brodo/pcmcia-2.6
[net-next-2.6.git] / fs / cifs / inode.c
CommitLineData
1da177e4
LT
1/*
2 * fs/cifs/inode.c
3 *
f19159dc 4 * Copyright (C) International Business Machines Corp., 2002,2010
1da177e4
LT
5 * 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>
1da177e4 22#include <linux/stat.h>
5a0e3ad6 23#include <linux/slab.h>
1da177e4
LT
24#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"
9451a9a5 32#include "fscache.h"
1da177e4 33
70eff55d 34
7962670e 35static void cifs_set_ops(struct inode *inode, const bool is_dfs_referral)
70eff55d
CH
36{
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:
bc5b6e24 62#ifdef CONFIG_CIFS_DFS_UPCALL
7962670e
IM
63 if (is_dfs_referral) {
64 inode->i_op = &cifs_dfs_referral_inode_operations;
65 } else {
bc5b6e24
SF
66#else /* NO DFS support, treat as a directory */
67 {
68#endif
7962670e
IM
69 inode->i_op = &cifs_dir_inode_ops;
70 inode->i_fop = &cifs_dir_ops;
71 }
70eff55d
CH
72 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
df2cf170
JL
82/* 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
f19159dc 90 cFYI(1, "%s: revalidating inode %llu", __func__, cifs_i->uniqueid);
df2cf170
JL
91
92 if (inode->i_state & I_NEW) {
f19159dc 93 cFYI(1, "%s: inode %llu is new", __func__, cifs_i->uniqueid);
df2cf170
JL
94 return;
95 }
96
97 /* don't bother with revalidation if we have an oplock */
98 if (cifs_i->clientCanCacheRead) {
f19159dc
SF
99 cFYI(1, "%s: inode %llu is oplocked", __func__,
100 cifs_i->uniqueid);
df2cf170
JL
101 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) {
f19159dc
SF
107 cFYI(1, "%s: inode %llu is unchanged", __func__,
108 cifs_i->uniqueid);
df2cf170
JL
109 return;
110 }
111
f19159dc
SF
112 cFYI(1, "%s: invalidating inode %llu mapping", __func__,
113 cifs_i->uniqueid);
df2cf170
JL
114 cifs_i->invalid_mapping = true;
115}
116
cc0bad75
JL
117/* populate an inode with info from a cifs_fattr struct */
118void
119cifs_fattr_to_inode(struct inode *inode, struct cifs_fattr *fattr)
75f12983 120{
cc0bad75 121 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
0b8f18e3
JL
122 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
123 unsigned long oldtime = cifs_i->time;
cc0bad75 124
df2cf170
JL
125 cifs_revalidate_cache(inode, fattr);
126
cc0bad75
JL
127 inode->i_atime = fattr->cf_atime;
128 inode->i_mtime = fattr->cf_mtime;
129 inode->i_ctime = fattr->cf_ctime;
cc0bad75
JL
130 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
0b8f18e3
JL
135 /* 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
cc0bad75 140 cifs_i->cifsAttrs = fattr->cf_cifsattrs;
75f12983 141
0b8f18e3
JL
142 if (fattr->cf_flags & CIFS_FATTR_NEED_REVAL)
143 cifs_i->time = 0;
144 else
145 cifs_i->time = jiffies;
146
b6b38f70
JP
147 cFYI(1, "inode 0x%p old_time=%ld new_time=%ld", inode,
148 oldtime, cifs_i->time);
0b8f18e3
JL
149
150 cifs_i->delete_pending = fattr->cf_flags & CIFS_FATTR_DELETE_PENDING;
cc0bad75 151
835a36ca 152 cifs_i->server_eof = fattr->cf_eof;
cc0bad75
JL
153 /*
154 * Can't safely change the file size here if the client is writing to
155 * it due to potential races.
156 */
157 spin_lock(&inode->i_lock);
158 if (is_size_safe_to_change(cifs_i, fattr->cf_eof)) {
159 i_size_write(inode, fattr->cf_eof);
160
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 */
166 inode->i_blocks = (512 - 1 + fattr->cf_bytes) >> 9;
167 }
168 spin_unlock(&inode->i_lock);
169
170 cifs_set_ops(inode, fattr->cf_flags & CIFS_FATTR_DFS_REFERRAL);
171}
172
4065c802
JL
173void
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
cc0bad75
JL
184/* 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);
75f12983
CH
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 */
cc0bad75 203 fattr->cf_mode &= ~S_IFMT;
75f12983
CH
204 switch (le32_to_cpu(info->Type)) {
205 case UNIX_FILE:
cc0bad75
JL
206 fattr->cf_mode |= S_IFREG;
207 fattr->cf_dtype = DT_REG;
75f12983
CH
208 break;
209 case UNIX_SYMLINK:
cc0bad75
JL
210 fattr->cf_mode |= S_IFLNK;
211 fattr->cf_dtype = DT_LNK;
75f12983
CH
212 break;
213 case UNIX_DIR:
cc0bad75
JL
214 fattr->cf_mode |= S_IFDIR;
215 fattr->cf_dtype = DT_DIR;
75f12983
CH
216 break;
217 case UNIX_CHARDEV:
cc0bad75
JL
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);
75f12983
CH
222 break;
223 case UNIX_BLOCKDEV:
cc0bad75
JL
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);
75f12983
CH
228 break;
229 case UNIX_FIFO:
cc0bad75
JL
230 fattr->cf_mode |= S_IFIFO;
231 fattr->cf_dtype = DT_FIFO;
75f12983
CH
232 break;
233 case UNIX_SOCKET:
cc0bad75
JL
234 fattr->cf_mode |= S_IFSOCK;
235 fattr->cf_dtype = DT_SOCK;
75f12983
CH
236 break;
237 default:
238 /* safest to call it a file if we do not know */
cc0bad75
JL
239 fattr->cf_mode |= S_IFREG;
240 fattr->cf_dtype = DT_REG;
b6b38f70 241 cFYI(1, "unknown type %d", le32_to_cpu(info->Type));
75f12983
CH
242 break;
243 }
244
cc0bad75
JL
245 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_OVERR_UID)
246 fattr->cf_uid = cifs_sb->mnt_uid;
75f12983 247 else
cc0bad75 248 fattr->cf_uid = le64_to_cpu(info->Uid);
75f12983 249
cc0bad75
JL
250 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_OVERR_GID)
251 fattr->cf_gid = cifs_sb->mnt_gid;
75f12983 252 else
cc0bad75 253 fattr->cf_gid = le64_to_cpu(info->Gid);
75f12983 254
cc0bad75 255 fattr->cf_nlink = le64_to_cpu(info->Nlinks);
75f12983
CH
256}
257
b9a3260f 258/*
cc0bad75
JL
259 * 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).
b9a3260f 264 */
f1230c97 265static void
cc0bad75 266cifs_create_dfs_fattr(struct cifs_fattr *fattr, struct super_block *sb)
0e4bbde9 267{
cc0bad75 268 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
0e4bbde9 269
b6b38f70 270 cFYI(1, "creating fake fattr for DFS referral");
cc0bad75
JL
271
272 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;
0e4bbde9
SF
281}
282
abab095d
JL
283int cifs_get_file_info_unix(struct file *filp)
284{
285 int rc;
286 int xid;
287 FILE_UNIX_BASIC_INFO find_data;
288 struct cifs_fattr fattr;
289 struct inode *inode = filp->f_path.dentry->d_inode;
290 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
291 struct cifsTconInfo *tcon = cifs_sb->tcon;
c21dfb69 292 struct cifsFileInfo *cfile = filp->private_data;
abab095d
JL
293
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
1da177e4 308int cifs_get_inode_info_unix(struct inode **pinode,
cc0bad75
JL
309 const unsigned char *full_path,
310 struct super_block *sb, int xid)
1da177e4 311{
cc0bad75 312 int rc;
0e4bbde9 313 FILE_UNIX_BASIC_INFO find_data;
cc0bad75
JL
314 struct cifs_fattr fattr;
315 struct cifsTconInfo *tcon;
1da177e4 316 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
1da177e4 317
cc0bad75 318 tcon = cifs_sb->tcon;
b6b38f70 319 cFYI(1, "Getting info on %s", full_path);
7962670e 320
1da177e4 321 /* could have done a find first instead but this returns more info */
cc0bad75 322 rc = CIFSSMBUnixQPathInfo(xid, tcon, full_path, &find_data,
737b758c
SF
323 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
324 CIFS_MOUNT_MAP_SPECIAL_CHR);
e911d0cc 325
cc0bad75
JL
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);
330 rc = 0;
331 } else {
332 return rc;
333 }
1da177e4 334
0e4bbde9 335 if (*pinode == NULL) {
cc0bad75 336 /* get new inode */
4065c802 337 cifs_fill_uniqueid(sb, &fattr);
cc0bad75
JL
338 *pinode = cifs_iget(sb, &fattr);
339 if (!*pinode)
0e4bbde9 340 rc = -ENOMEM;
cc0bad75
JL
341 } else {
342 /* we already have inode, update it */
343 cifs_fattr_to_inode(*pinode, &fattr);
0e4bbde9 344 }
1da177e4 345
1da177e4
LT
346 return rc;
347}
348
0b8f18e3
JL
349static int
350cifs_sfu_type(struct cifs_fattr *fattr, const unsigned char *path,
351 struct cifs_sb_info *cifs_sb, int xid)
d6e2f2a4
SF
352{
353 int rc;
4b18f2a9 354 int oplock = 0;
d6e2f2a4
SF
355 __u16 netfid;
356 struct cifsTconInfo *pTcon = cifs_sb->tcon;
86c96b4b 357 char buf[24];
d6e2f2a4 358 unsigned int bytes_read;
fb8c4b14 359 char *pbuf;
d6e2f2a4
SF
360
361 pbuf = buf;
362
0b8f18e3
JL
363 fattr->cf_mode &= ~S_IFMT;
364
365 if (fattr->cf_eof == 0) {
366 fattr->cf_mode |= S_IFIFO;
367 fattr->cf_dtype = DT_FIFO;
d6e2f2a4 368 return 0;
0b8f18e3
JL
369 } else if (fattr->cf_eof < 8) {
370 fattr->cf_mode |= S_IFREG;
371 fattr->cf_dtype = DT_REG;
d6e2f2a4
SF
372 return -EINVAL; /* EOPNOTSUPP? */
373 }
50c2f753 374
d6e2f2a4
SF
375 rc = CIFSSMBOpen(xid, pTcon, path, FILE_OPEN, GENERIC_READ,
376 CREATE_NOT_DIR, &netfid, &oplock, NULL,
377 cifs_sb->local_nls,
378 cifs_sb->mnt_cifs_flags &
379 CIFS_MOUNT_MAP_SPECIAL_CHR);
fb8c4b14 380 if (rc == 0) {
ec637e3f 381 int buf_type = CIFS_NO_BUFFER;
d6e2f2a4 382 /* Read header */
0b8f18e3 383 rc = CIFSSMBRead(xid, pTcon, netfid,
86c96b4b 384 24 /* length */, 0 /* offset */,
ec637e3f 385 &bytes_read, &pbuf, &buf_type);
4523cc30
SF
386 if ((rc == 0) && (bytes_read >= 8)) {
387 if (memcmp("IntxBLK", pbuf, 8) == 0) {
b6b38f70 388 cFYI(1, "Block device");
0b8f18e3
JL
389 fattr->cf_mode |= S_IFBLK;
390 fattr->cf_dtype = DT_BLK;
4523cc30 391 if (bytes_read == 24) {
86c96b4b
SF
392 /* 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));
0b8f18e3 397 fattr->cf_rdev = MKDEV(mjr, mnr);
86c96b4b 398 }
4523cc30 399 } else if (memcmp("IntxCHR", pbuf, 8) == 0) {
b6b38f70 400 cFYI(1, "Char device");
0b8f18e3
JL
401 fattr->cf_mode |= S_IFCHR;
402 fattr->cf_dtype = DT_CHR;
4523cc30 403 if (bytes_read == 24) {
86c96b4b
SF
404 /* we have enough to decode dev num */
405 __u64 mjr; /* major */
406 __u64 mnr; /* minor */
407 mjr = le64_to_cpu(*(__le64 *)(pbuf+8));
408 mnr = le64_to_cpu(*(__le64 *)(pbuf+16));
0b8f18e3 409 fattr->cf_rdev = MKDEV(mjr, mnr);
fb8c4b14 410 }
4523cc30 411 } else if (memcmp("IntxLNK", pbuf, 7) == 0) {
b6b38f70 412 cFYI(1, "Symlink");
0b8f18e3
JL
413 fattr->cf_mode |= S_IFLNK;
414 fattr->cf_dtype = DT_LNK;
86c96b4b 415 } else {
0b8f18e3
JL
416 fattr->cf_mode |= S_IFREG; /* file? */
417 fattr->cf_dtype = DT_REG;
fb8c4b14 418 rc = -EOPNOTSUPP;
86c96b4b 419 }
3020a1f5 420 } else {
0b8f18e3
JL
421 fattr->cf_mode |= S_IFREG; /* then it is a file */
422 fattr->cf_dtype = DT_REG;
fb8c4b14
SF
423 rc = -EOPNOTSUPP; /* or some unknown SFU type */
424 }
d6e2f2a4 425 CIFSSMBClose(xid, pTcon, netfid);
d6e2f2a4
SF
426 }
427 return rc;
d6e2f2a4
SF
428}
429
9e294f1c
SF
430#define SFBITS_MASK (S_ISVTX | S_ISGID | S_ISUID) /* SETFILEBITS valid bits */
431
0b8f18e3
JL
432/*
433 * Fetch mode bits as provided by SFU.
434 *
435 * FIXME: Doesn't this clobber the type bit we got from cifs_sfu_type ?
436 */
437static int cifs_sfu_mode(struct cifs_fattr *fattr, const unsigned char *path,
438 struct cifs_sb_info *cifs_sb, int xid)
9e294f1c 439{
3020a1f5 440#ifdef CONFIG_CIFS_XATTR
9e294f1c
SF
441 ssize_t rc;
442 char ea_value[4];
443 __u32 mode;
444
31c0519f 445 rc = CIFSSMBQAllEAs(xid, cifs_sb->tcon, path, "SETFILEBITS",
0b8f18e3
JL
446 ea_value, 4 /* size of buf */, cifs_sb->local_nls,
447 cifs_sb->mnt_cifs_flags &
448 CIFS_MOUNT_MAP_SPECIAL_CHR);
4523cc30 449 if (rc < 0)
9e294f1c
SF
450 return (int)rc;
451 else if (rc > 3) {
452 mode = le32_to_cpu(*((__le32 *)ea_value));
0b8f18e3 453 fattr->cf_mode &= ~SFBITS_MASK;
b6b38f70
JP
454 cFYI(1, "special bits 0%o org mode 0%o", mode,
455 fattr->cf_mode);
0b8f18e3 456 fattr->cf_mode = (mode & SFBITS_MASK) | fattr->cf_mode;
b6b38f70 457 cFYI(1, "special mode bits 0%o", mode);
9e294f1c 458 }
0b8f18e3
JL
459
460 return 0;
3020a1f5
SF
461#else
462 return -EOPNOTSUPP;
463#endif
9e294f1c
SF
464}
465
0b8f18e3 466/* Fill a cifs_fattr struct with info from FILE_ALL_INFO */
f1230c97 467static void
0b8f18e3
JL
468cifs_all_info_to_fattr(struct cifs_fattr *fattr, FILE_ALL_INFO *info,
469 struct cifs_sb_info *cifs_sb, bool adjust_tz)
b9a3260f 470{
0b8f18e3
JL
471 memset(fattr, 0, sizeof(*fattr));
472 fattr->cf_cifsattrs = le32_to_cpu(info->Attributes);
473 if (info->DeletePending)
474 fattr->cf_flags |= CIFS_FATTR_DELETE_PENDING;
475
476 if (info->LastAccessTime)
477 fattr->cf_atime = cifs_NTtimeToUnix(info->LastAccessTime);
478 else
479 fattr->cf_atime = CURRENT_TIME;
480
481 fattr->cf_ctime = cifs_NTtimeToUnix(info->ChangeTime);
482 fattr->cf_mtime = cifs_NTtimeToUnix(info->LastWriteTime);
483
484 if (adjust_tz) {
485 fattr->cf_ctime.tv_sec += cifs_sb->tcon->ses->server->timeAdj;
486 fattr->cf_mtime.tv_sec += cifs_sb->tcon->ses->server->timeAdj;
487 }
488
489 fattr->cf_eof = le64_to_cpu(info->EndOfFile);
490 fattr->cf_bytes = le64_to_cpu(info->AllocationSize);
491
492 if (fattr->cf_cifsattrs & ATTR_DIRECTORY) {
493 fattr->cf_mode = S_IFDIR | cifs_sb->mnt_dir_mode;
494 fattr->cf_dtype = DT_DIR;
495 } else {
496 fattr->cf_mode = S_IFREG | cifs_sb->mnt_file_mode;
497 fattr->cf_dtype = DT_REG;
0b8f18e3 498
d0c280d2
JL
499 /* clear write bits if ATTR_READONLY is set */
500 if (fattr->cf_cifsattrs & ATTR_READONLY)
501 fattr->cf_mode &= ~(S_IWUGO);
502 }
0b8f18e3
JL
503
504 fattr->cf_nlink = le32_to_cpu(info->NumberOfLinks);
505
506 fattr->cf_uid = cifs_sb->mnt_uid;
507 fattr->cf_gid = cifs_sb->mnt_gid;
b9a3260f
SF
508}
509
abab095d
JL
510int cifs_get_file_info(struct file *filp)
511{
512 int rc;
513 int xid;
514 FILE_ALL_INFO find_data;
515 struct cifs_fattr fattr;
516 struct inode *inode = filp->f_path.dentry->d_inode;
517 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
518 struct cifsTconInfo *tcon = cifs_sb->tcon;
c21dfb69 519 struct cifsFileInfo *cfile = filp->private_data;
abab095d
JL
520
521 xid = GetXid();
522 rc = CIFSSMBQFileInfo(xid, tcon, cfile->netfid, &find_data);
523 if (rc == -EOPNOTSUPP || rc == -EINVAL) {
524 /*
525 * FIXME: legacy server -- fall back to path-based call?
ff215713
SF
526 * for now, just skip revalidating and mark inode for
527 * immediate reval.
528 */
abab095d
JL
529 rc = 0;
530 CIFS_I(inode)->time = 0;
531 goto cgfi_exit;
532 } else if (rc == -EREMOTE) {
533 cifs_create_dfs_fattr(&fattr, inode->i_sb);
534 rc = 0;
535 } else if (rc)
536 goto cgfi_exit;
537
538 /*
539 * don't bother with SFU junk here -- just mark inode as needing
540 * revalidation.
541 */
542 cifs_all_info_to_fattr(&fattr, &find_data, cifs_sb, false);
543 fattr.cf_uniqueid = CIFS_I(inode)->uniqueid;
544 fattr.cf_flags |= CIFS_FATTR_NEED_REVAL;
545 cifs_fattr_to_inode(inode, &fattr);
546cgfi_exit:
547 FreeXid(xid);
548 return rc;
549}
550
1da177e4 551int cifs_get_inode_info(struct inode **pinode,
646dd539 552 const unsigned char *full_path, FILE_ALL_INFO *pfindData,
8b1327f6 553 struct super_block *sb, int xid, const __u16 *pfid)
1da177e4 554{
0b8f18e3 555 int rc = 0, tmprc;
1da177e4 556 struct cifsTconInfo *pTcon;
1da177e4 557 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
1da177e4 558 char *buf = NULL;
5ade9dea 559 bool adjustTZ = false;
0b8f18e3 560 struct cifs_fattr fattr;
1da177e4
LT
561
562 pTcon = cifs_sb->tcon;
b6b38f70 563 cFYI(1, "Getting info on %s", full_path);
1da177e4 564
d0d2f2df
SF
565 if ((pfindData == NULL) && (*pinode != NULL)) {
566 if (CIFS_I(*pinode)->clientCanCacheRead) {
b6b38f70 567 cFYI(1, "No need to revalidate cached inode sizes");
1da177e4
LT
568 return rc;
569 }
570 }
571
572 /* if file info not passed in then get it from server */
d0d2f2df 573 if (pfindData == NULL) {
1da177e4 574 buf = kmalloc(sizeof(FILE_ALL_INFO), GFP_KERNEL);
d0d2f2df 575 if (buf == NULL)
1da177e4
LT
576 return -ENOMEM;
577 pfindData = (FILE_ALL_INFO *)buf;
7962670e 578
1da177e4 579 /* could do find first instead but this returns more info */
7962670e 580 rc = CIFSSMBQPathInfo(xid, pTcon, full_path, pfindData,
acf1a1b1 581 0 /* not legacy */,
6b8edfe0 582 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
737b758c 583 CIFS_MOUNT_MAP_SPECIAL_CHR);
6b8edfe0
SF
584 /* BB optimize code so we do not make the above call
585 when server claims no NT SMB support and the above call
586 failed at least once - set flag in tcon or mount */
4523cc30 587 if ((rc == -EOPNOTSUPP) || (rc == -EINVAL)) {
7962670e 588 rc = SMBQueryInformation(xid, pTcon, full_path,
fb8c4b14 589 pfindData, cifs_sb->local_nls,
6b8edfe0
SF
590 cifs_sb->mnt_cifs_flags &
591 CIFS_MOUNT_MAP_SPECIAL_CHR);
4b18f2a9 592 adjustTZ = true;
6b8edfe0 593 }
1da177e4 594 }
0b8f18e3
JL
595
596 if (!rc) {
597 cifs_all_info_to_fattr(&fattr, (FILE_ALL_INFO *) pfindData,
598 cifs_sb, adjustTZ);
599 } else if (rc == -EREMOTE) {
600 cifs_create_dfs_fattr(&fattr, sb);
b9a3260f 601 rc = 0;
0b8f18e3 602 } else {
7962670e 603 goto cgii_exit;
0b8f18e3 604 }
1da177e4 605
0b8f18e3
JL
606 /*
607 * If an inode wasn't passed in, then get the inode number
608 *
609 * Is an i_ino of zero legal? Can we use that to check if the server
610 * supports returning inode numbers? Are there other sanity checks we
611 * can use to ensure that the server is really filling in that field?
612 *
613 * We can not use the IndexNumber field by default from Windows or
614 * Samba (in ALL_INFO buf) but we can request it explicitly. The SNIA
615 * CIFS spec claims that this value is unique within the scope of a
616 * share, and the windows docs hint that it's actually unique
617 * per-machine.
618 *
619 * There may be higher info levels that work but are there Windows
620 * server or network appliances for which IndexNumber field is not
621 * guaranteed unique?
622 */
b9a3260f 623 if (*pinode == NULL) {
b9a3260f
SF
624 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM) {
625 int rc1 = 0;
b9a3260f
SF
626
627 rc1 = CIFSGetSrvInodeNumber(xid, pTcon,
0b8f18e3 628 full_path, &fattr.cf_uniqueid,
737b758c
SF
629 cifs_sb->local_nls,
630 cifs_sb->mnt_cifs_flags &
631 CIFS_MOUNT_MAP_SPECIAL_CHR);
ec06aedd 632 if (rc1 || !fattr.cf_uniqueid) {
b6b38f70 633 cFYI(1, "GetSrvInodeNum rc %d", rc1);
0b8f18e3 634 fattr.cf_uniqueid = iunique(sb, ROOT_I);
ec06aedd 635 cifs_autodisable_serverino(cifs_sb);
132ac7b7 636 }
132ac7b7 637 } else {
0b8f18e3 638 fattr.cf_uniqueid = iunique(sb, ROOT_I);
132ac7b7 639 }
b9a3260f 640 } else {
0b8f18e3 641 fattr.cf_uniqueid = CIFS_I(*pinode)->uniqueid;
b9a3260f
SF
642 }
643
0b8f18e3
JL
644 /* query for SFU type info if supported and needed */
645 if (fattr.cf_cifsattrs & ATTR_SYSTEM &&
646 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_UNX_EMUL) {
647 tmprc = cifs_sfu_type(&fattr, full_path, cifs_sb, xid);
648 if (tmprc)
b6b38f70 649 cFYI(1, "cifs_sfu_type failed: %d", tmprc);
b9a3260f 650 }
1da177e4 651
4879b448 652#ifdef CONFIG_CIFS_EXPERIMENTAL
b9a3260f
SF
653 /* fill in 0777 bits from ACL */
654 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL) {
b6b38f70 655 cFYI(1, "Getting mode bits from ACL");
0b8f18e3 656 cifs_acl_to_fattr(cifs_sb, &fattr, *pinode, full_path, pfid);
b9a3260f 657 }
4879b448 658#endif
b9a3260f 659
0b8f18e3
JL
660 /* fill in remaining high mode bits e.g. SUID, VTX */
661 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_UNX_EMUL)
662 cifs_sfu_mode(&fattr, full_path, cifs_sb, xid);
b9a3260f 663
0b8f18e3
JL
664 if (!*pinode) {
665 *pinode = cifs_iget(sb, &fattr);
666 if (!*pinode)
667 rc = -ENOMEM;
668 } else {
669 cifs_fattr_to_inode(*pinode, &fattr);
670 }
b9a3260f 671
7962670e 672cgii_exit:
1da177e4
LT
673 kfree(buf);
674 return rc;
675}
676
7f8ed420
SF
677static const struct inode_operations cifs_ipc_inode_ops = {
678 .lookup = cifs_lookup,
679};
680
e4cce94c 681char *cifs_build_path_to_root(struct cifs_sb_info *cifs_sb)
8be0ed44
SF
682{
683 int pplen = cifs_sb->prepathlen;
684 int dfsplen;
685 char *full_path = NULL;
686
687 /* if no prefix path, simply set path to the root of share to "" */
688 if (pplen == 0) {
689 full_path = kmalloc(1, GFP_KERNEL);
690 if (full_path)
691 full_path[0] = 0;
692 return full_path;
693 }
694
695 if (cifs_sb->tcon && (cifs_sb->tcon->Flags & SMB_SHARE_IS_IN_DFS))
696 dfsplen = strnlen(cifs_sb->tcon->treeName, MAX_TREE_SIZE + 1);
697 else
698 dfsplen = 0;
699
700 full_path = kmalloc(dfsplen + pplen + 1, GFP_KERNEL);
701 if (full_path == NULL)
702 return full_path;
703
704 if (dfsplen) {
705 strncpy(full_path, cifs_sb->tcon->treeName, dfsplen);
706 /* switch slash direction in prepath depending on whether
707 * windows or posix style path names
708 */
709 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_POSIX_PATHS) {
710 int i;
711 for (i = 0; i < dfsplen; i++) {
712 if (full_path[i] == '\\')
713 full_path[i] = '/';
714 }
715 }
716 }
717 strncpy(full_path + dfsplen, cifs_sb->prepath, pplen);
718 full_path[dfsplen + pplen] = 0; /* add trailing null */
719 return full_path;
720}
721
cc0bad75
JL
722static int
723cifs_find_inode(struct inode *inode, void *opaque)
724{
725 struct cifs_fattr *fattr = (struct cifs_fattr *) opaque;
726
f30b9c11 727 /* don't match inode with different uniqueid */
cc0bad75
JL
728 if (CIFS_I(inode)->uniqueid != fattr->cf_uniqueid)
729 return 0;
730
f30b9c11
JL
731 /* don't match inode of different type */
732 if ((inode->i_mode & S_IFMT) != (fattr->cf_mode & S_IFMT))
733 return 0;
734
5acfec25
JL
735 /* if it's not a directory or has no dentries, then flag it */
736 if (S_ISDIR(inode->i_mode) && !list_empty(&inode->i_dentry))
3d694380 737 fattr->cf_flags |= CIFS_FATTR_INO_COLLISION;
3d694380 738
cc0bad75
JL
739 return 1;
740}
741
742static int
743cifs_init_inode(struct inode *inode, void *opaque)
744{
745 struct cifs_fattr *fattr = (struct cifs_fattr *) opaque;
746
747 CIFS_I(inode)->uniqueid = fattr->cf_uniqueid;
748 return 0;
749}
750
5acfec25
JL
751/*
752 * walk dentry list for an inode and report whether it has aliases that
753 * are hashed. We use this to determine if a directory inode can actually
754 * be used.
755 */
756static bool
757inode_has_hashed_dentries(struct inode *inode)
758{
759 struct dentry *dentry;
760
761 spin_lock(&dcache_lock);
762 list_for_each_entry(dentry, &inode->i_dentry, d_alias) {
763 if (!d_unhashed(dentry) || IS_ROOT(dentry)) {
764 spin_unlock(&dcache_lock);
765 return true;
766 }
767 }
768 spin_unlock(&dcache_lock);
769 return false;
770}
771
cc0bad75
JL
772/* Given fattrs, get a corresponding inode */
773struct inode *
774cifs_iget(struct super_block *sb, struct cifs_fattr *fattr)
775{
776 unsigned long hash;
777 struct inode *inode;
778
3d694380 779retry_iget5_locked:
b6b38f70 780 cFYI(1, "looking for uniqueid=%llu", fattr->cf_uniqueid);
cc0bad75
JL
781
782 /* hash down to 32-bits on 32-bit arch */
783 hash = cifs_uniqueid_to_ino_t(fattr->cf_uniqueid);
784
785 inode = iget5_locked(sb, hash, cifs_find_inode, cifs_init_inode, fattr);
cc0bad75 786 if (inode) {
5acfec25 787 /* was there a potentially problematic inode collision? */
3d694380 788 if (fattr->cf_flags & CIFS_FATTR_INO_COLLISION) {
3d694380 789 fattr->cf_flags &= ~CIFS_FATTR_INO_COLLISION;
5acfec25
JL
790
791 if (inode_has_hashed_dentries(inode)) {
792 cifs_autodisable_serverino(CIFS_SB(sb));
793 iput(inode);
794 fattr->cf_uniqueid = iunique(sb, ROOT_I);
795 goto retry_iget5_locked;
796 }
3d694380
JL
797 }
798
cc0bad75
JL
799 cifs_fattr_to_inode(inode, fattr);
800 if (sb->s_flags & MS_NOATIME)
801 inode->i_flags |= S_NOATIME | S_NOCMTIME;
802 if (inode->i_state & I_NEW) {
803 inode->i_ino = hash;
0ccd4802 804#ifdef CONFIG_CIFS_FSCACHE
9451a9a5
SJ
805 /* initialize per-inode cache cookie pointer */
806 CIFS_I(inode)->fscache = NULL;
0ccd4802 807#endif
cc0bad75
JL
808 unlock_new_inode(inode);
809 }
810 }
811
812 return inode;
813}
814
1da177e4 815/* gets root inode */
bd433d4c 816struct inode *cifs_root_iget(struct super_block *sb, unsigned long ino)
1da177e4 817{
ce634ab2 818 int xid;
1da177e4 819 struct cifs_sb_info *cifs_sb;
cc0bad75 820 struct inode *inode = NULL;
ce634ab2 821 long rc;
8be0ed44 822 char *full_path;
ce634ab2 823
cc0bad75 824 cifs_sb = CIFS_SB(sb);
e4cce94c 825 full_path = cifs_build_path_to_root(cifs_sb);
8be0ed44
SF
826 if (full_path == NULL)
827 return ERR_PTR(-ENOMEM);
c18c842b 828
8be0ed44 829 xid = GetXid();
0b8f18e3 830 if (cifs_sb->tcon->unix_ext)
cc0bad75 831 rc = cifs_get_inode_info_unix(&inode, full_path, sb, xid);
0b8f18e3
JL
832 else
833 rc = cifs_get_inode_info(&inode, full_path, NULL, sb,
8be0ed44 834 xid, NULL);
0b8f18e3
JL
835
836 if (!inode)
f0138a79 837 return ERR_PTR(rc);
cc0bad75 838
0ccd4802 839#ifdef CONFIG_CIFS_FSCACHE
d03382ce
SJ
840 /* populate tcon->resource_id */
841 cifs_sb->tcon->resource_id = CIFS_I(inode)->uniqueid;
0ccd4802 842#endif
d03382ce 843
7f8ed420 844 if (rc && cifs_sb->tcon->ipc) {
b6b38f70 845 cFYI(1, "ipc connection - fake read inode");
7f8ed420
SF
846 inode->i_mode |= S_IFDIR;
847 inode->i_nlink = 2;
848 inode->i_op = &cifs_ipc_inode_ops;
849 inode->i_fop = &simple_dir_operations;
850 inode->i_uid = cifs_sb->mnt_uid;
851 inode->i_gid = cifs_sb->mnt_gid;
ad661334 852 } else if (rc) {
8be0ed44 853 kfree(full_path);
ce634ab2
DH
854 _FreeXid(xid);
855 iget_failed(inode);
856 return ERR_PTR(rc);
7f8ed420
SF
857 }
858
ce634ab2 859
8be0ed44 860 kfree(full_path);
ce634ab2
DH
861 /* can not call macro FreeXid here since in a void func
862 * TODO: This is no longer true
863 */
1da177e4 864 _FreeXid(xid);
ce634ab2 865 return inode;
1da177e4
LT
866}
867
388e57b2
SF
868static int
869cifs_set_file_info(struct inode *inode, struct iattr *attrs, int xid,
870 char *full_path, __u32 dosattr)
871{
872 int rc;
873 int oplock = 0;
874 __u16 netfid;
875 __u32 netpid;
876 bool set_time = false;
877 struct cifsFileInfo *open_file;
878 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
879 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
880 struct cifsTconInfo *pTcon = cifs_sb->tcon;
881 FILE_BASIC_INFO info_buf;
882
1adcb710
SF
883 if (attrs == NULL)
884 return -EINVAL;
885
388e57b2
SF
886 if (attrs->ia_valid & ATTR_ATIME) {
887 set_time = true;
888 info_buf.LastAccessTime =
889 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_atime));
890 } else
891 info_buf.LastAccessTime = 0;
892
893 if (attrs->ia_valid & ATTR_MTIME) {
894 set_time = true;
895 info_buf.LastWriteTime =
896 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_mtime));
897 } else
898 info_buf.LastWriteTime = 0;
899
900 /*
901 * Samba throws this field away, but windows may actually use it.
902 * Do not set ctime unless other time stamps are changed explicitly
903 * (i.e. by utimes()) since we would then have a mix of client and
904 * server times.
905 */
906 if (set_time && (attrs->ia_valid & ATTR_CTIME)) {
b6b38f70 907 cFYI(1, "CIFS - CTIME changed");
388e57b2
SF
908 info_buf.ChangeTime =
909 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_ctime));
910 } else
911 info_buf.ChangeTime = 0;
912
913 info_buf.CreationTime = 0; /* don't change */
914 info_buf.Attributes = cpu_to_le32(dosattr);
915
916 /*
917 * If the file is already open for write, just use that fileid
918 */
919 open_file = find_writable_file(cifsInode);
920 if (open_file) {
921 netfid = open_file->netfid;
922 netpid = open_file->pid;
923 goto set_via_filehandle;
924 }
925
926 /*
927 * NT4 apparently returns success on this call, but it doesn't
928 * really work.
929 */
930 if (!(pTcon->ses->flags & CIFS_SES_NT4)) {
931 rc = CIFSSMBSetPathInfo(xid, pTcon, full_path,
932 &info_buf, cifs_sb->local_nls,
933 cifs_sb->mnt_cifs_flags &
934 CIFS_MOUNT_MAP_SPECIAL_CHR);
6b37faa1
JL
935 if (rc == 0) {
936 cifsInode->cifsAttrs = dosattr;
937 goto out;
938 } else if (rc != -EOPNOTSUPP && rc != -EINVAL)
388e57b2
SF
939 goto out;
940 }
941
b6b38f70
JP
942 cFYI(1, "calling SetFileInfo since SetPathInfo for "
943 "times not supported by this server");
388e57b2
SF
944 rc = CIFSSMBOpen(xid, pTcon, full_path, FILE_OPEN,
945 SYNCHRONIZE | FILE_WRITE_ATTRIBUTES,
946 CREATE_NOT_DIR, &netfid, &oplock,
947 NULL, cifs_sb->local_nls,
948 cifs_sb->mnt_cifs_flags &
949 CIFS_MOUNT_MAP_SPECIAL_CHR);
950
951 if (rc != 0) {
952 if (rc == -EIO)
953 rc = -EINVAL;
954 goto out;
955 }
956
957 netpid = current->tgid;
958
959set_via_filehandle:
960 rc = CIFSSMBSetFileInfo(xid, pTcon, &info_buf, netfid, netpid);
d388908e
SF
961 if (!rc)
962 cifsInode->cifsAttrs = dosattr;
963
388e57b2
SF
964 if (open_file == NULL)
965 CIFSSMBClose(xid, pTcon, netfid);
966 else
6ab409b5 967 cifsFileInfo_put(open_file);
388e57b2
SF
968out:
969 return rc;
970}
971
a12a1ac7
JL
972/*
973 * open the given file (if it isn't already), set the DELETE_ON_CLOSE bit
974 * and rename it to a random name that hopefully won't conflict with
975 * anything else.
976 */
977static int
3270958b 978cifs_rename_pending_delete(char *full_path, struct dentry *dentry, int xid)
a12a1ac7
JL
979{
980 int oplock = 0;
981 int rc;
982 __u16 netfid;
3270958b 983 struct inode *inode = dentry->d_inode;
a12a1ac7
JL
984 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
985 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
986 struct cifsTconInfo *tcon = cifs_sb->tcon;
3270958b
SF
987 __u32 dosattr, origattr;
988 FILE_BASIC_INFO *info_buf = NULL;
a12a1ac7
JL
989
990 rc = CIFSSMBOpen(xid, tcon, full_path, FILE_OPEN,
dd1db2de 991 DELETE|FILE_WRITE_ATTRIBUTES, CREATE_NOT_DIR,
a12a1ac7
JL
992 &netfid, &oplock, NULL, cifs_sb->local_nls,
993 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
994 if (rc != 0)
995 goto out;
996
3270958b
SF
997 origattr = cifsInode->cifsAttrs;
998 if (origattr == 0)
999 origattr |= ATTR_NORMAL;
1000
1001 dosattr = origattr & ~ATTR_READONLY;
a12a1ac7
JL
1002 if (dosattr == 0)
1003 dosattr |= ATTR_NORMAL;
1004 dosattr |= ATTR_HIDDEN;
1005
3270958b
SF
1006 /* set ATTR_HIDDEN and clear ATTR_READONLY, but only if needed */
1007 if (dosattr != origattr) {
1008 info_buf = kzalloc(sizeof(*info_buf), GFP_KERNEL);
1009 if (info_buf == NULL) {
1010 rc = -ENOMEM;
1011 goto out_close;
1012 }
1013 info_buf->Attributes = cpu_to_le32(dosattr);
1014 rc = CIFSSMBSetFileInfo(xid, tcon, info_buf, netfid,
1015 current->tgid);
1016 /* although we would like to mark the file hidden
1017 if that fails we will still try to rename it */
41346098 1018 if (rc != 0)
3270958b
SF
1019 cifsInode->cifsAttrs = dosattr;
1020 else
1021 dosattr = origattr; /* since not able to change them */
a12a1ac7 1022 }
a12a1ac7 1023
dd1db2de
JL
1024 /* rename the file */
1025 rc = CIFSSMBRenameOpenFile(xid, tcon, netfid, NULL, cifs_sb->local_nls,
a12a1ac7
JL
1026 cifs_sb->mnt_cifs_flags &
1027 CIFS_MOUNT_MAP_SPECIAL_CHR);
3270958b
SF
1028 if (rc != 0) {
1029 rc = -ETXTBSY;
1030 goto undo_setattr;
1031 }
6d22f098 1032
3270958b
SF
1033 /* try to set DELETE_ON_CLOSE */
1034 if (!cifsInode->delete_pending) {
1035 rc = CIFSSMBSetFileDisposition(xid, tcon, true, netfid,
1036 current->tgid);
1037 /*
1038 * some samba versions return -ENOENT when we try to set the
1039 * file disposition here. Likely a samba bug, but work around
1040 * it for now. This means that some cifsXXX files may hang
1041 * around after they shouldn't.
1042 *
1043 * BB: remove this hack after more servers have the fix
1044 */
1045 if (rc == -ENOENT)
1046 rc = 0;
1047 else if (rc != 0) {
1048 rc = -ETXTBSY;
1049 goto undo_rename;
1050 }
1051 cifsInode->delete_pending = true;
1052 }
7ce86d5a 1053
a12a1ac7
JL
1054out_close:
1055 CIFSSMBClose(xid, tcon, netfid);
1056out:
3270958b 1057 kfree(info_buf);
a12a1ac7 1058 return rc;
3270958b
SF
1059
1060 /*
1061 * reset everything back to the original state. Don't bother
1062 * dealing with errors here since we can't do anything about
1063 * them anyway.
1064 */
1065undo_rename:
1066 CIFSSMBRenameOpenFile(xid, tcon, netfid, dentry->d_name.name,
1067 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1068 CIFS_MOUNT_MAP_SPECIAL_CHR);
1069undo_setattr:
1070 if (dosattr != origattr) {
1071 info_buf->Attributes = cpu_to_le32(origattr);
1072 if (!CIFSSMBSetFileInfo(xid, tcon, info_buf, netfid,
1073 current->tgid))
1074 cifsInode->cifsAttrs = origattr;
1075 }
1076
1077 goto out_close;
a12a1ac7
JL
1078}
1079
ff694527
SF
1080
1081/*
1082 * If dentry->d_inode is null (usually meaning the cached dentry
1083 * is a negative dentry) then we would attempt a standard SMB delete, but
af901ca1
AGR
1084 * if that fails we can not attempt the fall back mechanisms on EACCESS
1085 * but will return the EACCESS to the caller. Note that the VFS does not call
ff694527
SF
1086 * unlink on negative dentries currently.
1087 */
5f0319a7 1088int cifs_unlink(struct inode *dir, struct dentry *dentry)
1da177e4
LT
1089{
1090 int rc = 0;
1091 int xid;
1da177e4 1092 char *full_path = NULL;
5f0319a7 1093 struct inode *inode = dentry->d_inode;
ff694527 1094 struct cifsInodeInfo *cifs_inode;
5f0319a7
JL
1095 struct super_block *sb = dir->i_sb;
1096 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
1097 struct cifsTconInfo *tcon = cifs_sb->tcon;
6050247d
SF
1098 struct iattr *attrs = NULL;
1099 __u32 dosattr = 0, origattr = 0;
1da177e4 1100
b6b38f70 1101 cFYI(1, "cifs_unlink, dir=0x%p, dentry=0x%p", dir, dentry);
1da177e4
LT
1102
1103 xid = GetXid();
1104
5f0319a7
JL
1105 /* Unlink can be called from rename so we can not take the
1106 * sb->s_vfs_rename_mutex here */
1107 full_path = build_path_from_dentry(dentry);
1da177e4 1108 if (full_path == NULL) {
0f3bc09e 1109 rc = -ENOMEM;
1da177e4 1110 FreeXid(xid);
0f3bc09e 1111 return rc;
1da177e4 1112 }
2d785a50 1113
5f0319a7 1114 if ((tcon->ses->capabilities & CAP_UNIX) &&
2d785a50 1115 (CIFS_UNIX_POSIX_PATH_OPS_CAP &
5f0319a7
JL
1116 le64_to_cpu(tcon->fsUnixInfo.Capability))) {
1117 rc = CIFSPOSIXDelFile(xid, tcon, full_path,
2d785a50 1118 SMB_POSIX_UNLINK_FILE_TARGET, cifs_sb->local_nls,
737b758c 1119 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
b6b38f70 1120 cFYI(1, "posix del rc %d", rc);
2d785a50
SF
1121 if ((rc == 0) || (rc == -ENOENT))
1122 goto psx_del_no_retry;
1123 }
1da177e4 1124
6050247d 1125retry_std_delete:
5f0319a7 1126 rc = CIFSSMBDelFile(xid, tcon, full_path, cifs_sb->local_nls,
2d785a50 1127 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
6050247d 1128
2d785a50 1129psx_del_no_retry:
1da177e4 1130 if (!rc) {
5f0319a7
JL
1131 if (inode)
1132 drop_nlink(inode);
1da177e4 1133 } else if (rc == -ENOENT) {
5f0319a7 1134 d_drop(dentry);
1da177e4 1135 } else if (rc == -ETXTBSY) {
3270958b 1136 rc = cifs_rename_pending_delete(full_path, dentry, xid);
a12a1ac7
JL
1137 if (rc == 0)
1138 drop_nlink(inode);
ff694527 1139 } else if ((rc == -EACCES) && (dosattr == 0) && inode) {
388e57b2
SF
1140 attrs = kzalloc(sizeof(*attrs), GFP_KERNEL);
1141 if (attrs == NULL) {
1142 rc = -ENOMEM;
1143 goto out_reval;
1da177e4 1144 }
388e57b2
SF
1145
1146 /* try to reset dos attributes */
ff694527
SF
1147 cifs_inode = CIFS_I(inode);
1148 origattr = cifs_inode->cifsAttrs;
6050247d
SF
1149 if (origattr == 0)
1150 origattr |= ATTR_NORMAL;
1151 dosattr = origattr & ~ATTR_READONLY;
388e57b2
SF
1152 if (dosattr == 0)
1153 dosattr |= ATTR_NORMAL;
1154 dosattr |= ATTR_HIDDEN;
1155
1156 rc = cifs_set_file_info(inode, attrs, xid, full_path, dosattr);
388e57b2
SF
1157 if (rc != 0)
1158 goto out_reval;
6050247d
SF
1159
1160 goto retry_std_delete;
1da177e4 1161 }
6050247d
SF
1162
1163 /* undo the setattr if we errored out and it's needed */
1164 if (rc != 0 && dosattr != 0)
1165 cifs_set_file_info(inode, attrs, xid, full_path, origattr);
1166
388e57b2 1167out_reval:
4523cc30 1168 if (inode) {
ff694527
SF
1169 cifs_inode = CIFS_I(inode);
1170 cifs_inode->time = 0; /* will force revalidate to get info
5f0319a7
JL
1171 when needed */
1172 inode->i_ctime = current_fs_time(sb);
06bcfedd 1173 }
5f0319a7 1174 dir->i_ctime = dir->i_mtime = current_fs_time(sb);
ff694527 1175 cifs_inode = CIFS_I(dir);
6050247d 1176 CIFS_I(dir)->time = 0; /* force revalidate of dir as well */
1da177e4
LT
1177
1178 kfree(full_path);
6050247d 1179 kfree(attrs);
1da177e4
LT
1180 FreeXid(xid);
1181 return rc;
1182}
1183
1184int cifs_mkdir(struct inode *inode, struct dentry *direntry, int mode)
1185{
6b37faa1 1186 int rc = 0, tmprc;
1da177e4
LT
1187 int xid;
1188 struct cifs_sb_info *cifs_sb;
1189 struct cifsTconInfo *pTcon;
1190 char *full_path = NULL;
1191 struct inode *newinode = NULL;
cc0bad75 1192 struct cifs_fattr fattr;
1da177e4 1193
b6b38f70 1194 cFYI(1, "In cifs_mkdir, mode = 0x%x inode = 0x%p", mode, inode);
1da177e4
LT
1195
1196 xid = GetXid();
1197
1198 cifs_sb = CIFS_SB(inode->i_sb);
1199 pTcon = cifs_sb->tcon;
1200
7f57356b 1201 full_path = build_path_from_dentry(direntry);
1da177e4 1202 if (full_path == NULL) {
0f3bc09e 1203 rc = -ENOMEM;
1da177e4 1204 FreeXid(xid);
0f3bc09e 1205 return rc;
1da177e4 1206 }
50c2f753 1207
fb8c4b14
SF
1208 if ((pTcon->ses->capabilities & CAP_UNIX) &&
1209 (CIFS_UNIX_POSIX_PATH_OPS_CAP &
2dd29d31
SF
1210 le64_to_cpu(pTcon->fsUnixInfo.Capability))) {
1211 u32 oplock = 0;
f6d09982 1212 FILE_UNIX_BASIC_INFO *pInfo =
2dd29d31 1213 kzalloc(sizeof(FILE_UNIX_BASIC_INFO), GFP_KERNEL);
fb8c4b14 1214 if (pInfo == NULL) {
2dd29d31
SF
1215 rc = -ENOMEM;
1216 goto mkdir_out;
1217 }
50c2f753 1218
ce3b0f8d 1219 mode &= ~current_umask();
2dd29d31
SF
1220 rc = CIFSPOSIXCreate(xid, pTcon, SMB_O_DIRECTORY | SMB_O_CREAT,
1221 mode, NULL /* netfid */, pInfo, &oplock,
fb8c4b14
SF
1222 full_path, cifs_sb->local_nls,
1223 cifs_sb->mnt_cifs_flags &
2dd29d31 1224 CIFS_MOUNT_MAP_SPECIAL_CHR);
c45d707f
SF
1225 if (rc == -EOPNOTSUPP) {
1226 kfree(pInfo);
1227 goto mkdir_retry_old;
1228 } else if (rc) {
b6b38f70 1229 cFYI(1, "posix mkdir returned 0x%x", rc);
2dd29d31
SF
1230 d_drop(direntry);
1231 } else {
8f2376ad
CG
1232 if (pInfo->Type == cpu_to_le32(-1)) {
1233 /* no return info, go query for it */
5a07cdf8 1234 kfree(pInfo);
fb8c4b14 1235 goto mkdir_get_info;
5a07cdf8 1236 }
fb8c4b14
SF
1237/*BB check (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID ) to see if need
1238 to set uid/gid */
2dd29d31
SF
1239 inc_nlink(inode);
1240 if (pTcon->nocase)
1241 direntry->d_op = &cifs_ci_dentry_ops;
1242 else
1243 direntry->d_op = &cifs_dentry_ops;
cbac3cba 1244
cc0bad75 1245 cifs_unix_basic_to_fattr(&fattr, pInfo, cifs_sb);
4065c802 1246 cifs_fill_uniqueid(inode->i_sb, &fattr);
cc0bad75
JL
1247 newinode = cifs_iget(inode->i_sb, &fattr);
1248 if (!newinode) {
5a07cdf8 1249 kfree(pInfo);
cbac3cba 1250 goto mkdir_get_info;
5a07cdf8 1251 }
6b37faa1 1252
2dd29d31 1253 d_instantiate(direntry, newinode);
cbac3cba 1254
cbac3cba 1255#ifdef CONFIG_CIFS_DEBUG2
b6b38f70
JP
1256 cFYI(1, "instantiated dentry %p %s to inode %p",
1257 direntry, direntry->d_name.name, newinode);
cbac3cba 1258
fb8c4b14 1259 if (newinode->i_nlink != 2)
b6b38f70
JP
1260 cFYI(1, "unexpected number of links %d",
1261 newinode->i_nlink);
cbac3cba 1262#endif
2dd29d31
SF
1263 }
1264 kfree(pInfo);
1265 goto mkdir_out;
fb8c4b14 1266 }
c45d707f 1267mkdir_retry_old:
1da177e4 1268 /* BB add setting the equivalent of mode via CreateX w/ACLs */
737b758c
SF
1269 rc = CIFSSMBMkDir(xid, pTcon, full_path, cifs_sb->local_nls,
1270 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
1da177e4 1271 if (rc) {
b6b38f70 1272 cFYI(1, "cifs_mkdir returned 0x%x", rc);
1da177e4
LT
1273 d_drop(direntry);
1274 } else {
fb8c4b14 1275mkdir_get_info:
d8c76e6f 1276 inc_nlink(inode);
c18c842b 1277 if (pTcon->unix_ext)
1da177e4 1278 rc = cifs_get_inode_info_unix(&newinode, full_path,
fb8c4b14 1279 inode->i_sb, xid);
1da177e4
LT
1280 else
1281 rc = cifs_get_inode_info(&newinode, full_path, NULL,
8b1327f6 1282 inode->i_sb, xid, NULL);
1da177e4 1283
b92327fe
SF
1284 if (pTcon->nocase)
1285 direntry->d_op = &cifs_ci_dentry_ops;
1286 else
1287 direntry->d_op = &cifs_dentry_ops;
1da177e4 1288 d_instantiate(direntry, newinode);
2dd29d31 1289 /* setting nlink not necessary except in cases where we
fb8c4b14 1290 * failed to get it from the server or was set bogus */
2dd29d31 1291 if ((direntry->d_inode) && (direntry->d_inode->i_nlink < 2))
fb8c4b14 1292 direntry->d_inode->i_nlink = 2;
95089910 1293
ce3b0f8d 1294 mode &= ~current_umask();
95089910
JL
1295 /* must turn on setgid bit if parent dir has it */
1296 if (inode->i_mode & S_ISGID)
1297 mode |= S_ISGID;
1298
c18c842b 1299 if (pTcon->unix_ext) {
4e1e7fb9
JL
1300 struct cifs_unix_set_info_args args = {
1301 .mode = mode,
1302 .ctime = NO_CHANGE_64,
1303 .atime = NO_CHANGE_64,
1304 .mtime = NO_CHANGE_64,
1305 .device = 0,
1306 };
d0d2f2df 1307 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID) {
a001e5b5 1308 args.uid = (__u64)current_fsuid();
95089910
JL
1309 if (inode->i_mode & S_ISGID)
1310 args.gid = (__u64)inode->i_gid;
1311 else
a001e5b5 1312 args.gid = (__u64)current_fsgid();
1da177e4 1313 } else {
4e1e7fb9
JL
1314 args.uid = NO_CHANGE_64;
1315 args.gid = NO_CHANGE_64;
1da177e4 1316 }
01ea95e3
JL
1317 CIFSSMBUnixSetPathInfo(xid, pTcon, full_path, &args,
1318 cifs_sb->local_nls,
1319 cifs_sb->mnt_cifs_flags &
1320 CIFS_MOUNT_MAP_SPECIAL_CHR);
3ce53fc4 1321 } else {
67750fb9
JL
1322 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL) &&
1323 (mode & S_IWUGO) == 0) {
1324 FILE_BASIC_INFO pInfo;
6b37faa1
JL
1325 struct cifsInodeInfo *cifsInode;
1326 u32 dosattrs;
1327
67750fb9 1328 memset(&pInfo, 0, sizeof(pInfo));
6b37faa1
JL
1329 cifsInode = CIFS_I(newinode);
1330 dosattrs = cifsInode->cifsAttrs|ATTR_READONLY;
1331 pInfo.Attributes = cpu_to_le32(dosattrs);
1332 tmprc = CIFSSMBSetPathInfo(xid, pTcon,
1333 full_path, &pInfo,
1334 cifs_sb->local_nls,
67750fb9
JL
1335 cifs_sb->mnt_cifs_flags &
1336 CIFS_MOUNT_MAP_SPECIAL_CHR);
6b37faa1
JL
1337 if (tmprc == 0)
1338 cifsInode->cifsAttrs = dosattrs;
67750fb9 1339 }
fb8c4b14 1340 if (direntry->d_inode) {
b0fd30d3
JL
1341 if (cifs_sb->mnt_cifs_flags &
1342 CIFS_MOUNT_DYNPERM)
1343 direntry->d_inode->i_mode =
1344 (mode | S_IFDIR);
4e94a105 1345
fb8c4b14 1346 if (cifs_sb->mnt_cifs_flags &
6473a559 1347 CIFS_MOUNT_SET_UID) {
fb8c4b14 1348 direntry->d_inode->i_uid =
a001e5b5 1349 current_fsuid();
95089910
JL
1350 if (inode->i_mode & S_ISGID)
1351 direntry->d_inode->i_gid =
1352 inode->i_gid;
1353 else
1354 direntry->d_inode->i_gid =
a001e5b5 1355 current_fsgid();
6473a559
SF
1356 }
1357 }
2a138ebb 1358 }
1da177e4 1359 }
fb8c4b14 1360mkdir_out:
1da177e4
LT
1361 kfree(full_path);
1362 FreeXid(xid);
1363 return rc;
1364}
1365
1366int cifs_rmdir(struct inode *inode, struct dentry *direntry)
1367{
1368 int rc = 0;
1369 int xid;
1370 struct cifs_sb_info *cifs_sb;
1371 struct cifsTconInfo *pTcon;
1372 char *full_path = NULL;
1373 struct cifsInodeInfo *cifsInode;
1374
b6b38f70 1375 cFYI(1, "cifs_rmdir, inode = 0x%p", inode);
1da177e4
LT
1376
1377 xid = GetXid();
1378
1379 cifs_sb = CIFS_SB(inode->i_sb);
1380 pTcon = cifs_sb->tcon;
1381
7f57356b 1382 full_path = build_path_from_dentry(direntry);
1da177e4 1383 if (full_path == NULL) {
0f3bc09e 1384 rc = -ENOMEM;
1da177e4 1385 FreeXid(xid);
0f3bc09e 1386 return rc;
1da177e4
LT
1387 }
1388
737b758c
SF
1389 rc = CIFSSMBRmDir(xid, pTcon, full_path, cifs_sb->local_nls,
1390 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
1da177e4
LT
1391
1392 if (!rc) {
9a53c3a7 1393 drop_nlink(inode);
3677db10 1394 spin_lock(&direntry->d_inode->i_lock);
fb8c4b14 1395 i_size_write(direntry->d_inode, 0);
ce71ec36 1396 clear_nlink(direntry->d_inode);
3677db10 1397 spin_unlock(&direntry->d_inode->i_lock);
1da177e4
LT
1398 }
1399
1400 cifsInode = CIFS_I(direntry->d_inode);
1401 cifsInode->time = 0; /* force revalidate to go get info when
1402 needed */
42c24544
SF
1403
1404 cifsInode = CIFS_I(inode);
1405 cifsInode->time = 0; /* force revalidate to get parent dir info
1406 since cached search results now invalid */
1407
1da177e4
LT
1408 direntry->d_inode->i_ctime = inode->i_ctime = inode->i_mtime =
1409 current_fs_time(inode->i_sb);
1410
1411 kfree(full_path);
1412 FreeXid(xid);
1413 return rc;
1414}
1415
ee2fd967
SF
1416static int
1417cifs_do_rename(int xid, struct dentry *from_dentry, const char *fromPath,
1418 struct dentry *to_dentry, const char *toPath)
1419{
1420 struct cifs_sb_info *cifs_sb = CIFS_SB(from_dentry->d_sb);
1421 struct cifsTconInfo *pTcon = cifs_sb->tcon;
1422 __u16 srcfid;
1423 int oplock, rc;
1424
1425 /* try path-based rename first */
1426 rc = CIFSSMBRename(xid, pTcon, fromPath, toPath, cifs_sb->local_nls,
1427 cifs_sb->mnt_cifs_flags &
1428 CIFS_MOUNT_MAP_SPECIAL_CHR);
1429
1430 /*
1431 * don't bother with rename by filehandle unless file is busy and
1432 * source Note that cross directory moves do not work with
1433 * rename by filehandle to various Windows servers.
1434 */
1435 if (rc == 0 || rc != -ETXTBSY)
1436 return rc;
1437
ed0e3ace
JL
1438 /* open-file renames don't work across directories */
1439 if (to_dentry->d_parent != from_dentry->d_parent)
1440 return rc;
1441
ee2fd967
SF
1442 /* open the file to be renamed -- we need DELETE perms */
1443 rc = CIFSSMBOpen(xid, pTcon, fromPath, FILE_OPEN, DELETE,
1444 CREATE_NOT_DIR, &srcfid, &oplock, NULL,
1445 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1446 CIFS_MOUNT_MAP_SPECIAL_CHR);
1447
1448 if (rc == 0) {
1449 rc = CIFSSMBRenameOpenFile(xid, pTcon, srcfid,
1450 (const char *) to_dentry->d_name.name,
1451 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1452 CIFS_MOUNT_MAP_SPECIAL_CHR);
1453
1454 CIFSSMBClose(xid, pTcon, srcfid);
1455 }
1456
1457 return rc;
1458}
1459
14121bdc
JL
1460int cifs_rename(struct inode *source_dir, struct dentry *source_dentry,
1461 struct inode *target_dir, struct dentry *target_dentry)
1da177e4 1462{
ee2fd967
SF
1463 char *fromName = NULL;
1464 char *toName = NULL;
639e7a91 1465 struct cifs_sb_info *cifs_sb;
14121bdc 1466 struct cifsTconInfo *tcon;
ee2fd967
SF
1467 FILE_UNIX_BASIC_INFO *info_buf_source = NULL;
1468 FILE_UNIX_BASIC_INFO *info_buf_target;
8d281efb 1469 int xid, rc, tmprc;
1da177e4 1470
639e7a91
JL
1471 cifs_sb = CIFS_SB(source_dir->i_sb);
1472 tcon = cifs_sb->tcon;
1da177e4 1473
ee2fd967
SF
1474 xid = GetXid();
1475
ee2fd967
SF
1476 /*
1477 * we already have the rename sem so we do not need to
1478 * grab it again here to protect the path integrity
1479 */
14121bdc 1480 fromName = build_path_from_dentry(source_dentry);
ee2fd967
SF
1481 if (fromName == NULL) {
1482 rc = -ENOMEM;
1483 goto cifs_rename_exit;
1484 }
1485
14121bdc 1486 toName = build_path_from_dentry(target_dentry);
ee2fd967 1487 if (toName == NULL) {
1da177e4
LT
1488 rc = -ENOMEM;
1489 goto cifs_rename_exit;
1490 }
1491
14121bdc
JL
1492 rc = cifs_do_rename(xid, source_dentry, fromName,
1493 target_dentry, toName);
ee2fd967 1494
14121bdc
JL
1495 if (rc == -EEXIST && tcon->unix_ext) {
1496 /*
1497 * Are src and dst hardlinks of same inode? We can
1498 * only tell with unix extensions enabled
1499 */
1500 info_buf_source =
1501 kmalloc(2 * sizeof(FILE_UNIX_BASIC_INFO),
1502 GFP_KERNEL);
1503 if (info_buf_source == NULL) {
1504 rc = -ENOMEM;
1505 goto cifs_rename_exit;
1506 }
1507
1508 info_buf_target = info_buf_source + 1;
8d281efb 1509 tmprc = CIFSSMBUnixQPathInfo(xid, tcon, fromName,
14121bdc 1510 info_buf_source,
639e7a91
JL
1511 cifs_sb->local_nls,
1512 cifs_sb->mnt_cifs_flags &
14121bdc 1513 CIFS_MOUNT_MAP_SPECIAL_CHR);
8d281efb 1514 if (tmprc != 0)
14121bdc 1515 goto unlink_target;
ee2fd967 1516
639e7a91
JL
1517 tmprc = CIFSSMBUnixQPathInfo(xid, tcon, toName,
1518 info_buf_target,
1519 cifs_sb->local_nls,
1520 cifs_sb->mnt_cifs_flags &
14121bdc
JL
1521 CIFS_MOUNT_MAP_SPECIAL_CHR);
1522
8d281efb 1523 if (tmprc == 0 && (info_buf_source->UniqueId ==
ae6884a9 1524 info_buf_target->UniqueId)) {
14121bdc 1525 /* same file, POSIX says that this is a noop */
ae6884a9 1526 rc = 0;
14121bdc 1527 goto cifs_rename_exit;
ae6884a9 1528 }
14121bdc 1529 } /* else ... BB we could add the same check for Windows by
ee2fd967 1530 checking the UniqueId via FILE_INTERNAL_INFO */
14121bdc 1531
ee2fd967 1532unlink_target:
fc6f3943
JL
1533 /* Try unlinking the target dentry if it's not negative */
1534 if (target_dentry->d_inode && (rc == -EACCES || rc == -EEXIST)) {
8d281efb 1535 tmprc = cifs_unlink(target_dir, target_dentry);
14121bdc
JL
1536 if (tmprc)
1537 goto cifs_rename_exit;
1538
14121bdc
JL
1539 rc = cifs_do_rename(xid, source_dentry, fromName,
1540 target_dentry, toName);
1da177e4
LT
1541 }
1542
1543cifs_rename_exit:
ee2fd967 1544 kfree(info_buf_source);
1da177e4
LT
1545 kfree(fromName);
1546 kfree(toName);
1547 FreeXid(xid);
1548 return rc;
1549}
1550
df2cf170
JL
1551static bool
1552cifs_inode_needs_reval(struct inode *inode)
1da177e4 1553{
df2cf170 1554 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
1da177e4 1555
df2cf170
JL
1556 if (cifs_i->clientCanCacheRead)
1557 return false;
1da177e4 1558
df2cf170
JL
1559 if (!lookupCacheEnabled)
1560 return true;
1da177e4 1561
df2cf170
JL
1562 if (cifs_i->time == 0)
1563 return true;
1da177e4 1564
df2cf170
JL
1565 /* FIXME: the actimeo should be tunable */
1566 if (time_after_eq(jiffies, cifs_i->time + HZ))
1567 return true;
1568
db19272e
JL
1569 /* hardlinked files w/ noserverino get "special" treatment */
1570 if (!(CIFS_SB(inode->i_sb)->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM) &&
1571 S_ISREG(inode->i_mode) && inode->i_nlink != 1)
1572 return true;
1573
df2cf170
JL
1574 return false;
1575}
1576
1577/* check invalid_mapping flag and zap the cache if it's set */
1578static void
1579cifs_invalidate_mapping(struct inode *inode)
1580{
1581 int rc;
1582 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
1583
1584 cifs_i->invalid_mapping = false;
1585
1586 /* write back any cached data */
1587 if (inode->i_mapping && inode->i_mapping->nrpages != 0) {
1588 rc = filemap_write_and_wait(inode->i_mapping);
1589 if (rc)
1590 cifs_i->write_behind_rc = rc;
1591 }
1592 invalidate_remote_inode(inode);
9451a9a5 1593 cifs_fscache_reset_inode_cookie(inode);
df2cf170
JL
1594}
1595
abab095d
JL
1596int cifs_revalidate_file(struct file *filp)
1597{
1598 int rc = 0;
1599 struct inode *inode = filp->f_path.dentry->d_inode;
1600
1601 if (!cifs_inode_needs_reval(inode))
1602 goto check_inval;
1603
1604 if (CIFS_SB(inode->i_sb)->tcon->unix_ext)
1605 rc = cifs_get_file_info_unix(filp);
1606 else
1607 rc = cifs_get_file_info(filp);
1608
1609check_inval:
1610 if (CIFS_I(inode)->invalid_mapping)
1611 cifs_invalidate_mapping(inode);
1612
1613 return rc;
1614}
1615
df2cf170
JL
1616/* revalidate a dentry's inode attributes */
1617int cifs_revalidate_dentry(struct dentry *dentry)
1618{
1619 int xid;
1620 int rc = 0;
1621 char *full_path = NULL;
1622 struct inode *inode = dentry->d_inode;
1623 struct super_block *sb = dentry->d_sb;
1624
1625 if (inode == NULL)
1626 return -ENOENT;
1da177e4
LT
1627
1628 xid = GetXid();
1629
df2cf170
JL
1630 if (!cifs_inode_needs_reval(inode))
1631 goto check_inval;
1da177e4
LT
1632
1633 /* can not safely grab the rename sem here if rename calls revalidate
1634 since that would deadlock */
df2cf170 1635 full_path = build_path_from_dentry(dentry);
1da177e4 1636 if (full_path == NULL) {
0f3bc09e 1637 rc = -ENOMEM;
df2cf170 1638 goto check_inval;
1da177e4
LT
1639 }
1640
f19159dc 1641 cFYI(1, "Revalidate: %s inode 0x%p count %d dentry: 0x%p d_time %ld "
df2cf170 1642 "jiffies %ld", full_path, inode, inode->i_count.counter,
f19159dc 1643 dentry, dentry->d_time, jiffies);
1da177e4 1644
df2cf170
JL
1645 if (CIFS_SB(sb)->tcon->unix_ext)
1646 rc = cifs_get_inode_info_unix(&inode, full_path, sb, xid);
1647 else
1648 rc = cifs_get_inode_info(&inode, full_path, NULL, sb,
1649 xid, NULL);
1da177e4 1650
df2cf170
JL
1651check_inval:
1652 if (CIFS_I(inode)->invalid_mapping)
1653 cifs_invalidate_mapping(inode);
50c2f753 1654
1da177e4
LT
1655 kfree(full_path);
1656 FreeXid(xid);
1657 return rc;
1658}
1659
1660int cifs_getattr(struct vfsmount *mnt, struct dentry *dentry,
1661 struct kstat *stat)
1662{
df2cf170 1663 int err = cifs_revalidate_dentry(dentry);
5fe14c85 1664 if (!err) {
1da177e4 1665 generic_fillattr(dentry->d_inode, stat);
5fe14c85 1666 stat->blksize = CIFS_MAX_MSGSIZE;
cc0bad75 1667 stat->ino = CIFS_I(dentry->d_inode)->uniqueid;
5fe14c85 1668 }
1da177e4
LT
1669 return err;
1670}
1671
1672static int cifs_truncate_page(struct address_space *mapping, loff_t from)
1673{
1674 pgoff_t index = from >> PAGE_CACHE_SHIFT;
1675 unsigned offset = from & (PAGE_CACHE_SIZE - 1);
1676 struct page *page;
1da177e4
LT
1677 int rc = 0;
1678
1679 page = grab_cache_page(mapping, index);
1680 if (!page)
1681 return -ENOMEM;
1682
eebd2aa3 1683 zero_user_segment(page, offset, PAGE_CACHE_SIZE);
1da177e4
LT
1684 unlock_page(page);
1685 page_cache_release(page);
1686 return rc;
1687}
1688
1b947463 1689static void cifs_setsize(struct inode *inode, loff_t offset)
3677db10 1690{
c08d3b0e 1691 loff_t oldsize;
3677db10 1692
ba6a46a0 1693 spin_lock(&inode->i_lock);
c08d3b0e 1694 oldsize = inode->i_size;
3677db10 1695 i_size_write(inode, offset);
ba6a46a0 1696 spin_unlock(&inode->i_lock);
1b947463 1697
c08d3b0e 1698 truncate_pagecache(inode, oldsize, offset);
3677db10
SF
1699}
1700
8efdbde6
JL
1701static int
1702cifs_set_file_size(struct inode *inode, struct iattr *attrs,
1703 int xid, char *full_path)
1704{
1705 int rc;
1706 struct cifsFileInfo *open_file;
1707 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
1708 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
1709 struct cifsTconInfo *pTcon = cifs_sb->tcon;
1710
1711 /*
1712 * To avoid spurious oplock breaks from server, in the case of
1713 * inodes that we already have open, avoid doing path based
1714 * setting of file size if we can do it by handle.
1715 * This keeps our caching token (oplock) and avoids timeouts
1716 * when the local oplock break takes longer to flush
1717 * writebehind data than the SMB timeout for the SetPathInfo
1718 * request would allow
1719 */
1720 open_file = find_writable_file(cifsInode);
1721 if (open_file) {
1722 __u16 nfid = open_file->netfid;
1723 __u32 npid = open_file->pid;
1724 rc = CIFSSMBSetFileSize(xid, pTcon, attrs->ia_size, nfid,
1725 npid, false);
6ab409b5 1726 cifsFileInfo_put(open_file);
b6b38f70 1727 cFYI(1, "SetFSize for attrs rc = %d", rc);
8efdbde6
JL
1728 if ((rc == -EINVAL) || (rc == -EOPNOTSUPP)) {
1729 unsigned int bytes_written;
1730 rc = CIFSSMBWrite(xid, pTcon, nfid, 0, attrs->ia_size,
1731 &bytes_written, NULL, NULL, 1);
b6b38f70 1732 cFYI(1, "Wrt seteof rc %d", rc);
8efdbde6
JL
1733 }
1734 } else
1735 rc = -EINVAL;
1736
1737 if (rc != 0) {
1738 /* Set file size by pathname rather than by handle
1739 either because no valid, writeable file handle for
1740 it was found or because there was an error setting
1741 it by handle */
1742 rc = CIFSSMBSetEOF(xid, pTcon, full_path, attrs->ia_size,
1743 false, cifs_sb->local_nls,
1744 cifs_sb->mnt_cifs_flags &
1745 CIFS_MOUNT_MAP_SPECIAL_CHR);
b6b38f70 1746 cFYI(1, "SetEOF by path (setattrs) rc = %d", rc);
8efdbde6
JL
1747 if ((rc == -EINVAL) || (rc == -EOPNOTSUPP)) {
1748 __u16 netfid;
1749 int oplock = 0;
1750
1751 rc = SMBLegacyOpen(xid, pTcon, full_path,
1752 FILE_OPEN, GENERIC_WRITE,
1753 CREATE_NOT_DIR, &netfid, &oplock, NULL,
1754 cifs_sb->local_nls,
1755 cifs_sb->mnt_cifs_flags &
1756 CIFS_MOUNT_MAP_SPECIAL_CHR);
1757 if (rc == 0) {
1758 unsigned int bytes_written;
1759 rc = CIFSSMBWrite(xid, pTcon, netfid, 0,
1760 attrs->ia_size,
1761 &bytes_written, NULL,
1762 NULL, 1);
b6b38f70 1763 cFYI(1, "wrt seteof rc %d", rc);
8efdbde6
JL
1764 CIFSSMBClose(xid, pTcon, netfid);
1765 }
1766 }
1767 }
1768
1769 if (rc == 0) {
fbec9ab9 1770 cifsInode->server_eof = attrs->ia_size;
1b947463 1771 cifs_setsize(inode, attrs->ia_size);
8efdbde6
JL
1772 cifs_truncate_page(inode->i_mapping, inode->i_size);
1773 }
1774
1775 return rc;
1776}
1777
3fe5c1dd
JL
1778static int
1779cifs_setattr_unix(struct dentry *direntry, struct iattr *attrs)
1780{
1781 int rc;
1782 int xid;
1783 char *full_path = NULL;
1784 struct inode *inode = direntry->d_inode;
1785 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
1786 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
1787 struct cifsTconInfo *pTcon = cifs_sb->tcon;
1788 struct cifs_unix_set_info_args *args = NULL;
3bbeeb3c 1789 struct cifsFileInfo *open_file;
3fe5c1dd 1790
b6b38f70
JP
1791 cFYI(1, "setattr_unix on file %s attrs->ia_valid=0x%x",
1792 direntry->d_name.name, attrs->ia_valid);
3fe5c1dd
JL
1793
1794 xid = GetXid();
1795
db78b877
CH
1796 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_PERM)
1797 attrs->ia_valid |= ATTR_FORCE;
1798
1799 rc = inode_change_ok(inode, attrs);
1800 if (rc < 0)
1801 goto out;
3fe5c1dd
JL
1802
1803 full_path = build_path_from_dentry(direntry);
1804 if (full_path == NULL) {
1805 rc = -ENOMEM;
1806 goto out;
1807 }
1808
0f4d634c
JL
1809 /*
1810 * Attempt to flush data before changing attributes. We need to do
1811 * this for ATTR_SIZE and ATTR_MTIME for sure, and if we change the
1812 * ownership or mode then we may also need to do this. Here, we take
1813 * the safe way out and just do the flush on all setattr requests. If
1814 * the flush returns error, store it to report later and continue.
1815 *
1816 * BB: This should be smarter. Why bother flushing pages that
1817 * will be truncated anyway? Also, should we error out here if
1818 * the flush returns error?
1819 */
1820 rc = filemap_write_and_wait(inode->i_mapping);
1821 if (rc != 0) {
1822 cifsInode->write_behind_rc = rc;
1823 rc = 0;
3fe5c1dd
JL
1824 }
1825
1826 if (attrs->ia_valid & ATTR_SIZE) {
1827 rc = cifs_set_file_size(inode, attrs, xid, full_path);
1828 if (rc != 0)
1829 goto out;
1830 }
1831
1832 /* skip mode change if it's just for clearing setuid/setgid */
1833 if (attrs->ia_valid & (ATTR_KILL_SUID|ATTR_KILL_SGID))
1834 attrs->ia_valid &= ~ATTR_MODE;
1835
1836 args = kmalloc(sizeof(*args), GFP_KERNEL);
1837 if (args == NULL) {
1838 rc = -ENOMEM;
1839 goto out;
1840 }
1841
1842 /* set up the struct */
1843 if (attrs->ia_valid & ATTR_MODE)
1844 args->mode = attrs->ia_mode;
1845 else
1846 args->mode = NO_CHANGE_64;
1847
1848 if (attrs->ia_valid & ATTR_UID)
1849 args->uid = attrs->ia_uid;
1850 else
1851 args->uid = NO_CHANGE_64;
1852
1853 if (attrs->ia_valid & ATTR_GID)
1854 args->gid = attrs->ia_gid;
1855 else
1856 args->gid = NO_CHANGE_64;
1857
1858 if (attrs->ia_valid & ATTR_ATIME)
1859 args->atime = cifs_UnixTimeToNT(attrs->ia_atime);
1860 else
1861 args->atime = NO_CHANGE_64;
1862
1863 if (attrs->ia_valid & ATTR_MTIME)
1864 args->mtime = cifs_UnixTimeToNT(attrs->ia_mtime);
1865 else
1866 args->mtime = NO_CHANGE_64;
1867
1868 if (attrs->ia_valid & ATTR_CTIME)
1869 args->ctime = cifs_UnixTimeToNT(attrs->ia_ctime);
1870 else
1871 args->ctime = NO_CHANGE_64;
1872
1873 args->device = 0;
3bbeeb3c
JL
1874 open_file = find_writable_file(cifsInode);
1875 if (open_file) {
1876 u16 nfid = open_file->netfid;
1877 u32 npid = open_file->pid;
1878 rc = CIFSSMBUnixSetFileInfo(xid, pTcon, args, nfid, npid);
6ab409b5 1879 cifsFileInfo_put(open_file);
3bbeeb3c
JL
1880 } else {
1881 rc = CIFSSMBUnixSetPathInfo(xid, pTcon, full_path, args,
01ea95e3
JL
1882 cifs_sb->local_nls,
1883 cifs_sb->mnt_cifs_flags &
1884 CIFS_MOUNT_MAP_SPECIAL_CHR);
3bbeeb3c 1885 }
3fe5c1dd 1886
1025774c
CH
1887 if (rc)
1888 goto out;
ccd4bb1b 1889
1025774c 1890 if ((attrs->ia_valid & ATTR_SIZE) &&
1b947463
CH
1891 attrs->ia_size != i_size_read(inode))
1892 truncate_setsize(inode, attrs->ia_size);
1025774c
CH
1893
1894 setattr_copy(inode, attrs);
1895 mark_inode_dirty(inode);
1896
1897 /* force revalidate when any of these times are set since some
1898 of the fs types (eg ext3, fat) do not have fine enough
1899 time granularity to match protocol, and we do not have a
1900 a way (yet) to query the server fs's time granularity (and
1901 whether it rounds times down).
1902 */
1903 if (attrs->ia_valid & (ATTR_MTIME | ATTR_CTIME))
1904 cifsInode->time = 0;
3fe5c1dd
JL
1905out:
1906 kfree(args);
1907 kfree(full_path);
1908 FreeXid(xid);
1909 return rc;
1910}
1911
0510eeb7
JL
1912static int
1913cifs_setattr_nounix(struct dentry *direntry, struct iattr *attrs)
1da177e4
LT
1914{
1915 int xid;
3fe5c1dd
JL
1916 struct inode *inode = direntry->d_inode;
1917 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
3fe5c1dd 1918 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
1da177e4
LT
1919 char *full_path = NULL;
1920 int rc = -EACCES;
feb3e20c 1921 __u32 dosattr = 0;
4e1e7fb9 1922 __u64 mode = NO_CHANGE_64;
3fe5c1dd 1923
1da177e4
LT
1924 xid = GetXid();
1925
b6b38f70
JP
1926 cFYI(1, "setattr on file %s attrs->iavalid 0x%x",
1927 direntry->d_name.name, attrs->ia_valid);
6473a559 1928
db78b877
CH
1929 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_PERM)
1930 attrs->ia_valid |= ATTR_FORCE;
1931
1932 rc = inode_change_ok(inode, attrs);
1933 if (rc < 0) {
1934 FreeXid(xid);
1935 return rc;
6473a559 1936 }
50c2f753 1937
7f57356b 1938 full_path = build_path_from_dentry(direntry);
1da177e4 1939 if (full_path == NULL) {
0f3bc09e 1940 rc = -ENOMEM;
1da177e4 1941 FreeXid(xid);
0f3bc09e 1942 return rc;
1da177e4 1943 }
1da177e4 1944
0f4d634c
JL
1945 /*
1946 * Attempt to flush data before changing attributes. We need to do
1947 * this for ATTR_SIZE and ATTR_MTIME for sure, and if we change the
1948 * ownership or mode then we may also need to do this. Here, we take
1949 * the safe way out and just do the flush on all setattr requests. If
1950 * the flush returns error, store it to report later and continue.
1951 *
1952 * BB: This should be smarter. Why bother flushing pages that
1953 * will be truncated anyway? Also, should we error out here if
1954 * the flush returns error?
1955 */
1956 rc = filemap_write_and_wait(inode->i_mapping);
1957 if (rc != 0) {
1958 cifsInode->write_behind_rc = rc;
1959 rc = 0;
50531444 1960 }
cea21805 1961
50531444 1962 if (attrs->ia_valid & ATTR_SIZE) {
8efdbde6
JL
1963 rc = cifs_set_file_size(inode, attrs, xid, full_path);
1964 if (rc != 0)
e30dcf3a 1965 goto cifs_setattr_exit;
1da177e4 1966 }
4ca691a8
JL
1967
1968 /*
1969 * Without unix extensions we can't send ownership changes to the
1970 * server, so silently ignore them. This is consistent with how
1971 * local DOS/Windows filesystems behave (VFAT, NTFS, etc). With
1972 * CIFSACL support + proper Windows to Unix idmapping, we may be
1973 * able to support this in the future.
1974 */
3fe5c1dd 1975 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID))
4ca691a8 1976 attrs->ia_valid &= ~(ATTR_UID | ATTR_GID);
1da177e4 1977
d32c4f26
JL
1978 /* skip mode change if it's just for clearing setuid/setgid */
1979 if (attrs->ia_valid & (ATTR_KILL_SUID|ATTR_KILL_SGID))
1980 attrs->ia_valid &= ~ATTR_MODE;
1981
1da177e4 1982 if (attrs->ia_valid & ATTR_MODE) {
b6b38f70 1983 cFYI(1, "Mode changed to 0%o", attrs->ia_mode);
1da177e4 1984 mode = attrs->ia_mode;
1da177e4
LT
1985 }
1986
3fe5c1dd 1987 if (attrs->ia_valid & ATTR_MODE) {
cdbce9c8 1988 rc = 0;
97837582
SF
1989#ifdef CONFIG_CIFS_EXPERIMENTAL
1990 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL)
02eadeff 1991 rc = mode_to_acl(inode, full_path, mode);
5132861a 1992 else
97837582 1993#endif
5132861a
JL
1994 if (((mode & S_IWUGO) == 0) &&
1995 (cifsInode->cifsAttrs & ATTR_READONLY) == 0) {
feb3e20c
JL
1996
1997 dosattr = cifsInode->cifsAttrs | ATTR_READONLY;
1998
5132861a
JL
1999 /* fix up mode if we're not using dynperm */
2000 if ((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM) == 0)
2001 attrs->ia_mode = inode->i_mode & ~S_IWUGO;
2002 } else if ((mode & S_IWUGO) &&
2003 (cifsInode->cifsAttrs & ATTR_READONLY)) {
feb3e20c
JL
2004
2005 dosattr = cifsInode->cifsAttrs & ~ATTR_READONLY;
2006 /* Attributes of 0 are ignored */
2007 if (dosattr == 0)
2008 dosattr |= ATTR_NORMAL;
5132861a
JL
2009
2010 /* reset local inode permissions to normal */
2011 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM)) {
2012 attrs->ia_mode &= ~(S_IALLUGO);
2013 if (S_ISDIR(inode->i_mode))
2014 attrs->ia_mode |=
2015 cifs_sb->mnt_dir_mode;
2016 else
2017 attrs->ia_mode |=
2018 cifs_sb->mnt_file_mode;
2019 }
2020 } else if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM)) {
2021 /* ignore mode change - ATTR_READONLY hasn't changed */
2022 attrs->ia_valid &= ~ATTR_MODE;
1da177e4 2023 }
1da177e4
LT
2024 }
2025
feb3e20c
JL
2026 if (attrs->ia_valid & (ATTR_MTIME|ATTR_ATIME|ATTR_CTIME) ||
2027 ((attrs->ia_valid & ATTR_MODE) && dosattr)) {
2028 rc = cifs_set_file_info(inode, attrs, xid, full_path, dosattr);
2029 /* BB: check for rc = -EOPNOTSUPP and switch to legacy mode */
1da177e4 2030
e30dcf3a
SF
2031 /* Even if error on time set, no sense failing the call if
2032 the server would set the time to a reasonable value anyway,
2033 and this check ensures that we are not being called from
2034 sys_utimes in which case we ought to fail the call back to
2035 the user when the server rejects the call */
fb8c4b14 2036 if ((rc) && (attrs->ia_valid &
feb3e20c 2037 (ATTR_MODE | ATTR_GID | ATTR_UID | ATTR_SIZE)))
e30dcf3a 2038 rc = 0;
1da177e4
LT
2039 }
2040
2041 /* do not need local check to inode_check_ok since the server does
2042 that */
1025774c
CH
2043 if (rc)
2044 goto cifs_setattr_exit;
2045
2046 if ((attrs->ia_valid & ATTR_SIZE) &&
1b947463
CH
2047 attrs->ia_size != i_size_read(inode))
2048 truncate_setsize(inode, attrs->ia_size);
1025774c
CH
2049
2050 setattr_copy(inode, attrs);
2051 mark_inode_dirty(inode);
2052 return 0;
2053
e30dcf3a 2054cifs_setattr_exit:
1da177e4
LT
2055 kfree(full_path);
2056 FreeXid(xid);
2057 return rc;
2058}
2059
0510eeb7
JL
2060int
2061cifs_setattr(struct dentry *direntry, struct iattr *attrs)
2062{
2063 struct inode *inode = direntry->d_inode;
2064 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
2065 struct cifsTconInfo *pTcon = cifs_sb->tcon;
2066
2067 if (pTcon->unix_ext)
2068 return cifs_setattr_unix(direntry, attrs);
2069
2070 return cifs_setattr_nounix(direntry, attrs);
2071
2072 /* BB: add cifs_setattr_legacy for really old servers */
2073}
2074
99ee4dbd 2075#if 0
1da177e4
LT
2076void cifs_delete_inode(struct inode *inode)
2077{
b6b38f70 2078 cFYI(1, "In cifs_delete_inode, inode = 0x%p", inode);
1da177e4
LT
2079 /* may have to add back in if and when safe distributed caching of
2080 directories added e.g. via FindNotify */
2081}
99ee4dbd 2082#endif