df: implement -B n<suff> and -B <suff> formats of -B option
[oweals/busybox.git] / util-linux / fstrim.c
1 /* vi: set sw=4 ts=4: */
2 /*
3  * fstrim.c - discard the part (or whole) of mounted filesystem.
4  *
5  * 03 March 2012 - Malek Degachi <malek-degachi@laposte.net>
6  * Adapted for busybox from util-linux-2.12a.
7  *
8  * Licensed under GPLv2 or later, see file LICENSE in this source tree.
9  */
10
11 //config:config FSTRIM
12 //config:       bool "fstrim"
13 //config:       default y
14 //config:       select PLATFORM_LINUX
15 //config:       help
16 //config:         Discard unused blocks on a mounted filesystem.
17
18 //applet:IF_FSTRIM(APPLET(fstrim, BB_DIR_SBIN, BB_SUID_DROP))
19
20 //kbuild:lib-$(CONFIG_FSTRIM) += fstrim.o
21
22 //usage:#define fstrim_trivial_usage
23 //usage:       "[OPTIONS] MOUNTPOINT"
24 //usage:#define fstrim_full_usage "\n\n"
25 //usage:        IF_LONG_OPTS(
26 //usage:       "        -o,--offset=OFFSET      Offset in bytes to discard from"
27 //usage:     "\n        -l,--length=LEN         Bytes to discard"
28 //usage:     "\n        -m,--minimum=MIN        Minimum extent length"
29 //usage:     "\n        -v,--verbose            Print number of discarded bytes"
30 //usage:        )
31 //usage:        IF_NOT_LONG_OPTS(
32 //usage:       "        -o OFFSET       Offset in bytes to discard from"
33 //usage:     "\n        -l LEN          Bytes to discard"
34 //usage:     "\n        -m MIN          Minimum extent length"
35 //usage:     "\n        -v              Print number of discarded bytes"
36 //usage:        )
37
38 #include "libbb.h"
39 #include <linux/fs.h>
40
41 #ifndef FITRIM
42 struct fstrim_range {
43         uint64_t start;
44         uint64_t len;
45         uint64_t minlen;
46 };
47 #define FITRIM          _IOWR('X', 121, struct fstrim_range)
48 #endif
49
50 int fstrim_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
51 int fstrim_main(int argc UNUSED_PARAM, char **argv)
52 {
53         struct fstrim_range range;
54         char *arg_o, *arg_l, *arg_m, *mp;
55         unsigned opts;
56         int fd;
57
58         enum {
59                 OPT_o = (1 << 0),
60                 OPT_l = (1 << 1),
61                 OPT_m = (1 << 2),
62                 OPT_v = (1 << 3),
63         };
64
65 #if ENABLE_LONG_OPTS
66         static const char getopt_longopts[] ALIGN1 =
67                 "offset\0"    Required_argument    "o"
68                 "length\0"    Required_argument    "l"
69                 "minimum\0"   Required_argument    "m"
70                 "verbose\0"   No_argument          "v"
71                 ;
72         applet_long_options = getopt_longopts;
73 #endif
74
75         opt_complementary = "=1"; /* exactly one non-option arg: the mountpoint */
76         opts = getopt32(argv, "o:l:m:v", &arg_o, &arg_l, &arg_m);
77
78         memset(&range, 0, sizeof(range));
79         range.len = ULLONG_MAX;
80
81         if (opts & OPT_o)
82                 range.start = xatoull_sfx(arg_o, kmg_i_suffixes);
83         if (opts & OPT_l)
84                 range.len = xatoull_sfx(arg_l, kmg_i_suffixes);
85         if (opts & OPT_m)
86                 range.minlen = xatoull_sfx(arg_m, kmg_i_suffixes);
87
88         mp = argv[optind];
89         if (find_block_device(mp)) {
90                 fd = xopen_nonblocking(mp);
91                 xioctl(fd, FITRIM, &range);
92                 if (ENABLE_FEATURE_CLEAN_UP)
93                         close(fd);
94
95                 if (opts & OPT_v)
96                         printf("%s: %llu bytes trimmed\n", mp, (unsigned long long)range.len);
97                 return EXIT_SUCCESS;
98         }
99         return EXIT_FAILURE;
100 }