wscript: Add check for --wrap linker flag
[vlendec/samba-autobuild/.git] / source3 / lib / sysquotas_4B.c
1 /*
2  * Unix SMB/CIFS implementation.
3  * System QUOTA function wrappers for QUOTACTL_4B
4
5  * Copyright (C) 2011 James Peach.
6  *
7  * This program is free software; you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation; either version 2 of the License, or
10  * (at your option) any later version.
11  *
12  * This program is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with this program; if not, write to the Free Software
19  * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
20  */
21
22 #include "includes.h"
23
24 #undef DBGC_CLASS
25 #define DBGC_CLASS DBGC_QUOTA
26
27 #ifndef HAVE_SYS_QUOTAS
28 #undef HAVE_QUOTACTL_4B
29 #endif
30
31 #ifdef HAVE_QUOTACTL_4B
32 /* int quotactl(const char *path, int cmd, int id, char *addr)
33  *
34  * This is used by many (all?) BSD-derived systems. This implementation has
35  * been developed and tested on Darwin, but may also work on other BSD systems.
36  */
37
38 #ifdef HAVE_SYS_TYPES_H
39 #include <sys/types.h>
40 #endif
41
42 #ifdef HAVE_SYS_QUOTA_H
43 #include <sys/quota.h>
44 #endif
45
46 #ifdef HAVE_UFS_UFS_QUOTA_H
47 #include <ufs/ufs/quota.h>
48 #endif
49
50 #if defined(DARWINOS)
51 /* WorkARound broken HFS access checks in hfs_quotactl. Darwin only(?) */
52 #define HFS_QUOTACTL_WAR 1
53 #endif
54
55 static void xlate_qblk_to_smb(const struct dqblk * const qblk,
56                         SMB_DISK_QUOTA *dp)
57 {
58         ZERO_STRUCTP(dp);
59
60         DEBUG(10, ("unix softlimit=%u hardlimit=%u curblock=%u\n",
61             (unsigned)qblk->dqb_bsoftlimit, (unsigned)qblk->dqb_bhardlimit,
62 #ifdef HAVE_STRUCT_DQBLK_DQB_CURBYTES
63             (unsigned)qblk->dqb_curbytes));
64 #else
65             (unsigned)qblk->dqb_curblocks));
66 #endif
67
68         DEBUGADD(10, ("unix softinodes=%u hardinodes=%u curinodes=%u\n",
69             (unsigned)qblk->dqb_isoftlimit, (unsigned)qblk->dqb_ihardlimit,
70             (unsigned)qblk->dqb_curinodes));
71
72 #ifdef HAVE_STRUCT_DQBLK_DQB_CURBYTES
73         /* On Darwin, quotas are counted in bytes. We report them
74          * in 512b blocks because various callers have assumptions
75          * about the block size.
76          */
77 #define XLATE_TO_BLOCKS(bytes) (((bytes) + 1) / 512)
78         dp->bsize = 512;
79
80         dp->softlimit = XLATE_TO_BLOCKS(qblk->dqb_bsoftlimit);
81         dp->hardlimit = XLATE_TO_BLOCKS(qblk->dqb_bhardlimit);
82         dp->curblocks = XLATE_TO_BLOCKS(qblk->dqb_curbytes);
83 #undef XLATE_TO_BLOCKS
84 #else
85         dp->bsize = DEV_BSIZE;
86
87         dp->softlimit = qblk->dqb_bsoftlimit;
88         dp->hardlimit = qblk->dqb_bhardlimit;
89         dp->curblocks = qblk->dqb_curblocks;
90 #endif
91
92         dp->ihardlimit = qblk->dqb_ihardlimit;
93         dp->isoftlimit = qblk->dqb_isoftlimit;
94         dp->curinodes = qblk->dqb_curinodes;
95
96         dp->qflags = QUOTAS_ENABLED | QUOTAS_DENY_DISK;
97
98         DEBUG(10, ("softlimit=%u hardlimit=%u curblock=%u\n",
99             (unsigned)dp->softlimit, (unsigned)dp->hardlimit,
100             (unsigned)dp->curblocks));
101
102         DEBUGADD(10, ("softinodes=%u hardinodes=%u curinodes=%u\n",
103             (unsigned)dp->isoftlimit, (unsigned)dp->ihardlimit,
104             (unsigned)dp->curinodes));
105
106 }
107
108 static void xlate_smb_to_qblk(const SMB_DISK_QUOTA * const dp,
109                         struct dqblk *qblk)
110 {
111         ZERO_STRUCTP(qblk);
112
113         qblk->dqb_bsoftlimit = dp->softlimit;
114         qblk->dqb_bhardlimit = dp->hardlimit;
115 #ifdef HAVE_STRUCT_DQBLK_DQB_CURBYTES
116         /* On Darwin, quotas are counted in bytes. */
117         qblk->dqb_bsoftlimit *= dp->bsize;
118         qblk->dqb_bhardlimit *= dp->bsize;
119 #endif
120         qblk->dqb_ihardlimit = dp->ihardlimit;
121         qblk->dqb_isoftlimit = dp->isoftlimit;
122 }
123
124 static int sys_quotactl_4B(const char * path, int cmd,
125                 int id, struct dqblk *qblk)
126 {
127         int ret;
128
129         /* NB: We must test GRPQUOTA here, because USRQUOTA is 0. */
130         DEBUG(10, ("%s quota for %s ID %u on %s\n",
131                     (cmd & QCMD(Q_GETQUOTA, 0)) ? "getting" : "setting",
132                     (cmd & QCMD(0, GRPQUOTA)) ? "group" : "user",
133                     (unsigned)id, path));
134
135 #ifdef HFS_QUOTACTL_WAR
136         become_root();
137 #endif  /* HFS_QUOTACTL_WAR */
138
139         ret = quotactl(path, cmd, id, qblk);
140         if (ret == -1) {
141                 /* ENOTSUP means quota support is not compiled in. EINVAL
142                  * means that quotas are not configured (commonly).
143                  */
144                 if (errno != ENOTSUP && errno != EINVAL) {
145                         DEBUG(0, ("failed to %s quota for %s ID %u on %s: %s\n",
146                                     (cmd & QCMD(Q_GETQUOTA, 0)) ? "get" : "set",
147                                     (cmd & QCMD(0, GRPQUOTA)) ? "group" : "user",
148                                     (unsigned)id, path, strerror(errno)));
149                 }
150
151 #ifdef HFS_QUOTACTL_WAR
152                 unbecome_root();
153 #endif  /* HFS_QUOTACTL_WAR */
154
155
156                 return -1;
157         }
158
159 #ifdef HFS_QUOTACTL_WAR
160                 unbecome_root();
161 #endif  /* HFS_QUOTACTL_WAR */
162
163         return 0;
164 }
165
166 int sys_get_vfs_quota(const char *path, const char *bdev,
167         enum SMB_QUOTA_TYPE qtype, unid_t id, SMB_DISK_QUOTA *dp)
168 {
169         int ret;
170         struct dqblk qblk;
171
172         ZERO_STRUCT(qblk);
173
174         switch (qtype) {
175         case SMB_USER_QUOTA_TYPE:
176                 /* Get quota for provided UID. */
177                 ret = sys_quotactl_4B(path, QCMD(Q_GETQUOTA, USRQUOTA),
178                                         id.uid, &qblk);
179                 break;
180         case SMB_USER_FS_QUOTA_TYPE:
181                 /* Get quota for current UID. */
182                 ret = sys_quotactl_4B(path, QCMD(Q_GETQUOTA, USRQUOTA),
183                                         geteuid(), &qblk);
184                 break;
185         case SMB_GROUP_QUOTA_TYPE:
186                 /* Get quota for provided GID. */
187                 ret = sys_quotactl_4B(path, QCMD(Q_GETQUOTA, GRPQUOTA),
188                                         id.gid, &qblk);
189                 break;
190         case SMB_GROUP_FS_QUOTA_TYPE:
191                 /* Get quota for current GID. */
192                 ret = sys_quotactl_4B(path, QCMD(Q_GETQUOTA, GRPQUOTA),
193                                         getegid(), &qblk);
194                 break;
195         default:
196                 DEBUG(0, ("cannot get unsupported quota type: %u\n",
197                             (unsigned)qtype));
198                 errno = ENOSYS;
199                 return -1;
200         }
201
202         if (ret == -1) {
203                 return -1;
204         }
205
206         xlate_qblk_to_smb(&qblk, dp);
207         dp->qtype = qtype;
208
209         return ret;
210 }
211
212 int sys_set_vfs_quota(const char *path, const char *bdev,
213         enum SMB_QUOTA_TYPE qtype, unid_t id, SMB_DISK_QUOTA *dp)
214 {
215         struct dqblk qblk;
216
217         xlate_smb_to_qblk(dp, &qblk);
218
219         switch (qtype) {
220         case SMB_USER_QUOTA_TYPE:
221                 /* Set quota for provided UID. */
222                 return sys_quotactl_4B(path, QCMD(Q_SETQUOTA, USRQUOTA),
223                                         id.uid, &qblk);
224         case SMB_USER_FS_QUOTA_TYPE:
225                 /* Set quota for current UID. */
226                 return sys_quotactl_4B(path, QCMD(Q_SETQUOTA, USRQUOTA),
227                                         geteuid(), &qblk);
228         case SMB_GROUP_QUOTA_TYPE:
229                 /* Set quota for provided GID. */
230                 return sys_quotactl_4B(path, QCMD(Q_SETQUOTA, GRPQUOTA),
231                                         id.gid, &qblk);
232         case SMB_GROUP_FS_QUOTA_TYPE:
233                 /* Set quota for current GID. */
234                 return sys_quotactl_4B(path, QCMD(Q_SETQUOTA, GRPQUOTA),
235                                         getegid(), &qblk);
236         default:
237                 DEBUG(0, ("cannot set unsupported quota type: %u\n",
238                             (unsigned)qtype));
239                 errno = ENOSYS;
240                 return -1;
241         }
242 }
243
244 #endif /* HAVE_QUOTACTL_4B */