NFSD: Store parent's stat in a separate value
authorKinglong Mee <kinglongmee@gmail.com>
Thu, 30 Jul 2015 13:51:54 +0000 (21:51 +0800)
committerJ. Bruce Fields <bfields@redhat.com>
Mon, 31 Aug 2015 19:11:05 +0000 (15:11 -0400)
After commit ae7095a7c4 (nfsd4: helper function for getting mounted_on
ino) we ignore the return value from get_parent_attributes().

Also, the following FATTR4_WORD2_LAYOUT_BLKSIZE uses stat.blksize, so to
avoid overwriting that, use an independent value for the parent's
attributes.

Signed-off-by: Kinglong Mee <kinglongmee@gmail.com>
Signed-off-by: J. Bruce Fields <bfields@redhat.com>
fs/nfsd/nfs4xdr.c

index 75e0563c09d1911d927501ee52b53a3bd988940e..1aa0dc38ed5840a4af573ca3bc296cdc8506ba47 100644 (file)
@@ -2675,6 +2675,9 @@ out_acl:
                *p++ = cpu_to_be32(stat.mtime.tv_nsec);
        }
        if (bmval1 & FATTR4_WORD1_MOUNTED_ON_FILEID) {
+               struct kstat parent_stat;
+               u64 ino = stat.ino;
+
                p = xdr_reserve_space(xdr, 8);
                if (!p)
                        goto out_resource;
@@ -2683,9 +2686,13 @@ out_acl:
                 * and this is the root of a cross-mounted filesystem.
                 */
                if (ignore_crossmnt == 0 &&
-                   dentry == exp->ex_path.mnt->mnt_root)
-                       get_parent_attributes(exp, &stat);
-               p = xdr_encode_hyper(p, stat.ino);
+                   dentry == exp->ex_path.mnt->mnt_root) {
+                       err = get_parent_attributes(exp, &parent_stat);
+                       if (err)
+                               goto out_nfserr;
+                       ino = parent_stat.ino;
+               }
+               p = xdr_encode_hyper(p, ino);
        }
 #ifdef CONFIG_NFSD_PNFS
        if ((bmval1 & FATTR4_WORD1_FS_LAYOUT_TYPES) ||