diff options
author | Josef Bacik <josef@redhat.com> | 2012-06-19 10:59:00 -0400 |
---|---|---|
committer | Chris Mason <chris.mason@fusionio.com> | 2012-07-02 15:36:23 -0400 |
commit | c3473e830074ef04f974f2829690942dd8580619 (patch) | |
tree | 21e5e5117bffcf4cdb01e6985346747aeccd331e /fs/sync.c | |
parent | 597a60fadedf9a40fdff8735054bf772b3dafd57 (diff) | |
download | op-kernel-dev-c3473e830074ef04f974f2829690942dd8580619.zip op-kernel-dev-c3473e830074ef04f974f2829690942dd8580619.tar.gz |
Btrfs: fix dio write vs buffered read race
Miao pointed out there's a problem with mixing dio writes and buffered
reads. If the read happens between us invalidating the page range and
actually locking the extent we can bring in pages into page cache. Then
once the write finishes if somebody tries to read again it will just find
uptodate pages and we'll read stale data. So we need to lock the extent and
check for uptodate bits in the range. If there are uptodate bits we need to
unlock and invalidate again. This will keep this race from happening since
we will hold the extent locked until we create the ordered extent, and then
teh read side always waits for ordered extents. There was also a race in
how we updated i_size, previously we were relying on the generic DIO stuff
to adjust the i_size after the DIO had completed, but this happens outside
of the extent lock which means reads could come in and not see the updated
i_size. So instead move this work into where we create the extents, and
then this way the update ordered i_size stuff works properly in the endio
handlers. Thanks,
Signed-off-by: Josef Bacik <josef@redhat.com>
Diffstat (limited to 'fs/sync.c')
0 files changed, 0 insertions, 0 deletions