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:
Linus Torvalds 2017-01-20 11:58:30 -08:00
commit 56ef18829e

View File

@ -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;