diff options
-rw-r--r-- | fs/ioctl.c | 39 | ||||
-rw-r--r-- | include/linux/fs.h | 8 |
2 files changed, 47 insertions, 0 deletions
diff --git a/fs/ioctl.c b/fs/ioctl.c index f855ea4fc888..e92fdbb3bc3a 100644 --- a/fs/ioctl.c +++ b/fs/ioctl.c | |||
@@ -530,6 +530,41 @@ static int ioctl_fsthaw(struct file *filp) | |||
530 | return thaw_super(sb); | 530 | return thaw_super(sb); |
531 | } | 531 | } |
532 | 532 | ||
533 | static int ioctl_fstrim(struct file *filp, void __user *argp) | ||
534 | { | ||
535 | struct super_block *sb = filp->f_path.dentry->d_inode->i_sb; | ||
536 | struct fstrim_range range; | ||
537 | int ret = 0; | ||
538 | |||
539 | if (!capable(CAP_SYS_ADMIN)) | ||
540 | return -EPERM; | ||
541 | |||
542 | /* If filesystem doesn't support trim feature, return. */ | ||
543 | if (sb->s_op->trim_fs == NULL) | ||
544 | return -EOPNOTSUPP; | ||
545 | |||
546 | /* If a blockdevice-backed filesystem isn't specified, return EINVAL. */ | ||
547 | if (sb->s_bdev == NULL) | ||
548 | return -EINVAL; | ||
549 | |||
550 | if (argp == NULL) { | ||
551 | range.start = 0; | ||
552 | range.len = ULLONG_MAX; | ||
553 | range.minlen = 0; | ||
554 | } else if (copy_from_user(&range, argp, sizeof(range))) | ||
555 | return -EFAULT; | ||
556 | |||
557 | ret = sb->s_op->trim_fs(sb, &range); | ||
558 | if (ret < 0) | ||
559 | return ret; | ||
560 | |||
561 | if ((argp != NULL) && | ||
562 | (copy_to_user(argp, &range, sizeof(range)))) | ||
563 | return -EFAULT; | ||
564 | |||
565 | return 0; | ||
566 | } | ||
567 | |||
533 | /* | 568 | /* |
534 | * When you add any new common ioctls to the switches above and below | 569 | * When you add any new common ioctls to the switches above and below |
535 | * please update compat_sys_ioctl() too. | 570 | * please update compat_sys_ioctl() too. |
@@ -580,6 +615,10 @@ int do_vfs_ioctl(struct file *filp, unsigned int fd, unsigned int cmd, | |||
580 | error = ioctl_fsthaw(filp); | 615 | error = ioctl_fsthaw(filp); |
581 | break; | 616 | break; |
582 | 617 | ||
618 | case FITRIM: | ||
619 | error = ioctl_fstrim(filp, argp); | ||
620 | break; | ||
621 | |||
583 | case FS_IOC_FIEMAP: | 622 | case FS_IOC_FIEMAP: |
584 | return ioctl_fiemap(filp, arg); | 623 | return ioctl_fiemap(filp, arg); |
585 | 624 | ||
diff --git a/include/linux/fs.h b/include/linux/fs.h index 63d069bd80b7..7008268e9b5a 100644 --- a/include/linux/fs.h +++ b/include/linux/fs.h | |||
@@ -32,6 +32,12 @@ | |||
32 | #define SEEK_END 2 /* seek relative to end of file */ | 32 | #define SEEK_END 2 /* seek relative to end of file */ |
33 | #define SEEK_MAX SEEK_END | 33 | #define SEEK_MAX SEEK_END |
34 | 34 | ||
35 | struct fstrim_range { | ||
36 | uint64_t start; | ||
37 | uint64_t len; | ||
38 | uint64_t minlen; | ||
39 | }; | ||
40 | |||
35 | /* And dynamically-tunable limits and defaults: */ | 41 | /* And dynamically-tunable limits and defaults: */ |
36 | struct files_stat_struct { | 42 | struct files_stat_struct { |
37 | int nr_files; /* read only */ | 43 | int nr_files; /* read only */ |
@@ -316,6 +322,7 @@ struct inodes_stat_t { | |||
316 | #define FIGETBSZ _IO(0x00,2) /* get the block size used for bmap */ | 322 | #define FIGETBSZ _IO(0x00,2) /* get the block size used for bmap */ |
317 | #define FIFREEZE _IOWR('X', 119, int) /* Freeze */ | 323 | #define FIFREEZE _IOWR('X', 119, int) /* Freeze */ |
318 | #define FITHAW _IOWR('X', 120, int) /* Thaw */ | 324 | #define FITHAW _IOWR('X', 120, int) /* Thaw */ |
325 | #define FITRIM _IOWR('X', 121, struct fstrim_range) /* Trim */ | ||
319 | 326 | ||
320 | #define FS_IOC_GETFLAGS _IOR('f', 1, long) | 327 | #define FS_IOC_GETFLAGS _IOR('f', 1, long) |
321 | #define FS_IOC_SETFLAGS _IOW('f', 2, long) | 328 | #define FS_IOC_SETFLAGS _IOW('f', 2, long) |
@@ -1581,6 +1588,7 @@ struct super_operations { | |||
1581 | ssize_t (*quota_write)(struct super_block *, int, const char *, size_t, loff_t); | 1588 | ssize_t (*quota_write)(struct super_block *, int, const char *, size_t, loff_t); |
1582 | #endif | 1589 | #endif |
1583 | int (*bdev_try_to_free_page)(struct super_block*, struct page*, gfp_t); | 1590 | int (*bdev_try_to_free_page)(struct super_block*, struct page*, gfp_t); |
1591 | int (*trim_fs) (struct super_block *, struct fstrim_range *); | ||
1584 | }; | 1592 | }; |
1585 | 1593 | ||
1586 | /* | 1594 | /* |