#define EMBEDDED_LEVELS 2
struct nameidata {
struct path path;
- union {
- struct qstr last;
- struct path link;
- };
+ struct qstr last;
struct path root;
struct inode *inode; /* path.dentry.d_inode */
unsigned int flags;
return 0;
/* Allowed if owner and follower match. */
- inode = nd->link.dentry->d_inode;
+ inode = nd->stack[0].link.dentry->d_inode;
if (uid_eq(current_cred()->fsuid, inode->i_uid))
return 0;
if (uid_eq(parent->i_uid, inode->i_uid))
return 0;
- audit_log_link_denied("follow_link", &nd->link);
- path_put(&nd->link);
+ audit_log_link_denied("follow_link", &nd->stack[0].link);
+ path_put(&nd->stack[0].link);
path_put(&nd->path);
return -EACCES;
}
const char *get_link(struct nameidata *nd)
{
struct saved *last = nd->stack + nd->depth;
- struct dentry *dentry = nd->link.dentry;
+ struct dentry *dentry = last->link.dentry;
struct inode *inode = dentry->d_inode;
int error;
const char *res;
BUG_ON(nd->flags & LOOKUP_RCU);
- last->link = nd->link;
last->cookie = NULL;
nd->depth++;
static int pick_link(struct nameidata *nd, struct path *link)
{
int error;
+ struct saved *last;
if (unlikely(nd->total_link_count++ >= MAXSYMLINKS)) {
path_to_nameidata(link, nd);
return -ELOOP;
return error;
}
- nd->link = *link;
+ last = nd->stack + nd->depth;
+ last->link = *link;
return 1;
}