// SPDX-License-Identifier: GPL-2.0-or-later
/*
* Copyright (c) 2013-2019 FUJITSU LIMITED. All rights reserved
* Author: DAN LI
* Author: Yang Xu
*
* This testcase checks basic flags of quotactl(2) for an XFS file system:
* 1) quotactl(2) succeeds to turn off xfs quota and get xfs quota off status
* for user.
* 2) quotactl(2) succeeds to turn on xfs quota and get xfs quota on status
* for usr.
* 3) quotactl(2) succeeds to set and use Q_XGETQUOTA to get xfs disk quota
* limits for user.
* 4) quotactl(2) succeeds to set and use Q_XGETNEXTQUOTA to get xfs disk
* quota limits greater than or equal to ID for user.
* 5) quotactl(2) succeeds to turn off xfs quota and get xfs quota off statv
* for user.
* 6) quotactl(2) succeeds to turn on xfs quota and get xfs quota on statv
* for user.
* 7) quotactl(2) succeeds to turn off xfs quota and get xfs quota off status
* for group.
* 8) quotactl(2) succeeds to turn on xfs quota and get xfs quota on status
* for group.
* 9) quotactl(2) succeeds to set and use Q_XGETQUOTA to get xfs disk quota
* limits for group.
* 10)quotactl(2) succeeds to set and use Q_XGETNEXTQUOTA to get xfs disk
* quota limits for group.
* 11)quotactl(2) succeeds to turn off xfs quota and get xfs quota off statv
* for group.
* 12)quotactl(2) succeeds to turn on xfs quota and get xfs quota on statv
* for group.
*/
#include "quotactl02.h"
#ifdef HAVE_XFS_XQM_H
static uint32_t qflagu = XFS_QUOTA_UDQ_ENFD;
static uint32_t qflagg = XFS_QUOTA_GDQ_ENFD;
static struct t_case {
int cmd;
void *addr;
void (*func_check)();
int check_subcmd;
int flag;
char *des;
char *tname;
} tcases[] = {
{QCMD(Q_XQUOTAOFF, USRQUOTA), &qflagu, check_qoff,
QCMD(Q_XGETQSTAT, USRQUOTA), 1,
"turn off xfs quota and get xfs quota off status for user",
"QCMD(Q_XGETQSTAT, USRQUOTA) off"},
{QCMD(Q_XQUOTAON, USRQUOTA), &qflagu, check_qon,
QCMD(Q_XGETQSTAT, USRQUOTA), 1,
"turn on xfs quota and get xfs quota on status for user",
"QCMD(Q_XGETQSTAT, USRQUOTA) on"},
{QCMD(Q_XSETQLIM, USRQUOTA), &set_dquota, check_qlim,
QCMD(Q_XGETQUOTA, USRQUOTA), 0, "Q_XGETQUOTA for user",
"QCMD(Q_XGETQUOTA, USRQUOTA) qlim"},
{QCMD(Q_XSETQLIM, USRQUOTA), &set_dquota, check_qlim,
QCMD(Q_XGETNEXTQUOTA, USRQUOTA), 0, "Q_XGETNEXTQUOTA for user",
"QCMD(Q_XGETNEXTQUOTA, USRQUOTA)"},
{QCMD(Q_XQUOTAOFF, USRQUOTA), &qflagu, check_qoffv,
QCMD(Q_XGETQSTATV, USRQUOTA), 1,
"turn off xfs quota and get xfs quota off statv for user",
"QCMD(Q_XGETQSTATV, USRQUOTA) off"},
{QCMD(Q_XQUOTAON, USRQUOTA), &qflagu, check_qonv,
QCMD(Q_XGETQSTATV, USRQUOTA), 1,
"turn on xfs quota and get xfs quota on statv for user",
"QCMD(Q_XGETQSTATV, USRQUOTA) on"},
{QCMD(Q_XQUOTAOFF, GRPQUOTA), &qflagg, check_qoff,
QCMD(Q_XGETQSTAT, GRPQUOTA), 1,
"turn off xfs quota and get xfs quota off status for group",
"QCMD(Q_XGETQSTAT, GRPQUOTA) off"},
{QCMD(Q_XQUOTAON, GRPQUOTA), &qflagg, check_qon,
QCMD(Q_XGETQSTAT, GRPQUOTA), 1,
"turn on xfs quota and get xfs quota on status for group",
"QCMD(Q_XGETQSTAT, GRPQUOTA) on"},
{QCMD(Q_XSETQLIM, GRPQUOTA), &set_dquota, check_qlim,
QCMD(Q_XGETQUOTA, GRPQUOTA), 0, "Q_XGETQUOTA for group",
"QCMD(Q_XGETQUOTA, GRPQUOTA) qlim"},
{QCMD(Q_XSETQLIM, GRPQUOTA), &set_dquota, check_qlim,
QCMD(Q_XGETNEXTQUOTA, GRPQUOTA), 0, "Q_XGETNEXTQUOTA for group",
"QCMD(Q_XGETNEXTQUOTA, GRPQUOTA)"},
{QCMD(Q_XQUOTAOFF, GRPQUOTA), &qflagg, check_qoffv,
QCMD(Q_XGETQSTATV, GRPQUOTA), 1,
"turn off xfs quota and get xfs quota off statv for group",
"QCMD(Q_XGETQSTATV, GRPQUOTA) off"},
{QCMD(Q_XQUOTAON, GRPQUOTA), &qflagg, check_qonv,
QCMD(Q_XGETQSTATV, GRPQUOTA), 1,
"turn on xfs quota and get xfs quota on statv for group",
"QCMD(Q_XGETQSTATV, GRPQUOTA) on"},
};
static void setup(void)
{
test_id = geteuid();
check_support_cmd(USRQUOTA);
check_support_cmd(GRPQUOTA);
}
static void verify_quota(unsigned int n)
{
struct t_case *tc = &tcases[n];
tst_res(TINFO, "Test #%d: %s", n, tc->tname);
if ((tc->check_subcmd == QCMD(Q_XGETNEXTQUOTA, USRQUOTA)
|| tc->check_subcmd == QCMD(Q_XGETNEXTQUOTA, GRPQUOTA))
&& x_getnextquota_nsup) {
tst_res(TCONF, "current system doesn't support this cmd");
return;
}
if ((tc->check_subcmd == QCMD(Q_XGETQSTATV, USRQUOTA)
|| tc->check_subcmd == QCMD(Q_XGETQSTATV, GRPQUOTA))
&& x_getstatv_nsup) {
tst_res(TCONF, "current system doesn't support this cmd");
return;
}
TEST(quotactl(tc->cmd, tst_device->dev, test_id, tc->addr));
if (TST_RET == -1) {
tst_res(TFAIL | TTERRNO, "quotactl() failed to %s", tc->des);
return;
}
if (tc->flag)
tc->func_check(tc->check_subcmd, tc->des, *(int *)(tc->addr));
else
tc->func_check(tc->check_subcmd, tc->des);
}
static struct tst_test test = {
.needs_root = 1,
.needs_kconfigs = kconfigs,
.test = verify_quota,
.tcnt = ARRAY_SIZE(tcases),
.mount_device = 1,
.dev_fs_type = "xfs",
.mntpoint = mntpoint,
.mnt_data = "usrquota,grpquota",
.setup = setup,
};
#else
TST_TEST_TCONF("System doesn't have ");
#endif