|
@@ -101,9 +101,12 @@ static int nfs4_stat_to_errno(int);
|
|
#define nfs4_path_maxsz (1 + ((3 + NFS4_MAXPATHLEN) >> 2))
|
|
#define nfs4_path_maxsz (1 + ((3 + NFS4_MAXPATHLEN) >> 2))
|
|
#define nfs4_owner_maxsz (1 + XDR_QUADLEN(IDMAP_NAMESZ))
|
|
#define nfs4_owner_maxsz (1 + XDR_QUADLEN(IDMAP_NAMESZ))
|
|
#define nfs4_group_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: */
|
|
/* 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 + \
|
|
#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 + \
|
|
#define nfs4_fattr_maxsz (nfs4_fattr_bitmap_maxsz + \
|
|
nfs4_fattr_value_maxsz)
|
|
nfs4_fattr_value_maxsz)
|
|
#define decode_getattr_maxsz (op_decode_hdr_maxsz + nfs4_fattr_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);
|
|
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)
|
|
static void encode_fsinfo(struct xdr_stream *xdr, const u32* bitmask, struct compound_hdr *hdr)
|
|
{
|
|
{
|
|
encode_getattr_three(xdr,
|
|
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_putfh(xdr, args->fh, &hdr);
|
|
encode_open(xdr, args, &hdr);
|
|
encode_open(xdr, args, &hdr);
|
|
encode_getfh(xdr, &hdr);
|
|
encode_getfh(xdr, &hdr);
|
|
- encode_getfattr(xdr, args->bitmask, &hdr);
|
|
|
|
|
|
+ encode_getfattr_open(xdr, args->bitmask, &hdr);
|
|
encode_nops(&hdr);
|
|
encode_nops(&hdr);
|
|
}
|
|
}
|
|
|
|
|
|
@@ -4186,6 +4199,110 @@ xdr_error:
|
|
return status;
|
|
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,
|
|
static int decode_getfattr_attrs(struct xdr_stream *xdr, uint32_t *bitmap,
|
|
struct nfs_fattr *fattr, struct nfs_fh *fh,
|
|
struct nfs_fattr *fattr, struct nfs_fh *fh,
|
|
struct nfs4_fs_locations *fs_loc,
|
|
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;
|
|
goto xdr_error;
|
|
fattr->valid |= status;
|
|
fattr->valid |= status;
|
|
|
|
|
|
|
|
+ status = decode_attr_mdsthreshold(xdr, bitmap, fattr->mdsthreshold);
|
|
|
|
+ if (status < 0)
|
|
|
|
+ goto xdr_error;
|
|
|
|
+
|
|
xdr_error:
|
|
xdr_error:
|
|
dprintk("%s: xdr returned %d\n", __func__, -status);
|
|
dprintk("%s: xdr returned %d\n", __func__, -status);
|
|
return status;
|
|
return status;
|