NFSv4.1 mdsthreshold attribute xdr
We only support one layout type per file system, so one threshold_item4 per mdsthreshold4. Signed-off-by: Andy Adamson <andros@netapp.com> Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
This commit is contained in:
parent
54ac471c83
commit
88034c3d88
125
fs/nfs/nfs4xdr.c
125
fs/nfs/nfs4xdr.c
@ -101,9 +101,12 @@ static int nfs4_stat_to_errno(int);
|
||||
#define nfs4_path_maxsz (1 + ((3 + NFS4_MAXPATHLEN) >> 2))
|
||||
#define nfs4_owner_maxsz (1 + XDR_QUADLEN(IDMAP_NAMESZ))
|
||||
#define nfs4_group_maxsz (1 + XDR_QUADLEN(IDMAP_NAMESZ))
|
||||
/* We support only one layout type per file system */
|
||||
#define decode_mdsthreshold_maxsz (1 + 1 + nfs4_fattr_bitmap_maxsz + 1 + 8)
|
||||
/* This is based on getfattr, which uses the most attributes: */
|
||||
#define nfs4_fattr_value_maxsz (1 + (1 + 2 + 2 + 4 + 2 + 1 + 1 + 2 + 2 + \
|
||||
3 + 3 + 3 + nfs4_owner_maxsz + nfs4_group_maxsz))
|
||||
3 + 3 + 3 + nfs4_owner_maxsz + \
|
||||
nfs4_group_maxsz + decode_mdsthreshold_maxsz))
|
||||
#define nfs4_fattr_maxsz (nfs4_fattr_bitmap_maxsz + \
|
||||
nfs4_fattr_value_maxsz)
|
||||
#define decode_getattr_maxsz (op_decode_hdr_maxsz + nfs4_fattr_maxsz)
|
||||
@ -1172,6 +1175,16 @@ static void encode_getfattr(struct xdr_stream *xdr, const u32* bitmask, struct c
|
||||
bitmask[1] & nfs4_fattr_bitmap[1], hdr);
|
||||
}
|
||||
|
||||
static void encode_getfattr_open(struct xdr_stream *xdr, const u32 *bitmask,
|
||||
struct compound_hdr *hdr)
|
||||
{
|
||||
encode_getattr_three(xdr,
|
||||
bitmask[0] & nfs4_fattr_bitmap[0],
|
||||
bitmask[1] & nfs4_fattr_bitmap[1],
|
||||
bitmask[2] & FATTR4_WORD2_MDSTHRESHOLD,
|
||||
hdr);
|
||||
}
|
||||
|
||||
static void encode_fsinfo(struct xdr_stream *xdr, const u32* bitmask, struct compound_hdr *hdr)
|
||||
{
|
||||
encode_getattr_three(xdr,
|
||||
@ -2164,7 +2177,7 @@ static void nfs4_xdr_enc_open(struct rpc_rqst *req, struct xdr_stream *xdr,
|
||||
encode_putfh(xdr, args->fh, &hdr);
|
||||
encode_open(xdr, args, &hdr);
|
||||
encode_getfh(xdr, &hdr);
|
||||
encode_getfattr(xdr, args->bitmask, &hdr);
|
||||
encode_getfattr_open(xdr, args->bitmask, &hdr);
|
||||
encode_nops(&hdr);
|
||||
}
|
||||
|
||||
@ -4186,6 +4199,110 @@ static int decode_pathconf(struct xdr_stream *xdr, struct nfs_pathconf *pathconf
|
||||
return status;
|
||||
}
|
||||
|
||||
static int decode_threshold_hint(struct xdr_stream *xdr,
|
||||
uint32_t *bitmap,
|
||||
uint64_t *res,
|
||||
uint32_t hint_bit)
|
||||
{
|
||||
__be32 *p;
|
||||
|
||||
*res = 0;
|
||||
if (likely(bitmap[0] & hint_bit)) {
|
||||
p = xdr_inline_decode(xdr, 8);
|
||||
if (unlikely(!p))
|
||||
goto out_overflow;
|
||||
xdr_decode_hyper(p, res);
|
||||
}
|
||||
return 0;
|
||||
out_overflow:
|
||||
print_overflow_msg(__func__, xdr);
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
static int decode_first_threshold_item4(struct xdr_stream *xdr,
|
||||
struct nfs4_threshold *res)
|
||||
{
|
||||
__be32 *p, *savep;
|
||||
uint32_t bitmap[3] = {0,}, attrlen;
|
||||
int status;
|
||||
|
||||
/* layout type */
|
||||
p = xdr_inline_decode(xdr, 4);
|
||||
if (unlikely(!p)) {
|
||||
print_overflow_msg(__func__, xdr);
|
||||
return -EIO;
|
||||
}
|
||||
res->l_type = be32_to_cpup(p);
|
||||
|
||||
/* thi_hintset bitmap */
|
||||
status = decode_attr_bitmap(xdr, bitmap);
|
||||
if (status < 0)
|
||||
goto xdr_error;
|
||||
|
||||
/* thi_hintlist length */
|
||||
status = decode_attr_length(xdr, &attrlen, &savep);
|
||||
if (status < 0)
|
||||
goto xdr_error;
|
||||
/* thi_hintlist */
|
||||
status = decode_threshold_hint(xdr, bitmap, &res->rd_sz, THRESHOLD_RD);
|
||||
if (status < 0)
|
||||
goto xdr_error;
|
||||
status = decode_threshold_hint(xdr, bitmap, &res->wr_sz, THRESHOLD_WR);
|
||||
if (status < 0)
|
||||
goto xdr_error;
|
||||
status = decode_threshold_hint(xdr, bitmap, &res->rd_io_sz,
|
||||
THRESHOLD_RD_IO);
|
||||
if (status < 0)
|
||||
goto xdr_error;
|
||||
status = decode_threshold_hint(xdr, bitmap, &res->wr_io_sz,
|
||||
THRESHOLD_WR_IO);
|
||||
if (status < 0)
|
||||
goto xdr_error;
|
||||
|
||||
status = verify_attr_len(xdr, savep, attrlen);
|
||||
res->bm = bitmap[0];
|
||||
|
||||
dprintk("%s bm=0x%x rd_sz=%llu wr_sz=%llu rd_io=%llu wr_io=%llu\n",
|
||||
__func__, res->bm, res->rd_sz, res->wr_sz, res->rd_io_sz,
|
||||
res->wr_io_sz);
|
||||
xdr_error:
|
||||
dprintk("%s ret=%d!\n", __func__, status);
|
||||
return status;
|
||||
}
|
||||
|
||||
/*
|
||||
* Thresholds on pNFS direct I/O vrs MDS I/O
|
||||
*/
|
||||
static int decode_attr_mdsthreshold(struct xdr_stream *xdr,
|
||||
uint32_t *bitmap,
|
||||
struct nfs4_threshold *res)
|
||||
{
|
||||
__be32 *p;
|
||||
int status = 0;
|
||||
uint32_t num;
|
||||
|
||||
if (unlikely(bitmap[2] & (FATTR4_WORD2_MDSTHRESHOLD - 1U)))
|
||||
return -EIO;
|
||||
if (likely(bitmap[2] & FATTR4_WORD2_MDSTHRESHOLD)) {
|
||||
p = xdr_inline_decode(xdr, 4);
|
||||
if (unlikely(!p))
|
||||
goto out_overflow;
|
||||
num = be32_to_cpup(p);
|
||||
if (num == 0)
|
||||
return 0;
|
||||
if (num > 1)
|
||||
printk(KERN_INFO "%s: Warning: Multiple pNFS layout "
|
||||
"drivers per filesystem not supported\n",
|
||||
__func__);
|
||||
|
||||
status = decode_first_threshold_item4(xdr, res);
|
||||
}
|
||||
return status;
|
||||
out_overflow:
|
||||
print_overflow_msg(__func__, xdr);
|
||||
return -EIO;
|
||||
}
|
||||
|
||||
static int decode_getfattr_attrs(struct xdr_stream *xdr, uint32_t *bitmap,
|
||||
struct nfs_fattr *fattr, struct nfs_fh *fh,
|
||||
struct nfs4_fs_locations *fs_loc,
|
||||
@ -4292,6 +4409,10 @@ static int decode_getfattr_attrs(struct xdr_stream *xdr, uint32_t *bitmap,
|
||||
goto xdr_error;
|
||||
fattr->valid |= status;
|
||||
|
||||
status = decode_attr_mdsthreshold(xdr, bitmap, fattr->mdsthreshold);
|
||||
if (status < 0)
|
||||
goto xdr_error;
|
||||
|
||||
xdr_error:
|
||||
dprintk("%s: xdr returned %d\n", __func__, -status);
|
||||
return status;
|
||||
|
@ -526,6 +526,13 @@ enum lock_type4 {
|
||||
#define FATTR4_WORD1_MOUNTED_ON_FILEID (1UL << 23)
|
||||
#define FATTR4_WORD1_FS_LAYOUT_TYPES (1UL << 30)
|
||||
#define FATTR4_WORD2_LAYOUT_BLKSIZE (1UL << 1)
|
||||
#define FATTR4_WORD2_MDSTHRESHOLD (1UL << 4)
|
||||
|
||||
/* MDS threshold bitmap bits */
|
||||
#define THRESHOLD_RD (1UL << 0)
|
||||
#define THRESHOLD_WR (1UL << 1)
|
||||
#define THRESHOLD_RD_IO (1UL << 2)
|
||||
#define THRESHOLD_WR_IO (1UL << 3)
|
||||
|
||||
#define NFSPROC4_NULL 0
|
||||
#define NFSPROC4_COMPOUND 1
|
||||
|
@ -35,6 +35,15 @@ static inline int nfs_fsid_equal(const struct nfs_fsid *a, const struct nfs_fsid
|
||||
return a->major == b->major && a->minor == b->minor;
|
||||
}
|
||||
|
||||
struct nfs4_threshold {
|
||||
__u32 bm;
|
||||
__u32 l_type;
|
||||
__u64 rd_sz;
|
||||
__u64 wr_sz;
|
||||
__u64 rd_io_sz;
|
||||
__u64 wr_io_sz;
|
||||
};
|
||||
|
||||
struct nfs_fattr {
|
||||
unsigned int valid; /* which fields are valid */
|
||||
umode_t mode;
|
||||
@ -67,6 +76,7 @@ struct nfs_fattr {
|
||||
unsigned long gencount;
|
||||
struct nfs4_string *owner_name;
|
||||
struct nfs4_string *group_name;
|
||||
struct nfs4_threshold *mdsthreshold; /* pNFS threshold hints */
|
||||
};
|
||||
|
||||
#define NFS_ATTR_FATTR_TYPE (1U << 0)
|
||||
|
Loading…
Reference in New Issue
Block a user