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:
Kent Overstreet 2023-12-22 21:10:32 -05:00
parent 0c0ba8e9c5
commit 806ebf2aa0
4 changed files with 58 additions and 48 deletions

View File

@ -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);

View File

@ -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));
goto got_sb; if (ret) {
sb = ERR_PTR(ret);
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);

View File

@ -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;
}

View File

@ -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 */