On 05/30/2016 09:59 AM, Andreas Gruenbacher wrote: > SELinux sometimes needs to load the security label of an inode without > knowing which dentry belongs to that inode (for example, in the > inode_permission hook). The security label is stored in an xattr; > getxattr currently requires both the dentry and the inode. > > So far, SELinux has been using d_find_alias to find any dentry for the > inode; there is no guarantee that d_find_alias finds a suitable dentry > on all types of filesystems, though. > > This patch changes SELinux calls getxattr with a NULL dentry when the > dentry is unknown. On filesystems that require a dentry, getxattr fails > with -ECHILD; on all others, it succeeds. > > Signed-off-by: Andreas Gruenbacher <agruenba@xxxxxxxxxx> > --- > fs/9p/acl.c | 3 +++ > fs/9p/xattr.c | 3 +++ > fs/cifs/xattr.c | 9 +++++++-- > fs/ecryptfs/inode.c | 8 ++++++-- > fs/overlayfs/inode.c | 6 +++++- > net/socket.c | 3 +++ > security/selinux/hooks.c | 43 +++++++++++++++---------------------------- > 7 files changed, 42 insertions(+), 33 deletions(-) > > diff --git a/fs/ecryptfs/inode.c b/fs/ecryptfs/inode.c > index 9d153b6..dd90e5d 100644 > --- a/fs/ecryptfs/inode.c > +++ b/fs/ecryptfs/inode.c > @@ -1043,8 +1043,12 @@ static ssize_t > ecryptfs_getxattr(struct dentry *dentry, struct inode *inode, > const char *name, void *value, size_t size) > { > - return ecryptfs_getxattr_lower(ecryptfs_dentry_to_lower(dentry), > - ecryptfs_inode_to_lower(inode), > + struct dentry *lower_dentry; > + struct inode *lower_inode; > + > + lower_dentry = dentry ? ecryptfs_dentry_to_lower(dentry) : NULL; > + lower_inode = ecryptfs_inode_to_lower(inode); > + return ecryptfs_getxattr_lower(lower_dentry, lower_inode, > name, value, size); ecryptfs_getxattr_lower() doesn't appear to handle a NULL lower_dentry. > diff --git a/security/selinux/hooks.c b/security/selinux/hooks.c > index a86d537..dd509c8 100644 > --- a/security/selinux/hooks.c > +++ b/security/selinux/hooks.c > @@ -1387,33 +1387,7 @@ static int inode_doinit_with_dentry(struct inode *inode, struct dentry *opt_dent > case SECURITY_FS_USE_NATIVE: > break; > case SECURITY_FS_USE_XATTR: > - if (!inode->i_op->getxattr) { > - isec->sid = sbsec->def_sid; > - break; > - } I don't think we can remove the above or we'll end up invoking inode->i_op->getxattr == NULL below. > - > - /* Need a dentry, since the xattr API requires one. > - Life would be simpler if we could just pass the inode. */ > - if (opt_dentry) { > - /* Called from d_instantiate or d_splice_alias. */ > - dentry = dget(opt_dentry); > - } else { > - /* Called from selinux_complete_init, try to find a dentry. */ > - dentry = d_find_alias(inode); > - } > - if (!dentry) { > - /* > - * this is can be hit on boot when a file is accessed > - * before the policy is loaded. When we load policy we > - * may find inodes that have no dentry on the > - * sbsec->isec_head list. No reason to complain as these > - * will get fixed up the next time we go through > - * inode_doinit with a dentry, before these inodes could > - * be used again by userspace. > - */ > - goto out_unlock; > - } > - > + dentry = dget(opt_dentry); > len = INITCONTEXTLEN; > context = kmalloc(len+1, GFP_NOFS); > if (!context) { > @@ -1448,7 +1422,20 @@ static int inode_doinit_with_dentry(struct inode *inode, struct dentry *opt_dent > } > dput(dentry); > if (rc < 0) { > - if (rc != -ENODATA) { > + if (rc == -ECHILD) { > + /* > + * The dentry is NULL and this ->getxattr > + * requires a dentry. We will keep trying > + * until we have a dentry and the label can be > + * loaded. > + * > + * (We could also remember when >getxattr > + * requires a dentry in the superblock if > + * retrying becomes too inefficient.) > + */ > + kfree(context); > + goto out_unlock; > + } else if (rc != -EOPNOTSUPP && rc != -ENODATA) { > printk(KERN_WARNING "SELinux: %s: getxattr returned " > "%d for dev=%s ino=%ld\n", __func__, > -rc, inode->i_sb->s_id, inode->i_ino); > -- To unsubscribe from this list: send the line "unsubscribe linux-fsdevel" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html