Merge branch 'overlayfs-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/mszeredi/vfs
Pull overlayfs fix from Miklos Szeredi: "This fixes a regression introduced in this cycle" * 'overlayfs-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/mszeredi/vfs: ovl: fix possible use after free on redirect dir lookup
This commit is contained in:
commit
56ef18829e
@ -154,29 +154,38 @@ out_err:
|
|||||||
static int ovl_lookup_layer(struct dentry *base, struct ovl_lookup_data *d,
|
static int ovl_lookup_layer(struct dentry *base, struct ovl_lookup_data *d,
|
||||||
struct dentry **ret)
|
struct dentry **ret)
|
||||||
{
|
{
|
||||||
const char *s = d->name.name;
|
/* Counting down from the end, since the prefix can change */
|
||||||
|
size_t rem = d->name.len - 1;
|
||||||
struct dentry *dentry = NULL;
|
struct dentry *dentry = NULL;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (*s != '/')
|
if (d->name.name[0] != '/')
|
||||||
return ovl_lookup_single(base, d, d->name.name, d->name.len,
|
return ovl_lookup_single(base, d, d->name.name, d->name.len,
|
||||||
0, "", ret);
|
0, "", ret);
|
||||||
|
|
||||||
while (*s++ == '/' && !IS_ERR_OR_NULL(base) && d_can_lookup(base)) {
|
while (!IS_ERR_OR_NULL(base) && d_can_lookup(base)) {
|
||||||
|
const char *s = d->name.name + d->name.len - rem;
|
||||||
const char *next = strchrnul(s, '/');
|
const char *next = strchrnul(s, '/');
|
||||||
size_t slen = strlen(s);
|
size_t thislen = next - s;
|
||||||
|
bool end = !next[0];
|
||||||
|
|
||||||
if (WARN_ON(slen > d->name.len) ||
|
/* Verify we did not go off the rails */
|
||||||
WARN_ON(strcmp(d->name.name + d->name.len - slen, s)))
|
if (WARN_ON(s[-1] != '/'))
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
err = ovl_lookup_single(base, d, s, next - s,
|
err = ovl_lookup_single(base, d, s, thislen,
|
||||||
d->name.len - slen, next, &base);
|
d->name.len - rem, next, &base);
|
||||||
dput(dentry);
|
dput(dentry);
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
dentry = base;
|
dentry = base;
|
||||||
s = next;
|
if (end)
|
||||||
|
break;
|
||||||
|
|
||||||
|
rem -= thislen + 1;
|
||||||
|
|
||||||
|
if (WARN_ON(rem >= d->name.len))
|
||||||
|
return -EIO;
|
||||||
}
|
}
|
||||||
*ret = dentry;
|
*ret = dentry;
|
||||||
return 0;
|
return 0;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user