btrfs-progs: kerncompat: disconnect assert and warning messages
[platform/upstream/btrfs-progs.git] / cmds-quota.c
1 /*
2  * Copyright (C) 2012 STRATO.  All rights reserved.
3  *
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.
7  *
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.
12  *
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.
17  */
18
19 #include <sys/ioctl.h>
20 #include <unistd.h>
21
22 #include "ctree.h"
23 #include "ioctl.h"
24
25 #include "commands.h"
26 #include "utils.h"
27
28 static const char * const quota_cmd_group_usage[] = {
29         "btrfs quota <command> [options] <path>",
30         NULL
31 };
32
33 static int quota_ctl(int cmd, int argc, char **argv)
34 {
35         int ret = 0;
36         int fd;
37         int e;
38         char *path = argv[1];
39         struct btrfs_ioctl_quota_ctl_args args;
40         DIR *dirstream = NULL;
41
42         if (check_argc_exact(argc, 2))
43                 return -1;
44
45         memset(&args, 0, sizeof(args));
46         args.cmd = cmd;
47
48         fd = btrfs_open_dir(path, &dirstream, 1);
49         if (fd < 0)
50                 return 1;
51
52         ret = ioctl(fd, BTRFS_IOC_QUOTA_CTL, &args);
53         e = errno;
54         close_file_or_dir(fd, dirstream);
55         if (ret < 0) {
56                 error("quota command failed: %s", strerror(e));
57                 return 1;
58         }
59         return 0;
60 }
61
62 static const char * const cmd_quota_enable_usage[] = {
63         "btrfs quota enable <path>",
64         "Enable subvolume quota support for a filesystem.",
65         "Any data already present on the filesystem will not count towards",
66         "the space usage numbers. It is recommended to enable quota for a",
67         "filesystem before writing any data to it.",
68         NULL
69 };
70
71 static int cmd_quota_enable(int argc, char **argv)
72 {
73         int ret;
74
75         clean_args_no_options(argc, argv, cmd_quota_enable_usage);
76
77         ret = quota_ctl(BTRFS_QUOTA_CTL_ENABLE, argc, argv);
78
79         if (ret < 0)
80                 usage(cmd_quota_enable_usage);
81         return ret;
82 }
83
84 static const char * const cmd_quota_disable_usage[] = {
85         "btrfs quota disable <path>",
86         "Disable subvolume quota support for a filesystem.",
87         NULL
88 };
89
90 static int cmd_quota_disable(int argc, char **argv)
91 {
92         int ret;
93
94         clean_args_no_options(argc, argv, cmd_quota_disable_usage);
95
96         ret = quota_ctl(BTRFS_QUOTA_CTL_DISABLE, argc, argv);
97
98         if (ret < 0)
99                 usage(cmd_quota_disable_usage);
100         return ret;
101 }
102
103 static const char * const cmd_quota_rescan_usage[] = {
104         "btrfs quota rescan [-sw] <path>",
105         "Trash all qgroup numbers and scan the metadata again with the current config.",
106         "",
107         "-s   show status of a running rescan operation",
108         "-w   wait for rescan operation to finish (can be already in progress)",
109         NULL
110 };
111
112 static int cmd_quota_rescan(int argc, char **argv)
113 {
114         int ret = 0;
115         int fd;
116         int e;
117         char *path = NULL;
118         struct btrfs_ioctl_quota_rescan_args args;
119         unsigned long ioctlnum = BTRFS_IOC_QUOTA_RESCAN;
120         DIR *dirstream = NULL;
121         int wait_for_completion = 0;
122
123         while (1) {
124                 int c = getopt(argc, argv, "sw");
125                 if (c < 0)
126                         break;
127                 switch (c) {
128                 case 's':
129                         ioctlnum = BTRFS_IOC_QUOTA_RESCAN_STATUS;
130                         break;
131                 case 'w':
132                         wait_for_completion = 1;
133                         break;
134                 default:
135                         usage(cmd_quota_rescan_usage);
136                 }
137         }
138
139         if (ioctlnum != BTRFS_IOC_QUOTA_RESCAN && wait_for_completion) {
140                 error("switch -w cannot be used with -s");
141                 return 1;
142         }
143
144         if (check_argc_exact(argc - optind, 1))
145                 usage(cmd_quota_rescan_usage);
146
147         memset(&args, 0, sizeof(args));
148
149         path = argv[optind];
150         fd = btrfs_open_dir(path, &dirstream, 1);
151         if (fd < 0)
152                 return 1;
153
154         ret = ioctl(fd, ioctlnum, &args);
155         e = errno;
156
157         if (ioctlnum == BTRFS_IOC_QUOTA_RESCAN_STATUS) {
158                 close_file_or_dir(fd, dirstream);
159                 if (ret < 0) {
160                         error("could not obtain quota rescan status: %s",
161                               strerror(e));
162                         return 1;
163                 }
164                 if (!args.flags)
165                         printf("no rescan operation in progress\n");
166                 else
167                         printf("rescan operation running (current key %lld)\n",
168                                 args.progress);
169                 return 0;
170         }
171
172         if (ret == 0) {
173                 printf("quota rescan started\n");
174                 fflush(stdout);
175         } else if (ret < 0 && (!wait_for_completion || e != EINPROGRESS)) {
176                 error("quota rescan failed: %s", strerror(e));
177                 close_file_or_dir(fd, dirstream);
178                 return 1;
179         }
180
181         if (wait_for_completion) {
182                 ret = ioctl(fd, BTRFS_IOC_QUOTA_RESCAN_WAIT, &args);
183                 e = errno;
184                 if (ret < 0) {
185                         error("quota rescan wait failed: %s",
186                               strerror(e));
187                         close_file_or_dir(fd, dirstream);
188                         return 1;
189                 }
190         }
191
192         close_file_or_dir(fd, dirstream);
193         return 0;
194 }
195
196 static const char quota_cmd_group_info[] =
197 "manage filesystem quota settings";
198
199 const struct cmd_group quota_cmd_group = {
200         quota_cmd_group_usage, quota_cmd_group_info, {
201                 { "enable", cmd_quota_enable, cmd_quota_enable_usage, NULL, 0 },
202                 { "disable", cmd_quota_disable, cmd_quota_disable_usage,
203                    NULL, 0 },
204                 { "rescan", cmd_quota_rescan, cmd_quota_rescan_usage, NULL, 0 },
205                 NULL_CMD_STRUCT
206         }
207 };
208
209 int cmd_quota(int argc, char **argv)
210 {
211         return handle_command_group(&quota_cmd_group, argc, argv);
212 }