Commit 63dff3e4 authored by Paul Moore's avatar Paul Moore

lsm: add the inode_free_security_rcu() LSM implementation hook

The LSM framework has an existing inode_free_security() hook which
is used by LSMs that manage state associated with an inode, but
due to the use of RCU to protect the inode, special care must be
taken to ensure that the LSMs do not fully release the inode state
until it is safe from a RCU perspective.

This patch implements a new inode_free_security_rcu() implementation
hook which is called when it is safe to free the LSM's internal inode
state.  Unfortunately, this new hook does not have access to the inode
itself as it may already be released, so the existing
inode_free_security() hook is retained for those LSMs which require
access to the inode.

Cc: stable@vger.kernel.org
Reported-by: syzbot+5446fbf332b0602ede0b@syzkaller.appspotmail.com
Closes: https://lore.kernel.org/r/00000000000076ba3b0617f65cc8@google.comSigned-off-by: default avatarPaul Moore <paul@paul-moore.com>
parent 711f5c5c
...@@ -114,6 +114,7 @@ LSM_HOOK(int, 0, path_notify, const struct path *path, u64 mask, ...@@ -114,6 +114,7 @@ LSM_HOOK(int, 0, path_notify, const struct path *path, u64 mask,
unsigned int obj_type) unsigned int obj_type)
LSM_HOOK(int, 0, inode_alloc_security, struct inode *inode) LSM_HOOK(int, 0, inode_alloc_security, struct inode *inode)
LSM_HOOK(void, LSM_RET_VOID, inode_free_security, struct inode *inode) LSM_HOOK(void, LSM_RET_VOID, inode_free_security, struct inode *inode)
LSM_HOOK(void, LSM_RET_VOID, inode_free_security_rcu, void *inode_security)
LSM_HOOK(int, -EOPNOTSUPP, inode_init_security, struct inode *inode, LSM_HOOK(int, -EOPNOTSUPP, inode_init_security, struct inode *inode,
struct inode *dir, const struct qstr *qstr, struct xattr *xattrs, struct inode *dir, const struct qstr *qstr, struct xattr *xattrs,
int *xattr_count) int *xattr_count)
......
...@@ -223,7 +223,7 @@ static inline void ima_inode_set_iint(const struct inode *inode, ...@@ -223,7 +223,7 @@ static inline void ima_inode_set_iint(const struct inode *inode,
struct ima_iint_cache *ima_iint_find(struct inode *inode); struct ima_iint_cache *ima_iint_find(struct inode *inode);
struct ima_iint_cache *ima_inode_get(struct inode *inode); struct ima_iint_cache *ima_inode_get(struct inode *inode);
void ima_inode_free(struct inode *inode); void ima_inode_free_rcu(void *inode_security);
void __init ima_iintcache_init(void); void __init ima_iintcache_init(void);
extern const int read_idmap[]; extern const int read_idmap[];
......
...@@ -109,22 +109,18 @@ struct ima_iint_cache *ima_inode_get(struct inode *inode) ...@@ -109,22 +109,18 @@ struct ima_iint_cache *ima_inode_get(struct inode *inode)
} }
/** /**
* ima_inode_free - Called on inode free * ima_inode_free_rcu - Called to free an inode via a RCU callback
* @inode: Pointer to the inode * @inode_security: The inode->i_security pointer
* *
* Free the iint associated with an inode. * Free the IMA data associated with an inode.
*/ */
void ima_inode_free(struct inode *inode) void ima_inode_free_rcu(void *inode_security)
{ {
struct ima_iint_cache *iint; struct ima_iint_cache **iint_p = inode_security + ima_blob_sizes.lbs_inode;
if (!IS_IMA(inode))
return;
iint = ima_iint_find(inode);
ima_inode_set_iint(inode, NULL);
ima_iint_free(iint); /* *iint_p should be NULL if !IS_IMA(inode) */
if (*iint_p)
ima_iint_free(*iint_p);
} }
static void ima_iint_init_once(void *foo) static void ima_iint_init_once(void *foo)
......
...@@ -1193,7 +1193,7 @@ static struct security_hook_list ima_hooks[] __ro_after_init = { ...@@ -1193,7 +1193,7 @@ static struct security_hook_list ima_hooks[] __ro_after_init = {
#ifdef CONFIG_INTEGRITY_ASYMMETRIC_KEYS #ifdef CONFIG_INTEGRITY_ASYMMETRIC_KEYS
LSM_HOOK_INIT(kernel_module_request, ima_kernel_module_request), LSM_HOOK_INIT(kernel_module_request, ima_kernel_module_request),
#endif #endif
LSM_HOOK_INIT(inode_free_security, ima_inode_free), LSM_HOOK_INIT(inode_free_security_rcu, ima_inode_free_rcu),
}; };
static const struct lsm_id ima_lsmid = { static const struct lsm_id ima_lsmid = {
......
...@@ -1207,13 +1207,16 @@ static int current_check_refer_path(struct dentry *const old_dentry, ...@@ -1207,13 +1207,16 @@ static int current_check_refer_path(struct dentry *const old_dentry,
/* Inode hooks */ /* Inode hooks */
static void hook_inode_free_security(struct inode *const inode) static void hook_inode_free_security_rcu(void *inode_security)
{ {
struct landlock_inode_security *inode_sec;
/* /*
* All inodes must already have been untied from their object by * All inodes must already have been untied from their object by
* release_inode() or hook_sb_delete(). * release_inode() or hook_sb_delete().
*/ */
WARN_ON_ONCE(landlock_inode(inode)->object); inode_sec = inode_security + landlock_blob_sizes.lbs_inode;
WARN_ON_ONCE(inode_sec->object);
} }
/* Super-block hooks */ /* Super-block hooks */
...@@ -1637,7 +1640,7 @@ static int hook_file_ioctl_compat(struct file *file, unsigned int cmd, ...@@ -1637,7 +1640,7 @@ static int hook_file_ioctl_compat(struct file *file, unsigned int cmd,
} }
static struct security_hook_list landlock_hooks[] __ro_after_init = { static struct security_hook_list landlock_hooks[] __ro_after_init = {
LSM_HOOK_INIT(inode_free_security, hook_inode_free_security), LSM_HOOK_INIT(inode_free_security_rcu, hook_inode_free_security_rcu),
LSM_HOOK_INIT(sb_delete, hook_sb_delete), LSM_HOOK_INIT(sb_delete, hook_sb_delete),
LSM_HOOK_INIT(sb_mount, hook_sb_mount), LSM_HOOK_INIT(sb_mount, hook_sb_mount),
......
...@@ -1609,9 +1609,8 @@ int security_inode_alloc(struct inode *inode) ...@@ -1609,9 +1609,8 @@ int security_inode_alloc(struct inode *inode)
static void inode_free_by_rcu(struct rcu_head *head) static void inode_free_by_rcu(struct rcu_head *head)
{ {
/* /* The rcu head is at the start of the inode blob */
* The rcu head is at the start of the inode blob call_void_hook(inode_free_security_rcu, head);
*/
kmem_cache_free(lsm_inode_cache, head); kmem_cache_free(lsm_inode_cache, head);
} }
...@@ -1619,23 +1618,24 @@ static void inode_free_by_rcu(struct rcu_head *head) ...@@ -1619,23 +1618,24 @@ static void inode_free_by_rcu(struct rcu_head *head)
* security_inode_free() - Free an inode's LSM blob * security_inode_free() - Free an inode's LSM blob
* @inode: the inode * @inode: the inode
* *
* Deallocate the inode security structure and set @inode->i_security to NULL. * Release any LSM resources associated with @inode, although due to the
* inode's RCU protections it is possible that the resources will not be
* fully released until after the current RCU grace period has elapsed.
*
* It is important for LSMs to note that despite being present in a call to
* security_inode_free(), @inode may still be referenced in a VFS path walk
* and calls to security_inode_permission() may be made during, or after,
* a call to security_inode_free(). For this reason the inode->i_security
* field is released via a call_rcu() callback and any LSMs which need to
* retain inode state for use in security_inode_permission() should only
* release that state in the inode_free_security_rcu() LSM hook callback.
*/ */
void security_inode_free(struct inode *inode) void security_inode_free(struct inode *inode)
{ {
call_void_hook(inode_free_security, inode); call_void_hook(inode_free_security, inode);
/* if (!inode->i_security)
* The inode may still be referenced in a path walk and return;
* a call to security_inode_permission() can be made call_rcu((struct rcu_head *)inode->i_security, inode_free_by_rcu);
* after inode_free_security() is called. Ideally, the VFS
* wouldn't do this, but fixing that is a much harder
* job. For now, simply free the i_security via RCU, and
* leave the current inode->i_security pointer intact.
* The inode will be freed after the RCU grace period too.
*/
if (inode->i_security)
call_rcu((struct rcu_head *)inode->i_security,
inode_free_by_rcu);
} }
/** /**
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment