1 /*        $NetBSD: quota_kernel.c,v 1.6 2014/06/28 22:27:50 dholland Exp $      */
2 /*-
3  * Copyright (c) 2012 The NetBSD Foundation, Inc.
4  * All rights reserved.
5  *
6  * This code is derived from software contributed to The NetBSD Foundation
7  * by David A. Holland.
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  *
18  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
19  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
20  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
21  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
22  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
23  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
24  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
25  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
26  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
27  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28  * POSSIBILITY OF SUCH DAMAGE.
29  */
30 
31 #include <sys/cdefs.h>
32 __RCSID("$NetBSD: quota_kernel.c,v 1.6 2014/06/28 22:27:50 dholland Exp $");
33 
34 #include <stdlib.h>
35 #include <err.h>
36 #include <errno.h>
37 #include <limits.h>
38 
39 #include <quota.h>
40 #include <sys/quotactl.h>
41 
42 #include "quotapvt.h"
43 
44 struct kernel_quotacursor {
45           /* just wrap the kernel interface type */
46           struct quotakcursor kcursor;
47 };
48 
49 static int
__quota_kernel_stat(struct quotahandle * qh,struct quotastat * stat)50 __quota_kernel_stat(struct quotahandle *qh, struct quotastat *stat)
51 {
52           struct quotactl_args args;
53 
54           args.qc_op = QUOTACTL_STAT;
55           args.u.stat.qc_info = stat;
56           return __quotactl(qh->qh_mountpoint, &args);
57 }
58 
59 const char *
__quota_kernel_getimplname(struct quotahandle * qh)60 __quota_kernel_getimplname(struct quotahandle *qh)
61 {
62           static struct quotastat stat;
63 
64           if (__quota_kernel_stat(qh, &stat)) {
65                     return NULL;
66           }
67           return stat.qs_implname;
68 }
69 
70 unsigned
__quota_kernel_getrestrictions(struct quotahandle * qh)71 __quota_kernel_getrestrictions(struct quotahandle *qh)
72 {
73           struct quotastat stat;
74 
75           if (__quota_kernel_stat(qh, &stat)) {
76                     /* XXX no particularly satisfactory thing to do here */
77                     return 0;
78           }
79           return stat.qs_restrictions;
80 }
81 
82 int
__quota_kernel_getnumidtypes(struct quotahandle * qh)83 __quota_kernel_getnumidtypes(struct quotahandle *qh)
84 {
85           struct quotastat stat;
86 
87           if (__quota_kernel_stat(qh, &stat)) {
88                     return 0;
89           }
90           return stat.qs_numidtypes;
91 }
92 
93 const char *
__quota_kernel_idtype_getname(struct quotahandle * qh,int idtype)94 __quota_kernel_idtype_getname(struct quotahandle *qh, int idtype)
95 {
96           static struct quotaidtypestat stat;
97           struct quotactl_args args;
98 
99           args.qc_op = QUOTACTL_IDTYPESTAT;
100           args.u.idtypestat.qc_idtype = idtype;
101           args.u.idtypestat.qc_info = &stat;
102           if (__quotactl(qh->qh_mountpoint, &args)) {
103                     return NULL;
104           }
105           return stat.qis_name;
106 }
107 
108 int
__quota_kernel_getnumobjtypes(struct quotahandle * qh)109 __quota_kernel_getnumobjtypes(struct quotahandle *qh)
110 {
111           struct quotastat stat;
112 
113           if (__quota_kernel_stat(qh, &stat)) {
114                     return 0;
115           }
116           return stat.qs_numobjtypes;
117 }
118 
119 const char *
__quota_kernel_objtype_getname(struct quotahandle * qh,int objtype)120 __quota_kernel_objtype_getname(struct quotahandle *qh, int objtype)
121 {
122           static struct quotaobjtypestat stat;
123           struct quotactl_args args;
124 
125           args.qc_op = QUOTACTL_OBJTYPESTAT;
126           args.u.objtypestat.qc_objtype = objtype;
127           args.u.objtypestat.qc_info = &stat;
128           if (__quotactl(qh->qh_mountpoint, &args)) {
129                     return NULL;
130           }
131           return stat.qos_name;
132 }
133 
134 int
__quota_kernel_objtype_isbytes(struct quotahandle * qh,int objtype)135 __quota_kernel_objtype_isbytes(struct quotahandle *qh, int objtype)
136 {
137           struct quotaobjtypestat stat;
138           struct quotactl_args args;
139 
140           args.qc_op = QUOTACTL_OBJTYPESTAT;
141           args.u.objtypestat.qc_objtype = objtype;
142           args.u.objtypestat.qc_info = &stat;
143           if (__quotactl(qh->qh_mountpoint, &args)) {
144                     return 0;
145           }
146           return stat.qos_isbytes;
147 }
148 
149 int
__quota_kernel_quotaon(struct quotahandle * qh,int idtype)150 __quota_kernel_quotaon(struct quotahandle *qh, int idtype)
151 {
152           struct quotactl_args args;
153           const char *file;
154           char path[PATH_MAX];
155 
156           /*
157            * Note that while it is an error to call quotaon on something
158            * that isn't a volume with old-style quotas that expects
159            * quotaon to be called, it's not our responsibility to check
160            * for that; the filesystem will. Also note that it is not an
161            * error to call quotaon repeatedly -- apparently this is to
162            * permit changing the quota file in use on the fly or
163            * something. So all we need to do here is ask the oldfiles
164            * code if the mount option was set in fstab and fetch back
165            * the filename.
166            */
167 
168           file = __quota_oldfiles_getquotafile(qh, idtype, path, sizeof(path));
169           if (file == NULL) {
170                     /*
171                      * This idtype (or maybe any idtype) was not enabled
172                      * in fstab.
173                      */
174                     errno = ENXIO;
175                     return -1;
176           }
177 
178           args.qc_op = QUOTACTL_QUOTAON;
179           args.u.quotaon.qc_idtype = idtype;
180           args.u.quotaon.qc_quotafile = file;
181           return __quotactl(qh->qh_mountpoint, &args);
182 }
183 
184 int
__quota_kernel_quotaoff(struct quotahandle * qh,int idtype)185 __quota_kernel_quotaoff(struct quotahandle *qh, int idtype)
186 {
187           struct quotactl_args args;
188 
189           args.qc_op = QUOTACTL_QUOTAOFF;
190           args.u.quotaoff.qc_idtype = idtype;
191           return __quotactl(qh->qh_mountpoint, &args);
192 }
193 
194 int
__quota_kernel_get(struct quotahandle * qh,const struct quotakey * qk,struct quotaval * qv)195 __quota_kernel_get(struct quotahandle *qh, const struct quotakey *qk,
196                        struct quotaval *qv)
197 {
198           struct quotactl_args args;
199 
200           args.qc_op = QUOTACTL_GET;
201           args.u.get.qc_key = qk;
202           args.u.get.qc_val = qv;
203           return __quotactl(qh->qh_mountpoint, &args);
204 }
205 
206 int
__quota_kernel_put(struct quotahandle * qh,const struct quotakey * qk,const struct quotaval * qv)207 __quota_kernel_put(struct quotahandle *qh, const struct quotakey *qk,
208                        const struct quotaval *qv)
209 {
210           struct quotactl_args args;
211 
212           args.qc_op = QUOTACTL_PUT;
213           args.u.put.qc_key = qk;
214           args.u.put.qc_val = qv;
215           return __quotactl(qh->qh_mountpoint, &args);
216 }
217 
218 int
__quota_kernel_delete(struct quotahandle * qh,const struct quotakey * qk)219 __quota_kernel_delete(struct quotahandle *qh, const struct quotakey *qk)
220 {
221           struct quotactl_args args;
222 
223           args.qc_op = QUOTACTL_DEL;
224           args.u.del.qc_key = qk;
225           return __quotactl(qh->qh_mountpoint, &args);
226 }
227 
228 struct kernel_quotacursor *
__quota_kernel_cursor_create(struct quotahandle * qh)229 __quota_kernel_cursor_create(struct quotahandle *qh)
230 {
231           struct quotactl_args args;
232           struct kernel_quotacursor *cursor;
233           int sverrno;
234 
235           cursor = malloc(sizeof(*cursor));
236           if (cursor == NULL) {
237                     return NULL;
238           }
239 
240           args.qc_op = QUOTACTL_CURSOROPEN;
241           args.u.cursoropen.qc_cursor = &cursor->kcursor;
242           if (__quotactl(qh->qh_mountpoint, &args)) {
243                     sverrno = errno;
244                     free(cursor);
245                     errno = sverrno;
246                     return NULL;
247           }
248           return cursor;
249 }
250 
251 void
__quota_kernel_cursor_destroy(struct quotahandle * qh,struct kernel_quotacursor * cursor)252 __quota_kernel_cursor_destroy(struct quotahandle *qh,
253                                     struct kernel_quotacursor *cursor)
254 {
255           struct quotactl_args args;
256 
257           args.qc_op = QUOTACTL_CURSORCLOSE;
258           args.u.cursorclose.qc_cursor = &cursor->kcursor;
259           if (__quotactl(qh->qh_mountpoint, &args)) {
260                     /* XXX should we really print from inside the library? */
261                     warn("__quotactl cursorclose");
262           }
263           free(cursor);
264 }
265 
266 int
__quota_kernel_cursor_skipidtype(struct quotahandle * qh,struct kernel_quotacursor * cursor,int idtype)267 __quota_kernel_cursor_skipidtype(struct quotahandle *qh,
268                                          struct kernel_quotacursor *cursor,
269                                          int idtype)
270 {
271           struct quotactl_args args;
272 
273           args.qc_op = QUOTACTL_CURSORSKIPIDTYPE;
274           args.u.cursorskipidtype.qc_cursor = &cursor->kcursor;
275           args.u.cursorskipidtype.qc_idtype = idtype;
276           return __quotactl(qh->qh_mountpoint, &args);
277 }
278 
279 int
__quota_kernel_cursor_get(struct quotahandle * qh,struct kernel_quotacursor * cursor,struct quotakey * key,struct quotaval * val)280 __quota_kernel_cursor_get(struct quotahandle *qh,
281                                 struct kernel_quotacursor *cursor,
282                                 struct quotakey *key, struct quotaval *val)
283 {
284           int ret;
285 
286           ret = __quota_kernel_cursor_getn(qh, cursor, key, val, 1);
287           if (ret < 0) {
288                     return -1;
289           }
290           return 0;
291 }
292 
293 int
__quota_kernel_cursor_getn(struct quotahandle * qh,struct kernel_quotacursor * cursor,struct quotakey * keys,struct quotaval * vals,unsigned maxnum)294 __quota_kernel_cursor_getn(struct quotahandle *qh,
295                                  struct kernel_quotacursor *cursor,
296                                  struct quotakey *keys, struct quotaval *vals,
297                                  unsigned maxnum)
298 {
299           struct quotactl_args args;
300           unsigned ret;
301 
302           if (maxnum > INT_MAX) {
303                     /* joker, eh? */
304                     errno = EINVAL;
305                     return -1;
306           }
307 
308           args.qc_op = QUOTACTL_CURSORGET;
309           args.u.cursorget.qc_cursor = &cursor->kcursor;
310           args.u.cursorget.qc_keys = keys;
311           args.u.cursorget.qc_vals = vals;
312           args.u.cursorget.qc_maxnum = maxnum;
313           args.u.cursorget.qc_ret = &ret;
314           if (__quotactl(qh->qh_mountpoint, &args) < 0) {
315                     return -1;
316           }
317           return ret;
318 }
319 
320 int
__quota_kernel_cursor_atend(struct quotahandle * qh,struct kernel_quotacursor * cursor)321 __quota_kernel_cursor_atend(struct quotahandle *qh,
322                                   struct kernel_quotacursor *cursor)
323 {
324           struct quotactl_args args;
325           int ret;
326 
327           args.qc_op = QUOTACTL_CURSORATEND;
328           args.u.cursoratend.qc_cursor = &cursor->kcursor;
329           args.u.cursoratend.qc_ret = &ret;
330           if (__quotactl(qh->qh_mountpoint, &args)) {
331                     /*
332                      * Return -1 so naive callers, who test for the return
333                      * value being nonzero, stop iterating, and
334                      * sophisticated callers can tell an error from
335                      * end-of-data.
336                      */
337                     return -1;
338           }
339           return ret;
340 }
341 
342 int
__quota_kernel_cursor_rewind(struct quotahandle * qh,struct kernel_quotacursor * cursor)343 __quota_kernel_cursor_rewind(struct quotahandle *qh,
344                                    struct kernel_quotacursor *cursor)
345 {
346           struct quotactl_args args;
347 
348           args.qc_op = QUOTACTL_CURSORREWIND;
349           args.u.cursorrewind.qc_cursor = &cursor->kcursor;
350           return __quotactl(qh->qh_mountpoint, &args);
351 }
352