/* $NetBSD: quota_open.c,v 1.7 2012/02/01 05:34:40 dholland Exp $ */ /*- * Copyright (c) 2011 The NetBSD Foundation, Inc. * All rights reserved. * * This code is derived from software contributed to The NetBSD Foundation * by David A. Holland. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include __RCSID("$NetBSD: quota_open.c,v 1.7 2012/02/01 05:34:40 dholland Exp $"); #include #include #include #include #include #include #include #include "quotapvt.h" struct quotahandle * quota_open(const char *path) { struct statvfs stv; struct quotahandle *qh; int mode; int serrno; /* * Probe order: * * 1. Check for NFS. NFS quota ops don't go to the kernel * at all but instead do RPCs to the NFS server's * rpc.rquotad, so it doesn't matter what the kernel * thinks. * * 2. Check if quotas are enabled in the mount flags. If * so, we can do quotactl calls. * * 3. Check if the volume is listed in fstab as one of * the filesystem types supported by quota_oldfiles.c, * and with the proper mount options to enable quotas. * * Note that (as of this writing) the mount options for * enabling quotas are accepted by mount for *all* filesystem * types and then ignored -- the kernel mount flag (ST_QUOTA / * MNT_QUOTA) gets set either by the filesystem based on its * own criteria, or for old-style quotas, during quotaon. The * quota filenames specified in fstab are not passed to or * known by the kernel except via quota_oldfiles.c! This is * generally gross but not easily fixed. */ if (statvfs(path, &stv) < 0) { return NULL; } __quota_oldfiles_load_fstab(); if (!strcmp(stv.f_fstypename, "nfs")) { mode = QUOTA_MODE_NFS; } else if ((stv.f_flag & ST_QUOTA) != 0) { mode = QUOTA_MODE_KERNEL; } else if (__quota_oldfiles_infstab(stv.f_mntonname)) { mode = QUOTA_MODE_OLDFILES; } else { errno = EOPNOTSUPP; return NULL; } qh = malloc(sizeof(*qh)); if (qh == NULL) { return NULL; } /* * Get the mount point from statvfs; this way the passed-in * path can be any path on the volume. */ qh->qh_mountpoint = strdup(stv.f_mntonname); if (qh->qh_mountpoint == NULL) { serrno = errno; free(qh); errno = serrno; return NULL; } qh->qh_mountdevice = strdup(stv.f_mntfromname); if (qh->qh_mountdevice == NULL) { serrno = errno; free(qh->qh_mountpoint); free(qh); errno = serrno; return NULL; } qh->qh_mode = mode; qh->qh_oldfilesopen = 0; qh->qh_userfile = -1; qh->qh_groupfile = -1; return qh; } const char * quota_getmountpoint(struct quotahandle *qh) { return qh->qh_mountpoint; } const char * quota_getmountdevice(struct quotahandle *qh) { return qh->qh_mountdevice; } void quota_close(struct quotahandle *qh) { if (qh->qh_userfile >= 0) { close(qh->qh_userfile); } if (qh->qh_groupfile >= 0) { close(qh->qh_groupfile); } free(qh->qh_mountdevice); free(qh->qh_mountpoint); free(qh); } int quota_quotaon(struct quotahandle *qh, int idtype) { switch (qh->qh_mode) { case QUOTA_MODE_NFS: errno = EOPNOTSUPP; break; case QUOTA_MODE_OLDFILES: return __quota_oldfiles_quotaon(qh, idtype); case QUOTA_MODE_KERNEL: return __quota_kernel_quotaon(qh, idtype); default: errno = EINVAL; break; } return -1; } int quota_quotaoff(struct quotahandle *qh, int idtype) { switch (qh->qh_mode) { case QUOTA_MODE_NFS: errno = EOPNOTSUPP; break; case QUOTA_MODE_OLDFILES: /* can't quotaoff if we haven't quotaon'd */ errno = ENOTCONN; break; case QUOTA_MODE_KERNEL: return __quota_kernel_quotaoff(qh, idtype); default: errno = EINVAL; break; } return -1; }