-
Notifications
You must be signed in to change notification settings - Fork 93
/
cmd_fs.c
159 lines (119 loc) · 3.57 KB
/
cmd_fs.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
#include <stdio.h>
#include <sys/ioctl.h>
#include <uuid/uuid.h>
#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 print_dev_usage(struct bch_ioctl_dev_usage *d, unsigned idx,
const char *label, enum units units)
{
char *name = NULL;
u64 available = d->nr_buckets;
unsigned i;
printf("\n");
printf_pad(20, "%s (device %u):", label, idx);
name = !d->dev ? strdup("(offline)")
: dev_to_path(d->dev)
?: strdup("(device not found)");
printf("%24s%12s\n", name, bch2_dev_state[d->state]);
free(name);
printf("%-20s%12s%12s%12s\n",
"", "data", "buckets", "fragmented");
for (i = BCH_DATA_SB; i < BCH_DATA_NR; i++) {
u64 frag = max((s64) d->buckets[i] * d->bucket_size -
(s64) d->sectors[i], 0LL);
printf_pad(20, " %s:", bch2_data_types[i]);
printf("%12s%12llu%12s\n",
pr_units(d->sectors[i], units),
d->buckets[i],
pr_units(frag, units));
if (i != BCH_DATA_CACHED)
available -= d->buckets[i];
}
printf_pad(20, " available:");
printf("%12s%12llu\n",
pr_units(available * d->bucket_size, units),
available);
printf_pad(20, " capacity:");
printf("%12s%12llu\n",
pr_units(d->nr_buckets * d->bucket_size, units),
d->nr_buckets);
}
struct dev_by_label {
unsigned idx;
char *label;
};
static int dev_by_label_cmp(const void *_l, const void *_r)
{
const struct dev_by_label *l = _l, *r = _r;
return strcmp(l->label, r->label);
}
static void print_fs_usage(const char *path, enum units units)
{
unsigned i, j;
char uuid[40];
struct bchfs_handle fs = bcache_fs_open(path);
struct bch_ioctl_usage *u = bchu_usage(fs);
uuid_unparse(fs.uuid.b, uuid);
printf("Filesystem %s:\n", uuid);
printf("%-20s%12s\n", "Size:", pr_units(u->fs.capacity, units));
printf("%-20s%12s\n", "Used:", pr_units(u->fs.used, units));
printf("%-20s%12s%12s%12s%12s\n",
"By replicas:", "1x", "2x", "3x", "4x");
for (j = BCH_DATA_BTREE; j < BCH_DATA_NR; j++) {
printf_pad(20, " %s:", bch2_data_types[j]);
for (i = 0; i < BCH_REPLICAS_MAX; i++)
printf("%12s", pr_units(u->fs.sectors[j][i], units));
printf("\n");
}
printf_pad(20, " %s:", "reserved");
for (i = 0; i < BCH_REPLICAS_MAX; i++)
printf("%12s", pr_units(u->fs.persistent_reserved[i], units));
printf("\n");
printf("%-20s%12s\n", " online reserved:", pr_units(u->fs.online_reserved, units));
darray(struct dev_by_label) devs_by_label;
darray_init(devs_by_label);
for (i = 0; i < u->nr_devices; i++) {
struct bch_ioctl_dev_usage *d = u->devs + i;
if (!d->alive)
continue;
char *label_attr = mprintf("dev-%u/label", i);
char *label = read_file_str(fs.sysfs_fd, label_attr);
free(label_attr);
darray_append(devs_by_label,
(struct dev_by_label) { i, label });
}
sort(&darray_item(devs_by_label, 0), darray_size(devs_by_label),
sizeof(darray_item(devs_by_label, 0)), dev_by_label_cmp, NULL);
struct dev_by_label *d;
darray_foreach(d, devs_by_label)
print_dev_usage(u->devs + d->idx, d->idx, d->label, units);
darray_foreach(d, devs_by_label)
free(d->label);
darray_free(devs_by_label);
free(u);
bcache_fs_close(fs);
}
int cmd_fs_usage(int argc, char *argv[])
{
enum units units = BYTES;
char *fs;
int opt;
while ((opt = getopt(argc, argv, "h")) != -1)
switch (opt) {
case 'h':
units = HUMAN_READABLE;
break;
}
args_shift(optind);
if (!argc) {
print_fs_usage(".", units);
} else {
while ((fs = arg_pop()))
print_fs_usage(fs, units);
}
return 0;
}