Misc fixes and cleanup for project quota
1) The Coverity Scan reports some issues for the project quota patch, including: 1.1) zfs_prop_get_userquota() directly uses the const quota type value as the condition check by wrong. 1.2) dmu_objset_userquota_get_ids() may cause dnode::dn_newgid to be overwritten by dnode::dn->dn_oldprojid. 2) This patch fixes related issues. It also enhances the logic for zfs_project_item_alloc() to avoid buffer overflow. 3) Skip project quota ability check if does not change project quota related things (id or flag). Otherwise, it will cause chattr (for other non project quota flags) operation failed if project quota disabled. Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov> Signed-off-by: Fan Yong <fan.yong@intel.com> Closes #7251 Closes #7265
This commit is contained in:
parent
dd3e1e3083
commit
2705ebf0a7
|
@ -43,7 +43,7 @@
|
||||||
|
|
||||||
typedef struct zfs_project_item {
|
typedef struct zfs_project_item {
|
||||||
list_node_t zpi_list;
|
list_node_t zpi_list;
|
||||||
char zpi_name[PATH_MAX];
|
char zpi_name[0];
|
||||||
} zfs_project_item_t;
|
} zfs_project_item_t;
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -51,7 +51,7 @@ zfs_project_item_alloc(list_t *head, const char *name)
|
||||||
{
|
{
|
||||||
zfs_project_item_t *zpi;
|
zfs_project_item_t *zpi;
|
||||||
|
|
||||||
zpi = safe_malloc(sizeof (zfs_project_item_t));
|
zpi = safe_malloc(sizeof (zfs_project_item_t) + strlen(name) + 1);
|
||||||
strcpy(zpi->zpi_name, name);
|
strcpy(zpi->zpi_name, name);
|
||||||
list_insert_tail(head, zpi);
|
list_insert_tail(head, zpi);
|
||||||
}
|
}
|
||||||
|
|
|
@ -3179,7 +3179,8 @@ zfs_prop_get_userquota(zfs_handle_t *zhp, const char *propname,
|
||||||
} else if (propvalue == 0 &&
|
} else if (propvalue == 0 &&
|
||||||
(type == ZFS_PROP_USERQUOTA || type == ZFS_PROP_GROUPQUOTA ||
|
(type == ZFS_PROP_USERQUOTA || type == ZFS_PROP_GROUPQUOTA ||
|
||||||
type == ZFS_PROP_USEROBJQUOTA || type == ZFS_PROP_GROUPOBJQUOTA ||
|
type == ZFS_PROP_USEROBJQUOTA || type == ZFS_PROP_GROUPOBJQUOTA ||
|
||||||
type == ZFS_PROP_PROJECTQUOTA || ZFS_PROP_PROJECTOBJQUOTA)) {
|
type == ZFS_PROP_PROJECTQUOTA ||
|
||||||
|
type == ZFS_PROP_PROJECTOBJQUOTA)) {
|
||||||
(void) strlcpy(propbuf, "none", proplen);
|
(void) strlcpy(propbuf, "none", proplen);
|
||||||
} else if (type == ZFS_PROP_USERQUOTA || type == ZFS_PROP_GROUPQUOTA ||
|
} else if (type == ZFS_PROP_USERQUOTA || type == ZFS_PROP_GROUPQUOTA ||
|
||||||
type == ZFS_PROP_USERUSED || type == ZFS_PROP_GROUPUSED ||
|
type == ZFS_PROP_USERUSED || type == ZFS_PROP_GROUPUSED ||
|
||||||
|
|
|
@ -2077,7 +2077,7 @@ dmu_objset_userquota_get_ids(dnode_t *dn, boolean_t before, dmu_tx_t *tx)
|
||||||
if (flags & DN_ID_OLD_EXIST) {
|
if (flags & DN_ID_OLD_EXIST) {
|
||||||
dn->dn_newuid = dn->dn_olduid;
|
dn->dn_newuid = dn->dn_olduid;
|
||||||
dn->dn_newgid = dn->dn_oldgid;
|
dn->dn_newgid = dn->dn_oldgid;
|
||||||
dn->dn_newgid = dn->dn_oldprojid;
|
dn->dn_newprojid = dn->dn_oldprojid;
|
||||||
} else {
|
} else {
|
||||||
dn->dn_newuid = 0;
|
dn->dn_newuid = 0;
|
||||||
dn->dn_newgid = 0;
|
dn->dn_newgid = 0;
|
||||||
|
|
|
@ -2891,6 +2891,8 @@ zfs_setattr(struct inode *ip, vattr_t *vap, int flags, cred_t *cr)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (XVA_ISSET_REQ(xvap, XAT_PROJINHERIT) &&
|
if (XVA_ISSET_REQ(xvap, XAT_PROJINHERIT) &&
|
||||||
|
(xoap->xoa_projinherit !=
|
||||||
|
((zp->z_pflags & ZFS_PROJINHERIT) != 0)) &&
|
||||||
(!dmu_objset_projectquota_enabled(os) ||
|
(!dmu_objset_projectquota_enabled(os) ||
|
||||||
(!S_ISREG(ip->i_mode) && !S_ISDIR(ip->i_mode)))) {
|
(!S_ISREG(ip->i_mode) && !S_ISDIR(ip->i_mode)))) {
|
||||||
ZFS_EXIT(zfsvfs);
|
ZFS_EXIT(zfsvfs);
|
||||||
|
|
Loading…
Reference in New Issue