diff mbox series

[15/20] gfs2_drevalidate(): use stable parent inode and name passed by caller

Message ID 20250110024303.4157645-15-viro@zeniv.linux.org.uk (mailing list archive)
State New
Headers show
Series [01/20] make sure that DNAME_INLINE_LEN is a multiple of word size | expand

Commit Message

Al Viro Jan. 10, 2025, 2:42 a.m. UTC
No need to mess with dget_parent() for the former; for the latter we really should
not rely upon ->d_name.name remaining stable.  Again, a UAF there.

Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
---
 fs/gfs2/dentry.c | 24 ++++++++----------------
 1 file changed, 8 insertions(+), 16 deletions(-)

Comments

Andreas Grünbacher Jan. 10, 2025, 7:20 p.m. UTC | #1
Am Fr., 10. Jan. 2025 um 03:44 Uhr schrieb Al Viro <viro@zeniv.linux.org.uk>:
> No need to mess with dget_parent() for the former; for the latter we really should
> not rely upon ->d_name.name remaining stable.  Again, a UAF there.
>
> Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
> ---
>  fs/gfs2/dentry.c | 24 ++++++++----------------
>  1 file changed, 8 insertions(+), 16 deletions(-)
>
> diff --git a/fs/gfs2/dentry.c b/fs/gfs2/dentry.c
> index 86c338901fab..95050e719233 100644
> --- a/fs/gfs2/dentry.c
> +++ b/fs/gfs2/dentry.c
> @@ -35,48 +35,40 @@
>  static int gfs2_drevalidate(struct inode *dir, const struct qstr *name,
>                             struct dentry *dentry, unsigned int flags)
>  {
> -       struct dentry *parent;
> -       struct gfs2_sbd *sdp;
> -       struct gfs2_inode *dip;
> +       struct gfs2_sbd *sdp = GFS2_SB(dir);
> +       struct gfs2_inode *dip = GFS2_I(dir);
>         struct inode *inode;
>         struct gfs2_holder d_gh;
>         struct gfs2_inode *ip = NULL;
> -       int error, valid = 0;
> +       int error, valid;
>         int had_lock = 0;
>
>         if (flags & LOOKUP_RCU)
>                 return -ECHILD;
>
> -       parent = dget_parent(dentry);
> -       sdp = GFS2_SB(d_inode(parent));
> -       dip = GFS2_I(d_inode(parent));
>         inode = d_inode(dentry);
>
>         if (inode) {
>                 if (is_bad_inode(inode))
> -                       goto out;
> +                       return 0;
>                 ip = GFS2_I(inode);
>         }
>
> -       if (sdp->sd_lockstruct.ls_ops->lm_mount == NULL) {
> -               valid = 1;
> -               goto out;
> -       }
> +       if (sdp->sd_lockstruct.ls_ops->lm_mount == NULL)
> +               return 1;
>
>         had_lock = (gfs2_glock_is_locked_by_me(dip->i_gl) != NULL);
>         if (!had_lock) {
>                 error = gfs2_glock_nq_init(dip->i_gl, LM_ST_SHARED, 0, &d_gh);
>                 if (error)
> -                       goto out;
> +                       return 0;
>         }
>
> -       error = gfs2_dir_check(d_inode(parent), &dentry->d_name, ip);
> +       error = gfs2_dir_check(dir, name, ip);
>         valid = inode ? !error : (error == -ENOENT);
>
>         if (!had_lock)
>                 gfs2_glock_dq_uninit(&d_gh);
> -out:
> -       dput(parent);
>         return valid;
>  }
>
> --
> 2.39.5

Reviewed-by: Andreas Gruenbacher <agruenba@redhat.com>

Thanks,
Andreas
diff mbox series

Patch

diff --git a/fs/gfs2/dentry.c b/fs/gfs2/dentry.c
index 86c338901fab..95050e719233 100644
--- a/fs/gfs2/dentry.c
+++ b/fs/gfs2/dentry.c
@@ -35,48 +35,40 @@ 
 static int gfs2_drevalidate(struct inode *dir, const struct qstr *name,
 			    struct dentry *dentry, unsigned int flags)
 {
-	struct dentry *parent;
-	struct gfs2_sbd *sdp;
-	struct gfs2_inode *dip;
+	struct gfs2_sbd *sdp = GFS2_SB(dir);
+	struct gfs2_inode *dip = GFS2_I(dir);
 	struct inode *inode;
 	struct gfs2_holder d_gh;
 	struct gfs2_inode *ip = NULL;
-	int error, valid = 0;
+	int error, valid;
 	int had_lock = 0;
 
 	if (flags & LOOKUP_RCU)
 		return -ECHILD;
 
-	parent = dget_parent(dentry);
-	sdp = GFS2_SB(d_inode(parent));
-	dip = GFS2_I(d_inode(parent));
 	inode = d_inode(dentry);
 
 	if (inode) {
 		if (is_bad_inode(inode))
-			goto out;
+			return 0;
 		ip = GFS2_I(inode);
 	}
 
-	if (sdp->sd_lockstruct.ls_ops->lm_mount == NULL) {
-		valid = 1;
-		goto out;
-	}
+	if (sdp->sd_lockstruct.ls_ops->lm_mount == NULL)
+		return 1;
 
 	had_lock = (gfs2_glock_is_locked_by_me(dip->i_gl) != NULL);
 	if (!had_lock) {
 		error = gfs2_glock_nq_init(dip->i_gl, LM_ST_SHARED, 0, &d_gh);
 		if (error)
-			goto out;
+			return 0;
 	}
 
-	error = gfs2_dir_check(d_inode(parent), &dentry->d_name, ip);
+	error = gfs2_dir_check(dir, name, ip);
 	valid = inode ? !error : (error == -ENOENT);
 
 	if (!had_lock)
 		gfs2_glock_dq_uninit(&d_gh);
-out:
-	dput(parent);
 	return valid;
 }