The patch titled nfs: Support vector I/O throughout the NFS direct I/O path has been added to the -mm tree. Its filename is nfs-support-vector-i-o-throughout-the-nfs-direct-i-o-path.patch See http://www.zip.com.au/~akpm/linux/patches/stuff/added-to-mm.txt to find out what to do about this ------------------------------------------------------ Subject: nfs: Support vector I/O throughout the NFS direct I/O path From: Chuck Lever <cel@xxxxxxxxxx> Now that the preliminaries are complete, it is safe to loop over all the segments in an iovec, dispatching all of the requests against a single nfs_direct_req. Test plan: Specialized test applications using vectored synchronous and asynchronous I/O. Signed-off-by: Chuck Lever <cel@xxxxxxxxxx> Cc: Trond Myklebust <trond.myklebust@xxxxxxxxxx> Signed-off-by: Andrew Morton <akpm@xxxxxxxx> --- fs/nfs/direct.c | 78 +++++++++++++++++++++++++++++----------------- 1 files changed, 50 insertions(+), 28 deletions(-) diff -puN fs/nfs/direct.c~nfs-support-vector-i-o-throughout-the-nfs-direct-i-o-path fs/nfs/direct.c --- 25/fs/nfs/direct.c~nfs-support-vector-i-o-throughout-the-nfs-direct-i-o-path Fri May 19 11:25:13 2006 +++ 25-akpm/fs/nfs/direct.c Fri May 19 11:25:13 2006 @@ -281,8 +281,6 @@ static int nfs_direct_read_schedule(stru unsigned int pgbase; unsigned int started = 0; - dreq->outstanding++; - pgbase = user_addr & ~PAGE_MASK; do { struct nfs_read_data *data; @@ -351,13 +349,30 @@ static int nfs_direct_read_schedule(stru count -= bytes; } while (count != 0); + return started; +} + +static int nfs_direct_read_schedule_iovec(struct nfs_direct_req *dreq, const struct iovec *iov, unsigned long nr_segs, loff_t pos) +{ + ssize_t started = 0; + unsigned long seg; + + dreq->outstanding++; + + for (seg = 0; seg < nr_segs; seg++) { + unsigned long user_addr = (unsigned long) iov[seg].iov_base; + size_t count = iov[seg].iov_len; + started += nfs_direct_read_schedule(dreq, user_addr, count, pos); + pos += count; + } + if (nfs_direct_dec_outstanding(dreq)) nfs_direct_complete(dreq); return started; } -static ssize_t nfs_direct_read(struct kiocb *iocb, unsigned long user_addr, size_t count, loff_t pos) +static ssize_t nfs_direct_read(struct kiocb *iocb, const struct iovec *iov, unsigned long nr_segs, loff_t pos) { ssize_t result = 0; sigset_t oldset; @@ -375,7 +390,7 @@ static ssize_t nfs_direct_read(struct ki dreq->iocb = iocb; rpc_clnt_sigmask(clnt, &oldset); - if (nfs_direct_read_schedule(dreq, user_addr, count, pos)) + if (nfs_direct_read_schedule_iovec(dreq, iov, nr_segs, pos)) result = nfs_direct_wait(dreq); rpc_clnt_sigunmask(clnt, &oldset); @@ -606,8 +621,6 @@ static int nfs_direct_write_schedule(str unsigned int pgbase; unsigned int started = 0; - dreq->outstanding++; - pgbase = user_addr & ~PAGE_MASK; do { struct nfs_write_data *data; @@ -679,13 +692,31 @@ static int nfs_direct_write_schedule(str count -= bytes; } while (count != 0); + return started; +} + + +static int nfs_direct_write_schedule_iovec(struct nfs_direct_req *dreq, const struct iovec *iov, unsigned long nr_segs, loff_t pos, int sync) +{ + ssize_t started = 0; + unsigned long seg; + + dreq->outstanding++; + + for (seg = 0; seg < nr_segs; seg++) { + unsigned long user_addr = (unsigned long) iov[seg].iov_base; + size_t count = iov[seg].iov_len; + started += nfs_direct_write_schedule(dreq, user_addr, count, pos, sync); + pos += count; + } + if (nfs_direct_dec_outstanding(dreq)) - nfs_direct_write_complete(dreq, inode); + nfs_direct_write_complete(dreq, dreq->inode); return started; } -static ssize_t nfs_direct_write(struct kiocb *iocb, unsigned long user_addr, size_t count, loff_t pos) +static ssize_t nfs_direct_write(struct kiocb *iocb, const struct iovec *iov, unsigned long nr_segs, loff_t pos, size_t count) { ssize_t result = 0; sigset_t oldset; @@ -711,7 +742,7 @@ static ssize_t nfs_direct_write(struct k nfs_begin_data_update(inode); rpc_clnt_sigmask(clnt, &oldset); - if (nfs_direct_write_schedule(dreq, user_addr, count, pos, sync)) + if (nfs_direct_write_schedule_iovec(dreq, iov, nr_segs, pos, sync)) result = nfs_direct_wait(dreq); rpc_clnt_sigunmask(clnt, &oldset); @@ -775,28 +806,22 @@ ssize_t nfs_file_direct_read(struct kioc ssize_t retval; struct file *file = iocb->ki_filp; struct address_space *mapping = file->f_mapping; - /* XXX: temporary */ - const char __user *buf = iov[0].iov_base; - size_t count = iov[0].iov_len; retval = check_access_ok(VERIFY_WRITE, iov, nr_segs); if (retval <= 0) goto out; nfs_add_stats(mapping->host, NFSIOS_DIRECTREADBYTES, retval); - dprintk("nfs: direct read(%s/%s, %lu@%Ld)\n", + dprintk("nfs: direct read(%s/%s, %zd@%Ld)\n", file->f_dentry->d_parent->d_name.name, file->f_dentry->d_name.name, - (unsigned long) count, (long long) pos); - - if (nr_segs != 1) - return -EINVAL; + retval, (long long) pos); retval = nfs_sync_mapping(mapping); if (retval) goto out; - retval = nfs_direct_read(iocb, (unsigned long) buf, count, pos); + retval = nfs_direct_read(iocb, iov, nr_segs, pos); if (retval > 0) iocb->ki_pos = pos + retval; @@ -832,35 +857,32 @@ out: ssize_t nfs_file_direct_write(struct kiocb *iocb, const struct iovec *iov, unsigned long nr_segs, loff_t pos) { - ssize_t retval; + ssize_t retval, count; struct file *file = iocb->ki_filp; struct address_space *mapping = file->f_mapping; - /* XXX: temporary */ - const char __user *buf = iov[0].iov_base; - size_t count = iov[0].iov_len; retval = check_access_ok(VERIFY_READ, iov, nr_segs); if (retval <= 0) goto out; nfs_add_stats(mapping->host, NFSIOS_DIRECTWRITTENBYTES, retval); - dfprintk(VFS, "nfs: direct write(%s/%s, %lu@%Ld)\n", + dfprintk(VFS, "nfs: direct write(%s/%s, %zd@%Ld)\n", file->f_dentry->d_parent->d_name.name, file->f_dentry->d_name.name, - (unsigned long) count, (long long) pos); - - if (nr_segs != 1) - return -EINVAL; + retval, (long long) pos); + count = retval; retval = generic_write_checks(file, &pos, &count, 0); if (retval) goto out; + if (!count) + goto out; /* return 0 */ retval = nfs_sync_mapping(mapping); if (retval) goto out; - retval = nfs_direct_write(iocb, (unsigned long) buf, count, pos); + retval = nfs_direct_write(iocb, iov, nr_segs, pos, count); /* * XXX: nfs_end_data_update() already ensures this file's _ Patches currently in -mm which might be from cel@xxxxxxxxxx are git-nfs.patch nfs-open-code-the-nfs-direct-write-rescheduler.patch nfs-remove-user_addr-and-user_count-from-nfs_direct_req.patch nfs-eliminate-nfs_get_user_pages.patch nfs-alloc-nfs_read-write_data-as-direct-i-o-is-scheduled.patch nfs-check-all-iov-segments-for-correct-memory-access-rights.patch nfs-support-vector-i-o-throughout-the-nfs-direct-i-o-path.patch - To unsubscribe from this list: send the line "unsubscribe mm-commits" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html