diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2015-02-17 18:48:33 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2015-02-17 18:48:33 -0500 |
commit | 66dc830d14a222c9214a8557e9feb1e4a67a3857 (patch) | |
tree | d5bd699150fecfe5b2ebfddd9db651389480937d /fs | |
parent | 05016b0f0a9d900e976db7f50a7761c0aefe5a1c (diff) | |
parent | dbe4e192a234cd6133d86fffb965d0f032c12ccc (diff) |
Merge branch 'iov_iter' of git://git.kernel.org/pub/scm/linux/kernel/git/viro/vfs
Pull iov_iter updates from Al Viro:
"More iov_iter work - missing counterpart of iov_iter_init() for
bvec-backed ones and vfs_read_iter()/vfs_write_iter() - wrappers for
sync calls of ->read_iter()/->write_iter()"
* 'iov_iter' of git://git.kernel.org/pub/scm/linux/kernel/git/viro/vfs:
fs: add vfs_iter_{read,write} helpers
new helper: iov_iter_bvec()
Diffstat (limited to 'fs')
-rw-r--r-- | fs/read_write.c | 46 | ||||
-rw-r--r-- | fs/splice.c | 23 |
2 files changed, 50 insertions, 19 deletions
diff --git a/fs/read_write.c b/fs/read_write.c index 4060691e78f7..8e1b68786d66 100644 --- a/fs/read_write.c +++ b/fs/read_write.c | |||
@@ -333,6 +333,52 @@ out_putf: | |||
333 | } | 333 | } |
334 | #endif | 334 | #endif |
335 | 335 | ||
336 | ssize_t vfs_iter_read(struct file *file, struct iov_iter *iter, loff_t *ppos) | ||
337 | { | ||
338 | struct kiocb kiocb; | ||
339 | ssize_t ret; | ||
340 | |||
341 | if (!file->f_op->read_iter) | ||
342 | return -EINVAL; | ||
343 | |||
344 | init_sync_kiocb(&kiocb, file); | ||
345 | kiocb.ki_pos = *ppos; | ||
346 | kiocb.ki_nbytes = iov_iter_count(iter); | ||
347 | |||
348 | iter->type |= READ; | ||
349 | ret = file->f_op->read_iter(&kiocb, iter); | ||
350 | if (ret == -EIOCBQUEUED) | ||
351 | ret = wait_on_sync_kiocb(&kiocb); | ||
352 | |||
353 | if (ret > 0) | ||
354 | *ppos = kiocb.ki_pos; | ||
355 | return ret; | ||
356 | } | ||
357 | EXPORT_SYMBOL(vfs_iter_read); | ||
358 | |||
359 | ssize_t vfs_iter_write(struct file *file, struct iov_iter *iter, loff_t *ppos) | ||
360 | { | ||
361 | struct kiocb kiocb; | ||
362 | ssize_t ret; | ||
363 | |||
364 | if (!file->f_op->write_iter) | ||
365 | return -EINVAL; | ||
366 | |||
367 | init_sync_kiocb(&kiocb, file); | ||
368 | kiocb.ki_pos = *ppos; | ||
369 | kiocb.ki_nbytes = iov_iter_count(iter); | ||
370 | |||
371 | iter->type |= WRITE; | ||
372 | ret = file->f_op->write_iter(&kiocb, iter); | ||
373 | if (ret == -EIOCBQUEUED) | ||
374 | ret = wait_on_sync_kiocb(&kiocb); | ||
375 | |||
376 | if (ret > 0) | ||
377 | *ppos = kiocb.ki_pos; | ||
378 | return ret; | ||
379 | } | ||
380 | EXPORT_SYMBOL(vfs_iter_write); | ||
381 | |||
336 | /* | 382 | /* |
337 | * rw_verify_area doesn't like huge counts. We limit | 383 | * rw_verify_area doesn't like huge counts. We limit |
338 | * them to something that fits in "int" so that others | 384 | * them to something that fits in "int" so that others |
diff --git a/fs/splice.c b/fs/splice.c index 75c6058eabf2..7968da96bebb 100644 --- a/fs/splice.c +++ b/fs/splice.c | |||
@@ -961,7 +961,6 @@ iter_file_splice_write(struct pipe_inode_info *pipe, struct file *out, | |||
961 | splice_from_pipe_begin(&sd); | 961 | splice_from_pipe_begin(&sd); |
962 | while (sd.total_len) { | 962 | while (sd.total_len) { |
963 | struct iov_iter from; | 963 | struct iov_iter from; |
964 | struct kiocb kiocb; | ||
965 | size_t left; | 964 | size_t left; |
966 | int n, idx; | 965 | int n, idx; |
967 | 966 | ||
@@ -1005,29 +1004,15 @@ iter_file_splice_write(struct pipe_inode_info *pipe, struct file *out, | |||
1005 | left -= this_len; | 1004 | left -= this_len; |
1006 | } | 1005 | } |
1007 | 1006 | ||
1008 | /* ... iov_iter */ | 1007 | iov_iter_bvec(&from, ITER_BVEC | WRITE, array, n, |
1009 | from.type = ITER_BVEC | WRITE; | 1008 | sd.total_len - left); |
1010 | from.bvec = array; | 1009 | ret = vfs_iter_write(out, &from, &sd.pos); |
1011 | from.nr_segs = n; | ||
1012 | from.count = sd.total_len - left; | ||
1013 | from.iov_offset = 0; | ||
1014 | |||
1015 | /* ... and iocb */ | ||
1016 | init_sync_kiocb(&kiocb, out); | ||
1017 | kiocb.ki_pos = sd.pos; | ||
1018 | kiocb.ki_nbytes = sd.total_len - left; | ||
1019 | |||
1020 | /* now, send it */ | ||
1021 | ret = out->f_op->write_iter(&kiocb, &from); | ||
1022 | if (-EIOCBQUEUED == ret) | ||
1023 | ret = wait_on_sync_kiocb(&kiocb); | ||
1024 | |||
1025 | if (ret <= 0) | 1010 | if (ret <= 0) |
1026 | break; | 1011 | break; |
1027 | 1012 | ||
1028 | sd.num_spliced += ret; | 1013 | sd.num_spliced += ret; |
1029 | sd.total_len -= ret; | 1014 | sd.total_len -= ret; |
1030 | *ppos = sd.pos = kiocb.ki_pos; | 1015 | *ppos = sd.pos; |
1031 | 1016 | ||
1032 | /* dismiss the fully eaten buffers, adjust the partial one */ | 1017 | /* dismiss the fully eaten buffers, adjust the partial one */ |
1033 | while (ret) { | 1018 | while (ret) { |