// SPDX-License-Identifier: GPL-2.0-or-later
/*
* Copyright (c) 2013-2021 FUJITSU LIMITED. All rights reserved
* Author: DAN LI
* Author: Yang Xu
*/
/*\
* [Description]
*
* This testcases checks that quotactl(2) on xfs filesystem succeeds to:
*
* - turn off xfs quota and get xfs quota off status for user
* - turn on xfs quota and get xfs quota on status for user
* - set and use Q_XGETQUOTA to get xfs disk quota limits for user
* - set and use Q_XGETNEXTQUOTA to get xfs disk quota limits greater than or
* equal to ID for user
* - turn off xfs quota and get xfs quota off statv for user
* - turn on xfs quota and get xfs quota on statv for user
* - turn off xfs quota and get xfs quota off status for group
* - turn on xfs quota and get xfs quota on status for group
* - set and use Q_XGETQUOTA to get xfs disk quota limits for group
* - set and use Q_XGETNEXTQUOTA to get xfs disk quota limits for group
* - turn off xfs quota and get xfs quota off statv for group
* - turn on xfs quota and get xfs quota on statv for gorup
*/
#include "quotactl02.h"
#include "quotactl_syscall_var.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)
{
quotactl_info();
fd = SAFE_OPEN(MNTPOINT, O_RDONLY);
check_support_cmd(USRQUOTA);
check_support_cmd(GRPQUOTA);
}
static void cleanup(void)
{
if (fd > -1)
SAFE_CLOSE(fd);
}
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;
}
TST_EXP_PASS_SILENT(do_quotactl(fd, tc->cmd, tst_device->dev, test_id, tc->addr),
"do_quotactl()");
if (!TST_PASS)
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 = (const char *[]) {
"CONFIG_XFS_QUOTA",
NULL
},
.test = verify_quota,
.tcnt = ARRAY_SIZE(tcases),
.mount_device = 1,
.dev_fs_type = "xfs",
.mntpoint = MNTPOINT,
.mnt_data = "usrquota,grpquota",
.setup = setup,
.cleanup = cleanup,
.test_variants = QUOTACTL_SYSCALL_VARIANTS,
};
#else
TST_TEST_TCONF("System doesn't have ");
#endif