2 * Copyright (C) 2007 Oracle. All rights reserved.
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public
6 * License v2 as published by the Free Software Foundation.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
11 * General Public License for more details.
13 * You should have received a copy of the GNU General Public
14 * License along with this program; if not, write to the
15 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
16 * Boston, MA 021110-1307, USA.
19 #include <linux/sched.h>
20 #include <linux/slab.h>
21 #include <linux/spinlock.h>
22 #include <linux/completion.h>
23 #include <linux/buffer_head.h>
24 #include <linux/kobject.h>
25 #include <linux/bug.h>
29 #include "transaction.h"
32 static inline struct btrfs_fs_info *to_fs_info(struct kobject *kobj);
34 static u64 get_features(struct btrfs_fs_info *fs_info,
35 enum btrfs_feature_set set)
37 struct btrfs_super_block *disk_super = fs_info->super_copy;
38 if (set == FEAT_COMPAT)
39 return btrfs_super_compat_flags(disk_super);
40 else if (set == FEAT_COMPAT_RO)
41 return btrfs_super_compat_ro_flags(disk_super);
43 return btrfs_super_incompat_flags(disk_super);
46 static ssize_t btrfs_feature_attr_show(struct kobject *kobj,
47 struct kobj_attribute *a, char *buf)
50 struct btrfs_fs_info *fs_info = to_fs_info(kobj);
52 struct btrfs_feature_attr *fa = to_btrfs_feature_attr(a);
53 u64 features = get_features(fs_info, fa->feature_set);
54 if (features & fa->feature_bit)
58 return snprintf(buf, PAGE_SIZE, "%d\n", val);
61 static umode_t btrfs_feature_visible(struct kobject *kobj,
62 struct attribute *attr, int unused)
64 struct btrfs_fs_info *fs_info = to_fs_info(kobj);
65 umode_t mode = attr->mode;
68 struct btrfs_feature_attr *fa;
71 fa = attr_to_btrfs_feature_attr(attr);
72 features = get_features(fs_info, fa->feature_set);
74 if (!(features & fa->feature_bit))
81 BTRFS_FEAT_ATTR_INCOMPAT(mixed_backref, MIXED_BACKREF);
82 BTRFS_FEAT_ATTR_INCOMPAT(default_subvol, DEFAULT_SUBVOL);
83 BTRFS_FEAT_ATTR_INCOMPAT(mixed_groups, MIXED_GROUPS);
84 BTRFS_FEAT_ATTR_INCOMPAT(compress_lzo, COMPRESS_LZO);
85 BTRFS_FEAT_ATTR_INCOMPAT(compress_lzov2, COMPRESS_LZOv2);
86 BTRFS_FEAT_ATTR_INCOMPAT(big_metadata, BIG_METADATA);
87 BTRFS_FEAT_ATTR_INCOMPAT(extended_iref, EXTENDED_IREF);
88 BTRFS_FEAT_ATTR_INCOMPAT(raid56, RAID56);
89 BTRFS_FEAT_ATTR_INCOMPAT(skinny_metadata, SKINNY_METADATA);
91 static struct attribute *btrfs_supported_feature_attrs[] = {
92 BTRFS_FEAT_ATTR_PTR(mixed_backref),
93 BTRFS_FEAT_ATTR_PTR(default_subvol),
94 BTRFS_FEAT_ATTR_PTR(mixed_groups),
95 BTRFS_FEAT_ATTR_PTR(compress_lzo),
96 BTRFS_FEAT_ATTR_PTR(compress_lzov2),
97 BTRFS_FEAT_ATTR_PTR(big_metadata),
98 BTRFS_FEAT_ATTR_PTR(extended_iref),
99 BTRFS_FEAT_ATTR_PTR(raid56),
100 BTRFS_FEAT_ATTR_PTR(skinny_metadata),
104 static const struct attribute_group btrfs_feature_attr_group = {
106 .is_visible = btrfs_feature_visible,
107 .attrs = btrfs_supported_feature_attrs,
110 static void btrfs_release_super_kobj(struct kobject *kobj)
112 struct btrfs_fs_info *fs_info = to_fs_info(kobj);
113 complete(&fs_info->kobj_unregister);
116 static struct kobj_type btrfs_ktype = {
117 .sysfs_ops = &kobj_sysfs_ops,
118 .release = btrfs_release_super_kobj,
121 static inline struct btrfs_fs_info *to_fs_info(struct kobject *kobj)
123 if (kobj->ktype != &btrfs_ktype)
125 return container_of(kobj, struct btrfs_fs_info, super_kobj);
128 void btrfs_sysfs_remove_one(struct btrfs_fs_info *fs_info)
130 kobject_del(&fs_info->super_kobj);
131 kobject_put(&fs_info->super_kobj);
132 wait_for_completion(&fs_info->kobj_unregister);
135 const char * const btrfs_feature_set_names[3] = {
136 [FEAT_COMPAT] = "compat",
137 [FEAT_COMPAT_RO] = "compat_ro",
138 [FEAT_INCOMPAT] = "incompat",
141 #define NUM_FEATURE_BITS 64
142 static char btrfs_unknown_feature_names[3][NUM_FEATURE_BITS][13];
143 static struct btrfs_feature_attr btrfs_feature_attrs[3][NUM_FEATURE_BITS];
145 static void init_feature_attrs(void)
147 struct btrfs_feature_attr *fa;
150 BUILD_BUG_ON(ARRAY_SIZE(btrfs_unknown_feature_names) !=
151 ARRAY_SIZE(btrfs_feature_attrs));
152 BUILD_BUG_ON(ARRAY_SIZE(btrfs_unknown_feature_names[0]) !=
153 ARRAY_SIZE(btrfs_feature_attrs[0]));
155 for (i = 0; btrfs_supported_feature_attrs[i]; i++) {
156 struct btrfs_feature_attr *sfa;
157 struct attribute *a = btrfs_supported_feature_attrs[i];
158 sfa = attr_to_btrfs_feature_attr(a);
159 fa = &btrfs_feature_attrs[sfa->feature_set][sfa->feature_bit];
161 fa->kobj_attr.attr.name = sfa->kobj_attr.attr.name;
164 for (set = 0; set < FEAT_MAX; set++) {
165 for (i = 0; i < ARRAY_SIZE(btrfs_feature_attrs[set]); i++) {
166 char *name = btrfs_unknown_feature_names[set][i];
167 fa = &btrfs_feature_attrs[set][i];
169 if (fa->kobj_attr.attr.name)
172 snprintf(name, 13, "%s:%u",
173 btrfs_feature_set_names[set], i);
175 fa->kobj_attr.attr.name = name;
176 fa->kobj_attr.attr.mode = S_IRUGO;
177 fa->feature_set = set;
178 fa->feature_bit = 1ULL << i;
183 static u64 supported_feature_masks[3] = {
184 [FEAT_COMPAT] = BTRFS_FEATURE_COMPAT_SUPP,
185 [FEAT_COMPAT_RO] = BTRFS_FEATURE_COMPAT_RO_SUPP,
186 [FEAT_INCOMPAT] = BTRFS_FEATURE_INCOMPAT_SUPP,
189 static int add_unknown_feature_attrs(struct btrfs_fs_info *fs_info)
193 for (set = 0; set < FEAT_MAX; set++) {
194 int i, count, ret, index = 0;
195 struct attribute **attrs;
196 struct attribute_group agroup = {
199 u64 features = get_features(fs_info, set);
200 features &= ~supported_feature_masks[set];
202 count = hweight64(features);
207 attrs = kcalloc(count + 1, sizeof(void *), GFP_KERNEL);
209 for (i = 0; i < NUM_FEATURE_BITS; i++) {
210 struct btrfs_feature_attr *fa;
212 if (!(features & (1ULL << i)))
215 fa = &btrfs_feature_attrs[set][i];
216 attrs[index++] = &fa->kobj_attr.attr;
220 agroup.attrs = attrs;
222 ret = sysfs_merge_group(&fs_info->super_kobj, &agroup);
230 /* /sys/fs/btrfs/ entry */
231 static struct kset *btrfs_kset;
233 int btrfs_sysfs_add_one(struct btrfs_fs_info *fs_info)
237 init_completion(&fs_info->kobj_unregister);
238 fs_info->super_kobj.kset = btrfs_kset;
239 error = kobject_init_and_add(&fs_info->super_kobj, &btrfs_ktype, NULL,
240 "%pU", fs_info->fsid);
242 error = sysfs_create_group(&fs_info->super_kobj,
243 &btrfs_feature_attr_group);
247 error = add_unknown_feature_attrs(fs_info);
253 btrfs_sysfs_remove_one(fs_info);
257 int btrfs_init_sysfs(void)
260 btrfs_kset = kset_create_and_add("btrfs", NULL, fs_kobj);
264 init_feature_attrs();
266 ret = sysfs_create_group(&btrfs_kset->kobj, &btrfs_feature_attr_group);
268 kset_unregister(btrfs_kset);
275 void btrfs_exit_sysfs(void)
277 sysfs_remove_group(&btrfs_kset->kobj, &btrfs_feature_attr_group);
278 kset_unregister(btrfs_kset);