]> git.hungrycats.org Git - linux/commitdiff
NFS: Fix a decoding problem in nfs3_decode_dirent
authorTrond Myklebust <Trond.Myklebust@netapp.com>
Fri, 18 Mar 2011 01:54:39 +0000 (21:54 -0400)
committerGreg Kroah-Hartman <gregkh@suse.de>
Wed, 23 Mar 2011 19:50:22 +0000 (12:50 -0700)
[This needs to be applied to 2.6.37 only. The bug in question was
inadvertently fixed by a series of cleanups in 2.6.38, but the patches
in question are too large to be backported. This patch is a minimal fix
that serves the same purpose.]

When we decode a filename followed by an 8-byte cookie, we need to
consider the fact that the filename and cookie are 32-bit word aligned.
Presently, we may end up copying insufficient amounts of data when
xdr_inline_decode() needs to invoke xdr_copy_to_scratch to deal
with a page boundary.

The following patch fixes the issue by first decoding the filename, and
then decoding the cookie.

Reported-by: Neil Brown <neilb@suse.de>
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Reviewed-by: NeilBrown <neilb@suse.de>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
fs/nfs/nfs2xdr.c
fs/nfs/nfs3xdr.c

index b382a1b5e7e4f8672acf2492c849cc13bd82ebc7..33a038d17992a3b2115644193cb83ba8d3d1feb0 100644 (file)
@@ -477,11 +477,13 @@ nfs_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, struct nfs_se
        entry->ino        = ntohl(*p++);
        entry->len        = ntohl(*p++);
 
-       p = xdr_inline_decode(xdr, entry->len + 4);
+       p = xdr_inline_decode(xdr, entry->len);
        if (unlikely(!p))
                goto out_overflow;
        entry->name       = (const char *) p;
-       p                += XDR_QUADLEN(entry->len);
+       p = xdr_inline_decode(xdr, 4);
+       if (unlikely(!p))
+               goto out_overflow;
        entry->prev_cookie        = entry->cookie;
        entry->cookie     = ntohl(*p++);
 
index ba91236c6ee74c0c3e42dba0ec3f7bbfd7c288c8..dcd934ffda0b9c3ab8d10cb34803b7032830eeea 100644 (file)
@@ -614,11 +614,13 @@ nfs3_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, struct nfs_s
        p = xdr_decode_hyper(p, &entry->ino);
        entry->len  = ntohl(*p++);
 
-       p = xdr_inline_decode(xdr, entry->len + 8);
+       p = xdr_inline_decode(xdr, entry->len);
        if (unlikely(!p))
                goto out_overflow;
        entry->name = (const char *) p;
-       p += XDR_QUADLEN(entry->len);
+       p = xdr_inline_decode(xdr, 8);
+       if (unlikely(!p))
+               goto out_overflow;
        entry->prev_cookie = entry->cookie;
        p = xdr_decode_hyper(p, &entry->cookie);