]> git.proxmox.com Git - mirror_ubuntu-artful-kernel.git/blame - fs/cifs/inode.c
[CIFS] relinquish fscache cookie before freeing CIFSTconInfo
[mirror_ubuntu-artful-kernel.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
727 if (CIFS_I(inode)->uniqueid != fattr->cf_uniqueid)
728 return 0;
729
3d694380
JL
730 /*
731 * uh oh -- it's a directory. We can't use it since hardlinked dirs are
732 * verboten. Disable serverino and return it as if it were found, the
733 * caller can discard it, generate a uniqueid and retry the find
734 */
735 if (S_ISDIR(inode->i_mode) && !list_empty(&inode->i_dentry)) {
736 fattr->cf_flags |= CIFS_FATTR_INO_COLLISION;
737 cifs_autodisable_serverino(CIFS_SB(inode->i_sb));
738 }
739
cc0bad75
JL
740 return 1;
741}
742
743static int
744cifs_init_inode(struct inode *inode, void *opaque)
745{
746 struct cifs_fattr *fattr = (struct cifs_fattr *) opaque;
747
748 CIFS_I(inode)->uniqueid = fattr->cf_uniqueid;
749 return 0;
750}
751
752/* Given fattrs, get a corresponding inode */
753struct inode *
754cifs_iget(struct super_block *sb, struct cifs_fattr *fattr)
755{
756 unsigned long hash;
757 struct inode *inode;
758
3d694380 759retry_iget5_locked:
b6b38f70 760 cFYI(1, "looking for uniqueid=%llu", fattr->cf_uniqueid);
cc0bad75
JL
761
762 /* hash down to 32-bits on 32-bit arch */
763 hash = cifs_uniqueid_to_ino_t(fattr->cf_uniqueid);
764
765 inode = iget5_locked(sb, hash, cifs_find_inode, cifs_init_inode, fattr);
cc0bad75 766 if (inode) {
3d694380
JL
767 /* was there a problematic inode number collision? */
768 if (fattr->cf_flags & CIFS_FATTR_INO_COLLISION) {
769 iput(inode);
770 fattr->cf_uniqueid = iunique(sb, ROOT_I);
771 fattr->cf_flags &= ~CIFS_FATTR_INO_COLLISION;
772 goto retry_iget5_locked;
773 }
774
cc0bad75
JL
775 cifs_fattr_to_inode(inode, fattr);
776 if (sb->s_flags & MS_NOATIME)
777 inode->i_flags |= S_NOATIME | S_NOCMTIME;
778 if (inode->i_state & I_NEW) {
779 inode->i_ino = hash;
0ccd4802 780#ifdef CONFIG_CIFS_FSCACHE
9451a9a5
SJ
781 /* initialize per-inode cache cookie pointer */
782 CIFS_I(inode)->fscache = NULL;
0ccd4802 783#endif
cc0bad75
JL
784 unlock_new_inode(inode);
785 }
786 }
787
788 return inode;
789}
790
1da177e4 791/* gets root inode */
bd433d4c 792struct inode *cifs_root_iget(struct super_block *sb, unsigned long ino)
1da177e4 793{
ce634ab2 794 int xid;
1da177e4 795 struct cifs_sb_info *cifs_sb;
cc0bad75 796 struct inode *inode = NULL;
ce634ab2 797 long rc;
8be0ed44 798 char *full_path;
ce634ab2 799
cc0bad75 800 cifs_sb = CIFS_SB(sb);
e4cce94c 801 full_path = cifs_build_path_to_root(cifs_sb);
8be0ed44
SF
802 if (full_path == NULL)
803 return ERR_PTR(-ENOMEM);
c18c842b 804
8be0ed44 805 xid = GetXid();
0b8f18e3 806 if (cifs_sb->tcon->unix_ext)
cc0bad75 807 rc = cifs_get_inode_info_unix(&inode, full_path, sb, xid);
0b8f18e3
JL
808 else
809 rc = cifs_get_inode_info(&inode, full_path, NULL, sb,
8be0ed44 810 xid, NULL);
0b8f18e3
JL
811
812 if (!inode)
813 return ERR_PTR(-ENOMEM);
cc0bad75 814
0ccd4802 815#ifdef CONFIG_CIFS_FSCACHE
d03382ce
SJ
816 /* populate tcon->resource_id */
817 cifs_sb->tcon->resource_id = CIFS_I(inode)->uniqueid;
0ccd4802 818#endif
d03382ce 819
7f8ed420 820 if (rc && cifs_sb->tcon->ipc) {
b6b38f70 821 cFYI(1, "ipc connection - fake read inode");
7f8ed420
SF
822 inode->i_mode |= S_IFDIR;
823 inode->i_nlink = 2;
824 inode->i_op = &cifs_ipc_inode_ops;
825 inode->i_fop = &simple_dir_operations;
826 inode->i_uid = cifs_sb->mnt_uid;
827 inode->i_gid = cifs_sb->mnt_gid;
ad661334 828 } else if (rc) {
8be0ed44 829 kfree(full_path);
ce634ab2
DH
830 _FreeXid(xid);
831 iget_failed(inode);
832 return ERR_PTR(rc);
7f8ed420
SF
833 }
834
ce634ab2 835
8be0ed44 836 kfree(full_path);
ce634ab2
DH
837 /* can not call macro FreeXid here since in a void func
838 * TODO: This is no longer true
839 */
1da177e4 840 _FreeXid(xid);
ce634ab2 841 return inode;
1da177e4
LT
842}
843
388e57b2
SF
844static int
845cifs_set_file_info(struct inode *inode, struct iattr *attrs, int xid,
846 char *full_path, __u32 dosattr)
847{
848 int rc;
849 int oplock = 0;
850 __u16 netfid;
851 __u32 netpid;
852 bool set_time = false;
853 struct cifsFileInfo *open_file;
854 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
855 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
856 struct cifsTconInfo *pTcon = cifs_sb->tcon;
857 FILE_BASIC_INFO info_buf;
858
1adcb710
SF
859 if (attrs == NULL)
860 return -EINVAL;
861
388e57b2
SF
862 if (attrs->ia_valid & ATTR_ATIME) {
863 set_time = true;
864 info_buf.LastAccessTime =
865 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_atime));
866 } else
867 info_buf.LastAccessTime = 0;
868
869 if (attrs->ia_valid & ATTR_MTIME) {
870 set_time = true;
871 info_buf.LastWriteTime =
872 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_mtime));
873 } else
874 info_buf.LastWriteTime = 0;
875
876 /*
877 * Samba throws this field away, but windows may actually use it.
878 * Do not set ctime unless other time stamps are changed explicitly
879 * (i.e. by utimes()) since we would then have a mix of client and
880 * server times.
881 */
882 if (set_time && (attrs->ia_valid & ATTR_CTIME)) {
b6b38f70 883 cFYI(1, "CIFS - CTIME changed");
388e57b2
SF
884 info_buf.ChangeTime =
885 cpu_to_le64(cifs_UnixTimeToNT(attrs->ia_ctime));
886 } else
887 info_buf.ChangeTime = 0;
888
889 info_buf.CreationTime = 0; /* don't change */
890 info_buf.Attributes = cpu_to_le32(dosattr);
891
892 /*
893 * If the file is already open for write, just use that fileid
894 */
895 open_file = find_writable_file(cifsInode);
896 if (open_file) {
897 netfid = open_file->netfid;
898 netpid = open_file->pid;
899 goto set_via_filehandle;
900 }
901
902 /*
903 * NT4 apparently returns success on this call, but it doesn't
904 * really work.
905 */
906 if (!(pTcon->ses->flags & CIFS_SES_NT4)) {
907 rc = CIFSSMBSetPathInfo(xid, pTcon, full_path,
908 &info_buf, cifs_sb->local_nls,
909 cifs_sb->mnt_cifs_flags &
910 CIFS_MOUNT_MAP_SPECIAL_CHR);
6b37faa1
JL
911 if (rc == 0) {
912 cifsInode->cifsAttrs = dosattr;
913 goto out;
914 } else if (rc != -EOPNOTSUPP && rc != -EINVAL)
388e57b2
SF
915 goto out;
916 }
917
b6b38f70
JP
918 cFYI(1, "calling SetFileInfo since SetPathInfo for "
919 "times not supported by this server");
388e57b2
SF
920 rc = CIFSSMBOpen(xid, pTcon, full_path, FILE_OPEN,
921 SYNCHRONIZE | FILE_WRITE_ATTRIBUTES,
922 CREATE_NOT_DIR, &netfid, &oplock,
923 NULL, cifs_sb->local_nls,
924 cifs_sb->mnt_cifs_flags &
925 CIFS_MOUNT_MAP_SPECIAL_CHR);
926
927 if (rc != 0) {
928 if (rc == -EIO)
929 rc = -EINVAL;
930 goto out;
931 }
932
933 netpid = current->tgid;
934
935set_via_filehandle:
936 rc = CIFSSMBSetFileInfo(xid, pTcon, &info_buf, netfid, netpid);
d388908e
SF
937 if (!rc)
938 cifsInode->cifsAttrs = dosattr;
939
388e57b2
SF
940 if (open_file == NULL)
941 CIFSSMBClose(xid, pTcon, netfid);
942 else
6ab409b5 943 cifsFileInfo_put(open_file);
388e57b2
SF
944out:
945 return rc;
946}
947
a12a1ac7
JL
948/*
949 * open the given file (if it isn't already), set the DELETE_ON_CLOSE bit
950 * and rename it to a random name that hopefully won't conflict with
951 * anything else.
952 */
953static int
3270958b 954cifs_rename_pending_delete(char *full_path, struct dentry *dentry, int xid)
a12a1ac7
JL
955{
956 int oplock = 0;
957 int rc;
958 __u16 netfid;
3270958b 959 struct inode *inode = dentry->d_inode;
a12a1ac7
JL
960 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
961 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
962 struct cifsTconInfo *tcon = cifs_sb->tcon;
3270958b
SF
963 __u32 dosattr, origattr;
964 FILE_BASIC_INFO *info_buf = NULL;
a12a1ac7
JL
965
966 rc = CIFSSMBOpen(xid, tcon, full_path, FILE_OPEN,
dd1db2de 967 DELETE|FILE_WRITE_ATTRIBUTES, CREATE_NOT_DIR,
a12a1ac7
JL
968 &netfid, &oplock, NULL, cifs_sb->local_nls,
969 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
970 if (rc != 0)
971 goto out;
972
3270958b
SF
973 origattr = cifsInode->cifsAttrs;
974 if (origattr == 0)
975 origattr |= ATTR_NORMAL;
976
977 dosattr = origattr & ~ATTR_READONLY;
a12a1ac7
JL
978 if (dosattr == 0)
979 dosattr |= ATTR_NORMAL;
980 dosattr |= ATTR_HIDDEN;
981
3270958b
SF
982 /* set ATTR_HIDDEN and clear ATTR_READONLY, but only if needed */
983 if (dosattr != origattr) {
984 info_buf = kzalloc(sizeof(*info_buf), GFP_KERNEL);
985 if (info_buf == NULL) {
986 rc = -ENOMEM;
987 goto out_close;
988 }
989 info_buf->Attributes = cpu_to_le32(dosattr);
990 rc = CIFSSMBSetFileInfo(xid, tcon, info_buf, netfid,
991 current->tgid);
992 /* although we would like to mark the file hidden
993 if that fails we will still try to rename it */
41346098 994 if (rc != 0)
3270958b
SF
995 cifsInode->cifsAttrs = dosattr;
996 else
997 dosattr = origattr; /* since not able to change them */
a12a1ac7 998 }
a12a1ac7 999
dd1db2de
JL
1000 /* rename the file */
1001 rc = CIFSSMBRenameOpenFile(xid, tcon, netfid, NULL, cifs_sb->local_nls,
a12a1ac7
JL
1002 cifs_sb->mnt_cifs_flags &
1003 CIFS_MOUNT_MAP_SPECIAL_CHR);
3270958b
SF
1004 if (rc != 0) {
1005 rc = -ETXTBSY;
1006 goto undo_setattr;
1007 }
6d22f098 1008
3270958b
SF
1009 /* try to set DELETE_ON_CLOSE */
1010 if (!cifsInode->delete_pending) {
1011 rc = CIFSSMBSetFileDisposition(xid, tcon, true, netfid,
1012 current->tgid);
1013 /*
1014 * some samba versions return -ENOENT when we try to set the
1015 * file disposition here. Likely a samba bug, but work around
1016 * it for now. This means that some cifsXXX files may hang
1017 * around after they shouldn't.
1018 *
1019 * BB: remove this hack after more servers have the fix
1020 */
1021 if (rc == -ENOENT)
1022 rc = 0;
1023 else if (rc != 0) {
1024 rc = -ETXTBSY;
1025 goto undo_rename;
1026 }
1027 cifsInode->delete_pending = true;
1028 }
7ce86d5a 1029
a12a1ac7
JL
1030out_close:
1031 CIFSSMBClose(xid, tcon, netfid);
1032out:
3270958b 1033 kfree(info_buf);
a12a1ac7 1034 return rc;
3270958b
SF
1035
1036 /*
1037 * reset everything back to the original state. Don't bother
1038 * dealing with errors here since we can't do anything about
1039 * them anyway.
1040 */
1041undo_rename:
1042 CIFSSMBRenameOpenFile(xid, tcon, netfid, dentry->d_name.name,
1043 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1044 CIFS_MOUNT_MAP_SPECIAL_CHR);
1045undo_setattr:
1046 if (dosattr != origattr) {
1047 info_buf->Attributes = cpu_to_le32(origattr);
1048 if (!CIFSSMBSetFileInfo(xid, tcon, info_buf, netfid,
1049 current->tgid))
1050 cifsInode->cifsAttrs = origattr;
1051 }
1052
1053 goto out_close;
a12a1ac7
JL
1054}
1055
ff694527
SF
1056
1057/*
1058 * If dentry->d_inode is null (usually meaning the cached dentry
1059 * is a negative dentry) then we would attempt a standard SMB delete, but
af901ca1
AGR
1060 * if that fails we can not attempt the fall back mechanisms on EACCESS
1061 * but will return the EACCESS to the caller. Note that the VFS does not call
ff694527
SF
1062 * unlink on negative dentries currently.
1063 */
5f0319a7 1064int cifs_unlink(struct inode *dir, struct dentry *dentry)
1da177e4
LT
1065{
1066 int rc = 0;
1067 int xid;
1da177e4 1068 char *full_path = NULL;
5f0319a7 1069 struct inode *inode = dentry->d_inode;
ff694527 1070 struct cifsInodeInfo *cifs_inode;
5f0319a7
JL
1071 struct super_block *sb = dir->i_sb;
1072 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
1073 struct cifsTconInfo *tcon = cifs_sb->tcon;
6050247d
SF
1074 struct iattr *attrs = NULL;
1075 __u32 dosattr = 0, origattr = 0;
1da177e4 1076
b6b38f70 1077 cFYI(1, "cifs_unlink, dir=0x%p, dentry=0x%p", dir, dentry);
1da177e4
LT
1078
1079 xid = GetXid();
1080
5f0319a7
JL
1081 /* Unlink can be called from rename so we can not take the
1082 * sb->s_vfs_rename_mutex here */
1083 full_path = build_path_from_dentry(dentry);
1da177e4 1084 if (full_path == NULL) {
0f3bc09e 1085 rc = -ENOMEM;
1da177e4 1086 FreeXid(xid);
0f3bc09e 1087 return rc;
1da177e4 1088 }
2d785a50 1089
5f0319a7 1090 if ((tcon->ses->capabilities & CAP_UNIX) &&
2d785a50 1091 (CIFS_UNIX_POSIX_PATH_OPS_CAP &
5f0319a7
JL
1092 le64_to_cpu(tcon->fsUnixInfo.Capability))) {
1093 rc = CIFSPOSIXDelFile(xid, tcon, full_path,
2d785a50 1094 SMB_POSIX_UNLINK_FILE_TARGET, cifs_sb->local_nls,
737b758c 1095 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
b6b38f70 1096 cFYI(1, "posix del rc %d", rc);
2d785a50
SF
1097 if ((rc == 0) || (rc == -ENOENT))
1098 goto psx_del_no_retry;
1099 }
1da177e4 1100
6050247d 1101retry_std_delete:
5f0319a7 1102 rc = CIFSSMBDelFile(xid, tcon, full_path, cifs_sb->local_nls,
2d785a50 1103 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
6050247d 1104
2d785a50 1105psx_del_no_retry:
1da177e4 1106 if (!rc) {
5f0319a7
JL
1107 if (inode)
1108 drop_nlink(inode);
1da177e4 1109 } else if (rc == -ENOENT) {
5f0319a7 1110 d_drop(dentry);
1da177e4 1111 } else if (rc == -ETXTBSY) {
3270958b 1112 rc = cifs_rename_pending_delete(full_path, dentry, xid);
a12a1ac7
JL
1113 if (rc == 0)
1114 drop_nlink(inode);
ff694527 1115 } else if ((rc == -EACCES) && (dosattr == 0) && inode) {
388e57b2
SF
1116 attrs = kzalloc(sizeof(*attrs), GFP_KERNEL);
1117 if (attrs == NULL) {
1118 rc = -ENOMEM;
1119 goto out_reval;
1da177e4 1120 }
388e57b2
SF
1121
1122 /* try to reset dos attributes */
ff694527
SF
1123 cifs_inode = CIFS_I(inode);
1124 origattr = cifs_inode->cifsAttrs;
6050247d
SF
1125 if (origattr == 0)
1126 origattr |= ATTR_NORMAL;
1127 dosattr = origattr & ~ATTR_READONLY;
388e57b2
SF
1128 if (dosattr == 0)
1129 dosattr |= ATTR_NORMAL;
1130 dosattr |= ATTR_HIDDEN;
1131
1132 rc = cifs_set_file_info(inode, attrs, xid, full_path, dosattr);
388e57b2
SF
1133 if (rc != 0)
1134 goto out_reval;
6050247d
SF
1135
1136 goto retry_std_delete;
1da177e4 1137 }
6050247d
SF
1138
1139 /* undo the setattr if we errored out and it's needed */
1140 if (rc != 0 && dosattr != 0)
1141 cifs_set_file_info(inode, attrs, xid, full_path, origattr);
1142
388e57b2 1143out_reval:
4523cc30 1144 if (inode) {
ff694527
SF
1145 cifs_inode = CIFS_I(inode);
1146 cifs_inode->time = 0; /* will force revalidate to get info
5f0319a7
JL
1147 when needed */
1148 inode->i_ctime = current_fs_time(sb);
06bcfedd 1149 }
5f0319a7 1150 dir->i_ctime = dir->i_mtime = current_fs_time(sb);
ff694527 1151 cifs_inode = CIFS_I(dir);
6050247d 1152 CIFS_I(dir)->time = 0; /* force revalidate of dir as well */
1da177e4
LT
1153
1154 kfree(full_path);
6050247d 1155 kfree(attrs);
1da177e4
LT
1156 FreeXid(xid);
1157 return rc;
1158}
1159
1160int cifs_mkdir(struct inode *inode, struct dentry *direntry, int mode)
1161{
6b37faa1 1162 int rc = 0, tmprc;
1da177e4
LT
1163 int xid;
1164 struct cifs_sb_info *cifs_sb;
1165 struct cifsTconInfo *pTcon;
1166 char *full_path = NULL;
1167 struct inode *newinode = NULL;
cc0bad75 1168 struct cifs_fattr fattr;
1da177e4 1169
b6b38f70 1170 cFYI(1, "In cifs_mkdir, mode = 0x%x inode = 0x%p", mode, inode);
1da177e4
LT
1171
1172 xid = GetXid();
1173
1174 cifs_sb = CIFS_SB(inode->i_sb);
1175 pTcon = cifs_sb->tcon;
1176
7f57356b 1177 full_path = build_path_from_dentry(direntry);
1da177e4 1178 if (full_path == NULL) {
0f3bc09e 1179 rc = -ENOMEM;
1da177e4 1180 FreeXid(xid);
0f3bc09e 1181 return rc;
1da177e4 1182 }
50c2f753 1183
fb8c4b14
SF
1184 if ((pTcon->ses->capabilities & CAP_UNIX) &&
1185 (CIFS_UNIX_POSIX_PATH_OPS_CAP &
2dd29d31
SF
1186 le64_to_cpu(pTcon->fsUnixInfo.Capability))) {
1187 u32 oplock = 0;
f6d09982 1188 FILE_UNIX_BASIC_INFO *pInfo =
2dd29d31 1189 kzalloc(sizeof(FILE_UNIX_BASIC_INFO), GFP_KERNEL);
fb8c4b14 1190 if (pInfo == NULL) {
2dd29d31
SF
1191 rc = -ENOMEM;
1192 goto mkdir_out;
1193 }
50c2f753 1194
ce3b0f8d 1195 mode &= ~current_umask();
2dd29d31
SF
1196 rc = CIFSPOSIXCreate(xid, pTcon, SMB_O_DIRECTORY | SMB_O_CREAT,
1197 mode, NULL /* netfid */, pInfo, &oplock,
fb8c4b14
SF
1198 full_path, cifs_sb->local_nls,
1199 cifs_sb->mnt_cifs_flags &
2dd29d31 1200 CIFS_MOUNT_MAP_SPECIAL_CHR);
c45d707f
SF
1201 if (rc == -EOPNOTSUPP) {
1202 kfree(pInfo);
1203 goto mkdir_retry_old;
1204 } else if (rc) {
b6b38f70 1205 cFYI(1, "posix mkdir returned 0x%x", rc);
2dd29d31
SF
1206 d_drop(direntry);
1207 } else {
8f2376ad
CG
1208 if (pInfo->Type == cpu_to_le32(-1)) {
1209 /* no return info, go query for it */
5a07cdf8 1210 kfree(pInfo);
fb8c4b14 1211 goto mkdir_get_info;
5a07cdf8 1212 }
fb8c4b14
SF
1213/*BB check (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID ) to see if need
1214 to set uid/gid */
2dd29d31
SF
1215 inc_nlink(inode);
1216 if (pTcon->nocase)
1217 direntry->d_op = &cifs_ci_dentry_ops;
1218 else
1219 direntry->d_op = &cifs_dentry_ops;
cbac3cba 1220
cc0bad75 1221 cifs_unix_basic_to_fattr(&fattr, pInfo, cifs_sb);
4065c802 1222 cifs_fill_uniqueid(inode->i_sb, &fattr);
cc0bad75
JL
1223 newinode = cifs_iget(inode->i_sb, &fattr);
1224 if (!newinode) {
5a07cdf8 1225 kfree(pInfo);
cbac3cba 1226 goto mkdir_get_info;
5a07cdf8 1227 }
6b37faa1 1228
2dd29d31 1229 d_instantiate(direntry, newinode);
cbac3cba 1230
cbac3cba 1231#ifdef CONFIG_CIFS_DEBUG2
b6b38f70
JP
1232 cFYI(1, "instantiated dentry %p %s to inode %p",
1233 direntry, direntry->d_name.name, newinode);
cbac3cba 1234
fb8c4b14 1235 if (newinode->i_nlink != 2)
b6b38f70
JP
1236 cFYI(1, "unexpected number of links %d",
1237 newinode->i_nlink);
cbac3cba 1238#endif
2dd29d31
SF
1239 }
1240 kfree(pInfo);
1241 goto mkdir_out;
fb8c4b14 1242 }
c45d707f 1243mkdir_retry_old:
1da177e4 1244 /* BB add setting the equivalent of mode via CreateX w/ACLs */
737b758c
SF
1245 rc = CIFSSMBMkDir(xid, pTcon, full_path, cifs_sb->local_nls,
1246 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
1da177e4 1247 if (rc) {
b6b38f70 1248 cFYI(1, "cifs_mkdir returned 0x%x", rc);
1da177e4
LT
1249 d_drop(direntry);
1250 } else {
fb8c4b14 1251mkdir_get_info:
d8c76e6f 1252 inc_nlink(inode);
c18c842b 1253 if (pTcon->unix_ext)
1da177e4 1254 rc = cifs_get_inode_info_unix(&newinode, full_path,
fb8c4b14 1255 inode->i_sb, xid);
1da177e4
LT
1256 else
1257 rc = cifs_get_inode_info(&newinode, full_path, NULL,
8b1327f6 1258 inode->i_sb, xid, NULL);
1da177e4 1259
b92327fe
SF
1260 if (pTcon->nocase)
1261 direntry->d_op = &cifs_ci_dentry_ops;
1262 else
1263 direntry->d_op = &cifs_dentry_ops;
1da177e4 1264 d_instantiate(direntry, newinode);
2dd29d31 1265 /* setting nlink not necessary except in cases where we
fb8c4b14 1266 * failed to get it from the server or was set bogus */
2dd29d31 1267 if ((direntry->d_inode) && (direntry->d_inode->i_nlink < 2))
fb8c4b14 1268 direntry->d_inode->i_nlink = 2;
95089910 1269
ce3b0f8d 1270 mode &= ~current_umask();
95089910
JL
1271 /* must turn on setgid bit if parent dir has it */
1272 if (inode->i_mode & S_ISGID)
1273 mode |= S_ISGID;
1274
c18c842b 1275 if (pTcon->unix_ext) {
4e1e7fb9
JL
1276 struct cifs_unix_set_info_args args = {
1277 .mode = mode,
1278 .ctime = NO_CHANGE_64,
1279 .atime = NO_CHANGE_64,
1280 .mtime = NO_CHANGE_64,
1281 .device = 0,
1282 };
d0d2f2df 1283 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID) {
a001e5b5 1284 args.uid = (__u64)current_fsuid();
95089910
JL
1285 if (inode->i_mode & S_ISGID)
1286 args.gid = (__u64)inode->i_gid;
1287 else
a001e5b5 1288 args.gid = (__u64)current_fsgid();
1da177e4 1289 } else {
4e1e7fb9
JL
1290 args.uid = NO_CHANGE_64;
1291 args.gid = NO_CHANGE_64;
1da177e4 1292 }
01ea95e3
JL
1293 CIFSSMBUnixSetPathInfo(xid, pTcon, full_path, &args,
1294 cifs_sb->local_nls,
1295 cifs_sb->mnt_cifs_flags &
1296 CIFS_MOUNT_MAP_SPECIAL_CHR);
3ce53fc4 1297 } else {
67750fb9
JL
1298 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL) &&
1299 (mode & S_IWUGO) == 0) {
1300 FILE_BASIC_INFO pInfo;
6b37faa1
JL
1301 struct cifsInodeInfo *cifsInode;
1302 u32 dosattrs;
1303
67750fb9 1304 memset(&pInfo, 0, sizeof(pInfo));
6b37faa1
JL
1305 cifsInode = CIFS_I(newinode);
1306 dosattrs = cifsInode->cifsAttrs|ATTR_READONLY;
1307 pInfo.Attributes = cpu_to_le32(dosattrs);
1308 tmprc = CIFSSMBSetPathInfo(xid, pTcon,
1309 full_path, &pInfo,
1310 cifs_sb->local_nls,
67750fb9
JL
1311 cifs_sb->mnt_cifs_flags &
1312 CIFS_MOUNT_MAP_SPECIAL_CHR);
6b37faa1
JL
1313 if (tmprc == 0)
1314 cifsInode->cifsAttrs = dosattrs;
67750fb9 1315 }
fb8c4b14 1316 if (direntry->d_inode) {
b0fd30d3
JL
1317 if (cifs_sb->mnt_cifs_flags &
1318 CIFS_MOUNT_DYNPERM)
1319 direntry->d_inode->i_mode =
1320 (mode | S_IFDIR);
4e94a105 1321
fb8c4b14 1322 if (cifs_sb->mnt_cifs_flags &
6473a559 1323 CIFS_MOUNT_SET_UID) {
fb8c4b14 1324 direntry->d_inode->i_uid =
a001e5b5 1325 current_fsuid();
95089910
JL
1326 if (inode->i_mode & S_ISGID)
1327 direntry->d_inode->i_gid =
1328 inode->i_gid;
1329 else
1330 direntry->d_inode->i_gid =
a001e5b5 1331 current_fsgid();
6473a559
SF
1332 }
1333 }
2a138ebb 1334 }
1da177e4 1335 }
fb8c4b14 1336mkdir_out:
1da177e4
LT
1337 kfree(full_path);
1338 FreeXid(xid);
1339 return rc;
1340}
1341
1342int cifs_rmdir(struct inode *inode, struct dentry *direntry)
1343{
1344 int rc = 0;
1345 int xid;
1346 struct cifs_sb_info *cifs_sb;
1347 struct cifsTconInfo *pTcon;
1348 char *full_path = NULL;
1349 struct cifsInodeInfo *cifsInode;
1350
b6b38f70 1351 cFYI(1, "cifs_rmdir, inode = 0x%p", inode);
1da177e4
LT
1352
1353 xid = GetXid();
1354
1355 cifs_sb = CIFS_SB(inode->i_sb);
1356 pTcon = cifs_sb->tcon;
1357
7f57356b 1358 full_path = build_path_from_dentry(direntry);
1da177e4 1359 if (full_path == NULL) {
0f3bc09e 1360 rc = -ENOMEM;
1da177e4 1361 FreeXid(xid);
0f3bc09e 1362 return rc;
1da177e4
LT
1363 }
1364
737b758c
SF
1365 rc = CIFSSMBRmDir(xid, pTcon, full_path, cifs_sb->local_nls,
1366 cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MAP_SPECIAL_CHR);
1da177e4
LT
1367
1368 if (!rc) {
9a53c3a7 1369 drop_nlink(inode);
3677db10 1370 spin_lock(&direntry->d_inode->i_lock);
fb8c4b14 1371 i_size_write(direntry->d_inode, 0);
ce71ec36 1372 clear_nlink(direntry->d_inode);
3677db10 1373 spin_unlock(&direntry->d_inode->i_lock);
1da177e4
LT
1374 }
1375
1376 cifsInode = CIFS_I(direntry->d_inode);
1377 cifsInode->time = 0; /* force revalidate to go get info when
1378 needed */
42c24544
SF
1379
1380 cifsInode = CIFS_I(inode);
1381 cifsInode->time = 0; /* force revalidate to get parent dir info
1382 since cached search results now invalid */
1383
1da177e4
LT
1384 direntry->d_inode->i_ctime = inode->i_ctime = inode->i_mtime =
1385 current_fs_time(inode->i_sb);
1386
1387 kfree(full_path);
1388 FreeXid(xid);
1389 return rc;
1390}
1391
ee2fd967
SF
1392static int
1393cifs_do_rename(int xid, struct dentry *from_dentry, const char *fromPath,
1394 struct dentry *to_dentry, const char *toPath)
1395{
1396 struct cifs_sb_info *cifs_sb = CIFS_SB(from_dentry->d_sb);
1397 struct cifsTconInfo *pTcon = cifs_sb->tcon;
1398 __u16 srcfid;
1399 int oplock, rc;
1400
1401 /* try path-based rename first */
1402 rc = CIFSSMBRename(xid, pTcon, fromPath, toPath, cifs_sb->local_nls,
1403 cifs_sb->mnt_cifs_flags &
1404 CIFS_MOUNT_MAP_SPECIAL_CHR);
1405
1406 /*
1407 * don't bother with rename by filehandle unless file is busy and
1408 * source Note that cross directory moves do not work with
1409 * rename by filehandle to various Windows servers.
1410 */
1411 if (rc == 0 || rc != -ETXTBSY)
1412 return rc;
1413
ed0e3ace
JL
1414 /* open-file renames don't work across directories */
1415 if (to_dentry->d_parent != from_dentry->d_parent)
1416 return rc;
1417
ee2fd967
SF
1418 /* open the file to be renamed -- we need DELETE perms */
1419 rc = CIFSSMBOpen(xid, pTcon, fromPath, FILE_OPEN, DELETE,
1420 CREATE_NOT_DIR, &srcfid, &oplock, NULL,
1421 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1422 CIFS_MOUNT_MAP_SPECIAL_CHR);
1423
1424 if (rc == 0) {
1425 rc = CIFSSMBRenameOpenFile(xid, pTcon, srcfid,
1426 (const char *) to_dentry->d_name.name,
1427 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
1428 CIFS_MOUNT_MAP_SPECIAL_CHR);
1429
1430 CIFSSMBClose(xid, pTcon, srcfid);
1431 }
1432
1433 return rc;
1434}
1435
14121bdc
JL
1436int cifs_rename(struct inode *source_dir, struct dentry *source_dentry,
1437 struct inode *target_dir, struct dentry *target_dentry)
1da177e4 1438{
ee2fd967
SF
1439 char *fromName = NULL;
1440 char *toName = NULL;
1da177e4
LT
1441 struct cifs_sb_info *cifs_sb_source;
1442 struct cifs_sb_info *cifs_sb_target;
14121bdc 1443 struct cifsTconInfo *tcon;
ee2fd967
SF
1444 FILE_UNIX_BASIC_INFO *info_buf_source = NULL;
1445 FILE_UNIX_BASIC_INFO *info_buf_target;
8d281efb 1446 int xid, rc, tmprc;
1da177e4 1447
14121bdc
JL
1448 cifs_sb_target = CIFS_SB(target_dir->i_sb);
1449 cifs_sb_source = CIFS_SB(source_dir->i_sb);
1450 tcon = cifs_sb_source->tcon;
1da177e4 1451
ee2fd967
SF
1452 xid = GetXid();
1453
1454 /*
1455 * BB: this might be allowed if same server, but different share.
1456 * Consider adding support for this
1457 */
14121bdc 1458 if (tcon != cifs_sb_target->tcon) {
ee2fd967
SF
1459 rc = -EXDEV;
1460 goto cifs_rename_exit;
1da177e4
LT
1461 }
1462
ee2fd967
SF
1463 /*
1464 * we already have the rename sem so we do not need to
1465 * grab it again here to protect the path integrity
1466 */
14121bdc 1467 fromName = build_path_from_dentry(source_dentry);
ee2fd967
SF
1468 if (fromName == NULL) {
1469 rc = -ENOMEM;
1470 goto cifs_rename_exit;
1471 }
1472
14121bdc 1473 toName = build_path_from_dentry(target_dentry);
ee2fd967 1474 if (toName == NULL) {
1da177e4
LT
1475 rc = -ENOMEM;
1476 goto cifs_rename_exit;
1477 }
1478
14121bdc
JL
1479 rc = cifs_do_rename(xid, source_dentry, fromName,
1480 target_dentry, toName);
ee2fd967 1481
14121bdc
JL
1482 if (rc == -EEXIST && tcon->unix_ext) {
1483 /*
1484 * Are src and dst hardlinks of same inode? We can
1485 * only tell with unix extensions enabled
1486 */
1487 info_buf_source =
1488 kmalloc(2 * sizeof(FILE_UNIX_BASIC_INFO),
1489 GFP_KERNEL);
1490 if (info_buf_source == NULL) {
1491 rc = -ENOMEM;
1492 goto cifs_rename_exit;
1493 }
1494
1495 info_buf_target = info_buf_source + 1;
8d281efb 1496 tmprc = CIFSSMBUnixQPathInfo(xid, tcon, fromName,
14121bdc
JL
1497 info_buf_source,
1498 cifs_sb_source->local_nls,
1499 cifs_sb_source->mnt_cifs_flags &
1500 CIFS_MOUNT_MAP_SPECIAL_CHR);
8d281efb 1501 if (tmprc != 0)
14121bdc 1502 goto unlink_target;
ee2fd967 1503
8d281efb 1504 tmprc = CIFSSMBUnixQPathInfo(xid, tcon,
14121bdc
JL
1505 toName, info_buf_target,
1506 cifs_sb_target->local_nls,
1507 /* remap based on source sb */
1508 cifs_sb_source->mnt_cifs_flags &
1509 CIFS_MOUNT_MAP_SPECIAL_CHR);
1510
8d281efb 1511 if (tmprc == 0 && (info_buf_source->UniqueId ==
ae6884a9 1512 info_buf_target->UniqueId)) {
14121bdc 1513 /* same file, POSIX says that this is a noop */
ae6884a9 1514 rc = 0;
14121bdc 1515 goto cifs_rename_exit;
ae6884a9 1516 }
14121bdc 1517 } /* else ... BB we could add the same check for Windows by
ee2fd967 1518 checking the UniqueId via FILE_INTERNAL_INFO */
14121bdc 1519
ee2fd967 1520unlink_target:
fc6f3943
JL
1521 /* Try unlinking the target dentry if it's not negative */
1522 if (target_dentry->d_inode && (rc == -EACCES || rc == -EEXIST)) {
8d281efb 1523 tmprc = cifs_unlink(target_dir, target_dentry);
14121bdc
JL
1524 if (tmprc)
1525 goto cifs_rename_exit;
1526
14121bdc
JL
1527 rc = cifs_do_rename(xid, source_dentry, fromName,
1528 target_dentry, toName);
1da177e4
LT
1529 }
1530
1531cifs_rename_exit:
ee2fd967 1532 kfree(info_buf_source);
1da177e4
LT
1533 kfree(fromName);
1534 kfree(toName);
1535 FreeXid(xid);
1536 return rc;
1537}
1538
df2cf170
JL
1539static bool
1540cifs_inode_needs_reval(struct inode *inode)
1da177e4 1541{
df2cf170 1542 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
1da177e4 1543
df2cf170
JL
1544 if (cifs_i->clientCanCacheRead)
1545 return false;
1da177e4 1546
df2cf170
JL
1547 if (!lookupCacheEnabled)
1548 return true;
1da177e4 1549
df2cf170
JL
1550 if (cifs_i->time == 0)
1551 return true;
1da177e4 1552
df2cf170
JL
1553 /* FIXME: the actimeo should be tunable */
1554 if (time_after_eq(jiffies, cifs_i->time + HZ))
1555 return true;
1556
db19272e
JL
1557 /* hardlinked files w/ noserverino get "special" treatment */
1558 if (!(CIFS_SB(inode->i_sb)->mnt_cifs_flags & CIFS_MOUNT_SERVER_INUM) &&
1559 S_ISREG(inode->i_mode) && inode->i_nlink != 1)
1560 return true;
1561
df2cf170
JL
1562 return false;
1563}
1564
1565/* check invalid_mapping flag and zap the cache if it's set */
1566static void
1567cifs_invalidate_mapping(struct inode *inode)
1568{
1569 int rc;
1570 struct cifsInodeInfo *cifs_i = CIFS_I(inode);
1571
1572 cifs_i->invalid_mapping = false;
1573
1574 /* write back any cached data */
1575 if (inode->i_mapping && inode->i_mapping->nrpages != 0) {
1576 rc = filemap_write_and_wait(inode->i_mapping);
1577 if (rc)
1578 cifs_i->write_behind_rc = rc;
1579 }
1580 invalidate_remote_inode(inode);
9451a9a5 1581 cifs_fscache_reset_inode_cookie(inode);
df2cf170
JL
1582}
1583
abab095d
JL
1584int cifs_revalidate_file(struct file *filp)
1585{
1586 int rc = 0;
1587 struct inode *inode = filp->f_path.dentry->d_inode;
1588
1589 if (!cifs_inode_needs_reval(inode))
1590 goto check_inval;
1591
1592 if (CIFS_SB(inode->i_sb)->tcon->unix_ext)
1593 rc = cifs_get_file_info_unix(filp);
1594 else
1595 rc = cifs_get_file_info(filp);
1596
1597check_inval:
1598 if (CIFS_I(inode)->invalid_mapping)
1599 cifs_invalidate_mapping(inode);
1600
1601 return rc;
1602}
1603
df2cf170
JL
1604/* revalidate a dentry's inode attributes */
1605int cifs_revalidate_dentry(struct dentry *dentry)
1606{
1607 int xid;
1608 int rc = 0;
1609 char *full_path = NULL;
1610 struct inode *inode = dentry->d_inode;
1611 struct super_block *sb = dentry->d_sb;
1612
1613 if (inode == NULL)
1614 return -ENOENT;
1da177e4
LT
1615
1616 xid = GetXid();
1617
df2cf170
JL
1618 if (!cifs_inode_needs_reval(inode))
1619 goto check_inval;
1da177e4
LT
1620
1621 /* can not safely grab the rename sem here if rename calls revalidate
1622 since that would deadlock */
df2cf170 1623 full_path = build_path_from_dentry(dentry);
1da177e4 1624 if (full_path == NULL) {
0f3bc09e 1625 rc = -ENOMEM;
df2cf170 1626 goto check_inval;
1da177e4
LT
1627 }
1628
f19159dc 1629 cFYI(1, "Revalidate: %s inode 0x%p count %d dentry: 0x%p d_time %ld "
df2cf170 1630 "jiffies %ld", full_path, inode, inode->i_count.counter,
f19159dc 1631 dentry, dentry->d_time, jiffies);
1da177e4 1632
df2cf170
JL
1633 if (CIFS_SB(sb)->tcon->unix_ext)
1634 rc = cifs_get_inode_info_unix(&inode, full_path, sb, xid);
1635 else
1636 rc = cifs_get_inode_info(&inode, full_path, NULL, sb,
1637 xid, NULL);
1da177e4 1638
df2cf170
JL
1639check_inval:
1640 if (CIFS_I(inode)->invalid_mapping)
1641 cifs_invalidate_mapping(inode);
50c2f753 1642
1da177e4
LT
1643 kfree(full_path);
1644 FreeXid(xid);
1645 return rc;
1646}
1647
1648int cifs_getattr(struct vfsmount *mnt, struct dentry *dentry,
1649 struct kstat *stat)
1650{
df2cf170 1651 int err = cifs_revalidate_dentry(dentry);
5fe14c85 1652 if (!err) {
1da177e4 1653 generic_fillattr(dentry->d_inode, stat);
5fe14c85 1654 stat->blksize = CIFS_MAX_MSGSIZE;
cc0bad75 1655 stat->ino = CIFS_I(dentry->d_inode)->uniqueid;
5fe14c85 1656 }
1da177e4
LT
1657 return err;
1658}
1659
1660static int cifs_truncate_page(struct address_space *mapping, loff_t from)
1661{
1662 pgoff_t index = from >> PAGE_CACHE_SHIFT;
1663 unsigned offset = from & (PAGE_CACHE_SIZE - 1);
1664 struct page *page;
1da177e4
LT
1665 int rc = 0;
1666
1667 page = grab_cache_page(mapping, index);
1668 if (!page)
1669 return -ENOMEM;
1670
eebd2aa3 1671 zero_user_segment(page, offset, PAGE_CACHE_SIZE);
1da177e4
LT
1672 unlock_page(page);
1673 page_cache_release(page);
1674 return rc;
1675}
1676
fb8c4b14 1677static int cifs_vmtruncate(struct inode *inode, loff_t offset)
3677db10 1678{
c08d3b0e 1679 loff_t oldsize;
1680 int err;
3677db10 1681
ba6a46a0 1682 spin_lock(&inode->i_lock);
c08d3b0e 1683 err = inode_newsize_ok(inode, offset);
1684 if (err) {
ba6a46a0 1685 spin_unlock(&inode->i_lock);
c08d3b0e 1686 goto out;
ba6a46a0 1687 }
c08d3b0e 1688
1689 oldsize = inode->i_size;
3677db10 1690 i_size_write(inode, offset);
ba6a46a0 1691 spin_unlock(&inode->i_lock);
c08d3b0e 1692 truncate_pagecache(inode, oldsize, offset);
acfa4380 1693 if (inode->i_op->truncate)
3677db10 1694 inode->i_op->truncate(inode);
c08d3b0e 1695out:
1696 return err;
3677db10
SF
1697}
1698
8efdbde6
JL
1699static int
1700cifs_set_file_size(struct inode *inode, struct iattr *attrs,
1701 int xid, char *full_path)
1702{
1703 int rc;
1704 struct cifsFileInfo *open_file;
1705 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
1706 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
1707 struct cifsTconInfo *pTcon = cifs_sb->tcon;
1708
1709 /*
1710 * To avoid spurious oplock breaks from server, in the case of
1711 * inodes that we already have open, avoid doing path based
1712 * setting of file size if we can do it by handle.
1713 * This keeps our caching token (oplock) and avoids timeouts
1714 * when the local oplock break takes longer to flush
1715 * writebehind data than the SMB timeout for the SetPathInfo
1716 * request would allow
1717 */
1718 open_file = find_writable_file(cifsInode);
1719 if (open_file) {
1720 __u16 nfid = open_file->netfid;
1721 __u32 npid = open_file->pid;
1722 rc = CIFSSMBSetFileSize(xid, pTcon, attrs->ia_size, nfid,
1723 npid, false);
6ab409b5 1724 cifsFileInfo_put(open_file);
b6b38f70 1725 cFYI(1, "SetFSize for attrs rc = %d", rc);
8efdbde6
JL
1726 if ((rc == -EINVAL) || (rc == -EOPNOTSUPP)) {
1727 unsigned int bytes_written;
1728 rc = CIFSSMBWrite(xid, pTcon, nfid, 0, attrs->ia_size,
1729 &bytes_written, NULL, NULL, 1);
b6b38f70 1730 cFYI(1, "Wrt seteof rc %d", rc);
8efdbde6
JL
1731 }
1732 } else
1733 rc = -EINVAL;
1734
1735 if (rc != 0) {
1736 /* Set file size by pathname rather than by handle
1737 either because no valid, writeable file handle for
1738 it was found or because there was an error setting
1739 it by handle */
1740 rc = CIFSSMBSetEOF(xid, pTcon, full_path, attrs->ia_size,
1741 false, cifs_sb->local_nls,
1742 cifs_sb->mnt_cifs_flags &
1743 CIFS_MOUNT_MAP_SPECIAL_CHR);
b6b38f70 1744 cFYI(1, "SetEOF by path (setattrs) rc = %d", rc);
8efdbde6
JL
1745 if ((rc == -EINVAL) || (rc == -EOPNOTSUPP)) {
1746 __u16 netfid;
1747 int oplock = 0;
1748
1749 rc = SMBLegacyOpen(xid, pTcon, full_path,
1750 FILE_OPEN, GENERIC_WRITE,
1751 CREATE_NOT_DIR, &netfid, &oplock, NULL,
1752 cifs_sb->local_nls,
1753 cifs_sb->mnt_cifs_flags &
1754 CIFS_MOUNT_MAP_SPECIAL_CHR);
1755 if (rc == 0) {
1756 unsigned int bytes_written;
1757 rc = CIFSSMBWrite(xid, pTcon, netfid, 0,
1758 attrs->ia_size,
1759 &bytes_written, NULL,
1760 NULL, 1);
b6b38f70 1761 cFYI(1, "wrt seteof rc %d", rc);
8efdbde6
JL
1762 CIFSSMBClose(xid, pTcon, netfid);
1763 }
1764 }
1765 }
1766
1767 if (rc == 0) {
fbec9ab9 1768 cifsInode->server_eof = attrs->ia_size;
8efdbde6
JL
1769 rc = cifs_vmtruncate(inode, attrs->ia_size);
1770 cifs_truncate_page(inode->i_mapping, inode->i_size);
1771 }
1772
1773 return rc;
1774}
1775
3fe5c1dd
JL
1776static int
1777cifs_setattr_unix(struct dentry *direntry, struct iattr *attrs)
1778{
1779 int rc;
1780 int xid;
1781 char *full_path = NULL;
1782 struct inode *inode = direntry->d_inode;
1783 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
1784 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
1785 struct cifsTconInfo *pTcon = cifs_sb->tcon;
1786 struct cifs_unix_set_info_args *args = NULL;
3bbeeb3c 1787 struct cifsFileInfo *open_file;
3fe5c1dd 1788
b6b38f70
JP
1789 cFYI(1, "setattr_unix on file %s attrs->ia_valid=0x%x",
1790 direntry->d_name.name, attrs->ia_valid);
3fe5c1dd
JL
1791
1792 xid = GetXid();
1793
1794 if ((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_PERM) == 0) {
1795 /* check if we have permission to change attrs */
1796 rc = inode_change_ok(inode, attrs);
1797 if (rc < 0)
1798 goto out;
1799 else
1800 rc = 0;
1801 }
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
ccd4bb1b 1887 if (!rc) {
3fe5c1dd 1888 rc = inode_setattr(inode, attrs);
ccd4bb1b
SF
1889
1890 /* force revalidate when any of these times are set since some
1891 of the fs types (eg ext3, fat) do not have fine enough
1892 time granularity to match protocol, and we do not have a
1893 a way (yet) to query the server fs's time granularity (and
1894 whether it rounds times down).
1895 */
1896 if (!rc && (attrs->ia_valid & (ATTR_MTIME | ATTR_CTIME)))
1897 cifsInode->time = 0;
1898 }
3fe5c1dd
JL
1899out:
1900 kfree(args);
1901 kfree(full_path);
1902 FreeXid(xid);
1903 return rc;
1904}
1905
0510eeb7
JL
1906static int
1907cifs_setattr_nounix(struct dentry *direntry, struct iattr *attrs)
1da177e4
LT
1908{
1909 int xid;
3fe5c1dd
JL
1910 struct inode *inode = direntry->d_inode;
1911 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
3fe5c1dd 1912 struct cifsInodeInfo *cifsInode = CIFS_I(inode);
1da177e4
LT
1913 char *full_path = NULL;
1914 int rc = -EACCES;
feb3e20c 1915 __u32 dosattr = 0;
4e1e7fb9 1916 __u64 mode = NO_CHANGE_64;
3fe5c1dd 1917
1da177e4
LT
1918 xid = GetXid();
1919
b6b38f70
JP
1920 cFYI(1, "setattr on file %s attrs->iavalid 0x%x",
1921 direntry->d_name.name, attrs->ia_valid);
6473a559 1922
2a138ebb 1923 if ((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NO_PERM) == 0) {
6473a559 1924 /* check if we have permission to change attrs */
02eadeff 1925 rc = inode_change_ok(inode, attrs);
fb8c4b14 1926 if (rc < 0) {
6473a559
SF
1927 FreeXid(xid);
1928 return rc;
1929 } else
1930 rc = 0;
1931 }
50c2f753 1932
7f57356b 1933 full_path = build_path_from_dentry(direntry);
1da177e4 1934 if (full_path == NULL) {
0f3bc09e 1935 rc = -ENOMEM;
1da177e4 1936 FreeXid(xid);
0f3bc09e 1937 return rc;
1da177e4 1938 }
1da177e4 1939
0f4d634c
JL
1940 /*
1941 * Attempt to flush data before changing attributes. We need to do
1942 * this for ATTR_SIZE and ATTR_MTIME for sure, and if we change the
1943 * ownership or mode then we may also need to do this. Here, we take
1944 * the safe way out and just do the flush on all setattr requests. If
1945 * the flush returns error, store it to report later and continue.
1946 *
1947 * BB: This should be smarter. Why bother flushing pages that
1948 * will be truncated anyway? Also, should we error out here if
1949 * the flush returns error?
1950 */
1951 rc = filemap_write_and_wait(inode->i_mapping);
1952 if (rc != 0) {
1953 cifsInode->write_behind_rc = rc;
1954 rc = 0;
50531444 1955 }
cea21805 1956
50531444 1957 if (attrs->ia_valid & ATTR_SIZE) {
8efdbde6
JL
1958 rc = cifs_set_file_size(inode, attrs, xid, full_path);
1959 if (rc != 0)
e30dcf3a 1960 goto cifs_setattr_exit;
1da177e4 1961 }
4ca691a8
JL
1962
1963 /*
1964 * Without unix extensions we can't send ownership changes to the
1965 * server, so silently ignore them. This is consistent with how
1966 * local DOS/Windows filesystems behave (VFAT, NTFS, etc). With
1967 * CIFSACL support + proper Windows to Unix idmapping, we may be
1968 * able to support this in the future.
1969 */
3fe5c1dd 1970 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_SET_UID))
4ca691a8 1971 attrs->ia_valid &= ~(ATTR_UID | ATTR_GID);
1da177e4 1972
d32c4f26
JL
1973 /* skip mode change if it's just for clearing setuid/setgid */
1974 if (attrs->ia_valid & (ATTR_KILL_SUID|ATTR_KILL_SGID))
1975 attrs->ia_valid &= ~ATTR_MODE;
1976
1da177e4 1977 if (attrs->ia_valid & ATTR_MODE) {
b6b38f70 1978 cFYI(1, "Mode changed to 0%o", attrs->ia_mode);
1da177e4 1979 mode = attrs->ia_mode;
1da177e4
LT
1980 }
1981
3fe5c1dd 1982 if (attrs->ia_valid & ATTR_MODE) {
cdbce9c8 1983 rc = 0;
97837582
SF
1984#ifdef CONFIG_CIFS_EXPERIMENTAL
1985 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_CIFS_ACL)
02eadeff 1986 rc = mode_to_acl(inode, full_path, mode);
5132861a 1987 else
97837582 1988#endif
5132861a
JL
1989 if (((mode & S_IWUGO) == 0) &&
1990 (cifsInode->cifsAttrs & ATTR_READONLY) == 0) {
feb3e20c
JL
1991
1992 dosattr = cifsInode->cifsAttrs | ATTR_READONLY;
1993
5132861a
JL
1994 /* fix up mode if we're not using dynperm */
1995 if ((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM) == 0)
1996 attrs->ia_mode = inode->i_mode & ~S_IWUGO;
1997 } else if ((mode & S_IWUGO) &&
1998 (cifsInode->cifsAttrs & ATTR_READONLY)) {
feb3e20c
JL
1999
2000 dosattr = cifsInode->cifsAttrs & ~ATTR_READONLY;
2001 /* Attributes of 0 are ignored */
2002 if (dosattr == 0)
2003 dosattr |= ATTR_NORMAL;
5132861a
JL
2004
2005 /* reset local inode permissions to normal */
2006 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM)) {
2007 attrs->ia_mode &= ~(S_IALLUGO);
2008 if (S_ISDIR(inode->i_mode))
2009 attrs->ia_mode |=
2010 cifs_sb->mnt_dir_mode;
2011 else
2012 attrs->ia_mode |=
2013 cifs_sb->mnt_file_mode;
2014 }
2015 } else if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DYNPERM)) {
2016 /* ignore mode change - ATTR_READONLY hasn't changed */
2017 attrs->ia_valid &= ~ATTR_MODE;
1da177e4 2018 }
1da177e4
LT
2019 }
2020
feb3e20c
JL
2021 if (attrs->ia_valid & (ATTR_MTIME|ATTR_ATIME|ATTR_CTIME) ||
2022 ((attrs->ia_valid & ATTR_MODE) && dosattr)) {
2023 rc = cifs_set_file_info(inode, attrs, xid, full_path, dosattr);
2024 /* BB: check for rc = -EOPNOTSUPP and switch to legacy mode */
1da177e4 2025
e30dcf3a
SF
2026 /* Even if error on time set, no sense failing the call if
2027 the server would set the time to a reasonable value anyway,
2028 and this check ensures that we are not being called from
2029 sys_utimes in which case we ought to fail the call back to
2030 the user when the server rejects the call */
fb8c4b14 2031 if ((rc) && (attrs->ia_valid &
feb3e20c 2032 (ATTR_MODE | ATTR_GID | ATTR_UID | ATTR_SIZE)))
e30dcf3a 2033 rc = 0;
1da177e4
LT
2034 }
2035
2036 /* do not need local check to inode_check_ok since the server does
2037 that */
2038 if (!rc)
02eadeff 2039 rc = inode_setattr(inode, attrs);
e30dcf3a 2040cifs_setattr_exit:
1da177e4
LT
2041 kfree(full_path);
2042 FreeXid(xid);
2043 return rc;
2044}
2045
0510eeb7
JL
2046int
2047cifs_setattr(struct dentry *direntry, struct iattr *attrs)
2048{
2049 struct inode *inode = direntry->d_inode;
2050 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
2051 struct cifsTconInfo *pTcon = cifs_sb->tcon;
2052
2053 if (pTcon->unix_ext)
2054 return cifs_setattr_unix(direntry, attrs);
2055
2056 return cifs_setattr_nounix(direntry, attrs);
2057
2058 /* BB: add cifs_setattr_legacy for really old servers */
2059}
2060
99ee4dbd 2061#if 0
1da177e4
LT
2062void cifs_delete_inode(struct inode *inode)
2063{
b6b38f70 2064 cFYI(1, "In cifs_delete_inode, inode = 0x%p", inode);
1da177e4
LT
2065 /* may have to add back in if and when safe distributed caching of
2066 directories added e.g. via FindNotify */
2067}
99ee4dbd 2068#endif