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:
+20
-17
@@ -73,6 +73,7 @@ int main(int argc, char *argv[])
|
||||
int cmd;
|
||||
int type = -1, typeflag = 0;
|
||||
int verbose = 0;
|
||||
int retval = 0;
|
||||
uint id = 0, idflag = 0;
|
||||
char *device = NULL;
|
||||
char *tmp;
|
||||
@@ -140,30 +141,32 @@ int main(int argc, char *argv[])
|
||||
cmd = QCMD(Q_GETNEXTQUOTA, type);
|
||||
if (quotactl(cmd, device, id, (void *)&dqb) < 0) {
|
||||
perror("Q_GETNEXTQUOTA");
|
||||
return 1;
|
||||
retval = 1;
|
||||
} else {
|
||||
/*
|
||||
* We only print id and inode limits because
|
||||
* block count varies depending on fs block size, etc;
|
||||
* this is just a sanity test that we can retrieve the quota,
|
||||
* and inode limits have the same units across both calls.
|
||||
*/
|
||||
printf("id %u\n", dqb.dqb_id);
|
||||
printf("ihard %llu\n",
|
||||
(unsigned long long)dqb.dqb_ihardlimit);
|
||||
printf("isoft %llu\n",
|
||||
(unsigned long long)dqb.dqb_isoftlimit);
|
||||
}
|
||||
|
||||
/*
|
||||
* We only print id and inode limits because
|
||||
* block count varies depending on fs block size, etc;
|
||||
* this is just a sanity test that we can retrieve the quota,
|
||||
* and inode limits have the same units across both calls.
|
||||
*/
|
||||
printf("id %u\n", dqb.dqb_id);
|
||||
printf("ihard %llu\n", (unsigned long long)dqb.dqb_ihardlimit);
|
||||
printf("isoft %llu\n", (unsigned long long)dqb.dqb_isoftlimit);
|
||||
|
||||
if (verbose)
|
||||
printf("====Q_XGETNEXTQUOTA====\n");
|
||||
cmd = QCMD(Q_XGETNEXTQUOTA, USRQUOTA);
|
||||
if (quotactl(cmd, device, id, (void *)&xqb) < 0) {
|
||||
perror("Q_XGETNEXTQUOTA");
|
||||
return 1;
|
||||
retval = 1;
|
||||
} else {
|
||||
printf("id %u\n", xqb.d_id);
|
||||
printf("ihard %llu\n", xqb.d_ino_hardlimit);
|
||||
printf("isoft %llu\n", xqb.d_ino_softlimit);
|
||||
}
|
||||
|
||||
printf("id %u\n", xqb.d_id);
|
||||
printf("ihard %llu\n", xqb.d_ino_hardlimit);
|
||||
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
|
||||
398 auto quick encrypt
|
||||
399 auto encrypt
|
||||
400 auto quick quota
|
||||
|
||||
Reference in New Issue
Block a user