blob: 06c3e83fa387fecf63b93e33a0455dac6da45f6d [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * fs/cifs/file.c
3 *
4 * vfs operations that deal with files
Steve Frenchfb8c4b12007-07-10 01:16:18 +00005 *
Steve Frenchf19159d2010-04-21 04:12:10 +00006 * Copyright (C) International Business Machines Corp., 2002,2010
Linus Torvalds1da177e2005-04-16 15:20:36 -07007 * Author(s): Steve French (sfrench@us.ibm.com)
Jeremy Allison7ee1af72006-08-02 21:56:33 +00008 * Jeremy Allison (jra@samba.org)
Linus Torvalds1da177e2005-04-16 15:20:36 -07009 *
10 * This library is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU Lesser General Public License as published
12 * by the Free Software Foundation; either version 2.1 of the License, or
13 * (at your option) any later version.
14 *
15 * This library is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
18 * the GNU Lesser General Public License for more details.
19 *
20 * You should have received a copy of the GNU Lesser General Public License
21 * along with this library; if not, write to the Free Software
22 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
23 */
24#include <linux/fs.h>
Steve French37c0eb42005-10-05 14:50:29 -070025#include <linux/backing-dev.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070026#include <linux/stat.h>
27#include <linux/fcntl.h>
28#include <linux/pagemap.h>
29#include <linux/pagevec.h>
Steve French37c0eb42005-10-05 14:50:29 -070030#include <linux/writeback.h>
Andrew Morton6f88cc22006-12-10 02:19:44 -080031#include <linux/task_io_accounting_ops.h>
Steve French23e7dd72005-10-20 13:44:56 -070032#include <linux/delay.h>
Jeff Layton3bc303c2009-09-21 06:47:50 -040033#include <linux/mount.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090034#include <linux/slab.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070035#include <asm/div64.h>
36#include "cifsfs.h"
37#include "cifspdu.h"
38#include "cifsglob.h"
39#include "cifsproto.h"
40#include "cifs_unicode.h"
41#include "cifs_debug.h"
42#include "cifs_fs_sb.h"
Suresh Jayaraman9451a9a2010-07-05 18:12:45 +053043#include "fscache.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070044
Linus Torvalds1da177e2005-04-16 15:20:36 -070045static inline int cifs_convert_flags(unsigned int flags)
46{
47 if ((flags & O_ACCMODE) == O_RDONLY)
48 return GENERIC_READ;
49 else if ((flags & O_ACCMODE) == O_WRONLY)
50 return GENERIC_WRITE;
51 else if ((flags & O_ACCMODE) == O_RDWR) {
52 /* GENERIC_ALL is too much permission to request
53 can cause unnecessary access denied on create */
54 /* return GENERIC_ALL; */
55 return (GENERIC_READ | GENERIC_WRITE);
56 }
57
Jeff Laytone10f7b52008-05-14 10:21:33 -070058 return (READ_CONTROL | FILE_WRITE_ATTRIBUTES | FILE_READ_ATTRIBUTES |
59 FILE_WRITE_EA | FILE_APPEND_DATA | FILE_WRITE_DATA |
60 FILE_READ_DATA);
Steve French7fc8f4e2009-02-23 20:43:11 +000061}
Jeff Laytone10f7b52008-05-14 10:21:33 -070062
Jeff Layton608712f2010-10-15 15:33:56 -040063static u32 cifs_posix_convert_flags(unsigned int flags)
Steve French7fc8f4e2009-02-23 20:43:11 +000064{
Jeff Layton608712f2010-10-15 15:33:56 -040065 u32 posix_flags = 0;
Jeff Laytone10f7b52008-05-14 10:21:33 -070066
Steve French7fc8f4e2009-02-23 20:43:11 +000067 if ((flags & O_ACCMODE) == O_RDONLY)
Jeff Layton608712f2010-10-15 15:33:56 -040068 posix_flags = SMB_O_RDONLY;
Steve French7fc8f4e2009-02-23 20:43:11 +000069 else if ((flags & O_ACCMODE) == O_WRONLY)
Jeff Layton608712f2010-10-15 15:33:56 -040070 posix_flags = SMB_O_WRONLY;
71 else if ((flags & O_ACCMODE) == O_RDWR)
72 posix_flags = SMB_O_RDWR;
73
74 if (flags & O_CREAT)
75 posix_flags |= SMB_O_CREAT;
76 if (flags & O_EXCL)
77 posix_flags |= SMB_O_EXCL;
78 if (flags & O_TRUNC)
79 posix_flags |= SMB_O_TRUNC;
80 /* be safe and imply O_SYNC for O_DSYNC */
Christoph Hellwig6b2f3d12009-10-27 11:05:28 +010081 if (flags & O_DSYNC)
Jeff Layton608712f2010-10-15 15:33:56 -040082 posix_flags |= SMB_O_SYNC;
Steve French7fc8f4e2009-02-23 20:43:11 +000083 if (flags & O_DIRECTORY)
Jeff Layton608712f2010-10-15 15:33:56 -040084 posix_flags |= SMB_O_DIRECTORY;
Steve French7fc8f4e2009-02-23 20:43:11 +000085 if (flags & O_NOFOLLOW)
Jeff Layton608712f2010-10-15 15:33:56 -040086 posix_flags |= SMB_O_NOFOLLOW;
Steve French7fc8f4e2009-02-23 20:43:11 +000087 if (flags & O_DIRECT)
Jeff Layton608712f2010-10-15 15:33:56 -040088 posix_flags |= SMB_O_DIRECT;
Steve French7fc8f4e2009-02-23 20:43:11 +000089
90 return posix_flags;
Linus Torvalds1da177e2005-04-16 15:20:36 -070091}
92
93static inline int cifs_get_disposition(unsigned int flags)
94{
95 if ((flags & (O_CREAT | O_EXCL)) == (O_CREAT | O_EXCL))
96 return FILE_CREATE;
97 else if ((flags & (O_CREAT | O_TRUNC)) == (O_CREAT | O_TRUNC))
98 return FILE_OVERWRITE_IF;
99 else if ((flags & O_CREAT) == O_CREAT)
100 return FILE_OPEN_IF;
Steve French55aa2e02006-05-30 18:09:31 +0000101 else if ((flags & O_TRUNC) == O_TRUNC)
102 return FILE_OVERWRITE;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700103 else
104 return FILE_OPEN;
105}
106
Jeff Laytondb460242010-06-16 13:40:17 -0400107static inline int cifs_open_inode_helper(struct inode *inode,
Suresh Jayaramana347ecb2010-09-17 19:43:10 +0530108 struct cifsTconInfo *pTcon, __u32 oplock, FILE_ALL_INFO *buf,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700109 char *full_path, int xid)
110{
Jeff Laytondb460242010-06-16 13:40:17 -0400111 struct cifsInodeInfo *pCifsInode = CIFS_I(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700112 struct timespec temp;
113 int rc;
114
Linus Torvalds1da177e2005-04-16 15:20:36 -0700115 if (pCifsInode->clientCanCacheRead) {
116 /* we have the inode open somewhere else
117 no need to discard cache data */
118 goto client_can_cache;
119 }
120
121 /* BB need same check in cifs_create too? */
122 /* if not oplocked, invalidate inode pages if mtime or file
123 size changed */
Jeff Layton07119a42009-05-27 09:37:33 -0400124 temp = cifs_NTtimeToUnix(buf->LastWriteTime);
Jeff Laytondb460242010-06-16 13:40:17 -0400125 if (timespec_equal(&inode->i_mtime, &temp) &&
126 (inode->i_size ==
Linus Torvalds1da177e2005-04-16 15:20:36 -0700127 (loff_t)le64_to_cpu(buf->EndOfFile))) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000128 cFYI(1, "inode unchanged on server");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700129 } else {
Jeff Laytondb460242010-06-16 13:40:17 -0400130 if (inode->i_mapping) {
Steve Frenchff2157132010-03-09 20:30:42 +0000131 /* BB no need to lock inode until after invalidate
132 since namei code should already have it locked? */
Jeff Laytondb460242010-06-16 13:40:17 -0400133 rc = filemap_write_and_wait(inode->i_mapping);
Jeff Laytoneb4b7562010-10-22 14:52:29 -0400134 mapping_set_error(inode->i_mapping, rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700135 }
Joe Perchesb6b38f72010-04-21 03:50:45 +0000136 cFYI(1, "invalidating remote inode since open detected it "
137 "changed");
Jeff Laytondb460242010-06-16 13:40:17 -0400138 invalidate_remote_inode(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700139 }
140
141client_can_cache:
Steve Frenchc18c8422007-07-18 23:21:09 +0000142 if (pTcon->unix_ext)
Jeff Laytondb460242010-06-16 13:40:17 -0400143 rc = cifs_get_inode_info_unix(&inode, full_path, inode->i_sb,
144 xid);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700145 else
Jeff Laytondb460242010-06-16 13:40:17 -0400146 rc = cifs_get_inode_info(&inode, full_path, buf, inode->i_sb,
147 xid, NULL);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700148
Pavel Shilovskyc6723622010-11-03 10:58:57 +0300149 cifs_set_oplock_level(pCifsInode, oplock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700150
151 return rc;
152}
153
Jeff Layton608712f2010-10-15 15:33:56 -0400154int cifs_posix_open(char *full_path, struct inode **pinode,
155 struct super_block *sb, int mode, unsigned int f_flags,
156 __u32 *poplock, __u16 *pnetfid, int xid)
157{
158 int rc;
159 FILE_UNIX_BASIC_INFO *presp_data;
160 __u32 posix_flags = 0;
161 struct cifs_sb_info *cifs_sb = CIFS_SB(sb);
162 struct cifs_fattr fattr;
163 struct tcon_link *tlink;
164 struct cifsTconInfo *tcon;
165
166 cFYI(1, "posix open %s", full_path);
167
168 presp_data = kzalloc(sizeof(FILE_UNIX_BASIC_INFO), GFP_KERNEL);
169 if (presp_data == NULL)
170 return -ENOMEM;
171
172 tlink = cifs_sb_tlink(cifs_sb);
173 if (IS_ERR(tlink)) {
174 rc = PTR_ERR(tlink);
175 goto posix_open_ret;
176 }
177
178 tcon = tlink_tcon(tlink);
179 mode &= ~current_umask();
180
181 posix_flags = cifs_posix_convert_flags(f_flags);
182 rc = CIFSPOSIXCreate(xid, tcon, posix_flags, mode, pnetfid, presp_data,
183 poplock, full_path, cifs_sb->local_nls,
184 cifs_sb->mnt_cifs_flags &
185 CIFS_MOUNT_MAP_SPECIAL_CHR);
186 cifs_put_tlink(tlink);
187
188 if (rc)
189 goto posix_open_ret;
190
191 if (presp_data->Type == cpu_to_le32(-1))
192 goto posix_open_ret; /* open ok, caller does qpathinfo */
193
194 if (!pinode)
195 goto posix_open_ret; /* caller does not need info */
196
197 cifs_unix_basic_to_fattr(&fattr, presp_data, cifs_sb);
198
199 /* get new inode and set it up */
200 if (*pinode == NULL) {
201 cifs_fill_uniqueid(sb, &fattr);
202 *pinode = cifs_iget(sb, &fattr);
203 if (!*pinode) {
204 rc = -ENOMEM;
205 goto posix_open_ret;
206 }
207 } else {
208 cifs_fattr_to_inode(*pinode, &fattr);
209 }
210
211posix_open_ret:
212 kfree(presp_data);
213 return rc;
214}
215
Jeff Layton15ecb432010-10-15 15:34:02 -0400216struct cifsFileInfo *
217cifs_new_fileinfo(__u16 fileHandle, struct file *file,
218 struct tcon_link *tlink, __u32 oplock)
219{
220 struct dentry *dentry = file->f_path.dentry;
221 struct inode *inode = dentry->d_inode;
222 struct cifsInodeInfo *pCifsInode = CIFS_I(inode);
223 struct cifsFileInfo *pCifsFile;
224
225 pCifsFile = kzalloc(sizeof(struct cifsFileInfo), GFP_KERNEL);
226 if (pCifsFile == NULL)
227 return pCifsFile;
228
Jeff Layton5f6dbc92010-10-15 15:34:06 -0400229 pCifsFile->count = 1;
Jeff Layton15ecb432010-10-15 15:34:02 -0400230 pCifsFile->netfid = fileHandle;
231 pCifsFile->pid = current->tgid;
232 pCifsFile->uid = current_fsuid();
233 pCifsFile->dentry = dget(dentry);
234 pCifsFile->f_flags = file->f_flags;
235 pCifsFile->invalidHandle = false;
Jeff Layton15ecb432010-10-15 15:34:02 -0400236 pCifsFile->tlink = cifs_get_tlink(tlink);
237 mutex_init(&pCifsFile->fh_mutex);
238 mutex_init(&pCifsFile->lock_mutex);
239 INIT_LIST_HEAD(&pCifsFile->llist);
Jeff Layton15ecb432010-10-15 15:34:02 -0400240 INIT_WORK(&pCifsFile->oplock_break, cifs_oplock_break);
241
Jeff Layton44772882010-10-15 15:34:03 -0400242 spin_lock(&cifs_file_list_lock);
Jeff Layton15ecb432010-10-15 15:34:02 -0400243 list_add(&pCifsFile->tlist, &(tlink_tcon(tlink)->openFileList));
244 /* if readable file instance put first in list*/
245 if (file->f_mode & FMODE_READ)
246 list_add(&pCifsFile->flist, &pCifsInode->openFileList);
247 else
248 list_add_tail(&pCifsFile->flist, &pCifsInode->openFileList);
Jeff Layton44772882010-10-15 15:34:03 -0400249 spin_unlock(&cifs_file_list_lock);
Jeff Layton15ecb432010-10-15 15:34:02 -0400250
Pavel Shilovskyc6723622010-11-03 10:58:57 +0300251 cifs_set_oplock_level(pCifsInode, oplock);
Jeff Layton15ecb432010-10-15 15:34:02 -0400252
253 file->private_data = pCifsFile;
254 return pCifsFile;
255}
256
Steve Frenchcdff08e2010-10-21 22:46:14 +0000257/*
258 * Release a reference on the file private data. This may involve closing
Jeff Layton5f6dbc92010-10-15 15:34:06 -0400259 * the filehandle out on the server. Must be called without holding
260 * cifs_file_list_lock.
Steve Frenchcdff08e2010-10-21 22:46:14 +0000261 */
Jeff Laytonb33879a2010-10-15 15:34:04 -0400262void cifsFileInfo_put(struct cifsFileInfo *cifs_file)
263{
Pavel Shilovskye66673e2010-11-02 12:00:42 +0300264 struct inode *inode = cifs_file->dentry->d_inode;
Steve Frenchcdff08e2010-10-21 22:46:14 +0000265 struct cifsTconInfo *tcon = tlink_tcon(cifs_file->tlink);
Pavel Shilovskye66673e2010-11-02 12:00:42 +0300266 struct cifsInodeInfo *cifsi = CIFS_I(inode);
Steve Frenchcdff08e2010-10-21 22:46:14 +0000267 struct cifsLockInfo *li, *tmp;
268
269 spin_lock(&cifs_file_list_lock);
Jeff Layton5f6dbc92010-10-15 15:34:06 -0400270 if (--cifs_file->count > 0) {
Steve Frenchcdff08e2010-10-21 22:46:14 +0000271 spin_unlock(&cifs_file_list_lock);
272 return;
Jeff Laytonb33879a2010-10-15 15:34:04 -0400273 }
Steve Frenchcdff08e2010-10-21 22:46:14 +0000274
275 /* remove it from the lists */
276 list_del(&cifs_file->flist);
277 list_del(&cifs_file->tlist);
278
279 if (list_empty(&cifsi->openFileList)) {
280 cFYI(1, "closing last open instance for inode %p",
281 cifs_file->dentry->d_inode);
Pavel Shilovskyc6723622010-11-03 10:58:57 +0300282 cifs_set_oplock_level(cifsi, 0);
Steve Frenchcdff08e2010-10-21 22:46:14 +0000283 }
284 spin_unlock(&cifs_file_list_lock);
285
286 if (!tcon->need_reconnect && !cifs_file->invalidHandle) {
287 int xid, rc;
288
289 xid = GetXid();
290 rc = CIFSSMBClose(xid, tcon, cifs_file->netfid);
291 FreeXid(xid);
292 }
293
294 /* Delete any outstanding lock records. We'll lose them when the file
295 * is closed anyway.
296 */
297 mutex_lock(&cifs_file->lock_mutex);
298 list_for_each_entry_safe(li, tmp, &cifs_file->llist, llist) {
299 list_del(&li->llist);
300 kfree(li);
301 }
302 mutex_unlock(&cifs_file->lock_mutex);
303
304 cifs_put_tlink(cifs_file->tlink);
305 dput(cifs_file->dentry);
306 kfree(cifs_file);
Jeff Laytonb33879a2010-10-15 15:34:04 -0400307}
308
Linus Torvalds1da177e2005-04-16 15:20:36 -0700309int cifs_open(struct inode *inode, struct file *file)
310{
311 int rc = -EACCES;
Jeff Layton590a3fe2009-09-12 11:54:28 -0400312 int xid;
313 __u32 oplock;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700314 struct cifs_sb_info *cifs_sb;
Steve French276a74a2009-03-03 18:00:34 +0000315 struct cifsTconInfo *tcon;
Jeff Layton7ffec372010-09-29 19:51:11 -0400316 struct tcon_link *tlink;
Jeff Layton6ca9f3b2010-06-16 13:40:16 -0400317 struct cifsFileInfo *pCifsFile = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700318 struct cifsInodeInfo *pCifsInode;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700319 char *full_path = NULL;
320 int desiredAccess;
321 int disposition;
322 __u16 netfid;
323 FILE_ALL_INFO *buf = NULL;
324
325 xid = GetXid();
326
327 cifs_sb = CIFS_SB(inode->i_sb);
Jeff Layton7ffec372010-09-29 19:51:11 -0400328 tlink = cifs_sb_tlink(cifs_sb);
329 if (IS_ERR(tlink)) {
330 FreeXid(xid);
331 return PTR_ERR(tlink);
332 }
333 tcon = tlink_tcon(tlink);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700334
Steve Frencha6ce4932009-04-09 01:14:32 +0000335 pCifsInode = CIFS_I(file->f_path.dentry->d_inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700336
Josef "Jeff" Sipeke6a00292006-12-08 02:36:48 -0800337 full_path = build_path_from_dentry(file->f_path.dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700338 if (full_path == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +0530339 rc = -ENOMEM;
Jeff Layton232341b2010-08-05 13:58:38 -0400340 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700341 }
342
Joe Perchesb6b38f72010-04-21 03:50:45 +0000343 cFYI(1, "inode = 0x%p file flags are 0x%x for %s",
344 inode, file->f_flags, full_path);
Steve French276a74a2009-03-03 18:00:34 +0000345
346 if (oplockEnabled)
347 oplock = REQ_OPLOCK;
348 else
349 oplock = 0;
350
Steve French64cc2c62009-03-04 19:54:08 +0000351 if (!tcon->broken_posix_open && tcon->unix_ext &&
352 (tcon->ses->capabilities & CAP_UNIX) &&
Steve French276a74a2009-03-03 18:00:34 +0000353 (CIFS_UNIX_POSIX_PATH_OPS_CAP &
354 le64_to_cpu(tcon->fsUnixInfo.Capability))) {
Steve French276a74a2009-03-03 18:00:34 +0000355 /* can not refresh inode info since size could be stale */
Jeff Layton2422f672010-06-16 13:40:16 -0400356 rc = cifs_posix_open(full_path, &inode, inode->i_sb,
Steve Frenchfa588e02010-04-22 19:21:55 +0000357 cifs_sb->mnt_file_mode /* ignored */,
Jeff Layton608712f2010-10-15 15:33:56 -0400358 file->f_flags, &oplock, &netfid, xid);
Steve French276a74a2009-03-03 18:00:34 +0000359 if (rc == 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000360 cFYI(1, "posix open succeeded");
Jeff Layton47c78b72010-06-16 13:40:17 -0400361
Jeff Laytonabfe1ee2010-10-15 15:33:58 -0400362 pCifsFile = cifs_new_fileinfo(netfid, file, tlink,
363 oplock);
Jeff Layton2422f672010-06-16 13:40:16 -0400364 if (pCifsFile == NULL) {
365 CIFSSMBClose(xid, tcon, netfid);
366 rc = -ENOMEM;
Jeff Layton2422f672010-06-16 13:40:16 -0400367 }
Suresh Jayaraman9451a9a2010-07-05 18:12:45 +0530368
369 cifs_fscache_set_inode_cookie(inode, file);
370
Steve French276a74a2009-03-03 18:00:34 +0000371 goto out;
Steve French64cc2c62009-03-04 19:54:08 +0000372 } else if ((rc == -EINVAL) || (rc == -EOPNOTSUPP)) {
373 if (tcon->ses->serverNOS)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000374 cERROR(1, "server %s of type %s returned"
Steve French64cc2c62009-03-04 19:54:08 +0000375 " unexpected error on SMB posix open"
376 ", disabling posix open support."
377 " Check if server update available.",
378 tcon->ses->serverName,
Joe Perchesb6b38f72010-04-21 03:50:45 +0000379 tcon->ses->serverNOS);
Steve French64cc2c62009-03-04 19:54:08 +0000380 tcon->broken_posix_open = true;
Steve French276a74a2009-03-03 18:00:34 +0000381 } else if ((rc != -EIO) && (rc != -EREMOTE) &&
382 (rc != -EOPNOTSUPP)) /* path not found or net err */
383 goto out;
Steve French64cc2c62009-03-04 19:54:08 +0000384 /* else fallthrough to retry open the old way on network i/o
385 or DFS errors */
Steve French276a74a2009-03-03 18:00:34 +0000386 }
387
Linus Torvalds1da177e2005-04-16 15:20:36 -0700388 desiredAccess = cifs_convert_flags(file->f_flags);
389
390/*********************************************************************
391 * open flag mapping table:
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000392 *
Linus Torvalds1da177e2005-04-16 15:20:36 -0700393 * POSIX Flag CIFS Disposition
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000394 * ---------- ----------------
Linus Torvalds1da177e2005-04-16 15:20:36 -0700395 * O_CREAT FILE_OPEN_IF
396 * O_CREAT | O_EXCL FILE_CREATE
397 * O_CREAT | O_TRUNC FILE_OVERWRITE_IF
398 * O_TRUNC FILE_OVERWRITE
399 * none of the above FILE_OPEN
400 *
401 * Note that there is not a direct match between disposition
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000402 * FILE_SUPERSEDE (ie create whether or not file exists although
Linus Torvalds1da177e2005-04-16 15:20:36 -0700403 * O_CREAT | O_TRUNC is similar but truncates the existing
404 * file rather than creating a new file as FILE_SUPERSEDE does
405 * (which uses the attributes / metadata passed in on open call)
406 *?
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000407 *? O_SYNC is a reasonable match to CIFS writethrough flag
Linus Torvalds1da177e2005-04-16 15:20:36 -0700408 *? and the read write flags match reasonably. O_LARGEFILE
409 *? is irrelevant because largefile support is always used
410 *? by this client. Flags O_APPEND, O_DIRECT, O_DIRECTORY,
411 * O_FASYNC, O_NOFOLLOW, O_NONBLOCK need further investigation
412 *********************************************************************/
413
414 disposition = cifs_get_disposition(file->f_flags);
415
Linus Torvalds1da177e2005-04-16 15:20:36 -0700416 /* BB pass O_SYNC flag through on file attributes .. BB */
417
418 /* Also refresh inode by passing in file_info buf returned by SMBOpen
419 and calling get_inode_info with returned buf (at least helps
420 non-Unix server case) */
421
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000422 /* BB we can not do this if this is the second open of a file
423 and the first handle has writebehind data, we might be
Linus Torvalds1da177e2005-04-16 15:20:36 -0700424 able to simply do a filemap_fdatawrite/filemap_fdatawait first */
425 buf = kmalloc(sizeof(FILE_ALL_INFO), GFP_KERNEL);
426 if (!buf) {
427 rc = -ENOMEM;
428 goto out;
429 }
Steve French5bafd762006-06-07 00:18:43 +0000430
Jeff Laytona6e8a842010-09-20 16:01:33 -0700431 if (tcon->ses->capabilities & CAP_NT_SMBS)
Steve French276a74a2009-03-03 18:00:34 +0000432 rc = CIFSSMBOpen(xid, tcon, full_path, disposition,
Steve French5bafd762006-06-07 00:18:43 +0000433 desiredAccess, CREATE_NOT_DIR, &netfid, &oplock, buf,
Steve French737b7582005-04-28 22:41:06 -0700434 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags
435 & CIFS_MOUNT_MAP_SPECIAL_CHR);
Steve French5bafd762006-06-07 00:18:43 +0000436 else
437 rc = -EIO; /* no NT SMB support fall into legacy open below */
438
Steve Frencha9d02ad2005-08-24 23:06:05 -0700439 if (rc == -EIO) {
440 /* Old server, try legacy style OpenX */
Steve French276a74a2009-03-03 18:00:34 +0000441 rc = SMBLegacyOpen(xid, tcon, full_path, disposition,
Steve Frencha9d02ad2005-08-24 23:06:05 -0700442 desiredAccess, CREATE_NOT_DIR, &netfid, &oplock, buf,
443 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags
444 & CIFS_MOUNT_MAP_SPECIAL_CHR);
445 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700446 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000447 cFYI(1, "cifs_open returned 0x%x", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700448 goto out;
449 }
Jeff Layton3321b792009-09-25 09:53:37 -0400450
Suresh Jayaramana347ecb2010-09-17 19:43:10 +0530451 rc = cifs_open_inode_helper(inode, tcon, oplock, buf, full_path, xid);
Jeff Layton47c78b72010-06-16 13:40:17 -0400452 if (rc != 0)
453 goto out;
454
Jeff Laytonabfe1ee2010-10-15 15:33:58 -0400455 pCifsFile = cifs_new_fileinfo(netfid, file, tlink, oplock);
Jeff Layton6ca9f3b2010-06-16 13:40:16 -0400456 if (pCifsFile == NULL) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700457 rc = -ENOMEM;
458 goto out;
459 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700460
Suresh Jayaraman9451a9a2010-07-05 18:12:45 +0530461 cifs_fscache_set_inode_cookie(inode, file);
462
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000463 if (oplock & CIFS_CREATE_ACTION) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700464 /* time to set mode which we can not set earlier due to
465 problems creating new read-only files */
Steve French276a74a2009-03-03 18:00:34 +0000466 if (tcon->unix_ext) {
Jeff Layton4e1e7fb2008-08-02 07:26:12 -0400467 struct cifs_unix_set_info_args args = {
468 .mode = inode->i_mode,
469 .uid = NO_CHANGE_64,
470 .gid = NO_CHANGE_64,
471 .ctime = NO_CHANGE_64,
472 .atime = NO_CHANGE_64,
473 .mtime = NO_CHANGE_64,
474 .device = 0,
475 };
Jeff Layton01ea95e2009-07-09 20:02:49 -0400476 CIFSSMBUnixSetPathInfo(xid, tcon, full_path, &args,
477 cifs_sb->local_nls,
478 cifs_sb->mnt_cifs_flags &
Steve French737b7582005-04-28 22:41:06 -0700479 CIFS_MOUNT_MAP_SPECIAL_CHR);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700480 }
481 }
482
483out:
484 kfree(buf);
485 kfree(full_path);
486 FreeXid(xid);
Jeff Layton7ffec372010-09-29 19:51:11 -0400487 cifs_put_tlink(tlink);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700488 return rc;
489}
490
Adrian Bunk04187262006-06-30 18:23:04 +0200491/* Try to reacquire byte range locks that were released when session */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700492/* to server was lost */
493static int cifs_relock_file(struct cifsFileInfo *cifsFile)
494{
495 int rc = 0;
496
497/* BB list all locks open on this file and relock */
498
499 return rc;
500}
501
Jeff Layton15886172010-10-15 15:33:59 -0400502static int cifs_reopen_file(struct cifsFileInfo *pCifsFile, bool can_flush)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700503{
504 int rc = -EACCES;
Jeff Layton590a3fe2009-09-12 11:54:28 -0400505 int xid;
506 __u32 oplock;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700507 struct cifs_sb_info *cifs_sb;
Steve French7fc8f4e2009-02-23 20:43:11 +0000508 struct cifsTconInfo *tcon;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700509 struct cifsInodeInfo *pCifsInode;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000510 struct inode *inode;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700511 char *full_path = NULL;
512 int desiredAccess;
513 int disposition = FILE_OPEN;
514 __u16 netfid;
515
Linus Torvalds1da177e2005-04-16 15:20:36 -0700516 xid = GetXid();
Jeff Laytonf0a71eb2009-06-27 07:04:55 -0400517 mutex_lock(&pCifsFile->fh_mutex);
Steve French4b18f2a2008-04-29 00:06:05 +0000518 if (!pCifsFile->invalidHandle) {
Jeff Laytonf0a71eb2009-06-27 07:04:55 -0400519 mutex_unlock(&pCifsFile->fh_mutex);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +0530520 rc = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700521 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +0530522 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700523 }
524
Jeff Layton15886172010-10-15 15:33:59 -0400525 inode = pCifsFile->dentry->d_inode;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700526 cifs_sb = CIFS_SB(inode->i_sb);
Jeff Layton13cfb732010-09-29 19:51:11 -0400527 tcon = tlink_tcon(pCifsFile->tlink);
Steve French3a9f4622007-04-04 17:10:24 +0000528
Linus Torvalds1da177e2005-04-16 15:20:36 -0700529/* can not grab rename sem here because various ops, including
530 those that already have the rename sem can end up causing writepage
531 to get called and if the server was down that means we end up here,
532 and we can never tell if the caller already has the rename_sem */
Jeff Layton15886172010-10-15 15:33:59 -0400533 full_path = build_path_from_dentry(pCifsFile->dentry);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700534 if (full_path == NULL) {
Steve French3a9f4622007-04-04 17:10:24 +0000535 rc = -ENOMEM;
Jeff Laytonf0a71eb2009-06-27 07:04:55 -0400536 mutex_unlock(&pCifsFile->fh_mutex);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700537 FreeXid(xid);
Steve French3a9f4622007-04-04 17:10:24 +0000538 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700539 }
540
Joe Perchesb6b38f72010-04-21 03:50:45 +0000541 cFYI(1, "inode = 0x%p file flags 0x%x for %s",
Jeff Layton15886172010-10-15 15:33:59 -0400542 inode, pCifsFile->f_flags, full_path);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700543
544 if (oplockEnabled)
545 oplock = REQ_OPLOCK;
546 else
Steve French4b18f2a2008-04-29 00:06:05 +0000547 oplock = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700548
Steve French7fc8f4e2009-02-23 20:43:11 +0000549 if (tcon->unix_ext && (tcon->ses->capabilities & CAP_UNIX) &&
550 (CIFS_UNIX_POSIX_PATH_OPS_CAP &
551 le64_to_cpu(tcon->fsUnixInfo.Capability))) {
Jeff Layton608712f2010-10-15 15:33:56 -0400552
553 /*
554 * O_CREAT, O_EXCL and O_TRUNC already had their effect on the
555 * original open. Must mask them off for a reopen.
556 */
Jeff Layton15886172010-10-15 15:33:59 -0400557 unsigned int oflags = pCifsFile->f_flags &
558 ~(O_CREAT | O_EXCL | O_TRUNC);
Jeff Layton608712f2010-10-15 15:33:56 -0400559
Jeff Layton2422f672010-06-16 13:40:16 -0400560 rc = cifs_posix_open(full_path, NULL, inode->i_sb,
Steve Frenchfa588e02010-04-22 19:21:55 +0000561 cifs_sb->mnt_file_mode /* ignored */,
562 oflags, &oplock, &netfid, xid);
Steve French7fc8f4e2009-02-23 20:43:11 +0000563 if (rc == 0) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000564 cFYI(1, "posix reopen succeeded");
Steve French7fc8f4e2009-02-23 20:43:11 +0000565 goto reopen_success;
566 }
567 /* fallthrough to retry open the old way on errors, especially
568 in the reconnect path it is important to retry hard */
569 }
570
Jeff Layton15886172010-10-15 15:33:59 -0400571 desiredAccess = cifs_convert_flags(pCifsFile->f_flags);
Steve French7fc8f4e2009-02-23 20:43:11 +0000572
Linus Torvalds1da177e2005-04-16 15:20:36 -0700573 /* Can not refresh inode by passing in file_info buf to be returned
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000574 by SMBOpen and then calling get_inode_info with returned buf
575 since file might have write behind data that needs to be flushed
Linus Torvalds1da177e2005-04-16 15:20:36 -0700576 and server version of file size can be stale. If we knew for sure
577 that inode was not dirty locally we could do this */
578
Steve French7fc8f4e2009-02-23 20:43:11 +0000579 rc = CIFSSMBOpen(xid, tcon, full_path, disposition, desiredAccess,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700580 CREATE_NOT_DIR, &netfid, &oplock, NULL,
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000581 cifs_sb->local_nls, cifs_sb->mnt_cifs_flags &
Steve French737b7582005-04-28 22:41:06 -0700582 CIFS_MOUNT_MAP_SPECIAL_CHR);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700583 if (rc) {
Jeff Laytonf0a71eb2009-06-27 07:04:55 -0400584 mutex_unlock(&pCifsFile->fh_mutex);
Joe Perchesb6b38f72010-04-21 03:50:45 +0000585 cFYI(1, "cifs_open returned 0x%x", rc);
586 cFYI(1, "oplock: %d", oplock);
Jeff Layton15886172010-10-15 15:33:59 -0400587 goto reopen_error_exit;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700588 }
Jeff Layton15886172010-10-15 15:33:59 -0400589
590reopen_success:
591 pCifsFile->netfid = netfid;
592 pCifsFile->invalidHandle = false;
593 mutex_unlock(&pCifsFile->fh_mutex);
594 pCifsInode = CIFS_I(inode);
595
596 if (can_flush) {
597 rc = filemap_write_and_wait(inode->i_mapping);
Jeff Laytoneb4b7562010-10-22 14:52:29 -0400598 mapping_set_error(inode->i_mapping, rc);
Jeff Layton15886172010-10-15 15:33:59 -0400599
Jeff Layton15886172010-10-15 15:33:59 -0400600 if (tcon->unix_ext)
601 rc = cifs_get_inode_info_unix(&inode,
602 full_path, inode->i_sb, xid);
603 else
604 rc = cifs_get_inode_info(&inode,
605 full_path, NULL, inode->i_sb,
606 xid, NULL);
607 } /* else we are writing out data to server already
608 and could deadlock if we tried to flush data, and
609 since we do not know if we have data that would
610 invalidate the current end of file on the server
611 we can not go to the server to get the new inod
612 info */
Pavel Shilovskye66673e2010-11-02 12:00:42 +0300613
Pavel Shilovskyc6723622010-11-03 10:58:57 +0300614 cifs_set_oplock_level(pCifsInode, oplock);
Pavel Shilovskye66673e2010-11-02 12:00:42 +0300615
Jeff Layton15886172010-10-15 15:33:59 -0400616 cifs_relock_file(pCifsFile);
617
618reopen_error_exit:
Linus Torvalds1da177e2005-04-16 15:20:36 -0700619 kfree(full_path);
620 FreeXid(xid);
621 return rc;
622}
623
624int cifs_close(struct inode *inode, struct file *file)
625{
Steve Frenchcdff08e2010-10-21 22:46:14 +0000626 cifsFileInfo_put(file->private_data);
627 file->private_data = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700628
Steve Frenchcdff08e2010-10-21 22:46:14 +0000629 /* return code from the ->release op is always ignored */
630 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700631}
632
633int cifs_closedir(struct inode *inode, struct file *file)
634{
635 int rc = 0;
636 int xid;
Joe Perchesc21dfb62010-07-12 13:50:14 -0700637 struct cifsFileInfo *pCFileStruct = file->private_data;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700638 char *ptmp;
639
Joe Perchesb6b38f72010-04-21 03:50:45 +0000640 cFYI(1, "Closedir inode = 0x%p", inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700641
642 xid = GetXid();
643
644 if (pCFileStruct) {
Jeff Layton13cfb732010-09-29 19:51:11 -0400645 struct cifsTconInfo *pTcon = tlink_tcon(pCFileStruct->tlink);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700646
Joe Perchesb6b38f72010-04-21 03:50:45 +0000647 cFYI(1, "Freeing private data in close dir");
Jeff Layton44772882010-10-15 15:34:03 -0400648 spin_lock(&cifs_file_list_lock);
Steve French4b18f2a2008-04-29 00:06:05 +0000649 if (!pCFileStruct->srch_inf.endOfSearch &&
650 !pCFileStruct->invalidHandle) {
651 pCFileStruct->invalidHandle = true;
Jeff Layton44772882010-10-15 15:34:03 -0400652 spin_unlock(&cifs_file_list_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700653 rc = CIFSFindClose(xid, pTcon, pCFileStruct->netfid);
Joe Perchesb6b38f72010-04-21 03:50:45 +0000654 cFYI(1, "Closing uncompleted readdir with rc %d",
655 rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700656 /* not much we can do if it fails anyway, ignore rc */
657 rc = 0;
Steve Frenchddb4cbf2008-11-20 20:00:44 +0000658 } else
Jeff Layton44772882010-10-15 15:34:03 -0400659 spin_unlock(&cifs_file_list_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700660 ptmp = pCFileStruct->srch_inf.ntwrk_buf_start;
661 if (ptmp) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000662 cFYI(1, "closedir free smb buf in srch struct");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700663 pCFileStruct->srch_inf.ntwrk_buf_start = NULL;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000664 if (pCFileStruct->srch_inf.smallBuf)
Steve Frenchd47d7c12006-02-28 03:45:48 +0000665 cifs_small_buf_release(ptmp);
666 else
667 cifs_buf_release(ptmp);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700668 }
Jeff Layton13cfb732010-09-29 19:51:11 -0400669 cifs_put_tlink(pCFileStruct->tlink);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700670 kfree(file->private_data);
671 file->private_data = NULL;
672 }
673 /* BB can we lock the filestruct while this is going on? */
674 FreeXid(xid);
675 return rc;
676}
677
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000678static int store_file_lock(struct cifsFileInfo *fid, __u64 len,
679 __u64 offset, __u8 lockType)
680{
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000681 struct cifsLockInfo *li =
682 kmalloc(sizeof(struct cifsLockInfo), GFP_KERNEL);
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000683 if (li == NULL)
684 return -ENOMEM;
685 li->offset = offset;
686 li->length = len;
687 li->type = lockType;
Roland Dreier796e5662007-05-03 04:33:45 +0000688 mutex_lock(&fid->lock_mutex);
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000689 list_add(&li->llist, &fid->llist);
Roland Dreier796e5662007-05-03 04:33:45 +0000690 mutex_unlock(&fid->lock_mutex);
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000691 return 0;
692}
693
Linus Torvalds1da177e2005-04-16 15:20:36 -0700694int cifs_lock(struct file *file, int cmd, struct file_lock *pfLock)
695{
696 int rc, xid;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700697 __u32 numLock = 0;
698 __u32 numUnlock = 0;
699 __u64 length;
Steve French4b18f2a2008-04-29 00:06:05 +0000700 bool wait_flag = false;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700701 struct cifs_sb_info *cifs_sb;
Steve French13a6e422008-12-02 17:24:33 +0000702 struct cifsTconInfo *tcon;
Steve French08547b02006-02-28 22:39:25 +0000703 __u16 netfid;
704 __u8 lockType = LOCKING_ANDX_LARGE_FILES;
Steve French13a6e422008-12-02 17:24:33 +0000705 bool posix_locking = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700706
707 length = 1 + pfLock->fl_end - pfLock->fl_start;
708 rc = -EACCES;
709 xid = GetXid();
710
Joe Perchesb6b38f72010-04-21 03:50:45 +0000711 cFYI(1, "Lock parm: 0x%x flockflags: "
Linus Torvalds1da177e2005-04-16 15:20:36 -0700712 "0x%x flocktype: 0x%x start: %lld end: %lld",
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000713 cmd, pfLock->fl_flags, pfLock->fl_type, pfLock->fl_start,
Joe Perchesb6b38f72010-04-21 03:50:45 +0000714 pfLock->fl_end);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700715
716 if (pfLock->fl_flags & FL_POSIX)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000717 cFYI(1, "Posix");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700718 if (pfLock->fl_flags & FL_FLOCK)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000719 cFYI(1, "Flock");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700720 if (pfLock->fl_flags & FL_SLEEP) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000721 cFYI(1, "Blocking lock");
Steve French4b18f2a2008-04-29 00:06:05 +0000722 wait_flag = true;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700723 }
724 if (pfLock->fl_flags & FL_ACCESS)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000725 cFYI(1, "Process suspended by mandatory locking - "
726 "not implemented yet");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700727 if (pfLock->fl_flags & FL_LEASE)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000728 cFYI(1, "Lease on file - not implemented yet");
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000729 if (pfLock->fl_flags &
Linus Torvalds1da177e2005-04-16 15:20:36 -0700730 (~(FL_POSIX | FL_FLOCK | FL_SLEEP | FL_ACCESS | FL_LEASE)))
Joe Perchesb6b38f72010-04-21 03:50:45 +0000731 cFYI(1, "Unknown lock flags 0x%x", pfLock->fl_flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700732
733 if (pfLock->fl_type == F_WRLCK) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000734 cFYI(1, "F_WRLCK ");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700735 numLock = 1;
736 } else if (pfLock->fl_type == F_UNLCK) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000737 cFYI(1, "F_UNLCK");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700738 numUnlock = 1;
Steve Frenchd47d7c12006-02-28 03:45:48 +0000739 /* Check if unlock includes more than
740 one lock range */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700741 } else if (pfLock->fl_type == F_RDLCK) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000742 cFYI(1, "F_RDLCK");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700743 lockType |= LOCKING_ANDX_SHARED_LOCK;
744 numLock = 1;
745 } else if (pfLock->fl_type == F_EXLCK) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000746 cFYI(1, "F_EXLCK");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700747 numLock = 1;
748 } else if (pfLock->fl_type == F_SHLCK) {
Joe Perchesb6b38f72010-04-21 03:50:45 +0000749 cFYI(1, "F_SHLCK");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700750 lockType |= LOCKING_ANDX_SHARED_LOCK;
751 numLock = 1;
752 } else
Joe Perchesb6b38f72010-04-21 03:50:45 +0000753 cFYI(1, "Unknown type of lock");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700754
Josef "Jeff" Sipeke6a00292006-12-08 02:36:48 -0800755 cifs_sb = CIFS_SB(file->f_path.dentry->d_sb);
Jeff Layton13cfb732010-09-29 19:51:11 -0400756 tcon = tlink_tcon(((struct cifsFileInfo *)file->private_data)->tlink);
Steve French08547b02006-02-28 22:39:25 +0000757 netfid = ((struct cifsFileInfo *)file->private_data)->netfid;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700758
Steve French13a6e422008-12-02 17:24:33 +0000759 if ((tcon->ses->capabilities & CAP_UNIX) &&
760 (CIFS_UNIX_FCNTL_CAP & le64_to_cpu(tcon->fsUnixInfo.Capability)) &&
Steve Frenchacc18aa2008-12-02 18:53:55 +0000761 ((cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NOPOSIXBRL) == 0))
Steve French13a6e422008-12-02 17:24:33 +0000762 posix_locking = 1;
Steve French08547b02006-02-28 22:39:25 +0000763 /* BB add code here to normalize offset and length to
764 account for negative length which we can not accept over the
765 wire */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700766 if (IS_GETLK(cmd)) {
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000767 if (posix_locking) {
Steve French08547b02006-02-28 22:39:25 +0000768 int posix_lock_type;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000769 if (lockType & LOCKING_ANDX_SHARED_LOCK)
Steve French08547b02006-02-28 22:39:25 +0000770 posix_lock_type = CIFS_RDLCK;
771 else
772 posix_lock_type = CIFS_WRLCK;
Steve French13a6e422008-12-02 17:24:33 +0000773 rc = CIFSSMBPosixLock(xid, tcon, netfid, 1 /* get */,
Steve Frenchfc94cdb2006-05-30 18:03:32 +0000774 length, pfLock,
Steve French08547b02006-02-28 22:39:25 +0000775 posix_lock_type, wait_flag);
776 FreeXid(xid);
777 return rc;
778 }
779
780 /* BB we could chain these into one lock request BB */
Steve French13a6e422008-12-02 17:24:33 +0000781 rc = CIFSSMBLock(xid, tcon, netfid, length, pfLock->fl_start,
Steve French08547b02006-02-28 22:39:25 +0000782 0, 1, lockType, 0 /* wait flag */ );
Linus Torvalds1da177e2005-04-16 15:20:36 -0700783 if (rc == 0) {
Steve French13a6e422008-12-02 17:24:33 +0000784 rc = CIFSSMBLock(xid, tcon, netfid, length,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700785 pfLock->fl_start, 1 /* numUnlock */ ,
786 0 /* numLock */ , lockType,
787 0 /* wait flag */ );
788 pfLock->fl_type = F_UNLCK;
789 if (rc != 0)
Joe Perchesb6b38f72010-04-21 03:50:45 +0000790 cERROR(1, "Error unlocking previously locked "
791 "range %d during test of lock", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700792 rc = 0;
793
794 } else {
795 /* if rc == ERR_SHARING_VIOLATION ? */
Pavel Shilovskyf05337c2010-04-05 09:59:14 +0400796 rc = 0;
797
798 if (lockType & LOCKING_ANDX_SHARED_LOCK) {
799 pfLock->fl_type = F_WRLCK;
800 } else {
801 rc = CIFSSMBLock(xid, tcon, netfid, length,
802 pfLock->fl_start, 0, 1,
803 lockType | LOCKING_ANDX_SHARED_LOCK,
804 0 /* wait flag */);
805 if (rc == 0) {
806 rc = CIFSSMBLock(xid, tcon, netfid,
807 length, pfLock->fl_start, 1, 0,
808 lockType |
809 LOCKING_ANDX_SHARED_LOCK,
810 0 /* wait flag */);
811 pfLock->fl_type = F_RDLCK;
812 if (rc != 0)
Steve Frenchf19159d2010-04-21 04:12:10 +0000813 cERROR(1, "Error unlocking "
Pavel Shilovskyf05337c2010-04-05 09:59:14 +0400814 "previously locked range %d "
Steve Frenchf19159d2010-04-21 04:12:10 +0000815 "during test of lock", rc);
Pavel Shilovskyf05337c2010-04-05 09:59:14 +0400816 rc = 0;
817 } else {
818 pfLock->fl_type = F_WRLCK;
819 rc = 0;
820 }
821 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700822 }
823
824 FreeXid(xid);
825 return rc;
826 }
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000827
828 if (!numLock && !numUnlock) {
829 /* if no lock or unlock then nothing
830 to do since we do not know what it is */
831 FreeXid(xid);
832 return -EOPNOTSUPP;
833 }
834
835 if (posix_locking) {
Steve French08547b02006-02-28 22:39:25 +0000836 int posix_lock_type;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000837 if (lockType & LOCKING_ANDX_SHARED_LOCK)
Steve French08547b02006-02-28 22:39:25 +0000838 posix_lock_type = CIFS_RDLCK;
839 else
840 posix_lock_type = CIFS_WRLCK;
Steve French50c2f752007-07-13 00:33:32 +0000841
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000842 if (numUnlock == 1)
Steve Frenchbeb84dc2006-03-03 23:36:34 +0000843 posix_lock_type = CIFS_UNLCK;
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000844
Steve French13a6e422008-12-02 17:24:33 +0000845 rc = CIFSSMBPosixLock(xid, tcon, netfid, 0 /* set */,
Steve Frenchfc94cdb2006-05-30 18:03:32 +0000846 length, pfLock,
Steve French08547b02006-02-28 22:39:25 +0000847 posix_lock_type, wait_flag);
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000848 } else {
Joe Perchesc21dfb62010-07-12 13:50:14 -0700849 struct cifsFileInfo *fid = file->private_data;
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000850
851 if (numLock) {
Steve French13a6e422008-12-02 17:24:33 +0000852 rc = CIFSSMBLock(xid, tcon, netfid, length,
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000853 pfLock->fl_start,
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000854 0, numLock, lockType, wait_flag);
855
856 if (rc == 0) {
857 /* For Windows locks we must store them. */
858 rc = store_file_lock(fid, length,
859 pfLock->fl_start, lockType);
860 }
861 } else if (numUnlock) {
862 /* For each stored lock that this unlock overlaps
863 completely, unlock it. */
864 int stored_rc = 0;
865 struct cifsLockInfo *li, *tmp;
866
Steve French6b70c952006-09-21 07:35:29 +0000867 rc = 0;
Roland Dreier796e5662007-05-03 04:33:45 +0000868 mutex_lock(&fid->lock_mutex);
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000869 list_for_each_entry_safe(li, tmp, &fid->llist, llist) {
870 if (pfLock->fl_start <= li->offset &&
Steve Frenchc19eb712007-08-24 03:22:48 +0000871 (pfLock->fl_start + length) >=
Jeff Layton39db8102007-08-24 03:16:51 +0000872 (li->offset + li->length)) {
Steve French13a6e422008-12-02 17:24:33 +0000873 stored_rc = CIFSSMBLock(xid, tcon,
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000874 netfid,
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000875 li->length, li->offset,
Steve French4b18f2a2008-04-29 00:06:05 +0000876 1, 0, li->type, false);
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000877 if (stored_rc)
878 rc = stored_rc;
Pavel Shilovsky2c964d12010-04-21 19:44:24 +0000879 else {
880 list_del(&li->llist);
881 kfree(li);
882 }
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000883 }
884 }
Roland Dreier796e5662007-05-03 04:33:45 +0000885 mutex_unlock(&fid->lock_mutex);
Jeremy Allison7ee1af72006-08-02 21:56:33 +0000886 }
887 }
888
Steve Frenchd634cc12005-08-26 14:42:59 -0500889 if (pfLock->fl_flags & FL_POSIX)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700890 posix_lock_file_wait(file, pfLock);
891 FreeXid(xid);
892 return rc;
893}
894
Jeff Laytonfbec9ab2009-04-03 13:44:00 -0400895/*
896 * Set the timeout on write requests past EOF. For some servers (Windows)
897 * these calls can be very long.
898 *
899 * If we're writing >10M past the EOF we give a 180s timeout. Anything less
900 * than that gets a 45s timeout. Writes not past EOF get 15s timeouts.
901 * The 10M cutoff is totally arbitrary. A better scheme for this would be
902 * welcome if someone wants to suggest one.
903 *
904 * We may be able to do a better job with this if there were some way to
905 * declare that a file should be sparse.
906 */
907static int
908cifs_write_timeout(struct cifsInodeInfo *cifsi, loff_t offset)
909{
910 if (offset <= cifsi->server_eof)
911 return CIFS_STD_OP;
912 else if (offset > (cifsi->server_eof + (10 * 1024 * 1024)))
913 return CIFS_VLONG_OP;
914 else
915 return CIFS_LONG_OP;
916}
917
918/* update the file size (if needed) after a write */
919static void
920cifs_update_eof(struct cifsInodeInfo *cifsi, loff_t offset,
921 unsigned int bytes_written)
922{
923 loff_t end_of_write = offset + bytes_written;
924
925 if (end_of_write > cifsi->server_eof)
926 cifsi->server_eof = end_of_write;
927}
928
Linus Torvalds1da177e2005-04-16 15:20:36 -0700929ssize_t cifs_user_write(struct file *file, const char __user *write_data,
930 size_t write_size, loff_t *poffset)
931{
Jiri Slaby50ae28f2010-11-01 16:08:55 +0100932 struct inode *inode = file->f_path.dentry->d_inode;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700933 int rc = 0;
934 unsigned int bytes_written = 0;
935 unsigned int total_written;
936 struct cifs_sb_info *cifs_sb;
937 struct cifsTconInfo *pTcon;
938 int xid, long_op;
939 struct cifsFileInfo *open_file;
Jiri Slaby50ae28f2010-11-01 16:08:55 +0100940 struct cifsInodeInfo *cifsi = CIFS_I(inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700941
Josef "Jeff" Sipeke6a00292006-12-08 02:36:48 -0800942 cifs_sb = CIFS_SB(file->f_path.dentry->d_sb);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700943
Joe Perchesb6b38f72010-04-21 03:50:45 +0000944 /* cFYI(1, " write %d bytes to offset %lld of %s", write_size,
945 *poffset, file->f_path.dentry->d_name.name); */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700946
947 if (file->private_data == NULL)
948 return -EBADF;
Jeff Laytonba00ba62010-09-20 16:01:31 -0700949
Joe Perchesc21dfb62010-07-12 13:50:14 -0700950 open_file = file->private_data;
Jeff Layton13cfb732010-09-29 19:51:11 -0400951 pTcon = tlink_tcon(open_file->tlink);
Steve French50c2f752007-07-13 00:33:32 +0000952
Jeff Layton838726c2008-08-28 07:54:59 -0400953 rc = generic_write_checks(file, poffset, &write_size, 0);
954 if (rc)
955 return rc;
956
Linus Torvalds1da177e2005-04-16 15:20:36 -0700957 xid = GetXid();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700958
Jeff Laytonfbec9ab2009-04-03 13:44:00 -0400959 long_op = cifs_write_timeout(cifsi, *poffset);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700960 for (total_written = 0; write_size > total_written;
961 total_written += bytes_written) {
962 rc = -EAGAIN;
963 while (rc == -EAGAIN) {
964 if (file->private_data == NULL) {
965 /* file has been closed on us */
966 FreeXid(xid);
967 /* if we have gotten here we have written some data
968 and blocked, and the file has been freed on us while
969 we blocked so return what we managed to write */
970 return total_written;
Steve Frenchfb8c4b12007-07-10 01:16:18 +0000971 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700972 if (open_file->invalidHandle) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700973 /* we could deadlock if we called
974 filemap_fdatawait from here so tell
975 reopen_file not to flush data to server
976 now */
Jeff Layton15886172010-10-15 15:33:59 -0400977 rc = cifs_reopen_file(open_file, false);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700978 if (rc != 0)
979 break;
980 }
981
982 rc = CIFSSMBWrite(xid, pTcon,
983 open_file->netfid,
984 min_t(const int, cifs_sb->wsize,
985 write_size - total_written),
986 *poffset, &bytes_written,
987 NULL, write_data + total_written, long_op);
988 }
989 if (rc || (bytes_written == 0)) {
990 if (total_written)
991 break;
992 else {
993 FreeXid(xid);
994 return rc;
995 }
Jeff Laytonfbec9ab2009-04-03 13:44:00 -0400996 } else {
997 cifs_update_eof(cifsi, *poffset, bytes_written);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700998 *poffset += bytes_written;
Jeff Laytonfbec9ab2009-04-03 13:44:00 -0400999 }
Steve French133672e2007-11-13 22:41:37 +00001000 long_op = CIFS_STD_OP; /* subsequent writes fast -
Linus Torvalds1da177e2005-04-16 15:20:36 -07001001 15 seconds is plenty */
1002 }
1003
Steve Frencha4544342005-08-24 13:59:35 -07001004 cifs_stats_bytes_written(pTcon, total_written);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001005
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001006/* Do not update local mtime - server will set its actual value on write
Jiri Slaby50ae28f2010-11-01 16:08:55 +01001007 * inode->i_ctime = inode->i_mtime =
1008 * current_fs_time(inode->i_sb);*/
1009 if (total_written > 0) {
1010 spin_lock(&inode->i_lock);
1011 if (*poffset > inode->i_size)
1012 i_size_write(inode, *poffset);
1013 spin_unlock(&inode->i_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001014 }
Jiri Slaby50ae28f2010-11-01 16:08:55 +01001015 mark_inode_dirty_sync(inode);
1016
Linus Torvalds1da177e2005-04-16 15:20:36 -07001017 FreeXid(xid);
1018 return total_written;
1019}
1020
Jeff Layton7da4b492010-10-15 15:34:00 -04001021static ssize_t cifs_write(struct cifsFileInfo *open_file,
1022 const char *write_data, size_t write_size,
1023 loff_t *poffset)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001024{
1025 int rc = 0;
1026 unsigned int bytes_written = 0;
1027 unsigned int total_written;
1028 struct cifs_sb_info *cifs_sb;
1029 struct cifsTconInfo *pTcon;
1030 int xid, long_op;
Jeff Layton7da4b492010-10-15 15:34:00 -04001031 struct dentry *dentry = open_file->dentry;
1032 struct cifsInodeInfo *cifsi = CIFS_I(dentry->d_inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001033
Jeff Layton7da4b492010-10-15 15:34:00 -04001034 cifs_sb = CIFS_SB(dentry->d_sb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001035
Joe Perchesb6b38f72010-04-21 03:50:45 +00001036 cFYI(1, "write %zd bytes to offset %lld of %s", write_size,
Jeff Layton7da4b492010-10-15 15:34:00 -04001037 *poffset, dentry->d_name.name);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001038
Jeff Layton13cfb732010-09-29 19:51:11 -04001039 pTcon = tlink_tcon(open_file->tlink);
Steve French50c2f752007-07-13 00:33:32 +00001040
Linus Torvalds1da177e2005-04-16 15:20:36 -07001041 xid = GetXid();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001042
Jeff Laytonfbec9ab2009-04-03 13:44:00 -04001043 long_op = cifs_write_timeout(cifsi, *poffset);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001044 for (total_written = 0; write_size > total_written;
1045 total_written += bytes_written) {
1046 rc = -EAGAIN;
1047 while (rc == -EAGAIN) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001048 if (open_file->invalidHandle) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001049 /* we could deadlock if we called
1050 filemap_fdatawait from here so tell
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001051 reopen_file not to flush data to
Linus Torvalds1da177e2005-04-16 15:20:36 -07001052 server now */
Jeff Layton15886172010-10-15 15:33:59 -04001053 rc = cifs_reopen_file(open_file, false);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001054 if (rc != 0)
1055 break;
1056 }
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001057 if (experimEnabled || (pTcon->ses->server &&
1058 ((pTcon->ses->server->secMode &
Steve French08775832006-05-30 18:08:26 +00001059 (SECMODE_SIGN_REQUIRED | SECMODE_SIGN_ENABLED))
Steve Frenchc01f36a2006-05-30 18:05:10 +00001060 == 0))) {
Steve French3e844692005-10-03 13:37:24 -07001061 struct kvec iov[2];
1062 unsigned int len;
1063
Steve French0ae0efa2005-10-10 10:57:19 -07001064 len = min((size_t)cifs_sb->wsize,
Steve French3e844692005-10-03 13:37:24 -07001065 write_size - total_written);
1066 /* iov[0] is reserved for smb header */
1067 iov[1].iov_base = (char *)write_data +
1068 total_written;
1069 iov[1].iov_len = len;
Steve Frenchd6e04ae2005-06-13 13:24:43 -05001070 rc = CIFSSMBWrite2(xid, pTcon,
Steve French3e844692005-10-03 13:37:24 -07001071 open_file->netfid, len,
Steve Frenchd6e04ae2005-06-13 13:24:43 -05001072 *poffset, &bytes_written,
Steve French3e844692005-10-03 13:37:24 -07001073 iov, 1, long_op);
Steve Frenchd6e04ae2005-06-13 13:24:43 -05001074 } else
Steve French60808232006-04-22 15:53:05 +00001075 rc = CIFSSMBWrite(xid, pTcon,
1076 open_file->netfid,
1077 min_t(const int, cifs_sb->wsize,
1078 write_size - total_written),
1079 *poffset, &bytes_written,
1080 write_data + total_written,
1081 NULL, long_op);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001082 }
1083 if (rc || (bytes_written == 0)) {
1084 if (total_written)
1085 break;
1086 else {
1087 FreeXid(xid);
1088 return rc;
1089 }
Jeff Laytonfbec9ab2009-04-03 13:44:00 -04001090 } else {
1091 cifs_update_eof(cifsi, *poffset, bytes_written);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001092 *poffset += bytes_written;
Jeff Laytonfbec9ab2009-04-03 13:44:00 -04001093 }
Steve French133672e2007-11-13 22:41:37 +00001094 long_op = CIFS_STD_OP; /* subsequent writes fast -
Linus Torvalds1da177e2005-04-16 15:20:36 -07001095 15 seconds is plenty */
1096 }
1097
Steve Frencha4544342005-08-24 13:59:35 -07001098 cifs_stats_bytes_written(pTcon, total_written);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001099
Jeff Layton7da4b492010-10-15 15:34:00 -04001100 if (total_written > 0) {
1101 spin_lock(&dentry->d_inode->i_lock);
1102 if (*poffset > dentry->d_inode->i_size)
1103 i_size_write(dentry->d_inode, *poffset);
1104 spin_unlock(&dentry->d_inode->i_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001105 }
Jeff Layton7da4b492010-10-15 15:34:00 -04001106 mark_inode_dirty_sync(dentry->d_inode);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001107 FreeXid(xid);
1108 return total_written;
1109}
1110
Steve French630f3f0c2007-10-25 21:17:17 +00001111#ifdef CONFIG_CIFS_EXPERIMENTAL
Jeff Layton6508d902010-09-29 19:51:11 -04001112struct cifsFileInfo *find_readable_file(struct cifsInodeInfo *cifs_inode,
1113 bool fsuid_only)
Steve French630f3f0c2007-10-25 21:17:17 +00001114{
1115 struct cifsFileInfo *open_file = NULL;
Jeff Layton6508d902010-09-29 19:51:11 -04001116 struct cifs_sb_info *cifs_sb = CIFS_SB(cifs_inode->vfs_inode.i_sb);
1117
1118 /* only filter by fsuid on multiuser mounts */
1119 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MULTIUSER))
1120 fsuid_only = false;
Steve French630f3f0c2007-10-25 21:17:17 +00001121
Jeff Layton44772882010-10-15 15:34:03 -04001122 spin_lock(&cifs_file_list_lock);
Steve French630f3f0c2007-10-25 21:17:17 +00001123 /* we could simply get the first_list_entry since write-only entries
1124 are always at the end of the list but since the first entry might
1125 have a close pending, we go through the whole list */
1126 list_for_each_entry(open_file, &cifs_inode->openFileList, flist) {
Jeff Layton6508d902010-09-29 19:51:11 -04001127 if (fsuid_only && open_file->uid != current_fsuid())
1128 continue;
Jeff Layton2e396b82010-10-15 15:34:01 -04001129 if (OPEN_FMODE(open_file->f_flags) & FMODE_READ) {
Steve French630f3f0c2007-10-25 21:17:17 +00001130 if (!open_file->invalidHandle) {
1131 /* found a good file */
1132 /* lock it so it will not be closed on us */
Dave Kleikamp6ab409b2009-08-31 11:07:12 -04001133 cifsFileInfo_get(open_file);
Jeff Layton44772882010-10-15 15:34:03 -04001134 spin_unlock(&cifs_file_list_lock);
Steve French630f3f0c2007-10-25 21:17:17 +00001135 return open_file;
1136 } /* else might as well continue, and look for
1137 another, or simply have the caller reopen it
1138 again rather than trying to fix this handle */
1139 } else /* write only file */
1140 break; /* write only files are last so must be done */
1141 }
Jeff Layton44772882010-10-15 15:34:03 -04001142 spin_unlock(&cifs_file_list_lock);
Steve French630f3f0c2007-10-25 21:17:17 +00001143 return NULL;
1144}
1145#endif
1146
Jeff Layton6508d902010-09-29 19:51:11 -04001147struct cifsFileInfo *find_writable_file(struct cifsInodeInfo *cifs_inode,
1148 bool fsuid_only)
Steve French6148a742005-10-05 12:23:19 -07001149{
1150 struct cifsFileInfo *open_file;
Jeff Laytond3892292010-11-02 16:22:50 -04001151 struct cifs_sb_info *cifs_sb;
Jeff Layton2846d382008-09-22 21:33:33 -04001152 bool any_available = false;
Steve Frenchdd99cd82005-10-05 19:32:49 -07001153 int rc;
Steve French6148a742005-10-05 12:23:19 -07001154
Steve French60808232006-04-22 15:53:05 +00001155 /* Having a null inode here (because mapping->host was set to zero by
1156 the VFS or MM) should not happen but we had reports of on oops (due to
1157 it being zero) during stress testcases so we need to check for it */
1158
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001159 if (cifs_inode == NULL) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001160 cERROR(1, "Null inode passed to cifs_writeable_file");
Steve French60808232006-04-22 15:53:05 +00001161 dump_stack();
1162 return NULL;
1163 }
1164
Jeff Laytond3892292010-11-02 16:22:50 -04001165 cifs_sb = CIFS_SB(cifs_inode->vfs_inode.i_sb);
1166
Jeff Layton6508d902010-09-29 19:51:11 -04001167 /* only filter by fsuid on multiuser mounts */
1168 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_MULTIUSER))
1169 fsuid_only = false;
1170
Jeff Layton44772882010-10-15 15:34:03 -04001171 spin_lock(&cifs_file_list_lock);
Steve French9b22b0b2007-10-02 01:11:08 +00001172refind_writable:
Steve French6148a742005-10-05 12:23:19 -07001173 list_for_each_entry(open_file, &cifs_inode->openFileList, flist) {
Jeff Layton6508d902010-09-29 19:51:11 -04001174 if (!any_available && open_file->pid != current->tgid)
1175 continue;
1176 if (fsuid_only && open_file->uid != current_fsuid())
1177 continue;
Jeff Layton2e396b82010-10-15 15:34:01 -04001178 if (OPEN_FMODE(open_file->f_flags) & FMODE_WRITE) {
Dave Kleikamp6ab409b2009-08-31 11:07:12 -04001179 cifsFileInfo_get(open_file);
Steve French9b22b0b2007-10-02 01:11:08 +00001180
1181 if (!open_file->invalidHandle) {
1182 /* found a good writable file */
Jeff Layton44772882010-10-15 15:34:03 -04001183 spin_unlock(&cifs_file_list_lock);
Steve French9b22b0b2007-10-02 01:11:08 +00001184 return open_file;
1185 }
Steve French8840dee2007-11-16 23:05:52 +00001186
Jeff Layton44772882010-10-15 15:34:03 -04001187 spin_unlock(&cifs_file_list_lock);
Steve Frenchcdff08e2010-10-21 22:46:14 +00001188
Steve French9b22b0b2007-10-02 01:11:08 +00001189 /* Had to unlock since following call can block */
Jeff Layton15886172010-10-15 15:33:59 -04001190 rc = cifs_reopen_file(open_file, false);
Steve Frenchcdff08e2010-10-21 22:46:14 +00001191 if (!rc)
1192 return open_file;
Steve French9b22b0b2007-10-02 01:11:08 +00001193
Steve Frenchcdff08e2010-10-21 22:46:14 +00001194 /* if it fails, try another handle if possible */
Joe Perchesb6b38f72010-04-21 03:50:45 +00001195 cFYI(1, "wp failed on reopen file");
Dave Kleikamp6ab409b2009-08-31 11:07:12 -04001196 cifsFileInfo_put(open_file);
Steve French8840dee2007-11-16 23:05:52 +00001197
Steve Frenchcdff08e2010-10-21 22:46:14 +00001198 spin_lock(&cifs_file_list_lock);
1199
Steve French9b22b0b2007-10-02 01:11:08 +00001200 /* else we simply continue to the next entry. Thus
1201 we do not loop on reopen errors. If we
1202 can not reopen the file, for example if we
1203 reconnected to a server with another client
1204 racing to delete or lock the file we would not
1205 make progress if we restarted before the beginning
1206 of the loop here. */
Steve French6148a742005-10-05 12:23:19 -07001207 }
1208 }
Jeff Layton2846d382008-09-22 21:33:33 -04001209 /* couldn't find useable FH with same pid, try any available */
1210 if (!any_available) {
1211 any_available = true;
1212 goto refind_writable;
1213 }
Jeff Layton44772882010-10-15 15:34:03 -04001214 spin_unlock(&cifs_file_list_lock);
Steve French6148a742005-10-05 12:23:19 -07001215 return NULL;
1216}
1217
Linus Torvalds1da177e2005-04-16 15:20:36 -07001218static int cifs_partialpagewrite(struct page *page, unsigned from, unsigned to)
1219{
1220 struct address_space *mapping = page->mapping;
1221 loff_t offset = (loff_t)page->index << PAGE_CACHE_SHIFT;
1222 char *write_data;
1223 int rc = -EFAULT;
1224 int bytes_written = 0;
1225 struct cifs_sb_info *cifs_sb;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001226 struct inode *inode;
Steve French6148a742005-10-05 12:23:19 -07001227 struct cifsFileInfo *open_file;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001228
1229 if (!mapping || !mapping->host)
1230 return -EFAULT;
1231
1232 inode = page->mapping->host;
1233 cifs_sb = CIFS_SB(inode->i_sb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001234
1235 offset += (loff_t)from;
1236 write_data = kmap(page);
1237 write_data += from;
1238
1239 if ((to > PAGE_CACHE_SIZE) || (from > to)) {
1240 kunmap(page);
1241 return -EIO;
1242 }
1243
1244 /* racing with truncate? */
1245 if (offset > mapping->host->i_size) {
1246 kunmap(page);
1247 return 0; /* don't care */
1248 }
1249
1250 /* check to make sure that we are not extending the file */
1251 if (mapping->host->i_size - offset < (loff_t)to)
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001252 to = (unsigned)(mapping->host->i_size - offset);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001253
Jeff Layton6508d902010-09-29 19:51:11 -04001254 open_file = find_writable_file(CIFS_I(mapping->host), false);
Steve French6148a742005-10-05 12:23:19 -07001255 if (open_file) {
Jeff Layton7da4b492010-10-15 15:34:00 -04001256 bytes_written = cifs_write(open_file, write_data,
1257 to - from, &offset);
Dave Kleikamp6ab409b2009-08-31 11:07:12 -04001258 cifsFileInfo_put(open_file);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001259 /* Does mm or vfs already set times? */
Steve French6148a742005-10-05 12:23:19 -07001260 inode->i_atime = inode->i_mtime = current_fs_time(inode->i_sb);
Steve Frenchbb5a9a02007-12-31 04:21:29 +00001261 if ((bytes_written > 0) && (offset))
Steve French6148a742005-10-05 12:23:19 -07001262 rc = 0;
Steve Frenchbb5a9a02007-12-31 04:21:29 +00001263 else if (bytes_written < 0)
1264 rc = bytes_written;
Steve French6148a742005-10-05 12:23:19 -07001265 } else {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001266 cFYI(1, "No writeable filehandles for inode");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001267 rc = -EIO;
1268 }
1269
1270 kunmap(page);
1271 return rc;
1272}
1273
Linus Torvalds1da177e2005-04-16 15:20:36 -07001274static int cifs_writepages(struct address_space *mapping,
Steve French37c0eb42005-10-05 14:50:29 -07001275 struct writeback_control *wbc)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001276{
Steve French37c0eb42005-10-05 14:50:29 -07001277 unsigned int bytes_to_write;
1278 unsigned int bytes_written;
1279 struct cifs_sb_info *cifs_sb;
1280 int done = 0;
OGAWA Hirofumi111ebb62006-06-23 02:03:26 -07001281 pgoff_t end;
Steve French37c0eb42005-10-05 14:50:29 -07001282 pgoff_t index;
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001283 int range_whole = 0;
1284 struct kvec *iov;
Steve French84d2f072005-10-12 15:32:05 -07001285 int len;
Steve French37c0eb42005-10-05 14:50:29 -07001286 int n_iov = 0;
1287 pgoff_t next;
1288 int nr_pages;
1289 __u64 offset = 0;
Steve French23e7dd72005-10-20 13:44:56 -07001290 struct cifsFileInfo *open_file;
Jeff Laytonba00ba62010-09-20 16:01:31 -07001291 struct cifsTconInfo *tcon;
Jeff Laytonfbec9ab2009-04-03 13:44:00 -04001292 struct cifsInodeInfo *cifsi = CIFS_I(mapping->host);
Steve French37c0eb42005-10-05 14:50:29 -07001293 struct page *page;
1294 struct pagevec pvec;
1295 int rc = 0;
1296 int scanned = 0;
Jeff Laytonfbec9ab2009-04-03 13:44:00 -04001297 int xid, long_op;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001298
Steve French37c0eb42005-10-05 14:50:29 -07001299 cifs_sb = CIFS_SB(mapping->host->i_sb);
Steve French50c2f752007-07-13 00:33:32 +00001300
Steve French37c0eb42005-10-05 14:50:29 -07001301 /*
1302 * If wsize is smaller that the page cache size, default to writing
1303 * one page at a time via cifs_writepage
1304 */
1305 if (cifs_sb->wsize < PAGE_CACHE_SIZE)
1306 return generic_writepages(mapping, wbc);
1307
Steve French9a0c8232007-02-02 04:21:57 +00001308 iov = kmalloc(32 * sizeof(struct kvec), GFP_KERNEL);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001309 if (iov == NULL)
Steve French9a0c8232007-02-02 04:21:57 +00001310 return generic_writepages(mapping, wbc);
1311
Steve French37c0eb42005-10-05 14:50:29 -07001312 /*
Jeff Laytonf3983c22010-09-22 16:17:40 -07001313 * if there's no open file, then this is likely to fail too,
1314 * but it'll at least handle the return. Maybe it should be
1315 * a BUG() instead?
Steve French37c0eb42005-10-05 14:50:29 -07001316 */
Jeff Layton6508d902010-09-29 19:51:11 -04001317 open_file = find_writable_file(CIFS_I(mapping->host), false);
Jeff Laytonf3983c22010-09-22 16:17:40 -07001318 if (!open_file) {
Steve French9a0c8232007-02-02 04:21:57 +00001319 kfree(iov);
Jeff Laytonf3983c22010-09-22 16:17:40 -07001320 return generic_writepages(mapping, wbc);
Steve French37c0eb42005-10-05 14:50:29 -07001321 }
1322
Jeff Layton13cfb732010-09-29 19:51:11 -04001323 tcon = tlink_tcon(open_file->tlink);
Jeff Laytonf3983c22010-09-22 16:17:40 -07001324 if (!experimEnabled && tcon->ses->server->secMode &
1325 (SECMODE_SIGN_REQUIRED | SECMODE_SIGN_ENABLED)) {
1326 cifsFileInfo_put(open_file);
Dan Carpenter6b035902010-10-27 23:19:32 +02001327 kfree(iov);
Jeff Laytonf3983c22010-09-22 16:17:40 -07001328 return generic_writepages(mapping, wbc);
1329 }
1330 cifsFileInfo_put(open_file);
1331
Linus Torvalds1da177e2005-04-16 15:20:36 -07001332 xid = GetXid();
1333
Steve French37c0eb42005-10-05 14:50:29 -07001334 pagevec_init(&pvec, 0);
OGAWA Hirofumi111ebb62006-06-23 02:03:26 -07001335 if (wbc->range_cyclic) {
Steve French37c0eb42005-10-05 14:50:29 -07001336 index = mapping->writeback_index; /* Start from prev offset */
OGAWA Hirofumi111ebb62006-06-23 02:03:26 -07001337 end = -1;
1338 } else {
1339 index = wbc->range_start >> PAGE_CACHE_SHIFT;
1340 end = wbc->range_end >> PAGE_CACHE_SHIFT;
1341 if (wbc->range_start == 0 && wbc->range_end == LLONG_MAX)
1342 range_whole = 1;
Steve French37c0eb42005-10-05 14:50:29 -07001343 scanned = 1;
1344 }
1345retry:
1346 while (!done && (index <= end) &&
1347 (nr_pages = pagevec_lookup_tag(&pvec, mapping, &index,
1348 PAGECACHE_TAG_DIRTY,
1349 min(end - index, (pgoff_t)PAGEVEC_SIZE - 1) + 1))) {
1350 int first;
1351 unsigned int i;
1352
Steve French37c0eb42005-10-05 14:50:29 -07001353 first = -1;
1354 next = 0;
1355 n_iov = 0;
1356 bytes_to_write = 0;
1357
1358 for (i = 0; i < nr_pages; i++) {
1359 page = pvec.pages[i];
1360 /*
1361 * At this point we hold neither mapping->tree_lock nor
1362 * lock on the page itself: the page may be truncated or
1363 * invalidated (changing page->mapping to NULL), or even
1364 * swizzled back from swapper_space to tmpfs file
1365 * mapping
1366 */
1367
1368 if (first < 0)
1369 lock_page(page);
Nick Piggin529ae9a2008-08-02 12:01:03 +02001370 else if (!trylock_page(page))
Steve French37c0eb42005-10-05 14:50:29 -07001371 break;
1372
1373 if (unlikely(page->mapping != mapping)) {
1374 unlock_page(page);
1375 break;
1376 }
1377
OGAWA Hirofumi111ebb62006-06-23 02:03:26 -07001378 if (!wbc->range_cyclic && page->index > end) {
Steve French37c0eb42005-10-05 14:50:29 -07001379 done = 1;
1380 unlock_page(page);
1381 break;
1382 }
1383
1384 if (next && (page->index != next)) {
1385 /* Not next consecutive page */
1386 unlock_page(page);
1387 break;
1388 }
1389
1390 if (wbc->sync_mode != WB_SYNC_NONE)
1391 wait_on_page_writeback(page);
1392
1393 if (PageWriteback(page) ||
Linus Torvaldscb876f42006-12-23 16:19:07 -08001394 !clear_page_dirty_for_io(page)) {
Steve French37c0eb42005-10-05 14:50:29 -07001395 unlock_page(page);
1396 break;
1397 }
Steve French84d2f072005-10-12 15:32:05 -07001398
Linus Torvaldscb876f42006-12-23 16:19:07 -08001399 /*
1400 * This actually clears the dirty bit in the radix tree.
1401 * See cifs_writepage() for more commentary.
1402 */
1403 set_page_writeback(page);
1404
Steve French84d2f072005-10-12 15:32:05 -07001405 if (page_offset(page) >= mapping->host->i_size) {
1406 done = 1;
1407 unlock_page(page);
Linus Torvaldscb876f42006-12-23 16:19:07 -08001408 end_page_writeback(page);
Steve French84d2f072005-10-12 15:32:05 -07001409 break;
1410 }
1411
Steve French37c0eb42005-10-05 14:50:29 -07001412 /*
1413 * BB can we get rid of this? pages are held by pvec
1414 */
1415 page_cache_get(page);
1416
Steve French84d2f072005-10-12 15:32:05 -07001417 len = min(mapping->host->i_size - page_offset(page),
1418 (loff_t)PAGE_CACHE_SIZE);
1419
Steve French37c0eb42005-10-05 14:50:29 -07001420 /* reserve iov[0] for the smb header */
1421 n_iov++;
1422 iov[n_iov].iov_base = kmap(page);
Steve French84d2f072005-10-12 15:32:05 -07001423 iov[n_iov].iov_len = len;
1424 bytes_to_write += len;
Steve French37c0eb42005-10-05 14:50:29 -07001425
1426 if (first < 0) {
1427 first = i;
1428 offset = page_offset(page);
1429 }
1430 next = page->index + 1;
1431 if (bytes_to_write + PAGE_CACHE_SIZE > cifs_sb->wsize)
1432 break;
1433 }
1434 if (n_iov) {
Jeff Layton6508d902010-09-29 19:51:11 -04001435 open_file = find_writable_file(CIFS_I(mapping->host),
1436 false);
Steve French23e7dd72005-10-20 13:44:56 -07001437 if (!open_file) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001438 cERROR(1, "No writable handles for inode");
Steve French23e7dd72005-10-20 13:44:56 -07001439 rc = -EBADF;
Steve French1047abc2005-10-11 19:58:06 -07001440 } else {
Jeff Laytonfbec9ab2009-04-03 13:44:00 -04001441 long_op = cifs_write_timeout(cifsi, offset);
Jeff Laytonf3983c22010-09-22 16:17:40 -07001442 rc = CIFSSMBWrite2(xid, tcon, open_file->netfid,
Steve French23e7dd72005-10-20 13:44:56 -07001443 bytes_to_write, offset,
1444 &bytes_written, iov, n_iov,
Jeff Laytonfbec9ab2009-04-03 13:44:00 -04001445 long_op);
Dave Kleikamp6ab409b2009-08-31 11:07:12 -04001446 cifsFileInfo_put(open_file);
Jeff Laytonfbec9ab2009-04-03 13:44:00 -04001447 cifs_update_eof(cifsi, offset, bytes_written);
Steve French37c0eb42005-10-05 14:50:29 -07001448 }
Jeff Laytonf3983c22010-09-22 16:17:40 -07001449
1450 if (rc || bytes_written < bytes_to_write) {
1451 cERROR(1, "Write2 ret %d, wrote %d",
1452 rc, bytes_written);
Jeff Laytoneb4b7562010-10-22 14:52:29 -04001453 mapping_set_error(mapping, rc);
Jeff Laytonf3983c22010-09-22 16:17:40 -07001454 } else {
1455 cifs_stats_bytes_written(tcon, bytes_written);
1456 }
1457
Steve French37c0eb42005-10-05 14:50:29 -07001458 for (i = 0; i < n_iov; i++) {
1459 page = pvec.pages[first + i];
Steve Frencheb9bdaa2006-01-27 15:11:47 -08001460 /* Should we also set page error on
1461 success rc but too little data written? */
1462 /* BB investigate retry logic on temporary
1463 server crash cases and how recovery works
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001464 when page marked as error */
1465 if (rc)
Steve Frencheb9bdaa2006-01-27 15:11:47 -08001466 SetPageError(page);
Steve French37c0eb42005-10-05 14:50:29 -07001467 kunmap(page);
1468 unlock_page(page);
Linus Torvaldscb876f42006-12-23 16:19:07 -08001469 end_page_writeback(page);
Steve French37c0eb42005-10-05 14:50:29 -07001470 page_cache_release(page);
1471 }
1472 if ((wbc->nr_to_write -= n_iov) <= 0)
1473 done = 1;
1474 index = next;
Dave Kleikampb066a482008-11-18 03:49:05 +00001475 } else
1476 /* Need to re-find the pages we skipped */
1477 index = pvec.pages[0]->index + 1;
1478
Steve French37c0eb42005-10-05 14:50:29 -07001479 pagevec_release(&pvec);
1480 }
1481 if (!scanned && !done) {
1482 /*
1483 * We hit the last page and there is more work to be done: wrap
1484 * back to the start of the file
1485 */
1486 scanned = 1;
1487 index = 0;
1488 goto retry;
1489 }
OGAWA Hirofumi111ebb62006-06-23 02:03:26 -07001490 if (wbc->range_cyclic || (range_whole && wbc->nr_to_write > 0))
Steve French37c0eb42005-10-05 14:50:29 -07001491 mapping->writeback_index = index;
1492
Linus Torvalds1da177e2005-04-16 15:20:36 -07001493 FreeXid(xid);
Steve French9a0c8232007-02-02 04:21:57 +00001494 kfree(iov);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001495 return rc;
1496}
Linus Torvalds1da177e2005-04-16 15:20:36 -07001497
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001498static int cifs_writepage(struct page *page, struct writeback_control *wbc)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001499{
1500 int rc = -EFAULT;
1501 int xid;
1502
1503 xid = GetXid();
1504/* BB add check for wbc flags */
1505 page_cache_get(page);
Steve Frenchad7a2922008-02-07 23:25:02 +00001506 if (!PageUptodate(page))
Joe Perchesb6b38f72010-04-21 03:50:45 +00001507 cFYI(1, "ppw - page not up to date");
Linus Torvaldscb876f42006-12-23 16:19:07 -08001508
1509 /*
1510 * Set the "writeback" flag, and clear "dirty" in the radix tree.
1511 *
1512 * A writepage() implementation always needs to do either this,
1513 * or re-dirty the page with "redirty_page_for_writepage()" in
1514 * the case of a failure.
1515 *
1516 * Just unlocking the page will cause the radix tree tag-bits
1517 * to fail to update with the state of the page correctly.
1518 */
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001519 set_page_writeback(page);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001520 rc = cifs_partialpagewrite(page, 0, PAGE_CACHE_SIZE);
1521 SetPageUptodate(page); /* BB add check for error and Clearuptodate? */
1522 unlock_page(page);
Linus Torvaldscb876f42006-12-23 16:19:07 -08001523 end_page_writeback(page);
1524 page_cache_release(page);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001525 FreeXid(xid);
1526 return rc;
1527}
1528
Nick Piggind9414772008-09-24 11:32:59 -04001529static int cifs_write_end(struct file *file, struct address_space *mapping,
1530 loff_t pos, unsigned len, unsigned copied,
1531 struct page *page, void *fsdata)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001532{
Nick Piggind9414772008-09-24 11:32:59 -04001533 int rc;
1534 struct inode *inode = mapping->host;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001535
Joe Perchesb6b38f72010-04-21 03:50:45 +00001536 cFYI(1, "write_end for page %p from pos %lld with %d bytes",
1537 page, pos, copied);
Steve Frenchad7a2922008-02-07 23:25:02 +00001538
Jeff Laytona98ee8c2008-11-26 19:32:33 +00001539 if (PageChecked(page)) {
1540 if (copied == len)
1541 SetPageUptodate(page);
1542 ClearPageChecked(page);
1543 } else if (!PageUptodate(page) && copied == PAGE_CACHE_SIZE)
Nick Piggind9414772008-09-24 11:32:59 -04001544 SetPageUptodate(page);
1545
Linus Torvalds1da177e2005-04-16 15:20:36 -07001546 if (!PageUptodate(page)) {
Nick Piggind9414772008-09-24 11:32:59 -04001547 char *page_data;
1548 unsigned offset = pos & (PAGE_CACHE_SIZE - 1);
1549 int xid;
1550
1551 xid = GetXid();
Linus Torvalds1da177e2005-04-16 15:20:36 -07001552 /* this is probably better than directly calling
1553 partialpage_write since in this function the file handle is
1554 known which we might as well leverage */
1555 /* BB check if anything else missing out of ppw
1556 such as updating last write time */
1557 page_data = kmap(page);
Jeff Layton7da4b492010-10-15 15:34:00 -04001558 rc = cifs_write(file->private_data, page_data + offset,
1559 copied, &pos);
Nick Piggind9414772008-09-24 11:32:59 -04001560 /* if (rc < 0) should we set writebehind rc? */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001561 kunmap(page);
Nick Piggind9414772008-09-24 11:32:59 -04001562
1563 FreeXid(xid);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001564 } else {
Nick Piggind9414772008-09-24 11:32:59 -04001565 rc = copied;
1566 pos += copied;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001567 set_page_dirty(page);
1568 }
1569
Nick Piggind9414772008-09-24 11:32:59 -04001570 if (rc > 0) {
1571 spin_lock(&inode->i_lock);
1572 if (pos > inode->i_size)
1573 i_size_write(inode, pos);
1574 spin_unlock(&inode->i_lock);
1575 }
1576
1577 unlock_page(page);
1578 page_cache_release(page);
1579
Linus Torvalds1da177e2005-04-16 15:20:36 -07001580 return rc;
1581}
1582
Christoph Hellwig7ea80852010-05-26 17:53:25 +02001583int cifs_fsync(struct file *file, int datasync)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001584{
1585 int xid;
1586 int rc = 0;
Steve Frenchb298f222009-02-21 21:17:43 +00001587 struct cifsTconInfo *tcon;
Joe Perchesc21dfb62010-07-12 13:50:14 -07001588 struct cifsFileInfo *smbfile = file->private_data;
Josef "Jeff" Sipeke6a00292006-12-08 02:36:48 -08001589 struct inode *inode = file->f_path.dentry->d_inode;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001590
1591 xid = GetXid();
1592
Joe Perchesb6b38f72010-04-21 03:50:45 +00001593 cFYI(1, "Sync file - name: %s datasync: 0x%x",
Christoph Hellwig7ea80852010-05-26 17:53:25 +02001594 file->f_path.dentry->d_name.name, datasync);
Steve French50c2f752007-07-13 00:33:32 +00001595
Jeff Laytoncea21802007-11-20 23:19:03 +00001596 rc = filemap_write_and_wait(inode->i_mapping);
1597 if (rc == 0) {
Jeff Laytoneb4b7562010-10-22 14:52:29 -04001598 struct cifs_sb_info *cifs_sb = CIFS_SB(inode->i_sb);
1599
Jeff Layton13cfb732010-09-29 19:51:11 -04001600 tcon = tlink_tcon(smbfile->tlink);
Jeff Laytoneb4b7562010-10-22 14:52:29 -04001601 if (!(cifs_sb->mnt_cifs_flags & CIFS_MOUNT_NOSSYNC))
Steve Frenchb298f222009-02-21 21:17:43 +00001602 rc = CIFSSMBFlush(xid, tcon, smbfile->netfid);
Jeff Laytoncea21802007-11-20 23:19:03 +00001603 }
Steve Frenchb298f222009-02-21 21:17:43 +00001604
Linus Torvalds1da177e2005-04-16 15:20:36 -07001605 FreeXid(xid);
1606 return rc;
1607}
1608
NeilBrown3978d7172006-03-26 01:37:17 -08001609/* static void cifs_sync_page(struct page *page)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001610{
1611 struct address_space *mapping;
1612 struct inode *inode;
1613 unsigned long index = page->index;
1614 unsigned int rpages = 0;
1615 int rc = 0;
1616
Steve Frenchf19159d2010-04-21 04:12:10 +00001617 cFYI(1, "sync page %p", page);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001618 mapping = page->mapping;
1619 if (!mapping)
1620 return 0;
1621 inode = mapping->host;
1622 if (!inode)
NeilBrown3978d7172006-03-26 01:37:17 -08001623 return; */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001624
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001625/* fill in rpages then
Linus Torvalds1da177e2005-04-16 15:20:36 -07001626 result = cifs_pagein_inode(inode, index, rpages); */ /* BB finish */
1627
Joe Perchesb6b38f72010-04-21 03:50:45 +00001628/* cFYI(1, "rpages is %d for sync page of Index %ld", rpages, index);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001629
NeilBrown3978d7172006-03-26 01:37:17 -08001630#if 0
Linus Torvalds1da177e2005-04-16 15:20:36 -07001631 if (rc < 0)
1632 return rc;
1633 return 0;
NeilBrown3978d7172006-03-26 01:37:17 -08001634#endif
Linus Torvalds1da177e2005-04-16 15:20:36 -07001635} */
1636
1637/*
1638 * As file closes, flush all cached write data for this inode checking
1639 * for write behind errors.
1640 */
Miklos Szeredi75e1fcc2006-06-23 02:05:12 -07001641int cifs_flush(struct file *file, fl_owner_t id)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001642{
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001643 struct inode *inode = file->f_path.dentry->d_inode;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001644 int rc = 0;
1645
Jeff Laytoneb4b7562010-10-22 14:52:29 -04001646 if (file->f_mode & FMODE_WRITE)
Jeff Laytond3f13222010-10-15 15:34:07 -04001647 rc = filemap_write_and_wait(inode->i_mapping);
Steve French50c2f752007-07-13 00:33:32 +00001648
Joe Perchesb6b38f72010-04-21 03:50:45 +00001649 cFYI(1, "Flush inode %p file %p rc %d", inode, file, rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001650
1651 return rc;
1652}
1653
1654ssize_t cifs_user_read(struct file *file, char __user *read_data,
1655 size_t read_size, loff_t *poffset)
1656{
1657 int rc = -EACCES;
1658 unsigned int bytes_read = 0;
1659 unsigned int total_read = 0;
1660 unsigned int current_read_size;
1661 struct cifs_sb_info *cifs_sb;
1662 struct cifsTconInfo *pTcon;
1663 int xid;
1664 struct cifsFileInfo *open_file;
1665 char *smb_read_data;
1666 char __user *current_offset;
1667 struct smb_com_read_rsp *pSMBr;
1668
1669 xid = GetXid();
Josef "Jeff" Sipeke6a00292006-12-08 02:36:48 -08001670 cifs_sb = CIFS_SB(file->f_path.dentry->d_sb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001671
1672 if (file->private_data == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301673 rc = -EBADF;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001674 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301675 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001676 }
Joe Perchesc21dfb62010-07-12 13:50:14 -07001677 open_file = file->private_data;
Jeff Layton13cfb732010-09-29 19:51:11 -04001678 pTcon = tlink_tcon(open_file->tlink);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001679
Steve Frenchad7a2922008-02-07 23:25:02 +00001680 if ((file->f_flags & O_ACCMODE) == O_WRONLY)
Joe Perchesb6b38f72010-04-21 03:50:45 +00001681 cFYI(1, "attempting read on write only file instance");
Steve Frenchad7a2922008-02-07 23:25:02 +00001682
Linus Torvalds1da177e2005-04-16 15:20:36 -07001683 for (total_read = 0, current_offset = read_data;
1684 read_size > total_read;
1685 total_read += bytes_read, current_offset += bytes_read) {
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001686 current_read_size = min_t(const int, read_size - total_read,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001687 cifs_sb->rsize);
1688 rc = -EAGAIN;
1689 smb_read_data = NULL;
1690 while (rc == -EAGAIN) {
Steve Frenchec637e32005-12-12 20:53:18 -08001691 int buf_type = CIFS_NO_BUFFER;
Steve Frenchcdff08e2010-10-21 22:46:14 +00001692 if (open_file->invalidHandle) {
Jeff Layton15886172010-10-15 15:33:59 -04001693 rc = cifs_reopen_file(open_file, true);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001694 if (rc != 0)
1695 break;
1696 }
Steve Frenchbfa0d752005-08-31 21:50:37 -07001697 rc = CIFSSMBRead(xid, pTcon,
Steve Frenchec637e32005-12-12 20:53:18 -08001698 open_file->netfid,
1699 current_read_size, *poffset,
1700 &bytes_read, &smb_read_data,
1701 &buf_type);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001702 pSMBr = (struct smb_com_read_rsp *)smb_read_data;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001703 if (smb_read_data) {
Steve French93544cc2006-02-14 22:30:52 -06001704 if (copy_to_user(current_offset,
1705 smb_read_data +
1706 4 /* RFC1001 length field */ +
1707 le16_to_cpu(pSMBr->DataOffset),
Steve Frenchad7a2922008-02-07 23:25:02 +00001708 bytes_read))
Steve French93544cc2006-02-14 22:30:52 -06001709 rc = -EFAULT;
Steve French93544cc2006-02-14 22:30:52 -06001710
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001711 if (buf_type == CIFS_SMALL_BUFFER)
Steve Frenchec637e32005-12-12 20:53:18 -08001712 cifs_small_buf_release(smb_read_data);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001713 else if (buf_type == CIFS_LARGE_BUFFER)
Steve Frenchec637e32005-12-12 20:53:18 -08001714 cifs_buf_release(smb_read_data);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001715 smb_read_data = NULL;
1716 }
1717 }
1718 if (rc || (bytes_read == 0)) {
1719 if (total_read) {
1720 break;
1721 } else {
1722 FreeXid(xid);
1723 return rc;
1724 }
1725 } else {
Steve Frencha4544342005-08-24 13:59:35 -07001726 cifs_stats_bytes_read(pTcon, bytes_read);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001727 *poffset += bytes_read;
1728 }
1729 }
1730 FreeXid(xid);
1731 return total_read;
1732}
1733
1734
1735static ssize_t cifs_read(struct file *file, char *read_data, size_t read_size,
1736 loff_t *poffset)
1737{
1738 int rc = -EACCES;
1739 unsigned int bytes_read = 0;
1740 unsigned int total_read;
1741 unsigned int current_read_size;
1742 struct cifs_sb_info *cifs_sb;
1743 struct cifsTconInfo *pTcon;
1744 int xid;
1745 char *current_offset;
1746 struct cifsFileInfo *open_file;
Steve Frenchec637e32005-12-12 20:53:18 -08001747 int buf_type = CIFS_NO_BUFFER;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001748
1749 xid = GetXid();
Josef "Jeff" Sipeke6a00292006-12-08 02:36:48 -08001750 cifs_sb = CIFS_SB(file->f_path.dentry->d_sb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001751
1752 if (file->private_data == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301753 rc = -EBADF;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001754 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301755 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001756 }
Joe Perchesc21dfb62010-07-12 13:50:14 -07001757 open_file = file->private_data;
Jeff Layton13cfb732010-09-29 19:51:11 -04001758 pTcon = tlink_tcon(open_file->tlink);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001759
1760 if ((file->f_flags & O_ACCMODE) == O_WRONLY)
Joe Perchesb6b38f72010-04-21 03:50:45 +00001761 cFYI(1, "attempting read on write only file instance");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001762
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001763 for (total_read = 0, current_offset = read_data;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001764 read_size > total_read;
1765 total_read += bytes_read, current_offset += bytes_read) {
1766 current_read_size = min_t(const int, read_size - total_read,
1767 cifs_sb->rsize);
Steve Frenchf9f5c8172005-09-15 23:06:38 -07001768 /* For windows me and 9x we do not want to request more
1769 than it negotiated since it will refuse the read then */
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001770 if ((pTcon->ses) &&
Steve Frenchf9f5c8172005-09-15 23:06:38 -07001771 !(pTcon->ses->capabilities & CAP_LARGE_FILES)) {
1772 current_read_size = min_t(const int, current_read_size,
1773 pTcon->ses->server->maxBuf - 128);
1774 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001775 rc = -EAGAIN;
1776 while (rc == -EAGAIN) {
Steve Frenchcdff08e2010-10-21 22:46:14 +00001777 if (open_file->invalidHandle) {
Jeff Layton15886172010-10-15 15:33:59 -04001778 rc = cifs_reopen_file(open_file, true);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001779 if (rc != 0)
1780 break;
1781 }
Steve Frenchbfa0d752005-08-31 21:50:37 -07001782 rc = CIFSSMBRead(xid, pTcon,
Steve Frenchec637e32005-12-12 20:53:18 -08001783 open_file->netfid,
1784 current_read_size, *poffset,
1785 &bytes_read, &current_offset,
1786 &buf_type);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001787 }
1788 if (rc || (bytes_read == 0)) {
1789 if (total_read) {
1790 break;
1791 } else {
1792 FreeXid(xid);
1793 return rc;
1794 }
1795 } else {
Steve Frencha4544342005-08-24 13:59:35 -07001796 cifs_stats_bytes_read(pTcon, total_read);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001797 *poffset += bytes_read;
1798 }
1799 }
1800 FreeXid(xid);
1801 return total_read;
1802}
1803
1804int cifs_file_mmap(struct file *file, struct vm_area_struct *vma)
1805{
Linus Torvalds1da177e2005-04-16 15:20:36 -07001806 int rc, xid;
1807
1808 xid = GetXid();
Jeff Laytonabab0952010-02-12 07:44:18 -05001809 rc = cifs_revalidate_file(file);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001810 if (rc) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001811 cFYI(1, "Validation prior to mmap failed, error=%d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001812 FreeXid(xid);
1813 return rc;
1814 }
1815 rc = generic_file_mmap(file, vma);
1816 FreeXid(xid);
1817 return rc;
1818}
1819
1820
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001821static void cifs_copy_cache_pages(struct address_space *mapping,
Nick Piggin315e9952010-04-21 03:18:28 +00001822 struct list_head *pages, int bytes_read, char *data)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001823{
1824 struct page *page;
1825 char *target;
1826
1827 while (bytes_read > 0) {
1828 if (list_empty(pages))
1829 break;
1830
1831 page = list_entry(pages->prev, struct page, lru);
1832 list_del(&page->lru);
1833
Nick Piggin315e9952010-04-21 03:18:28 +00001834 if (add_to_page_cache_lru(page, mapping, page->index,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001835 GFP_KERNEL)) {
1836 page_cache_release(page);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001837 cFYI(1, "Add page cache failed");
Steve French3079ca62005-06-09 14:44:07 -07001838 data += PAGE_CACHE_SIZE;
1839 bytes_read -= PAGE_CACHE_SIZE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001840 continue;
1841 }
Jeff Layton06b43672010-06-01 10:54:45 -04001842 page_cache_release(page);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001843
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001844 target = kmap_atomic(page, KM_USER0);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001845
1846 if (PAGE_CACHE_SIZE > bytes_read) {
1847 memcpy(target, data, bytes_read);
1848 /* zero the tail end of this partial page */
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001849 memset(target + bytes_read, 0,
Linus Torvalds1da177e2005-04-16 15:20:36 -07001850 PAGE_CACHE_SIZE - bytes_read);
1851 bytes_read = 0;
1852 } else {
1853 memcpy(target, data, PAGE_CACHE_SIZE);
1854 bytes_read -= PAGE_CACHE_SIZE;
1855 }
1856 kunmap_atomic(target, KM_USER0);
1857
1858 flush_dcache_page(page);
1859 SetPageUptodate(page);
1860 unlock_page(page);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001861 data += PAGE_CACHE_SIZE;
Suresh Jayaraman9dc06552010-07-05 18:13:11 +05301862
1863 /* add page to FS-Cache */
1864 cifs_readpage_to_fscache(mapping->host, page);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001865 }
1866 return;
1867}
1868
1869static int cifs_readpages(struct file *file, struct address_space *mapping,
1870 struct list_head *page_list, unsigned num_pages)
1871{
1872 int rc = -EACCES;
1873 int xid;
1874 loff_t offset;
1875 struct page *page;
1876 struct cifs_sb_info *cifs_sb;
1877 struct cifsTconInfo *pTcon;
Steve French2c2130e2007-10-12 19:10:28 +00001878 unsigned int bytes_read = 0;
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001879 unsigned int read_size, i;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001880 char *smb_read_data = NULL;
1881 struct smb_com_read_rsp *pSMBr;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001882 struct cifsFileInfo *open_file;
Steve Frenchec637e32005-12-12 20:53:18 -08001883 int buf_type = CIFS_NO_BUFFER;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001884
1885 xid = GetXid();
1886 if (file->private_data == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301887 rc = -EBADF;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001888 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05301889 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001890 }
Joe Perchesc21dfb62010-07-12 13:50:14 -07001891 open_file = file->private_data;
Josef "Jeff" Sipeke6a00292006-12-08 02:36:48 -08001892 cifs_sb = CIFS_SB(file->f_path.dentry->d_sb);
Jeff Layton13cfb732010-09-29 19:51:11 -04001893 pTcon = tlink_tcon(open_file->tlink);
Steve Frenchbfa0d752005-08-31 21:50:37 -07001894
Suresh Jayaraman56698232010-07-05 18:13:25 +05301895 /*
1896 * Reads as many pages as possible from fscache. Returns -ENOBUFS
1897 * immediately if the cookie is negative
1898 */
1899 rc = cifs_readpages_from_fscache(mapping->host, mapping, page_list,
1900 &num_pages);
1901 if (rc == 0)
1902 goto read_complete;
1903
Steve Frenchf19159d2010-04-21 04:12:10 +00001904 cFYI(DBG2, "rpages: num pages %d", num_pages);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001905 for (i = 0; i < num_pages; ) {
1906 unsigned contig_pages;
1907 struct page *tmp_page;
1908 unsigned long expected_index;
1909
1910 if (list_empty(page_list))
1911 break;
1912
1913 page = list_entry(page_list->prev, struct page, lru);
1914 offset = (loff_t)page->index << PAGE_CACHE_SHIFT;
1915
1916 /* count adjacent pages that we will read into */
1917 contig_pages = 0;
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001918 expected_index =
Linus Torvalds1da177e2005-04-16 15:20:36 -07001919 list_entry(page_list->prev, struct page, lru)->index;
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001920 list_for_each_entry_reverse(tmp_page, page_list, lru) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001921 if (tmp_page->index == expected_index) {
1922 contig_pages++;
1923 expected_index++;
1924 } else
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001925 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001926 }
1927 if (contig_pages + i > num_pages)
1928 contig_pages = num_pages - i;
1929
1930 /* for reads over a certain size could initiate async
1931 read ahead */
1932
1933 read_size = contig_pages * PAGE_CACHE_SIZE;
1934 /* Read size needs to be in multiples of one page */
1935 read_size = min_t(const unsigned int, read_size,
1936 cifs_sb->rsize & PAGE_CACHE_MASK);
Joe Perchesb6b38f72010-04-21 03:50:45 +00001937 cFYI(DBG2, "rpages: read size 0x%x contiguous pages %d",
1938 read_size, contig_pages);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001939 rc = -EAGAIN;
1940 while (rc == -EAGAIN) {
Steve Frenchcdff08e2010-10-21 22:46:14 +00001941 if (open_file->invalidHandle) {
Jeff Layton15886172010-10-15 15:33:59 -04001942 rc = cifs_reopen_file(open_file, true);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001943 if (rc != 0)
1944 break;
1945 }
1946
Steve Frenchbfa0d752005-08-31 21:50:37 -07001947 rc = CIFSSMBRead(xid, pTcon,
Steve Frenchec637e32005-12-12 20:53:18 -08001948 open_file->netfid,
1949 read_size, offset,
1950 &bytes_read, &smb_read_data,
1951 &buf_type);
Steve Frencha9d02ad2005-08-24 23:06:05 -07001952 /* BB more RC checks ? */
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001953 if (rc == -EAGAIN) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001954 if (smb_read_data) {
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001955 if (buf_type == CIFS_SMALL_BUFFER)
Steve Frenchec637e32005-12-12 20:53:18 -08001956 cifs_small_buf_release(smb_read_data);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001957 else if (buf_type == CIFS_LARGE_BUFFER)
Steve Frenchec637e32005-12-12 20:53:18 -08001958 cifs_buf_release(smb_read_data);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001959 smb_read_data = NULL;
1960 }
1961 }
1962 }
1963 if ((rc < 0) || (smb_read_data == NULL)) {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001964 cFYI(1, "Read error in readpages: %d", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001965 break;
1966 } else if (bytes_read > 0) {
Andrew Morton6f88cc22006-12-10 02:19:44 -08001967 task_io_account_read(bytes_read);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001968 pSMBr = (struct smb_com_read_rsp *)smb_read_data;
1969 cifs_copy_cache_pages(mapping, page_list, bytes_read,
1970 smb_read_data + 4 /* RFC1001 hdr */ +
Nick Piggin315e9952010-04-21 03:18:28 +00001971 le16_to_cpu(pSMBr->DataOffset));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001972
1973 i += bytes_read >> PAGE_CACHE_SHIFT;
Steve Frencha4544342005-08-24 13:59:35 -07001974 cifs_stats_bytes_read(pTcon, bytes_read);
Steve French2c2130e2007-10-12 19:10:28 +00001975 if ((bytes_read & PAGE_CACHE_MASK) != bytes_read) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001976 i++; /* account for partial page */
1977
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001978 /* server copy of file can have smaller size
Linus Torvalds1da177e2005-04-16 15:20:36 -07001979 than client */
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001980 /* BB do we need to verify this common case ?
1981 this case is ok - if we are at server EOF
Linus Torvalds1da177e2005-04-16 15:20:36 -07001982 we will hit it on next read */
1983
OGAWA Hirofumi05ac9d42006-11-02 22:07:08 -08001984 /* break; */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001985 }
1986 } else {
Joe Perchesb6b38f72010-04-21 03:50:45 +00001987 cFYI(1, "No bytes read (%d) at offset %lld . "
Steve Frenchf19159d2010-04-21 04:12:10 +00001988 "Cleaning remaining pages from readahead list",
Joe Perchesb6b38f72010-04-21 03:50:45 +00001989 bytes_read, offset);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001990 /* BB turn off caching and do new lookup on
Linus Torvalds1da177e2005-04-16 15:20:36 -07001991 file size at server? */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001992 break;
1993 }
1994 if (smb_read_data) {
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001995 if (buf_type == CIFS_SMALL_BUFFER)
Steve Frenchec637e32005-12-12 20:53:18 -08001996 cifs_small_buf_release(smb_read_data);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00001997 else if (buf_type == CIFS_LARGE_BUFFER)
Steve Frenchec637e32005-12-12 20:53:18 -08001998 cifs_buf_release(smb_read_data);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001999 smb_read_data = NULL;
2000 }
2001 bytes_read = 0;
2002 }
2003
Linus Torvalds1da177e2005-04-16 15:20:36 -07002004/* need to free smb_read_data buf before exit */
2005 if (smb_read_data) {
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002006 if (buf_type == CIFS_SMALL_BUFFER)
Steve French47c886b2006-01-18 14:20:39 -08002007 cifs_small_buf_release(smb_read_data);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002008 else if (buf_type == CIFS_LARGE_BUFFER)
Steve French47c886b2006-01-18 14:20:39 -08002009 cifs_buf_release(smb_read_data);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002010 smb_read_data = NULL;
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002011 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002012
Suresh Jayaraman56698232010-07-05 18:13:25 +05302013read_complete:
Linus Torvalds1da177e2005-04-16 15:20:36 -07002014 FreeXid(xid);
2015 return rc;
2016}
2017
2018static int cifs_readpage_worker(struct file *file, struct page *page,
2019 loff_t *poffset)
2020{
2021 char *read_data;
2022 int rc;
2023
Suresh Jayaraman56698232010-07-05 18:13:25 +05302024 /* Is the page cached? */
2025 rc = cifs_readpage_from_fscache(file->f_path.dentry->d_inode, page);
2026 if (rc == 0)
2027 goto read_complete;
2028
Linus Torvalds1da177e2005-04-16 15:20:36 -07002029 page_cache_get(page);
2030 read_data = kmap(page);
2031 /* for reads over a certain size could initiate async read ahead */
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002032
Linus Torvalds1da177e2005-04-16 15:20:36 -07002033 rc = cifs_read(file, read_data, PAGE_CACHE_SIZE, poffset);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002034
Linus Torvalds1da177e2005-04-16 15:20:36 -07002035 if (rc < 0)
2036 goto io_error;
2037 else
Joe Perchesb6b38f72010-04-21 03:50:45 +00002038 cFYI(1, "Bytes read %d", rc);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002039
Josef "Jeff" Sipeke6a00292006-12-08 02:36:48 -08002040 file->f_path.dentry->d_inode->i_atime =
2041 current_fs_time(file->f_path.dentry->d_inode->i_sb);
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002042
Linus Torvalds1da177e2005-04-16 15:20:36 -07002043 if (PAGE_CACHE_SIZE > rc)
2044 memset(read_data + rc, 0, PAGE_CACHE_SIZE - rc);
2045
2046 flush_dcache_page(page);
2047 SetPageUptodate(page);
Suresh Jayaraman9dc06552010-07-05 18:13:11 +05302048
2049 /* send this page to the cache */
2050 cifs_readpage_to_fscache(file->f_path.dentry->d_inode, page);
2051
Linus Torvalds1da177e2005-04-16 15:20:36 -07002052 rc = 0;
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002053
Linus Torvalds1da177e2005-04-16 15:20:36 -07002054io_error:
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002055 kunmap(page);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002056 page_cache_release(page);
Suresh Jayaraman56698232010-07-05 18:13:25 +05302057
2058read_complete:
Linus Torvalds1da177e2005-04-16 15:20:36 -07002059 return rc;
2060}
2061
2062static int cifs_readpage(struct file *file, struct page *page)
2063{
2064 loff_t offset = (loff_t)page->index << PAGE_CACHE_SHIFT;
2065 int rc = -EACCES;
2066 int xid;
2067
2068 xid = GetXid();
2069
2070 if (file->private_data == NULL) {
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05302071 rc = -EBADF;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002072 FreeXid(xid);
Suresh Jayaraman0f3bc092009-06-25 18:12:34 +05302073 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002074 }
2075
Joe Perchesb6b38f72010-04-21 03:50:45 +00002076 cFYI(1, "readpage %p at offset %d 0x%x\n",
2077 page, (int)offset, (int)offset);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002078
2079 rc = cifs_readpage_worker(file, page, &offset);
2080
2081 unlock_page(page);
2082
2083 FreeXid(xid);
2084 return rc;
2085}
2086
Steve Frencha403a0a2007-07-26 15:54:16 +00002087static int is_inode_writable(struct cifsInodeInfo *cifs_inode)
2088{
2089 struct cifsFileInfo *open_file;
2090
Jeff Layton44772882010-10-15 15:34:03 -04002091 spin_lock(&cifs_file_list_lock);
Steve Frencha403a0a2007-07-26 15:54:16 +00002092 list_for_each_entry(open_file, &cifs_inode->openFileList, flist) {
Jeff Layton2e396b82010-10-15 15:34:01 -04002093 if (OPEN_FMODE(open_file->f_flags) & FMODE_WRITE) {
Jeff Layton44772882010-10-15 15:34:03 -04002094 spin_unlock(&cifs_file_list_lock);
Steve Frencha403a0a2007-07-26 15:54:16 +00002095 return 1;
2096 }
2097 }
Jeff Layton44772882010-10-15 15:34:03 -04002098 spin_unlock(&cifs_file_list_lock);
Steve Frencha403a0a2007-07-26 15:54:16 +00002099 return 0;
2100}
2101
Linus Torvalds1da177e2005-04-16 15:20:36 -07002102/* We do not want to update the file size from server for inodes
2103 open for write - to avoid races with writepage extending
2104 the file - in the future we could consider allowing
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002105 refreshing the inode only on increases in the file size
Linus Torvalds1da177e2005-04-16 15:20:36 -07002106 but this is tricky to do without racing with writebehind
2107 page caching in the current Linux kernel design */
Steve French4b18f2a2008-04-29 00:06:05 +00002108bool is_size_safe_to_change(struct cifsInodeInfo *cifsInode, __u64 end_of_file)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002109{
Steve Frencha403a0a2007-07-26 15:54:16 +00002110 if (!cifsInode)
Steve French4b18f2a2008-04-29 00:06:05 +00002111 return true;
Steve French23e7dd72005-10-20 13:44:56 -07002112
Steve Frencha403a0a2007-07-26 15:54:16 +00002113 if (is_inode_writable(cifsInode)) {
2114 /* This inode is open for write at least once */
Steve Frenchc32a0b62006-01-12 14:41:28 -08002115 struct cifs_sb_info *cifs_sb;
2116
Steve Frenchc32a0b62006-01-12 14:41:28 -08002117 cifs_sb = CIFS_SB(cifsInode->vfs_inode.i_sb);
Steve Frenchad7a2922008-02-07 23:25:02 +00002118 if (cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DIRECT_IO) {
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002119 /* since no page cache to corrupt on directio
Steve Frenchc32a0b62006-01-12 14:41:28 -08002120 we can change size safely */
Steve French4b18f2a2008-04-29 00:06:05 +00002121 return true;
Steve Frenchc32a0b62006-01-12 14:41:28 -08002122 }
2123
Steve Frenchfb8c4b12007-07-10 01:16:18 +00002124 if (i_size_read(&cifsInode->vfs_inode) < end_of_file)
Steve French4b18f2a2008-04-29 00:06:05 +00002125 return true;
Steve French7ba52632007-02-08 18:14:13 +00002126
Steve French4b18f2a2008-04-29 00:06:05 +00002127 return false;
Steve French23e7dd72005-10-20 13:44:56 -07002128 } else
Steve French4b18f2a2008-04-29 00:06:05 +00002129 return true;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002130}
2131
Nick Piggind9414772008-09-24 11:32:59 -04002132static int cifs_write_begin(struct file *file, struct address_space *mapping,
2133 loff_t pos, unsigned len, unsigned flags,
2134 struct page **pagep, void **fsdata)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002135{
Nick Piggind9414772008-09-24 11:32:59 -04002136 pgoff_t index = pos >> PAGE_CACHE_SHIFT;
2137 loff_t offset = pos & (PAGE_CACHE_SIZE - 1);
Jeff Laytona98ee8c2008-11-26 19:32:33 +00002138 loff_t page_start = pos & PAGE_MASK;
2139 loff_t i_size;
2140 struct page *page;
2141 int rc = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002142
Joe Perchesb6b38f72010-04-21 03:50:45 +00002143 cFYI(1, "write_begin from %lld len %d", (long long)pos, len);
Nick Piggind9414772008-09-24 11:32:59 -04002144
Nick Piggin54566b22009-01-04 12:00:53 -08002145 page = grab_cache_page_write_begin(mapping, index, flags);
Jeff Laytona98ee8c2008-11-26 19:32:33 +00002146 if (!page) {
2147 rc = -ENOMEM;
2148 goto out;
2149 }
Nick Piggind9414772008-09-24 11:32:59 -04002150
Jeff Laytona98ee8c2008-11-26 19:32:33 +00002151 if (PageUptodate(page))
2152 goto out;
Steve French8a236262007-03-06 00:31:00 +00002153
Jeff Laytona98ee8c2008-11-26 19:32:33 +00002154 /*
2155 * If we write a full page it will be up to date, no need to read from
2156 * the server. If the write is short, we'll end up doing a sync write
2157 * instead.
2158 */
2159 if (len == PAGE_CACHE_SIZE)
2160 goto out;
2161
2162 /*
2163 * optimize away the read when we have an oplock, and we're not
2164 * expecting to use any of the data we'd be reading in. That
2165 * is, when the page lies beyond the EOF, or straddles the EOF
2166 * and the write will cover all of the existing data.
2167 */
2168 if (CIFS_I(mapping->host)->clientCanCacheRead) {
2169 i_size = i_size_read(mapping->host);
2170 if (page_start >= i_size ||
2171 (offset == 0 && (pos + len) >= i_size)) {
2172 zero_user_segments(page, 0, offset,
2173 offset + len,
2174 PAGE_CACHE_SIZE);
2175 /*
2176 * PageChecked means that the parts of the page
2177 * to which we're not writing are considered up
2178 * to date. Once the data is copied to the
2179 * page, it can be set uptodate.
2180 */
2181 SetPageChecked(page);
2182 goto out;
2183 }
2184 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002185
Nick Piggind9414772008-09-24 11:32:59 -04002186 if ((file->f_flags & O_ACCMODE) != O_WRONLY) {
Jeff Laytona98ee8c2008-11-26 19:32:33 +00002187 /*
2188 * might as well read a page, it is fast enough. If we get
2189 * an error, we don't need to return it. cifs_write_end will
2190 * do a sync write instead since PG_uptodate isn't set.
2191 */
2192 cifs_readpage_worker(file, page, &page_start);
Steve French8a236262007-03-06 00:31:00 +00002193 } else {
2194 /* we could try using another file handle if there is one -
2195 but how would we lock it to prevent close of that handle
2196 racing with this read? In any case
Nick Piggind9414772008-09-24 11:32:59 -04002197 this will be written out by write_end so is fine */
Steve French8a236262007-03-06 00:31:00 +00002198 }
Jeff Laytona98ee8c2008-11-26 19:32:33 +00002199out:
2200 *pagep = page;
2201 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002202}
2203
Suresh Jayaraman85f2d6b2010-07-05 18:13:00 +05302204static int cifs_release_page(struct page *page, gfp_t gfp)
2205{
2206 if (PagePrivate(page))
2207 return 0;
2208
2209 return cifs_fscache_release_page(page, gfp);
2210}
2211
2212static void cifs_invalidate_page(struct page *page, unsigned long offset)
2213{
2214 struct cifsInodeInfo *cifsi = CIFS_I(page->mapping->host);
2215
2216 if (offset == 0)
2217 cifs_fscache_invalidate_page(page, &cifsi->vfs_inode);
2218}
2219
Tejun Heo9b646972010-07-20 22:09:02 +02002220void cifs_oplock_break(struct work_struct *work)
Jeff Layton3bc303c2009-09-21 06:47:50 -04002221{
2222 struct cifsFileInfo *cfile = container_of(work, struct cifsFileInfo,
2223 oplock_break);
Jeff Laytona5e18bc2010-10-11 15:07:18 -04002224 struct inode *inode = cfile->dentry->d_inode;
Jeff Layton3bc303c2009-09-21 06:47:50 -04002225 struct cifsInodeInfo *cinode = CIFS_I(inode);
Jeff Laytoneb4b7562010-10-22 14:52:29 -04002226 int rc = 0;
Jeff Layton3bc303c2009-09-21 06:47:50 -04002227
2228 if (inode && S_ISREG(inode->i_mode)) {
Steve Frenchd54ff732010-04-27 04:38:15 +00002229 if (cinode->clientCanCacheRead)
Al Viro8737c932009-12-24 06:47:55 -05002230 break_lease(inode, O_RDONLY);
Steve Frenchd54ff732010-04-27 04:38:15 +00002231 else
Al Viro8737c932009-12-24 06:47:55 -05002232 break_lease(inode, O_WRONLY);
Jeff Layton3bc303c2009-09-21 06:47:50 -04002233 rc = filemap_fdatawrite(inode->i_mapping);
2234 if (cinode->clientCanCacheRead == 0) {
Jeff Laytoneb4b7562010-10-22 14:52:29 -04002235 rc = filemap_fdatawait(inode->i_mapping);
2236 mapping_set_error(inode->i_mapping, rc);
Jeff Layton3bc303c2009-09-21 06:47:50 -04002237 invalidate_remote_inode(inode);
2238 }
Joe Perchesb6b38f72010-04-21 03:50:45 +00002239 cFYI(1, "Oplock flush inode %p rc %d", inode, rc);
Jeff Layton3bc303c2009-09-21 06:47:50 -04002240 }
2241
2242 /*
2243 * releasing stale oplock after recent reconnect of smb session using
2244 * a now incorrect file handle is not a data integrity issue but do
2245 * not bother sending an oplock release if session to server still is
2246 * disconnected since oplock already released by the server
2247 */
Steve Frenchcdff08e2010-10-21 22:46:14 +00002248 if (!cfile->oplock_break_cancelled) {
Jeff Layton13cfb732010-09-29 19:51:11 -04002249 rc = CIFSSMBLock(0, tlink_tcon(cfile->tlink), cfile->netfid, 0,
2250 0, 0, 0, LOCKING_ANDX_OPLOCK_RELEASE, false);
Joe Perchesb6b38f72010-04-21 03:50:45 +00002251 cFYI(1, "Oplock release rc = %d", rc);
Jeff Layton3bc303c2009-09-21 06:47:50 -04002252 }
Tejun Heo9b646972010-07-20 22:09:02 +02002253
2254 /*
2255 * We might have kicked in before is_valid_oplock_break()
2256 * finished grabbing reference for us. Make sure it's done by
Suresh Jayaraman6573e9b2010-10-18 23:52:18 +05302257 * waiting for cifs_file_list_lock.
Tejun Heo9b646972010-07-20 22:09:02 +02002258 */
Jeff Layton44772882010-10-15 15:34:03 -04002259 spin_lock(&cifs_file_list_lock);
2260 spin_unlock(&cifs_file_list_lock);
Tejun Heo9b646972010-07-20 22:09:02 +02002261
2262 cifs_oplock_break_put(cfile);
Jeff Layton3bc303c2009-09-21 06:47:50 -04002263}
2264
Jeff Layton5f6dbc92010-10-15 15:34:06 -04002265/* must be called while holding cifs_file_list_lock */
Tejun Heo9b646972010-07-20 22:09:02 +02002266void cifs_oplock_break_get(struct cifsFileInfo *cfile)
Jeff Layton3bc303c2009-09-21 06:47:50 -04002267{
Jeff Laytond7c86ff2010-10-11 15:07:19 -04002268 cifs_sb_active(cfile->dentry->d_sb);
Jeff Layton3bc303c2009-09-21 06:47:50 -04002269 cifsFileInfo_get(cfile);
Jeff Layton3bc303c2009-09-21 06:47:50 -04002270}
2271
Tejun Heo9b646972010-07-20 22:09:02 +02002272void cifs_oplock_break_put(struct cifsFileInfo *cfile)
Jeff Layton3bc303c2009-09-21 06:47:50 -04002273{
Jeff Layton3bc303c2009-09-21 06:47:50 -04002274 cifsFileInfo_put(cfile);
Jeff Laytond7c86ff2010-10-11 15:07:19 -04002275 cifs_sb_deactive(cfile->dentry->d_sb);
Jeff Layton3bc303c2009-09-21 06:47:50 -04002276}
2277
Christoph Hellwigf5e54d62006-06-28 04:26:44 -07002278const struct address_space_operations cifs_addr_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002279 .readpage = cifs_readpage,
2280 .readpages = cifs_readpages,
2281 .writepage = cifs_writepage,
Steve French37c0eb42005-10-05 14:50:29 -07002282 .writepages = cifs_writepages,
Nick Piggind9414772008-09-24 11:32:59 -04002283 .write_begin = cifs_write_begin,
2284 .write_end = cifs_write_end,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002285 .set_page_dirty = __set_page_dirty_nobuffers,
Suresh Jayaraman85f2d6b2010-07-05 18:13:00 +05302286 .releasepage = cifs_release_page,
2287 .invalidatepage = cifs_invalidate_page,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002288 /* .sync_page = cifs_sync_page, */
2289 /* .direct_IO = */
2290};
Dave Kleikamp273d81d2006-06-01 19:41:23 +00002291
2292/*
2293 * cifs_readpages requires the server to support a buffer large enough to
2294 * contain the header plus one complete page of data. Otherwise, we need
2295 * to leave cifs_readpages out of the address space operations.
2296 */
Christoph Hellwigf5e54d62006-06-28 04:26:44 -07002297const struct address_space_operations cifs_addr_ops_smallbuf = {
Dave Kleikamp273d81d2006-06-01 19:41:23 +00002298 .readpage = cifs_readpage,
2299 .writepage = cifs_writepage,
2300 .writepages = cifs_writepages,
Nick Piggind9414772008-09-24 11:32:59 -04002301 .write_begin = cifs_write_begin,
2302 .write_end = cifs_write_end,
Dave Kleikamp273d81d2006-06-01 19:41:23 +00002303 .set_page_dirty = __set_page_dirty_nobuffers,
Suresh Jayaraman85f2d6b2010-07-05 18:13:00 +05302304 .releasepage = cifs_release_page,
2305 .invalidatepage = cifs_invalidate_page,
Dave Kleikamp273d81d2006-06-01 19:41:23 +00002306 /* .sync_page = cifs_sync_page, */
2307 /* .direct_IO = */
2308};