1 1.7 dholland /* $NetBSD: quota_open.c,v 1.7 2012/02/01 05:34:40 dholland Exp $ */ 2 1.1 dholland /*- 3 1.1 dholland * Copyright (c) 2011 The NetBSD Foundation, Inc. 4 1.1 dholland * All rights reserved. 5 1.1 dholland * 6 1.1 dholland * This code is derived from software contributed to The NetBSD Foundation 7 1.1 dholland * by David A. Holland. 8 1.1 dholland * 9 1.1 dholland * Redistribution and use in source and binary forms, with or without 10 1.1 dholland * modification, are permitted provided that the following conditions 11 1.1 dholland * are met: 12 1.1 dholland * 1. Redistributions of source code must retain the above copyright 13 1.1 dholland * notice, this list of conditions and the following disclaimer. 14 1.1 dholland * 2. Redistributions in binary form must reproduce the above copyright 15 1.1 dholland * notice, this list of conditions and the following disclaimer in the 16 1.1 dholland * documentation and/or other materials provided with the distribution. 17 1.1 dholland * 18 1.1 dholland * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 19 1.1 dholland * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 20 1.1 dholland * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 21 1.1 dholland * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 22 1.1 dholland * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 23 1.1 dholland * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 24 1.1 dholland * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 25 1.1 dholland * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 26 1.1 dholland * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 27 1.1 dholland * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 28 1.1 dholland * POSSIBILITY OF SUCH DAMAGE. 29 1.1 dholland */ 30 1.1 dholland 31 1.1 dholland #include <sys/cdefs.h> 32 1.7 dholland __RCSID("$NetBSD: quota_open.c,v 1.7 2012/02/01 05:34:40 dholland Exp $"); 33 1.1 dholland 34 1.3 dholland #include <sys/types.h> 35 1.3 dholland #include <sys/statvfs.h> 36 1.2 dholland #include <stdlib.h> 37 1.1 dholland #include <string.h> 38 1.4 dholland #include <unistd.h> 39 1.1 dholland #include <errno.h> 40 1.1 dholland 41 1.1 dholland #include <quota.h> 42 1.2 dholland #include "quotapvt.h" 43 1.1 dholland 44 1.1 dholland struct quotahandle * 45 1.1 dholland quota_open(const char *path) 46 1.1 dholland { 47 1.3 dholland 48 1.3 dholland struct statvfs stv; 49 1.2 dholland struct quotahandle *qh; 50 1.5 dholland int mode; 51 1.2 dholland int serrno; 52 1.2 dholland 53 1.5 dholland /* 54 1.5 dholland * Probe order: 55 1.5 dholland * 56 1.5 dholland * 1. Check for NFS. NFS quota ops don't go to the kernel 57 1.5 dholland * at all but instead do RPCs to the NFS server's 58 1.5 dholland * rpc.rquotad, so it doesn't matter what the kernel 59 1.5 dholland * thinks. 60 1.5 dholland * 61 1.5 dholland * 2. Check if quotas are enabled in the mount flags. If 62 1.5 dholland * so, we can do quotactl calls. 63 1.5 dholland * 64 1.5 dholland * 3. Check if the volume is listed in fstab as one of 65 1.5 dholland * the filesystem types supported by quota_oldfiles.c, 66 1.5 dholland * and with the proper mount options to enable quotas. 67 1.6 dholland * 68 1.6 dholland * Note that (as of this writing) the mount options for 69 1.6 dholland * enabling quotas are accepted by mount for *all* filesystem 70 1.6 dholland * types and then ignored -- the kernel mount flag (ST_QUOTA / 71 1.6 dholland * MNT_QUOTA) gets set either by the filesystem based on its 72 1.6 dholland * own criteria, or for old-style quotas, during quotaon. The 73 1.6 dholland * quota filenames specified in fstab are not passed to or 74 1.6 dholland * known by the kernel except via quota_oldfiles.c! This is 75 1.6 dholland * generally gross but not easily fixed. 76 1.5 dholland */ 77 1.5 dholland 78 1.3 dholland if (statvfs(path, &stv) < 0) { 79 1.3 dholland return NULL; 80 1.3 dholland } 81 1.3 dholland 82 1.5 dholland __quota_oldfiles_load_fstab(); 83 1.5 dholland 84 1.3 dholland if (!strcmp(stv.f_fstypename, "nfs")) { 85 1.5 dholland mode = QUOTA_MODE_NFS; 86 1.5 dholland } else if ((stv.f_flag & ST_QUOTA) != 0) { 87 1.7 dholland mode = QUOTA_MODE_KERNEL; 88 1.5 dholland } else if (__quota_oldfiles_infstab(stv.f_mntonname)) { 89 1.5 dholland mode = QUOTA_MODE_OLDFILES; 90 1.3 dholland } else { 91 1.5 dholland errno = EOPNOTSUPP; 92 1.5 dholland return NULL; 93 1.3 dholland } 94 1.3 dholland 95 1.2 dholland qh = malloc(sizeof(*qh)); 96 1.2 dholland if (qh == NULL) { 97 1.2 dholland return NULL; 98 1.2 dholland } 99 1.3 dholland 100 1.3 dholland /* 101 1.3 dholland * Get the mount point from statvfs; this way the passed-in 102 1.3 dholland * path can be any path on the volume. 103 1.3 dholland */ 104 1.3 dholland 105 1.3 dholland qh->qh_mountpoint = strdup(stv.f_mntonname); 106 1.2 dholland if (qh->qh_mountpoint == NULL) { 107 1.2 dholland serrno = errno; 108 1.2 dholland free(qh); 109 1.2 dholland errno = serrno; 110 1.2 dholland return NULL; 111 1.2 dholland } 112 1.3 dholland 113 1.3 dholland qh->qh_mountdevice = strdup(stv.f_mntfromname); 114 1.3 dholland if (qh->qh_mountdevice == NULL) { 115 1.3 dholland serrno = errno; 116 1.3 dholland free(qh->qh_mountpoint); 117 1.3 dholland free(qh); 118 1.3 dholland errno = serrno; 119 1.3 dholland return NULL; 120 1.3 dholland } 121 1.3 dholland 122 1.5 dholland qh->qh_mode = mode; 123 1.3 dholland 124 1.5 dholland qh->qh_oldfilesopen = 0; 125 1.4 dholland qh->qh_userfile = -1; 126 1.4 dholland qh->qh_groupfile = -1; 127 1.4 dholland 128 1.2 dholland return qh; 129 1.1 dholland } 130 1.1 dholland 131 1.1 dholland const char * 132 1.1 dholland quota_getmountpoint(struct quotahandle *qh) 133 1.1 dholland { 134 1.2 dholland return qh->qh_mountpoint; 135 1.1 dholland } 136 1.1 dholland 137 1.1 dholland const char * 138 1.1 dholland quota_getmountdevice(struct quotahandle *qh) 139 1.1 dholland { 140 1.3 dholland return qh->qh_mountdevice; 141 1.1 dholland } 142 1.1 dholland 143 1.1 dholland void 144 1.1 dholland quota_close(struct quotahandle *qh) 145 1.1 dholland { 146 1.4 dholland if (qh->qh_userfile >= 0) { 147 1.4 dholland close(qh->qh_userfile); 148 1.4 dholland } 149 1.4 dholland if (qh->qh_groupfile >= 0) { 150 1.4 dholland close(qh->qh_groupfile); 151 1.4 dholland } 152 1.3 dholland free(qh->qh_mountdevice); 153 1.2 dholland free(qh->qh_mountpoint); 154 1.2 dholland free(qh); 155 1.1 dholland } 156 1.6 dholland 157 1.6 dholland int 158 1.6 dholland quota_quotaon(struct quotahandle *qh, int idtype) 159 1.6 dholland { 160 1.6 dholland switch (qh->qh_mode) { 161 1.6 dholland case QUOTA_MODE_NFS: 162 1.6 dholland errno = EOPNOTSUPP; 163 1.6 dholland break; 164 1.6 dholland case QUOTA_MODE_OLDFILES: 165 1.6 dholland return __quota_oldfiles_quotaon(qh, idtype); 166 1.7 dholland case QUOTA_MODE_KERNEL: 167 1.7 dholland return __quota_kernel_quotaon(qh, idtype); 168 1.6 dholland default: 169 1.6 dholland errno = EINVAL; 170 1.6 dholland break; 171 1.6 dholland } 172 1.6 dholland return -1; 173 1.6 dholland } 174 1.6 dholland 175 1.6 dholland int 176 1.6 dholland quota_quotaoff(struct quotahandle *qh, int idtype) 177 1.6 dholland { 178 1.6 dholland switch (qh->qh_mode) { 179 1.6 dholland case QUOTA_MODE_NFS: 180 1.6 dholland errno = EOPNOTSUPP; 181 1.6 dholland break; 182 1.6 dholland case QUOTA_MODE_OLDFILES: 183 1.6 dholland /* can't quotaoff if we haven't quotaon'd */ 184 1.6 dholland errno = ENOTCONN; 185 1.6 dholland break; 186 1.7 dholland case QUOTA_MODE_KERNEL: 187 1.7 dholland return __quota_kernel_quotaoff(qh, idtype); 188 1.6 dholland default: 189 1.6 dholland errno = EINVAL; 190 1.6 dholland break; 191 1.6 dholland } 192 1.6 dholland return -1; 193 1.6 dholland } 194