aboutsummaryrefslogtreecommitdiffstats
path: root/fs/nfs/direct.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/nfs/direct.c')
-rw-r--r--fs/nfs/direct.c54
1 files changed, 31 insertions, 23 deletions
diff --git a/fs/nfs/direct.c b/fs/nfs/direct.c
index df86e526702f..bcbc213b4033 100644
--- a/fs/nfs/direct.c
+++ b/fs/nfs/direct.c
@@ -7,11 +7,11 @@
7 * 7 *
8 * There are important applications whose performance or correctness 8 * There are important applications whose performance or correctness
9 * depends on uncached access to file data. Database clusters 9 * depends on uncached access to file data. Database clusters
10 * (multiple copies of the same instance running on separate hosts) 10 * (multiple copies of the same instance running on separate hosts)
11 * implement their own cache coherency protocol that subsumes file 11 * implement their own cache coherency protocol that subsumes file
12 * system cache protocols. Applications that process datasets 12 * system cache protocols. Applications that process datasets
13 * considerably larger than the client's memory do not always benefit 13 * considerably larger than the client's memory do not always benefit
14 * from a local cache. A streaming video server, for instance, has no 14 * from a local cache. A streaming video server, for instance, has no
15 * need to cache the contents of a file. 15 * need to cache the contents of a file.
16 * 16 *
17 * When an application requests uncached I/O, all read and write requests 17 * When an application requests uncached I/O, all read and write requests
@@ -34,6 +34,7 @@
34 * 08 Jun 2003 Port to 2.5 APIs --cel 34 * 08 Jun 2003 Port to 2.5 APIs --cel
35 * 31 Mar 2004 Handle direct I/O without VFS support --cel 35 * 31 Mar 2004 Handle direct I/O without VFS support --cel
36 * 15 Sep 2004 Parallel async reads --cel 36 * 15 Sep 2004 Parallel async reads --cel
37 * 04 May 2005 support O_DIRECT with aio --cel
37 * 38 *
38 */ 39 */
39 40
@@ -67,11 +68,11 @@ static kmem_cache_t *nfs_direct_cachep;
67 */ 68 */
68struct nfs_direct_req { 69struct nfs_direct_req {
69 struct kref kref; /* release manager */ 70 struct kref kref; /* release manager */
70 struct list_head list; /* nfs_read_data structs */ 71 struct list_head list; /* nfs_read/write_data structs */
71 struct file * filp; /* file descriptor */ 72 struct file * filp; /* file descriptor */
72 struct kiocb * iocb; /* controlling i/o request */ 73 struct kiocb * iocb; /* controlling i/o request */
73 wait_queue_head_t wait; /* wait for i/o completion */ 74 wait_queue_head_t wait; /* wait for i/o completion */
74 struct inode * inode; /* target file of I/O */ 75 struct inode * inode; /* target file of i/o */
75 struct page ** pages; /* pages in our buffer */ 76 struct page ** pages; /* pages in our buffer */
76 unsigned int npages; /* count of pages */ 77 unsigned int npages; /* count of pages */
77 atomic_t complete, /* i/os we're waiting for */ 78 atomic_t complete, /* i/os we're waiting for */
@@ -110,7 +111,6 @@ static inline int nfs_get_user_pages(int rw, unsigned long user_addr, size_t siz
110 size_t array_size; 111 size_t array_size;
111 112
112 /* set an arbitrary limit to prevent type overflow */ 113 /* set an arbitrary limit to prevent type overflow */
113 /* XXX: this can probably be as large as INT_MAX */
114 if (size > MAX_DIRECTIO_SIZE) { 114 if (size > MAX_DIRECTIO_SIZE) {
115 *pages = NULL; 115 *pages = NULL;
116 return -EFBIG; 116 return -EFBIG;
@@ -294,7 +294,7 @@ static const struct rpc_call_ops nfs_read_direct_ops = {
294 * For each nfs_read_data struct that was allocated on the list, dispatch 294 * For each nfs_read_data struct that was allocated on the list, dispatch
295 * an NFS READ operation 295 * an NFS READ operation
296 */ 296 */
297static void nfs_direct_read_schedule(struct nfs_direct_req *dreq, unsigned long user_addr, size_t count, loff_t file_offset) 297static void nfs_direct_read_schedule(struct nfs_direct_req *dreq, unsigned long user_addr, size_t count, loff_t pos)
298{ 298{
299 struct file *file = dreq->filp; 299 struct file *file = dreq->filp;
300 struct inode *inode = file->f_mapping->host; 300 struct inode *inode = file->f_mapping->host;
@@ -322,7 +322,7 @@ static void nfs_direct_read_schedule(struct nfs_direct_req *dreq, unsigned long
322 data->cred = ctx->cred; 322 data->cred = ctx->cred;
323 data->args.fh = NFS_FH(inode); 323 data->args.fh = NFS_FH(inode);
324 data->args.context = ctx; 324 data->args.context = ctx;
325 data->args.offset = file_offset; 325 data->args.offset = pos;
326 data->args.pgbase = pgbase; 326 data->args.pgbase = pgbase;
327 data->args.pages = &pages[curpage]; 327 data->args.pages = &pages[curpage];
328 data->args.count = bytes; 328 data->args.count = bytes;
@@ -347,7 +347,7 @@ static void nfs_direct_read_schedule(struct nfs_direct_req *dreq, unsigned long
347 bytes, 347 bytes,
348 (unsigned long long)data->args.offset); 348 (unsigned long long)data->args.offset);
349 349
350 file_offset += bytes; 350 pos += bytes;
351 pgbase += bytes; 351 pgbase += bytes;
352 curpage += pgbase >> PAGE_SHIFT; 352 curpage += pgbase >> PAGE_SHIFT;
353 pgbase &= ~PAGE_MASK; 353 pgbase &= ~PAGE_MASK;
@@ -356,7 +356,7 @@ static void nfs_direct_read_schedule(struct nfs_direct_req *dreq, unsigned long
356 } while (count != 0); 356 } while (count != 0);
357} 357}
358 358
359static ssize_t nfs_direct_read(struct kiocb *iocb, unsigned long user_addr, size_t count, loff_t file_offset, struct page **pages, unsigned int nr_pages) 359static ssize_t nfs_direct_read(struct kiocb *iocb, unsigned long user_addr, size_t count, loff_t pos, struct page **pages, unsigned int nr_pages)
360{ 360{
361 ssize_t result; 361 ssize_t result;
362 sigset_t oldset; 362 sigset_t oldset;
@@ -377,7 +377,7 @@ static ssize_t nfs_direct_read(struct kiocb *iocb, unsigned long user_addr, size
377 377
378 nfs_add_stats(inode, NFSIOS_DIRECTREADBYTES, count); 378 nfs_add_stats(inode, NFSIOS_DIRECTREADBYTES, count);
379 rpc_clnt_sigmask(clnt, &oldset); 379 rpc_clnt_sigmask(clnt, &oldset);
380 nfs_direct_read_schedule(dreq, user_addr, count, file_offset); 380 nfs_direct_read_schedule(dreq, user_addr, count, pos);
381 result = nfs_direct_wait(dreq); 381 result = nfs_direct_wait(dreq);
382 rpc_clnt_sigunmask(clnt, &oldset); 382 rpc_clnt_sigunmask(clnt, &oldset);
383 383
@@ -459,7 +459,7 @@ static const struct rpc_call_ops nfs_write_direct_ops = {
459 * XXX: For now, support only FILE_SYNC writes. Later we may add 459 * XXX: For now, support only FILE_SYNC writes. Later we may add
460 * support for UNSTABLE + COMMIT. 460 * support for UNSTABLE + COMMIT.
461 */ 461 */
462static void nfs_direct_write_schedule(struct nfs_direct_req *dreq, unsigned long user_addr, size_t count, loff_t file_offset) 462static void nfs_direct_write_schedule(struct nfs_direct_req *dreq, unsigned long user_addr, size_t count, loff_t pos)
463{ 463{
464 struct file *file = dreq->filp; 464 struct file *file = dreq->filp;
465 struct inode *inode = file->f_mapping->host; 465 struct inode *inode = file->f_mapping->host;
@@ -487,7 +487,7 @@ static void nfs_direct_write_schedule(struct nfs_direct_req *dreq, unsigned long
487 data->cred = ctx->cred; 487 data->cred = ctx->cred;
488 data->args.fh = NFS_FH(inode); 488 data->args.fh = NFS_FH(inode);
489 data->args.context = ctx; 489 data->args.context = ctx;
490 data->args.offset = file_offset; 490 data->args.offset = pos;
491 data->args.pgbase = pgbase; 491 data->args.pgbase = pgbase;
492 data->args.pages = &pages[curpage]; 492 data->args.pages = &pages[curpage];
493 data->args.count = bytes; 493 data->args.count = bytes;
@@ -513,7 +513,7 @@ static void nfs_direct_write_schedule(struct nfs_direct_req *dreq, unsigned long
513 bytes, 513 bytes,
514 (unsigned long long)data->args.offset); 514 (unsigned long long)data->args.offset);
515 515
516 file_offset += bytes; 516 pos += bytes;
517 pgbase += bytes; 517 pgbase += bytes;
518 curpage += pgbase >> PAGE_SHIFT; 518 curpage += pgbase >> PAGE_SHIFT;
519 pgbase &= ~PAGE_MASK; 519 pgbase &= ~PAGE_MASK;
@@ -522,7 +522,7 @@ static void nfs_direct_write_schedule(struct nfs_direct_req *dreq, unsigned long
522 } while (count != 0); 522 } while (count != 0);
523} 523}
524 524
525static ssize_t nfs_direct_write(struct kiocb *iocb, unsigned long user_addr, size_t count, loff_t file_offset, struct page **pages, int nr_pages) 525static ssize_t nfs_direct_write(struct kiocb *iocb, unsigned long user_addr, size_t count, loff_t pos, struct page **pages, int nr_pages)
526{ 526{
527 ssize_t result; 527 ssize_t result;
528 sigset_t oldset; 528 sigset_t oldset;
@@ -546,7 +546,7 @@ static ssize_t nfs_direct_write(struct kiocb *iocb, unsigned long user_addr, siz
546 nfs_begin_data_update(inode); 546 nfs_begin_data_update(inode);
547 547
548 rpc_clnt_sigmask(clnt, &oldset); 548 rpc_clnt_sigmask(clnt, &oldset);
549 nfs_direct_write_schedule(dreq, user_addr, count, file_offset); 549 nfs_direct_write_schedule(dreq, user_addr, count, pos);
550 result = nfs_direct_wait(dreq); 550 result = nfs_direct_wait(dreq);
551 rpc_clnt_sigunmask(clnt, &oldset); 551 rpc_clnt_sigunmask(clnt, &oldset);
552 552
@@ -557,18 +557,18 @@ static ssize_t nfs_direct_write(struct kiocb *iocb, unsigned long user_addr, siz
557 * nfs_file_direct_read - file direct read operation for NFS files 557 * nfs_file_direct_read - file direct read operation for NFS files
558 * @iocb: target I/O control block 558 * @iocb: target I/O control block
559 * @buf: user's buffer into which to read data 559 * @buf: user's buffer into which to read data
560 * count: number of bytes to read 560 * @count: number of bytes to read
561 * pos: byte offset in file where reading starts 561 * @pos: byte offset in file where reading starts
562 * 562 *
563 * We use this function for direct reads instead of calling 563 * We use this function for direct reads instead of calling
564 * generic_file_aio_read() in order to avoid gfar's check to see if 564 * generic_file_aio_read() in order to avoid gfar's check to see if
565 * the request starts before the end of the file. For that check 565 * the request starts before the end of the file. For that check
566 * to work, we must generate a GETATTR before each direct read, and 566 * to work, we must generate a GETATTR before each direct read, and
567 * even then there is a window between the GETATTR and the subsequent 567 * even then there is a window between the GETATTR and the subsequent
568 * READ where the file size could change. So our preference is simply 568 * READ where the file size could change. Our preference is simply
569 * to do all reads the application wants, and the server will take 569 * to do all reads the application wants, and the server will take
570 * care of managing the end of file boundary. 570 * care of managing the end of file boundary.
571 * 571 *
572 * This function also eliminates unnecessarily updating the file's 572 * This function also eliminates unnecessarily updating the file's
573 * atime locally, as the NFS server sets the file's atime, and this 573 * atime locally, as the NFS server sets the file's atime, and this
574 * client must read the updated atime from the server back into its 574 * client must read the updated atime from the server back into its
@@ -621,8 +621,8 @@ out:
621 * nfs_file_direct_write - file direct write operation for NFS files 621 * nfs_file_direct_write - file direct write operation for NFS files
622 * @iocb: target I/O control block 622 * @iocb: target I/O control block
623 * @buf: user's buffer from which to write data 623 * @buf: user's buffer from which to write data
624 * count: number of bytes to write 624 * @count: number of bytes to write
625 * pos: byte offset in file where writing starts 625 * @pos: byte offset in file where writing starts
626 * 626 *
627 * We use this function for direct writes instead of calling 627 * We use this function for direct writes instead of calling
628 * generic_file_aio_write() in order to avoid taking the inode 628 * generic_file_aio_write() in order to avoid taking the inode
@@ -703,6 +703,10 @@ out:
703 return retval; 703 return retval;
704} 704}
705 705
706/**
707 * nfs_init_directcache - create a slab cache for nfs_direct_req structures
708 *
709 */
706int nfs_init_directcache(void) 710int nfs_init_directcache(void)
707{ 711{
708 nfs_direct_cachep = kmem_cache_create("nfs_direct_cache", 712 nfs_direct_cachep = kmem_cache_create("nfs_direct_cache",
@@ -715,6 +719,10 @@ int nfs_init_directcache(void)
715 return 0; 719 return 0;
716} 720}
717 721
722/**
723 * nfs_init_directcache - destroy the slab cache for nfs_direct_req structures
724 *
725 */
718void nfs_destroy_directcache(void) 726void nfs_destroy_directcache(void)
719{ 727{
720 if (kmem_cache_destroy(nfs_direct_cachep)) 728 if (kmem_cache_destroy(nfs_direct_cachep))