Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
e5a2217e
Commit
e5a2217e
authored
Jul 18, 2008
by
Chris Mason
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix btrfs_wait_ordered_extent_range to properly wait
Signed-off-by:
Chris Mason
<
chris.mason@oracle.com
>
parent
7f3c74fb
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
49 additions
and
25 deletions
+49
-25
fs/btrfs/file-item.c
fs/btrfs/file-item.c
+2
-1
fs/btrfs/inode.c
fs/btrfs/inode.c
+20
-6
fs/btrfs/ordered-data.c
fs/btrfs/ordered-data.c
+27
-18
No files found.
fs/btrfs/file-item.c
View file @
e5a2217e
...
@@ -161,7 +161,8 @@ int btrfs_csum_one_bio(struct btrfs_root *root, struct inode *inode,
...
@@ -161,7 +161,8 @@ int btrfs_csum_one_bio(struct btrfs_root *root, struct inode *inode,
while
(
bio_index
<
bio
->
bi_vcnt
)
{
while
(
bio_index
<
bio
->
bi_vcnt
)
{
offset
=
page_offset
(
bvec
->
bv_page
)
+
bvec
->
bv_offset
;
offset
=
page_offset
(
bvec
->
bv_page
)
+
bvec
->
bv_offset
;
if
(
offset
>=
ordered
->
file_offset
+
ordered
->
len
)
{
if
(
offset
>=
ordered
->
file_offset
+
ordered
->
len
||
offset
<
ordered
->
file_offset
)
{
unsigned
long
bytes_left
;
unsigned
long
bytes_left
;
sums
->
len
=
this_sum_bytes
;
sums
->
len
=
this_sum_bytes
;
this_sum_bytes
=
0
;
this_sum_bytes
=
0
;
...
...
fs/btrfs/inode.c
View file @
e5a2217e
...
@@ -128,7 +128,9 @@ static int cow_file_range(struct inode *inode, u64 start, u64 end)
...
@@ -128,7 +128,9 @@ static int cow_file_range(struct inode *inode, u64 start, u64 end)
goto
out
;
goto
out
;
BUG_ON
(
num_bytes
>
btrfs_super_total_bytes
(
&
root
->
fs_info
->
super_copy
));
BUG_ON
(
num_bytes
>
btrfs_super_total_bytes
(
&
root
->
fs_info
->
super_copy
));
mutex_lock
(
&
BTRFS_I
(
inode
)
->
extent_mutex
);
btrfs_drop_extent_cache
(
inode
,
start
,
start
+
num_bytes
-
1
);
btrfs_drop_extent_cache
(
inode
,
start
,
start
+
num_bytes
-
1
);
mutex_unlock
(
&
BTRFS_I
(
inode
)
->
extent_mutex
);
while
(
num_bytes
>
0
)
{
while
(
num_bytes
>
0
)
{
cur_alloc_size
=
min
(
num_bytes
,
root
->
fs_info
->
max_extent
);
cur_alloc_size
=
min
(
num_bytes
,
root
->
fs_info
->
max_extent
);
...
@@ -144,6 +146,7 @@ static int cow_file_range(struct inode *inode, u64 start, u64 end)
...
@@ -144,6 +146,7 @@ static int cow_file_range(struct inode *inode, u64 start, u64 end)
em
->
len
=
ins
.
offset
;
em
->
len
=
ins
.
offset
;
em
->
block_start
=
ins
.
objectid
;
em
->
block_start
=
ins
.
objectid
;
em
->
bdev
=
root
->
fs_info
->
fs_devices
->
latest_bdev
;
em
->
bdev
=
root
->
fs_info
->
fs_devices
->
latest_bdev
;
mutex_lock
(
&
BTRFS_I
(
inode
)
->
extent_mutex
);
set_bit
(
EXTENT_FLAG_PINNED
,
&
em
->
flags
);
set_bit
(
EXTENT_FLAG_PINNED
,
&
em
->
flags
);
while
(
1
)
{
while
(
1
)
{
spin_lock
(
&
em_tree
->
lock
);
spin_lock
(
&
em_tree
->
lock
);
...
@@ -156,6 +159,7 @@ static int cow_file_range(struct inode *inode, u64 start, u64 end)
...
@@ -156,6 +159,7 @@ static int cow_file_range(struct inode *inode, u64 start, u64 end)
btrfs_drop_extent_cache
(
inode
,
start
,
btrfs_drop_extent_cache
(
inode
,
start
,
start
+
ins
.
offset
-
1
);
start
+
ins
.
offset
-
1
);
}
}
mutex_unlock
(
&
BTRFS_I
(
inode
)
->
extent_mutex
);
cur_alloc_size
=
ins
.
offset
;
cur_alloc_size
=
ins
.
offset
;
ret
=
btrfs_add_ordered_extent
(
inode
,
start
,
ins
.
objectid
,
ret
=
btrfs_add_ordered_extent
(
inode
,
start
,
ins
.
objectid
,
...
@@ -487,6 +491,8 @@ static int btrfs_finish_ordered_io(struct inode *inode, u64 start, u64 end)
...
@@ -487,6 +491,8 @@ static int btrfs_finish_ordered_io(struct inode *inode, u64 start, u64 end)
struct
extent_map_tree
*
em_tree
=
&
BTRFS_I
(
inode
)
->
extent_tree
;
struct
extent_map_tree
*
em_tree
=
&
BTRFS_I
(
inode
)
->
extent_tree
;
struct
extent_map
*
em
;
struct
extent_map
*
em
;
u64
alloc_hint
=
0
;
u64
alloc_hint
=
0
;
u64
clear_start
;
u64
clear_end
;
struct
list_head
list
;
struct
list_head
list
;
struct
btrfs_key
ins
;
struct
btrfs_key
ins
;
int
ret
;
int
ret
;
...
@@ -509,12 +515,14 @@ static int btrfs_finish_ordered_io(struct inode *inode, u64 start, u64 end)
...
@@ -509,12 +515,14 @@ static int btrfs_finish_ordered_io(struct inode *inode, u64 start, u64 end)
ins
.
objectid
=
ordered_extent
->
start
;
ins
.
objectid
=
ordered_extent
->
start
;
ins
.
offset
=
ordered_extent
->
len
;
ins
.
offset
=
ordered_extent
->
len
;
ins
.
type
=
BTRFS_EXTENT_ITEM_KEY
;
ins
.
type
=
BTRFS_EXTENT_ITEM_KEY
;
ret
=
btrfs_alloc_reserved_extent
(
trans
,
root
,
root
->
root_key
.
objectid
,
ret
=
btrfs_alloc_reserved_extent
(
trans
,
root
,
root
->
root_key
.
objectid
,
trans
->
transid
,
inode
->
i_ino
,
trans
->
transid
,
inode
->
i_ino
,
ordered_extent
->
file_offset
,
&
ins
);
ordered_extent
->
file_offset
,
&
ins
);
BUG_ON
(
ret
);
BUG_ON
(
ret
);
mutex_lock
(
&
BTRFS_I
(
inode
)
->
extent_mutex
);
mutex_lock
(
&
BTRFS_I
(
inode
)
->
extent_mutex
);
ret
=
btrfs_drop_extents
(
trans
,
root
,
inode
,
ret
=
btrfs_drop_extents
(
trans
,
root
,
inode
,
ordered_extent
->
file_offset
,
ordered_extent
->
file_offset
,
ordered_extent
->
file_offset
+
ordered_extent
->
file_offset
+
...
@@ -528,13 +536,19 @@ static int btrfs_finish_ordered_io(struct inode *inode, u64 start, u64 end)
...
@@ -528,13 +536,19 @@ static int btrfs_finish_ordered_io(struct inode *inode, u64 start, u64 end)
ordered_extent
->
len
,
0
);
ordered_extent
->
len
,
0
);
BUG_ON
(
ret
);
BUG_ON
(
ret
);
spin_lock
(
&
em_tree
->
lock
);
spin_lock
(
&
em_tree
->
lock
);
em
=
lookup_extent_mapping
(
em_tree
,
ordered_extent
->
file_offset
,
clear_start
=
ordered_extent
->
file_offset
;
ordered_extent
->
len
);
clear_end
=
ordered_extent
->
file_offset
+
ordered_extent
->
len
;
if
(
em
)
{
while
(
clear_start
<
clear_end
)
{
clear_bit
(
EXTENT_FLAG_PINNED
,
&
em
->
flags
);
em
=
lookup_extent_mapping
(
em_tree
,
clear_start
,
free_extent_map
(
em
);
clear_end
-
clear_start
);
if
(
em
)
{
clear_bit
(
EXTENT_FLAG_PINNED
,
&
em
->
flags
);
clear_start
=
em
->
start
+
em
->
len
;
free_extent_map
(
em
);
}
else
{
break
;
}
}
}
spin_unlock
(
&
em_tree
->
lock
);
spin_unlock
(
&
em_tree
->
lock
);
...
...
fs/btrfs/ordered-data.c
View file @
e5a2217e
...
@@ -324,22 +324,37 @@ void btrfs_start_ordered_extent(struct inode *inode,
...
@@ -324,22 +324,37 @@ void btrfs_start_ordered_extent(struct inode *inode,
void
btrfs_wait_ordered_range
(
struct
inode
*
inode
,
u64
start
,
u64
len
)
void
btrfs_wait_ordered_range
(
struct
inode
*
inode
,
u64
start
,
u64
len
)
{
{
u64
end
;
u64
end
;
u64
orig_end
;
u64
wait_end
;
struct
btrfs_ordered_extent
*
ordered
;
struct
btrfs_ordered_extent
*
ordered
;
int
found
;
u64
mask
=
BTRFS_I
(
inode
)
->
root
->
sectorsize
-
1
;
int
should_wait
=
0
;
if
(
start
+
len
<
start
)
{
again:
wait_end
=
(
inode
->
i_size
+
mask
)
&
~
mask
;
if
(
start
+
len
<
start
)
orig_end
=
(
u64
)
-
1
;
end
=
(
u64
)
-
1
;
}
else
{
else
orig_end
=
start
+
len
-
1
;
end
=
start
+
len
-
1
;
wait_end
=
orig_end
;
found
=
0
;
}
/* start IO across the range first to instantiate any delalloc
* extents
*/
#if LINUX_VERSION_CODE < KERNEL_VERSION(2,6,22)
do_sync_file_range
(
file
,
start
,
wait_end
,
SYNC_FILE_RANGE_WRITE
);
#else
do_sync_mapping_range
(
inode
->
i_mapping
,
start
,
wait_end
,
SYNC_FILE_RANGE_WRITE
);
#endif
end
=
orig_end
;
wait_on_extent_writeback
(
&
BTRFS_I
(
inode
)
->
io_tree
,
start
,
orig_end
);
while
(
1
)
{
while
(
1
)
{
ordered
=
btrfs_lookup_first_ordered_extent
(
inode
,
end
);
ordered
=
btrfs_lookup_first_ordered_extent
(
inode
,
end
);
if
(
!
ordered
)
{
if
(
!
ordered
)
{
break
;
break
;
}
}
if
(
ordered
->
file_offset
>
=
start
+
len
)
{
if
(
ordered
->
file_offset
>
orig_end
)
{
btrfs_put_ordered_extent
(
ordered
);
btrfs_put_ordered_extent
(
ordered
);
break
;
break
;
}
}
...
@@ -347,21 +362,15 @@ void btrfs_wait_ordered_range(struct inode *inode, u64 start, u64 len)
...
@@ -347,21 +362,15 @@ void btrfs_wait_ordered_range(struct inode *inode, u64 start, u64 len)
btrfs_put_ordered_extent
(
ordered
);
btrfs_put_ordered_extent
(
ordered
);
break
;
break
;
}
}
btrfs_start_ordered_extent
(
inode
,
ordered
,
should_wait
);
btrfs_start_ordered_extent
(
inode
,
ordered
,
1
);
found
++
;
end
=
ordered
->
file_offset
;
end
=
ordered
->
file_offset
;
btrfs_put_ordered_extent
(
ordered
);
btrfs_put_ordered_extent
(
ordered
);
if
(
end
==
0
)
if
(
end
==
0
||
end
==
start
)
break
;
break
;
end
--
;
end
--
;
}
}
if
(
should_wait
&&
found
)
{
should_wait
=
0
;
goto
again
;
}
}
}
/*
/*
* find an ordered extent corresponding to file_offset. return NULL if
* find an ordered extent corresponding to file_offset. return NULL if
* nothing is found, otherwise take a reference on the extent and return it
* nothing is found, otherwise take a reference on the extent and return it
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment