1*84d9c625SLionel Sambuc /* $NetBSD: ulfs_quota2_subr.c,v 1.6 2013/06/06 00:51:50 dholland Exp $ */
2*84d9c625SLionel Sambuc /* from NetBSD: quota2_subr.c,v 1.5 2012/02/05 14:19:04 dholland Exp */
3*84d9c625SLionel Sambuc
4*84d9c625SLionel Sambuc /*-
5*84d9c625SLionel Sambuc * Copyright (c) 2010, 2011 Manuel Bouyer
6*84d9c625SLionel Sambuc * All rights reserved.
7*84d9c625SLionel Sambuc *
8*84d9c625SLionel Sambuc * Redistribution and use in source and binary forms, with or without
9*84d9c625SLionel Sambuc * modification, are permitted provided that the following conditions
10*84d9c625SLionel Sambuc * are met:
11*84d9c625SLionel Sambuc * 1. Redistributions of source code must retain the above copyright
12*84d9c625SLionel Sambuc * notice, this list of conditions and the following disclaimer.
13*84d9c625SLionel Sambuc * 2. Redistributions in binary form must reproduce the above copyright
14*84d9c625SLionel Sambuc * notice, this list of conditions and the following disclaimer in the
15*84d9c625SLionel Sambuc * documentation and/or other materials provided with the distribution.
16*84d9c625SLionel Sambuc *
17*84d9c625SLionel Sambuc * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
18*84d9c625SLionel Sambuc * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
19*84d9c625SLionel Sambuc * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
20*84d9c625SLionel Sambuc * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
21*84d9c625SLionel Sambuc * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22*84d9c625SLionel Sambuc * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23*84d9c625SLionel Sambuc * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24*84d9c625SLionel Sambuc * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
25*84d9c625SLionel Sambuc * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26*84d9c625SLionel Sambuc * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27*84d9c625SLionel Sambuc * POSSIBILITY OF SUCH DAMAGE.
28*84d9c625SLionel Sambuc */
29*84d9c625SLionel Sambuc
30*84d9c625SLionel Sambuc #include <sys/cdefs.h>
31*84d9c625SLionel Sambuc __KERNEL_RCSID(0, "$NetBSD: ulfs_quota2_subr.c,v 1.6 2013/06/06 00:51:50 dholland Exp $");
32*84d9c625SLionel Sambuc
33*84d9c625SLionel Sambuc #include <sys/param.h>
34*84d9c625SLionel Sambuc #include <sys/time.h>
35*84d9c625SLionel Sambuc
36*84d9c625SLionel Sambuc #include <ufs/lfs/lfs.h>
37*84d9c625SLionel Sambuc #include <ufs/lfs/lfs_extern.h>
38*84d9c625SLionel Sambuc #include <ufs/lfs/ulfs_inode.h>
39*84d9c625SLionel Sambuc #include <ufs/lfs/ulfs_dinode.h>
40*84d9c625SLionel Sambuc #include <ufs/lfs/ulfs_bswap.h>
41*84d9c625SLionel Sambuc #include <ufs/lfs/ulfs_quota2.h>
42*84d9c625SLionel Sambuc
43*84d9c625SLionel Sambuc #ifndef _KERNEL
44*84d9c625SLionel Sambuc #include <string.h>
45*84d9c625SLionel Sambuc #endif
46*84d9c625SLionel Sambuc
47*84d9c625SLionel Sambuc void
lfsquota2_addfreeq2e(struct quota2_header * q2h,void * bp,uint64_t baseoff,uint64_t bsize,int ns)48*84d9c625SLionel Sambuc lfsquota2_addfreeq2e(struct quota2_header *q2h, void *bp, uint64_t baseoff,
49*84d9c625SLionel Sambuc uint64_t bsize, int ns)
50*84d9c625SLionel Sambuc {
51*84d9c625SLionel Sambuc uint64_t blkoff = baseoff % bsize;
52*84d9c625SLionel Sambuc int i, nq2e;
53*84d9c625SLionel Sambuc struct quota2_entry *q2e;
54*84d9c625SLionel Sambuc
55*84d9c625SLionel Sambuc q2e = (void *)((char *)bp + blkoff);
56*84d9c625SLionel Sambuc nq2e = (bsize - blkoff) / sizeof(*q2e);
57*84d9c625SLionel Sambuc for (i = 0; i < nq2e; i++) {
58*84d9c625SLionel Sambuc q2e[i].q2e_next = q2h->q2h_free;
59*84d9c625SLionel Sambuc q2h->q2h_free = ulfs_rw64(i * sizeof(*q2e) + baseoff, ns);
60*84d9c625SLionel Sambuc }
61*84d9c625SLionel Sambuc }
62*84d9c625SLionel Sambuc
63*84d9c625SLionel Sambuc void
lfsquota2_create_blk0(uint64_t bsize,void * bp,int q2h_hash_shift,int type,int ns)64*84d9c625SLionel Sambuc lfsquota2_create_blk0(uint64_t bsize, void *bp, int q2h_hash_shift, int type,
65*84d9c625SLionel Sambuc int ns)
66*84d9c625SLionel Sambuc {
67*84d9c625SLionel Sambuc struct quota2_header *q2h;
68*84d9c625SLionel Sambuc const int quota2_hash_size = 1 << q2h_hash_shift;
69*84d9c625SLionel Sambuc const int quota2_full_header_size = sizeof(struct quota2_header) +
70*84d9c625SLionel Sambuc sizeof(q2h->q2h_entries[0]) * quota2_hash_size;
71*84d9c625SLionel Sambuc int i;
72*84d9c625SLionel Sambuc
73*84d9c625SLionel Sambuc memset(bp, 0, bsize);
74*84d9c625SLionel Sambuc q2h = bp;
75*84d9c625SLionel Sambuc q2h->q2h_magic_number = ulfs_rw32(Q2_HEAD_MAGIC, ns);
76*84d9c625SLionel Sambuc q2h->q2h_type = type;
77*84d9c625SLionel Sambuc q2h->q2h_hash_shift = q2h_hash_shift;
78*84d9c625SLionel Sambuc q2h->q2h_hash_size = ulfs_rw16(quota2_hash_size, ns);
79*84d9c625SLionel Sambuc /* setup defaut entry: unlimited, 7 days grace */
80*84d9c625SLionel Sambuc for (i = 0; i < N_QL; i++) {
81*84d9c625SLionel Sambuc q2h->q2h_defentry.q2e_val[i].q2v_hardlimit =
82*84d9c625SLionel Sambuc q2h->q2h_defentry.q2e_val[i].q2v_softlimit =
83*84d9c625SLionel Sambuc ulfs_rw64(UQUAD_MAX, ns);
84*84d9c625SLionel Sambuc q2h->q2h_defentry.q2e_val[i].q2v_grace =
85*84d9c625SLionel Sambuc ulfs_rw64(7ULL * 24ULL * 3600ULL, ns);
86*84d9c625SLionel Sambuc }
87*84d9c625SLionel Sambuc
88*84d9c625SLionel Sambuc /* first quota entry, after the hash table */
89*84d9c625SLionel Sambuc lfsquota2_addfreeq2e(q2h, bp, quota2_full_header_size, bsize, ns);
90*84d9c625SLionel Sambuc }
91*84d9c625SLionel Sambuc
92*84d9c625SLionel Sambuc void
lfsquota2_ulfs_rwq2v(const struct quota2_val * s,struct quota2_val * d,int needswap)93*84d9c625SLionel Sambuc lfsquota2_ulfs_rwq2v(const struct quota2_val *s, struct quota2_val *d, int needswap)
94*84d9c625SLionel Sambuc {
95*84d9c625SLionel Sambuc d->q2v_hardlimit = ulfs_rw64(s->q2v_hardlimit, needswap);
96*84d9c625SLionel Sambuc d->q2v_softlimit = ulfs_rw64(s->q2v_softlimit, needswap);
97*84d9c625SLionel Sambuc d->q2v_cur = ulfs_rw64(s->q2v_cur, needswap);
98*84d9c625SLionel Sambuc d->q2v_time = ulfs_rw64(s->q2v_time, needswap);
99*84d9c625SLionel Sambuc d->q2v_grace = ulfs_rw64(s->q2v_grace, needswap);
100*84d9c625SLionel Sambuc }
101*84d9c625SLionel Sambuc
102*84d9c625SLionel Sambuc void
lfsquota2_ulfs_rwq2e(const struct quota2_entry * s,struct quota2_entry * d,int needswap)103*84d9c625SLionel Sambuc lfsquota2_ulfs_rwq2e(const struct quota2_entry *s, struct quota2_entry *d,
104*84d9c625SLionel Sambuc int needswap)
105*84d9c625SLionel Sambuc {
106*84d9c625SLionel Sambuc lfsquota2_ulfs_rwq2v(&s->q2e_val[QL_BLOCK], &d->q2e_val[QL_BLOCK],
107*84d9c625SLionel Sambuc needswap);
108*84d9c625SLionel Sambuc lfsquota2_ulfs_rwq2v(&s->q2e_val[QL_FILE], &d->q2e_val[QL_FILE],
109*84d9c625SLionel Sambuc needswap);
110*84d9c625SLionel Sambuc d->q2e_uid = ulfs_rw32(s->q2e_uid, needswap);
111*84d9c625SLionel Sambuc }
112*84d9c625SLionel Sambuc
113*84d9c625SLionel Sambuc int
lfsquota_check_limit(uint64_t cur,uint64_t change,uint64_t soft,uint64_t hard,time_t expire,time_t now)114*84d9c625SLionel Sambuc lfsquota_check_limit(uint64_t cur, uint64_t change, uint64_t soft, uint64_t hard,
115*84d9c625SLionel Sambuc time_t expire, time_t now)
116*84d9c625SLionel Sambuc {
117*84d9c625SLionel Sambuc if (cur + change > hard) {
118*84d9c625SLionel Sambuc if (cur <= soft)
119*84d9c625SLionel Sambuc return (QL_F_CROSS | QL_S_DENY_HARD);
120*84d9c625SLionel Sambuc return QL_S_DENY_HARD;
121*84d9c625SLionel Sambuc } else if (cur + change > soft) {
122*84d9c625SLionel Sambuc if (cur <= soft)
123*84d9c625SLionel Sambuc return (QL_F_CROSS | QL_S_ALLOW_SOFT);
124*84d9c625SLionel Sambuc if (now > expire) {
125*84d9c625SLionel Sambuc return QL_S_DENY_GRACE;
126*84d9c625SLionel Sambuc }
127*84d9c625SLionel Sambuc return QL_S_ALLOW_SOFT;
128*84d9c625SLionel Sambuc }
129*84d9c625SLionel Sambuc return QL_S_ALLOW_OK;
130*84d9c625SLionel Sambuc }
131