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
nexedi
linux
Commits
8a4b83cc
Commit
8a4b83cc
authored
Mar 24, 2008
by
Chris Mason
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Btrfs: Add support for device scanning and detection ioctls
Signed-off-by:
Chris Mason
<
chris.mason@oracle.com
>
parent
239b14b3
Changes
7
Show whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
333 additions
and
44 deletions
+333
-44
fs/btrfs/ctree.h
fs/btrfs/ctree.h
+19
-2
fs/btrfs/disk-io.c
fs/btrfs/disk-io.c
+15
-9
fs/btrfs/disk-io.h
fs/btrfs/disk-io.h
+3
-1
fs/btrfs/ioctl.h
fs/btrfs/ioctl.h
+5
-1
fs/btrfs/super.c
fs/btrfs/super.c
+44
-17
fs/btrfs/volumes.c
fs/btrfs/volumes.c
+222
-14
fs/btrfs/volumes.h
fs/btrfs/volumes.h
+25
-0
No files found.
fs/btrfs/ctree.h
View file @
8a4b83cc
...
@@ -37,7 +37,7 @@ extern struct kmem_cache *btrfs_transaction_cachep;
...
@@ -37,7 +37,7 @@ extern struct kmem_cache *btrfs_transaction_cachep;
extern
struct
kmem_cache
*
btrfs_bit_radix_cachep
;
extern
struct
kmem_cache
*
btrfs_bit_radix_cachep
;
extern
struct
kmem_cache
*
btrfs_path_cachep
;
extern
struct
kmem_cache
*
btrfs_path_cachep
;
#define BTRFS_MAGIC "_B
4
RfS_M"
#define BTRFS_MAGIC "_B
5
RfS_M"
#define BTRFS_MAX_LEVEL 8
#define BTRFS_MAX_LEVEL 8
...
@@ -238,6 +238,7 @@ struct btrfs_super_block {
...
@@ -238,6 +238,7 @@ struct btrfs_super_block {
__le64
total_bytes
;
__le64
total_bytes
;
__le64
bytes_used
;
__le64
bytes_used
;
__le64
root_dir_objectid
;
__le64
root_dir_objectid
;
__le64
num_devices
;
__le32
sectorsize
;
__le32
sectorsize
;
__le32
nodesize
;
__le32
nodesize
;
__le32
leafsize
;
__le32
leafsize
;
...
@@ -440,6 +441,7 @@ struct btrfs_block_group_cache {
...
@@ -440,6 +441,7 @@ struct btrfs_block_group_cache {
};
};
struct
btrfs_device
;
struct
btrfs_device
;
struct
btrfs_fs_devices
;
struct
btrfs_fs_info
{
struct
btrfs_fs_info
{
u8
fsid
[
BTRFS_FSID_SIZE
];
u8
fsid
[
BTRFS_FSID_SIZE
];
struct
btrfs_root
*
extent_root
;
struct
btrfs_root
*
extent_root
;
...
@@ -489,7 +491,7 @@ struct btrfs_fs_info {
...
@@ -489,7 +491,7 @@ struct btrfs_fs_info {
u64
total_pinned
;
u64
total_pinned
;
struct
list_head
dirty_cowonly_roots
;
struct
list_head
dirty_cowonly_roots
;
struct
list_head
devices
;
struct
btrfs_fs_devices
*
fs_
devices
;
struct
list_head
space_info
;
struct
list_head
space_info
;
spinlock_t
delalloc_lock
;
spinlock_t
delalloc_lock
;
spinlock_t
new_trans_lock
;
spinlock_t
new_trans_lock
;
...
@@ -677,6 +679,19 @@ BTRFS_SETGET_FUNCS(device_io_width, struct btrfs_dev_item, io_width, 32);
...
@@ -677,6 +679,19 @@ BTRFS_SETGET_FUNCS(device_io_width, struct btrfs_dev_item, io_width, 32);
BTRFS_SETGET_FUNCS
(
device_sector_size
,
struct
btrfs_dev_item
,
sector_size
,
32
);
BTRFS_SETGET_FUNCS
(
device_sector_size
,
struct
btrfs_dev_item
,
sector_size
,
32
);
BTRFS_SETGET_FUNCS
(
device_id
,
struct
btrfs_dev_item
,
devid
,
64
);
BTRFS_SETGET_FUNCS
(
device_id
,
struct
btrfs_dev_item
,
devid
,
64
);
BTRFS_SETGET_STACK_FUNCS
(
stack_device_type
,
struct
btrfs_dev_item
,
type
,
64
);
BTRFS_SETGET_STACK_FUNCS
(
stack_device_total_bytes
,
struct
btrfs_dev_item
,
total_bytes
,
64
);
BTRFS_SETGET_STACK_FUNCS
(
stack_device_bytes_used
,
struct
btrfs_dev_item
,
bytes_used
,
64
);
BTRFS_SETGET_STACK_FUNCS
(
stack_device_io_align
,
struct
btrfs_dev_item
,
io_align
,
32
);
BTRFS_SETGET_STACK_FUNCS
(
stack_device_io_width
,
struct
btrfs_dev_item
,
io_width
,
32
);
BTRFS_SETGET_STACK_FUNCS
(
stack_device_sector_size
,
struct
btrfs_dev_item
,
sector_size
,
32
);
BTRFS_SETGET_STACK_FUNCS
(
stack_device_id
,
struct
btrfs_dev_item
,
devid
,
64
);
static
inline
char
*
btrfs_device_uuid
(
struct
btrfs_dev_item
*
d
)
static
inline
char
*
btrfs_device_uuid
(
struct
btrfs_dev_item
*
d
)
{
{
return
(
char
*
)
d
+
offsetof
(
struct
btrfs_dev_item
,
uuid
);
return
(
char
*
)
d
+
offsetof
(
struct
btrfs_dev_item
,
uuid
);
...
@@ -1106,6 +1121,8 @@ BTRFS_SETGET_STACK_FUNCS(super_stripesize, struct btrfs_super_block,
...
@@ -1106,6 +1121,8 @@ BTRFS_SETGET_STACK_FUNCS(super_stripesize, struct btrfs_super_block,
stripesize
,
32
);
stripesize
,
32
);
BTRFS_SETGET_STACK_FUNCS
(
super_root_dir
,
struct
btrfs_super_block
,
BTRFS_SETGET_STACK_FUNCS
(
super_root_dir
,
struct
btrfs_super_block
,
root_dir_objectid
,
64
);
root_dir_objectid
,
64
);
BTRFS_SETGET_STACK_FUNCS
(
super_num_devices
,
struct
btrfs_super_block
,
num_devices
,
64
);
static
inline
unsigned
long
btrfs_leaf_data
(
struct
extent_buffer
*
l
)
static
inline
unsigned
long
btrfs_leaf_data
(
struct
extent_buffer
*
l
)
{
{
...
...
fs/btrfs/disk-io.c
View file @
8a4b83cc
...
@@ -365,12 +365,12 @@ static int close_all_devices(struct btrfs_fs_info *fs_info)
...
@@ -365,12 +365,12 @@ static int close_all_devices(struct btrfs_fs_info *fs_info)
struct
list_head
*
next
;
struct
list_head
*
next
;
struct
btrfs_device
*
device
;
struct
btrfs_device
*
device
;
list
=
&
fs_info
->
devices
;
list
=
&
fs_info
->
fs_devices
->
devices
;
while
(
!
list_empty
(
list
))
{
list_for_each
(
next
,
list
)
{
next
=
list
->
next
;
list_del
(
next
);
device
=
list_entry
(
next
,
struct
btrfs_device
,
dev_list
);
device
=
list_entry
(
next
,
struct
btrfs_device
,
dev_list
);
kfree
(
device
);
if
(
device
->
bdev
&&
device
->
bdev
!=
fs_info
->
sb
->
s_bdev
)
close_bdev_excl
(
device
->
bdev
);
device
->
bdev
=
NULL
;
}
}
return
0
;
return
0
;
}
}
...
@@ -655,7 +655,8 @@ static int add_hasher(struct btrfs_fs_info *info, char *type) {
...
@@ -655,7 +655,8 @@ static int add_hasher(struct btrfs_fs_info *info, char *type) {
return 0;
return 0;
}
}
#endif
#endif
struct
btrfs_root
*
open_ctree
(
struct
super_block
*
sb
)
struct
btrfs_root
*
open_ctree
(
struct
super_block
*
sb
,
struct
btrfs_fs_devices
*
fs_devices
)
{
{
u32
sectorsize
;
u32
sectorsize
;
u32
nodesize
;
u32
nodesize
;
...
@@ -697,8 +698,8 @@ struct btrfs_root *open_ctree(struct super_block *sb)
...
@@ -697,8 +698,8 @@ struct btrfs_root *open_ctree(struct super_block *sb)
fs_info
->
extent_root
=
extent_root
;
fs_info
->
extent_root
=
extent_root
;
fs_info
->
chunk_root
=
chunk_root
;
fs_info
->
chunk_root
=
chunk_root
;
fs_info
->
dev_root
=
dev_root
;
fs_info
->
dev_root
=
dev_root
;
fs_info
->
fs_devices
=
fs_devices
;
INIT_LIST_HEAD
(
&
fs_info
->
dirty_cowonly_roots
);
INIT_LIST_HEAD
(
&
fs_info
->
dirty_cowonly_roots
);
INIT_LIST_HEAD
(
&
fs_info
->
devices
);
INIT_LIST_HEAD
(
&
fs_info
->
space_info
);
INIT_LIST_HEAD
(
&
fs_info
->
space_info
);
btrfs_mapping_init
(
&
fs_info
->
mapping_tree
);
btrfs_mapping_init
(
&
fs_info
->
mapping_tree
);
fs_info
->
sb
=
sb
;
fs_info
->
sb
=
sb
;
...
@@ -779,6 +780,12 @@ struct btrfs_root *open_ctree(struct super_block *sb)
...
@@ -779,6 +780,12 @@ struct btrfs_root *open_ctree(struct super_block *sb)
if
(
!
btrfs_super_root
(
disk_super
))
if
(
!
btrfs_super_root
(
disk_super
))
goto
fail_sb_buffer
;
goto
fail_sb_buffer
;
if
(
btrfs_super_num_devices
(
disk_super
)
!=
fs_devices
->
num_devices
)
{
printk
(
"Btrfs: wanted %llu devices, but found %llu
\n
"
,
(
unsigned
long
long
)
btrfs_super_num_devices
(
disk_super
),
(
unsigned
long
long
)
fs_devices
->
num_devices
);
goto
fail_sb_buffer
;
}
nodesize
=
btrfs_super_nodesize
(
disk_super
);
nodesize
=
btrfs_super_nodesize
(
disk_super
);
leafsize
=
btrfs_super_leafsize
(
disk_super
);
leafsize
=
btrfs_super_leafsize
(
disk_super
);
sectorsize
=
btrfs_super_sectorsize
(
disk_super
);
sectorsize
=
btrfs_super_sectorsize
(
disk_super
);
...
@@ -799,8 +806,6 @@ struct btrfs_root *open_ctree(struct super_block *sb)
...
@@ -799,8 +806,6 @@ struct btrfs_root *open_ctree(struct super_block *sb)
}
}
mutex_lock
(
&
fs_info
->
fs_mutex
);
mutex_lock
(
&
fs_info
->
fs_mutex
);
ret
=
btrfs_read_super_device
(
tree_root
,
fs_info
->
sb_buffer
);
BUG_ON
(
ret
);
ret
=
btrfs_read_sys_array
(
tree_root
);
ret
=
btrfs_read_sys_array
(
tree_root
);
BUG_ON
(
ret
);
BUG_ON
(
ret
);
...
@@ -859,6 +864,7 @@ struct btrfs_root *open_ctree(struct super_block *sb)
...
@@ -859,6 +864,7 @@ struct btrfs_root *open_ctree(struct super_block *sb)
fail_iput:
fail_iput:
iput
(
fs_info
->
btree_inode
);
iput
(
fs_info
->
btree_inode
);
fail:
fail:
close_all_devices
(
fs_info
);
kfree
(
extent_root
);
kfree
(
extent_root
);
kfree
(
tree_root
);
kfree
(
tree_root
);
kfree
(
fs_info
);
kfree
(
fs_info
);
...
...
fs/btrfs/disk-io.h
View file @
8a4b83cc
...
@@ -21,6 +21,7 @@
...
@@ -21,6 +21,7 @@
#define BTRFS_SUPER_INFO_OFFSET (16 * 1024)
#define BTRFS_SUPER_INFO_OFFSET (16 * 1024)
struct
btrfs_device
;
struct
btrfs_device
;
struct
btrfs_fs_devices
;
struct
extent_buffer
*
read_tree_block
(
struct
btrfs_root
*
root
,
u64
bytenr
,
struct
extent_buffer
*
read_tree_block
(
struct
btrfs_root
*
root
,
u64
bytenr
,
u32
blocksize
);
u32
blocksize
);
...
@@ -29,7 +30,8 @@ struct extent_buffer *btrfs_find_create_tree_block(struct btrfs_root *root,
...
@@ -29,7 +30,8 @@ struct extent_buffer *btrfs_find_create_tree_block(struct btrfs_root *root,
u64
bytenr
,
u32
blocksize
);
u64
bytenr
,
u32
blocksize
);
int
clean_tree_block
(
struct
btrfs_trans_handle
*
trans
,
int
clean_tree_block
(
struct
btrfs_trans_handle
*
trans
,
struct
btrfs_root
*
root
,
struct
extent_buffer
*
buf
);
struct
btrfs_root
*
root
,
struct
extent_buffer
*
buf
);
struct
btrfs_root
*
open_ctree
(
struct
super_block
*
sb
);
struct
btrfs_root
*
open_ctree
(
struct
super_block
*
sb
,
struct
btrfs_fs_devices
*
fs_devices
);
int
close_ctree
(
struct
btrfs_root
*
root
);
int
close_ctree
(
struct
btrfs_root
*
root
);
int
write_ctree_super
(
struct
btrfs_trans_handle
*
trans
,
int
write_ctree_super
(
struct
btrfs_trans_handle
*
trans
,
struct
btrfs_root
*
root
);
struct
btrfs_root
*
root
);
...
...
fs/btrfs/ioctl.h
View file @
8a4b83cc
...
@@ -22,8 +22,10 @@
...
@@ -22,8 +22,10 @@
#define BTRFS_IOCTL_MAGIC 0x94
#define BTRFS_IOCTL_MAGIC 0x94
#define BTRFS_VOL_NAME_MAX 255
#define BTRFS_VOL_NAME_MAX 255
#define BTRFS_PATH_NAME_MAX 4095
struct
btrfs_ioctl_vol_args
{
struct
btrfs_ioctl_vol_args
{
char
name
[
BTRFS_
VOL
_NAME_MAX
+
1
];
char
name
[
BTRFS_
PATH
_NAME_MAX
+
1
];
};
};
#define BTRFS_IOC_SNAP_CREATE _IOW(BTRFS_IOCTL_MAGIC, 1, \
#define BTRFS_IOC_SNAP_CREATE _IOW(BTRFS_IOCTL_MAGIC, 1, \
...
@@ -32,4 +34,6 @@ struct btrfs_ioctl_vol_args {
...
@@ -32,4 +34,6 @@ struct btrfs_ioctl_vol_args {
struct btrfs_ioctl_vol_args)
struct btrfs_ioctl_vol_args)
#define BTRFS_IOC_RESIZE _IOW(BTRFS_IOCTL_MAGIC, 3, \
#define BTRFS_IOC_RESIZE _IOW(BTRFS_IOCTL_MAGIC, 3, \
struct btrfs_ioctl_vol_args)
struct btrfs_ioctl_vol_args)
#define BTRFS_IOC_SCAN_DEV _IOW(BTRFS_IOCTL_MAGIC, 4, \
struct btrfs_ioctl_vol_args)
#endif
#endif
fs/btrfs/super.c
View file @
8a4b83cc
...
@@ -44,6 +44,7 @@
...
@@ -44,6 +44,7 @@
#include "ioctl.h"
#include "ioctl.h"
#include "print-tree.h"
#include "print-tree.h"
#include "xattr.h"
#include "xattr.h"
#include "volumes.h"
#define BTRFS_SUPER_MAGIC 0x9123683E
#define BTRFS_SUPER_MAGIC 0x9123683E
...
@@ -216,7 +217,9 @@ static int parse_options (char * options,
...
@@ -216,7 +217,9 @@ static int parse_options (char * options,
return
1
;
return
1
;
}
}
static
int
btrfs_fill_super
(
struct
super_block
*
sb
,
void
*
data
,
int
silent
)
static
int
btrfs_fill_super
(
struct
super_block
*
sb
,
struct
btrfs_fs_devices
*
fs_devices
,
void
*
data
,
int
silent
)
{
{
struct
inode
*
inode
;
struct
inode
*
inode
;
struct
dentry
*
root_dentry
;
struct
dentry
*
root_dentry
;
...
@@ -231,7 +234,7 @@ static int btrfs_fill_super(struct super_block * sb, void * data, int silent)
...
@@ -231,7 +234,7 @@ static int btrfs_fill_super(struct super_block * sb, void * data, int silent)
sb
->
s_xattr
=
btrfs_xattr_handlers
;
sb
->
s_xattr
=
btrfs_xattr_handlers
;
sb
->
s_time_gran
=
1
;
sb
->
s_time_gran
=
1
;
tree_root
=
open_ctree
(
sb
);
tree_root
=
open_ctree
(
sb
,
fs_devices
);
if
(
!
tree_root
||
IS_ERR
(
tree_root
))
{
if
(
!
tree_root
||
IS_ERR
(
tree_root
))
{
printk
(
"btrfs: open_ctree failed
\n
"
);
printk
(
"btrfs: open_ctree failed
\n
"
);
...
@@ -334,18 +337,23 @@ static int test_bdev_super(struct super_block *s, void *data)
...
@@ -334,18 +337,23 @@ static int test_bdev_super(struct super_block *s, void *data)
int
btrfs_get_sb_bdev
(
struct
file_system_type
*
fs_type
,
int
btrfs_get_sb_bdev
(
struct
file_system_type
*
fs_type
,
int
flags
,
const
char
*
dev_name
,
void
*
data
,
int
flags
,
const
char
*
dev_name
,
void
*
data
,
int
(
*
fill_super
)(
struct
super_block
*
,
void
*
,
int
),
struct
vfsmount
*
mnt
,
const
char
*
subvol
)
struct
vfsmount
*
mnt
,
const
char
*
subvol
)
{
{
struct
block_device
*
bdev
=
NULL
;
struct
block_device
*
bdev
=
NULL
;
struct
super_block
*
s
;
struct
super_block
*
s
;
struct
dentry
*
root
;
struct
dentry
*
root
;
struct
btrfs_fs_devices
*
fs_devices
=
NULL
;
int
error
=
0
;
int
error
=
0
;
bdev
=
open_bdev_excl
(
dev_name
,
flags
,
fs_type
);
error
=
btrfs_scan_one_device
(
dev_name
,
flags
,
fs_type
,
&
fs_devices
);
if
(
IS_ERR
(
bdev
))
if
(
error
)
return
PTR_ERR
(
bdev
);
return
error
;
error
=
btrfs_open_devices
(
fs_devices
,
flags
,
fs_type
);
if
(
error
)
return
error
;
bdev
=
fs_devices
->
lowest_bdev
;
/*
/*
* once the super is inserted into the list by sget, s_umount
* once the super is inserted into the list by sget, s_umount
* will protect the lockfs code from trying to start a snapshot
* will protect the lockfs code from trying to start a snapshot
...
@@ -372,7 +380,8 @@ int btrfs_get_sb_bdev(struct file_system_type *fs_type,
...
@@ -372,7 +380,8 @@ int btrfs_get_sb_bdev(struct file_system_type *fs_type,
s
->
s_flags
=
flags
;
s
->
s_flags
=
flags
;
strlcpy
(
s
->
s_id
,
bdevname
(
bdev
,
b
),
sizeof
(
s
->
s_id
));
strlcpy
(
s
->
s_id
,
bdevname
(
bdev
,
b
),
sizeof
(
s
->
s_id
));
sb_set_blocksize
(
s
,
block_size
(
bdev
));
sb_set_blocksize
(
s
,
block_size
(
bdev
));
error
=
fill_super
(
s
,
data
,
flags
&
MS_SILENT
?
1
:
0
);
error
=
btrfs_fill_super
(
s
,
fs_devices
,
data
,
flags
&
MS_SILENT
?
1
:
0
);
if
(
error
)
{
if
(
error
)
{
up_write
(
&
s
->
s_umount
);
up_write
(
&
s
->
s_umount
);
deactivate_super
(
s
);
deactivate_super
(
s
);
...
@@ -408,7 +417,7 @@ int btrfs_get_sb_bdev(struct file_system_type *fs_type,
...
@@ -408,7 +417,7 @@ int btrfs_get_sb_bdev(struct file_system_type *fs_type,
error_s:
error_s:
error
=
PTR_ERR
(
s
);
error
=
PTR_ERR
(
s
);
error_bdev:
error_bdev:
close_bdev_excl
(
bdev
);
btrfs_close_devices
(
fs_devices
);
error:
error:
return
error
;
return
error
;
}
}
...
@@ -421,8 +430,7 @@ static int btrfs_get_sb(struct file_system_type *fs_type,
...
@@ -421,8 +430,7 @@ static int btrfs_get_sb(struct file_system_type *fs_type,
char
*
subvol_name
=
NULL
;
char
*
subvol_name
=
NULL
;
parse_options
((
char
*
)
data
,
NULL
,
&
subvol_name
);
parse_options
((
char
*
)
data
,
NULL
,
&
subvol_name
);
ret
=
btrfs_get_sb_bdev
(
fs_type
,
flags
,
dev_name
,
data
,
ret
=
btrfs_get_sb_bdev
(
fs_type
,
flags
,
dev_name
,
data
,
mnt
,
btrfs_fill_super
,
mnt
,
subvol_name
?
subvol_name
:
"default"
);
subvol_name
?
subvol_name
:
"default"
);
if
(
subvol_name
)
if
(
subvol_name
)
kfree
(
subvol_name
);
kfree
(
subvol_name
);
...
@@ -445,13 +453,6 @@ static int btrfs_statfs(struct dentry *dentry, struct kstatfs *buf)
...
@@ -445,13 +453,6 @@ static int btrfs_statfs(struct dentry *dentry, struct kstatfs *buf)
return
0
;
return
0
;
}
}
static
long
btrfs_control_ioctl
(
struct
file
*
file
,
unsigned
int
cmd
,
unsigned
long
arg
)
{
printk
(
"btrfs control ioctl %d
\n
"
,
cmd
);
return
0
;
}
static
struct
file_system_type
btrfs_fs_type
=
{
static
struct
file_system_type
btrfs_fs_type
=
{
.
owner
=
THIS_MODULE
,
.
owner
=
THIS_MODULE
,
.
name
=
"btrfs"
,
.
name
=
"btrfs"
,
...
@@ -460,6 +461,31 @@ static struct file_system_type btrfs_fs_type = {
...
@@ -460,6 +461,31 @@ static struct file_system_type btrfs_fs_type = {
.
fs_flags
=
FS_REQUIRES_DEV
,
.
fs_flags
=
FS_REQUIRES_DEV
,
};
};
static
long
btrfs_control_ioctl
(
struct
file
*
file
,
unsigned
int
cmd
,
unsigned
long
arg
)
{
struct
btrfs_ioctl_vol_args
*
vol
;
struct
btrfs_fs_devices
*
fs_devices
;
int
ret
;
int
len
;
vol
=
kmalloc
(
sizeof
(
*
vol
),
GFP_KERNEL
);
if
(
copy_from_user
(
vol
,
(
void
__user
*
)
arg
,
sizeof
(
*
vol
)))
{
ret
=
-
EFAULT
;
goto
out
;
}
len
=
strnlen
(
vol
->
name
,
BTRFS_PATH_NAME_MAX
);
switch
(
cmd
)
{
case
BTRFS_IOC_SCAN_DEV
:
ret
=
btrfs_scan_one_device
(
vol
->
name
,
MS_RDONLY
,
&
btrfs_fs_type
,
&
fs_devices
);
break
;
}
out:
kfree
(
vol
);
return
0
;
}
static
void
btrfs_write_super_lockfs
(
struct
super_block
*
sb
)
static
void
btrfs_write_super_lockfs
(
struct
super_block
*
sb
)
{
{
struct
btrfs_root
*
root
=
btrfs_sb
(
sb
);
struct
btrfs_root
*
root
=
btrfs_sb
(
sb
);
...
@@ -567,6 +593,7 @@ static void __exit exit_btrfs_fs(void)
...
@@ -567,6 +593,7 @@ static void __exit exit_btrfs_fs(void)
btrfs_interface_exit
();
btrfs_interface_exit
();
unregister_filesystem
(
&
btrfs_fs_type
);
unregister_filesystem
(
&
btrfs_fs_type
);
btrfs_exit_sysfs
();
btrfs_exit_sysfs
();
btrfs_cleanup_fs_uuids
();
}
}
module_init
(
init_btrfs_fs
)
module_init
(
init_btrfs_fs
)
...
...
fs/btrfs/volumes.c
View file @
8a4b83cc
...
@@ -17,6 +17,7 @@
...
@@ -17,6 +17,7 @@
*/
*/
#include <linux/sched.h>
#include <linux/sched.h>
#include <linux/bio.h>
#include <linux/bio.h>
#include <linux/buffer_head.h>
#include "ctree.h"
#include "ctree.h"
#include "extent_map.h"
#include "extent_map.h"
#include "disk-io.h"
#include "disk-io.h"
...
@@ -28,6 +29,215 @@ struct map_lookup {
...
@@ -28,6 +29,215 @@ struct map_lookup {
struct
btrfs_device
*
dev
;
struct
btrfs_device
*
dev
;
u64
physical
;
u64
physical
;
};
};
static
DEFINE_MUTEX
(
uuid_mutex
);
static
LIST_HEAD
(
fs_uuids
);
int
btrfs_cleanup_fs_uuids
(
void
)
{
struct
btrfs_fs_devices
*
fs_devices
;
struct
list_head
*
uuid_cur
;
struct
list_head
*
devices_cur
;
struct
btrfs_device
*
dev
;
list_for_each
(
uuid_cur
,
&
fs_uuids
)
{
fs_devices
=
list_entry
(
uuid_cur
,
struct
btrfs_fs_devices
,
list
);
while
(
!
list_empty
(
&
fs_devices
->
devices
))
{
devices_cur
=
fs_devices
->
devices
.
next
;
dev
=
list_entry
(
devices_cur
,
struct
btrfs_device
,
dev_list
);
printk
(
"uuid cleanup finds %s
\n
"
,
dev
->
name
);
if
(
dev
->
bdev
)
{
printk
(
"closing
\n
"
);
close_bdev_excl
(
dev
->
bdev
);
}
list_del
(
&
dev
->
dev_list
);
kfree
(
dev
);
}
}
return
0
;
}
static
struct
btrfs_device
*
__find_device
(
struct
list_head
*
head
,
u64
devid
)
{
struct
btrfs_device
*
dev
;
struct
list_head
*
cur
;
list_for_each
(
cur
,
head
)
{
dev
=
list_entry
(
cur
,
struct
btrfs_device
,
dev_list
);
if
(
dev
->
devid
==
devid
)
return
dev
;
}
return
NULL
;
}
static
struct
btrfs_fs_devices
*
find_fsid
(
u8
*
fsid
)
{
struct
list_head
*
cur
;
struct
btrfs_fs_devices
*
fs_devices
;
list_for_each
(
cur
,
&
fs_uuids
)
{
fs_devices
=
list_entry
(
cur
,
struct
btrfs_fs_devices
,
list
);
if
(
memcmp
(
fsid
,
fs_devices
->
fsid
,
BTRFS_FSID_SIZE
)
==
0
)
return
fs_devices
;
}
return
NULL
;
}
static
int
device_list_add
(
const
char
*
path
,
struct
btrfs_super_block
*
disk_super
,
u64
devid
,
struct
btrfs_fs_devices
**
fs_devices_ret
)
{
struct
btrfs_device
*
device
;
struct
btrfs_fs_devices
*
fs_devices
;
u64
found_transid
=
btrfs_super_generation
(
disk_super
);
fs_devices
=
find_fsid
(
disk_super
->
fsid
);
if
(
!
fs_devices
)
{
fs_devices
=
kmalloc
(
sizeof
(
*
fs_devices
),
GFP_NOFS
);
if
(
!
fs_devices
)
return
-
ENOMEM
;
INIT_LIST_HEAD
(
&
fs_devices
->
devices
);
list_add
(
&
fs_devices
->
list
,
&
fs_uuids
);
memcpy
(
fs_devices
->
fsid
,
disk_super
->
fsid
,
BTRFS_FSID_SIZE
);
fs_devices
->
latest_devid
=
devid
;
fs_devices
->
latest_trans
=
found_transid
;
fs_devices
->
lowest_devid
=
(
u64
)
-
1
;
fs_devices
->
num_devices
=
0
;
device
=
NULL
;
}
else
{
device
=
__find_device
(
&
fs_devices
->
devices
,
devid
);
}
if
(
!
device
)
{
device
=
kzalloc
(
sizeof
(
*
device
),
GFP_NOFS
);
if
(
!
device
)
{
/* we can safely leave the fs_devices entry around */
return
-
ENOMEM
;
}
device
->
devid
=
devid
;
device
->
name
=
kstrdup
(
path
,
GFP_NOFS
);
if
(
!
device
->
name
)
{
kfree
(
device
);
return
-
ENOMEM
;
}
list_add
(
&
device
->
dev_list
,
&
fs_devices
->
devices
);
fs_devices
->
num_devices
++
;
}
if
(
found_transid
>
fs_devices
->
latest_trans
)
{
fs_devices
->
latest_devid
=
devid
;
fs_devices
->
latest_trans
=
found_transid
;
}
if
(
fs_devices
->
lowest_devid
>
devid
)
{
fs_devices
->
lowest_devid
=
devid
;
printk
(
"lowest devid now %Lu
\n
"
,
devid
);
}
*
fs_devices_ret
=
fs_devices
;
return
0
;
}
int
btrfs_close_devices
(
struct
btrfs_fs_devices
*
fs_devices
)
{
struct
list_head
*
head
=
&
fs_devices
->
devices
;
struct
list_head
*
cur
;
struct
btrfs_device
*
device
;
mutex_lock
(
&
uuid_mutex
);
list_for_each
(
cur
,
head
)
{
device
=
list_entry
(
cur
,
struct
btrfs_device
,
dev_list
);
if
(
device
->
bdev
)
{
close_bdev_excl
(
device
->
bdev
);
printk
(
"close devices closes %s
\n
"
,
device
->
name
);
}
device
->
bdev
=
NULL
;
}
mutex_unlock
(
&
uuid_mutex
);
return
0
;
}
int
btrfs_open_devices
(
struct
btrfs_fs_devices
*
fs_devices
,
int
flags
,
void
*
holder
)
{
struct
block_device
*
bdev
;
struct
list_head
*
head
=
&
fs_devices
->
devices
;
struct
list_head
*
cur
;
struct
btrfs_device
*
device
;
int
ret
;
mutex_lock
(
&
uuid_mutex
);
list_for_each
(
cur
,
head
)
{
device
=
list_entry
(
cur
,
struct
btrfs_device
,
dev_list
);
bdev
=
open_bdev_excl
(
device
->
name
,
flags
,
holder
);
printk
(
"opening %s devid %Lu
\n
"
,
device
->
name
,
device
->
devid
);
if
(
IS_ERR
(
bdev
))
{
printk
(
"open %s failed
\n
"
,
device
->
name
);
ret
=
PTR_ERR
(
bdev
);
goto
fail
;
}
if
(
device
->
devid
==
fs_devices
->
latest_devid
)
fs_devices
->
latest_bdev
=
bdev
;
if
(
device
->
devid
==
fs_devices
->
lowest_devid
)
{
fs_devices
->
lowest_bdev
=
bdev
;
printk
(
"lowest bdev %s
\n
"
,
device
->
name
);
}
device
->
bdev
=
bdev
;
}
mutex_unlock
(
&
uuid_mutex
);
return
0
;
fail:
mutex_unlock
(
&
uuid_mutex
);
btrfs_close_devices
(
fs_devices
);
return
ret
;
}
int
btrfs_scan_one_device
(
const
char
*
path
,
int
flags
,
void
*
holder
,
struct
btrfs_fs_devices
**
fs_devices_ret
)
{
struct
btrfs_super_block
*
disk_super
;
struct
block_device
*
bdev
;
struct
buffer_head
*
bh
;
int
ret
;
u64
devid
;
mutex_lock
(
&
uuid_mutex
);
printk
(
"scan one opens %s
\n
"
,
path
);
bdev
=
open_bdev_excl
(
path
,
flags
,
holder
);
if
(
IS_ERR
(
bdev
))
{
printk
(
"open failed
\n
"
);
ret
=
PTR_ERR
(
bdev
);
goto
error
;
}
ret
=
set_blocksize
(
bdev
,
4096
);
if
(
ret
)
goto
error_close
;
bh
=
__bread
(
bdev
,
BTRFS_SUPER_INFO_OFFSET
/
4096
,
4096
);
if
(
!
bh
)
{
ret
=
-
EIO
;
goto
error_close
;
}
disk_super
=
(
struct
btrfs_super_block
*
)
bh
->
b_data
;
if
(
strncmp
((
char
*
)(
&
disk_super
->
magic
),
BTRFS_MAGIC
,
sizeof
(
disk_super
->
magic
)))
{
printk
(
"no btrfs found on %s
\n
"
,
path
);
ret
=
-
ENOENT
;
goto
error_brelse
;
}
devid
=
le64_to_cpu
(
disk_super
->
dev_item
.
devid
);
printk
(
"found device %Lu on %s
\n
"
,
devid
,
path
);
ret
=
device_list_add
(
path
,
disk_super
,
devid
,
fs_devices_ret
);
error_brelse:
brelse
(
bh
);
error_close:
close_bdev_excl
(
bdev
);
printk
(
"scan one closes bdev %s
\n
"
,
path
);
error:
mutex_unlock
(
&
uuid_mutex
);
return
ret
;
}
/*
/*
* this uses a pretty simple search, the expectation is that it is
* this uses a pretty simple search, the expectation is that it is
...
@@ -56,6 +266,10 @@ static int find_free_dev_extent(struct btrfs_trans_handle *trans,
...
@@ -56,6 +266,10 @@ static int find_free_dev_extent(struct btrfs_trans_handle *trans,
/* FIXME use last free of some kind */
/* FIXME use last free of some kind */
/* we don't want to overwrite the superblock on the drive,
* so we make sure to start at an offset of at least 1MB
*/
search_start
=
max
((
u64
)
1024
*
1024
,
search_start
);
key
.
objectid
=
device
->
devid
;
key
.
objectid
=
device
->
devid
;
key
.
offset
=
search_start
;
key
.
offset
=
search_start
;
key
.
type
=
BTRFS_DEV_EXTENT_KEY
;
key
.
type
=
BTRFS_DEV_EXTENT_KEY
;
...
@@ -285,6 +499,7 @@ int btrfs_add_device(struct btrfs_trans_handle *trans,
...
@@ -285,6 +499,7 @@ int btrfs_add_device(struct btrfs_trans_handle *trans,
leaf
=
path
->
nodes
[
0
];
leaf
=
path
->
nodes
[
0
];
dev_item
=
btrfs_item_ptr
(
leaf
,
path
->
slots
[
0
],
struct
btrfs_dev_item
);
dev_item
=
btrfs_item_ptr
(
leaf
,
path
->
slots
[
0
],
struct
btrfs_dev_item
);
device
->
devid
=
free_devid
;
btrfs_set_device_id
(
leaf
,
dev_item
,
device
->
devid
);
btrfs_set_device_id
(
leaf
,
dev_item
,
device
->
devid
);
btrfs_set_device_type
(
leaf
,
dev_item
,
device
->
type
);
btrfs_set_device_type
(
leaf
,
dev_item
,
device
->
type
);
btrfs_set_device_io_align
(
leaf
,
dev_item
,
device
->
io_align
);
btrfs_set_device_io_align
(
leaf
,
dev_item
,
device
->
io_align
);
...
@@ -382,7 +597,7 @@ int btrfs_alloc_chunk(struct btrfs_trans_handle *trans,
...
@@ -382,7 +597,7 @@ int btrfs_alloc_chunk(struct btrfs_trans_handle *trans,
struct
btrfs_device
*
device
=
NULL
;
struct
btrfs_device
*
device
=
NULL
;
struct
btrfs_chunk
*
chunk
;
struct
btrfs_chunk
*
chunk
;
struct
list_head
private_devs
;
struct
list_head
private_devs
;
struct
list_head
*
dev_list
=
&
extent_root
->
fs_info
->
devices
;
struct
list_head
*
dev_list
=
&
extent_root
->
fs_info
->
fs_devices
->
devices
;
struct
list_head
*
cur
;
struct
list_head
*
cur
;
struct
extent_map_tree
*
em_tree
;
struct
extent_map_tree
*
em_tree
;
struct
map_lookup
*
map
;
struct
map_lookup
*
map
;
...
@@ -449,7 +664,7 @@ int btrfs_alloc_chunk(struct btrfs_trans_handle *trans,
...
@@ -449,7 +664,7 @@ int btrfs_alloc_chunk(struct btrfs_trans_handle *trans,
key
.
objectid
,
key
.
objectid
,
calc_size
,
&
dev_offset
);
calc_size
,
&
dev_offset
);
BUG_ON
(
ret
);
BUG_ON
(
ret
);
printk
(
"alloc chunk size %Lu from dev %Lu
\n
"
,
calc_size
,
device
->
devid
);
device
->
bytes_used
+=
calc_size
;
device
->
bytes_used
+=
calc_size
;
ret
=
btrfs_update_device
(
trans
,
device
);
ret
=
btrfs_update_device
(
trans
,
device
);
BUG_ON
(
ret
);
BUG_ON
(
ret
);
...
@@ -592,17 +807,9 @@ int btrfs_map_bio(struct btrfs_root *root, int rw, struct bio *bio)
...
@@ -592,17 +807,9 @@ int btrfs_map_bio(struct btrfs_root *root, int rw, struct bio *bio)
struct
btrfs_device
*
btrfs_find_device
(
struct
btrfs_root
*
root
,
u64
devid
)
struct
btrfs_device
*
btrfs_find_device
(
struct
btrfs_root
*
root
,
u64
devid
)
{
{
struct
btrfs_device
*
dev
;
struct
list_head
*
head
=
&
root
->
fs_info
->
fs_devices
->
devices
;
struct
list_head
*
cur
=
root
->
fs_info
->
devices
.
next
;
struct
list_head
*
head
=
&
root
->
fs_info
->
devices
;
while
(
cur
!=
head
)
{
return
__find_device
(
head
,
devid
);
dev
=
list_entry
(
cur
,
struct
btrfs_device
,
dev_list
);
if
(
dev
->
devid
==
devid
)
return
dev
;
cur
=
cur
->
next
;
}
return
NULL
;
}
}
static
int
read_one_chunk
(
struct
btrfs_root
*
root
,
struct
btrfs_key
*
key
,
static
int
read_one_chunk
(
struct
btrfs_root
*
root
,
struct
btrfs_key
*
key
,
...
@@ -699,15 +906,16 @@ static int read_one_dev(struct btrfs_root *root,
...
@@ -699,15 +906,16 @@ static int read_one_dev(struct btrfs_root *root,
devid
=
btrfs_device_id
(
leaf
,
dev_item
);
devid
=
btrfs_device_id
(
leaf
,
dev_item
);
device
=
btrfs_find_device
(
root
,
devid
);
device
=
btrfs_find_device
(
root
,
devid
);
if
(
!
device
)
{
if
(
!
device
)
{
printk
(
"warning devid %Lu not found already
\n
"
,
devid
);
device
=
kmalloc
(
sizeof
(
*
device
),
GFP_NOFS
);
device
=
kmalloc
(
sizeof
(
*
device
),
GFP_NOFS
);
if
(
!
device
)
if
(
!
device
)
return
-
ENOMEM
;
return
-
ENOMEM
;
list_add
(
&
device
->
dev_list
,
&
root
->
fs_info
->
devices
);
list_add
(
&
device
->
dev_list
,
&
root
->
fs_info
->
fs_devices
->
devices
);
}
}
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
->
bdev
=
root
->
fs_info
->
sb
->
s_bdev
;
ret
=
0
;
ret
=
0
;
#if 0
#if 0
ret = btrfs_open_device(device);
ret = btrfs_open_device(device);
...
...
fs/btrfs/volumes.h
View file @
8a4b83cc
...
@@ -24,6 +24,8 @@ struct btrfs_device {
...
@@ -24,6 +24,8 @@ struct btrfs_device {
struct
block_device
*
bdev
;
struct
block_device
*
bdev
;
char
*
name
;
/* the internal btrfs device id */
/* the internal btrfs device id */
u64
devid
;
u64
devid
;
...
@@ -49,6 +51,20 @@ struct btrfs_device {
...
@@ -49,6 +51,20 @@ struct btrfs_device {
u8
uuid
[
BTRFS_DEV_UUID_SIZE
];
u8
uuid
[
BTRFS_DEV_UUID_SIZE
];
};
};
struct
btrfs_fs_devices
{
u8
fsid
[
BTRFS_FSID_SIZE
];
/* FS specific uuid */
/* the device with this id has the most recent coyp of the super */
u64
latest_devid
;
u64
latest_trans
;
u64
lowest_devid
;
u64
num_devices
;
struct
block_device
*
latest_bdev
;
struct
block_device
*
lowest_bdev
;
struct
list_head
devices
;
struct
list_head
list
;
};
int
btrfs_alloc_dev_extent
(
struct
btrfs_trans_handle
*
trans
,
int
btrfs_alloc_dev_extent
(
struct
btrfs_trans_handle
*
trans
,
struct
btrfs_device
*
device
,
struct
btrfs_device
*
device
,
u64
owner
,
u64
num_bytes
,
u64
*
start
);
u64
owner
,
u64
num_bytes
,
u64
*
start
);
...
@@ -67,4 +83,13 @@ int btrfs_read_super_device(struct btrfs_root *root, struct extent_buffer *buf);
...
@@ -67,4 +83,13 @@ int btrfs_read_super_device(struct btrfs_root *root, struct extent_buffer *buf);
int
btrfs_map_block
(
struct
btrfs_mapping_tree
*
map_tree
,
int
btrfs_map_block
(
struct
btrfs_mapping_tree
*
map_tree
,
u64
logical
,
u64
*
phys
,
u64
*
length
,
u64
logical
,
u64
*
phys
,
u64
*
length
,
struct
btrfs_device
**
dev
);
struct
btrfs_device
**
dev
);
int
btrfs_open_devices
(
struct
btrfs_fs_devices
*
fs_devices
,
int
flags
,
void
*
holder
);
int
btrfs_scan_one_device
(
const
char
*
path
,
int
flags
,
void
*
holder
,
struct
btrfs_fs_devices
**
fs_devices_ret
);
int
btrfs_close_devices
(
struct
btrfs_fs_devices
*
fs_devices
);
int
btrfs_add_device
(
struct
btrfs_trans_handle
*
trans
,
struct
btrfs_root
*
root
,
struct
btrfs_device
*
device
);
int
btrfs_cleanup_fs_uuids
(
void
);
#endif
#endif
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