summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--mm/filemap.c24
1 files changed, 10 insertions, 14 deletions
diff --git a/mm/filemap.c b/mm/filemap.c
index 67c4df82aa41..cd863c4ff9ae 100644
--- a/mm/filemap.c
+++ b/mm/filemap.c
@@ -2240,15 +2240,16 @@ static int filemap_update_page(struct kiocb *iocb,
struct inode *inode = mapping->host;
int error;
- if (iocb->ki_flags & IOCB_WAITQ) {
- error = lock_page_async(page, iocb->ki_waitq);
- if (error)
- return error;
- } else {
- if (!trylock_page(page)) {
+ if (!trylock_page(page)) {
+ if (iocb->ki_flags & (IOCB_NOWAIT | IOCB_NOIO))
+ return -EAGAIN;
+ if (!(iocb->ki_flags & IOCB_WAITQ)) {
put_and_wait_on_page_locked(page, TASK_KILLABLE);
return AOP_TRUNCATED_PAGE;
}
+ error = __lock_page_async(page, iocb->ki_waitq);
+ if (error)
+ return error;
}
if (!page->mapping)
@@ -2366,14 +2367,9 @@ got_pages:
}
if (!PageUptodate(page)) {
- if ((iocb->ki_flags & IOCB_NOWAIT) ||
- ((iocb->ki_flags & IOCB_WAITQ) && pvec->nr > 1)) {
- put_page(page);
- pvec->nr--;
- err = -EAGAIN;
- goto err;
- }
-
+ if ((iocb->ki_flags & IOCB_WAITQ) &&
+ pagevec_count(pvec) > 1)
+ iocb->ki_flags |= IOCB_NOWAIT;
err = filemap_update_page(iocb, mapping, iter, page,
pg_pos, pg_count);
if (err) {