aboutsummaryrefslogtreecommitdiffstats
path: root/mm/filemap.c
diff options
context:
space:
mode:
Diffstat (limited to 'mm/filemap.c')
-rw-r--r--mm/filemap.c36
1 files changed, 31 insertions, 5 deletions
diff --git a/mm/filemap.c b/mm/filemap.c
index 35e12d186566..45a2d18df849 100644
--- a/mm/filemap.c
+++ b/mm/filemap.c
@@ -1275,7 +1275,7 @@ generic_file_aio_read(struct kiocb *iocb, const struct iovec *iov,
1275{ 1275{
1276 struct file *filp = iocb->ki_filp; 1276 struct file *filp = iocb->ki_filp;
1277 ssize_t retval; 1277 ssize_t retval;
1278 unsigned long seg; 1278 unsigned long seg = 0;
1279 size_t count; 1279 size_t count;
1280 loff_t *ppos = &iocb->ki_pos; 1280 loff_t *ppos = &iocb->ki_pos;
1281 1281
@@ -1302,21 +1302,47 @@ generic_file_aio_read(struct kiocb *iocb, const struct iovec *iov,
1302 retval = mapping->a_ops->direct_IO(READ, iocb, 1302 retval = mapping->a_ops->direct_IO(READ, iocb,
1303 iov, pos, nr_segs); 1303 iov, pos, nr_segs);
1304 } 1304 }
1305 if (retval > 0) 1305 if (retval > 0) {
1306 *ppos = pos + retval; 1306 *ppos = pos + retval;
1307 if (retval) { 1307 count -= retval;
1308 }
1309
1310 /*
1311 * Btrfs can have a short DIO read if we encounter
1312 * compressed extents, so if there was an error, or if
1313 * we've already read everything we wanted to, or if
1314 * there was a short read because we hit EOF, go ahead
1315 * and return. Otherwise fallthrough to buffered io for
1316 * the rest of the read.
1317 */
1318 if (retval < 0 || !count || *ppos >= size) {
1308 file_accessed(filp); 1319 file_accessed(filp);
1309 goto out; 1320 goto out;
1310 } 1321 }
1311 } 1322 }
1312 } 1323 }
1313 1324
1325 count = retval;
1314 for (seg = 0; seg < nr_segs; seg++) { 1326 for (seg = 0; seg < nr_segs; seg++) {
1315 read_descriptor_t desc; 1327 read_descriptor_t desc;
1328 loff_t offset = 0;
1329
1330 /*
1331 * If we did a short DIO read we need to skip the section of the
1332 * iov that we've already read data into.
1333 */
1334 if (count) {
1335 if (count > iov[seg].iov_len) {
1336 count -= iov[seg].iov_len;
1337 continue;
1338 }
1339 offset = count;
1340 count = 0;
1341 }
1316 1342
1317 desc.written = 0; 1343 desc.written = 0;
1318 desc.arg.buf = iov[seg].iov_base; 1344 desc.arg.buf = iov[seg].iov_base + offset;
1319 desc.count = iov[seg].iov_len; 1345 desc.count = iov[seg].iov_len - offset;
1320 if (desc.count == 0) 1346 if (desc.count == 0)
1321 continue; 1347 continue;
1322 desc.error = 0; 1348 desc.error = 0;