return rc;
if (id > 0)
cmd->u.pccc_add.pccc_flags |= PCC_DATASET_ROPCC;
+ } else if (strcmp(key, "proj_quota") == 0) {
+ rc = kstrtoul(val, 10, &id);
+ if (rc)
+ return rc;
+ if (id > 0)
+ cmd->u.pccc_add.pccc_flags |= PCC_DATASET_PROJ_QUOTA;
} else {
return -EINVAL;
}
/*
* Reset uid, gid or size for the PCC copy masked by @valid.
- * TODO: Set the project ID for PCC copy.
*/
int pcc_inode_reset_iattr(struct dentry *dentry, unsigned int valid,
kuid_t uid, kgid_t gid, loff_t size)
RETURN(rc);
}
+/* Set the project ID for PCC copy.*/
+int pcc_file_reset_projid(struct pcc_dataset *dataset, struct file *file,
+ __u32 projid)
+{
+ struct fsxattr fsx;
+ mm_segment_t old_fs;
+ int rc;
+
+ ENTRY;
+
+ if (!(dataset->pccd_flags & PCC_DATASET_PROJ_QUOTA))
+ RETURN(0);
+
+ if (!file->f_op->unlocked_ioctl)
+ RETURN(-ENOIOCTLCMD);
+
+ memset(&fsx, 0, sizeof(struct fsxattr));
+ fsx.fsx_projid = projid;
+ old_fs = get_fs();
+ set_fs(get_ds());
+ rc = file->f_op->unlocked_ioctl(file, FS_IOC_FSSETXATTR,
+ (unsigned long)&fsx);
+ set_fs(old_fs);
+
+ RETURN(rc);
+}
+
+int pcc_inode_reset_projid(struct pcc_dataset *dataset, struct dentry *dentry,
+ __u32 projid)
+{
+ struct path path;
+ struct file *file;
+ int rc;
+
+ ENTRY;
+
+ if (!(dataset->pccd_flags & PCC_DATASET_PROJ_QUOTA))
+ RETURN(0);
+
+ path.mnt = dataset->pccd_path.mnt;
+ path.dentry = dentry;
+ file = dentry_open(&path, O_WRONLY | O_LARGEFILE, current_cred());
+ if (IS_ERR_OR_NULL(file)) {
+ rc = file == NULL ? -EINVAL : PTR_ERR(file);
+ RETURN(rc);
+ }
+
+ rc = pcc_file_reset_projid(dataset, file, projid);
+ fput(file);
+ RETURN(rc);
+}
+
int pcc_inode_create(struct super_block *sb, struct pcc_dataset *dataset,
struct lu_fid *fid, struct dentry **pcc_dentry)
{
if (rc)
GOTO(out_put, rc);
+ rc = pcc_inode_reset_projid(pca->pca_dataset, pcc_dentry,
+ ll_i2info(inode)->lli_projid);
+ if (rc)
+ GOTO(out_put, rc);
+
pcc_inode_attach_set(super, pca->pca_dataset, ll_i2info(inode),
pcci, pcc_dentry, LU_PCC_READWRITE);
if (rc)
GOTO(out_fput, rc);
+ rc = pcc_file_reset_projid(dataset, pcc_filp,
+ ll_i2info(inode)->lli_projid);
+ if (rc)
+ GOTO(out_fput, rc);
+
ret = pcc_copy_data(file, pcc_filp);
if (ret < 0)
GOTO(out_fput, rc = ret);
PCC_DATASET_ROPCC = 0x20,
/* PCC backend provides caching services for both RW-PCC and RO-PCC */
PCC_DATASET_PCC_ALL = PCC_DATASET_RWPCC | PCC_DATASET_ROPCC,
+ /* Set the project ID for the PCC copy */
+ PCC_DATASET_PROJ_QUOTA = 0x40,
};
struct pcc_dataset {
stack_trap "umount_loopdev $facet $mntpt" EXIT
}
+setup_loopdev_project() {
+ local facet=$1
+ local file=$2
+ local mntpt=$3
+ local size=${4:-50}
+
+ do_facet $facet mkdir -p $mntpt || error "mkdir -p $mntpt failed"
+ stack_trap "do_facet $facet rm -rf $mntpt" EXIT
+ do_facet $facet dd if=/dev/zero of=$file bs=1M count=$size
+ stack_trap "do_facet $facet rm -f $file" EXIT
+ do_facet $facet mkfs.ext4 -O project,quota $file ||
+ error "mkfs.ext4 -O project,quota $file failed"
+ do_facet $facet file $file
+ do_facet $facet mount -t ext4 -o loop,prjquota $file $mntpt ||
+ error "mount -o loop,prjquota $file $mntpt failed"
+ stack_trap "umount_loopdev $facet $mntpt" EXIT
+ do_facet $facet mount | grep $mntpt
+}
+
lpcc_rw_test() {
local restore="$1"
local project="$2"
}
run_test 35 "Auto PCC-RO attach in atomic_open"
+test_36() {
+ quotaon --help |& grep -q 'project quotas' ||
+ skip "Not support project quota on local filesystem"
+
+ ! is_project_quota_supported &&
+ skip "project quota is not supported"
+
+ enable_project_quota
+
+ local loopfile="$TMP/$tfile"
+ local mntpt="/mnt/pcc.$tdir"
+ local hsm_root="$mntpt/$tdir"
+ local file=$DIR/$tfile
+ local id=100
+
+ setup_loopdev_project $SINGLEAGT $loopfile $mntpt 50
+ do_facet $SINGLEAGT quotaon -Ppv $mntpt
+ do_facet $SINGLEAGT setquota -P $id 0 4096 0 0 $mntpt ||
+ error "setquota -P $id on $mntpt failed"
+ do_facet $SINGLEAGT repquota -Pvs $mntpt
+
+ do_facet $SINGLEAGT mkdir $hsm_root || error "mkdir $hsm_root failed"
+ setup_pcc_mapping $SINGLEAGT \
+ "projid={$id}\ roid=$HSM_ARCHIVE_NUMBER\ proj_quota=1\ ropcc=1"
+ do_facet $SINGLEAGT $LCTL pcc list $MOUNT
+
+ do_facet $SINGLEAGT dd if=/dev/zero of=$file bs=1M count=2 ||
+ error "Write $file failed"
+ $LFS project -p $id $file || error "failed to set project for $file"
+ $LFS project -d $file
+ do_facet $SINGLEAGT dd if=$file of=/dev/null bs=1M count=2 ||
+ error "Read $file failed"
+ do_facet $SINGLEAGT $LFS pcc state $file
+ check_lpcc_state $file "readonly"
+ do_facet $SINGLEAGT repquota -Pvs $mntpt
+ do_facet $SINGLEAGT dd if=/dev/zero of=$file bs=1M count=5 ||
+ error "Write $file failed"
+ check_lpcc_state $file "none"
+ do_facet $SINGLEAGT dd if=$file of=/dev/null bs=1M count=5 ||
+ error "Read $file failed"
+ do_facet $SINGLEAGT repquota -Pvs $mntpt
+ do_facet $SINGLEAGT $LFS pcc state $file
+ check_lpcc_state $file "none"
+}
+run_test 36 "Test Project quota on loop PCC device"
+
complete $SECONDS
check_and_cleanup_lustre
exit_status