blob: fd0ae0295d4db28b9b8c79029d3dcf6342d3df86 [file] [log] [blame]
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +00001/*
2 * Copyright (c) 1991, 1992 Paul Kranenburg <pk@cs.few.eur.nl>
3 * Copyright (c) 1993 Branko Lankester <branko@hacktic.nl>
4 * Copyright (c) 1993, 1994, 1995, 1996 Rick Sladkey <jrs@world.std.com>
5 * Copyright (c) 1996-1999 Wichert Akkerman <wichert@cistron.nl>
6 * Copyright (c) 2005, 2006 Dmitry V. Levin <ldv@altlinux.org>
7 * All rights reserved.
8 *
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted provided that the following conditions
11 * are met:
12 * 1. Redistributions of source code must retain the above copyright
13 * notice, this list of conditions and the following disclaimer.
14 * 2. Redistributions in binary form must reproduce the above copyright
15 * notice, this list of conditions and the following disclaimer in the
16 * documentation and/or other materials provided with the distribution.
17 * 3. The name of the author may not be used to endorse or promote products
18 * derived from this software without specific prior written permission.
19 *
20 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
21 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
22 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
23 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
24 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
25 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
29 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +000030 */
31
32#include "defs.h"
33
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +000034#define SUBCMDMASK 0x00ff
35#define SUBCMDSHIFT 8
36#define QCMD_CMD(cmd) ((u_int32_t)(cmd) >> SUBCMDSHIFT)
37#define QCMD_TYPE(cmd) ((u_int32_t)(cmd) & SUBCMDMASK)
38
39#define OLD_CMD(cmd) ((u_int32_t)(cmd) << 8)
40#define NEW_CMD(cmd) ((u_int32_t)(cmd) | 0x800000)
41#define XQM_CMD(cmd) ((u_int32_t)(cmd) | ('X'<<8))
42
43#define Q_V1_QUOTAON OLD_CMD(0x1)
44#define Q_V1_QUOTAOFF OLD_CMD(0x2)
45#define Q_V1_GETQUOTA OLD_CMD(0x3)
46#define Q_V1_SETQUOTA OLD_CMD(0x4)
47#define Q_V1_SETUSE OLD_CMD(0x5)
48#define Q_V1_SYNC OLD_CMD(0x6)
49#define Q_SETQLIM OLD_CMD(0x7)
50#define Q_V1_GETSTATS OLD_CMD(0x8)
51#define Q_V1_RSQUASH OLD_CMD(0x10)
52
53#define Q_V2_GETQUOTA OLD_CMD(0xD)
54#define Q_V2_SETQUOTA OLD_CMD(0xE)
55#define Q_V2_SETUSE OLD_CMD(0xF)
56#define Q_V2_GETINFO OLD_CMD(0x9)
57#define Q_V2_SETINFO OLD_CMD(0xA)
58#define Q_V2_SETGRACE OLD_CMD(0xB)
59#define Q_V2_SETFLAGS OLD_CMD(0xC)
60#define Q_V2_GETSTATS OLD_CMD(0x11)
61
62#define Q_SYNC NEW_CMD(0x1)
63#define Q_QUOTAON NEW_CMD(0x2)
64#define Q_QUOTAOFF NEW_CMD(0x3)
65#define Q_GETFMT NEW_CMD(0x4)
66#define Q_GETINFO NEW_CMD(0x5)
67#define Q_SETINFO NEW_CMD(0x6)
68#define Q_GETQUOTA NEW_CMD(0x7)
69#define Q_SETQUOTA NEW_CMD(0x8)
70
71#define Q_XQUOTAON XQM_CMD(0x1)
72#define Q_XQUOTAOFF XQM_CMD(0x2)
73#define Q_XGETQUOTA XQM_CMD(0x3)
74#define Q_XSETQLIM XQM_CMD(0x4)
75#define Q_XGETQSTAT XQM_CMD(0x5)
76#define Q_XQUOTARM XQM_CMD(0x6)
77#define Q_XQUOTASYNC XQM_CMD(0x7)
78
Dmitry V. Levin0ed617b2014-04-25 23:30:54 +000079#include "xlat/quotacmds.h"
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +000080
81#define USRQUOTA 0
82#define GRPQUOTA 1
83
Dmitry V. Levin0ed617b2014-04-25 23:30:54 +000084#include "xlat/quotatypes.h"
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +000085
86/* Quota format identifiers */
87#define QFMT_VFS_OLD 1
88#define QFMT_VFS_V0 2
89
Dmitry V. Levin0ed617b2014-04-25 23:30:54 +000090#include "xlat/quota_formats.h"
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +000091
92#define XFS_QUOTA_UDQ_ACCT (1<<0) /* user quota accounting */
93#define XFS_QUOTA_UDQ_ENFD (1<<1) /* user quota limits enforcement */
94#define XFS_QUOTA_GDQ_ACCT (1<<2) /* group quota accounting */
95#define XFS_QUOTA_GDQ_ENFD (1<<3) /* group quota limits enforcement */
96
97#define XFS_USER_QUOTA (1<<0) /* user quota type */
98#define XFS_PROJ_QUOTA (1<<1) /* (IRIX) project quota type */
99#define XFS_GROUP_QUOTA (1<<2) /* group quota type */
100
Dmitry V. Levin0ed617b2014-04-25 23:30:54 +0000101#include "xlat/xfs_quota_flags.h"
Dmitry V. Levin0ed617b2014-04-25 23:30:54 +0000102#include "xlat/xfs_dqblk_flags.h"
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000103
104/*
105 * Following flags are used to specify which fields are valid
106 */
107#define QIF_BLIMITS 1
108#define QIF_SPACE 2
109#define QIF_ILIMITS 4
110#define QIF_INODES 8
111#define QIF_BTIME 16
112#define QIF_ITIME 32
113
Dmitry V. Levin0ed617b2014-04-25 23:30:54 +0000114#include "xlat/if_dqblk_valid.h"
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000115
116struct if_dqblk
117{
118 u_int64_t dqb_bhardlimit;
119 u_int64_t dqb_bsoftlimit;
120 u_int64_t dqb_curspace;
121 u_int64_t dqb_ihardlimit;
122 u_int64_t dqb_isoftlimit;
123 u_int64_t dqb_curinodes;
124 u_int64_t dqb_btime;
125 u_int64_t dqb_itime;
126 u_int32_t dqb_valid;
127};
128
129struct v1_dqblk
130{
131 u_int32_t dqb_bhardlimit; /* absolute limit on disk blks alloc */
132 u_int32_t dqb_bsoftlimit; /* preferred limit on disk blks */
133 u_int32_t dqb_curblocks; /* current block count */
134 u_int32_t dqb_ihardlimit; /* maximum # allocated inodes */
135 u_int32_t dqb_isoftlimit; /* preferred inode limit */
136 u_int32_t dqb_curinodes; /* current # allocated inodes */
137 time_t dqb_btime; /* time limit for excessive disk use */
138 time_t dqb_itime; /* time limit for excessive files */
139};
140
141struct v2_dqblk
142{
143 unsigned int dqb_ihardlimit;
144 unsigned int dqb_isoftlimit;
145 unsigned int dqb_curinodes;
146 unsigned int dqb_bhardlimit;
147 unsigned int dqb_bsoftlimit;
148 u_int64_t dqb_curspace;
149 time_t dqb_btime;
150 time_t dqb_itime;
151};
152
153struct xfs_dqblk
154{
155 int8_t d_version; /* version of this structure */
156 int8_t d_flags; /* XFS_{USER,PROJ,GROUP}_QUOTA */
157 u_int16_t d_fieldmask; /* field specifier */
158 u_int32_t d_id; /* user, project, or group ID */
159 u_int64_t d_blk_hardlimit; /* absolute limit on disk blks */
160 u_int64_t d_blk_softlimit; /* preferred limit on disk blks */
161 u_int64_t d_ino_hardlimit; /* maximum # allocated inodes */
162 u_int64_t d_ino_softlimit; /* preferred inode limit */
163 u_int64_t d_bcount; /* # disk blocks owned by the user */
164 u_int64_t d_icount; /* # inodes owned by the user */
165 int32_t d_itimer; /* zero if within inode limits */
166 int32_t d_btimer; /* similar to above; for disk blocks */
167 u_int16_t d_iwarns; /* # warnings issued wrt num inodes */
168 u_int16_t d_bwarns; /* # warnings issued wrt disk blocks */
169 int32_t d_padding2; /* padding2 - for future use */
170 u_int64_t d_rtb_hardlimit; /* absolute limit on realtime blks */
171 u_int64_t d_rtb_softlimit; /* preferred limit on RT disk blks */
172 u_int64_t d_rtbcount; /* # realtime blocks owned */
173 int32_t d_rtbtimer; /* similar to above; for RT disk blks */
174 u_int16_t d_rtbwarns; /* # warnings issued wrt RT disk blks */
175 int16_t d_padding3; /* padding3 - for future use */
176 char d_padding4[8]; /* yet more padding */
177};
178
179/*
180 * Following flags are used to specify which fields are valid
181 */
182#define IIF_BGRACE 1
183#define IIF_IGRACE 2
184#define IIF_FLAGS 4
185
Dmitry V. Levin0ed617b2014-04-25 23:30:54 +0000186#include "xlat/if_dqinfo_valid.h"
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000187
188struct if_dqinfo
189{
190 u_int64_t dqi_bgrace;
191 u_int64_t dqi_igrace;
192 u_int32_t dqi_flags;
193 u_int32_t dqi_valid;
194};
195
196struct v2_dqinfo
197{
198 unsigned int dqi_bgrace;
199 unsigned int dqi_igrace;
200 unsigned int dqi_flags;
201 unsigned int dqi_blocks;
202 unsigned int dqi_free_blk;
203 unsigned int dqi_free_entry;
204};
205
206struct v1_dqstats
207{
208 u_int32_t lookups;
209 u_int32_t drops;
210 u_int32_t reads;
211 u_int32_t writes;
212 u_int32_t cache_hits;
213 u_int32_t allocated_dquots;
214 u_int32_t free_dquots;
215 u_int32_t syncs;
216};
217
218struct v2_dqstats
219{
220 u_int32_t lookups;
221 u_int32_t drops;
222 u_int32_t reads;
223 u_int32_t writes;
224 u_int32_t cache_hits;
225 u_int32_t allocated_dquots;
226 u_int32_t free_dquots;
227 u_int32_t syncs;
228 u_int32_t version;
229};
230
231typedef struct fs_qfilestat
232{
233 u_int64_t qfs_ino; /* inode number */
234 u_int64_t qfs_nblks; /* number of BBs 512-byte-blks */
235 u_int32_t qfs_nextents; /* number of extents */
236} fs_qfilestat_t;
237
238struct xfs_dqstats
239{
240 int8_t qs_version; /* version number for future changes */
241 u_int16_t qs_flags; /* XFS_QUOTA_{U,P,G}DQ_{ACCT,ENFD} */
242 int8_t qs_pad; /* unused */
243 fs_qfilestat_t qs_uquota; /* user quota storage information */
244 fs_qfilestat_t qs_gquota; /* group quota storage information */
245 u_int32_t qs_incoredqs; /* number of dquots incore */
246 int32_t qs_btimelimit; /* limit for blks timer */
247 int32_t qs_itimelimit; /* limit for inodes timer */
248 int32_t qs_rtbtimelimit; /* limit for rt blks timer */
249 u_int16_t qs_bwarnlimit; /* limit for num warnings */
250 u_int16_t qs_iwarnlimit; /* limit for num warnings */
251};
252
253static void
254decode_cmd_data(struct tcb *tcp, u_int32_t cmd, unsigned long data)
255{
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200256 switch (cmd) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000257 case Q_GETQUOTA:
258 case Q_SETQUOTA:
259 {
260 struct if_dqblk dq;
261
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200262 if (cmd == Q_GETQUOTA && syserror(tcp)) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000263 tprintf("%#lx", data);
264 break;
265 }
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200266 if (umove(tcp, data, &dq) < 0) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000267 tprintf("{???} %#lx", data);
268 break;
269 }
270 tprintf("{bhardlimit=%" PRIu64 ", ", dq.dqb_bhardlimit);
271 tprintf("bsoftlimit=%" PRIu64 ", ", dq.dqb_bsoftlimit);
272 tprintf("curspace=%" PRIu64 ", ", dq.dqb_curspace);
273 tprintf("ihardlimit=%" PRIu64 ", ", dq.dqb_ihardlimit);
274 tprintf("isoftlimit=%" PRIu64 ", ", dq.dqb_isoftlimit);
275 tprintf("curinodes=%" PRIu64 ", ", dq.dqb_curinodes);
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200276 if (!abbrev(tcp)) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000277 tprintf("btime=%" PRIu64 ", ", dq.dqb_btime);
278 tprintf("itime=%" PRIu64 ", ", dq.dqb_itime);
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200279 tprints("valid=");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000280 printflags(if_dqblk_valid,
281 dq.dqb_valid, "QIF_???");
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200282 tprints("}");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000283 } else
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200284 tprints("...}");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000285 break;
286 }
287 case Q_V1_GETQUOTA:
288 case Q_V1_SETQUOTA:
289 {
290 struct v1_dqblk dq;
291
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200292 if (cmd == Q_V1_GETQUOTA && syserror(tcp)) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000293 tprintf("%#lx", data);
294 break;
295 }
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200296 if (umove(tcp, data, &dq) < 0) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000297 tprintf("{???} %#lx", data);
298 break;
299 }
300 tprintf("{bhardlimit=%u, ", dq.dqb_bhardlimit);
301 tprintf("bsoftlimit=%u, ", dq.dqb_bsoftlimit);
302 tprintf("curblocks=%u, ", dq.dqb_curblocks);
303 tprintf("ihardlimit=%u, ", dq.dqb_ihardlimit);
304 tprintf("isoftlimit=%u, ", dq.dqb_isoftlimit);
305 tprintf("curinodes=%u, ", dq.dqb_curinodes);
H.J. Lu0b315b62012-02-03 10:16:03 -0800306 tprintf("btime=%lu, ", (long) dq.dqb_btime);
307 tprintf("itime=%lu}", (long) dq.dqb_itime);
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000308 break;
309 }
310 case Q_V2_GETQUOTA:
311 case Q_V2_SETQUOTA:
312 {
313 struct v2_dqblk dq;
314
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200315 if (cmd == Q_V2_GETQUOTA && syserror(tcp)) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000316 tprintf("%#lx", data);
317 break;
318 }
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200319 if (umove(tcp, data, &dq) < 0) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000320 tprintf("{???} %#lx", data);
321 break;
322 }
323 tprintf("{ihardlimit=%u, ", dq.dqb_ihardlimit);
324 tprintf("isoftlimit=%u, ", dq.dqb_isoftlimit);
325 tprintf("curinodes=%u, ", dq.dqb_curinodes);
326 tprintf("bhardlimit=%u, ", dq.dqb_bhardlimit);
327 tprintf("bsoftlimit=%u, ", dq.dqb_bsoftlimit);
328 tprintf("curspace=%" PRIu64 ", ", dq.dqb_curspace);
H.J. Lu0b315b62012-02-03 10:16:03 -0800329 tprintf("btime=%lu, ", (long) dq.dqb_btime);
330 tprintf("itime=%lu}", (long) dq.dqb_itime);
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000331 break;
332 }
333 case Q_XGETQUOTA:
334 case Q_XSETQLIM:
335 {
336 struct xfs_dqblk dq;
337
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200338 if (cmd == Q_XGETQUOTA && syserror(tcp)) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000339 tprintf("%#lx", data);
340 break;
341 }
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200342 if (umove(tcp, data, &dq) < 0) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000343 tprintf("{???} %#lx", data);
344 break;
345 }
346 tprintf("{version=%d, ", dq.d_version);
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200347 tprints("flags=");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000348 printflags(xfs_dqblk_flags,
349 dq.d_flags, "XFS_???_QUOTA");
350 tprintf(", fieldmask=%#x, ", dq.d_fieldmask);
351 tprintf("id=%u, ", dq.d_id);
352 tprintf("blk_hardlimit=%" PRIu64 ", ", dq.d_blk_hardlimit);
353 tprintf("blk_softlimit=%" PRIu64 ", ", dq.d_blk_softlimit);
354 tprintf("ino_hardlimit=%" PRIu64 ", ", dq.d_ino_hardlimit);
355 tprintf("ino_softlimit=%" PRIu64 ", ", dq.d_ino_softlimit);
356 tprintf("bcount=%" PRIu64 ", ", dq.d_bcount);
357 tprintf("icount=%" PRIu64 ", ", dq.d_icount);
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200358 if (!abbrev(tcp)) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000359 tprintf("itimer=%d, ", dq.d_itimer);
360 tprintf("btimer=%d, ", dq.d_btimer);
361 tprintf("iwarns=%u, ", dq.d_iwarns);
362 tprintf("bwarns=%u, ", dq.d_bwarns);
363 tprintf("rtbcount=%" PRIu64 ", ", dq.d_rtbcount);
364 tprintf("rtbtimer=%d, ", dq.d_rtbtimer);
365 tprintf("rtbwarns=%u}", dq.d_rtbwarns);
366 } else
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200367 tprints("...}");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000368 break;
369 }
370 case Q_GETFMT:
371 {
372 u_int32_t fmt;
373
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200374 if (syserror(tcp)) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000375 tprintf("%#lx", data);
376 break;
377 }
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200378 if (umove(tcp, data, &fmt) < 0) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000379 tprintf("{???} %#lx", data);
380 break;
381 }
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200382 tprints("{");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000383 printxval(quota_formats, fmt, "QFMT_VFS_???");
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200384 tprints("}");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000385 break;
386 }
387 case Q_GETINFO:
388 case Q_SETINFO:
389 {
390 struct if_dqinfo dq;
391
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200392 if (cmd == Q_GETINFO && syserror(tcp)) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000393 tprintf("%#lx", data);
394 break;
395 }
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200396 if (umove(tcp, data, &dq) < 0) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000397 tprintf("{???} %#lx", data);
398 break;
399 }
400 tprintf("{bgrace=%" PRIu64 ", ", dq.dqi_bgrace);
401 tprintf("igrace=%" PRIu64 ", ", dq.dqi_igrace);
402 tprintf("flags=%#x, ", dq.dqi_flags);
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200403 tprints("valid=");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000404 printflags(if_dqinfo_valid, dq.dqi_valid, "IIF_???");
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200405 tprints("}");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000406 break;
407 }
408 case Q_V2_GETINFO:
409 case Q_V2_SETINFO:
410 {
411 struct v2_dqinfo dq;
412
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200413 if (cmd == Q_V2_GETINFO && syserror(tcp)) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000414 tprintf("%#lx", data);
415 break;
416 }
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200417 if (umove(tcp, data, &dq) < 0) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000418 tprintf("{???} %#lx", data);
419 break;
420 }
421 tprintf("{bgrace=%u, ", dq.dqi_bgrace);
422 tprintf("igrace=%u, ", dq.dqi_igrace);
423 tprintf("flags=%#x, ", dq.dqi_flags);
424 tprintf("blocks=%u, ", dq.dqi_blocks);
425 tprintf("free_blk=%u, ", dq.dqi_free_blk);
426 tprintf("free_entry=%u}", dq.dqi_free_entry);
427 break;
428 }
429 case Q_V1_GETSTATS:
430 {
431 struct v1_dqstats dq;
432
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200433 if (syserror(tcp)) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000434 tprintf("%#lx", data);
435 break;
436 }
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200437 if (umove(tcp, data, &dq) < 0) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000438 tprintf("{???} %#lx", data);
439 break;
440 }
441 tprintf("{lookups=%u, ", dq.lookups);
442 tprintf("drops=%u, ", dq.drops);
443 tprintf("reads=%u, ", dq.reads);
444 tprintf("writes=%u, ", dq.writes);
445 tprintf("cache_hits=%u, ", dq.cache_hits);
446 tprintf("allocated_dquots=%u, ", dq.allocated_dquots);
447 tprintf("free_dquots=%u, ", dq.free_dquots);
448 tprintf("syncs=%u}", dq.syncs);
449 break;
450 }
451 case Q_V2_GETSTATS:
452 {
453 struct v2_dqstats dq;
454
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200455 if (syserror(tcp)) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000456 tprintf("%#lx", data);
457 break;
458 }
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200459 if (umove(tcp, data, &dq) < 0) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000460 tprintf("{???} %#lx", data);
461 break;
462 }
463 tprintf("{lookups=%u, ", dq.lookups);
464 tprintf("drops=%u, ", dq.drops);
465 tprintf("reads=%u, ", dq.reads);
466 tprintf("writes=%u, ", dq.writes);
467 tprintf("cache_hits=%u, ", dq.cache_hits);
468 tprintf("allocated_dquots=%u, ", dq.allocated_dquots);
469 tprintf("free_dquots=%u, ", dq.free_dquots);
470 tprintf("syncs=%u, ", dq.syncs);
471 tprintf("version=%u}", dq.version);
472 break;
473 }
474 case Q_XGETQSTAT:
475 {
476 struct xfs_dqstats dq;
477
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200478 if (syserror(tcp)) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000479 tprintf("%#lx", data);
480 break;
481 }
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200482 if (umove(tcp, data, &dq) < 0) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000483 tprintf("{???} %#lx", data);
484 break;
485 }
486 tprintf("{version=%d, ", dq.qs_version);
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200487 if (abbrev(tcp)) {
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200488 tprints("...}");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000489 break;
490 }
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200491 tprints("flags=");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000492 printflags(xfs_quota_flags,
493 dq.qs_flags, "XFS_QUOTA_???");
494 tprintf(", incoredqs=%u, ", dq.qs_incoredqs);
495 tprintf("u_ino=%" PRIu64 ", ", dq.qs_uquota.qfs_ino);
496 tprintf("u_nblks=%" PRIu64 ", ", dq.qs_uquota.qfs_nblks);
497 tprintf("u_nextents=%u, ", dq.qs_uquota.qfs_nextents);
498 tprintf("g_ino=%" PRIu64 ", ", dq.qs_gquota.qfs_ino);
499 tprintf("g_nblks=%" PRIu64 ", ", dq.qs_gquota.qfs_nblks);
500 tprintf("g_nextents=%u, ", dq.qs_gquota.qfs_nextents);
501 tprintf("btimelimit=%d, ", dq.qs_btimelimit);
502 tprintf("itimelimit=%d, ", dq.qs_itimelimit);
503 tprintf("rtbtimelimit=%d, ", dq.qs_rtbtimelimit);
504 tprintf("bwarnlimit=%u, ", dq.qs_bwarnlimit);
505 tprintf("iwarnlimit=%u}", dq.qs_iwarnlimit);
506 break;
507 }
508 case Q_XQUOTAON:
509 {
510 u_int32_t flag;
511
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200512 if (umove(tcp, data, &flag) < 0) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000513 tprintf("{???} %#lx", data);
514 break;
515 }
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200516 tprints("{");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000517 printflags(xfs_quota_flags, flag, "XFS_QUOTA_???");
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200518 tprints("}");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000519 break;
520 }
521 default:
522 tprintf("%#lx", data);
523 break;
524 }
525}
526
Dmitry V. Levina0bd3742015-04-07 01:36:50 +0000527SYS_FUNC(quotactl)
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000528{
529 /*
530 * The Linux kernel only looks at the low 32 bits of command and id
531 * arguments, but on some 64-bit architectures (s390x) this word
532 * will have been sign-extended when we see it. The high 1 bits
533 * don't mean anything, so don't confuse the output with them.
534 */
535 u_int32_t qcmd = tcp->u_arg[0];
536 u_int32_t cmd = QCMD_CMD(qcmd);
537 u_int32_t type = QCMD_TYPE(qcmd);
538 u_int32_t id = tcp->u_arg[2];
539
540 if (!verbose(tcp))
541 return printargs(tcp);
542
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200543 if (entering(tcp)) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000544 printxval(quotacmds, cmd, "Q_???");
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200545 tprints("|");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000546 printxval(quotatypes, type, "???QUOTA");
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200547 tprints(", ");
Dmitry V. Levin79439662012-10-26 23:43:13 +0000548 printpath(tcp, tcp->u_arg[1]);
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200549 tprints(", ");
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200550 switch (cmd) {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000551 case Q_V1_QUOTAON:
552 case Q_QUOTAON:
553 printxval(quota_formats, id, "QFMT_VFS_???");
554 break;
555 case Q_V1_GETQUOTA:
556 case Q_V2_GETQUOTA:
557 case Q_GETQUOTA:
558 case Q_V1_SETQUOTA:
559 case Q_V2_SETQUOTA:
560 case Q_V1_SETUSE:
561 case Q_V2_SETUSE:
562 case Q_SETQLIM:
563 case Q_SETQUOTA:
564 case Q_XGETQUOTA:
565 case Q_XSETQLIM:
566 tprintf("%u", id);
567 break;
568 default:
569 tprintf("%#lx", tcp->u_arg[2]);
570 break;
571 }
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200572 tprints(", ");
Denys Vlasenko7b609d52011-06-22 14:32:43 +0200573 } else {
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000574 if (!tcp->u_arg[3])
Denys Vlasenko60fe8c12011-09-01 10:00:28 +0200575 tprints("NULL");
Dmitry V. Levinef5c3d62006-10-16 00:26:47 +0000576 else
577 decode_cmd_data(tcp, cmd, tcp->u_arg[3]);
578 }
579 return 0;
580}