xref: /aosp_15_r20/external/e2fsprogs/debugfs/quota.c (revision 6a54128f25917bfc36a8a6e9d722c04a0b4641b6)
1*6a54128fSAndroid Build Coastguard Worker /*
2*6a54128fSAndroid Build Coastguard Worker  * quota.c --- debugfs quota commands
3*6a54128fSAndroid Build Coastguard Worker  *
4*6a54128fSAndroid Build Coastguard Worker  * Copyright (C) 2014 Theodore Ts'o.  This file may be redistributed
5*6a54128fSAndroid Build Coastguard Worker  * under the terms of the GNU Public License.
6*6a54128fSAndroid Build Coastguard Worker  */
7*6a54128fSAndroid Build Coastguard Worker 
8*6a54128fSAndroid Build Coastguard Worker #include "config.h"
9*6a54128fSAndroid Build Coastguard Worker #include <stdio.h>
10*6a54128fSAndroid Build Coastguard Worker #include <unistd.h>
11*6a54128fSAndroid Build Coastguard Worker #include <stdlib.h>
12*6a54128fSAndroid Build Coastguard Worker #include <ctype.h>
13*6a54128fSAndroid Build Coastguard Worker #include <string.h>
14*6a54128fSAndroid Build Coastguard Worker #include <time.h>
15*6a54128fSAndroid Build Coastguard Worker #ifdef HAVE_ERRNO_H
16*6a54128fSAndroid Build Coastguard Worker #include <errno.h>
17*6a54128fSAndroid Build Coastguard Worker #endif
18*6a54128fSAndroid Build Coastguard Worker #include <sys/types.h>
19*6a54128fSAndroid Build Coastguard Worker #ifdef HAVE_GETOPT_H
20*6a54128fSAndroid Build Coastguard Worker #include <getopt.h>
21*6a54128fSAndroid Build Coastguard Worker #else
22*6a54128fSAndroid Build Coastguard Worker extern int optind;
23*6a54128fSAndroid Build Coastguard Worker extern char *optarg;
24*6a54128fSAndroid Build Coastguard Worker #endif
25*6a54128fSAndroid Build Coastguard Worker 
26*6a54128fSAndroid Build Coastguard Worker #include "debugfs.h"
27*6a54128fSAndroid Build Coastguard Worker 
28*6a54128fSAndroid Build Coastguard Worker const char *quota_type[] = { "user", "group", "project", NULL };
29*6a54128fSAndroid Build Coastguard Worker 
load_quota_ctx(char * progname)30*6a54128fSAndroid Build Coastguard Worker static int load_quota_ctx(char *progname)
31*6a54128fSAndroid Build Coastguard Worker {
32*6a54128fSAndroid Build Coastguard Worker 	errcode_t	retval;
33*6a54128fSAndroid Build Coastguard Worker 
34*6a54128fSAndroid Build Coastguard Worker 	if (check_fs_open(progname))
35*6a54128fSAndroid Build Coastguard Worker 		return 1;
36*6a54128fSAndroid Build Coastguard Worker 
37*6a54128fSAndroid Build Coastguard Worker 	if (!ext2fs_has_feature_quota(current_fs->super)) {
38*6a54128fSAndroid Build Coastguard Worker 		com_err(progname, 0, "quota feature not enabled");
39*6a54128fSAndroid Build Coastguard Worker 		return 1;
40*6a54128fSAndroid Build Coastguard Worker 	}
41*6a54128fSAndroid Build Coastguard Worker 
42*6a54128fSAndroid Build Coastguard Worker 	if (current_qctx)
43*6a54128fSAndroid Build Coastguard Worker 		return 0;
44*6a54128fSAndroid Build Coastguard Worker 
45*6a54128fSAndroid Build Coastguard Worker 	retval = quota_init_context(&current_qctx, current_fs, 0);
46*6a54128fSAndroid Build Coastguard Worker 	if (retval) {
47*6a54128fSAndroid Build Coastguard Worker 		com_err(current_fs->device_name, retval,
48*6a54128fSAndroid Build Coastguard Worker 			"while trying to load quota information");
49*6a54128fSAndroid Build Coastguard Worker 		return 1;
50*6a54128fSAndroid Build Coastguard Worker 	}
51*6a54128fSAndroid Build Coastguard Worker 	return 0;
52*6a54128fSAndroid Build Coastguard Worker }
53*6a54128fSAndroid Build Coastguard Worker 
parse_quota_type(const char * cmdname,const char * str)54*6a54128fSAndroid Build Coastguard Worker static int parse_quota_type(const char *cmdname, const char *str)
55*6a54128fSAndroid Build Coastguard Worker {
56*6a54128fSAndroid Build Coastguard Worker 	errcode_t	retval;
57*6a54128fSAndroid Build Coastguard Worker 	char		*t;
58*6a54128fSAndroid Build Coastguard Worker 	int		flags = 0;
59*6a54128fSAndroid Build Coastguard Worker 	int		i;
60*6a54128fSAndroid Build Coastguard Worker 
61*6a54128fSAndroid Build Coastguard Worker 	for (i = 0; i < MAXQUOTAS; i++) {
62*6a54128fSAndroid Build Coastguard Worker 		if (strcasecmp(str, quota_type[i]) == 0)
63*6a54128fSAndroid Build Coastguard Worker 			break;
64*6a54128fSAndroid Build Coastguard Worker 	}
65*6a54128fSAndroid Build Coastguard Worker 	if (i >= MAXQUOTAS) {
66*6a54128fSAndroid Build Coastguard Worker 		i = strtol(str, &t, 0);
67*6a54128fSAndroid Build Coastguard Worker 		if (*t)
68*6a54128fSAndroid Build Coastguard Worker 			i = -1;
69*6a54128fSAndroid Build Coastguard Worker 	}
70*6a54128fSAndroid Build Coastguard Worker 	if (i < 0 || i >= MAXQUOTAS) {
71*6a54128fSAndroid Build Coastguard Worker 		com_err(0, 0, "Invalid quota type: %s", str);
72*6a54128fSAndroid Build Coastguard Worker 		printf("Valid quota types are: ");
73*6a54128fSAndroid Build Coastguard Worker 		for (i = 0; i < MAXQUOTAS; i++)
74*6a54128fSAndroid Build Coastguard Worker 			printf("%s ", quota_type[i]);
75*6a54128fSAndroid Build Coastguard Worker 		printf("\n");
76*6a54128fSAndroid Build Coastguard Worker 		return -1;
77*6a54128fSAndroid Build Coastguard Worker 	}
78*6a54128fSAndroid Build Coastguard Worker 
79*6a54128fSAndroid Build Coastguard Worker 	if (current_fs->flags & EXT2_FLAG_RW)
80*6a54128fSAndroid Build Coastguard Worker 		flags |= EXT2_FILE_WRITE;
81*6a54128fSAndroid Build Coastguard Worker 
82*6a54128fSAndroid Build Coastguard Worker 	retval = quota_file_open(current_qctx, NULL, 0, i, -1, flags);
83*6a54128fSAndroid Build Coastguard Worker 	if (retval) {
84*6a54128fSAndroid Build Coastguard Worker 		com_err(cmdname, retval,
85*6a54128fSAndroid Build Coastguard Worker 			"while opening quota inode (type %d)", i);
86*6a54128fSAndroid Build Coastguard Worker 		return -1;
87*6a54128fSAndroid Build Coastguard Worker 	}
88*6a54128fSAndroid Build Coastguard Worker 	return i;
89*6a54128fSAndroid Build Coastguard Worker }
90*6a54128fSAndroid Build Coastguard Worker 
91*6a54128fSAndroid Build Coastguard Worker 
list_quota_callback(struct dquot * dq,void * cb_data EXT2FS_ATTR ((unused)))92*6a54128fSAndroid Build Coastguard Worker static int list_quota_callback(struct dquot *dq,
93*6a54128fSAndroid Build Coastguard Worker 			       void *cb_data EXT2FS_ATTR((unused)))
94*6a54128fSAndroid Build Coastguard Worker {
95*6a54128fSAndroid Build Coastguard Worker 	printf("%10u   %8lld %8lld %8lld    %8lld %8lld %8lld\n",
96*6a54128fSAndroid Build Coastguard Worker 	       dq->dq_id, (long long)dq->dq_dqb.dqb_curspace,
97*6a54128fSAndroid Build Coastguard Worker 	       (long long)dq->dq_dqb.dqb_bsoftlimit,
98*6a54128fSAndroid Build Coastguard Worker 	       (long long)dq->dq_dqb.dqb_bhardlimit,
99*6a54128fSAndroid Build Coastguard Worker 	       (long long)dq->dq_dqb.dqb_curinodes,
100*6a54128fSAndroid Build Coastguard Worker 	       (long long)dq->dq_dqb.dqb_isoftlimit,
101*6a54128fSAndroid Build Coastguard Worker 	       (long long)dq->dq_dqb.dqb_ihardlimit);
102*6a54128fSAndroid Build Coastguard Worker 	return 0;
103*6a54128fSAndroid Build Coastguard Worker }
104*6a54128fSAndroid Build Coastguard Worker 
do_list_quota(int argc,char * argv[],int sci_idx EXT2FS_ATTR ((unused)),void * infop EXT2FS_ATTR ((unused)))105*6a54128fSAndroid Build Coastguard Worker void do_list_quota(int argc, char *argv[], int sci_idx EXT2FS_ATTR((unused)),
106*6a54128fSAndroid Build Coastguard Worker 		   void *infop EXT2FS_ATTR((unused)))
107*6a54128fSAndroid Build Coastguard Worker {
108*6a54128fSAndroid Build Coastguard Worker 	errcode_t	retval;
109*6a54128fSAndroid Build Coastguard Worker 	int		type;
110*6a54128fSAndroid Build Coastguard Worker 	struct quota_handle *qh;
111*6a54128fSAndroid Build Coastguard Worker 
112*6a54128fSAndroid Build Coastguard Worker 	if (load_quota_ctx(argv[0]))
113*6a54128fSAndroid Build Coastguard Worker 		return;
114*6a54128fSAndroid Build Coastguard Worker 
115*6a54128fSAndroid Build Coastguard Worker 	if (argc != 2) {
116*6a54128fSAndroid Build Coastguard Worker 		com_err(0, 0, "Usage: list_quota <quota_type>\n");
117*6a54128fSAndroid Build Coastguard Worker 		return;
118*6a54128fSAndroid Build Coastguard Worker 	}
119*6a54128fSAndroid Build Coastguard Worker 
120*6a54128fSAndroid Build Coastguard Worker 	type = parse_quota_type(argv[0], argv[1]);
121*6a54128fSAndroid Build Coastguard Worker 	if (type < 0)
122*6a54128fSAndroid Build Coastguard Worker 		return;
123*6a54128fSAndroid Build Coastguard Worker 
124*6a54128fSAndroid Build Coastguard Worker 	printf("%7s %2s   %8s %8s %8s    %8s %8s %8s\n",
125*6a54128fSAndroid Build Coastguard Worker 	       quota_type[type], "id",
126*6a54128fSAndroid Build Coastguard Worker 	       "space", "quota", "limit", "inodes", "quota", "limit");
127*6a54128fSAndroid Build Coastguard Worker 	qh = current_qctx->quota_file[type];
128*6a54128fSAndroid Build Coastguard Worker 	retval = qh->qh_ops->scan_dquots(qh, list_quota_callback, NULL);
129*6a54128fSAndroid Build Coastguard Worker 	if (retval) {
130*6a54128fSAndroid Build Coastguard Worker 		com_err(argv[0], retval, "while scanning dquots");
131*6a54128fSAndroid Build Coastguard Worker 		return;
132*6a54128fSAndroid Build Coastguard Worker 	}
133*6a54128fSAndroid Build Coastguard Worker }
134*6a54128fSAndroid Build Coastguard Worker 
do_get_quota(int argc,char * argv[],int sci_idx EXT2FS_ATTR ((unused)),void * infop EXT2FS_ATTR ((unused)))135*6a54128fSAndroid Build Coastguard Worker void do_get_quota(int argc, char *argv[], int sci_idx EXT2FS_ATTR((unused)),
136*6a54128fSAndroid Build Coastguard Worker 		  void *infop EXT2FS_ATTR((unused)))
137*6a54128fSAndroid Build Coastguard Worker {
138*6a54128fSAndroid Build Coastguard Worker 	int		err, type;
139*6a54128fSAndroid Build Coastguard Worker 	struct quota_handle *qh;
140*6a54128fSAndroid Build Coastguard Worker 	struct dquot	*dq;
141*6a54128fSAndroid Build Coastguard Worker 	qid_t		id;
142*6a54128fSAndroid Build Coastguard Worker 
143*6a54128fSAndroid Build Coastguard Worker 	if (load_quota_ctx(argv[0]))
144*6a54128fSAndroid Build Coastguard Worker 		return;
145*6a54128fSAndroid Build Coastguard Worker 
146*6a54128fSAndroid Build Coastguard Worker 	if (argc != 3) {
147*6a54128fSAndroid Build Coastguard Worker 		com_err(0, 0, "Usage: get_quota <quota_type> <id>\n");
148*6a54128fSAndroid Build Coastguard Worker 		return;
149*6a54128fSAndroid Build Coastguard Worker 	}
150*6a54128fSAndroid Build Coastguard Worker 
151*6a54128fSAndroid Build Coastguard Worker 	type = parse_quota_type(argv[0], argv[1]);
152*6a54128fSAndroid Build Coastguard Worker 	if (type < 0)
153*6a54128fSAndroid Build Coastguard Worker 		return;
154*6a54128fSAndroid Build Coastguard Worker 
155*6a54128fSAndroid Build Coastguard Worker 	id = parse_ulong(argv[2], argv[0], "id", &err);
156*6a54128fSAndroid Build Coastguard Worker 	if (err)
157*6a54128fSAndroid Build Coastguard Worker 		return;
158*6a54128fSAndroid Build Coastguard Worker 
159*6a54128fSAndroid Build Coastguard Worker 	printf("%7s %2s   %8s %8s %8s    %8s %8s %8s\n",
160*6a54128fSAndroid Build Coastguard Worker 	       quota_type[type], "id",
161*6a54128fSAndroid Build Coastguard Worker 	       "space", "quota", "limit", "inodes", "quota", "limit");
162*6a54128fSAndroid Build Coastguard Worker 
163*6a54128fSAndroid Build Coastguard Worker 	qh = current_qctx->quota_file[type];
164*6a54128fSAndroid Build Coastguard Worker 
165*6a54128fSAndroid Build Coastguard Worker 	dq = qh->qh_ops->read_dquot(qh, id);
166*6a54128fSAndroid Build Coastguard Worker 	if (dq) {
167*6a54128fSAndroid Build Coastguard Worker 		list_quota_callback(dq, NULL);
168*6a54128fSAndroid Build Coastguard Worker 		ext2fs_free_mem(&dq);
169*6a54128fSAndroid Build Coastguard Worker 	} else {
170*6a54128fSAndroid Build Coastguard Worker 		com_err(argv[0], 0, "couldn't read quota record");
171*6a54128fSAndroid Build Coastguard Worker 	}
172*6a54128fSAndroid Build Coastguard Worker }
173