1 // SPDX-License-Identifier: GPL-2.0
3 #include "disk_groups.h"
4 #include "sb-members.h"
7 #include <linux/sort.h>
9 static int group_cmp(const void *_l, const void *_r)
11 const struct bch_disk_group *l = _l;
12 const struct bch_disk_group *r = _r;
14 return ((BCH_GROUP_DELETED(l) > BCH_GROUP_DELETED(r)) -
15 (BCH_GROUP_DELETED(l) < BCH_GROUP_DELETED(r))) ?:
16 ((BCH_GROUP_PARENT(l) > BCH_GROUP_PARENT(r)) -
17 (BCH_GROUP_PARENT(l) < BCH_GROUP_PARENT(r))) ?:
18 strncmp(l->label, r->label, sizeof(l->label));
21 static int bch2_sb_disk_groups_validate(struct bch_sb *sb, struct bch_sb_field *f,
22 enum bch_validate_flags flags, struct printbuf *err)
24 struct bch_sb_field_disk_groups *groups =
25 field_to_type(f, disk_groups);
26 struct bch_disk_group *g, *sorted = NULL;
27 unsigned nr_groups = disk_groups_nr(groups);
31 for (i = 0; i < sb->nr_devices; i++) {
32 struct bch_member m = bch2_sb_member_get(sb, i);
35 if (!BCH_MEMBER_GROUP(&m))
38 group_id = BCH_MEMBER_GROUP(&m) - 1;
40 if (group_id >= nr_groups) {
41 prt_printf(err, "disk %u has invalid label %u (have %u)",
42 i, group_id, nr_groups);
43 return -BCH_ERR_invalid_sb_disk_groups;
46 if (BCH_GROUP_DELETED(&groups->entries[group_id])) {
47 prt_printf(err, "disk %u has deleted label %u", i, group_id);
48 return -BCH_ERR_invalid_sb_disk_groups;
55 for (i = 0; i < nr_groups; i++) {
56 g = groups->entries + i;
58 if (BCH_GROUP_DELETED(g))
61 len = strnlen(g->label, sizeof(g->label));
63 prt_printf(err, "label %u empty", i);
64 return -BCH_ERR_invalid_sb_disk_groups;
68 sorted = kmalloc_array(nr_groups, sizeof(*sorted), GFP_KERNEL);
70 return -BCH_ERR_ENOMEM_disk_groups_validate;
72 memcpy(sorted, groups->entries, nr_groups * sizeof(*sorted));
73 sort(sorted, nr_groups, sizeof(*sorted), group_cmp, NULL);
75 for (g = sorted; g + 1 < sorted + nr_groups; g++)
76 if (!BCH_GROUP_DELETED(g) &&
77 !group_cmp(&g[0], &g[1])) {
78 prt_printf(err, "duplicate label %llu.%.*s",
80 (int) sizeof(g->label), g->label);
81 ret = -BCH_ERR_invalid_sb_disk_groups;
89 void bch2_disk_groups_to_text(struct printbuf *out, struct bch_fs *c)
94 struct bch_disk_groups_cpu *g = rcu_dereference(c->disk_groups);
98 for (unsigned i = 0; i < g->nr; i++) {
100 prt_printf(out, " ");
102 if (g->entries[i].deleted) {
103 prt_printf(out, "[deleted]");
107 prt_printf(out, "[parent %d devs", g->entries[i].parent);
108 for_each_member_device_rcu(c, ca, &g->entries[i].devs)
109 prt_printf(out, " %s", ca->name);
110 prt_printf(out, "]");
118 static void bch2_sb_disk_groups_to_text(struct printbuf *out,
120 struct bch_sb_field *f)
122 struct bch_sb_field_disk_groups *groups =
123 field_to_type(f, disk_groups);
124 struct bch_disk_group *g;
125 unsigned nr_groups = disk_groups_nr(groups);
127 for (g = groups->entries;
128 g < groups->entries + nr_groups;
130 if (g != groups->entries)
131 prt_printf(out, " ");
133 if (BCH_GROUP_DELETED(g))
134 prt_printf(out, "[deleted]");
136 prt_printf(out, "[parent %llu name %s]",
137 BCH_GROUP_PARENT(g), g->label);
141 const struct bch_sb_field_ops bch_sb_field_ops_disk_groups = {
142 .validate = bch2_sb_disk_groups_validate,
143 .to_text = bch2_sb_disk_groups_to_text
146 int bch2_sb_disk_groups_to_cpu(struct bch_fs *c)
148 struct bch_sb_field_disk_groups *groups;
149 struct bch_disk_groups_cpu *cpu_g, *old_g;
150 unsigned i, g, nr_groups;
152 lockdep_assert_held(&c->sb_lock);
154 groups = bch2_sb_field_get(c->disk_sb.sb, disk_groups);
155 nr_groups = disk_groups_nr(groups);
160 cpu_g = kzalloc(struct_size(cpu_g, entries, nr_groups), GFP_KERNEL);
162 return -BCH_ERR_ENOMEM_disk_groups_to_cpu;
164 cpu_g->nr = nr_groups;
166 for (i = 0; i < nr_groups; i++) {
167 struct bch_disk_group *src = &groups->entries[i];
168 struct bch_disk_group_cpu *dst = &cpu_g->entries[i];
170 dst->deleted = BCH_GROUP_DELETED(src);
171 dst->parent = BCH_GROUP_PARENT(src);
172 memcpy(dst->label, src->label, sizeof(dst->label));
175 for (i = 0; i < c->disk_sb.sb->nr_devices; i++) {
176 struct bch_member m = bch2_sb_member_get(c->disk_sb.sb, i);
177 struct bch_disk_group_cpu *dst;
179 if (!bch2_member_alive(&m))
182 g = BCH_MEMBER_GROUP(&m);
184 dst = &cpu_g->entries[g - 1];
185 __set_bit(i, dst->devs.d);
190 old_g = rcu_dereference_protected(c->disk_groups,
191 lockdep_is_held(&c->sb_lock));
192 rcu_assign_pointer(c->disk_groups, cpu_g);
194 kfree_rcu(old_g, rcu);
199 const struct bch_devs_mask *bch2_target_to_mask(struct bch_fs *c, unsigned target)
201 struct target t = target_decode(target);
202 struct bch_devs_mask *devs;
211 struct bch_dev *ca = t.dev < c->sb.nr_devices
212 ? rcu_dereference(c->devs[t.dev])
214 devs = ca ? &ca->self : NULL;
218 struct bch_disk_groups_cpu *g = rcu_dereference(c->disk_groups);
220 devs = g && t.group < g->nr && !g->entries[t.group].deleted
221 ? &g->entries[t.group].devs
234 bool bch2_dev_in_target(struct bch_fs *c, unsigned dev, unsigned target)
236 struct target t = target_decode(target);
244 struct bch_disk_groups_cpu *g;
245 const struct bch_devs_mask *m;
249 g = rcu_dereference(c->disk_groups);
250 m = g && t.group < g->nr && !g->entries[t.group].deleted
251 ? &g->entries[t.group].devs
254 ret = m ? test_bit(dev, m->d) : false;
264 static int __bch2_disk_group_find(struct bch_sb_field_disk_groups *groups,
266 const char *name, unsigned namelen)
268 unsigned i, nr_groups = disk_groups_nr(groups);
270 if (!namelen || namelen > BCH_SB_LABEL_SIZE)
273 for (i = 0; i < nr_groups; i++) {
274 struct bch_disk_group *g = groups->entries + i;
276 if (BCH_GROUP_DELETED(g))
279 if (!BCH_GROUP_DELETED(g) &&
280 BCH_GROUP_PARENT(g) == parent &&
281 strnlen(g->label, sizeof(g->label)) == namelen &&
282 !memcmp(name, g->label, namelen))
289 static int __bch2_disk_group_add(struct bch_sb_handle *sb, unsigned parent,
290 const char *name, unsigned namelen)
292 struct bch_sb_field_disk_groups *groups =
293 bch2_sb_field_get(sb->sb, disk_groups);
294 unsigned i, nr_groups = disk_groups_nr(groups);
295 struct bch_disk_group *g;
297 if (!namelen || namelen > BCH_SB_LABEL_SIZE)
301 i < nr_groups && !BCH_GROUP_DELETED(&groups->entries[i]);
305 if (i == nr_groups) {
307 (sizeof(struct bch_sb_field_disk_groups) +
308 sizeof(struct bch_disk_group) * (nr_groups + 1)) /
311 groups = bch2_sb_field_resize(sb, disk_groups, u64s);
313 return -BCH_ERR_ENOSPC_disk_label_add;
315 nr_groups = disk_groups_nr(groups);
318 BUG_ON(i >= nr_groups);
320 g = &groups->entries[i];
322 memcpy(g->label, name, namelen);
323 if (namelen < sizeof(g->label))
324 g->label[namelen] = '\0';
325 SET_BCH_GROUP_DELETED(g, 0);
326 SET_BCH_GROUP_PARENT(g, parent);
327 SET_BCH_GROUP_DATA_ALLOWED(g, ~0);
332 int bch2_disk_path_find(struct bch_sb_handle *sb, const char *name)
334 struct bch_sb_field_disk_groups *groups =
335 bch2_sb_field_get(sb->sb, disk_groups);
339 const char *next = strchrnul(name, '.');
340 unsigned len = next - name;
345 v = __bch2_disk_group_find(groups, v + 1, name, len);
347 } while (*name && v >= 0);
352 int bch2_disk_path_find_or_create(struct bch_sb_handle *sb, const char *name)
354 struct bch_sb_field_disk_groups *groups;
359 const char *next = strchrnul(name, '.');
360 unsigned len = next - name;
365 groups = bch2_sb_field_get(sb->sb, disk_groups);
367 v = __bch2_disk_group_find(groups, parent, name, len);
369 v = __bch2_disk_group_add(sb, parent, name, len);
375 } while (*name && v >= 0);
380 void bch2_disk_path_to_text(struct printbuf *out, struct bch_fs *c, unsigned v)
382 struct bch_disk_groups_cpu *groups;
383 struct bch_disk_group_cpu *g;
389 groups = rcu_dereference(c->disk_groups);
394 if (nr == ARRAY_SIZE(path))
400 g = groups->entries + v;
415 g = groups->entries + v;
417 prt_printf(out, "%.*s", (int) sizeof(g->label), g->label);
419 prt_printf(out, ".");
426 prt_printf(out, "invalid label %u", v);
430 void bch2_disk_path_to_text_sb(struct printbuf *out, struct bch_sb *sb, unsigned v)
432 struct bch_sb_field_disk_groups *groups =
433 bch2_sb_field_get(sb, disk_groups);
434 struct bch_disk_group *g;
439 if (nr == ARRAY_SIZE(path))
442 if (v >= disk_groups_nr(groups))
445 g = groups->entries + v;
447 if (BCH_GROUP_DELETED(g))
452 if (!BCH_GROUP_PARENT(g))
455 v = BCH_GROUP_PARENT(g) - 1;
460 g = groups->entries + v;
462 prt_printf(out, "%.*s", (int) sizeof(g->label), g->label);
464 prt_printf(out, ".");
468 prt_printf(out, "invalid label %u", v);
471 int __bch2_dev_group_set(struct bch_fs *c, struct bch_dev *ca, const char *name)
473 struct bch_member *mi;
476 if (!strlen(name) || !strcmp(name, "none"))
479 v = bch2_disk_path_find_or_create(&c->disk_sb, name);
483 ret = bch2_sb_disk_groups_to_cpu(c);
487 mi = bch2_members_v2_get_mut(c->disk_sb.sb, ca->dev_idx);
488 SET_BCH_MEMBER_GROUP(mi, v + 1);
492 int bch2_dev_group_set(struct bch_fs *c, struct bch_dev *ca, const char *name)
496 mutex_lock(&c->sb_lock);
497 ret = __bch2_dev_group_set(c, ca, name) ?:
499 mutex_unlock(&c->sb_lock);
504 int bch2_opt_target_parse(struct bch_fs *c, const char *val, u64 *res,
505 struct printbuf *err)
514 return -BCH_ERR_option_needs_open_fs;
516 if (!strlen(val) || !strcmp(val, "none")) {
521 /* Is it a device? */
522 ca = bch2_dev_lookup(c, val);
524 *res = dev_to_target(ca->dev_idx);
529 mutex_lock(&c->sb_lock);
530 g = bch2_disk_path_find(&c->disk_sb, val);
531 mutex_unlock(&c->sb_lock);
534 *res = group_to_target(g);
541 void bch2_target_to_text(struct printbuf *out, struct bch_fs *c, unsigned v)
543 struct target t = target_decode(v);
547 prt_printf(out, "none");
554 ca = t.dev < c->sb.nr_devices
555 ? rcu_dereference(c->devs[t.dev])
558 if (ca && percpu_ref_tryget(&ca->io_ref)) {
559 prt_printf(out, "/dev/%s", ca->name);
560 percpu_ref_put(&ca->io_ref);
562 prt_printf(out, "offline device %u", t.dev);
564 prt_printf(out, "invalid device %u", t.dev);
572 bch2_disk_path_to_text(out, c, t.group);
579 static void bch2_target_to_text_sb(struct printbuf *out, struct bch_sb *sb, unsigned v)
581 struct target t = target_decode(v);
585 prt_printf(out, "none");
588 struct bch_member m = bch2_sb_member_get(sb, t.dev);
590 if (bch2_member_exists(sb, t.dev)) {
591 prt_printf(out, "Device ");
592 pr_uuid(out, m.uuid.b);
593 prt_printf(out, " (%u)", t.dev);
595 prt_printf(out, "Bad device %u", t.dev);
600 bch2_disk_path_to_text_sb(out, sb, t.group);
607 void bch2_opt_target_to_text(struct printbuf *out,
613 bch2_target_to_text(out, c, v);
615 bch2_target_to_text_sb(out, sb, v);