bcachefs: Convert split_devs() to darray
Bit of cleanup & modernization: also moving this code to util.c, it'll be used by userspace as well. Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
This commit is contained in:
parent
0c0ba8e9c5
commit
806ebf2aa0
@ -20,6 +20,7 @@ struct { \
|
|||||||
#define DARRAY(_type) DARRAY_PREALLOCATED(_type, 0)
|
#define DARRAY(_type) DARRAY_PREALLOCATED(_type, 0)
|
||||||
|
|
||||||
typedef DARRAY(char) darray_char;
|
typedef DARRAY(char) darray_char;
|
||||||
|
typedef DARRAY(char *) darray_str;
|
||||||
|
|
||||||
int __bch2_darray_resize(darray_char *, size_t, size_t, gfp_t);
|
int __bch2_darray_resize(darray_char *, size_t, size_t, gfp_t);
|
||||||
|
|
||||||
|
@ -1624,31 +1624,6 @@ static struct bch_fs *bch2_path_to_fs(const char *path)
|
|||||||
return c ?: ERR_PTR(-ENOENT);
|
return c ?: ERR_PTR(-ENOENT);
|
||||||
}
|
}
|
||||||
|
|
||||||
static char **split_devs(const char *_dev_name, unsigned *nr)
|
|
||||||
{
|
|
||||||
char *dev_name = NULL, **devs = NULL, *s;
|
|
||||||
size_t i = 0, nr_devs = 0;
|
|
||||||
|
|
||||||
dev_name = kstrdup(_dev_name, GFP_KERNEL);
|
|
||||||
if (!dev_name)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
for (s = dev_name; s; s = strchr(s + 1, ':'))
|
|
||||||
nr_devs++;
|
|
||||||
|
|
||||||
devs = kcalloc(nr_devs + 1, sizeof(const char *), GFP_KERNEL);
|
|
||||||
if (!devs) {
|
|
||||||
kfree(dev_name);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
while ((s = strsep(&dev_name, ":")))
|
|
||||||
devs[i++] = s;
|
|
||||||
|
|
||||||
*nr = nr_devs;
|
|
||||||
return devs;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int bch2_remount(struct super_block *sb, int *flags, char *data)
|
static int bch2_remount(struct super_block *sb, int *flags, char *data)
|
||||||
{
|
{
|
||||||
struct bch_fs *c = sb->s_fs_info;
|
struct bch_fs *c = sb->s_fs_info;
|
||||||
@ -1801,17 +1776,18 @@ static int bch2_noset_super(struct super_block *s, void *data)
|
|||||||
return -EBUSY;
|
return -EBUSY;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
typedef DARRAY(struct bch_fs *) darray_fs;
|
||||||
|
|
||||||
static int bch2_test_super(struct super_block *s, void *data)
|
static int bch2_test_super(struct super_block *s, void *data)
|
||||||
{
|
{
|
||||||
struct bch_fs *c = s->s_fs_info;
|
struct bch_fs *c = s->s_fs_info;
|
||||||
struct bch_fs **devs = data;
|
darray_fs *d = data;
|
||||||
unsigned i;
|
|
||||||
|
|
||||||
if (!c)
|
if (!c)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
for (i = 0; devs[i]; i++)
|
darray_for_each(*d, i)
|
||||||
if (c != devs[i])
|
if (c != *i)
|
||||||
return false;
|
return false;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -1823,9 +1799,6 @@ static struct dentry *bch2_mount(struct file_system_type *fs_type,
|
|||||||
struct super_block *sb;
|
struct super_block *sb;
|
||||||
struct inode *vinode;
|
struct inode *vinode;
|
||||||
struct bch_opts opts = bch2_opts_empty();
|
struct bch_opts opts = bch2_opts_empty();
|
||||||
char **devs;
|
|
||||||
struct bch_fs **devs_to_fs = NULL;
|
|
||||||
unsigned nr_devs;
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
opt_set(opts, read_only, (flags & SB_RDONLY) != 0);
|
opt_set(opts, read_only, (flags & SB_RDONLY) != 0);
|
||||||
@ -1837,25 +1810,25 @@ static struct dentry *bch2_mount(struct file_system_type *fs_type,
|
|||||||
if (!dev_name || strlen(dev_name) == 0)
|
if (!dev_name || strlen(dev_name) == 0)
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
devs = split_devs(dev_name, &nr_devs);
|
darray_str devs;
|
||||||
if (!devs)
|
ret = bch2_split_devs(dev_name, &devs);
|
||||||
return ERR_PTR(-ENOMEM);
|
if (ret)
|
||||||
|
return ERR_PTR(ret);
|
||||||
|
|
||||||
devs_to_fs = kcalloc(nr_devs + 1, sizeof(void *), GFP_KERNEL);
|
darray_fs devs_to_fs = {};
|
||||||
if (!devs_to_fs) {
|
darray_for_each(devs, i) {
|
||||||
sb = ERR_PTR(-ENOMEM);
|
ret = darray_push(&devs_to_fs, bch2_path_to_fs(*i));
|
||||||
|
if (ret) {
|
||||||
|
sb = ERR_PTR(ret);
|
||||||
goto got_sb;
|
goto got_sb;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for (unsigned i = 0; i < nr_devs; i++)
|
sb = sget(fs_type, bch2_test_super, bch2_noset_super, flags|SB_NOSEC, &devs_to_fs);
|
||||||
devs_to_fs[i] = bch2_path_to_fs(devs[i]);
|
|
||||||
|
|
||||||
sb = sget(fs_type, bch2_test_super, bch2_noset_super,
|
|
||||||
flags|SB_NOSEC, devs_to_fs);
|
|
||||||
if (!IS_ERR(sb))
|
if (!IS_ERR(sb))
|
||||||
goto got_sb;
|
goto got_sb;
|
||||||
|
|
||||||
c = bch2_fs_open(devs, nr_devs, opts);
|
c = bch2_fs_open(devs.data, devs.nr, opts);
|
||||||
if (IS_ERR(c)) {
|
if (IS_ERR(c)) {
|
||||||
sb = ERR_CAST(c);
|
sb = ERR_CAST(c);
|
||||||
goto got_sb;
|
goto got_sb;
|
||||||
@ -1875,9 +1848,8 @@ static struct dentry *bch2_mount(struct file_system_type *fs_type,
|
|||||||
if (IS_ERR(sb))
|
if (IS_ERR(sb))
|
||||||
bch2_fs_stop(c);
|
bch2_fs_stop(c);
|
||||||
got_sb:
|
got_sb:
|
||||||
kfree(devs_to_fs);
|
darray_exit(&devs_to_fs);
|
||||||
kfree(devs[0]);
|
bch2_darray_str_exit(&devs);
|
||||||
kfree(devs);
|
|
||||||
|
|
||||||
if (IS_ERR(sb)) {
|
if (IS_ERR(sb)) {
|
||||||
ret = PTR_ERR(sb);
|
ret = PTR_ERR(sb);
|
||||||
|
@ -1174,3 +1174,37 @@ u64 *bch2_acc_percpu_u64s(u64 __percpu *p, unsigned nr)
|
|||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void bch2_darray_str_exit(darray_str *d)
|
||||||
|
{
|
||||||
|
darray_for_each(*d, i)
|
||||||
|
kfree(*i);
|
||||||
|
darray_exit(d);
|
||||||
|
}
|
||||||
|
|
||||||
|
int bch2_split_devs(const char *_dev_name, darray_str *ret)
|
||||||
|
{
|
||||||
|
darray_init(ret);
|
||||||
|
|
||||||
|
char *dev_name = kstrdup(_dev_name, GFP_KERNEL), *s = dev_name;
|
||||||
|
if (!dev_name)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
while ((s = strsep(&dev_name, ":"))) {
|
||||||
|
char *p = kstrdup(s, GFP_KERNEL);
|
||||||
|
if (!p)
|
||||||
|
goto err;
|
||||||
|
|
||||||
|
if (darray_push(ret, p)) {
|
||||||
|
kfree(p);
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
kfree(dev_name);
|
||||||
|
return 0;
|
||||||
|
err:
|
||||||
|
bch2_darray_str_exit(ret);
|
||||||
|
kfree(dev_name);
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
@ -863,4 +863,7 @@ static inline bool qstr_eq(const struct qstr l, const struct qstr r)
|
|||||||
return l.len == r.len && !memcmp(l.name, r.name, l.len);
|
return l.len == r.len && !memcmp(l.name, r.name, l.len);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void bch2_darray_str_exit(darray_str *);
|
||||||
|
int bch2_split_devs(const char *, darray_str *);
|
||||||
|
|
||||||
#endif /* _BCACHEFS_UTIL_H */
|
#endif /* _BCACHEFS_UTIL_H */
|
||||||
|
Loading…
Reference in New Issue
Block a user