#include #include #include #include "ccan/darray/darray.h" #include "linux/sort.h" #include "libbcachefs/bcachefs_ioctl.h" #include "libbcachefs/opts.h" #include "cmds.h" #include "libbcachefs.h" static void dev_usage_type_to_text(struct printbuf *out, const char *type, unsigned bucket_size, u64 buckets, u64 sectors) { u64 frag = max((s64) buckets * bucket_size - (s64) sectors, 0LL); pr_buf(out, "%s:", type); pr_tab(out); pr_sectors(out, sectors); pr_tab_rjust(out); pr_buf(out, "%llu", buckets); pr_tab_rjust(out); pr_sectors(out, frag); pr_tab_rjust(out); pr_newline(out); } static void dev_usage_to_text(struct printbuf *out, struct bchfs_handle fs, struct dev_name *d) { struct bch_ioctl_dev_usage u = bchu_dev_usage(fs, d->idx); unsigned i; pr_newline(out); pr_buf(out, "%s (device %u):", d->label ?: "(no label)", d->idx); pr_tab(out); pr_buf(out, "%s", d->dev ?: "(device not found)"); pr_tab_rjust(out); pr_buf(out, "%s", bch2_member_states[u.state]); pr_tab_rjust(out); pr_newline(out); pr_indent_push(out, 2); pr_tab(out); pr_buf(out, "data"); pr_tab_rjust(out); pr_buf(out, "buckets"); pr_tab_rjust(out); pr_buf(out, "fragmented"); pr_tab_rjust(out); pr_newline(out); for (i = BCH_DATA_sb; i < BCH_DATA_NR; i++) dev_usage_type_to_text(out, bch2_data_types[i], u.bucket_size, u.buckets[i], u.sectors[i]); dev_usage_type_to_text(out, "erasure coded", u.bucket_size, u.ec_buckets, u.ec_sectors); pr_buf(out, "available:"); pr_tab(out); pr_sectors(out, u.available_buckets * u.bucket_size); pr_tab_rjust(out); pr_buf(out, "%llu", u.available_buckets); pr_tab_rjust(out); pr_newline(out); pr_buf(out, "capacity:"); pr_tab(out); pr_sectors(out, u.nr_buckets * u.bucket_size); pr_tab_rjust(out); pr_buf(out, "%llu", u.nr_buckets); pr_tab_rjust(out); pr_indent_pop(out, 2); pr_newline(out); } static int dev_by_label_cmp(const void *_l, const void *_r) { const struct dev_name *l = _l, *r = _r; return (l->label && r->label ? strcmp(l->label, r->label) : 0) ?: (l->dev && r->dev ? strcmp(l->dev, r->dev) : 0) ?: cmp_int(l->idx, r->idx); } static struct dev_name *dev_idx_to_name(dev_names *dev_names, unsigned idx) { struct dev_name *dev; darray_foreach(dev, *dev_names) if (dev->idx == idx) return dev; return NULL; } static void replicas_usage_to_text(struct printbuf *out, const struct bch_replicas_usage *r, dev_names *dev_names) { unsigned i; if (!r->sectors) return; char devs[4096], *d = devs; *d++ = '['; for (i = 0; i < r->r.nr_devs; i++) { unsigned dev_idx = r->r.devs[i]; struct dev_name *dev = dev_idx_to_name(dev_names, dev_idx); if (i) *d++ = ' '; d += dev && dev->dev ? sprintf(d, "%s", dev->dev) : sprintf(d, "%u", dev_idx); } *d++ = ']'; *d++ = '\0'; pr_buf(out, "%s: ", bch2_data_types[r->r.data_type]); pr_tab(out); pr_buf(out, "%u/%u ", r->r.nr_required, r->r.nr_devs); pr_tab(out); pr_buf(out, "%s ", devs); pr_tab(out); pr_sectors(out, r->sectors); pr_tab_rjust(out); pr_newline(out); } #define for_each_usage_replica(_u, _r) \ for (_r = (_u)->replicas; \ _r != (void *) (_u)->replicas + (_u)->replica_entries_bytes;\ _r = replicas_usage_next(_r), \ BUG_ON((void *) _r > (void *) (_u)->replicas + (_u)->replica_entries_bytes)) static void fs_usage_to_text(struct printbuf *out, const char *path) { unsigned i; struct bchfs_handle fs = bcache_fs_open(path); struct dev_name *dev; dev_names dev_names = bchu_fs_get_devices(fs); struct bch_ioctl_fs_usage *u = bchu_fs_usage(fs); pr_buf(out, "Filesystem: "); pr_uuid(out, fs.uuid.b); pr_newline(out); out->tabstops[0] = 20; out->tabstops[1] = 36; pr_buf(out, "Size:"); pr_tab(out); pr_sectors(out, u->capacity); pr_tab_rjust(out); pr_newline(out); pr_buf(out, "Used:"); pr_tab(out); pr_sectors(out, u->used); pr_tab_rjust(out); pr_newline(out); pr_buf(out, "Online reserved:"); pr_tab(out); pr_sectors(out, u->online_reserved); pr_tab_rjust(out); pr_newline(out); pr_newline(out); out->tabstops[0] = 16; out->tabstops[1] = 32; out->tabstops[2] = 50; out->tabstops[3] = 68; pr_buf(out, "Data type"); pr_tab(out); pr_buf(out, "Required/total"); pr_tab(out); pr_buf(out, "Devices"); pr_newline(out); for (i = 0; i < BCH_REPLICAS_MAX; i++) { if (!u->persistent_reserved[i]) continue; pr_buf(out, "reserved:"); pr_tab(out); pr_buf(out, "%u/%u ", 1, i); pr_tab(out); pr_buf(out, "[] "); pr_sectors(out, u->persistent_reserved[i]); pr_tab_rjust(out); pr_newline(out); } struct bch_replicas_usage *r; for_each_usage_replica(u, r) if (r->r.data_type < BCH_DATA_user) replicas_usage_to_text(out, r, &dev_names); for_each_usage_replica(u, r) if (r->r.data_type == BCH_DATA_user && r->r.nr_required <= 1) replicas_usage_to_text(out, r, &dev_names); for_each_usage_replica(u, r) if (r->r.data_type == BCH_DATA_user && r->r.nr_required > 1) replicas_usage_to_text(out, r, &dev_names); for_each_usage_replica(u, r) if (r->r.data_type > BCH_DATA_user) replicas_usage_to_text(out, r, &dev_names); free(u); sort(&darray_item(dev_names, 0), darray_size(dev_names), sizeof(darray_item(dev_names, 0)), dev_by_label_cmp, NULL); out->tabstops[0] = 16; out->tabstops[1] = 36; out->tabstops[2] = 52; out->tabstops[3] = 68; darray_foreach(dev, dev_names) dev_usage_to_text(out, fs, dev); darray_foreach(dev, dev_names) { free(dev->dev); free(dev->label); } darray_free(dev_names); bcache_fs_close(fs); } int fs_usage(void) { puts("bcachefs fs - manage a running filesystem\n" "Usage: bcachefs fs [OPTION]... path\n" "\n" "Commands:\n" " usage show disk usage\n" "\n" "Report bugs to "); return 0; } int cmd_fs_usage(int argc, char *argv[]) { enum printbuf_units units = PRINTBUF_UNITS_BYTES; struct printbuf buf = PRINTBUF; char *fs; int opt; while ((opt = getopt(argc, argv, "h")) != -1) switch (opt) { case 'h': units = PRINTBUF_UNITS_HUMAN_READABLE; break; } args_shift(optind); if (!argc) { printbuf_reset(&buf); buf.units = units; fs_usage_to_text(&buf, "."); printf("%s", buf.buf); } else { while ((fs = arg_pop())) { printbuf_reset(&buf); buf.units = units; fs_usage_to_text(&buf, fs); printf("%s", buf.buf); } } printbuf_exit(&buf); return 0; }