mirror of
https://github.com/samba-team/samba.git
synced 2025-01-14 19:24:43 +03:00
af5dcaa740
most BSD systems have ufs/ufs/quota.h and they count the quota in blocks, not bytes and have slightly different dqblk struct members.
239 lines
6.3 KiB
C
239 lines
6.3 KiB
C
/*
|
|
* Unix SMB/CIFS implementation.
|
|
* System QUOTA function wrappers for QUOTACTL_4B
|
|
|
|
* Copyright (C) 2011 James Peach.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*/
|
|
|
|
#include "includes.h"
|
|
|
|
#undef DBGC_CLASS
|
|
#define DBGC_CLASS DBGC_QUOTA
|
|
|
|
#ifndef HAVE_SYS_QUOTAS
|
|
#undef HAVE_QUOTACTL_4B
|
|
#endif
|
|
|
|
#ifdef HAVE_QUOTACTL_4B
|
|
/* int quotactl(const char *path, int cmd, int id, char *addr)
|
|
*
|
|
* This is used by many (all?) BSD-derived systems. This implementation has
|
|
* been developed and tested on Darwin, but may also work on other BSD systems.
|
|
*/
|
|
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
#include <sys/types.h>
|
|
#endif
|
|
|
|
#ifdef HAVE_SYS_QUOTA_H
|
|
#include <sys/quota.h>
|
|
#endif
|
|
|
|
#ifdef HAVE_UFS_UFS_QUOTA_H
|
|
#include <ufs/ufs/quota.h>
|
|
#endif
|
|
|
|
#if defined(DARWINOS)
|
|
/* WorkARound broken HFS access checks in hfs_quotactl. Darwin only(?) */
|
|
#define HFS_QUOTACTL_WAR 1
|
|
#endif
|
|
|
|
static void xlate_qblk_to_smb(const struct dqblk * const qblk,
|
|
SMB_DISK_QUOTA *dp)
|
|
{
|
|
ZERO_STRUCTP(dp);
|
|
|
|
DEBUG(10, ("unix softlimit=%u hardlimit=%u curblock=%u\n",
|
|
(unsigned)qblk->dqb_bsoftlimit, (unsigned)qblk->dqb_bhardlimit,
|
|
#ifdef HAVE_STRUCT_DQBLK_DQB_CURBYTES
|
|
(unsigned)qblk->dqb_curbytes));
|
|
#else
|
|
(unsigned)qblk->dqb_curblocks));
|
|
#endif
|
|
|
|
DEBUGADD(10, ("unix softinodes=%u hardinodes=%u curinodes=%u\n",
|
|
(unsigned)qblk->dqb_isoftlimit, (unsigned)qblk->dqb_ihardlimit,
|
|
(unsigned)qblk->dqb_curinodes));
|
|
|
|
#ifdef HAVE_STRUCT_DQBLK_DQB_CURBYTES
|
|
/* On Darwin, quotas are counted in bytes. We report them
|
|
* in 512b blocks because various callers have assumptions
|
|
* about the block size.
|
|
*/
|
|
#define XLATE_TO_BLOCKS(bytes) (((bytes) + 1) / 512)
|
|
dp->bsize = 512;
|
|
|
|
dp->softlimit = XLATE_TO_BLOCKS(qblk->dqb_bsoftlimit);
|
|
dp->hardlimit = XLATE_TO_BLOCKS(qblk->dqb_bhardlimit);
|
|
dp->curblocks = XLATE_TO_BLOCKS(qblk->dqb_curbytes);
|
|
#undef XLATE_TO_BLOCKS
|
|
#endif
|
|
|
|
dp->ihardlimit = qblk->dqb_ihardlimit;
|
|
dp->isoftlimit = qblk->dqb_isoftlimit;
|
|
dp->curinodes = qblk->dqb_curinodes;
|
|
|
|
dp->qflags = QUOTAS_ENABLED | QUOTAS_DENY_DISK;
|
|
|
|
DEBUG(10, ("softlimit=%u hardlimit=%u curblock=%u\n",
|
|
(unsigned)dp->softlimit, (unsigned)dp->hardlimit,
|
|
(unsigned)dp->curblocks));
|
|
|
|
DEBUGADD(10, ("softinodes=%u hardinodes=%u curinodes=%u\n",
|
|
(unsigned)dp->isoftlimit, (unsigned)dp->ihardlimit,
|
|
(unsigned)dp->curinodes));
|
|
|
|
}
|
|
|
|
static void xlate_smb_to_qblk(const SMB_DISK_QUOTA * const dp,
|
|
struct dqblk *qblk)
|
|
{
|
|
ZERO_STRUCTP(qblk);
|
|
|
|
qblk->dqb_bsoftlimit = dp->softlimit;
|
|
qblk->dqb_bhardlimit = dp->hardlimit;
|
|
#ifdef HAVE_STRUCT_DQBLK_DQB_CURBYTES
|
|
/* On Darwin, quotas are counted in bytes. */
|
|
qblk->dqb_bsoftlimit *= dp->bsize;
|
|
qblk->dqb_bhardlimit *= dp->bsize;
|
|
#endif
|
|
qblk->dqb_ihardlimit = dp->ihardlimit;
|
|
qblk->dqb_isoftlimit = dp->isoftlimit;
|
|
}
|
|
|
|
static int sys_quotactl_4B(const char * path, int cmd,
|
|
int id, struct dqblk *qblk)
|
|
{
|
|
int ret;
|
|
|
|
/* NB: We must test GRPQUOTA here, because USRQUOTA is 0. */
|
|
DEBUG(10, ("%s quota for %s ID %u on %s\n",
|
|
(cmd & QCMD(Q_GETQUOTA, 0)) ? "getting" : "setting",
|
|
(cmd & QCMD(0, GRPQUOTA)) ? "group" : "user",
|
|
(unsigned)id, path));
|
|
|
|
#ifdef HFS_QUOTACTL_WAR
|
|
become_root();
|
|
#endif /* HFS_QUOTACTL_WAR */
|
|
|
|
ret = quotactl(path, cmd, id, qblk);
|
|
if (ret == -1) {
|
|
/* ENOTSUP means quota support is not compiled in. EINVAL
|
|
* means that quotas are not configured (commonly).
|
|
*/
|
|
if (errno != ENOTSUP && errno != EINVAL) {
|
|
DEBUG(0, ("failed to %s quota for %s ID %u on %s: %s\n",
|
|
(cmd & QCMD(Q_GETQUOTA, 0)) ? "get" : "set",
|
|
(cmd & QCMD(0, GRPQUOTA)) ? "group" : "user",
|
|
(unsigned)id, path, strerror(errno)));
|
|
}
|
|
|
|
#ifdef HFS_QUOTACTL_WAR
|
|
unbecome_root();
|
|
#endif /* HFS_QUOTACTL_WAR */
|
|
|
|
|
|
return -1;
|
|
}
|
|
|
|
#ifdef HFS_QUOTACTL_WAR
|
|
unbecome_root();
|
|
#endif /* HFS_QUOTACTL_WAR */
|
|
|
|
return 0;
|
|
}
|
|
|
|
int sys_get_vfs_quota(const char *path, const char *bdev,
|
|
enum SMB_QUOTA_TYPE qtype, unid_t id, SMB_DISK_QUOTA *dp)
|
|
{
|
|
int ret;
|
|
struct dqblk qblk;
|
|
|
|
ZERO_STRUCT(qblk);
|
|
|
|
switch (qtype) {
|
|
case SMB_USER_QUOTA_TYPE:
|
|
/* Get quota for provided UID. */
|
|
ret = sys_quotactl_4B(path, QCMD(Q_GETQUOTA, USRQUOTA),
|
|
id.uid, &qblk);
|
|
break;
|
|
case SMB_USER_FS_QUOTA_TYPE:
|
|
/* Get quota for current UID. */
|
|
ret = sys_quotactl_4B(path, QCMD(Q_GETQUOTA, USRQUOTA),
|
|
geteuid(), &qblk);
|
|
break;
|
|
case SMB_GROUP_QUOTA_TYPE:
|
|
/* Get quota for provided GID. */
|
|
ret = sys_quotactl_4B(path, QCMD(Q_GETQUOTA, GRPQUOTA),
|
|
id.gid, &qblk);
|
|
break;
|
|
case SMB_GROUP_FS_QUOTA_TYPE:
|
|
/* Get quota for current GID. */
|
|
ret = sys_quotactl_4B(path, QCMD(Q_GETQUOTA, GRPQUOTA),
|
|
getegid(), &qblk);
|
|
break;
|
|
default:
|
|
DEBUG(0, ("cannot get unsupported quota type: %u\n",
|
|
(unsigned)qtype));
|
|
errno = ENOSYS;
|
|
return -1;
|
|
}
|
|
|
|
if (ret == -1) {
|
|
return -1;
|
|
}
|
|
|
|
xlate_qblk_to_smb(&qblk, dp);
|
|
dp->qtype = qtype;
|
|
|
|
return ret;
|
|
}
|
|
|
|
int sys_set_vfs_quota(const char *path, const char *bdev,
|
|
enum SMB_QUOTA_TYPE qtype, unid_t id, SMB_DISK_QUOTA *dp)
|
|
{
|
|
struct dqblk qblk;
|
|
|
|
xlate_smb_to_qblk(dp, &qblk);
|
|
|
|
switch (qtype) {
|
|
case SMB_USER_QUOTA_TYPE:
|
|
/* Set quota for provided UID. */
|
|
return sys_quotactl_4B(path, QCMD(Q_SETQUOTA, USRQUOTA),
|
|
id.uid, &qblk);
|
|
case SMB_USER_FS_QUOTA_TYPE:
|
|
/* Set quota for current UID. */
|
|
return sys_quotactl_4B(path, QCMD(Q_SETQUOTA, USRQUOTA),
|
|
geteuid(), &qblk);
|
|
case SMB_GROUP_QUOTA_TYPE:
|
|
/* Set quota for provided GID. */
|
|
return sys_quotactl_4B(path, QCMD(Q_SETQUOTA, GRPQUOTA),
|
|
id.gid, &qblk);
|
|
case SMB_GROUP_FS_QUOTA_TYPE:
|
|
/* Set quota for current GID. */
|
|
return sys_quotactl_4B(path, QCMD(Q_SETQUOTA, GRPQUOTA),
|
|
getegid(), &qblk);
|
|
default:
|
|
DEBUG(0, ("cannot set unsupported quota type: %u\n",
|
|
(unsigned)qtype));
|
|
errno = ENOSYS;
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
#endif /* HAVE_QUOTACTL_4B */
|