mirror of
https://github.com/linux-apfs/apfstests.git
synced 2026-05-01 15:01:44 -07:00
generic: test GETNEXTQUOTA near INT_MAX
XFS kernel code had a bug where GETNEXTQUOTA-type quotactls requesting an ID near UINT_MAX could overflow and return 0 as the "next" active ID. This test checks that by creating an active quota near UINT_MAX, then asking for the next one after it. The proper answer is ENOENT, but if we wrap we'll return ID 0. This also changes test-nextquota.c so that it checks both GETNEXTQUOTA and XGETNEXTQUOTA even if one fails; it stores the failure conditions and returns 1 if either of them fails. Signed-off-by: Eric Sandeen <sandeen@redhat.com> Reviewed-by: Eryu Guan <eguan@redhat.com> Signed-off-by: Eryu Guan <eguan@redhat.com>
This commit is contained in:
+12
-9
@@ -73,6 +73,7 @@ int main(int argc, char *argv[])
|
|||||||
int cmd;
|
int cmd;
|
||||||
int type = -1, typeflag = 0;
|
int type = -1, typeflag = 0;
|
||||||
int verbose = 0;
|
int verbose = 0;
|
||||||
|
int retval = 0;
|
||||||
uint id = 0, idflag = 0;
|
uint id = 0, idflag = 0;
|
||||||
char *device = NULL;
|
char *device = NULL;
|
||||||
char *tmp;
|
char *tmp;
|
||||||
@@ -140,9 +141,8 @@ int main(int argc, char *argv[])
|
|||||||
cmd = QCMD(Q_GETNEXTQUOTA, type);
|
cmd = QCMD(Q_GETNEXTQUOTA, type);
|
||||||
if (quotactl(cmd, device, id, (void *)&dqb) < 0) {
|
if (quotactl(cmd, device, id, (void *)&dqb) < 0) {
|
||||||
perror("Q_GETNEXTQUOTA");
|
perror("Q_GETNEXTQUOTA");
|
||||||
return 1;
|
retval = 1;
|
||||||
}
|
} else {
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We only print id and inode limits because
|
* We only print id and inode limits because
|
||||||
* block count varies depending on fs block size, etc;
|
* block count varies depending on fs block size, etc;
|
||||||
@@ -150,20 +150,23 @@ int main(int argc, char *argv[])
|
|||||||
* and inode limits have the same units across both calls.
|
* and inode limits have the same units across both calls.
|
||||||
*/
|
*/
|
||||||
printf("id %u\n", dqb.dqb_id);
|
printf("id %u\n", dqb.dqb_id);
|
||||||
printf("ihard %llu\n", (unsigned long long)dqb.dqb_ihardlimit);
|
printf("ihard %llu\n",
|
||||||
printf("isoft %llu\n", (unsigned long long)dqb.dqb_isoftlimit);
|
(unsigned long long)dqb.dqb_ihardlimit);
|
||||||
|
printf("isoft %llu\n",
|
||||||
|
(unsigned long long)dqb.dqb_isoftlimit);
|
||||||
|
}
|
||||||
|
|
||||||
if (verbose)
|
if (verbose)
|
||||||
printf("====Q_XGETNEXTQUOTA====\n");
|
printf("====Q_XGETNEXTQUOTA====\n");
|
||||||
cmd = QCMD(Q_XGETNEXTQUOTA, USRQUOTA);
|
cmd = QCMD(Q_XGETNEXTQUOTA, USRQUOTA);
|
||||||
if (quotactl(cmd, device, id, (void *)&xqb) < 0) {
|
if (quotactl(cmd, device, id, (void *)&xqb) < 0) {
|
||||||
perror("Q_XGETNEXTQUOTA");
|
perror("Q_XGETNEXTQUOTA");
|
||||||
return 1;
|
retval = 1;
|
||||||
}
|
} else {
|
||||||
|
|
||||||
printf("id %u\n", xqb.d_id);
|
printf("id %u\n", xqb.d_id);
|
||||||
printf("ihard %llu\n", xqb.d_ino_hardlimit);
|
printf("ihard %llu\n", xqb.d_ino_hardlimit);
|
||||||
printf("isoft %llu\n", xqb.d_ino_softlimit);
|
printf("isoft %llu\n", xqb.d_ino_softlimit);
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|||||||
Executable
+93
@@ -0,0 +1,93 @@
|
|||||||
|
#! /bin/bash
|
||||||
|
# FS QA Test 394
|
||||||
|
#
|
||||||
|
# test out high quota ids retrieved by Q_GETNEXTQUOTA
|
||||||
|
# Request for next ID near 2^32 should not wrap to 0
|
||||||
|
#
|
||||||
|
# Designed to use the new Q_GETNEXTQUOTA quotactl
|
||||||
|
#
|
||||||
|
#-----------------------------------------------------------------------
|
||||||
|
# Copyright (c) 2016 Red Hat, Inc. All Rights Reserved.
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it would 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 the Free Software Foundation,
|
||||||
|
# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
#-----------------------------------------------------------------------
|
||||||
|
#
|
||||||
|
|
||||||
|
seq=`basename $0`
|
||||||
|
seqres=$RESULT_DIR/$seq
|
||||||
|
echo "QA output created by $seq"
|
||||||
|
|
||||||
|
here=`pwd`
|
||||||
|
tmp=/tmp/$$
|
||||||
|
status=1 # failure is the default!
|
||||||
|
trap "_cleanup; exit \$status" 0 1 2 3 15
|
||||||
|
|
||||||
|
_cleanup()
|
||||||
|
{
|
||||||
|
cd /
|
||||||
|
rm -f $tmp.*
|
||||||
|
}
|
||||||
|
|
||||||
|
# get standard environment, filters and checks
|
||||||
|
. ./common/rc
|
||||||
|
. ./common/filter
|
||||||
|
. ./common/quota
|
||||||
|
|
||||||
|
# remove previous $seqres.full before test
|
||||||
|
rm -f $seqres.full
|
||||||
|
|
||||||
|
# real QA test starts here
|
||||||
|
|
||||||
|
_supported_fs generic
|
||||||
|
_supported_os Linux
|
||||||
|
_require_quota
|
||||||
|
_require_scratch
|
||||||
|
|
||||||
|
_scratch_mkfs >> $seqres.full 2>&1
|
||||||
|
|
||||||
|
MOUNT_OPTIONS="-o usrquota,grpquota"
|
||||||
|
|
||||||
|
_qmount
|
||||||
|
|
||||||
|
# Ok, do we even have GETNEXTQUOTA? Querying ID 0 should work.
|
||||||
|
$here/src/test-nextquota -i 0 -u -d $SCRATCH_DEV &> $seqres.full || \
|
||||||
|
_notrun "No GETNEXTQUOTA support"
|
||||||
|
|
||||||
|
echo "Launch all quotas"
|
||||||
|
|
||||||
|
# We want to create a block of quotas for an id very near
|
||||||
|
# 2^32, then ask for the next quota after it. The returned
|
||||||
|
# ID should not overflow to 0.
|
||||||
|
|
||||||
|
# Populate with 2^32-4
|
||||||
|
ID=4294967292
|
||||||
|
setquota -u $ID $ID $ID $ID $ID $SCRATCH_MNT
|
||||||
|
touch ${SCRATCH_MNT}/${ID}
|
||||||
|
chown ${ID} ${SCRATCH_MNT}/${ID}
|
||||||
|
|
||||||
|
# remount just for kicks, make sure we get it off disk
|
||||||
|
_scratch_unmount
|
||||||
|
_qmount
|
||||||
|
|
||||||
|
# Ask for the next quota after $ID; should get nothing back
|
||||||
|
# If kernelspace wraps, we'll get 0 back.
|
||||||
|
for TYPE in u g; do
|
||||||
|
let NEXT=ID+1
|
||||||
|
echo "Ask for ID after $NEXT expecting nothing"
|
||||||
|
$here/src/test-nextquota -i $NEXT -${TYPE} -d $SCRATCH_DEV
|
||||||
|
done
|
||||||
|
|
||||||
|
# success, all done
|
||||||
|
status=0
|
||||||
|
exit
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
QA output created by 400
|
||||||
|
Launch all quotas
|
||||||
|
Ask for ID after 4294967293 expecting nothing
|
||||||
|
Q_GETNEXTQUOTA: No such file or directory
|
||||||
|
Q_XGETNEXTQUOTA: No such file or directory
|
||||||
|
Ask for ID after 4294967293 expecting nothing
|
||||||
|
Q_GETNEXTQUOTA: No such file or directory
|
||||||
|
Q_XGETNEXTQUOTA: No such file or directory
|
||||||
@@ -402,3 +402,4 @@
|
|||||||
397 auto quick encrypt
|
397 auto quick encrypt
|
||||||
398 auto quick encrypt
|
398 auto quick encrypt
|
||||||
399 auto encrypt
|
399 auto encrypt
|
||||||
|
400 auto quick quota
|
||||||
|
|||||||
Reference in New Issue
Block a user