Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
L
linux-davinci
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Redmine
Redmine
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Operations
Operations
Metrics
Environments
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
linux
linux-davinci
Commits
f510cfec
Commit
f510cfec
authored
Oct 15, 2007
by
Chris Mason
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Btrfs: Fix extent_buffer and extent_state leaks
Signed-off-by:
Chris Mason
<
chris.mason@oracle.com
>
parent
ae5252bd
Changes
7
Hide whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
139 additions
and
87 deletions
+139
-87
fs/btrfs/ctree.c
fs/btrfs/ctree.c
+8
-2
fs/btrfs/ctree.h
fs/btrfs/ctree.h
+1
-1
fs/btrfs/disk-io.c
fs/btrfs/disk-io.c
+14
-3
fs/btrfs/extent-tree.c
fs/btrfs/extent-tree.c
+58
-68
fs/btrfs/extent_map.c
fs/btrfs/extent_map.c
+53
-11
fs/btrfs/extent_map.h
fs/btrfs/extent_map.h
+2
-0
fs/btrfs/transaction.c
fs/btrfs/transaction.c
+3
-2
No files found.
fs/btrfs/ctree.c
View file @
f510cfec
...
@@ -87,6 +87,7 @@ static int __btrfs_cow_block(struct btrfs_trans_handle *trans,
...
@@ -87,6 +87,7 @@ static int __btrfs_cow_block(struct btrfs_trans_handle *trans,
if
(
IS_ERR
(
cow
))
if
(
IS_ERR
(
cow
))
return
PTR_ERR
(
cow
);
return
PTR_ERR
(
cow
);
cow
->
alloc_addr
=
(
unsigned
long
)
__builtin_return_address
(
0
);
if
(
buf
->
len
!=
root
->
sectorsize
||
cow
->
len
!=
root
->
sectorsize
)
if
(
buf
->
len
!=
root
->
sectorsize
||
cow
->
len
!=
root
->
sectorsize
)
WARN_ON
(
1
);
WARN_ON
(
1
);
...
@@ -132,6 +133,7 @@ int btrfs_cow_block(struct btrfs_trans_handle *trans,
...
@@ -132,6 +133,7 @@ int btrfs_cow_block(struct btrfs_trans_handle *trans,
struct
extent_buffer
**
cow_ret
)
struct
extent_buffer
**
cow_ret
)
{
{
u64
search_start
;
u64
search_start
;
int
ret
;
if
(
trans
->
transaction
!=
root
->
fs_info
->
running_transaction
)
{
if
(
trans
->
transaction
!=
root
->
fs_info
->
running_transaction
)
{
printk
(
KERN_CRIT
"trans %Lu running %Lu
\n
"
,
trans
->
transid
,
printk
(
KERN_CRIT
"trans %Lu running %Lu
\n
"
,
trans
->
transid
,
root
->
fs_info
->
running_transaction
->
transid
);
root
->
fs_info
->
running_transaction
->
transid
);
...
@@ -148,8 +150,10 @@ int btrfs_cow_block(struct btrfs_trans_handle *trans,
...
@@ -148,8 +150,10 @@ int btrfs_cow_block(struct btrfs_trans_handle *trans,
}
}
search_start
=
extent_buffer_blocknr
(
buf
)
&
~
((
u64
)
65535
);
search_start
=
extent_buffer_blocknr
(
buf
)
&
~
((
u64
)
65535
);
ret
urn
__btrfs_cow_block
(
trans
,
root
,
buf
,
parent
,
ret
=
__btrfs_cow_block
(
trans
,
root
,
buf
,
parent
,
parent_slot
,
cow_ret
,
search_start
,
0
);
parent_slot
,
cow_ret
,
search_start
,
0
);
(
*
cow_ret
)
->
alloc_addr
=
(
unsigned
long
)
__builtin_return_address
(
0
);
return
ret
;
}
}
static
int
close_blocks
(
u64
blocknr
,
u64
other
)
static
int
close_blocks
(
u64
blocknr
,
u64
other
)
...
@@ -1013,8 +1017,10 @@ again:
...
@@ -1013,8 +1017,10 @@ again:
if
(
sret
)
if
(
sret
)
return
sret
;
return
sret
;
b
=
p
->
nodes
[
level
];
b
=
p
->
nodes
[
level
];
if
(
!
b
)
if
(
!
b
)
{
btrfs_release_path
(
NULL
,
p
);
goto
again
;
goto
again
;
}
slot
=
p
->
slots
[
level
];
slot
=
p
->
slots
[
level
];
BUG_ON
(
btrfs_header_nritems
(
b
)
==
1
);
BUG_ON
(
btrfs_header_nritems
(
b
)
==
1
);
}
}
...
...
fs/btrfs/ctree.h
View file @
f510cfec
...
@@ -303,8 +303,8 @@ struct btrfs_fs_info {
...
@@ -303,8 +303,8 @@ struct btrfs_fs_info {
struct
radix_tree_root
pinned_radix
;
struct
radix_tree_root
pinned_radix
;
struct
radix_tree_root
block_group_radix
;
struct
radix_tree_root
block_group_radix
;
struct
radix_tree_root
block_group_data_radix
;
struct
radix_tree_root
block_group_data_radix
;
struct
radix_tree_root
extent_map_radix
;
struct
radix_tree_root
extent_ins_radix
;
struct
radix_tree_root
extent_ins_radix
;
struct
extent_map_tree
free_space_cache
;
u64
generation
;
u64
generation
;
u64
last_trans_committed
;
u64
last_trans_committed
;
struct
btrfs_transaction
*
running_transaction
;
struct
btrfs_transaction
*
running_transaction
;
...
...
fs/btrfs/disk-io.c
View file @
f510cfec
...
@@ -46,18 +46,25 @@ struct extent_buffer *btrfs_find_tree_block(struct btrfs_root *root,
...
@@ -46,18 +46,25 @@ struct extent_buffer *btrfs_find_tree_block(struct btrfs_root *root,
u64
blocknr
)
u64
blocknr
)
{
{
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
return
find_extent_buffer
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
struct
extent_buffer
*
eb
;
eb
=
find_extent_buffer
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
blocknr
*
root
->
sectorsize
,
blocknr
*
root
->
sectorsize
,
root
->
sectorsize
,
GFP_NOFS
);
root
->
sectorsize
,
GFP_NOFS
);
if
(
eb
)
eb
->
alloc_addr
=
(
unsigned
long
)
__builtin_return_address
(
0
);
return
eb
;
}
}
struct
extent_buffer
*
btrfs_find_create_tree_block
(
struct
btrfs_root
*
root
,
struct
extent_buffer
*
btrfs_find_create_tree_block
(
struct
btrfs_root
*
root
,
u64
blocknr
)
u64
blocknr
)
{
{
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
return
alloc_extent_buffer
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
struct
extent_buffer
*
eb
;
eb
=
alloc_extent_buffer
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
blocknr
*
root
->
sectorsize
,
blocknr
*
root
->
sectorsize
,
root
->
sectorsize
,
GFP_NOFS
);
root
->
sectorsize
,
GFP_NOFS
);
eb
->
alloc_addr
=
(
unsigned
long
)
__builtin_return_address
(
0
);
return
eb
;
}
}
struct
extent_map
*
btree_get_extent
(
struct
inode
*
inode
,
struct
page
*
page
,
struct
extent_map
*
btree_get_extent
(
struct
inode
*
inode
,
struct
page
*
page
,
...
@@ -226,6 +233,7 @@ struct extent_buffer *read_tree_block(struct btrfs_root *root, u64 blocknr)
...
@@ -226,6 +233,7 @@ struct extent_buffer *read_tree_block(struct btrfs_root *root, u64 blocknr)
return
NULL
;
return
NULL
;
read_extent_buffer_pages
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
read_extent_buffer_pages
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
buf
,
1
);
buf
,
1
);
buf
->
alloc_addr
=
(
unsigned
long
)
__builtin_return_address
(
0
);
return
buf
;
return
buf
;
}
}
...
@@ -426,7 +434,6 @@ struct btrfs_root *open_ctree(struct super_block *sb)
...
@@ -426,7 +434,6 @@ struct btrfs_root *open_ctree(struct super_block *sb)
}
}
init_bit_radix
(
&
fs_info
->
pinned_radix
);
init_bit_radix
(
&
fs_info
->
pinned_radix
);
init_bit_radix
(
&
fs_info
->
pending_del_radix
);
init_bit_radix
(
&
fs_info
->
pending_del_radix
);
init_bit_radix
(
&
fs_info
->
extent_map_radix
);
init_bit_radix
(
&
fs_info
->
extent_ins_radix
);
init_bit_radix
(
&
fs_info
->
extent_ins_radix
);
INIT_RADIX_TREE
(
&
fs_info
->
fs_roots_radix
,
GFP_NOFS
);
INIT_RADIX_TREE
(
&
fs_info
->
fs_roots_radix
,
GFP_NOFS
);
INIT_RADIX_TREE
(
&
fs_info
->
block_group_radix
,
GFP_KERNEL
);
INIT_RADIX_TREE
(
&
fs_info
->
block_group_radix
,
GFP_KERNEL
);
...
@@ -449,6 +456,8 @@ struct btrfs_root *open_ctree(struct super_block *sb)
...
@@ -449,6 +456,8 @@ struct btrfs_root *open_ctree(struct super_block *sb)
extent_map_tree_init
(
&
BTRFS_I
(
fs_info
->
btree_inode
)
->
extent_tree
,
extent_map_tree_init
(
&
BTRFS_I
(
fs_info
->
btree_inode
)
->
extent_tree
,
fs_info
->
btree_inode
->
i_mapping
,
fs_info
->
btree_inode
->
i_mapping
,
GFP_NOFS
);
GFP_NOFS
);
extent_map_tree_init
(
&
fs_info
->
free_space_cache
,
fs_info
->
btree_inode
->
i_mapping
,
GFP_NOFS
);
fs_info
->
do_barriers
=
1
;
fs_info
->
do_barriers
=
1
;
fs_info
->
closing
=
0
;
fs_info
->
closing
=
0
;
...
@@ -594,8 +603,10 @@ int close_ctree(struct btrfs_root *root)
...
@@ -594,8 +603,10 @@ int close_ctree(struct btrfs_root *root)
if
(
fs_info
->
extent_root
->
node
)
if
(
fs_info
->
extent_root
->
node
)
free_extent_buffer
(
fs_info
->
extent_root
->
node
);
free_extent_buffer
(
fs_info
->
extent_root
->
node
);
if
(
fs_info
->
tree_root
->
node
)
if
(
fs_info
->
tree_root
->
node
)
free_extent_buffer
(
fs_info
->
tree_root
->
node
);
free_extent_buffer
(
fs_info
->
tree_root
->
node
);
free_extent_buffer
(
fs_info
->
sb_buffer
);
free_extent_buffer
(
fs_info
->
sb_buffer
);
truncate_inode_pages
(
fs_info
->
btree_inode
->
i_mapping
,
0
);
truncate_inode_pages
(
fs_info
->
btree_inode
->
i_mapping
,
0
);
iput
(
fs_info
->
btree_inode
);
iput
(
fs_info
->
btree_inode
);
...
...
fs/btrfs/extent-tree.c
View file @
f510cfec
...
@@ -34,21 +34,19 @@ static int cache_block_group(struct btrfs_root *root,
...
@@ -34,21 +34,19 @@ static int cache_block_group(struct btrfs_root *root,
int
ret
;
int
ret
;
struct
btrfs_key
key
;
struct
btrfs_key
key
;
struct
extent_buffer
*
leaf
;
struct
extent_buffer
*
leaf
;
struct
radix_tree_root
*
extent_radix
;
struct
extent_map_tree
*
free_space_cache
;
int
slot
;
int
slot
;
u64
i
;
u64
last
=
0
;
u64
last
=
0
;
u64
hole_size
;
u64
hole_size
;
u64
first_free
;
u64
first_free
;
int
found
=
0
;
int
found
=
0
;
root
=
root
->
fs_info
->
extent_root
;
root
=
root
->
fs_info
->
extent_root
;
extent_radix
=
&
root
->
fs_info
->
extent_map_radix
;
free_space_cache
=
&
root
->
fs_info
->
free_space_cache
;
if
(
block_group
->
cached
)
if
(
block_group
->
cached
)
return
0
;
return
0
;
if
(
block_group
->
data
)
return
0
;
path
=
btrfs_alloc_path
();
path
=
btrfs_alloc_path
();
if
(
!
path
)
if
(
!
path
)
return
-
ENOMEM
;
return
-
ENOMEM
;
...
@@ -98,9 +96,11 @@ static int cache_block_group(struct btrfs_root *root,
...
@@ -98,9 +96,11 @@ static int cache_block_group(struct btrfs_root *root,
last
=
first_free
;
last
=
first_free
;
found
=
1
;
found
=
1
;
}
}
hole_size
=
key
.
objectid
-
last
;
if
(
key
.
objectid
>
last
)
{
for
(
i
=
0
;
i
<
hole_size
;
i
++
)
{
hole_size
=
key
.
objectid
-
last
;
set_radix_bit
(
extent_radix
,
last
+
i
);
set_extent_dirty
(
free_space_cache
,
last
,
last
+
hole_size
-
1
,
GFP_NOFS
);
}
}
last
=
key
.
objectid
+
key
.
offset
;
last
=
key
.
objectid
+
key
.
offset
;
}
}
...
@@ -114,9 +114,8 @@ next:
...
@@ -114,9 +114,8 @@ next:
block_group
->
key
.
offset
>
last
)
{
block_group
->
key
.
offset
>
last
)
{
hole_size
=
block_group
->
key
.
objectid
+
hole_size
=
block_group
->
key
.
objectid
+
block_group
->
key
.
offset
-
last
;
block_group
->
key
.
offset
-
last
;
for
(
i
=
0
;
i
<
hole_size
;
i
++
)
{
set_extent_dirty
(
free_space_cache
,
last
,
set_radix_bit
(
extent_radix
,
last
+
i
);
last
+
hole_size
-
1
,
GFP_NOFS
);
}
}
}
block_group
->
cached
=
1
;
block_group
->
cached
=
1
;
err:
err:
...
@@ -150,47 +149,33 @@ struct btrfs_block_group_cache *btrfs_lookup_block_group(struct
...
@@ -150,47 +149,33 @@ struct btrfs_block_group_cache *btrfs_lookup_block_group(struct
return
NULL
;
return
NULL
;
}
}
static
u64
leaf_range
(
struct
btrfs_root
*
root
)
{
u64
size
=
BTRFS_LEAF_DATA_SIZE
(
root
);
do_div
(
size
,
sizeof
(
struct
btrfs_extent_item
)
+
sizeof
(
struct
btrfs_item
));
return
size
;
}
static
u64
find_search_start
(
struct
btrfs_root
*
root
,
static
u64
find_search_start
(
struct
btrfs_root
*
root
,
struct
btrfs_block_group_cache
**
cache_ret
,
struct
btrfs_block_group_cache
**
cache_ret
,
u64
search_start
,
int
num
)
u64
search_start
,
int
num
,
int
data
)
{
{
unsigned
long
gang
[
8
];
int
ret
;
int
ret
;
struct
btrfs_block_group_cache
*
cache
=
*
cache_ret
;
struct
btrfs_block_group_cache
*
cache
=
*
cache_ret
;
u64
last
=
max
(
search_start
,
cache
->
key
.
objectid
);
u64
last
=
max
(
search_start
,
cache
->
key
.
objectid
);
u64
start
=
0
;
u64
end
=
0
;
if
(
cache
->
data
)
goto
out
;
again:
again:
ret
=
cache_block_group
(
root
,
cache
);
ret
=
cache_block_group
(
root
,
cache
);
if
(
ret
)
if
(
ret
)
goto
out
;
goto
out
;
while
(
1
)
{
while
(
1
)
{
ret
=
find_first_
radix_bit
(
&
root
->
fs_info
->
extent_map_radix
,
ret
=
find_first_
extent_bit
(
&
root
->
fs_info
->
free_space_cache
,
gang
,
last
,
ARRAY_SIZE
(
gang
)
);
last
,
&
start
,
&
end
,
EXTENT_DIRTY
);
if
(
!
ret
)
if
(
ret
)
goto
out
;
goto
out
;
last
=
gang
[
ret
-
1
]
+
1
;
if
(
num
>
1
)
{
start
=
max
(
last
,
start
);
if
(
ret
!=
ARRAY_SIZE
(
gang
))
{
last
=
end
+
1
;
goto
new_group
;
if
(
end
+
1
-
start
<
num
)
}
continue
;
if
(
gang
[
ret
-
1
]
-
gang
[
0
]
>
leaf_range
(
root
))
{
if
(
start
+
num
>
cache
->
key
.
objectid
+
cache
->
key
.
offset
)
continue
;
}
}
if
(
gang
[
0
]
>=
cache
->
key
.
objectid
+
cache
->
key
.
offset
)
{
goto
new_group
;
goto
new_group
;
}
return
start
;
return
gang
[
0
];
}
}
out:
out:
return
max
(
cache
->
last_alloc
,
search_start
);
return
max
(
cache
->
last_alloc
,
search_start
);
...
@@ -202,7 +187,7 @@ new_group:
...
@@ -202,7 +187,7 @@ new_group:
return
max
((
*
cache_ret
)
->
last_alloc
,
search_start
);
return
max
((
*
cache_ret
)
->
last_alloc
,
search_start
);
}
}
cache
=
btrfs_find_block_group
(
root
,
cache
,
cache
=
btrfs_find_block_group
(
root
,
cache
,
last
+
cache
->
key
.
offset
-
1
,
0
,
0
);
last
+
cache
->
key
.
offset
-
1
,
data
,
0
);
*
cache_ret
=
cache
;
*
cache_ret
=
cache
;
goto
again
;
goto
again
;
}
}
...
@@ -625,7 +610,6 @@ static int update_block_group(struct btrfs_trans_handle *trans,
...
@@ -625,7 +610,6 @@ static int update_block_group(struct btrfs_trans_handle *trans,
u64
total
=
num
;
u64
total
=
num
;
u64
old_val
;
u64
old_val
;
u64
block_in_group
;
u64
block_in_group
;
u64
i
;
int
ret
;
int
ret
;
while
(
total
)
{
while
(
total
)
{
...
@@ -644,12 +628,6 @@ static int update_block_group(struct btrfs_trans_handle *trans,
...
@@ -644,12 +628,6 @@ static int update_block_group(struct btrfs_trans_handle *trans,
if
(
alloc
)
{
if
(
alloc
)
{
if
(
blocknr
>
cache
->
last_alloc
)
if
(
blocknr
>
cache
->
last_alloc
)
cache
->
last_alloc
=
blocknr
;
cache
->
last_alloc
=
blocknr
;
if
(
!
cache
->
data
)
{
for
(
i
=
0
;
i
<
num
;
i
++
)
{
clear_radix_bit
(
&
info
->
extent_map_radix
,
blocknr
+
i
);
}
}
if
(
cache
->
data
!=
data
&&
if
(
cache
->
data
!=
data
&&
old_val
<
(
cache
->
key
.
offset
>>
1
))
{
old_val
<
(
cache
->
key
.
offset
>>
1
))
{
cache
->
data
=
data
;
cache
->
data
=
data
;
...
@@ -677,11 +655,10 @@ static int update_block_group(struct btrfs_trans_handle *trans,
...
@@ -677,11 +655,10 @@ static int update_block_group(struct btrfs_trans_handle *trans,
old_val
-=
num
;
old_val
-=
num
;
if
(
blocknr
<
cache
->
first_free
)
if
(
blocknr
<
cache
->
first_free
)
cache
->
first_free
=
blocknr
;
cache
->
first_free
=
blocknr
;
if
(
!
cache
->
data
&&
mark_free
)
{
if
(
mark_free
)
{
for
(
i
=
0
;
i
<
num
;
i
++
)
{
set_extent_dirty
(
&
info
->
free_space_cache
,
set_radix_bit
(
&
info
->
extent_map_radix
,
blocknr
,
blocknr
+
num
-
1
,
blocknr
+
i
);
GFP_NOFS
);
}
}
}
if
(
old_val
<
(
cache
->
key
.
offset
>>
1
)
&&
if
(
old_val
<
(
cache
->
key
.
offset
>>
1
)
&&
old_val
+
num
>=
(
cache
->
key
.
offset
>>
1
))
{
old_val
+
num
>=
(
cache
->
key
.
offset
>>
1
))
{
...
@@ -732,7 +709,9 @@ int btrfs_finish_extent_commit(struct btrfs_trans_handle *trans,
...
@@ -732,7 +709,9 @@ int btrfs_finish_extent_commit(struct btrfs_trans_handle *trans,
int
ret
;
int
ret
;
int
i
;
int
i
;
struct
radix_tree_root
*
pinned_radix
=
&
root
->
fs_info
->
pinned_radix
;
struct
radix_tree_root
*
pinned_radix
=
&
root
->
fs_info
->
pinned_radix
;
struct
radix_tree_root
*
extent_radix
=
&
root
->
fs_info
->
extent_map_radix
;
struct
extent_map_tree
*
free_space_cache
;
free_space_cache
=
&
root
->
fs_info
->
free_space_cache
;
while
(
1
)
{
while
(
1
)
{
ret
=
find_first_radix_bit
(
unpin_radix
,
gang
,
0
,
ret
=
find_first_radix_bit
(
unpin_radix
,
gang
,
0
,
...
@@ -751,8 +730,11 @@ int btrfs_finish_extent_commit(struct btrfs_trans_handle *trans,
...
@@ -751,8 +730,11 @@ int btrfs_finish_extent_commit(struct btrfs_trans_handle *trans,
block_group
->
pinned
--
;
block_group
->
pinned
--
;
if
(
gang
[
i
]
<
block_group
->
last_alloc
)
if
(
gang
[
i
]
<
block_group
->
last_alloc
)
block_group
->
last_alloc
=
gang
[
i
];
block_group
->
last_alloc
=
gang
[
i
];
if
(
!
block_group
->
data
)
if
(
!
block_group
->
data
)
{
set_radix_bit
(
extent_radix
,
gang
[
i
]);
set_extent_dirty
(
free_space_cache
,
gang
[
i
],
gang
[
i
],
GFP_NOFS
);
}
}
}
}
}
}
}
...
@@ -995,6 +977,7 @@ static int find_free_extent(struct btrfs_trans_handle *trans, struct btrfs_root
...
@@ -995,6 +977,7 @@ static int find_free_extent(struct btrfs_trans_handle *trans, struct btrfs_root
struct
btrfs_block_group_cache
*
block_group
;
struct
btrfs_block_group_cache
*
block_group
;
int
full_scan
=
0
;
int
full_scan
=
0
;
int
wrapped
=
0
;
int
wrapped
=
0
;
u64
cached_search_start
=
0
;
WARN_ON
(
num_blocks
<
1
);
WARN_ON
(
num_blocks
<
1
);
btrfs_set_key_type
(
ins
,
BTRFS_EXTENT_ITEM_KEY
);
btrfs_set_key_type
(
ins
,
BTRFS_EXTENT_ITEM_KEY
);
...
@@ -1017,11 +1000,9 @@ static int find_free_extent(struct btrfs_trans_handle *trans, struct btrfs_root
...
@@ -1017,11 +1000,9 @@ static int find_free_extent(struct btrfs_trans_handle *trans, struct btrfs_root
path
=
btrfs_alloc_path
();
path
=
btrfs_alloc_path
();
check_failed:
check_failed:
if
(
!
block_group
->
data
)
search_start
=
find_search_start
(
root
,
&
block_group
,
search_start
=
find_search_start
(
root
,
&
block_group
,
search_start
,
total_needed
,
data
);
search_start
,
total_needed
);
cached_search_start
=
search_start
;
else
if
(
!
full_scan
)
search_start
=
max
(
block_group
->
last_alloc
,
search_start
);
btrfs_init_path
(
path
);
btrfs_init_path
(
path
);
ins
->
objectid
=
search_start
;
ins
->
objectid
=
search_start
;
...
@@ -1097,6 +1078,7 @@ check_failed:
...
@@ -1097,6 +1078,7 @@ check_failed:
start_found
=
1
;
start_found
=
1
;
last_block
=
key
.
objectid
+
key
.
offset
;
last_block
=
key
.
objectid
+
key
.
offset
;
if
(
!
full_scan
&&
last_block
>=
block_group
->
key
.
objectid
+
if
(
!
full_scan
&&
last_block
>=
block_group
->
key
.
objectid
+
block_group
->
key
.
offset
)
{
block_group
->
key
.
offset
)
{
btrfs_release_path
(
root
,
path
);
btrfs_release_path
(
root
,
path
);
...
@@ -1138,6 +1120,9 @@ check_pending:
...
@@ -1138,6 +1120,9 @@ check_pending:
}
}
ins
->
offset
=
num_blocks
;
ins
->
offset
=
num_blocks
;
btrfs_free_path
(
path
);
btrfs_free_path
(
path
);
if
(
0
&&
ins
->
objectid
!=
cached_search_start
)
{
printk
(
"
\t
cached was %Lu found %Lu
\n
"
,
cached_search_start
,
ins
->
objectid
);
}
return
0
;
return
0
;
new_group:
new_group:
...
@@ -1209,6 +1194,10 @@ int btrfs_alloc_extent(struct btrfs_trans_handle *trans,
...
@@ -1209,6 +1194,10 @@ int btrfs_alloc_extent(struct btrfs_trans_handle *trans,
btrfs_set_root_used
(
&
root
->
root_item
,
root_blocks_used
+
btrfs_set_root_used
(
&
root
->
root_item
,
root_blocks_used
+
num_blocks
);
num_blocks
);
clear_extent_dirty
(
&
root
->
fs_info
->
free_space_cache
,
ins
->
objectid
,
ins
->
objectid
+
ins
->
offset
-
1
,
GFP_NOFS
);
if
(
root
==
extent_root
)
{
if
(
root
==
extent_root
)
{
BUG_ON
(
num_blocks
!=
1
);
BUG_ON
(
num_blocks
!=
1
);
set_radix_bit
(
&
root
->
fs_info
->
extent_ins_radix
,
ins
->
objectid
);
set_radix_bit
(
&
root
->
fs_info
->
extent_ins_radix
,
ins
->
objectid
);
...
@@ -1227,6 +1216,7 @@ int btrfs_alloc_extent(struct btrfs_trans_handle *trans,
...
@@ -1227,6 +1216,7 @@ int btrfs_alloc_extent(struct btrfs_trans_handle *trans,
BUG_ON
(
ret
);
BUG_ON
(
ret
);
finish_current_insert
(
trans
,
extent_root
);
finish_current_insert
(
trans
,
extent_root
);
pending_ret
=
del_pending_extents
(
trans
,
extent_root
);
pending_ret
=
del_pending_extents
(
trans
,
extent_root
);
if
(
ret
)
{
if
(
ret
)
{
return
ret
;
return
ret
;
}
}
...
@@ -1265,6 +1255,7 @@ struct extent_buffer *btrfs_alloc_free_block(struct btrfs_trans_handle *trans,
...
@@ -1265,6 +1255,7 @@ struct extent_buffer *btrfs_alloc_free_block(struct btrfs_trans_handle *trans,
return
ERR_PTR
(
-
ENOMEM
);
return
ERR_PTR
(
-
ENOMEM
);
}
}
btrfs_set_buffer_uptodate
(
buf
);
btrfs_set_buffer_uptodate
(
buf
);
buf
->
alloc_addr
=
(
unsigned
long
)
__builtin_return_address
(
0
);
set_extent_dirty
(
&
trans
->
transaction
->
dirty_pages
,
buf
->
start
,
set_extent_dirty
(
&
trans
->
transaction
->
dirty_pages
,
buf
->
start
,
buf
->
start
+
buf
->
len
-
1
,
GFP_NOFS
);
buf
->
start
+
buf
->
len
-
1
,
GFP_NOFS
);
/*
/*
...
@@ -1492,6 +1483,7 @@ int btrfs_drop_snapshot(struct btrfs_trans_handle *trans, struct btrfs_root
...
@@ -1492,6 +1483,7 @@ int btrfs_drop_snapshot(struct btrfs_trans_handle *trans, struct btrfs_root
orig_level
=
level
;
orig_level
=
level
;
if
(
btrfs_disk_key_objectid
(
&
root_item
->
drop_progress
)
==
0
)
{
if
(
btrfs_disk_key_objectid
(
&
root_item
->
drop_progress
)
==
0
)
{
path
->
nodes
[
level
]
=
root
->
node
;
path
->
nodes
[
level
]
=
root
->
node
;
extent_buffer_get
(
root
->
node
);
path
->
slots
[
level
]
=
0
;
path
->
slots
[
level
]
=
0
;
}
else
{
}
else
{
struct
btrfs_key
key
;
struct
btrfs_key
key
;
...
@@ -1524,7 +1516,6 @@ int btrfs_drop_snapshot(struct btrfs_trans_handle *trans, struct btrfs_root
...
@@ -1524,7 +1516,6 @@ int btrfs_drop_snapshot(struct btrfs_trans_handle *trans, struct btrfs_root
if
(
wret
<
0
)
if
(
wret
<
0
)
ret
=
wret
;
ret
=
wret
;
ret
=
-
EAGAIN
;
ret
=
-
EAGAIN
;
extent_buffer_get
(
root
->
node
);
break
;
break
;
}
}
for
(
i
=
0
;
i
<=
orig_level
;
i
++
)
{
for
(
i
=
0
;
i
<=
orig_level
;
i
++
)
{
...
@@ -1562,8 +1553,8 @@ int btrfs_free_block_groups(struct btrfs_fs_info *info)
...
@@ -1562,8 +1553,8 @@ int btrfs_free_block_groups(struct btrfs_fs_info *info)
{
{
int
ret
;
int
ret
;
int
ret2
;
int
ret2
;
u
nsigned
long
gang
[
16
]
;
u
64
start
;
int
i
;
u64
end
;
ret
=
free_block_group_radix
(
&
info
->
block_group_radix
);
ret
=
free_block_group_radix
(
&
info
->
block_group_radix
);
ret2
=
free_block_group_radix
(
&
info
->
block_group_data_radix
);
ret2
=
free_block_group_radix
(
&
info
->
block_group_data_radix
);
...
@@ -1573,13 +1564,12 @@ int btrfs_free_block_groups(struct btrfs_fs_info *info)
...
@@ -1573,13 +1564,12 @@ int btrfs_free_block_groups(struct btrfs_fs_info *info)
return
ret2
;
return
ret2
;
while
(
1
)
{
while
(
1
)
{
ret
=
find_first_
radix_bit
(
&
info
->
extent_map_radix
,
ret
=
find_first_
extent_bit
(
&
info
->
free_space_cache
,
0
,
gang
,
0
,
ARRAY_SIZE
(
gang
)
);
&
start
,
&
end
,
EXTENT_DIRTY
);
if
(
!
ret
)
if
(
ret
)
break
;
break
;
for
(
i
=
0
;
i
<
ret
;
i
++
)
{
clear_extent_dirty
(
&
info
->
free_space_cache
,
start
,
clear_radix_bit
(
&
info
->
extent_map_radix
,
gang
[
i
]);
end
,
GFP_NOFS
);
}
}
}
return
0
;
return
0
;
}
}
...
...
fs/btrfs/extent_map.c
View file @
f510cfec
...
@@ -19,8 +19,13 @@ struct kmem_cache *btrfs_cache_create(const char *name, size_t size,
...
@@ -19,8 +19,13 @@ struct kmem_cache *btrfs_cache_create(const char *name, size_t size,
static
struct
kmem_cache
*
extent_map_cache
;
static
struct
kmem_cache
*
extent_map_cache
;
static
struct
kmem_cache
*
extent_state_cache
;
static
struct
kmem_cache
*
extent_state_cache
;
static
struct
kmem_cache
*
extent_buffer_cache
;
static
struct
kmem_cache
*
extent_buffer_cache
;
static
LIST_HEAD
(
extent_buffers
);
static
LIST_HEAD
(
extent_buffers
);
static
LIST_HEAD
(
buffers
);
static
LIST_HEAD
(
states
);
static
spinlock_t
extent_buffers_lock
;
static
spinlock_t
extent_buffers_lock
;
static
spinlock_t
state_lock
=
SPIN_LOCK_UNLOCKED
;
static
int
nr_extent_buffers
;
static
int
nr_extent_buffers
;
#define MAX_EXTENT_BUFFER_CACHE 128
#define MAX_EXTENT_BUFFER_CACHE 128
...
@@ -48,6 +53,7 @@ void __init extent_map_init(void)
...
@@ -48,6 +53,7 @@ void __init extent_map_init(void)
void
__exit
extent_map_exit
(
void
)
void
__exit
extent_map_exit
(
void
)
{
{
struct
extent_buffer
*
eb
;
struct
extent_buffer
*
eb
;
struct
extent_state
*
state
;
while
(
!
list_empty
(
&
extent_buffers
))
{
while
(
!
list_empty
(
&
extent_buffers
))
{
eb
=
list_entry
(
extent_buffers
.
next
,
eb
=
list_entry
(
extent_buffers
.
next
,
...
@@ -55,6 +61,22 @@ void __exit extent_map_exit(void)
...
@@ -55,6 +61,22 @@ void __exit extent_map_exit(void)
list_del
(
&
eb
->
list
);
list_del
(
&
eb
->
list
);
kmem_cache_free
(
extent_buffer_cache
,
eb
);
kmem_cache_free
(
extent_buffer_cache
,
eb
);
}
}
while
(
!
list_empty
(
&
states
))
{
state
=
list_entry
(
states
.
next
,
struct
extent_state
,
list
);
printk
(
"state leak: start %Lu end %Lu state %lu in tree %d refs %d
\n
"
,
state
->
start
,
state
->
end
,
state
->
state
,
state
->
in_tree
,
atomic_read
(
&
state
->
refs
));
list_del
(
&
state
->
list
);
kmem_cache_free
(
extent_state_cache
,
state
);
}
while
(
!
list_empty
(
&
buffers
))
{
eb
=
list_entry
(
buffers
.
next
,
struct
extent_buffer
,
leak_list
);
printk
(
"buffer leak start %Lu len %lu return %lX
\n
"
,
eb
->
start
,
eb
->
len
,
eb
->
alloc_addr
);
list_del
(
&
eb
->
leak_list
);
kmem_cache_free
(
extent_buffer_cache
,
eb
);
}
if
(
extent_map_cache
)
if
(
extent_map_cache
)
kmem_cache_destroy
(
extent_map_cache
);
kmem_cache_destroy
(
extent_map_cache
);
if
(
extent_state_cache
)
if
(
extent_state_cache
)
...
@@ -101,12 +123,19 @@ EXPORT_SYMBOL(free_extent_map);
...
@@ -101,12 +123,19 @@ EXPORT_SYMBOL(free_extent_map);
struct
extent_state
*
alloc_extent_state
(
gfp_t
mask
)
struct
extent_state
*
alloc_extent_state
(
gfp_t
mask
)
{
{
struct
extent_state
*
state
;
struct
extent_state
*
state
;
unsigned
long
flags
;
state
=
kmem_cache_alloc
(
extent_state_cache
,
mask
);
state
=
kmem_cache_alloc
(
extent_state_cache
,
mask
);
if
(
!
state
||
IS_ERR
(
state
))
if
(
!
state
||
IS_ERR
(
state
))
return
state
;
return
state
;
state
->
state
=
0
;
state
->
state
=
0
;
state
->
in_tree
=
0
;
state
->
in_tree
=
0
;
state
->
private
=
0
;
state
->
private
=
0
;
spin_lock_irqsave
(
&
state_lock
,
flags
);
list_add
(
&
state
->
list
,
&
states
);
spin_unlock_irqrestore
(
&
state_lock
,
flags
);
atomic_set
(
&
state
->
refs
,
1
);
atomic_set
(
&
state
->
refs
,
1
);
init_waitqueue_head
(
&
state
->
wq
);
init_waitqueue_head
(
&
state
->
wq
);
return
state
;
return
state
;
...
@@ -115,10 +144,14 @@ EXPORT_SYMBOL(alloc_extent_state);
...
@@ -115,10 +144,14 @@ EXPORT_SYMBOL(alloc_extent_state);
void
free_extent_state
(
struct
extent_state
*
state
)
void
free_extent_state
(
struct
extent_state
*
state
)
{
{
unsigned
long
flags
;
if
(
!
state
)
if
(
!
state
)
return
;
return
;
if
(
atomic_dec_and_test
(
&
state
->
refs
))
{
if
(
atomic_dec_and_test
(
&
state
->
refs
))
{
WARN_ON
(
state
->
in_tree
);
WARN_ON
(
state
->
in_tree
);
spin_lock_irqsave
(
&
state_lock
,
flags
);
list_del
(
&
state
->
list
);
spin_unlock_irqrestore
(
&
state_lock
,
flags
);
kmem_cache_free
(
extent_state_cache
,
state
);
kmem_cache_free
(
extent_state_cache
,
state
);
}
}
}
}
...
@@ -361,10 +394,6 @@ static int insert_state(struct extent_map_tree *tree,
...
@@ -361,10 +394,6 @@ static int insert_state(struct extent_map_tree *tree,
state
->
state
|=
bits
;
state
->
state
|=
bits
;
state
->
start
=
start
;
state
->
start
=
start
;
state
->
end
=
end
;
state
->
end
=
end
;
if
((
end
&
4095
)
==
0
)
{
printk
(
"insert state %Lu %Lu strange end
\n
"
,
start
,
end
);
WARN_ON
(
1
);
}
node
=
tree_insert
(
&
tree
->
state
,
end
,
&
state
->
rb_node
);
node
=
tree_insert
(
&
tree
->
state
,
end
,
&
state
->
rb_node
);
if
(
node
)
{
if
(
node
)
{
struct
extent_state
*
found
;
struct
extent_state
*
found
;
...
@@ -399,11 +428,7 @@ static int split_state(struct extent_map_tree *tree, struct extent_state *orig,
...
@@ -399,11 +428,7 @@ static int split_state(struct extent_map_tree *tree, struct extent_state *orig,
prealloc
->
end
=
split
-
1
;
prealloc
->
end
=
split
-
1
;
prealloc
->
state
=
orig
->
state
;
prealloc
->
state
=
orig
->
state
;
orig
->
start
=
split
;
orig
->
start
=
split
;
if
((
prealloc
->
end
&
4095
)
==
0
)
{
printk
(
"insert state %Lu %Lu strange end
\n
"
,
prealloc
->
start
,
prealloc
->
end
);
WARN_ON
(
1
);
}
node
=
tree_insert
(
&
tree
->
state
,
prealloc
->
end
,
&
prealloc
->
rb_node
);
node
=
tree_insert
(
&
tree
->
state
,
prealloc
->
end
,
&
prealloc
->
rb_node
);
if
(
node
)
{
if
(
node
)
{
struct
extent_state
*
found
;
struct
extent_state
*
found
;
...
@@ -957,6 +982,7 @@ int find_first_extent_bit(struct extent_map_tree *tree, u64 start,
...
@@ -957,6 +982,7 @@ int find_first_extent_bit(struct extent_map_tree *tree, u64 start,
*
start_ret
=
state
->
start
;
*
start_ret
=
state
->
start
;
*
end_ret
=
state
->
end
;
*
end_ret
=
state
->
end
;
ret
=
0
;
ret
=
0
;
break
;
}
}
node
=
rb_next
(
node
);
node
=
rb_next
(
node
);
if
(
!
node
)
if
(
!
node
)
...
@@ -1877,6 +1903,7 @@ sector_t extent_bmap(struct address_space *mapping, sector_t iblock,
...
@@ -1877,6 +1903,7 @@ sector_t extent_bmap(struct address_space *mapping, sector_t iblock,
static
struct
extent_buffer
*
__alloc_extent_buffer
(
gfp_t
mask
)
static
struct
extent_buffer
*
__alloc_extent_buffer
(
gfp_t
mask
)
{
{
struct
extent_buffer
*
eb
=
NULL
;
struct
extent_buffer
*
eb
=
NULL
;
spin_lock
(
&
extent_buffers_lock
);
spin_lock
(
&
extent_buffers_lock
);
if
(
!
list_empty
(
&
extent_buffers
))
{
if
(
!
list_empty
(
&
extent_buffers
))
{
eb
=
list_entry
(
extent_buffers
.
next
,
struct
extent_buffer
,
eb
=
list_entry
(
extent_buffers
.
next
,
struct
extent_buffer
,
...
@@ -1886,15 +1913,26 @@ static struct extent_buffer *__alloc_extent_buffer(gfp_t mask)
...
@@ -1886,15 +1913,26 @@ static struct extent_buffer *__alloc_extent_buffer(gfp_t mask)
nr_extent_buffers
--
;
nr_extent_buffers
--
;
}
}
spin_unlock
(
&
extent_buffers_lock
);
spin_unlock
(
&
extent_buffers_lock
);
if
(
eb
)
{
if
(
eb
)
{
memset
(
eb
,
0
,
sizeof
(
*
eb
));
memset
(
eb
,
0
,
sizeof
(
*
eb
));
return
eb
;
}
else
{
eb
=
kmem_cache_zalloc
(
extent_buffer_cache
,
mask
);
}
}
return
kmem_cache_zalloc
(
extent_buffer_cache
,
mask
);
spin_lock
(
&
extent_buffers_lock
);
list_add
(
&
eb
->
leak_list
,
&
buffers
);
spin_unlock
(
&
extent_buffers_lock
);
return
eb
;
}
}
static
void
__free_extent_buffer
(
struct
extent_buffer
*
eb
)
static
void
__free_extent_buffer
(
struct
extent_buffer
*
eb
)
{
{
spin_lock
(
&
extent_buffers_lock
);
list_del_init
(
&
eb
->
leak_list
);
spin_unlock
(
&
extent_buffers_lock
);
if
(
nr_extent_buffers
>=
MAX_EXTENT_BUFFER_CACHE
)
{
if
(
nr_extent_buffers
>=
MAX_EXTENT_BUFFER_CACHE
)
{
kmem_cache_free
(
extent_buffer_cache
,
eb
);
kmem_cache_free
(
extent_buffer_cache
,
eb
);
}
else
{
}
else
{
...
@@ -1933,6 +1971,7 @@ struct extent_buffer *alloc_extent_buffer(struct extent_map_tree *tree,
...
@@ -1933,6 +1971,7 @@ struct extent_buffer *alloc_extent_buffer(struct extent_map_tree *tree,
if
(
!
eb
||
IS_ERR
(
eb
))
if
(
!
eb
||
IS_ERR
(
eb
))
return
NULL
;
return
NULL
;
eb
->
alloc_addr
=
__builtin_return_address
(
0
);
eb
->
start
=
start
;
eb
->
start
=
start
;
eb
->
len
=
len
;
eb
->
len
=
len
;
atomic_set
(
&
eb
->
refs
,
1
);
atomic_set
(
&
eb
->
refs
,
1
);
...
@@ -1947,6 +1986,7 @@ struct extent_buffer *alloc_extent_buffer(struct extent_map_tree *tree,
...
@@ -1947,6 +1986,7 @@ struct extent_buffer *alloc_extent_buffer(struct extent_map_tree *tree,
eb
->
start
&=
~
((
u64
)
PAGE_CACHE_SIZE
-
1
);
eb
->
start
&=
~
((
u64
)
PAGE_CACHE_SIZE
-
1
);
goto
fail
;
goto
fail
;
}
}
set_page_extent_mapped
(
p
);
if
(
i
==
0
)
if
(
i
==
0
)
eb
->
first_page
=
p
;
eb
->
first_page
=
p
;
if
(
!
PageUptodate
(
p
))
if
(
!
PageUptodate
(
p
))
...
@@ -1978,6 +2018,7 @@ struct extent_buffer *find_extent_buffer(struct extent_map_tree *tree,
...
@@ -1978,6 +2018,7 @@ struct extent_buffer *find_extent_buffer(struct extent_map_tree *tree,
if
(
!
eb
||
IS_ERR
(
eb
))
if
(
!
eb
||
IS_ERR
(
eb
))
return
NULL
;
return
NULL
;
eb
->
alloc_addr
=
__builtin_return_address
(
0
);
eb
->
start
=
start
;
eb
->
start
=
start
;
eb
->
len
=
len
;
eb
->
len
=
len
;
atomic_set
(
&
eb
->
refs
,
1
);
atomic_set
(
&
eb
->
refs
,
1
);
...
@@ -1992,6 +2033,7 @@ struct extent_buffer *find_extent_buffer(struct extent_map_tree *tree,
...
@@ -1992,6 +2033,7 @@ struct extent_buffer *find_extent_buffer(struct extent_map_tree *tree,
eb
->
start
&=
~
((
u64
)
PAGE_CACHE_SIZE
-
1
);
eb
->
start
&=
~
((
u64
)
PAGE_CACHE_SIZE
-
1
);
goto
fail
;
goto
fail
;
}
}
set_page_extent_mapped
(
p
);
if
(
i
==
0
)
if
(
i
==
0
)
eb
->
first_page
=
p
;
eb
->
first_page
=
p
;
}
}
...
...
fs/btrfs/extent_map.h
View file @
f510cfec
...
@@ -68,7 +68,9 @@ struct extent_buffer {
...
@@ -68,7 +68,9 @@ struct extent_buffer {
atomic_t
refs
;
atomic_t
refs
;
int
flags
;
int
flags
;
struct
list_head
list
;
struct
list_head
list
;
struct
list_head
leak_list
;
struct
page
*
first_page
;
struct
page
*
first_page
;
unsigned
long
alloc_addr
;
};
};
typedef
struct
extent_map
*
(
get_extent_t
)(
struct
inode
*
inode
,
typedef
struct
extent_map
*
(
get_extent_t
)(
struct
inode
*
inode
,
...
...
fs/btrfs/transaction.c
View file @
f510cfec
...
@@ -352,7 +352,7 @@ int btrfs_defrag_root(struct btrfs_root *root, int cacheonly)
...
@@ -352,7 +352,7 @@ int btrfs_defrag_root(struct btrfs_root *root, int cacheonly)
return
0
;
return
0
;
trans
=
btrfs_start_transaction
(
root
,
1
);
trans
=
btrfs_start_transaction
(
root
,
1
);
while
(
1
)
{
while
(
0
)
{
root
->
defrag_running
=
1
;
root
->
defrag_running
=
1
;
ret
=
btrfs_defrag_leaves
(
trans
,
root
,
cacheonly
);
ret
=
btrfs_defrag_leaves
(
trans
,
root
,
cacheonly
);
nr
=
trans
->
blocks_used
;
nr
=
trans
->
blocks_used
;
...
@@ -394,7 +394,7 @@ int btrfs_defrag_dirty_roots(struct btrfs_fs_info *info)
...
@@ -394,7 +394,7 @@ int btrfs_defrag_dirty_roots(struct btrfs_fs_info *info)
for
(
i
=
0
;
i
<
ret
;
i
++
)
{
for
(
i
=
0
;
i
<
ret
;
i
++
)
{
root
=
gang
[
i
];
root
=
gang
[
i
];
last
=
root
->
root_key
.
objectid
+
1
;
last
=
root
->
root_key
.
objectid
+
1
;
//
btrfs_defrag_root(root, 1);
btrfs_defrag_root
(
root
,
1
);
}
}
}
}
// btrfs_defrag_root(info->extent_root, 1);
// btrfs_defrag_root(info->extent_root, 1);
...
@@ -462,6 +462,7 @@ static int drop_dirty_roots(struct btrfs_root *tree_root,
...
@@ -462,6 +462,7 @@ static int drop_dirty_roots(struct btrfs_root *tree_root,
ret
=
btrfs_end_transaction
(
trans
,
tree_root
);
ret
=
btrfs_end_transaction
(
trans
,
tree_root
);
BUG_ON
(
ret
);
BUG_ON
(
ret
);
free_extent_buffer
(
dirty
->
root
->
node
);
kfree
(
dirty
->
root
);
kfree
(
dirty
->
root
);
kfree
(
dirty
);
kfree
(
dirty
);
mutex_unlock
(
&
tree_root
->
fs_info
->
fs_mutex
);
mutex_unlock
(
&
tree_root
->
fs_info
->
fs_mutex
);
...
...
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