8 #include <linux/compat.h>
9 #include <linux/mount.h>
11 #define FS_IOC_GOINGDOWN _IOR('X', 125, __u32)
15 /* bcachefs inode flags -> vfs inode flags: */
16 static const unsigned bch_flags_to_vfs[] = {
17 [__BCH_INODE_SYNC] = S_SYNC,
18 [__BCH_INODE_IMMUTABLE] = S_IMMUTABLE,
19 [__BCH_INODE_APPEND] = S_APPEND,
20 [__BCH_INODE_NOATIME] = S_NOATIME,
23 /* bcachefs inode flags -> FS_IOC_GETFLAGS: */
24 static const unsigned bch_flags_to_uflags[] = {
25 [__BCH_INODE_SYNC] = FS_SYNC_FL,
26 [__BCH_INODE_IMMUTABLE] = FS_IMMUTABLE_FL,
27 [__BCH_INODE_APPEND] = FS_APPEND_FL,
28 [__BCH_INODE_NODUMP] = FS_NODUMP_FL,
29 [__BCH_INODE_NOATIME] = FS_NOATIME_FL,
32 /* bcachefs inode flags -> FS_IOC_FSGETXATTR: */
33 static const unsigned bch_flags_to_xflags[] = {
34 [__BCH_INODE_SYNC] = FS_XFLAG_SYNC,
35 [__BCH_INODE_IMMUTABLE] = FS_XFLAG_IMMUTABLE,
36 [__BCH_INODE_APPEND] = FS_XFLAG_APPEND,
37 [__BCH_INODE_NODUMP] = FS_XFLAG_NODUMP,
38 [__BCH_INODE_NOATIME] = FS_XFLAG_NOATIME,
39 //[__BCH_INODE_PROJINHERIT] = FS_XFLAG_PROJINHERIT;
42 #define map_flags(_map, _in) \
44 unsigned _i, _out = 0; \
46 for (_i = 0; _i < ARRAY_SIZE(_map); _i++) \
47 if ((_in) & (1 << _i)) \
52 #define map_flags_rev(_map, _in) \
54 unsigned _i, _out = 0; \
56 for (_i = 0; _i < ARRAY_SIZE(_map); _i++) \
57 if ((_in) & _map[_i]) { \
64 #define set_flags(_map, _in, _out) \
68 for (_i = 0; _i < ARRAY_SIZE(_map); _i++) \
69 if ((_in) & (1 << _i)) \
72 (_out) &= ~_map[_i]; \
75 /* Set VFS inode flags from bcachefs inode: */
76 void bch2_inode_flags_to_vfs(struct bch_inode_info *inode)
78 set_flags(bch_flags_to_vfs, inode->ei_inode.bi_flags, inode->v.i_flags);
81 static int bch2_inode_flags_set(struct bch_inode_info *inode,
82 struct bch_inode_unpacked *bi,
86 * We're relying on btree locking here for exclusion with other ioctl
87 * calls - use the flags in the btree (@bi), not inode->i_flags:
89 unsigned newflags = *((unsigned *) p);
90 unsigned oldflags = bi->bi_flags;
92 if (((newflags ^ oldflags) & (BCH_INODE_APPEND|BCH_INODE_IMMUTABLE)) &&
93 !capable(CAP_LINUX_IMMUTABLE))
96 if (!S_ISREG(inode->v.i_mode) &&
97 !S_ISDIR(inode->v.i_mode) &&
98 (newflags & (BCH_INODE_NODUMP|BCH_INODE_NOATIME)) != newflags)
101 bi->bi_flags = newflags;
102 inode->v.i_ctime = current_time(&inode->v);
106 static int bch2_ioc_getflags(struct bch_inode_info *inode, int __user *arg)
108 unsigned flags = map_flags(bch_flags_to_uflags, inode->ei_inode.bi_flags);
110 return put_user(flags, arg);
113 static int bch2_ioc_setflags(struct bch_fs *c,
115 struct bch_inode_info *inode,
118 unsigned flags, uflags;
121 if (get_user(uflags, (int __user *) arg))
124 flags = map_flags_rev(bch_flags_to_uflags, uflags);
128 ret = mnt_want_write_file(file);
132 inode_lock(&inode->v);
133 if (!inode_owner_or_capable(&inode->v)) {
138 mutex_lock(&inode->ei_update_lock);
139 ret = __bch2_write_inode(c, inode, bch2_inode_flags_set, &flags);
142 bch2_inode_flags_to_vfs(inode);
143 mutex_unlock(&inode->ei_update_lock);
146 inode_unlock(&inode->v);
147 mnt_drop_write_file(file);
151 static int bch2_ioc_fsgetxattr(struct bch_inode_info *inode,
152 struct fsxattr __user *arg)
154 struct fsxattr fa = { 0 };
156 fa.fsx_xflags = map_flags(bch_flags_to_xflags, inode->ei_inode.bi_flags);
158 return copy_to_user(arg, &fa, sizeof(fa));
161 static int bch2_ioc_fssetxattr(struct bch_fs *c,
163 struct bch_inode_info *inode,
164 struct fsxattr __user *arg)
170 if (copy_from_user(&fa, arg, sizeof(fa)))
173 flags = map_flags_rev(bch_flags_to_xflags, fa.fsx_xflags);
177 ret = mnt_want_write_file(file);
181 inode_lock(&inode->v);
182 if (!inode_owner_or_capable(&inode->v)) {
187 mutex_lock(&inode->ei_update_lock);
188 ret = __bch2_write_inode(c, inode, bch2_inode_flags_set, &flags);
190 bch2_inode_flags_to_vfs(inode);
191 mutex_unlock(&inode->ei_update_lock);
193 inode_unlock(&inode->v);
194 mnt_drop_write_file(file);
198 long bch2_fs_file_ioctl(struct file *file, unsigned cmd, unsigned long arg)
200 struct bch_inode_info *inode = file_bch_inode(file);
201 struct super_block *sb = inode->v.i_sb;
202 struct bch_fs *c = sb->s_fs_info;
205 case FS_IOC_GETFLAGS:
206 return bch2_ioc_getflags(inode, (int __user *) arg);
208 case FS_IOC_SETFLAGS:
209 return bch2_ioc_setflags(c, file, inode, (int __user *) arg);
211 case FS_IOC_FSGETXATTR:
212 return bch2_ioc_fsgetxattr(inode, (void __user *) arg);
213 case FS_IOC_FSSETXATTR:
214 return bch2_ioc_fssetxattr(c, file, inode, (void __user *) arg);
216 case FS_IOC_GETVERSION:
218 case FS_IOC_SETVERSION:
221 case FS_IOC_GOINGDOWN:
222 if (!capable(CAP_SYS_ADMIN))
225 down_write(&sb->s_umount);
226 sb->s_flags |= MS_RDONLY;
227 bch2_fs_emergency_read_only(c);
228 up_write(&sb->s_umount);
232 return bch2_fs_ioctl(c, cmd, (void __user *) arg);
237 long bch2_compat_fs_ioctl(struct file *file, unsigned cmd, unsigned long arg)
239 /* These are just misnamed, they actually get/put from/to user an int */
241 case FS_IOC_GETFLAGS:
242 cmd = FS_IOC_GETFLAGS;
244 case FS_IOC32_SETFLAGS:
245 cmd = FS_IOC_SETFLAGS;
250 return bch2_fs_file_ioctl(file, cmd, (unsigned long) compat_ptr(arg));
254 #endif /* NO_BCACHEFS_FS */