From 1001b638aaa498739ab30cf7da97db16c1ad1e72 Mon Sep 17 00:00:00 2001 From: Daniel McNeil Date: Sat, 16 Apr 2005 15:25:50 -0700 Subject: [PATCH] --- yaml --- r: 137 b: refs/heads/master c: 29504ff3be784372c4e2f7e31681a3e0292c4d9a h: refs/heads/master i: 135: 7c6f1415b1d06d5349f0468e579c04190a0ae3fc v: v3 --- [refs] | 2 +- trunk/fs/direct-io.c | 20 +++++++++++++++++--- 2 files changed, 18 insertions(+), 4 deletions(-) diff --git a/[refs] b/[refs] index 48c066b53d6a..1e04a804ea99 100644 --- a/[refs] +++ b/[refs] @@ -1,2 +1,2 @@ --- -refs/heads/master: 1f08ad02379530e1c970d3d104343b9907b4d1b4 +refs/heads/master: 29504ff3be784372c4e2f7e31681a3e0292c4d9a diff --git a/trunk/fs/direct-io.c b/trunk/fs/direct-io.c index 5a674a0c7146..1d55e7e67342 100644 --- a/trunk/fs/direct-io.c +++ b/trunk/fs/direct-io.c @@ -66,6 +66,7 @@ struct dio { struct bio *bio; /* bio under assembly */ struct inode *inode; int rw; + loff_t i_size; /* i_size when submitted */ int lock_type; /* doesn't change */ unsigned blkbits; /* doesn't change */ unsigned blkfactor; /* When we're using an alignment which @@ -230,17 +231,29 @@ static void finished_one_bio(struct dio *dio) spin_lock_irqsave(&dio->bio_lock, flags); if (dio->bio_count == 1) { if (dio->is_async) { + ssize_t transferred; + loff_t offset; + /* * Last reference to the dio is going away. * Drop spinlock and complete the DIO. */ spin_unlock_irqrestore(&dio->bio_lock, flags); - dio_complete(dio, dio->block_in_file << dio->blkbits, - dio->result); + + /* Check for short read case */ + transferred = dio->result; + offset = dio->iocb->ki_pos; + + if ((dio->rw == READ) && + ((offset + transferred) > dio->i_size)) + transferred = dio->i_size - offset; + + dio_complete(dio, offset, transferred); + /* Complete AIO later if falling back to buffered i/o */ if (dio->result == dio->size || ((dio->rw == READ) && dio->result)) { - aio_complete(dio->iocb, dio->result, 0); + aio_complete(dio->iocb, transferred, 0); kfree(dio); return; } else { @@ -951,6 +964,7 @@ direct_io_worker(int rw, struct kiocb *iocb, struct inode *inode, dio->page_errors = 0; dio->result = 0; dio->iocb = iocb; + dio->i_size = i_size_read(inode); /* * BIO completion state.