Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openeuler
Kernel
提交
4dc11904
K
Kernel
项目概览
openeuler
/
Kernel
1 年多 前同步成功
通知
8
Star
0
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
DevOps
流水线
流水线任务
计划
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
K
Kernel
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
DevOps
DevOps
流水线
流水线任务
计划
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
流水线任务
提交
Issue看板
提交
4dc11904
编写于
10月 15, 2007
作者:
C
Chris Mason
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
Btrfs: Add an extent buffer LRU to reduce radix tree hits
Signed-off-by:
N
Chris Mason
<
chris.mason@oracle.com
>
上级
e19caa5f
变更
6
显示空白变更内容
内联
并排
Showing
6 changed file
with
128 addition
and
105 deletion
+128
-105
fs/btrfs/ctree.c
fs/btrfs/ctree.c
+0
-3
fs/btrfs/disk-io.c
fs/btrfs/disk-io.c
+11
-14
fs/btrfs/extent-tree.c
fs/btrfs/extent-tree.c
+0
-1
fs/btrfs/extent_map.c
fs/btrfs/extent_map.c
+106
-77
fs/btrfs/extent_map.h
fs/btrfs/extent_map.h
+9
-7
fs/btrfs/transaction.c
fs/btrfs/transaction.c
+2
-3
未找到文件。
fs/btrfs/ctree.c
浏览文件 @
4dc11904
...
@@ -88,8 +88,6 @@ static int __btrfs_cow_block(struct btrfs_trans_handle *trans,
...
@@ -88,8 +88,6 @@ 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
);
copy_extent_buffer
(
cow
,
buf
,
0
,
0
,
cow
->
len
);
copy_extent_buffer
(
cow
,
buf
,
0
,
0
,
cow
->
len
);
btrfs_set_header_bytenr
(
cow
,
cow
->
start
);
btrfs_set_header_bytenr
(
cow
,
cow
->
start
);
btrfs_set_header_generation
(
cow
,
trans
->
transid
);
btrfs_set_header_generation
(
cow
,
trans
->
transid
);
...
@@ -151,7 +149,6 @@ int btrfs_cow_block(struct btrfs_trans_handle *trans,
...
@@ -151,7 +149,6 @@ int btrfs_cow_block(struct btrfs_trans_handle *trans,
search_start
=
buf
->
start
&
~
((
u64
)
BTRFS_BLOCK_GROUP_SIZE
-
1
);
search_start
=
buf
->
start
&
~
((
u64
)
BTRFS_BLOCK_GROUP_SIZE
-
1
);
ret
=
__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
;
return
ret
;
}
}
...
...
fs/btrfs/disk-io.c
浏览文件 @
4dc11904
...
@@ -50,8 +50,6 @@ struct extent_buffer *btrfs_find_tree_block(struct btrfs_root *root,
...
@@ -50,8 +50,6 @@ struct extent_buffer *btrfs_find_tree_block(struct btrfs_root *root,
struct
extent_buffer
*
eb
;
struct
extent_buffer
*
eb
;
eb
=
find_extent_buffer
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
eb
=
find_extent_buffer
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
bytenr
,
blocksize
,
GFP_NOFS
);
bytenr
,
blocksize
,
GFP_NOFS
);
if
(
eb
)
eb
->
alloc_addr
=
(
unsigned
long
)
__builtin_return_address
(
0
);
return
eb
;
return
eb
;
}
}
...
@@ -63,7 +61,6 @@ struct extent_buffer *btrfs_find_create_tree_block(struct btrfs_root *root,
...
@@ -63,7 +61,6 @@ struct extent_buffer *btrfs_find_create_tree_block(struct btrfs_root *root,
eb
=
alloc_extent_buffer
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
eb
=
alloc_extent_buffer
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
bytenr
,
blocksize
,
GFP_NOFS
);
bytenr
,
blocksize
,
GFP_NOFS
);
eb
->
alloc_addr
=
(
unsigned
long
)
__builtin_return_address
(
0
);
return
eb
;
return
eb
;
}
}
...
@@ -234,7 +231,6 @@ struct extent_buffer *read_tree_block(struct btrfs_root *root, u64 bytenr,
...
@@ -234,7 +231,6 @@ struct extent_buffer *read_tree_block(struct btrfs_root *root, u64 bytenr,
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
;
}
}
...
@@ -638,6 +634,7 @@ int close_ctree(struct btrfs_root *root)
...
@@ -638,6 +634,7 @@ int close_ctree(struct btrfs_root *root)
btrfs_free_block_groups
(
root
->
fs_info
);
btrfs_free_block_groups
(
root
->
fs_info
);
del_fs_roots
(
fs_info
);
del_fs_roots
(
fs_info
);
extent_map_tree_cleanup
(
&
BTRFS_I
(
fs_info
->
btree_inode
)
->
extent_tree
);
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
);
kfree
(
fs_info
->
extent_root
);
kfree
(
fs_info
->
extent_root
);
...
@@ -647,20 +644,20 @@ int close_ctree(struct btrfs_root *root)
...
@@ -647,20 +644,20 @@ int close_ctree(struct btrfs_root *root)
int
btrfs_buffer_uptodate
(
struct
extent_buffer
*
buf
)
int
btrfs_buffer_uptodate
(
struct
extent_buffer
*
buf
)
{
{
struct
inode
*
btree_inode
=
buf
->
pages
[
0
]
->
mapping
->
host
;
struct
inode
*
btree_inode
=
buf
->
last_page
->
mapping
->
host
;
return
extent_buffer_uptodate
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
buf
);
return
extent_buffer_uptodate
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
buf
);
}
}
int
btrfs_set_buffer_uptodate
(
struct
extent_buffer
*
buf
)
int
btrfs_set_buffer_uptodate
(
struct
extent_buffer
*
buf
)
{
{
struct
inode
*
btree_inode
=
buf
->
pages
[
0
]
->
mapping
->
host
;
struct
inode
*
btree_inode
=
buf
->
last_page
->
mapping
->
host
;
return
set_extent_buffer_uptodate
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
return
set_extent_buffer_uptodate
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
buf
);
buf
);
}
}
void
btrfs_mark_buffer_dirty
(
struct
extent_buffer
*
buf
)
void
btrfs_mark_buffer_dirty
(
struct
extent_buffer
*
buf
)
{
{
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
pages
[
0
]
->
mapping
->
host
)
->
root
;
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
last_page
->
mapping
->
host
)
->
root
;
u64
transid
=
btrfs_header_generation
(
buf
);
u64
transid
=
btrfs_header_generation
(
buf
);
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
...
@@ -681,7 +678,7 @@ void btrfs_btree_balance_dirty(struct btrfs_root *root, unsigned long nr)
...
@@ -681,7 +678,7 @@ void btrfs_btree_balance_dirty(struct btrfs_root *root, unsigned long nr)
void
btrfs_set_buffer_defrag
(
struct
extent_buffer
*
buf
)
void
btrfs_set_buffer_defrag
(
struct
extent_buffer
*
buf
)
{
{
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
pages
[
0
]
->
mapping
->
host
)
->
root
;
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
last_page
->
mapping
->
host
)
->
root
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
set_extent_bits
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
buf
->
start
,
set_extent_bits
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
buf
->
start
,
buf
->
start
+
buf
->
len
-
1
,
EXTENT_DEFRAG
,
GFP_NOFS
);
buf
->
start
+
buf
->
len
-
1
,
EXTENT_DEFRAG
,
GFP_NOFS
);
...
@@ -689,7 +686,7 @@ void btrfs_set_buffer_defrag(struct extent_buffer *buf)
...
@@ -689,7 +686,7 @@ void btrfs_set_buffer_defrag(struct extent_buffer *buf)
void
btrfs_set_buffer_defrag_done
(
struct
extent_buffer
*
buf
)
void
btrfs_set_buffer_defrag_done
(
struct
extent_buffer
*
buf
)
{
{
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
pages
[
0
]
->
mapping
->
host
)
->
root
;
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
last_page
->
mapping
->
host
)
->
root
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
set_extent_bits
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
buf
->
start
,
set_extent_bits
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
buf
->
start
,
buf
->
start
+
buf
->
len
-
1
,
EXTENT_DEFRAG_DONE
,
buf
->
start
+
buf
->
len
-
1
,
EXTENT_DEFRAG_DONE
,
...
@@ -698,7 +695,7 @@ void btrfs_set_buffer_defrag_done(struct extent_buffer *buf)
...
@@ -698,7 +695,7 @@ void btrfs_set_buffer_defrag_done(struct extent_buffer *buf)
int
btrfs_buffer_defrag
(
struct
extent_buffer
*
buf
)
int
btrfs_buffer_defrag
(
struct
extent_buffer
*
buf
)
{
{
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
pages
[
0
]
->
mapping
->
host
)
->
root
;
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
last_page
->
mapping
->
host
)
->
root
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
return
test_range_bit
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
return
test_range_bit
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
buf
->
start
,
buf
->
start
+
buf
->
len
-
1
,
EXTENT_DEFRAG
,
0
);
buf
->
start
,
buf
->
start
+
buf
->
len
-
1
,
EXTENT_DEFRAG
,
0
);
...
@@ -706,7 +703,7 @@ int btrfs_buffer_defrag(struct extent_buffer *buf)
...
@@ -706,7 +703,7 @@ int btrfs_buffer_defrag(struct extent_buffer *buf)
int
btrfs_buffer_defrag_done
(
struct
extent_buffer
*
buf
)
int
btrfs_buffer_defrag_done
(
struct
extent_buffer
*
buf
)
{
{
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
pages
[
0
]
->
mapping
->
host
)
->
root
;
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
last_page
->
mapping
->
host
)
->
root
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
return
test_range_bit
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
return
test_range_bit
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
buf
->
start
,
buf
->
start
+
buf
->
len
-
1
,
buf
->
start
,
buf
->
start
+
buf
->
len
-
1
,
...
@@ -715,7 +712,7 @@ int btrfs_buffer_defrag_done(struct extent_buffer *buf)
...
@@ -715,7 +712,7 @@ int btrfs_buffer_defrag_done(struct extent_buffer *buf)
int
btrfs_clear_buffer_defrag_done
(
struct
extent_buffer
*
buf
)
int
btrfs_clear_buffer_defrag_done
(
struct
extent_buffer
*
buf
)
{
{
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
pages
[
0
]
->
mapping
->
host
)
->
root
;
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
last_page
->
mapping
->
host
)
->
root
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
return
clear_extent_bits
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
return
clear_extent_bits
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
buf
->
start
,
buf
->
start
+
buf
->
len
-
1
,
buf
->
start
,
buf
->
start
+
buf
->
len
-
1
,
...
@@ -724,7 +721,7 @@ int btrfs_clear_buffer_defrag_done(struct extent_buffer *buf)
...
@@ -724,7 +721,7 @@ int btrfs_clear_buffer_defrag_done(struct extent_buffer *buf)
int
btrfs_clear_buffer_defrag
(
struct
extent_buffer
*
buf
)
int
btrfs_clear_buffer_defrag
(
struct
extent_buffer
*
buf
)
{
{
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
pages
[
0
]
->
mapping
->
host
)
->
root
;
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
last_page
->
mapping
->
host
)
->
root
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
return
clear_extent_bits
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
return
clear_extent_bits
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
buf
->
start
,
buf
->
start
+
buf
->
len
-
1
,
buf
->
start
,
buf
->
start
+
buf
->
len
-
1
,
...
@@ -733,7 +730,7 @@ int btrfs_clear_buffer_defrag(struct extent_buffer *buf)
...
@@ -733,7 +730,7 @@ int btrfs_clear_buffer_defrag(struct extent_buffer *buf)
int
btrfs_read_buffer
(
struct
extent_buffer
*
buf
)
int
btrfs_read_buffer
(
struct
extent_buffer
*
buf
)
{
{
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
pages
[
0
]
->
mapping
->
host
)
->
root
;
struct
btrfs_root
*
root
=
BTRFS_I
(
buf
->
last_page
->
mapping
->
host
)
->
root
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
return
read_extent_buffer_pages
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
return
read_extent_buffer_pages
(
&
BTRFS_I
(
btree_inode
)
->
extent_tree
,
buf
,
1
);
buf
,
1
);
...
...
fs/btrfs/extent-tree.c
浏览文件 @
4dc11904
...
@@ -1176,7 +1176,6 @@ struct extent_buffer *btrfs_alloc_free_block(struct btrfs_trans_handle *trans,
...
@@ -1176,7 +1176,6 @@ 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
);
btrfs_set_buffer_defrag
(
buf
);
btrfs_set_buffer_defrag
(
buf
);
...
...
fs/btrfs/extent_map.c
浏览文件 @
4dc11904
...
@@ -8,6 +8,7 @@
...
@@ -8,6 +8,7 @@
#include <linux/module.h>
#include <linux/module.h>
#include <linux/spinlock.h>
#include <linux/spinlock.h>
#include <linux/blkdev.h>
#include <linux/blkdev.h>
#include <linux/swap.h>
#include "extent_map.h"
#include "extent_map.h"
/* temporary define until extent_map moves out of btrfs */
/* temporary define until extent_map moves out of btrfs */
...
@@ -20,14 +21,11 @@ static struct kmem_cache *extent_map_cache;
...
@@ -20,14 +21,11 @@ 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
(
buffers
);
static
LIST_HEAD
(
buffers
);
static
LIST_HEAD
(
states
);
static
LIST_HEAD
(
states
);
static
spinlock_t
extent_buffers_lock
;
static
spinlock_t
state_lock
=
SPIN_LOCK_UNLOCKED
;
static
spinlock_t
state_lock
=
SPIN_LOCK_UNLOCKED
;
static
int
nr_extent_buffers
;
#define BUFFER_LRU_MAX 64
#define MAX_EXTENT_BUFFER_CACHE 128
struct
tree_entry
{
struct
tree_entry
{
u64
start
;
u64
start
;
...
@@ -47,20 +45,12 @@ void __init extent_map_init(void)
...
@@ -47,20 +45,12 @@ void __init extent_map_init(void)
extent_buffer_cache
=
btrfs_cache_create
(
"extent_buffers"
,
extent_buffer_cache
=
btrfs_cache_create
(
"extent_buffers"
,
sizeof
(
struct
extent_buffer
),
0
,
sizeof
(
struct
extent_buffer
),
0
,
NULL
);
NULL
);
spin_lock_init
(
&
extent_buffers_lock
);
}
}
void
__exit
extent_map_exit
(
void
)
void
__exit
extent_map_exit
(
void
)
{
{
struct
extent_buffer
*
eb
;
struct
extent_state
*
state
;
struct
extent_state
*
state
;
while
(
!
list_empty
(
&
extent_buffers
))
{
eb
=
list_entry
(
extent_buffers
.
next
,
struct
extent_buffer
,
list
);
list_del
(
&
eb
->
list
);
kmem_cache_free
(
extent_buffer_cache
,
eb
);
}
while
(
!
list_empty
(
&
states
))
{
while
(
!
list_empty
(
&
states
))
{
state
=
list_entry
(
states
.
next
,
struct
extent_state
,
list
);
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
));
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
));
...
@@ -68,14 +58,6 @@ void __exit extent_map_exit(void)
...
@@ -68,14 +58,6 @@ void __exit extent_map_exit(void)
kmem_cache_free
(
extent_state_cache
,
state
);
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
);
...
@@ -92,10 +74,25 @@ void extent_map_tree_init(struct extent_map_tree *tree,
...
@@ -92,10 +74,25 @@ void extent_map_tree_init(struct extent_map_tree *tree,
tree
->
state
.
rb_node
=
NULL
;
tree
->
state
.
rb_node
=
NULL
;
tree
->
ops
=
NULL
;
tree
->
ops
=
NULL
;
rwlock_init
(
&
tree
->
lock
);
rwlock_init
(
&
tree
->
lock
);
spin_lock_init
(
&
tree
->
lru_lock
);
tree
->
mapping
=
mapping
;
tree
->
mapping
=
mapping
;
INIT_LIST_HEAD
(
&
tree
->
buffer_lru
);
tree
->
lru_size
=
0
;
}
}
EXPORT_SYMBOL
(
extent_map_tree_init
);
EXPORT_SYMBOL
(
extent_map_tree_init
);
void
extent_map_tree_cleanup
(
struct
extent_map_tree
*
tree
)
{
struct
extent_buffer
*
eb
;
while
(
!
list_empty
(
&
tree
->
buffer_lru
))
{
eb
=
list_entry
(
tree
->
buffer_lru
.
next
,
struct
extent_buffer
,
lru
);
list_del
(
&
eb
->
lru
);
free_extent_buffer
(
eb
);
}
}
EXPORT_SYMBOL
(
extent_map_tree_cleanup
);
struct
extent_map
*
alloc_extent_map
(
gfp_t
mask
)
struct
extent_map
*
alloc_extent_map
(
gfp_t
mask
)
{
{
struct
extent_map
*
em
;
struct
extent_map
*
em
;
...
@@ -1915,66 +1912,99 @@ sector_t extent_bmap(struct address_space *mapping, sector_t iblock,
...
@@ -1915,66 +1912,99 @@ sector_t extent_bmap(struct address_space *mapping, sector_t iblock,
return
(
em
->
block_start
+
start
-
em
->
start
)
>>
inode
->
i_blkbits
;
return
(
em
->
block_start
+
start
-
em
->
start
)
>>
inode
->
i_blkbits
;
}
}
static
struct
extent_buffer
*
__alloc_extent_buffer
(
gfp_t
mask
)
static
int
add_lru
(
struct
extent_map_tree
*
tree
,
struct
extent_buffer
*
eb
)
{
{
struct
extent_buffer
*
eb
=
NULL
;
if
(
list_empty
(
&
eb
->
lru
))
{
extent_buffer_get
(
eb
);
spin_lock
(
&
extent_buffers_lock
);
list_add
(
&
eb
->
lru
,
&
tree
->
buffer_lru
);
if
(
!
list_empty
(
&
extent_buffers
))
{
tree
->
lru_size
++
;
eb
=
list_entry
(
extent_buffers
.
next
,
struct
extent_buffer
,
if
(
tree
->
lru_size
>=
BUFFER_LRU_MAX
)
{
list
);
struct
extent_buffer
*
rm
;
list_del
(
&
eb
->
list
);
rm
=
list_entry
(
tree
->
buffer_lru
.
prev
,
WARN_ON
(
nr_extent_buffers
==
0
);
struct
extent_buffer
,
lru
);
nr_extent_buffers
--
;
tree
->
lru_size
--
;
list_del
(
&
rm
->
lru
);
free_extent_buffer
(
rm
);
}
}
spin_unlock
(
&
extent_buffers_lock
);
}
else
list_move
(
&
eb
->
lru
,
&
tree
->
buffer_lru
);
return
0
;
}
static
struct
extent_buffer
*
find_lru
(
struct
extent_map_tree
*
tree
,
u64
start
,
unsigned
long
len
)
{
struct
list_head
*
lru
=
&
tree
->
buffer_lru
;
struct
list_head
*
cur
=
lru
->
next
;
struct
extent_buffer
*
eb
;
if
(
eb
)
{
if
(
list_empty
(
lru
))
memset
(
eb
,
0
,
sizeof
(
*
eb
));
return
NULL
;
}
else
{
eb
=
kmem_cache_zalloc
(
extent_buffer_cache
,
mask
);
}
spin_lock
(
&
extent_buffers_lock
);
list_add
(
&
eb
->
leak_list
,
&
buffers
);
spin_unlock
(
&
extent_buffers_lock
);
do
{
eb
=
list_entry
(
cur
,
struct
extent_buffer
,
lru
);
if
(
eb
->
start
==
start
&&
eb
->
len
==
len
)
{
extent_buffer_get
(
eb
);
return
eb
;
return
eb
;
}
cur
=
cur
->
next
;
}
while
(
cur
!=
lru
);
return
NULL
;
}
}
static
void
__free_extent_buffer
(
struct
extent_buffer
*
eb
)
static
inline
unsigned
long
num_extent_pages
(
u64
start
,
u64
len
)
{
{
return
((
start
+
len
+
PAGE_CACHE_SIZE
-
1
)
>>
PAGE_CACHE_SHIFT
)
-
spin_lock
(
&
extent_buffers_lock
);
(
start
>>
PAGE_CACHE_SHIFT
);
list_del_init
(
&
eb
->
leak_list
);
spin_unlock
(
&
extent_buffers_lock
);
if
(
nr_extent_buffers
>=
MAX_EXTENT_BUFFER_CACHE
)
{
kmem_cache_free
(
extent_buffer_cache
,
eb
);
}
else
{
spin_lock
(
&
extent_buffers_lock
);
list_add
(
&
eb
->
list
,
&
extent_buffers
);
nr_extent_buffers
++
;
spin_unlock
(
&
extent_buffers_lock
);
}
}
}
static
inline
struct
page
*
extent_buffer_page
(
struct
extent_buffer
*
eb
,
int
i
)
static
inline
struct
page
*
extent_buffer_page
(
struct
extent_buffer
*
eb
,
unsigned
long
i
)
{
{
struct
page
*
p
;
struct
page
*
p
;
if
(
i
<
EXTENT_INLINE_PAGES
)
if
(
i
==
0
)
return
eb
->
pages
[
i
]
;
return
eb
->
last_page
;
i
+=
eb
->
start
>>
PAGE_CACHE_SHIFT
;
i
+=
eb
->
start
>>
PAGE_CACHE_SHIFT
;
p
=
find_get_page
(
eb
->
pages
[
0
]
->
mapping
,
i
);
p
=
find_get_page
(
eb
->
last_page
->
mapping
,
i
);
page_cache_release
(
p
);
page_cache_release
(
p
);
return
p
;
return
p
;
}
}
static
inline
unsigned
long
num_extent_pages
(
u64
start
,
u64
len
)
static
struct
extent_buffer
*
__alloc_extent_buffer
(
struct
extent_map_tree
*
tree
,
u64
start
,
unsigned
long
len
,
gfp_t
mask
)
{
{
return
((
start
+
len
+
PAGE_CACHE_SIZE
-
1
)
>>
PAGE_CACHE_SHIFT
)
-
struct
extent_buffer
*
eb
=
NULL
;
(
start
>>
PAGE_CACHE_SHIFT
);
spin_lock
(
&
tree
->
lru_lock
);
eb
=
find_lru
(
tree
,
start
,
len
);
if
(
eb
)
goto
lru_add
;
spin_unlock
(
&
tree
->
lru_lock
);
if
(
eb
)
{
memset
(
eb
,
0
,
sizeof
(
*
eb
));
}
else
{
eb
=
kmem_cache_zalloc
(
extent_buffer_cache
,
mask
);
}
INIT_LIST_HEAD
(
&
eb
->
lru
);
eb
->
start
=
start
;
eb
->
len
=
len
;
atomic_set
(
&
eb
->
refs
,
1
);
spin_lock
(
&
tree
->
lru_lock
);
lru_add:
add_lru
(
tree
,
eb
);
spin_unlock
(
&
tree
->
lru_lock
);
return
eb
;
}
}
static
void
__free_extent_buffer
(
struct
extent_buffer
*
eb
)
{
kmem_cache_free
(
extent_buffer_cache
,
eb
);
}
struct
extent_buffer
*
alloc_extent_buffer
(
struct
extent_map_tree
*
tree
,
struct
extent_buffer
*
alloc_extent_buffer
(
struct
extent_map_tree
*
tree
,
u64
start
,
unsigned
long
len
,
u64
start
,
unsigned
long
len
,
gfp_t
mask
)
gfp_t
mask
)
...
@@ -1987,14 +2017,12 @@ struct extent_buffer *alloc_extent_buffer(struct extent_map_tree *tree,
...
@@ -1987,14 +2017,12 @@ struct extent_buffer *alloc_extent_buffer(struct extent_map_tree *tree,
struct
address_space
*
mapping
=
tree
->
mapping
;
struct
address_space
*
mapping
=
tree
->
mapping
;
int
uptodate
=
0
;
int
uptodate
=
0
;
eb
=
__alloc_extent_buffer
(
mask
);
eb
=
__alloc_extent_buffer
(
tree
,
start
,
len
,
mask
);
if
(
!
eb
||
IS_ERR
(
eb
))
if
(
!
eb
||
IS_ERR
(
eb
))
return
NULL
;
return
NULL
;
eb
->
alloc_addr
=
(
unsigned
long
)
__builtin_return_address
(
0
);
if
(
eb
->
flags
&
EXTENT_BUFFER_FILLED
)
eb
->
start
=
start
;
return
eb
;
eb
->
len
=
len
;
atomic_set
(
&
eb
->
refs
,
1
);
for
(
i
=
0
;
i
<
num_pages
;
i
++
,
index
++
)
{
for
(
i
=
0
;
i
<
num_pages
;
i
++
,
index
++
)
{
p
=
find_or_create_page
(
mapping
,
index
,
mask
|
__GFP_HIGHMEM
);
p
=
find_or_create_page
(
mapping
,
index
,
mask
|
__GFP_HIGHMEM
);
...
@@ -2008,14 +2036,15 @@ struct extent_buffer *alloc_extent_buffer(struct extent_map_tree *tree,
...
@@ -2008,14 +2036,15 @@ struct extent_buffer *alloc_extent_buffer(struct extent_map_tree *tree,
goto
fail
;
goto
fail
;
}
}
set_page_extent_mapped
(
p
);
set_page_extent_mapped
(
p
);
if
(
i
<
EXTENT_INLINE_PAGES
)
if
(
i
==
0
)
eb
->
pages
[
i
]
=
p
;
eb
->
last_page
=
p
;
if
(
!
PageUptodate
(
p
))
if
(
!
PageUptodate
(
p
))
uptodate
=
0
;
uptodate
=
0
;
unlock_page
(
p
);
unlock_page
(
p
);
}
}
if
(
uptodate
)
if
(
uptodate
)
eb
->
flags
|=
EXTENT_UPTODATE
;
eb
->
flags
|=
EXTENT_UPTODATE
;
eb
->
flags
|=
EXTENT_BUFFER_FILLED
;
return
eb
;
return
eb
;
fail:
fail:
free_extent_buffer
(
eb
);
free_extent_buffer
(
eb
);
...
@@ -2035,14 +2064,12 @@ struct extent_buffer *find_extent_buffer(struct extent_map_tree *tree,
...
@@ -2035,14 +2064,12 @@ struct extent_buffer *find_extent_buffer(struct extent_map_tree *tree,
struct
address_space
*
mapping
=
tree
->
mapping
;
struct
address_space
*
mapping
=
tree
->
mapping
;
int
uptodate
=
1
;
int
uptodate
=
1
;
eb
=
__alloc_extent_buffer
(
mask
);
eb
=
__alloc_extent_buffer
(
tree
,
start
,
len
,
mask
);
if
(
!
eb
||
IS_ERR
(
eb
))
if
(
!
eb
||
IS_ERR
(
eb
))
return
NULL
;
return
NULL
;
eb
->
alloc_addr
=
(
unsigned
long
)
__builtin_return_address
(
0
);
if
(
eb
->
flags
&
EXTENT_BUFFER_FILLED
)
eb
->
start
=
start
;
return
eb
;
eb
->
len
=
len
;
atomic_set
(
&
eb
->
refs
,
1
);
for
(
i
=
0
;
i
<
num_pages
;
i
++
,
index
++
)
{
for
(
i
=
0
;
i
<
num_pages
;
i
++
,
index
++
)
{
p
=
find_lock_page
(
mapping
,
index
);
p
=
find_lock_page
(
mapping
,
index
);
...
@@ -2055,14 +2082,15 @@ struct extent_buffer *find_extent_buffer(struct extent_map_tree *tree,
...
@@ -2055,14 +2082,15 @@ struct extent_buffer *find_extent_buffer(struct extent_map_tree *tree,
goto
fail
;
goto
fail
;
}
}
set_page_extent_mapped
(
p
);
set_page_extent_mapped
(
p
);
if
(
i
<
EXTENT_INLINE_PAGES
)
if
(
i
==
0
)
eb
->
pages
[
i
]
=
p
;
eb
->
last_page
=
p
;
if
(
!
PageUptodate
(
p
))
if
(
!
PageUptodate
(
p
))
uptodate
=
0
;
uptodate
=
0
;
unlock_page
(
p
);
unlock_page
(
p
);
}
}
if
(
uptodate
)
if
(
uptodate
)
eb
->
flags
|=
EXTENT_UPTODATE
;
eb
->
flags
|=
EXTENT_UPTODATE
;
eb
->
flags
|=
EXTENT_BUFFER_FILLED
;
return
eb
;
return
eb
;
fail:
fail:
free_extent_buffer
(
eb
);
free_extent_buffer
(
eb
);
...
@@ -2231,6 +2259,7 @@ int read_extent_buffer_pages(struct extent_map_tree *tree,
...
@@ -2231,6 +2259,7 @@ int read_extent_buffer_pages(struct extent_map_tree *tree,
ret
=
-
EIO
;
ret
=
-
EIO
;
}
}
}
}
if
(
!
ret
)
eb
->
flags
|=
EXTENT_UPTODATE
;
eb
->
flags
|=
EXTENT_UPTODATE
;
return
ret
;
return
ret
;
}
}
...
...
fs/btrfs/extent_map.h
浏览文件 @
4dc11904
...
@@ -16,6 +16,7 @@
...
@@ -16,6 +16,7 @@
#define EXTENT_DELALLOC (1 << 5)
#define EXTENT_DELALLOC (1 << 5)
#define EXTENT_DEFRAG (1 << 6)
#define EXTENT_DEFRAG (1 << 6)
#define EXTENT_DEFRAG_DONE (1 << 7)
#define EXTENT_DEFRAG_DONE (1 << 7)
#define EXTENT_BUFFER_FILLED (1 << 8)
#define EXTENT_IOBITS (EXTENT_LOCKED | EXTENT_WRITEBACK)
#define EXTENT_IOBITS (EXTENT_LOCKED | EXTENT_WRITEBACK)
...
@@ -33,6 +34,9 @@ struct extent_map_tree {
...
@@ -33,6 +34,9 @@ struct extent_map_tree {
struct
address_space
*
mapping
;
struct
address_space
*
mapping
;
rwlock_t
lock
;
rwlock_t
lock
;
struct
extent_map_ops
*
ops
;
struct
extent_map_ops
*
ops
;
spinlock_t
lru_lock
;
struct
list_head
buffer_lru
;
int
lru_size
;
};
};
/* note, this must start with the same fields as fs/extent_map.c:tree_entry */
/* note, this must start with the same fields as fs/extent_map.c:tree_entry */
...
@@ -64,20 +68,17 @@ struct extent_state {
...
@@ -64,20 +68,17 @@ struct extent_state {
struct
list_head
list
;
struct
list_head
list
;
};
};
#define EXTENT_INLINE_PAGES 32
struct
extent_buffer
{
struct
extent_buffer
{
u64
start
;
u64
start
;
unsigned
long
len
;
unsigned
long
len
;
atomic_t
refs
;
int
flags
;
struct
list_head
list
;
struct
list_head
leak_list
;
unsigned
long
alloc_addr
;
char
*
map_token
;
char
*
map_token
;
char
*
kaddr
;
char
*
kaddr
;
unsigned
long
map_start
;
unsigned
long
map_start
;
unsigned
long
map_len
;
unsigned
long
map_len
;
struct
page
*
pages
[
EXTENT_INLINE_PAGES
];
struct
page
*
last_page
;
struct
list_head
lru
;
atomic_t
refs
;
int
flags
;
};
};
typedef
struct
extent_map
*
(
get_extent_t
)(
struct
inode
*
inode
,
typedef
struct
extent_map
*
(
get_extent_t
)(
struct
inode
*
inode
,
...
@@ -88,6 +89,7 @@ typedef struct extent_map *(get_extent_t)(struct inode *inode,
...
@@ -88,6 +89,7 @@ typedef struct extent_map *(get_extent_t)(struct inode *inode,
void
extent_map_tree_init
(
struct
extent_map_tree
*
tree
,
void
extent_map_tree_init
(
struct
extent_map_tree
*
tree
,
struct
address_space
*
mapping
,
gfp_t
mask
);
struct
address_space
*
mapping
,
gfp_t
mask
);
void
extent_map_tree_cleanup
(
struct
extent_map_tree
*
tree
);
struct
extent_map
*
lookup_extent_mapping
(
struct
extent_map_tree
*
tree
,
struct
extent_map
*
lookup_extent_mapping
(
struct
extent_map_tree
*
tree
,
u64
start
,
u64
end
);
u64
start
,
u64
end
);
int
add_extent_mapping
(
struct
extent_map_tree
*
tree
,
int
add_extent_mapping
(
struct
extent_map_tree
*
tree
,
...
...
fs/btrfs/transaction.c
浏览文件 @
4dc11904
...
@@ -443,8 +443,7 @@ static int drop_dirty_roots(struct btrfs_root *tree_root,
...
@@ -443,8 +443,7 @@ static int drop_dirty_roots(struct btrfs_root *tree_root,
BUG_ON
(
ret
);
BUG_ON
(
ret
);
mutex_unlock
(
&
tree_root
->
fs_info
->
fs_mutex
);
mutex_unlock
(
&
tree_root
->
fs_info
->
fs_mutex
);
btrfs_btree_balance_dirty
(
tree_root
,
nr
);
btrfs_btree_balance_dirty
(
tree_root
,
nr
);
schedule
();
cond_resched
();
mutex_lock
(
&
tree_root
->
fs_info
->
fs_mutex
);
mutex_lock
(
&
tree_root
->
fs_info
->
fs_mutex
);
}
}
BUG_ON
(
ret
);
BUG_ON
(
ret
);
...
@@ -471,7 +470,7 @@ static int drop_dirty_roots(struct btrfs_root *tree_root,
...
@@ -471,7 +470,7 @@ static int drop_dirty_roots(struct btrfs_root *tree_root,
mutex_unlock
(
&
tree_root
->
fs_info
->
fs_mutex
);
mutex_unlock
(
&
tree_root
->
fs_info
->
fs_mutex
);
btrfs_btree_balance_dirty
(
tree_root
,
nr
);
btrfs_btree_balance_dirty
(
tree_root
,
nr
);
schedule
();
cond_resched
();
}
}
return
ret
;
return
ret
;
}
}
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录