|
@@ -4725,14 +4725,13 @@ static int decode_getfattr(struct xdr_stream *xdr, struct nfs_fattr *fattr,
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
/*
|
|
- * Decode potentially multiple layout types. Currently we only support
|
|
|
|
- * one layout driver per file system.
|
|
|
|
|
|
+ * Decode potentially multiple layout types.
|
|
*/
|
|
*/
|
|
-static int decode_first_pnfs_layout_type(struct xdr_stream *xdr,
|
|
|
|
|
|
+static int decode_pnfs_layout_types(struct xdr_stream *xdr,
|
|
uint32_t *layouttype)
|
|
uint32_t *layouttype)
|
|
{
|
|
{
|
|
__be32 *p;
|
|
__be32 *p;
|
|
- int num;
|
|
|
|
|
|
+ uint32_t num, i;
|
|
|
|
|
|
p = xdr_inline_decode(xdr, 4);
|
|
p = xdr_inline_decode(xdr, 4);
|
|
if (unlikely(!p))
|
|
if (unlikely(!p))
|
|
@@ -4741,18 +4740,17 @@ static int decode_first_pnfs_layout_type(struct xdr_stream *xdr,
|
|
|
|
|
|
/* pNFS is not supported by the underlying file system */
|
|
/* pNFS is not supported by the underlying file system */
|
|
if (num == 0) {
|
|
if (num == 0) {
|
|
- *layouttype = 0;
|
|
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
- if (num > 1)
|
|
|
|
- printk(KERN_INFO "NFS: %s: Warning: Multiple pNFS layout "
|
|
|
|
- "drivers per filesystem not supported\n", __func__);
|
|
|
|
|
|
+ if (num > NFS_MAX_LAYOUT_TYPES)
|
|
|
|
+ printk(KERN_INFO "NFS: %s: Warning: Too many (%d) pNFS layout types\n", __func__, num);
|
|
|
|
|
|
/* Decode and set first layout type, move xdr->p past unused types */
|
|
/* Decode and set first layout type, move xdr->p past unused types */
|
|
p = xdr_inline_decode(xdr, num * 4);
|
|
p = xdr_inline_decode(xdr, num * 4);
|
|
if (unlikely(!p))
|
|
if (unlikely(!p))
|
|
goto out_overflow;
|
|
goto out_overflow;
|
|
- *layouttype = be32_to_cpup(p);
|
|
|
|
|
|
+ for(i = 0; i < num && i < NFS_MAX_LAYOUT_TYPES; i++)
|
|
|
|
+ layouttype[i] = be32_to_cpup(p++);
|
|
return 0;
|
|
return 0;
|
|
out_overflow:
|
|
out_overflow:
|
|
print_overflow_msg(__func__, xdr);
|
|
print_overflow_msg(__func__, xdr);
|
|
@@ -4772,10 +4770,9 @@ static int decode_attr_pnfstype(struct xdr_stream *xdr, uint32_t *bitmap,
|
|
if (unlikely(bitmap[1] & (FATTR4_WORD1_FS_LAYOUT_TYPES - 1U)))
|
|
if (unlikely(bitmap[1] & (FATTR4_WORD1_FS_LAYOUT_TYPES - 1U)))
|
|
return -EIO;
|
|
return -EIO;
|
|
if (bitmap[1] & FATTR4_WORD1_FS_LAYOUT_TYPES) {
|
|
if (bitmap[1] & FATTR4_WORD1_FS_LAYOUT_TYPES) {
|
|
- status = decode_first_pnfs_layout_type(xdr, layouttype);
|
|
|
|
|
|
+ status = decode_pnfs_layout_types(xdr, layouttype);
|
|
bitmap[1] &= ~FATTR4_WORD1_FS_LAYOUT_TYPES;
|
|
bitmap[1] &= ~FATTR4_WORD1_FS_LAYOUT_TYPES;
|
|
- } else
|
|
|
|
- *layouttype = 0;
|
|
|
|
|
|
+ }
|
|
return status;
|
|
return status;
|
|
}
|
|
}
|
|
|
|
|
|
@@ -4856,7 +4853,7 @@ static int decode_fsinfo(struct xdr_stream *xdr, struct nfs_fsinfo *fsinfo)
|
|
status = decode_attr_time_delta(xdr, bitmap, &fsinfo->time_delta);
|
|
status = decode_attr_time_delta(xdr, bitmap, &fsinfo->time_delta);
|
|
if (status != 0)
|
|
if (status != 0)
|
|
goto xdr_error;
|
|
goto xdr_error;
|
|
- status = decode_attr_pnfstype(xdr, bitmap, &fsinfo->layouttype);
|
|
|
|
|
|
+ status = decode_attr_pnfstype(xdr, bitmap, fsinfo->layouttype);
|
|
if (status != 0)
|
|
if (status != 0)
|
|
goto xdr_error;
|
|
goto xdr_error;
|
|
|
|
|