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
afd582ac
Commit
afd582ac
authored
Oct 24, 2011
by
David Sterba
Browse files
Options
Browse Files
Download
Plain Diff
Merge remote-tracking branch 'remotes/josef/for-chris' into btrfs-next-stable
parents
c3b92c87
016fc6a6
Changes
16
Expand all
Hide whitespace changes
Inline
Side-by-side
Showing
16 changed files
with
1557 additions
and
1058 deletions
+1557
-1058
fs/btrfs/btrfs_inode.h
fs/btrfs/btrfs_inode.h
+9
-8
fs/btrfs/ctree.h
fs/btrfs/ctree.h
+27
-29
fs/btrfs/disk-io.c
fs/btrfs/disk-io.c
+4
-4
fs/btrfs/extent-tree.c
fs/btrfs/extent-tree.c
+364
-304
fs/btrfs/extent_io.c
fs/btrfs/extent_io.c
+193
-1
fs/btrfs/extent_io.h
fs/btrfs/extent_io.h
+3
-0
fs/btrfs/file.c
fs/btrfs/file.c
+18
-7
fs/btrfs/free-space-cache.c
fs/btrfs/free-space-cache.c
+545
-357
fs/btrfs/inode-map.c
fs/btrfs/inode-map.c
+4
-2
fs/btrfs/inode.c
fs/btrfs/inode.c
+133
-153
fs/btrfs/ioctl.c
fs/btrfs/ioctl.c
+13
-7
fs/btrfs/relocation.c
fs/btrfs/relocation.c
+7
-12
fs/btrfs/super.c
fs/btrfs/super.c
+153
-92
fs/btrfs/transaction.c
fs/btrfs/transaction.c
+38
-78
fs/btrfs/volumes.c
fs/btrfs/volumes.c
+35
-4
fs/btrfs/xattr.c
fs/btrfs/xattr.c
+11
-0
No files found.
fs/btrfs/btrfs_inode.h
View file @
afd582ac
...
@@ -103,11 +103,6 @@ struct btrfs_inode {
...
@@ -103,11 +103,6 @@ struct btrfs_inode {
*/
*/
u64
delalloc_bytes
;
u64
delalloc_bytes
;
/* total number of bytes that may be used for this inode for
* delalloc
*/
u64
reserved_bytes
;
/*
/*
* the size of the file stored in the metadata on disk. data=ordered
* the size of the file stored in the metadata on disk. data=ordered
* means the in-memory i_size might be larger than the size on disk
* means the in-memory i_size might be larger than the size on disk
...
@@ -115,9 +110,6 @@ struct btrfs_inode {
...
@@ -115,9 +110,6 @@ struct btrfs_inode {
*/
*/
u64
disk_i_size
;
u64
disk_i_size
;
/* flags field from the on disk inode */
u32
flags
;
/*
/*
* if this is a directory then index_cnt is the counter for the index
* if this is a directory then index_cnt is the counter for the index
* number for new files that are created
* number for new files that are created
...
@@ -131,6 +123,15 @@ struct btrfs_inode {
...
@@ -131,6 +123,15 @@ struct btrfs_inode {
*/
*/
u64
last_unlink_trans
;
u64
last_unlink_trans
;
/*
* Number of bytes outstanding that are going to need csums. This is
* used in ENOSPC accounting.
*/
u64
csum_bytes
;
/* flags field from the on disk inode */
u32
flags
;
/*
/*
* Counters to keep track of the number of extent item's we may use due
* Counters to keep track of the number of extent item's we may use due
* to delalloc and such. outstanding_extents is the number of extent
* to delalloc and such. outstanding_extents is the number of extent
...
...
fs/btrfs/ctree.h
View file @
afd582ac
...
@@ -30,6 +30,7 @@
...
@@ -30,6 +30,7 @@
#include <linux/kobject.h>
#include <linux/kobject.h>
#include <trace/events/btrfs.h>
#include <trace/events/btrfs.h>
#include <asm/kmap_types.h>
#include <asm/kmap_types.h>
#include <linux/pagemap.h>
#include "extent_io.h"
#include "extent_io.h"
#include "extent_map.h"
#include "extent_map.h"
#include "async-thread.h"
#include "async-thread.h"
...
@@ -772,14 +773,8 @@ struct btrfs_space_info {
...
@@ -772,14 +773,8 @@ struct btrfs_space_info {
struct
btrfs_block_rsv
{
struct
btrfs_block_rsv
{
u64
size
;
u64
size
;
u64
reserved
;
u64
reserved
;
u64
freed
[
2
];
struct
btrfs_space_info
*
space_info
;
struct
btrfs_space_info
*
space_info
;
struct
list_head
list
;
spinlock_t
lock
;
spinlock_t
lock
;
atomic_t
usage
;
unsigned
int
priority
:
8
;
unsigned
int
durable
:
1
;
unsigned
int
refill_used
:
1
;
unsigned
int
full
:
1
;
unsigned
int
full
:
1
;
};
};
...
@@ -840,10 +835,10 @@ struct btrfs_block_group_cache {
...
@@ -840,10 +835,10 @@ struct btrfs_block_group_cache {
spinlock_t
lock
;
spinlock_t
lock
;
u64
pinned
;
u64
pinned
;
u64
reserved
;
u64
reserved
;
u64
reserved_pinned
;
u64
bytes_super
;
u64
bytes_super
;
u64
flags
;
u64
flags
;
u64
sectorsize
;
u64
sectorsize
;
u64
cache_generation
;
unsigned
int
ro
:
1
;
unsigned
int
ro
:
1
;
unsigned
int
dirty
:
1
;
unsigned
int
dirty
:
1
;
unsigned
int
iref
:
1
;
unsigned
int
iref
:
1
;
...
@@ -899,6 +894,10 @@ struct btrfs_fs_info {
...
@@ -899,6 +894,10 @@ struct btrfs_fs_info {
spinlock_t
block_group_cache_lock
;
spinlock_t
block_group_cache_lock
;
struct
rb_root
block_group_cache_tree
;
struct
rb_root
block_group_cache_tree
;
/* keep track of unallocated space */
spinlock_t
free_chunk_lock
;
u64
free_chunk_space
;
struct
extent_io_tree
freed_extents
[
2
];
struct
extent_io_tree
freed_extents
[
2
];
struct
extent_io_tree
*
pinned_extents
;
struct
extent_io_tree
*
pinned_extents
;
...
@@ -919,11 +918,6 @@ struct btrfs_fs_info {
...
@@ -919,11 +918,6 @@ struct btrfs_fs_info {
struct
btrfs_block_rsv
empty_block_rsv
;
struct
btrfs_block_rsv
empty_block_rsv
;
/* list of block reservations that cross multiple transactions */
struct
list_head
durable_block_rsv_list
;
struct
mutex
durable_block_rsv_mutex
;
u64
generation
;
u64
generation
;
u64
last_trans_committed
;
u64
last_trans_committed
;
...
@@ -2129,6 +2123,11 @@ static inline bool btrfs_mixed_space_info(struct btrfs_space_info *space_info)
...
@@ -2129,6 +2123,11 @@ static inline bool btrfs_mixed_space_info(struct btrfs_space_info *space_info)
(
space_info
->
flags
&
BTRFS_BLOCK_GROUP_DATA
));
(
space_info
->
flags
&
BTRFS_BLOCK_GROUP_DATA
));
}
}
static
inline
gfp_t
btrfs_alloc_write_mask
(
struct
address_space
*
mapping
)
{
return
mapping_gfp_mask
(
mapping
)
&
~
__GFP_FS
;
}
/* extent-tree.c */
/* extent-tree.c */
static
inline
u64
btrfs_calc_trans_metadata_size
(
struct
btrfs_root
*
root
,
static
inline
u64
btrfs_calc_trans_metadata_size
(
struct
btrfs_root
*
root
,
unsigned
num_items
)
unsigned
num_items
)
...
@@ -2137,6 +2136,17 @@ static inline u64 btrfs_calc_trans_metadata_size(struct btrfs_root *root,
...
@@ -2137,6 +2136,17 @@ static inline u64 btrfs_calc_trans_metadata_size(struct btrfs_root *root,
3
*
num_items
;
3
*
num_items
;
}
}
/*
* Doing a truncate won't result in new nodes or leaves, just what we need for
* COW.
*/
static
inline
u64
btrfs_calc_trunc_metadata_size
(
struct
btrfs_root
*
root
,
unsigned
num_items
)
{
return
(
root
->
leafsize
+
root
->
nodesize
*
(
BTRFS_MAX_LEVEL
-
1
))
*
num_items
;
}
void
btrfs_put_block_group
(
struct
btrfs_block_group_cache
*
cache
);
void
btrfs_put_block_group
(
struct
btrfs_block_group_cache
*
cache
);
int
btrfs_run_delayed_refs
(
struct
btrfs_trans_handle
*
trans
,
int
btrfs_run_delayed_refs
(
struct
btrfs_trans_handle
*
trans
,
struct
btrfs_root
*
root
,
unsigned
long
count
);
struct
btrfs_root
*
root
,
unsigned
long
count
);
...
@@ -2196,8 +2206,6 @@ int btrfs_free_extent(struct btrfs_trans_handle *trans,
...
@@ -2196,8 +2206,6 @@ int btrfs_free_extent(struct btrfs_trans_handle *trans,
u64
root_objectid
,
u64
owner
,
u64
offset
);
u64
root_objectid
,
u64
owner
,
u64
offset
);
int
btrfs_free_reserved_extent
(
struct
btrfs_root
*
root
,
u64
start
,
u64
len
);
int
btrfs_free_reserved_extent
(
struct
btrfs_root
*
root
,
u64
start
,
u64
len
);
int
btrfs_update_reserved_bytes
(
struct
btrfs_block_group_cache
*
cache
,
u64
num_bytes
,
int
reserve
,
int
sinfo
);
int
btrfs_prepare_extent_commit
(
struct
btrfs_trans_handle
*
trans
,
int
btrfs_prepare_extent_commit
(
struct
btrfs_trans_handle
*
trans
,
struct
btrfs_root
*
root
);
struct
btrfs_root
*
root
);
int
btrfs_finish_extent_commit
(
struct
btrfs_trans_handle
*
trans
,
int
btrfs_finish_extent_commit
(
struct
btrfs_trans_handle
*
trans
,
...
@@ -2240,25 +2248,20 @@ void btrfs_init_block_rsv(struct btrfs_block_rsv *rsv);
...
@@ -2240,25 +2248,20 @@ void btrfs_init_block_rsv(struct btrfs_block_rsv *rsv);
struct
btrfs_block_rsv
*
btrfs_alloc_block_rsv
(
struct
btrfs_root
*
root
);
struct
btrfs_block_rsv
*
btrfs_alloc_block_rsv
(
struct
btrfs_root
*
root
);
void
btrfs_free_block_rsv
(
struct
btrfs_root
*
root
,
void
btrfs_free_block_rsv
(
struct
btrfs_root
*
root
,
struct
btrfs_block_rsv
*
rsv
);
struct
btrfs_block_rsv
*
rsv
);
void
btrfs_add_durable_block_rsv
(
struct
btrfs_fs_info
*
fs_info
,
int
btrfs_block_rsv_add
(
struct
btrfs_root
*
root
,
struct
btrfs_block_rsv
*
rsv
);
int
btrfs_block_rsv_add
(
struct
btrfs_trans_handle
*
trans
,
struct
btrfs_root
*
root
,
struct
btrfs_block_rsv
*
block_rsv
,
struct
btrfs_block_rsv
*
block_rsv
,
u64
num_bytes
);
u64
num_bytes
);
int
btrfs_block_rsv_check
(
struct
btrfs_trans_handle
*
trans
,
int
btrfs_block_rsv_check
(
struct
btrfs_root
*
root
,
struct
btrfs_root
*
root
,
struct
btrfs_block_rsv
*
block_rsv
,
int
min_factor
);
int
btrfs_block_rsv_refill
(
struct
btrfs_root
*
root
,
struct
btrfs_block_rsv
*
block_rsv
,
struct
btrfs_block_rsv
*
block_rsv
,
u64
min_reserved
,
int
min_factor
);
u64
min_reserved
);
int
btrfs_block_rsv_migrate
(
struct
btrfs_block_rsv
*
src_rsv
,
int
btrfs_block_rsv_migrate
(
struct
btrfs_block_rsv
*
src_rsv
,
struct
btrfs_block_rsv
*
dst_rsv
,
struct
btrfs_block_rsv
*
dst_rsv
,
u64
num_bytes
);
u64
num_bytes
);
void
btrfs_block_rsv_release
(
struct
btrfs_root
*
root
,
void
btrfs_block_rsv_release
(
struct
btrfs_root
*
root
,
struct
btrfs_block_rsv
*
block_rsv
,
struct
btrfs_block_rsv
*
block_rsv
,
u64
num_bytes
);
u64
num_bytes
);
int
btrfs_truncate_reserve_metadata
(
struct
btrfs_trans_handle
*
trans
,
struct
btrfs_root
*
root
,
struct
btrfs_block_rsv
*
rsv
);
int
btrfs_set_block_group_ro
(
struct
btrfs_root
*
root
,
int
btrfs_set_block_group_ro
(
struct
btrfs_root
*
root
,
struct
btrfs_block_group_cache
*
cache
);
struct
btrfs_block_group_cache
*
cache
);
int
btrfs_set_block_group_rw
(
struct
btrfs_root
*
root
,
int
btrfs_set_block_group_rw
(
struct
btrfs_root
*
root
,
...
@@ -2579,11 +2582,6 @@ int btrfs_update_inode(struct btrfs_trans_handle *trans,
...
@@ -2579,11 +2582,6 @@ int btrfs_update_inode(struct btrfs_trans_handle *trans,
int
btrfs_orphan_add
(
struct
btrfs_trans_handle
*
trans
,
struct
inode
*
inode
);
int
btrfs_orphan_add
(
struct
btrfs_trans_handle
*
trans
,
struct
inode
*
inode
);
int
btrfs_orphan_del
(
struct
btrfs_trans_handle
*
trans
,
struct
inode
*
inode
);
int
btrfs_orphan_del
(
struct
btrfs_trans_handle
*
trans
,
struct
inode
*
inode
);
int
btrfs_orphan_cleanup
(
struct
btrfs_root
*
root
);
int
btrfs_orphan_cleanup
(
struct
btrfs_root
*
root
);
void
btrfs_orphan_pre_snapshot
(
struct
btrfs_trans_handle
*
trans
,
struct
btrfs_pending_snapshot
*
pending
,
u64
*
bytes_to_reserve
);
void
btrfs_orphan_post_snapshot
(
struct
btrfs_trans_handle
*
trans
,
struct
btrfs_pending_snapshot
*
pending
);
void
btrfs_orphan_commit_root
(
struct
btrfs_trans_handle
*
trans
,
void
btrfs_orphan_commit_root
(
struct
btrfs_trans_handle
*
trans
,
struct
btrfs_root
*
root
);
struct
btrfs_root
*
root
);
int
btrfs_cont_expand
(
struct
inode
*
inode
,
loff_t
oldsize
,
loff_t
size
);
int
btrfs_cont_expand
(
struct
inode
*
inode
,
loff_t
oldsize
,
loff_t
size
);
...
...
fs/btrfs/disk-io.c
View file @
afd582ac
...
@@ -1648,6 +1648,7 @@ struct btrfs_root *open_ctree(struct super_block *sb,
...
@@ -1648,6 +1648,7 @@ struct btrfs_root *open_ctree(struct super_block *sb,
spin_lock_init
(
&
fs_info
->
fs_roots_radix_lock
);
spin_lock_init
(
&
fs_info
->
fs_roots_radix_lock
);
spin_lock_init
(
&
fs_info
->
delayed_iput_lock
);
spin_lock_init
(
&
fs_info
->
delayed_iput_lock
);
spin_lock_init
(
&
fs_info
->
defrag_inodes_lock
);
spin_lock_init
(
&
fs_info
->
defrag_inodes_lock
);
spin_lock_init
(
&
fs_info
->
free_chunk_lock
);
mutex_init
(
&
fs_info
->
reloc_mutex
);
mutex_init
(
&
fs_info
->
reloc_mutex
);
init_completion
(
&
fs_info
->
kobj_unregister
);
init_completion
(
&
fs_info
->
kobj_unregister
);
...
@@ -1665,8 +1666,6 @@ struct btrfs_root *open_ctree(struct super_block *sb,
...
@@ -1665,8 +1666,6 @@ struct btrfs_root *open_ctree(struct super_block *sb,
btrfs_init_block_rsv
(
&
fs_info
->
trans_block_rsv
);
btrfs_init_block_rsv
(
&
fs_info
->
trans_block_rsv
);
btrfs_init_block_rsv
(
&
fs_info
->
chunk_block_rsv
);
btrfs_init_block_rsv
(
&
fs_info
->
chunk_block_rsv
);
btrfs_init_block_rsv
(
&
fs_info
->
empty_block_rsv
);
btrfs_init_block_rsv
(
&
fs_info
->
empty_block_rsv
);
INIT_LIST_HEAD
(
&
fs_info
->
durable_block_rsv_list
);
mutex_init
(
&
fs_info
->
durable_block_rsv_mutex
);
atomic_set
(
&
fs_info
->
nr_async_submits
,
0
);
atomic_set
(
&
fs_info
->
nr_async_submits
,
0
);
atomic_set
(
&
fs_info
->
async_delalloc_pages
,
0
);
atomic_set
(
&
fs_info
->
async_delalloc_pages
,
0
);
atomic_set
(
&
fs_info
->
async_submit_draining
,
0
);
atomic_set
(
&
fs_info
->
async_submit_draining
,
0
);
...
@@ -1677,6 +1676,7 @@ struct btrfs_root *open_ctree(struct super_block *sb,
...
@@ -1677,6 +1676,7 @@ struct btrfs_root *open_ctree(struct super_block *sb,
fs_info
->
metadata_ratio
=
0
;
fs_info
->
metadata_ratio
=
0
;
fs_info
->
defrag_inodes
=
RB_ROOT
;
fs_info
->
defrag_inodes
=
RB_ROOT
;
fs_info
->
trans_no_join
=
0
;
fs_info
->
trans_no_join
=
0
;
fs_info
->
free_chunk_space
=
0
;
fs_info
->
thread_pool_size
=
min_t
(
unsigned
long
,
fs_info
->
thread_pool_size
=
min_t
(
unsigned
long
,
num_online_cpus
()
+
2
,
8
);
num_online_cpus
()
+
2
,
8
);
...
@@ -2545,8 +2545,6 @@ int close_ctree(struct btrfs_root *root)
...
@@ -2545,8 +2545,6 @@ int close_ctree(struct btrfs_root *root)
/* clear out the rbtree of defraggable inodes */
/* clear out the rbtree of defraggable inodes */
btrfs_run_defrag_inodes
(
root
->
fs_info
);
btrfs_run_defrag_inodes
(
root
->
fs_info
);
btrfs_put_block_group_cache
(
fs_info
);
/*
/*
* Here come 2 situations when btrfs is broken to flip readonly:
* Here come 2 situations when btrfs is broken to flip readonly:
*
*
...
@@ -2572,6 +2570,8 @@ int close_ctree(struct btrfs_root *root)
...
@@ -2572,6 +2570,8 @@ int close_ctree(struct btrfs_root *root)
printk
(
KERN_ERR
"btrfs: commit super ret %d
\n
"
,
ret
);
printk
(
KERN_ERR
"btrfs: commit super ret %d
\n
"
,
ret
);
}
}
btrfs_put_block_group_cache
(
fs_info
);
kthread_stop
(
root
->
fs_info
->
transaction_kthread
);
kthread_stop
(
root
->
fs_info
->
transaction_kthread
);
kthread_stop
(
root
->
fs_info
->
cleaner_kthread
);
kthread_stop
(
root
->
fs_info
->
cleaner_kthread
);
...
...
fs/btrfs/extent-tree.c
View file @
afd582ac
This diff is collapsed.
Click to expand it.
fs/btrfs/extent_io.c
View file @
afd582ac
...
@@ -894,6 +894,194 @@ int set_extent_bit(struct extent_io_tree *tree, u64 start, u64 end,
...
@@ -894,6 +894,194 @@ int set_extent_bit(struct extent_io_tree *tree, u64 start, u64 end,
goto
again
;
goto
again
;
}
}
/**
* convert_extent - convert all bits in a given range from one bit to another
* @tree: the io tree to search
* @start: the start offset in bytes
* @end: the end offset in bytes (inclusive)
* @bits: the bits to set in this range
* @clear_bits: the bits to clear in this range
* @mask: the allocation mask
*
* This will go through and set bits for the given range. If any states exist
* already in this range they are set with the given bit and cleared of the
* clear_bits. This is only meant to be used by things that are mergeable, ie
* converting from say DELALLOC to DIRTY. This is not meant to be used with
* boundary bits like LOCK.
*/
int
convert_extent_bit
(
struct
extent_io_tree
*
tree
,
u64
start
,
u64
end
,
int
bits
,
int
clear_bits
,
gfp_t
mask
)
{
struct
extent_state
*
state
;
struct
extent_state
*
prealloc
=
NULL
;
struct
rb_node
*
node
;
int
err
=
0
;
u64
last_start
;
u64
last_end
;
again:
if
(
!
prealloc
&&
(
mask
&
__GFP_WAIT
))
{
prealloc
=
alloc_extent_state
(
mask
);
if
(
!
prealloc
)
return
-
ENOMEM
;
}
spin_lock
(
&
tree
->
lock
);
/*
* this search will find all the extents that end after
* our range starts.
*/
node
=
tree_search
(
tree
,
start
);
if
(
!
node
)
{
prealloc
=
alloc_extent_state_atomic
(
prealloc
);
if
(
!
prealloc
)
return
-
ENOMEM
;
err
=
insert_state
(
tree
,
prealloc
,
start
,
end
,
&
bits
);
prealloc
=
NULL
;
BUG_ON
(
err
==
-
EEXIST
);
goto
out
;
}
state
=
rb_entry
(
node
,
struct
extent_state
,
rb_node
);
hit_next:
last_start
=
state
->
start
;
last_end
=
state
->
end
;
/*
* | ---- desired range ---- |
* | state |
*
* Just lock what we found and keep going
*/
if
(
state
->
start
==
start
&&
state
->
end
<=
end
)
{
struct
rb_node
*
next_node
;
set_state_bits
(
tree
,
state
,
&
bits
);
clear_state_bit
(
tree
,
state
,
&
clear_bits
,
0
);
merge_state
(
tree
,
state
);
if
(
last_end
==
(
u64
)
-
1
)
goto
out
;
start
=
last_end
+
1
;
next_node
=
rb_next
(
&
state
->
rb_node
);
if
(
next_node
&&
start
<
end
&&
prealloc
&&
!
need_resched
())
{
state
=
rb_entry
(
next_node
,
struct
extent_state
,
rb_node
);
if
(
state
->
start
==
start
)
goto
hit_next
;
}
goto
search_again
;
}
/*
* | ---- desired range ---- |
* | state |
* or
* | ------------- state -------------- |
*
* We need to split the extent we found, and may flip bits on
* second half.
*
* If the extent we found extends past our
* range, we just split and search again. It'll get split
* again the next time though.
*
* If the extent we found is inside our range, we set the
* desired bit on it.
*/
if
(
state
->
start
<
start
)
{
prealloc
=
alloc_extent_state_atomic
(
prealloc
);
if
(
!
prealloc
)
return
-
ENOMEM
;
err
=
split_state
(
tree
,
state
,
prealloc
,
start
);
BUG_ON
(
err
==
-
EEXIST
);
prealloc
=
NULL
;
if
(
err
)
goto
out
;
if
(
state
->
end
<=
end
)
{
set_state_bits
(
tree
,
state
,
&
bits
);
clear_state_bit
(
tree
,
state
,
&
clear_bits
,
0
);
merge_state
(
tree
,
state
);
if
(
last_end
==
(
u64
)
-
1
)
goto
out
;
start
=
last_end
+
1
;
}
goto
search_again
;
}
/*
* | ---- desired range ---- |
* | state | or | state |
*
* There's a hole, we need to insert something in it and
* ignore the extent we found.
*/
if
(
state
->
start
>
start
)
{
u64
this_end
;
if
(
end
<
last_start
)
this_end
=
end
;
else
this_end
=
last_start
-
1
;
prealloc
=
alloc_extent_state_atomic
(
prealloc
);
if
(
!
prealloc
)
return
-
ENOMEM
;
/*
* Avoid to free 'prealloc' if it can be merged with
* the later extent.
*/
err
=
insert_state
(
tree
,
prealloc
,
start
,
this_end
,
&
bits
);
BUG_ON
(
err
==
-
EEXIST
);
if
(
err
)
{
free_extent_state
(
prealloc
);
prealloc
=
NULL
;
goto
out
;
}
prealloc
=
NULL
;
start
=
this_end
+
1
;
goto
search_again
;
}
/*
* | ---- desired range ---- |
* | state |
* We need to split the extent, and set the bit
* on the first half
*/
if
(
state
->
start
<=
end
&&
state
->
end
>
end
)
{
prealloc
=
alloc_extent_state_atomic
(
prealloc
);
if
(
!
prealloc
)
return
-
ENOMEM
;
err
=
split_state
(
tree
,
state
,
prealloc
,
end
+
1
);
BUG_ON
(
err
==
-
EEXIST
);
set_state_bits
(
tree
,
prealloc
,
&
bits
);
clear_state_bit
(
tree
,
prealloc
,
&
clear_bits
,
0
);
merge_state
(
tree
,
prealloc
);
prealloc
=
NULL
;
goto
out
;
}
goto
search_again
;
out:
spin_unlock
(
&
tree
->
lock
);
if
(
prealloc
)
free_extent_state
(
prealloc
);
return
err
;
search_again:
if
(
start
>
end
)
goto
out
;
spin_unlock
(
&
tree
->
lock
);
if
(
mask
&
__GFP_WAIT
)
cond_resched
();
goto
again
;
}
/* wrappers around set/clear extent bit */
/* wrappers around set/clear extent bit */
int
set_extent_dirty
(
struct
extent_io_tree
*
tree
,
u64
start
,
u64
end
,
int
set_extent_dirty
(
struct
extent_io_tree
*
tree
,
u64
start
,
u64
end
,
gfp_t
mask
)
gfp_t
mask
)
...
@@ -2136,6 +2324,7 @@ static int __extent_writepage(struct page *page, struct writeback_control *wbc,
...
@@ -2136,6 +2324,7 @@ static int __extent_writepage(struct page *page, struct writeback_control *wbc,
int
compressed
;
int
compressed
;
int
write_flags
;
int
write_flags
;
unsigned
long
nr_written
=
0
;
unsigned
long
nr_written
=
0
;
bool
fill_delalloc
=
true
;
if
(
wbc
->
sync_mode
==
WB_SYNC_ALL
)
if
(
wbc
->
sync_mode
==
WB_SYNC_ALL
)
write_flags
=
WRITE_SYNC
;
write_flags
=
WRITE_SYNC
;
...
@@ -2166,10 +2355,13 @@ static int __extent_writepage(struct page *page, struct writeback_control *wbc,
...
@@ -2166,10 +2355,13 @@ static int __extent_writepage(struct page *page, struct writeback_control *wbc,
set_page_extent_mapped
(
page
);
set_page_extent_mapped
(
page
);
if
(
!
tree
->
ops
||
!
tree
->
ops
->
fill_delalloc
)
fill_delalloc
=
false
;
delalloc_start
=
start
;
delalloc_start
=
start
;
delalloc_end
=
0
;
delalloc_end
=
0
;
page_started
=
0
;
page_started
=
0
;
if
(
!
epd
->
extent_locked
)
{
if
(
!
epd
->
extent_locked
&&
fill_delalloc
)
{
u64
delalloc_to_write
=
0
;
u64
delalloc_to_write
=
0
;
/*
/*
* make sure the wbc mapping index is at least updated
* make sure the wbc mapping index is at least updated
...
...
fs/btrfs/extent_io.h
View file @
afd582ac
...
@@ -17,6 +17,7 @@
...
@@ -17,6 +17,7 @@
#define EXTENT_NODATASUM (1 << 10)
#define EXTENT_NODATASUM (1 << 10)
#define EXTENT_DO_ACCOUNTING (1 << 11)
#define EXTENT_DO_ACCOUNTING (1 << 11)
#define EXTENT_FIRST_DELALLOC (1 << 12)
#define EXTENT_FIRST_DELALLOC (1 << 12)
#define EXTENT_NEED_WAIT (1 << 13)
#define EXTENT_IOBITS (EXTENT_LOCKED | EXTENT_WRITEBACK)
#define EXTENT_IOBITS (EXTENT_LOCKED | EXTENT_WRITEBACK)
#define EXTENT_CTLBITS (EXTENT_DO_ACCOUNTING | EXTENT_FIRST_DELALLOC)
#define EXTENT_CTLBITS (EXTENT_DO_ACCOUNTING | EXTENT_FIRST_DELALLOC)
...
@@ -214,6 +215,8 @@ int set_extent_dirty(struct extent_io_tree *tree, u64 start, u64 end,
...
@@ -214,6 +215,8 @@ int set_extent_dirty(struct extent_io_tree *tree, u64 start, u64 end,
gfp_t
mask
);
gfp_t
mask
);
int
clear_extent_dirty
(
struct
extent_io_tree
*
tree
,
u64
start
,
u64
end
,
int
clear_extent_dirty
(
struct
extent_io_tree
*
tree
,
u64
start
,
u64
end
,
gfp_t
mask
);
gfp_t
mask
);
int
convert_extent_bit
(
struct
extent_io_tree
*
tree
,
u64
start
,
u64
end
,
int
bits
,
int
clear_bits
,
gfp_t
mask
);
int
set_extent_delalloc
(
struct
extent_io_tree
*
tree
,
u64
start
,
u64
end
,
int
set_extent_delalloc
(
struct
extent_io_tree
*
tree
,
u64
start
,
u64
end
,
struct
extent_state
**
cached_state
,
gfp_t
mask
);
struct
extent_state
**
cached_state
,
gfp_t
mask
);
int
find_first_extent_bit
(
struct
extent_io_tree
*
tree
,
u64
start
,
int
find_first_extent_bit
(
struct
extent_io_tree
*
tree
,
u64
start
,
...
...
fs/btrfs/file.c
View file @
afd582ac
...
@@ -1069,6 +1069,7 @@ static noinline int prepare_pages(struct btrfs_root *root, struct file *file,
...
@@ -1069,6 +1069,7 @@ static noinline int prepare_pages(struct btrfs_root *root, struct file *file,
int
i
;
int
i
;
unsigned
long
index
=
pos
>>
PAGE_CACHE_SHIFT
;
unsigned
long
index
=
pos
>>
PAGE_CACHE_SHIFT
;
struct
inode
*
inode
=
fdentry
(
file
)
->
d_inode
;
struct
inode
*
inode
=
fdentry
(
file
)
->
d_inode
;
gfp_t
mask
=
btrfs_alloc_write_mask
(
inode
->
i_mapping
);
int
err
=
0
;
int
err
=
0
;
int
faili
=
0
;
int
faili
=
0
;
u64
start_pos
;
u64
start_pos
;
...
@@ -1080,7 +1081,7 @@ static noinline int prepare_pages(struct btrfs_root *root, struct file *file,
...
@@ -1080,7 +1081,7 @@ static noinline int prepare_pages(struct btrfs_root *root, struct file *file,
again:
again:
for
(
i
=
0
;
i
<
num_pages
;
i
++
)
{
for
(
i
=
0
;
i
<
num_pages
;
i
++
)
{
pages
[
i
]
=
find_or_create_page
(
inode
->
i_mapping
,
index
+
i
,
pages
[
i
]
=
find_or_create_page
(
inode
->
i_mapping
,
index
+
i
,
GFP_NOFS
);
mask
);
if
(
!
pages
[
i
])
{
if
(
!
pages
[
i
])
{
faili
=
i
-
1
;
faili
=
i
-
1
;
err
=
-
ENOMEM
;
err
=
-
ENOMEM
;
...
@@ -1615,10 +1616,6 @@ static long btrfs_fallocate(struct file *file, int mode,
...
@@ -1615,10 +1616,6 @@ static long btrfs_fallocate(struct file *file, int mode,
goto
out
;
goto
out
;
}
}
ret
=
btrfs_check_data_free_space
(
inode
,
alloc_end
-
alloc_start
);
if
(
ret
)
goto
out
;
locked_end
=
alloc_end
-
1
;
locked_end
=
alloc_end
-
1
;
while
(
1
)
{
while
(
1
)
{
struct
btrfs_ordered_extent
*
ordered
;
struct
btrfs_ordered_extent
*
ordered
;
...
@@ -1664,11 +1661,27 @@ static long btrfs_fallocate(struct file *file, int mode,
...
@@ -1664,11 +1661,27 @@ static long btrfs_fallocate(struct file *file, int mode,
if
(
em
->
block_start
==
EXTENT_MAP_HOLE
||
if
(
em
->
block_start
==
EXTENT_MAP_HOLE
||
(
cur_offset
>=
inode
->
i_size
&&
(
cur_offset
>=
inode
->
i_size
&&
!
test_bit
(
EXTENT_FLAG_PREALLOC
,
&
em
->
flags
)))
{
!
test_bit
(
EXTENT_FLAG_PREALLOC
,
&
em
->
flags
)))
{
/*
* Make sure we have enough space before we do the
* allocation.
*/
ret
=
btrfs_check_data_free_space
(
inode
,
last_byte
-
cur_offset
);
if
(
ret
)
{
free_extent_map
(
em
);
break
;
}
ret
=
btrfs_prealloc_file_range
(
inode
,
mode
,
cur_offset
,
ret
=
btrfs_prealloc_file_range
(
inode
,
mode
,
cur_offset
,
last_byte
-
cur_offset
,
last_byte
-
cur_offset
,
1
<<
inode
->
i_blkbits
,
1
<<
inode
->
i_blkbits
,
offset
+
len
,
offset
+
len
,
&
alloc_hint
);
&
alloc_hint
);
/* Let go of our reservation. */
btrfs_free_reserved_data_space
(
inode
,
last_byte
-
cur_offset
);
if
(
ret
<
0
)
{
if
(
ret
<
0
)
{
free_extent_map
(
em
);
free_extent_map
(
em
);
break
;
break
;
...
@@ -1694,8 +1707,6 @@ static long btrfs_fallocate(struct file *file, int mode,
...
@@ -1694,8 +1707,6 @@ static long btrfs_fallocate(struct file *file, int mode,
}
}
unlock_extent_cached
(
&
BTRFS_I
(
inode
)
->
io_tree
,
alloc_start
,
locked_end
,
unlock_extent_cached
(
&
BTRFS_I
(
inode
)
->
io_tree
,
alloc_start
,
locked_end
,
&
cached_state
,
GFP_NOFS
);
&
cached_state
,
GFP_NOFS
);
btrfs_free_reserved_data_space
(
inode
,
alloc_end
-
alloc_start
);
out:
out:
mutex_unlock
(
&
inode
->
i_mutex
);
mutex_unlock
(
&
inode
->
i_mutex
);
return
ret
;
return
ret
;
...
...
fs/btrfs/free-space-cache.c
View file @
afd582ac
This diff is collapsed.
Click to expand it.
fs/btrfs/inode-map.c
View file @
afd582ac
...
@@ -465,14 +465,16 @@ int btrfs_save_ino_cache(struct btrfs_root *root,
...
@@ -465,14 +465,16 @@ int btrfs_save_ino_cache(struct btrfs_root *root,
/* Just to make sure we have enough space */
/* Just to make sure we have enough space */
prealloc
+=
8
*
PAGE_CACHE_SIZE
;
prealloc
+=
8
*
PAGE_CACHE_SIZE
;
ret
=
btrfs_
check_data_fre
e_space
(
inode
,
prealloc
);
ret
=
btrfs_
delalloc_reserv
e_space
(
inode
,
prealloc
);
if
(
ret
)
if
(
ret
)
goto
out_put
;
goto
out_put
;
ret
=
btrfs_prealloc_file_range_trans
(
inode
,
trans
,
0
,
0
,
prealloc
,
ret
=
btrfs_prealloc_file_range_trans
(
inode
,
trans
,
0
,
0
,
prealloc
,
prealloc
,
prealloc
,
&
alloc_hint
);
prealloc
,
prealloc
,
&
alloc_hint
);
if
(
ret
)
if
(
ret
)
{
btrfs_delalloc_release_space
(
inode
,
prealloc
);
goto
out_put
;
goto
out_put
;
}
btrfs_free_reserved_data_space
(
inode
,
prealloc
);
btrfs_free_reserved_data_space
(
inode
,
prealloc
);
out_put:
out_put:
...
...
fs/btrfs/inode.c
View file @
afd582ac
This diff is collapsed.
Click to expand it.
fs/btrfs/ioctl.c
View file @
afd582ac
...
@@ -117,7 +117,7 @@ void btrfs_update_iflags(struct inode *inode)
...
@@ -117,7 +117,7 @@ void btrfs_update_iflags(struct inode *inode)
/*
/*
* Inherit flags from the parent inode.
* Inherit flags from the parent inode.
*
*
*
Unlike extN we don't have any flags we don't want to inherit currently
.
*
Currently only the compression flags and the cow flags are inherited
.
*/
*/
void
btrfs_inherit_iflags
(
struct
inode
*
inode
,
struct
inode
*
dir
)
void
btrfs_inherit_iflags
(
struct
inode
*
inode
,
struct
inode
*
dir
)
{
{
...
@@ -128,12 +128,17 @@ void btrfs_inherit_iflags(struct inode *inode, struct inode *dir)
...
@@ -128,12 +128,17 @@ void btrfs_inherit_iflags(struct inode *inode, struct inode *dir)
flags
=
BTRFS_I
(
dir
)
->
flags
;
flags
=
BTRFS_I
(
dir
)
->
flags
;
if
(
S_ISREG
(
inode
->
i_mode
))
if
(
flags
&
BTRFS_INODE_NOCOMPRESS
)
{
flags
&=
~
BTRFS_INODE_DIRSYNC
;
BTRFS_I
(
inode
)
->
flags
&=
~
BTRFS_INODE_COMPRESS
;
else
if
(
!
S_ISDIR
(
inode
->
i_mode
))
BTRFS_I
(
inode
)
->
flags
|=
BTRFS_INODE_NOCOMPRESS
;
flags
&=
(
BTRFS_INODE_NODUMP
|
BTRFS_INODE_NOATIME
);
}
else
if
(
flags
&
BTRFS_INODE_COMPRESS
)
{
BTRFS_I
(
inode
)
->
flags
&=
~
BTRFS_INODE_NOCOMPRESS
;
BTRFS_I
(
inode
)
->
flags
|=
BTRFS_INODE_COMPRESS
;
}
if
(
flags
&
BTRFS_INODE_NODATACOW
)
BTRFS_I
(
inode
)
->
flags
|=
BTRFS_INODE_NODATACOW
;
BTRFS_I
(
inode
)
->
flags
=
flags
;
btrfs_update_iflags
(
inode
);
btrfs_update_iflags
(
inode
);
}
}
...
@@ -843,6 +848,7 @@ static int cluster_pages_for_defrag(struct inode *inode,
...
@@ -843,6 +848,7 @@ static int cluster_pages_for_defrag(struct inode *inode,
int
i_done
;
int
i_done
;
struct
btrfs_ordered_extent
*
ordered
;
struct
btrfs_ordered_extent
*
ordered
;
struct
extent_state
*
cached_state
=
NULL
;
struct
extent_state
*
cached_state
=
NULL
;
gfp_t
mask
=
btrfs_alloc_write_mask
(
inode
->
i_mapping
);
if
(
isize
==
0
)
if
(
isize
==
0
)
return
0
;
return
0
;
...
@@ -860,7 +866,7 @@ static int cluster_pages_for_defrag(struct inode *inode,
...
@@ -860,7 +866,7 @@ static int cluster_pages_for_defrag(struct inode *inode,
for
(
i
=
0
;
i
<
num_pages
;
i
++
)
{
for
(
i
=
0
;
i
<
num_pages
;
i
++
)
{
struct
page
*
page
;
struct
page
*
page
;
page
=
find_or_create_page
(
inode
->
i_mapping
,
page
=
find_or_create_page
(
inode
->
i_mapping
,
start_index
+
i
,
GFP_NOFS
);
start_index
+
i
,
mask
);
if
(
!
page
)
if
(
!
page
)
break
;
break
;
...
...
fs/btrfs/relocation.c
View file @
afd582ac
...
@@ -2041,8 +2041,7 @@ static noinline_for_stack int merge_reloc_root(struct reloc_control *rc,
...
@@ -2041,8 +2041,7 @@ static noinline_for_stack int merge_reloc_root(struct reloc_control *rc,
BUG_ON
(
IS_ERR
(
trans
));
BUG_ON
(
IS_ERR
(
trans
));
trans
->
block_rsv
=
rc
->
block_rsv
;
trans
->
block_rsv
=
rc
->
block_rsv
;
ret
=
btrfs_block_rsv_check
(
trans
,
root
,
rc
->
block_rsv
,
ret
=
btrfs_block_rsv_refill
(
root
,
rc
->
block_rsv
,
min_reserved
);
min_reserved
,
0
);
if
(
ret
)
{
if
(
ret
)
{
BUG_ON
(
ret
!=
-
EAGAIN
);
BUG_ON
(
ret
!=
-
EAGAIN
);
ret
=
btrfs_commit_transaction
(
trans
,
root
);
ret
=
btrfs_commit_transaction
(
trans
,
root
);
...
@@ -2152,8 +2151,7 @@ int prepare_to_merge(struct reloc_control *rc, int err)
...
@@ -2152,8 +2151,7 @@ int prepare_to_merge(struct reloc_control *rc, int err)
again:
again:
if
(
!
err
)
{
if
(
!
err
)
{
num_bytes
=
rc
->
merging_rsv_size
;
num_bytes
=
rc
->
merging_rsv_size
;
ret
=
btrfs_block_rsv_add
(
NULL
,
root
,
rc
->
block_rsv
,
ret
=
btrfs_block_rsv_add
(
root
,
rc
->
block_rsv
,
num_bytes
);
num_bytes
);
if
(
ret
)
if
(
ret
)
err
=
ret
;
err
=
ret
;
}
}
...
@@ -2427,7 +2425,7 @@ static int reserve_metadata_space(struct btrfs_trans_handle *trans,
...
@@ -2427,7 +2425,7 @@ static int reserve_metadata_space(struct btrfs_trans_handle *trans,
num_bytes
=
calcu_metadata_size
(
rc
,
node
,
1
)
*
2
;
num_bytes
=
calcu_metadata_size
(
rc
,
node
,
1
)
*
2
;
trans
->
block_rsv
=
rc
->
block_rsv
;
trans
->
block_rsv
=
rc
->
block_rsv
;
ret
=
btrfs_block_rsv_add
(
trans
,
root
,
rc
->
block_rsv
,
num_bytes
);
ret
=
btrfs_block_rsv_add
(
root
,
rc
->
block_rsv
,
num_bytes
);
if
(
ret
)
{
if
(
ret
)
{
if
(
ret
==
-
EAGAIN
)
if
(
ret
==
-
EAGAIN
)
rc
->
commit_transaction
=
1
;
rc
->
commit_transaction
=
1
;
...
@@ -2922,6 +2920,7 @@ static int relocate_file_extent_cluster(struct inode *inode,
...
@@ -2922,6 +2920,7 @@ static int relocate_file_extent_cluster(struct inode *inode,
unsigned
long
last_index
;
unsigned
long
last_index
;
struct
page
*
page
;
struct
page
*
page
;
struct
file_ra_state
*
ra
;
struct
file_ra_state
*
ra
;
gfp_t
mask
=
btrfs_alloc_write_mask
(
inode
->
i_mapping
);
int
nr
=
0
;
int
nr
=
0
;
int
ret
=
0
;
int
ret
=
0
;
...
@@ -2956,7 +2955,7 @@ static int relocate_file_extent_cluster(struct inode *inode,
...
@@ -2956,7 +2955,7 @@ static int relocate_file_extent_cluster(struct inode *inode,
ra
,
NULL
,
index
,
ra
,
NULL
,
index
,
last_index
+
1
-
index
);
last_index
+
1
-
index
);
page
=
find_or_create_page
(
inode
->
i_mapping
,
index
,
page
=
find_or_create_page
(
inode
->
i_mapping
,
index
,
GFP_NOFS
);
mask
);
if
(
!
page
)
{
if
(
!
page
)
{
btrfs_delalloc_release_metadata
(
inode
,
btrfs_delalloc_release_metadata
(
inode
,
PAGE_CACHE_SIZE
);
PAGE_CACHE_SIZE
);
...
@@ -3645,14 +3644,11 @@ int prepare_to_relocate(struct reloc_control *rc)
...
@@ -3645,14 +3644,11 @@ int prepare_to_relocate(struct reloc_control *rc)
* btrfs_init_reloc_root will use them when there
* btrfs_init_reloc_root will use them when there
* is no reservation in transaction handle.
* is no reservation in transaction handle.
*/
*/
ret
=
btrfs_block_rsv_add
(
NULL
,
rc
->
extent_root
,
rc
->
block_rsv
,
ret
=
btrfs_block_rsv_add
(
rc
->
extent_root
,
rc
->
block_rsv
,
rc
->
extent_root
->
nodesize
*
256
);
rc
->
extent_root
->
nodesize
*
256
);
if
(
ret
)
if
(
ret
)
return
ret
;
return
ret
;
rc
->
block_rsv
->
refill_used
=
1
;
btrfs_add_durable_block_rsv
(
rc
->
extent_root
->
fs_info
,
rc
->
block_rsv
);
memset
(
&
rc
->
cluster
,
0
,
sizeof
(
rc
->
cluster
));
memset
(
&
rc
->
cluster
,
0
,
sizeof
(
rc
->
cluster
));
rc
->
search_start
=
rc
->
block_group
->
key
.
objectid
;
rc
->
search_start
=
rc
->
block_group
->
key
.
objectid
;
rc
->
extents_found
=
0
;
rc
->
extents_found
=
0
;
...
@@ -3777,8 +3773,7 @@ static noinline_for_stack int relocate_block_group(struct reloc_control *rc)
...
@@ -3777,8 +3773,7 @@ static noinline_for_stack int relocate_block_group(struct reloc_control *rc)
}
}
}
}
ret
=
btrfs_block_rsv_check
(
trans
,
rc
->
extent_root
,
ret
=
btrfs_block_rsv_check
(
rc
->
extent_root
,
rc
->
block_rsv
,
5
);
rc
->
block_rsv
,
0
,
5
);
if
(
ret
<
0
)
{
if
(
ret
<
0
)
{
if
(
ret
!=
-
EAGAIN
)
{
if
(
ret
!=
-
EAGAIN
)
{
err
=
ret
;
err
=
ret
;
...
...
fs/btrfs/super.c
View file @
afd582ac
...
@@ -40,6 +40,7 @@
...
@@ -40,6 +40,7 @@
#include <linux/magic.h>
#include <linux/magic.h>
#include <linux/slab.h>
#include <linux/slab.h>
#include <linux/cleancache.h>
#include <linux/cleancache.h>
#include <linux/mnt_namespace.h>
#include "compat.h"
#include "compat.h"
#include "delayed-inode.h"
#include "delayed-inode.h"
#include "ctree.h"
#include "ctree.h"
...
@@ -58,6 +59,7 @@
...
@@ -58,6 +59,7 @@
#include <trace/events/btrfs.h>
#include <trace/events/btrfs.h>
static
const
struct
super_operations
btrfs_super_ops
;
static
const
struct
super_operations
btrfs_super_ops
;
static
struct
file_system_type
btrfs_fs_type
;
static
const
char
*
btrfs_decode_error
(
struct
btrfs_fs_info
*
fs_info
,
int
errno
,
static
const
char
*
btrfs_decode_error
(
struct
btrfs_fs_info
*
fs_info
,
int
errno
,
char
nbuf
[
16
])
char
nbuf
[
16
])
...
@@ -162,7 +164,7 @@ enum {
...
@@ -162,7 +164,7 @@ enum {
Opt_notreelog
,
Opt_ratio
,
Opt_flushoncommit
,
Opt_discard
,
Opt_notreelog
,
Opt_ratio
,
Opt_flushoncommit
,
Opt_discard
,
Opt_space_cache
,
Opt_clear_cache
,
Opt_user_subvol_rm_allowed
,
Opt_space_cache
,
Opt_clear_cache
,
Opt_user_subvol_rm_allowed
,
Opt_enospc_debug
,
Opt_subvolrootid
,
Opt_defrag
,
Opt_enospc_debug
,
Opt_subvolrootid
,
Opt_defrag
,
Opt_inode_cache
,
Opt_err
,
Opt_inode_cache
,
Opt_
no_space_cache
,
Opt_
err
,
};
};
static
match_table_t
tokens
=
{
static
match_table_t
tokens
=
{
...
@@ -195,6 +197,7 @@ static match_table_t tokens = {
...
@@ -195,6 +197,7 @@ static match_table_t tokens = {
{
Opt_subvolrootid
,
"subvolrootid=%d"
},
{
Opt_subvolrootid
,
"subvolrootid=%d"
},
{
Opt_defrag
,
"autodefrag"
},
{
Opt_defrag
,
"autodefrag"
},
{
Opt_inode_cache
,
"inode_cache"
},
{
Opt_inode_cache
,
"inode_cache"
},
{
Opt_no_space_cache
,
"no_space_cache"
},
{
Opt_err
,
NULL
},
{
Opt_err
,
NULL
},
};
};
...
@@ -206,14 +209,19 @@ int btrfs_parse_options(struct btrfs_root *root, char *options)
...
@@ -206,14 +209,19 @@ int btrfs_parse_options(struct btrfs_root *root, char *options)
{
{
struct
btrfs_fs_info
*
info
=
root
->
fs_info
;
struct
btrfs_fs_info
*
info
=
root
->
fs_info
;
substring_t
args
[
MAX_OPT_ARGS
];
substring_t
args
[
MAX_OPT_ARGS
];
char
*
p
,
*
num
,
*
orig
;
char
*
p
,
*
num
,
*
orig
=
NULL
;
u64
cache_gen
;
int
intarg
;
int
intarg
;
int
ret
=
0
;
int
ret
=
0
;
char
*
compress_type
;
char
*
compress_type
;
bool
compress_force
=
false
;
bool
compress_force
=
false
;
cache_gen
=
btrfs_super_cache_generation
(
&
root
->
fs_info
->
super_copy
);
if
(
cache_gen
)
btrfs_set_opt
(
info
->
mount_opt
,
SPACE_CACHE
);
if
(
!
options
)
if
(
!
options
)
return
0
;
goto
out
;
/*
/*
* strsep changes the string, duplicate it because parse_options
* strsep changes the string, duplicate it because parse_options
...
@@ -360,9 +368,12 @@ int btrfs_parse_options(struct btrfs_root *root, char *options)
...
@@ -360,9 +368,12 @@ int btrfs_parse_options(struct btrfs_root *root, char *options)
btrfs_set_opt
(
info
->
mount_opt
,
DISCARD
);
btrfs_set_opt
(
info
->
mount_opt
,
DISCARD
);
break
;
break
;
case
Opt_space_cache
:
case
Opt_space_cache
:
printk
(
KERN_INFO
"btrfs: enabling disk space caching
\n
"
);
btrfs_set_opt
(
info
->
mount_opt
,
SPACE_CACHE
);
btrfs_set_opt
(
info
->
mount_opt
,
SPACE_CACHE
);
break
;
break
;
case
Opt_no_space_cache
:
printk
(
KERN_INFO
"btrfs: disabling disk space caching
\n
"
);
btrfs_clear_opt
(
info
->
mount_opt
,
SPACE_CACHE
);
break
;
case
Opt_inode_cache
:
case
Opt_inode_cache
:
printk
(
KERN_INFO
"btrfs: enabling inode map caching
\n
"
);
printk
(
KERN_INFO
"btrfs: enabling inode map caching
\n
"
);
btrfs_set_opt
(
info
->
mount_opt
,
INODE_MAP_CACHE
);
btrfs_set_opt
(
info
->
mount_opt
,
INODE_MAP_CACHE
);
...
@@ -391,6 +402,8 @@ int btrfs_parse_options(struct btrfs_root *root, char *options)
...
@@ -391,6 +402,8 @@ int btrfs_parse_options(struct btrfs_root *root, char *options)
}
}
}
}
out:
out:
if
(
!
ret
&&
btrfs_test_opt
(
root
,
SPACE_CACHE
))
printk
(
KERN_INFO
"btrfs: disk space caching is enabled
\n
"
);
kfree
(
orig
);
kfree
(
orig
);
return
ret
;
return
ret
;
}
}
...
@@ -411,7 +424,7 @@ static int btrfs_parse_early_options(const char *options, fmode_t flags,
...
@@ -411,7 +424,7 @@ static int btrfs_parse_early_options(const char *options, fmode_t flags,
int
intarg
;
int
intarg
;
if
(
!
options
)
if
(
!
options
)
goto
out
;
return
0
;
/*
/*
* strsep changes the string, duplicate it because parse_options
* strsep changes the string, duplicate it because parse_options
...
@@ -460,26 +473,15 @@ static int btrfs_parse_early_options(const char *options, fmode_t flags,
...
@@ -460,26 +473,15 @@ static int btrfs_parse_early_options(const char *options, fmode_t flags,
error
=
btrfs_scan_one_device
(
match_strdup
(
&
args
[
0
]),
error
=
btrfs_scan_one_device
(
match_strdup
(
&
args
[
0
]),
flags
,
holder
,
fs_devices
);
flags
,
holder
,
fs_devices
);
if
(
error
)
if
(
error
)
goto
out
_free_opts
;
goto
out
;
break
;
break
;
default:
default:
break
;
break
;
}
}
}
}
out_free_opts
:
out
:
kfree
(
orig
);
kfree
(
orig
);
out:
/*
* If no subvolume name is specified we use the default one. Allocate
* a copy of the string "." here so that code later in the
* mount path doesn't care if it's the default volume or another one.
*/
if
(
!*
subvol_name
)
{
*
subvol_name
=
kstrdup
(
"."
,
GFP_KERNEL
);
if
(
!*
subvol_name
)
return
-
ENOMEM
;
}
return
error
;
return
error
;
}
}
...
@@ -492,7 +494,6 @@ static struct dentry *get_default_root(struct super_block *sb,
...
@@ -492,7 +494,6 @@ static struct dentry *get_default_root(struct super_block *sb,
struct
btrfs_path
*
path
;
struct
btrfs_path
*
path
;
struct
btrfs_key
location
;
struct
btrfs_key
location
;
struct
inode
*
inode
;
struct
inode
*
inode
;
struct
dentry
*
dentry
;
u64
dir_id
;
u64
dir_id
;
int
new
=
0
;
int
new
=
0
;
...
@@ -566,29 +567,7 @@ static struct dentry *get_default_root(struct super_block *sb,
...
@@ -566,29 +567,7 @@ static struct dentry *get_default_root(struct super_block *sb,
return
dget
(
sb
->
s_root
);
return
dget
(
sb
->
s_root
);
}
}
if
(
new
)
{
return
d_obtain_alias
(
inode
);
const
struct
qstr
name
=
{
.
name
=
"/"
,
.
len
=
1
};
/*
* New inode, we need to make the dentry a sibling of s_root so
* everything gets cleaned up properly on unmount.
*/
dentry
=
d_alloc
(
sb
->
s_root
,
&
name
);
if
(
!
dentry
)
{
iput
(
inode
);
return
ERR_PTR
(
-
ENOMEM
);
}
d_splice_alias
(
inode
,
dentry
);
}
else
{
/*
* We found the inode in cache, just find a dentry for it and
* put the reference to the inode we just got.
*/
dentry
=
d_find_alias
(
inode
);
iput
(
inode
);
}
return
dentry
;
}
}
static
int
btrfs_fill_super
(
struct
super_block
*
sb
,
static
int
btrfs_fill_super
(
struct
super_block
*
sb
,
...
@@ -719,6 +698,8 @@ static int btrfs_show_options(struct seq_file *seq, struct vfsmount *vfs)
...
@@ -719,6 +698,8 @@ static int btrfs_show_options(struct seq_file *seq, struct vfsmount *vfs)
seq_puts
(
seq
,
",noacl"
);
seq_puts
(
seq
,
",noacl"
);
if
(
btrfs_test_opt
(
root
,
SPACE_CACHE
))
if
(
btrfs_test_opt
(
root
,
SPACE_CACHE
))
seq_puts
(
seq
,
",space_cache"
);
seq_puts
(
seq
,
",space_cache"
);
else
seq_puts
(
seq
,
",no_space_cache"
);
if
(
btrfs_test_opt
(
root
,
CLEAR_CACHE
))
if
(
btrfs_test_opt
(
root
,
CLEAR_CACHE
))
seq_puts
(
seq
,
",clear_cache"
);
seq_puts
(
seq
,
",clear_cache"
);
if
(
btrfs_test_opt
(
root
,
USER_SUBVOL_RM_ALLOWED
))
if
(
btrfs_test_opt
(
root
,
USER_SUBVOL_RM_ALLOWED
))
...
@@ -753,6 +734,118 @@ static int btrfs_set_super(struct super_block *s, void *data)
...
@@ -753,6 +734,118 @@ static int btrfs_set_super(struct super_block *s, void *data)
return
set_anon_super
(
s
,
data
);
return
set_anon_super
(
s
,
data
);
}
}
/*
* This will strip out the subvol=%s argument for an argument string and add
* subvolid=0 to make sure we get the actual tree root for path walking to the
* subvol we want.
*/
static
char
*
setup_root_args
(
char
*
args
)
{
unsigned
copied
=
0
;
unsigned
len
=
strlen
(
args
)
+
2
;
char
*
pos
;
char
*
ret
;
/*
* We need the same args as before, but minus
*
* subvol=a
*
* and add
*
* subvolid=0
*
* which is a difference of 2 characters, so we allocate strlen(args) +
* 2 characters.
*/
ret
=
kzalloc
(
len
*
sizeof
(
char
),
GFP_NOFS
);
if
(
!
ret
)
return
NULL
;
pos
=
strstr
(
args
,
"subvol="
);
/* This shouldn't happen, but just in case.. */
if
(
!
pos
)
{
kfree
(
ret
);
return
NULL
;
}
/*
* The subvol=<> arg is not at the front of the string, copy everybody
* up to that into ret.
*/
if
(
pos
!=
args
)
{
*
pos
=
'\0'
;
strcpy
(
ret
,
args
);
copied
+=
strlen
(
args
);
pos
++
;
}
strncpy
(
ret
+
copied
,
"subvolid=0"
,
len
-
copied
);
/* Length of subvolid=0 */
copied
+=
10
;
/*
* If there is no , after the subvol= option then we know there's no
* other options and we can just return.
*/
pos
=
strchr
(
pos
,
','
);
if
(
!
pos
)
return
ret
;
/* Copy the rest of the arguments into our buffer */
strncpy
(
ret
+
copied
,
pos
,
len
-
copied
);
copied
+=
strlen
(
pos
);
return
ret
;
}
static
struct
dentry
*
mount_subvol
(
const
char
*
subvol_name
,
int
flags
,
const
char
*
device_name
,
char
*
data
)
{
struct
super_block
*
s
;
struct
dentry
*
root
;
struct
vfsmount
*
mnt
;
struct
mnt_namespace
*
ns_private
;
char
*
newargs
;
struct
path
path
;
int
error
;
newargs
=
setup_root_args
(
data
);
if
(
!
newargs
)
return
ERR_PTR
(
-
ENOMEM
);
mnt
=
vfs_kern_mount
(
&
btrfs_fs_type
,
flags
,
device_name
,
newargs
);
kfree
(
newargs
);
if
(
IS_ERR
(
mnt
))
return
ERR_CAST
(
mnt
);
ns_private
=
create_mnt_ns
(
mnt
);
if
(
IS_ERR
(
ns_private
))
{
mntput
(
mnt
);
return
ERR_CAST
(
ns_private
);
}
/*
* This will trigger the automount of the subvol so we can just
* drop the mnt we have here and return the dentry that we
* found.
*/
error
=
vfs_path_lookup
(
mnt
->
mnt_root
,
mnt
,
subvol_name
,
LOOKUP_FOLLOW
,
&
path
);
put_mnt_ns
(
ns_private
);
if
(
error
)
return
ERR_PTR
(
error
);
/* Get a ref to the sb and the dentry we found and return it */
s
=
path
.
mnt
->
mnt_sb
;
atomic_inc
(
&
s
->
s_active
);
root
=
dget
(
path
.
dentry
);
path_put
(
&
path
);
down_write
(
&
s
->
s_umount
);
return
root
;
}
/*
/*
* Find a superblock for the given device / mount point.
* Find a superblock for the given device / mount point.
...
@@ -784,13 +877,19 @@ static struct dentry *btrfs_mount(struct file_system_type *fs_type, int flags,
...
@@ -784,13 +877,19 @@ static struct dentry *btrfs_mount(struct file_system_type *fs_type, int flags,
if
(
error
)
if
(
error
)
return
ERR_PTR
(
error
);
return
ERR_PTR
(
error
);
if
(
subvol_name
)
{
root
=
mount_subvol
(
subvol_name
,
flags
,
device_name
,
data
);
kfree
(
subvol_name
);
return
root
;
}
error
=
btrfs_scan_one_device
(
device_name
,
mode
,
fs_type
,
&
fs_devices
);
error
=
btrfs_scan_one_device
(
device_name
,
mode
,
fs_type
,
&
fs_devices
);
if
(
error
)
if
(
error
)
goto
error_free_subvol_name
;
return
ERR_PTR
(
error
)
;
error
=
btrfs_open_devices
(
fs_devices
,
mode
,
fs_type
);
error
=
btrfs_open_devices
(
fs_devices
,
mode
,
fs_type
);
if
(
error
)
if
(
error
)
goto
error_free_subvol_name
;
return
ERR_PTR
(
error
)
;
if
(
!
(
flags
&
MS_RDONLY
)
&&
fs_devices
->
rw_devices
==
0
)
{
if
(
!
(
flags
&
MS_RDONLY
)
&&
fs_devices
->
rw_devices
==
0
)
{
error
=
-
EACCES
;
error
=
-
EACCES
;
...
@@ -815,14 +914,15 @@ static struct dentry *btrfs_mount(struct file_system_type *fs_type, int flags,
...
@@ -815,14 +914,15 @@ static struct dentry *btrfs_mount(struct file_system_type *fs_type, int flags,
bdev
=
fs_devices
->
latest_bdev
;
bdev
=
fs_devices
->
latest_bdev
;
s
=
sget
(
fs_type
,
btrfs_test_super
,
btrfs_set_super
,
tree_root
);
s
=
sget
(
fs_type
,
btrfs_test_super
,
btrfs_set_super
,
tree_root
);
if
(
IS_ERR
(
s
))
if
(
IS_ERR
(
s
))
{
goto
error_s
;
error
=
PTR_ERR
(
s
);
goto
error_close_devices
;
}
if
(
s
->
s_root
)
{
if
(
s
->
s_root
)
{
if
((
flags
^
s
->
s_flags
)
&
MS_RDONLY
)
{
if
((
flags
^
s
->
s_flags
)
&
MS_RDONLY
)
{
deactivate_locked_super
(
s
);
deactivate_locked_super
(
s
);
error
=
-
EBUSY
;
return
ERR_PTR
(
-
EBUSY
);
goto
error_close_devices
;
}
}
btrfs_close_devices
(
fs_devices
);
btrfs_close_devices
(
fs_devices
);
...
@@ -837,64 +937,25 @@ static struct dentry *btrfs_mount(struct file_system_type *fs_type, int flags,
...
@@ -837,64 +937,25 @@ static struct dentry *btrfs_mount(struct file_system_type *fs_type, int flags,
flags
&
MS_SILENT
?
1
:
0
);
flags
&
MS_SILENT
?
1
:
0
);
if
(
error
)
{
if
(
error
)
{
deactivate_locked_super
(
s
);
deactivate_locked_super
(
s
);
goto
error_free_subvol_name
;
return
ERR_PTR
(
error
)
;
}
}
btrfs_sb
(
s
)
->
fs_info
->
bdev_holder
=
fs_type
;
btrfs_sb
(
s
)
->
fs_info
->
bdev_holder
=
fs_type
;
s
->
s_flags
|=
MS_ACTIVE
;
s
->
s_flags
|=
MS_ACTIVE
;
}
}
/* if they gave us a subvolume name bind mount into that */
root
=
get_default_root
(
s
,
subvol_objectid
);
if
(
strcmp
(
subvol_name
,
"."
))
{
if
(
IS_ERR
(
root
))
{
struct
dentry
*
new_root
;
deactivate_locked_super
(
s
);
return
root
;
root
=
get_default_root
(
s
,
subvol_rootid
);
if
(
IS_ERR
(
root
))
{
error
=
PTR_ERR
(
root
);
deactivate_locked_super
(
s
);
goto
error_free_subvol_name
;
}
mutex_lock
(
&
root
->
d_inode
->
i_mutex
);
new_root
=
lookup_one_len
(
subvol_name
,
root
,
strlen
(
subvol_name
));
mutex_unlock
(
&
root
->
d_inode
->
i_mutex
);
if
(
IS_ERR
(
new_root
))
{
dput
(
root
);
deactivate_locked_super
(
s
);
error
=
PTR_ERR
(
new_root
);
goto
error_free_subvol_name
;
}
if
(
!
new_root
->
d_inode
)
{
dput
(
root
);
dput
(
new_root
);
deactivate_locked_super
(
s
);
error
=
-
ENXIO
;
goto
error_free_subvol_name
;
}
dput
(
root
);
root
=
new_root
;
}
else
{
root
=
get_default_root
(
s
,
subvol_objectid
);
if
(
IS_ERR
(
root
))
{
error
=
PTR_ERR
(
root
);
deactivate_locked_super
(
s
);
goto
error_free_subvol_name
;
}
}
}
kfree
(
subvol_name
);
return
root
;
return
root
;
error_s:
error
=
PTR_ERR
(
s
);
error_close_devices:
error_close_devices:
btrfs_close_devices
(
fs_devices
);
btrfs_close_devices
(
fs_devices
);
kfree
(
fs_info
);
kfree
(
fs_info
);
kfree
(
tree_root
);
kfree
(
tree_root
);
error_free_subvol_name:
kfree
(
subvol_name
);
return
ERR_PTR
(
error
);
return
ERR_PTR
(
error
);
}
}
...
...
fs/btrfs/transaction.c
View file @
afd582ac
...
@@ -275,7 +275,7 @@ static struct btrfs_trans_handle *start_transaction(struct btrfs_root *root,
...
@@ -275,7 +275,7 @@ static struct btrfs_trans_handle *start_transaction(struct btrfs_root *root,
*/
*/
if
(
num_items
>
0
&&
root
!=
root
->
fs_info
->
chunk_root
)
{
if
(
num_items
>
0
&&
root
!=
root
->
fs_info
->
chunk_root
)
{
num_bytes
=
btrfs_calc_trans_metadata_size
(
root
,
num_items
);
num_bytes
=
btrfs_calc_trans_metadata_size
(
root
,
num_items
);
ret
=
btrfs_block_rsv_add
(
NULL
,
root
,
ret
=
btrfs_block_rsv_add
(
root
,
&
root
->
fs_info
->
trans_block_rsv
,
&
root
->
fs_info
->
trans_block_rsv
,
num_bytes
);
num_bytes
);
if
(
ret
)
if
(
ret
)
...
@@ -418,8 +418,8 @@ static int should_end_transaction(struct btrfs_trans_handle *trans,
...
@@ -418,8 +418,8 @@ static int should_end_transaction(struct btrfs_trans_handle *trans,
struct
btrfs_root
*
root
)
struct
btrfs_root
*
root
)
{
{
int
ret
;
int
ret
;
ret
=
btrfs_block_rsv_check
(
trans
,
root
,
&
root
->
fs_info
->
global_block_rsv
,
0
,
5
);
ret
=
btrfs_block_rsv_check
(
root
,
&
root
->
fs_info
->
global_block_rsv
,
5
);
return
ret
?
1
:
0
;
return
ret
?
1
:
0
;
}
}
...
@@ -427,17 +427,26 @@ int btrfs_should_end_transaction(struct btrfs_trans_handle *trans,
...
@@ -427,17 +427,26 @@ int btrfs_should_end_transaction(struct btrfs_trans_handle *trans,
struct
btrfs_root
*
root
)
struct
btrfs_root
*
root
)
{
{
struct
btrfs_transaction
*
cur_trans
=
trans
->
transaction
;
struct
btrfs_transaction
*
cur_trans
=
trans
->
transaction
;
struct
btrfs_block_rsv
*
rsv
=
trans
->
block_rsv
;
int
updates
;
int
updates
;
smp_mb
();
smp_mb
();
if
(
cur_trans
->
blocked
||
cur_trans
->
delayed_refs
.
flushing
)
if
(
cur_trans
->
blocked
||
cur_trans
->
delayed_refs
.
flushing
)
return
1
;
return
1
;
/*
* We need to do this in case we're deleting csums so the global block
* rsv get's used instead of the csum block rsv.
*/
trans
->
block_rsv
=
NULL
;
updates
=
trans
->
delayed_ref_updates
;
updates
=
trans
->
delayed_ref_updates
;
trans
->
delayed_ref_updates
=
0
;
trans
->
delayed_ref_updates
=
0
;
if
(
updates
)
if
(
updates
)
btrfs_run_delayed_refs
(
trans
,
root
,
updates
);
btrfs_run_delayed_refs
(
trans
,
root
,
updates
);
trans
->
block_rsv
=
rsv
;
return
should_end_transaction
(
trans
,
root
);
return
should_end_transaction
(
trans
,
root
);
}
}
...
@@ -453,6 +462,8 @@ static int __btrfs_end_transaction(struct btrfs_trans_handle *trans,
...
@@ -453,6 +462,8 @@ static int __btrfs_end_transaction(struct btrfs_trans_handle *trans,
return
0
;
return
0
;
}
}
btrfs_trans_release_metadata
(
trans
,
root
);
trans
->
block_rsv
=
NULL
;
while
(
count
<
4
)
{
while
(
count
<
4
)
{
unsigned
long
cur
=
trans
->
delayed_ref_updates
;
unsigned
long
cur
=
trans
->
delayed_ref_updates
;
trans
->
delayed_ref_updates
=
0
;
trans
->
delayed_ref_updates
=
0
;
...
@@ -473,8 +484,6 @@ static int __btrfs_end_transaction(struct btrfs_trans_handle *trans,
...
@@ -473,8 +484,6 @@ static int __btrfs_end_transaction(struct btrfs_trans_handle *trans,
count
++
;
count
++
;
}
}
btrfs_trans_release_metadata
(
trans
,
root
);
if
(
lock
&&
!
atomic_read
(
&
root
->
fs_info
->
open_ioctl_trans
)
&&
if
(
lock
&&
!
atomic_read
(
&
root
->
fs_info
->
open_ioctl_trans
)
&&
should_end_transaction
(
trans
,
root
))
{
should_end_transaction
(
trans
,
root
))
{
trans
->
transaction
->
blocked
=
1
;
trans
->
transaction
->
blocked
=
1
;
...
@@ -562,50 +571,21 @@ int btrfs_end_transaction_dmeta(struct btrfs_trans_handle *trans,
...
@@ -562,50 +571,21 @@ int btrfs_end_transaction_dmeta(struct btrfs_trans_handle *trans,
int
btrfs_write_marked_extents
(
struct
btrfs_root
*
root
,
int
btrfs_write_marked_extents
(
struct
btrfs_root
*
root
,
struct
extent_io_tree
*
dirty_pages
,
int
mark
)
struct
extent_io_tree
*
dirty_pages
,
int
mark
)
{
{
int
ret
;
int
err
=
0
;
int
err
=
0
;
int
werr
=
0
;
int
werr
=
0
;
struct
page
*
page
;
struct
address_space
*
mapping
=
root
->
fs_info
->
btree_inode
->
i_mapping
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
u64
start
=
0
;
u64
start
=
0
;
u64
end
;
u64
end
;
unsigned
long
index
;
while
(
1
)
{
ret
=
find_first_extent_bit
(
dirty_pages
,
start
,
&
start
,
&
end
,
mark
);
if
(
ret
)
break
;
while
(
start
<=
end
)
{
cond_resched
();
index
=
start
>>
PAGE_CACHE_SHIFT
;
start
=
(
u64
)(
index
+
1
)
<<
PAGE_CACHE_SHIFT
;
page
=
find_get_page
(
btree_inode
->
i_mapping
,
index
);
if
(
!
page
)
continue
;
btree_lock_page_hook
(
page
);
if
(
!
page
->
mapping
)
{
unlock_page
(
page
);
page_cache_release
(
page
);
continue
;
}
if
(
PageWriteback
(
page
))
{
while
(
!
find_first_extent_bit
(
dirty_pages
,
start
,
&
start
,
&
end
,
if
(
PageDirty
(
page
))
mark
))
{
wait_on_page_writeback
(
page
);
convert_extent_bit
(
dirty_pages
,
start
,
end
,
EXTENT_NEED_WAIT
,
mark
,
else
{
GFP_NOFS
);
unlock_page
(
page
);
err
=
filemap_fdatawrite_range
(
mapping
,
start
,
end
);
page_cache_release
(
page
);
if
(
err
)
continue
;
werr
=
err
;
}
cond_resched
();
}
start
=
end
+
1
;
err
=
write_one_page
(
page
,
0
);
if
(
err
)
werr
=
err
;
page_cache_release
(
page
);
}
}
}
if
(
err
)
if
(
err
)
werr
=
err
;
werr
=
err
;
...
@@ -621,39 +601,20 @@ int btrfs_write_marked_extents(struct btrfs_root *root,
...
@@ -621,39 +601,20 @@ int btrfs_write_marked_extents(struct btrfs_root *root,
int
btrfs_wait_marked_extents
(
struct
btrfs_root
*
root
,
int
btrfs_wait_marked_extents
(
struct
btrfs_root
*
root
,
struct
extent_io_tree
*
dirty_pages
,
int
mark
)
struct
extent_io_tree
*
dirty_pages
,
int
mark
)
{
{
int
ret
;
int
err
=
0
;
int
err
=
0
;
int
werr
=
0
;
int
werr
=
0
;
struct
page
*
page
;
struct
address_space
*
mapping
=
root
->
fs_info
->
btree_inode
->
i_mapping
;
struct
inode
*
btree_inode
=
root
->
fs_info
->
btree_inode
;
u64
start
=
0
;
u64
start
=
0
;
u64
end
;
u64
end
;
unsigned
long
index
;
while
(
1
)
{
ret
=
find_first_extent_bit
(
dirty_pages
,
start
,
&
start
,
&
end
,
mark
);
if
(
ret
)
break
;
clear_extent_bits
(
dirty_pages
,
start
,
end
,
mark
,
GFP_NOFS
);
while
(
!
find_first_extent_bit
(
dirty_pages
,
start
,
&
start
,
&
end
,
while
(
start
<=
end
)
{
EXTENT_NEED_WAIT
))
{
index
=
start
>>
PAGE_CACHE_SHIFT
;
clear_extent_bits
(
dirty_pages
,
start
,
end
,
EXTENT_NEED_WAIT
,
GFP_NOFS
);
start
=
(
u64
)(
index
+
1
)
<<
PAGE_CACHE_SHIFT
;
err
=
filemap_fdatawait_range
(
mapping
,
start
,
end
);
page
=
find_get_page
(
btree_inode
->
i_mapping
,
index
);
if
(
err
)
if
(
!
page
)
werr
=
err
;
continue
;
cond_resched
();
if
(
PageDirty
(
page
))
{
start
=
end
+
1
;
btree_lock_page_hook
(
page
);
wait_on_page_writeback
(
page
);
err
=
write_one_page
(
page
,
0
);
if
(
err
)
werr
=
err
;
}
wait_on_page_writeback
(
page
);
page_cache_release
(
page
);
cond_resched
();
}
}
}
if
(
err
)
if
(
err
)
werr
=
err
;
werr
=
err
;
...
@@ -911,10 +872,9 @@ static noinline int create_pending_snapshot(struct btrfs_trans_handle *trans,
...
@@ -911,10 +872,9 @@ static noinline int create_pending_snapshot(struct btrfs_trans_handle *trans,
}
}
btrfs_reloc_pre_snapshot
(
trans
,
pending
,
&
to_reserve
);
btrfs_reloc_pre_snapshot
(
trans
,
pending
,
&
to_reserve
);
btrfs_orphan_pre_snapshot
(
trans
,
pending
,
&
to_reserve
);
if
(
to_reserve
>
0
)
{
if
(
to_reserve
>
0
)
{
ret
=
btrfs_block_rsv_add
(
trans
,
root
,
&
pending
->
block_rsv
,
ret
=
btrfs_block_rsv_add
(
root
,
&
pending
->
block_rsv
,
to_reserve
);
to_reserve
);
if
(
ret
)
{
if
(
ret
)
{
pending
->
error
=
ret
;
pending
->
error
=
ret
;
...
@@ -1002,7 +962,6 @@ static noinline int create_pending_snapshot(struct btrfs_trans_handle *trans,
...
@@ -1002,7 +962,6 @@ static noinline int create_pending_snapshot(struct btrfs_trans_handle *trans,
BUG_ON
(
IS_ERR
(
pending
->
snap
));
BUG_ON
(
IS_ERR
(
pending
->
snap
));
btrfs_reloc_post_snapshot
(
trans
,
pending
);
btrfs_reloc_post_snapshot
(
trans
,
pending
);
btrfs_orphan_post_snapshot
(
trans
,
pending
);
fail:
fail:
kfree
(
new_root_item
);
kfree
(
new_root_item
);
trans
->
block_rsv
=
rsv
;
trans
->
block_rsv
=
rsv
;
...
@@ -1043,7 +1002,7 @@ static void update_super_roots(struct btrfs_root *root)
...
@@ -1043,7 +1002,7 @@ static void update_super_roots(struct btrfs_root *root)
super
->
root
=
root_item
->
bytenr
;
super
->
root
=
root_item
->
bytenr
;
super
->
generation
=
root_item
->
generation
;
super
->
generation
=
root_item
->
generation
;
super
->
root_level
=
root_item
->
level
;
super
->
root_level
=
root_item
->
level
;
if
(
super
->
cache_generation
!=
0
||
btrfs_test_opt
(
root
,
SPACE_CACHE
))
if
(
btrfs_test_opt
(
root
,
SPACE_CACHE
))
super
->
cache_generation
=
root_item
->
generation
;
super
->
cache_generation
=
root_item
->
generation
;
}
}
...
@@ -1168,14 +1127,15 @@ int btrfs_commit_transaction(struct btrfs_trans_handle *trans,
...
@@ -1168,14 +1127,15 @@ int btrfs_commit_transaction(struct btrfs_trans_handle *trans,
btrfs_run_ordered_operations
(
root
,
0
);
btrfs_run_ordered_operations
(
root
,
0
);
btrfs_trans_release_metadata
(
trans
,
root
);
trans
->
block_rsv
=
NULL
;
/* make a pass through all the delayed refs we have so far
/* make a pass through all the delayed refs we have so far
* any runnings procs may add more while we are here
* any runnings procs may add more while we are here
*/
*/
ret
=
btrfs_run_delayed_refs
(
trans
,
root
,
0
);
ret
=
btrfs_run_delayed_refs
(
trans
,
root
,
0
);
BUG_ON
(
ret
);
BUG_ON
(
ret
);
btrfs_trans_release_metadata
(
trans
,
root
);
cur_trans
=
trans
->
transaction
;
cur_trans
=
trans
->
transaction
;
/*
/*
* set the flushing flag so procs in this transaction have to
* set the flushing flag so procs in this transaction have to
...
...
fs/btrfs/volumes.c
View file @
afd582ac
...
@@ -1013,8 +1013,13 @@ static int btrfs_free_dev_extent(struct btrfs_trans_handle *trans,
...
@@ -1013,8 +1013,13 @@ static int btrfs_free_dev_extent(struct btrfs_trans_handle *trans,
}
}
BUG_ON
(
ret
);
BUG_ON
(
ret
);
if
(
device
->
bytes_used
>
0
)
if
(
device
->
bytes_used
>
0
)
{
device
->
bytes_used
-=
btrfs_dev_extent_length
(
leaf
,
extent
);
u64
len
=
btrfs_dev_extent_length
(
leaf
,
extent
);
device
->
bytes_used
-=
len
;
spin_lock
(
&
root
->
fs_info
->
free_chunk_lock
);
root
->
fs_info
->
free_chunk_space
+=
len
;
spin_unlock
(
&
root
->
fs_info
->
free_chunk_lock
);
}
ret
=
btrfs_del_item
(
trans
,
root
,
path
);
ret
=
btrfs_del_item
(
trans
,
root
,
path
);
out:
out:
...
@@ -1356,6 +1361,11 @@ int btrfs_rm_device(struct btrfs_root *root, char *device_path)
...
@@ -1356,6 +1361,11 @@ int btrfs_rm_device(struct btrfs_root *root, char *device_path)
if
(
ret
)
if
(
ret
)
goto
error_undo
;
goto
error_undo
;
spin_lock
(
&
root
->
fs_info
->
free_chunk_lock
);
root
->
fs_info
->
free_chunk_space
=
device
->
total_bytes
-
device
->
bytes_used
;
spin_unlock
(
&
root
->
fs_info
->
free_chunk_lock
);
device
->
in_fs_metadata
=
0
;
device
->
in_fs_metadata
=
0
;
btrfs_scrub_cancel_dev
(
root
,
device
);
btrfs_scrub_cancel_dev
(
root
,
device
);
...
@@ -1691,6 +1701,10 @@ int btrfs_init_new_device(struct btrfs_root *root, char *device_path)
...
@@ -1691,6 +1701,10 @@ int btrfs_init_new_device(struct btrfs_root *root, char *device_path)
root
->
fs_info
->
fs_devices
->
num_can_discard
++
;
root
->
fs_info
->
fs_devices
->
num_can_discard
++
;
root
->
fs_info
->
fs_devices
->
total_rw_bytes
+=
device
->
total_bytes
;
root
->
fs_info
->
fs_devices
->
total_rw_bytes
+=
device
->
total_bytes
;
spin_lock
(
&
root
->
fs_info
->
free_chunk_lock
);
root
->
fs_info
->
free_chunk_space
+=
device
->
total_bytes
;
spin_unlock
(
&
root
->
fs_info
->
free_chunk_lock
);
if
(
!
blk_queue_nonrot
(
bdev_get_queue
(
bdev
)))
if
(
!
blk_queue_nonrot
(
bdev_get_queue
(
bdev
)))
root
->
fs_info
->
fs_devices
->
rotating
=
1
;
root
->
fs_info
->
fs_devices
->
rotating
=
1
;
...
@@ -2192,8 +2206,12 @@ int btrfs_shrink_device(struct btrfs_device *device, u64 new_size)
...
@@ -2192,8 +2206,12 @@ int btrfs_shrink_device(struct btrfs_device *device, u64 new_size)
lock_chunks
(
root
);
lock_chunks
(
root
);
device
->
total_bytes
=
new_size
;
device
->
total_bytes
=
new_size
;
if
(
device
->
writeable
)
if
(
device
->
writeable
)
{
device
->
fs_devices
->
total_rw_bytes
-=
diff
;
device
->
fs_devices
->
total_rw_bytes
-=
diff
;
spin_lock
(
&
root
->
fs_info
->
free_chunk_lock
);
root
->
fs_info
->
free_chunk_space
-=
diff
;
spin_unlock
(
&
root
->
fs_info
->
free_chunk_lock
);
}
unlock_chunks
(
root
);
unlock_chunks
(
root
);
again:
again:
...
@@ -2257,6 +2275,9 @@ int btrfs_shrink_device(struct btrfs_device *device, u64 new_size)
...
@@ -2257,6 +2275,9 @@ int btrfs_shrink_device(struct btrfs_device *device, u64 new_size)
device
->
total_bytes
=
old_size
;
device
->
total_bytes
=
old_size
;
if
(
device
->
writeable
)
if
(
device
->
writeable
)
device
->
fs_devices
->
total_rw_bytes
+=
diff
;
device
->
fs_devices
->
total_rw_bytes
+=
diff
;
spin_lock
(
&
root
->
fs_info
->
free_chunk_lock
);
root
->
fs_info
->
free_chunk_space
+=
diff
;
spin_unlock
(
&
root
->
fs_info
->
free_chunk_lock
);
unlock_chunks
(
root
);
unlock_chunks
(
root
);
goto
done
;
goto
done
;
}
}
...
@@ -2615,6 +2636,11 @@ static int __finish_chunk_alloc(struct btrfs_trans_handle *trans,
...
@@ -2615,6 +2636,11 @@ static int __finish_chunk_alloc(struct btrfs_trans_handle *trans,
index
++
;
index
++
;
}
}
spin_lock
(
&
extent_root
->
fs_info
->
free_chunk_lock
);
extent_root
->
fs_info
->
free_chunk_space
-=
(
stripe_size
*
map
->
num_stripes
);
spin_unlock
(
&
extent_root
->
fs_info
->
free_chunk_lock
);
index
=
0
;
index
=
0
;
stripe
=
&
chunk
->
stripe
;
stripe
=
&
chunk
->
stripe
;
while
(
index
<
map
->
num_stripes
)
{
while
(
index
<
map
->
num_stripes
)
{
...
@@ -3616,8 +3642,13 @@ static int read_one_dev(struct btrfs_root *root,
...
@@ -3616,8 +3642,13 @@ static int read_one_dev(struct btrfs_root *root,
fill_device_from_item
(
leaf
,
dev_item
,
device
);
fill_device_from_item
(
leaf
,
dev_item
,
device
);
device
->
dev_root
=
root
->
fs_info
->
dev_root
;
device
->
dev_root
=
root
->
fs_info
->
dev_root
;
device
->
in_fs_metadata
=
1
;
device
->
in_fs_metadata
=
1
;
if
(
device
->
writeable
)
if
(
device
->
writeable
)
{
device
->
fs_devices
->
total_rw_bytes
+=
device
->
total_bytes
;
device
->
fs_devices
->
total_rw_bytes
+=
device
->
total_bytes
;
spin_lock
(
&
root
->
fs_info
->
free_chunk_lock
);
root
->
fs_info
->
free_chunk_space
+=
device
->
total_bytes
-
device
->
bytes_used
;
spin_unlock
(
&
root
->
fs_info
->
free_chunk_lock
);
}
ret
=
0
;
ret
=
0
;
return
ret
;
return
ret
;
}
}
...
...
fs/btrfs/xattr.c
View file @
afd582ac
...
@@ -127,6 +127,17 @@ static int do_setxattr(struct btrfs_trans_handle *trans,
...
@@ -127,6 +127,17 @@ static int do_setxattr(struct btrfs_trans_handle *trans,
again:
again:
ret
=
btrfs_insert_xattr_item
(
trans
,
root
,
path
,
btrfs_ino
(
inode
),
ret
=
btrfs_insert_xattr_item
(
trans
,
root
,
path
,
btrfs_ino
(
inode
),
name
,
name_len
,
value
,
size
);
name
,
name_len
,
value
,
size
);
/*
* If we're setting an xattr to a new value but the new value is say
* exactly BTRFS_MAX_XATTR_SIZE, we could end up with EOVERFLOW getting
* back from split_leaf. This is because it thinks we'll be extending
* the existing item size, but we're asking for enough space to add the
* item itself. So if we get EOVERFLOW just set ret to EEXIST and let
* the rest of the function figure it out.
*/
if
(
ret
==
-
EOVERFLOW
)
ret
=
-
EEXIST
;
if
(
ret
==
-
EEXIST
)
{
if
(
ret
==
-
EEXIST
)
{
if
(
flags
&
XATTR_CREATE
)
if
(
flags
&
XATTR_CREATE
)
goto
out
;
goto
out
;
...
...
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