|
|
|
@@ -10,6 +10,7 @@
|
|
|
|
|
#include <linux/fs.h>
|
|
|
|
|
#include <linux/namei.h>
|
|
|
|
|
#include <linux/xattr.h>
|
|
|
|
|
#include <linux/ratelimit.h>
|
|
|
|
|
#include "overlayfs.h"
|
|
|
|
|
#include "ovl_entry.h"
|
|
|
|
|
|
|
|
|
@@ -19,8 +20,66 @@ struct ovl_lookup_data {
|
|
|
|
|
bool opaque;
|
|
|
|
|
bool stop;
|
|
|
|
|
bool last;
|
|
|
|
|
char *redirect;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static int ovl_check_redirect(struct dentry *dentry, struct ovl_lookup_data *d,
|
|
|
|
|
size_t prelen, const char *post)
|
|
|
|
|
{
|
|
|
|
|
int res;
|
|
|
|
|
char *s, *next, *buf = NULL;
|
|
|
|
|
|
|
|
|
|
res = vfs_getxattr(dentry, OVL_XATTR_REDIRECT, NULL, 0);
|
|
|
|
|
if (res < 0) {
|
|
|
|
|
if (res == -ENODATA || res == -EOPNOTSUPP)
|
|
|
|
|
return 0;
|
|
|
|
|
goto fail;
|
|
|
|
|
}
|
|
|
|
|
buf = kzalloc(prelen + res + strlen(post) + 1, GFP_TEMPORARY);
|
|
|
|
|
if (!buf)
|
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
|
|
if (res == 0)
|
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
|
|
res = vfs_getxattr(dentry, OVL_XATTR_REDIRECT, buf, res);
|
|
|
|
|
if (res < 0)
|
|
|
|
|
goto fail;
|
|
|
|
|
if (res == 0)
|
|
|
|
|
goto invalid;
|
|
|
|
|
if (buf[0] == '/') {
|
|
|
|
|
for (s = buf; *s++ == '/'; s = next) {
|
|
|
|
|
next = strchrnul(s, '/');
|
|
|
|
|
if (s == next)
|
|
|
|
|
goto invalid;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (strchr(buf, '/') != NULL)
|
|
|
|
|
goto invalid;
|
|
|
|
|
|
|
|
|
|
memmove(buf + prelen, buf, res);
|
|
|
|
|
memcpy(buf, d->name.name, prelen);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
strcat(buf, post);
|
|
|
|
|
kfree(d->redirect);
|
|
|
|
|
d->redirect = buf;
|
|
|
|
|
d->name.name = d->redirect;
|
|
|
|
|
d->name.len = strlen(d->redirect);
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
err_free:
|
|
|
|
|
kfree(buf);
|
|
|
|
|
return 0;
|
|
|
|
|
fail:
|
|
|
|
|
pr_warn_ratelimited("overlayfs: failed to get redirect (%i)\n", res);
|
|
|
|
|
goto err_free;
|
|
|
|
|
invalid:
|
|
|
|
|
pr_warn_ratelimited("overlayfs: invalid redirect (%s)\n", buf);
|
|
|
|
|
goto err_free;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static bool ovl_is_opaquedir(struct dentry *dentry)
|
|
|
|
|
{
|
|
|
|
|
int res;
|
|
|
|
@@ -38,6 +97,7 @@ static bool ovl_is_opaquedir(struct dentry *dentry)
|
|
|
|
|
|
|
|
|
|
static int ovl_lookup_single(struct dentry *base, struct ovl_lookup_data *d,
|
|
|
|
|
const char *name, unsigned int namelen,
|
|
|
|
|
size_t prelen, const char *post,
|
|
|
|
|
struct dentry **ret)
|
|
|
|
|
{
|
|
|
|
|
struct dentry *this;
|
|
|
|
@@ -74,6 +134,9 @@ static int ovl_lookup_single(struct dentry *base, struct ovl_lookup_data *d,
|
|
|
|
|
d->stop = d->opaque = true;
|
|
|
|
|
goto out;
|
|
|
|
|
}
|
|
|
|
|
err = ovl_check_redirect(this, d, prelen, post);
|
|
|
|
|
if (err)
|
|
|
|
|
goto out_err;
|
|
|
|
|
out:
|
|
|
|
|
*ret = this;
|
|
|
|
|
return 0;
|
|
|
|
@@ -91,7 +154,32 @@ out_err:
|
|
|
|
|
static int ovl_lookup_layer(struct dentry *base, struct ovl_lookup_data *d,
|
|
|
|
|
struct dentry **ret)
|
|
|
|
|
{
|
|
|
|
|
return ovl_lookup_single(base, d, d->name.name, d->name.len, ret);
|
|
|
|
|
const char *s = d->name.name;
|
|
|
|
|
struct dentry *dentry = NULL;
|
|
|
|
|
int err;
|
|
|
|
|
|
|
|
|
|
if (*s != '/')
|
|
|
|
|
return ovl_lookup_single(base, d, d->name.name, d->name.len,
|
|
|
|
|
0, "", ret);
|
|
|
|
|
|
|
|
|
|
while (*s++ == '/' && !IS_ERR_OR_NULL(base) && d_can_lookup(base)) {
|
|
|
|
|
const char *next = strchrnul(s, '/');
|
|
|
|
|
size_t slen = strlen(s);
|
|
|
|
|
|
|
|
|
|
if (WARN_ON(slen > d->name.len) ||
|
|
|
|
|
WARN_ON(strcmp(d->name.name + d->name.len - slen, s)))
|
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
|
|
err = ovl_lookup_single(base, d, s, next - s,
|
|
|
|
|
d->name.len - slen, next, &base);
|
|
|
|
|
dput(dentry);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
dentry = base;
|
|
|
|
|
s = next;
|
|
|
|
|
}
|
|
|
|
|
*ret = dentry;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
@@ -127,6 +215,7 @@ struct dentry *ovl_lookup(struct inode *dir, struct dentry *dentry,
|
|
|
|
|
unsigned int ctr = 0;
|
|
|
|
|
struct inode *inode = NULL;
|
|
|
|
|
bool upperopaque = false;
|
|
|
|
|
char *upperredirect = NULL;
|
|
|
|
|
struct dentry *this;
|
|
|
|
|
unsigned int i;
|
|
|
|
|
int err;
|
|
|
|
@@ -136,6 +225,7 @@ struct dentry *ovl_lookup(struct inode *dir, struct dentry *dentry,
|
|
|
|
|
.opaque = false,
|
|
|
|
|
.stop = false,
|
|
|
|
|
.last = !poe->numlower,
|
|
|
|
|
.redirect = NULL,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (dentry->d_name.len > ofs->namelen)
|
|
|
|
@@ -153,12 +243,20 @@ struct dentry *ovl_lookup(struct inode *dir, struct dentry *dentry,
|
|
|
|
|
err = -EREMOTE;
|
|
|
|
|
goto out;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (d.redirect) {
|
|
|
|
|
upperredirect = kstrdup(d.redirect, GFP_KERNEL);
|
|
|
|
|
if (!upperredirect)
|
|
|
|
|
goto out_put_upper;
|
|
|
|
|
if (d.redirect[0] == '/')
|
|
|
|
|
poe = dentry->d_sb->s_root->d_fsdata;
|
|
|
|
|
}
|
|
|
|
|
upperopaque = d.opaque;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!d.stop && poe->numlower) {
|
|
|
|
|
err = -ENOMEM;
|
|
|
|
|
stack = kcalloc(poe->numlower, sizeof(struct path),
|
|
|
|
|
stack = kcalloc(ofs->numlower, sizeof(struct path),
|
|
|
|
|
GFP_TEMPORARY);
|
|
|
|
|
if (!stack)
|
|
|
|
|
goto out_put_upper;
|
|
|
|
@@ -178,6 +276,22 @@ struct dentry *ovl_lookup(struct inode *dir, struct dentry *dentry,
|
|
|
|
|
stack[ctr].dentry = this;
|
|
|
|
|
stack[ctr].mnt = lowerpath.mnt;
|
|
|
|
|
ctr++;
|
|
|
|
|
|
|
|
|
|
if (d.stop)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
if (d.redirect &&
|
|
|
|
|
d.redirect[0] == '/' &&
|
|
|
|
|
poe != dentry->d_sb->s_root->d_fsdata) {
|
|
|
|
|
poe = dentry->d_sb->s_root->d_fsdata;
|
|
|
|
|
|
|
|
|
|
/* Find the current layer on the root dentry */
|
|
|
|
|
for (i = 0; i < poe->numlower; i++)
|
|
|
|
|
if (poe->lowerstack[i].mnt == lowerpath.mnt)
|
|
|
|
|
break;
|
|
|
|
|
if (WARN_ON(i == poe->numlower))
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
oe = ovl_alloc_entry(ctr);
|
|
|
|
@@ -208,9 +322,11 @@ struct dentry *ovl_lookup(struct inode *dir, struct dentry *dentry,
|
|
|
|
|
|
|
|
|
|
revert_creds(old_cred);
|
|
|
|
|
oe->opaque = upperopaque;
|
|
|
|
|
oe->redirect = upperredirect;
|
|
|
|
|
oe->__upperdentry = upperdentry;
|
|
|
|
|
memcpy(oe->lowerstack, stack, sizeof(struct path) * ctr);
|
|
|
|
|
kfree(stack);
|
|
|
|
|
kfree(d.redirect);
|
|
|
|
|
dentry->d_fsdata = oe;
|
|
|
|
|
d_add(dentry, inode);
|
|
|
|
|
|
|
|
|
@@ -224,7 +340,9 @@ out_put:
|
|
|
|
|
kfree(stack);
|
|
|
|
|
out_put_upper:
|
|
|
|
|
dput(upperdentry);
|
|
|
|
|
kfree(upperredirect);
|
|
|
|
|
out:
|
|
|
|
|
kfree(d.redirect);
|
|
|
|
|
revert_creds(old_cred);
|
|
|
|
|
return ERR_PTR(err);
|
|
|
|
|
}
|
|
|
|
|