Commit a2d2329e authored by Christian Brauner's avatar Christian Brauner

ima: handle idmapped mounts

IMA does sometimes access the inode's i_uid and compares it against the
rules' fowner. Enable IMA to handle idmapped mounts by passing down the
mount's user namespace. We simply make use of the helpers we introduced
before. If the initial user namespace is passed nothing changes so
non-idmapped mounts will see identical behavior as before.

Link: https://lore.kernel.org/r/20210121131959.646623-27-christian.brauner@ubuntu.comSigned-off-by: default avatarChristian Brauner <christian.brauner@ubuntu.com>
parent 3cee6079
...@@ -401,7 +401,7 @@ int notify_change(struct user_namespace *mnt_userns, struct dentry *dentry, ...@@ -401,7 +401,7 @@ int notify_change(struct user_namespace *mnt_userns, struct dentry *dentry,
if (!error) { if (!error) {
fsnotify_change(dentry, ia_valid); fsnotify_change(dentry, ia_valid);
ima_inode_post_setattr(dentry); ima_inode_post_setattr(mnt_userns, dentry);
evm_inode_post_setattr(dentry, ia_valid); evm_inode_post_setattr(dentry, ia_valid);
} }
......
...@@ -3422,7 +3422,7 @@ struct dentry *vfs_tmpfile(struct user_namespace *mnt_userns, ...@@ -3422,7 +3422,7 @@ struct dentry *vfs_tmpfile(struct user_namespace *mnt_userns,
inode->i_state |= I_LINKABLE; inode->i_state |= I_LINKABLE;
spin_unlock(&inode->i_lock); spin_unlock(&inode->i_lock);
} }
ima_post_create_tmpfile(inode); ima_post_create_tmpfile(mnt_userns, inode);
return child; return child;
out_err: out_err:
...@@ -3750,7 +3750,7 @@ static long do_mknodat(int dfd, const char __user *filename, umode_t mode, ...@@ -3750,7 +3750,7 @@ static long do_mknodat(int dfd, const char __user *filename, umode_t mode,
error = vfs_create(mnt_userns, path.dentry->d_inode, error = vfs_create(mnt_userns, path.dentry->d_inode,
dentry, mode, true); dentry, mode, true);
if (!error) if (!error)
ima_post_path_mknod(dentry); ima_post_path_mknod(mnt_userns, dentry);
break; break;
case S_IFCHR: case S_IFBLK: case S_IFCHR: case S_IFBLK:
error = vfs_mknod(mnt_userns, path.dentry->d_inode, error = vfs_mknod(mnt_userns, path.dentry->d_inode,
......
...@@ -16,7 +16,8 @@ struct linux_binprm; ...@@ -16,7 +16,8 @@ struct linux_binprm;
#ifdef CONFIG_IMA #ifdef CONFIG_IMA
extern int ima_bprm_check(struct linux_binprm *bprm); extern int ima_bprm_check(struct linux_binprm *bprm);
extern int ima_file_check(struct file *file, int mask); extern int ima_file_check(struct file *file, int mask);
extern void ima_post_create_tmpfile(struct inode *inode); extern void ima_post_create_tmpfile(struct user_namespace *mnt_userns,
struct inode *inode);
extern void ima_file_free(struct file *file); extern void ima_file_free(struct file *file);
extern int ima_file_mmap(struct file *file, unsigned long prot); extern int ima_file_mmap(struct file *file, unsigned long prot);
extern int ima_file_mprotect(struct vm_area_struct *vma, unsigned long prot); extern int ima_file_mprotect(struct vm_area_struct *vma, unsigned long prot);
...@@ -27,7 +28,8 @@ extern int ima_read_file(struct file *file, enum kernel_read_file_id id, ...@@ -27,7 +28,8 @@ extern int ima_read_file(struct file *file, enum kernel_read_file_id id,
bool contents); bool contents);
extern int ima_post_read_file(struct file *file, void *buf, loff_t size, extern int ima_post_read_file(struct file *file, void *buf, loff_t size,
enum kernel_read_file_id id); enum kernel_read_file_id id);
extern void ima_post_path_mknod(struct dentry *dentry); extern void ima_post_path_mknod(struct user_namespace *mnt_userns,
struct dentry *dentry);
extern int ima_file_hash(struct file *file, char *buf, size_t buf_size); extern int ima_file_hash(struct file *file, char *buf, size_t buf_size);
extern int ima_inode_hash(struct inode *inode, char *buf, size_t buf_size); extern int ima_inode_hash(struct inode *inode, char *buf, size_t buf_size);
extern void ima_kexec_cmdline(int kernel_fd, const void *buf, int size); extern void ima_kexec_cmdline(int kernel_fd, const void *buf, int size);
...@@ -68,7 +70,8 @@ static inline int ima_file_check(struct file *file, int mask) ...@@ -68,7 +70,8 @@ static inline int ima_file_check(struct file *file, int mask)
return 0; return 0;
} }
static inline void ima_post_create_tmpfile(struct inode *inode) static inline void ima_post_create_tmpfile(struct user_namespace *mnt_userns,
struct inode *inode)
{ {
} }
...@@ -112,7 +115,8 @@ static inline int ima_post_read_file(struct file *file, void *buf, loff_t size, ...@@ -112,7 +115,8 @@ static inline int ima_post_read_file(struct file *file, void *buf, loff_t size,
return 0; return 0;
} }
static inline void ima_post_path_mknod(struct dentry *dentry) static inline void ima_post_path_mknod(struct user_namespace *mnt_userns,
struct dentry *dentry)
{ {
return; return;
} }
...@@ -153,7 +157,8 @@ static inline void ima_post_key_create_or_update(struct key *keyring, ...@@ -153,7 +157,8 @@ static inline void ima_post_key_create_or_update(struct key *keyring,
#ifdef CONFIG_IMA_APPRAISE #ifdef CONFIG_IMA_APPRAISE
extern bool is_ima_appraise_enabled(void); extern bool is_ima_appraise_enabled(void);
extern void ima_inode_post_setattr(struct dentry *dentry); extern void ima_inode_post_setattr(struct user_namespace *mnt_userns,
struct dentry *dentry);
extern int ima_inode_setxattr(struct dentry *dentry, const char *xattr_name, extern int ima_inode_setxattr(struct dentry *dentry, const char *xattr_name,
const void *xattr_value, size_t xattr_value_len); const void *xattr_value, size_t xattr_value_len);
extern int ima_inode_removexattr(struct dentry *dentry, const char *xattr_name); extern int ima_inode_removexattr(struct dentry *dentry, const char *xattr_name);
...@@ -163,7 +168,8 @@ static inline bool is_ima_appraise_enabled(void) ...@@ -163,7 +168,8 @@ static inline bool is_ima_appraise_enabled(void)
return 0; return 0;
} }
static inline void ima_inode_post_setattr(struct dentry *dentry) static inline void ima_inode_post_setattr(struct user_namespace *mnt_userns,
struct dentry *dentry)
{ {
return; return;
} }
......
...@@ -253,8 +253,9 @@ static inline void ima_process_queued_keys(void) {} ...@@ -253,8 +253,9 @@ static inline void ima_process_queued_keys(void) {}
#endif /* CONFIG_IMA_QUEUE_EARLY_BOOT_KEYS */ #endif /* CONFIG_IMA_QUEUE_EARLY_BOOT_KEYS */
/* LIM API function definitions */ /* LIM API function definitions */
int ima_get_action(struct inode *inode, const struct cred *cred, u32 secid, int ima_get_action(struct user_namespace *mnt_userns, struct inode *inode,
int mask, enum ima_hooks func, int *pcr, const struct cred *cred, u32 secid, int mask,
enum ima_hooks func, int *pcr,
struct ima_template_desc **template_desc, struct ima_template_desc **template_desc,
const char *keyring); const char *keyring);
int ima_must_measure(struct inode *inode, int mask, enum ima_hooks func); int ima_must_measure(struct inode *inode, int mask, enum ima_hooks func);
...@@ -266,7 +267,8 @@ void ima_store_measurement(struct integrity_iint_cache *iint, struct file *file, ...@@ -266,7 +267,8 @@ void ima_store_measurement(struct integrity_iint_cache *iint, struct file *file,
struct evm_ima_xattr_data *xattr_value, struct evm_ima_xattr_data *xattr_value,
int xattr_len, const struct modsig *modsig, int pcr, int xattr_len, const struct modsig *modsig, int pcr,
struct ima_template_desc *template_desc); struct ima_template_desc *template_desc);
void process_buffer_measurement(struct inode *inode, const void *buf, int size, void process_buffer_measurement(struct user_namespace *mnt_userns,
struct inode *inode, const void *buf, int size,
const char *eventname, enum ima_hooks func, const char *eventname, enum ima_hooks func,
int pcr, const char *keyring); int pcr, const char *keyring);
void ima_audit_measurement(struct integrity_iint_cache *iint, void ima_audit_measurement(struct integrity_iint_cache *iint,
...@@ -281,8 +283,9 @@ void ima_free_template_entry(struct ima_template_entry *entry); ...@@ -281,8 +283,9 @@ void ima_free_template_entry(struct ima_template_entry *entry);
const char *ima_d_path(const struct path *path, char **pathbuf, char *filename); const char *ima_d_path(const struct path *path, char **pathbuf, char *filename);
/* IMA policy related functions */ /* IMA policy related functions */
int ima_match_policy(struct inode *inode, const struct cred *cred, u32 secid, int ima_match_policy(struct user_namespace *mnt_userns, struct inode *inode,
enum ima_hooks func, int mask, int flags, int *pcr, const struct cred *cred, u32 secid, enum ima_hooks func,
int mask, int flags, int *pcr,
struct ima_template_desc **template_desc, struct ima_template_desc **template_desc,
const char *keyring); const char *keyring);
void ima_init_policy(void); void ima_init_policy(void);
...@@ -313,7 +316,8 @@ int ima_appraise_measurement(enum ima_hooks func, ...@@ -313,7 +316,8 @@ int ima_appraise_measurement(enum ima_hooks func,
struct file *file, const unsigned char *filename, struct file *file, const unsigned char *filename,
struct evm_ima_xattr_data *xattr_value, struct evm_ima_xattr_data *xattr_value,
int xattr_len, const struct modsig *modsig); int xattr_len, const struct modsig *modsig);
int ima_must_appraise(struct inode *inode, int mask, enum ima_hooks func); int ima_must_appraise(struct user_namespace *mnt_userns, struct inode *inode,
int mask, enum ima_hooks func);
void ima_update_xattr(struct integrity_iint_cache *iint, struct file *file); void ima_update_xattr(struct integrity_iint_cache *iint, struct file *file);
enum integrity_status ima_get_cache_status(struct integrity_iint_cache *iint, enum integrity_status ima_get_cache_status(struct integrity_iint_cache *iint,
enum ima_hooks func); enum ima_hooks func);
...@@ -340,7 +344,8 @@ static inline int ima_appraise_measurement(enum ima_hooks func, ...@@ -340,7 +344,8 @@ static inline int ima_appraise_measurement(enum ima_hooks func,
return INTEGRITY_UNKNOWN; return INTEGRITY_UNKNOWN;
} }
static inline int ima_must_appraise(struct inode *inode, int mask, static inline int ima_must_appraise(struct user_namespace *mnt_userns,
struct inode *inode, int mask,
enum ima_hooks func) enum ima_hooks func)
{ {
return 0; return 0;
......
...@@ -162,6 +162,7 @@ void ima_add_violation(struct file *file, const unsigned char *filename, ...@@ -162,6 +162,7 @@ void ima_add_violation(struct file *file, const unsigned char *filename,
/** /**
* ima_get_action - appraise & measure decision based on policy. * ima_get_action - appraise & measure decision based on policy.
* @mnt_userns: user namespace of the mount the inode was found from
* @inode: pointer to the inode associated with the object being validated * @inode: pointer to the inode associated with the object being validated
* @cred: pointer to credentials structure to validate * @cred: pointer to credentials structure to validate
* @secid: secid of the task being validated * @secid: secid of the task being validated
...@@ -183,8 +184,9 @@ void ima_add_violation(struct file *file, const unsigned char *filename, ...@@ -183,8 +184,9 @@ void ima_add_violation(struct file *file, const unsigned char *filename,
* Returns IMA_MEASURE, IMA_APPRAISE mask. * Returns IMA_MEASURE, IMA_APPRAISE mask.
* *
*/ */
int ima_get_action(struct inode *inode, const struct cred *cred, u32 secid, int ima_get_action(struct user_namespace *mnt_userns, struct inode *inode,
int mask, enum ima_hooks func, int *pcr, const struct cred *cred, u32 secid, int mask,
enum ima_hooks func, int *pcr,
struct ima_template_desc **template_desc, struct ima_template_desc **template_desc,
const char *keyring) const char *keyring)
{ {
...@@ -192,8 +194,8 @@ int ima_get_action(struct inode *inode, const struct cred *cred, u32 secid, ...@@ -192,8 +194,8 @@ int ima_get_action(struct inode *inode, const struct cred *cred, u32 secid,
flags &= ima_policy_flag; flags &= ima_policy_flag;
return ima_match_policy(inode, cred, secid, func, mask, flags, pcr, return ima_match_policy(mnt_userns, inode, cred, secid, func, mask,
template_desc, keyring); flags, pcr, template_desc, keyring);
} }
/* /*
......
...@@ -68,7 +68,8 @@ bool is_ima_appraise_enabled(void) ...@@ -68,7 +68,8 @@ bool is_ima_appraise_enabled(void)
* *
* Return 1 to appraise or hash * Return 1 to appraise or hash
*/ */
int ima_must_appraise(struct inode *inode, int mask, enum ima_hooks func) int ima_must_appraise(struct user_namespace *mnt_userns, struct inode *inode,
int mask, enum ima_hooks func)
{ {
u32 secid; u32 secid;
...@@ -76,8 +77,8 @@ int ima_must_appraise(struct inode *inode, int mask, enum ima_hooks func) ...@@ -76,8 +77,8 @@ int ima_must_appraise(struct inode *inode, int mask, enum ima_hooks func)
return 0; return 0;
security_task_getsecid(current, &secid); security_task_getsecid(current, &secid);
return ima_match_policy(inode, current_cred(), secid, func, mask, return ima_match_policy(mnt_userns, inode, current_cred(), secid, func,
IMA_APPRAISE | IMA_HASH, NULL, NULL, NULL); mask, IMA_APPRAISE | IMA_HASH, NULL, NULL, NULL);
} }
static int ima_fix_xattr(struct dentry *dentry, static int ima_fix_xattr(struct dentry *dentry,
...@@ -350,7 +351,7 @@ int ima_check_blacklist(struct integrity_iint_cache *iint, ...@@ -350,7 +351,7 @@ int ima_check_blacklist(struct integrity_iint_cache *iint,
rc = is_binary_blacklisted(digest, digestsize); rc = is_binary_blacklisted(digest, digestsize);
if ((rc == -EPERM) && (iint->flags & IMA_MEASURE)) if ((rc == -EPERM) && (iint->flags & IMA_MEASURE))
process_buffer_measurement(NULL, digest, digestsize, process_buffer_measurement(&init_user_ns, NULL, digest, digestsize,
"blacklisted-hash", NONE, "blacklisted-hash", NONE,
pcr, NULL); pcr, NULL);
} }
...@@ -501,6 +502,7 @@ void ima_update_xattr(struct integrity_iint_cache *iint, struct file *file) ...@@ -501,6 +502,7 @@ void ima_update_xattr(struct integrity_iint_cache *iint, struct file *file)
/** /**
* ima_inode_post_setattr - reflect file metadata changes * ima_inode_post_setattr - reflect file metadata changes
* @mnt_userns: user namespace of the mount the inode was found from
* @dentry: pointer to the affected dentry * @dentry: pointer to the affected dentry
* *
* Changes to a dentry's metadata might result in needing to appraise. * Changes to a dentry's metadata might result in needing to appraise.
...@@ -508,7 +510,8 @@ void ima_update_xattr(struct integrity_iint_cache *iint, struct file *file) ...@@ -508,7 +510,8 @@ void ima_update_xattr(struct integrity_iint_cache *iint, struct file *file)
* This function is called from notify_change(), which expects the caller * This function is called from notify_change(), which expects the caller
* to lock the inode's i_mutex. * to lock the inode's i_mutex.
*/ */
void ima_inode_post_setattr(struct dentry *dentry) void ima_inode_post_setattr(struct user_namespace *mnt_userns,
struct dentry *dentry)
{ {
struct inode *inode = d_backing_inode(dentry); struct inode *inode = d_backing_inode(dentry);
struct integrity_iint_cache *iint; struct integrity_iint_cache *iint;
...@@ -518,7 +521,7 @@ void ima_inode_post_setattr(struct dentry *dentry) ...@@ -518,7 +521,7 @@ void ima_inode_post_setattr(struct dentry *dentry)
|| !(inode->i_opflags & IOP_XATTR)) || !(inode->i_opflags & IOP_XATTR))
return; return;
action = ima_must_appraise(inode, MAY_ACCESS, POST_SETATTR); action = ima_must_appraise(mnt_userns, inode, MAY_ACCESS, POST_SETATTR);
if (!action) if (!action)
__vfs_removexattr(&init_user_ns, dentry, XATTR_NAME_IMA); __vfs_removexattr(&init_user_ns, dentry, XATTR_NAME_IMA);
iint = integrity_iint_find(inode); iint = integrity_iint_find(inode);
......
...@@ -10,6 +10,7 @@ ...@@ -10,6 +10,7 @@
*/ */
#include <keys/asymmetric-type.h> #include <keys/asymmetric-type.h>
#include <linux/user_namespace.h>
#include "ima.h" #include "ima.h"
/** /**
...@@ -58,7 +59,7 @@ void ima_post_key_create_or_update(struct key *keyring, struct key *key, ...@@ -58,7 +59,7 @@ void ima_post_key_create_or_update(struct key *keyring, struct key *key,
* if the IMA policy is configured to measure a key linked * if the IMA policy is configured to measure a key linked
* to the given keyring. * to the given keyring.
*/ */
process_buffer_measurement(NULL, payload, payload_len, process_buffer_measurement(&init_user_ns, NULL, payload, payload_len,
keyring->description, KEY_CHECK, 0, keyring->description, KEY_CHECK, 0,
keyring->description); keyring->description);
} }
...@@ -218,8 +218,8 @@ static int process_measurement(struct file *file, const struct cred *cred, ...@@ -218,8 +218,8 @@ static int process_measurement(struct file *file, const struct cred *cred,
* bitmask based on the appraise/audit/measurement policy. * bitmask based on the appraise/audit/measurement policy.
* Included is the appraise submask. * Included is the appraise submask.
*/ */
action = ima_get_action(inode, cred, secid, mask, func, &pcr, action = ima_get_action(file_mnt_user_ns(file), inode, cred, secid,
&template_desc, NULL); mask, func, &pcr, &template_desc, NULL);
violation_check = ((func == FILE_CHECK || func == MMAP_CHECK) && violation_check = ((func == FILE_CHECK || func == MMAP_CHECK) &&
(ima_policy_flag & IMA_MEASURE)); (ima_policy_flag & IMA_MEASURE));
if (!action && !violation_check) if (!action && !violation_check)
...@@ -431,8 +431,9 @@ int ima_file_mprotect(struct vm_area_struct *vma, unsigned long prot) ...@@ -431,8 +431,9 @@ int ima_file_mprotect(struct vm_area_struct *vma, unsigned long prot)
security_task_getsecid(current, &secid); security_task_getsecid(current, &secid);
inode = file_inode(vma->vm_file); inode = file_inode(vma->vm_file);
action = ima_get_action(inode, current_cred(), secid, MAY_EXEC, action = ima_get_action(file_mnt_user_ns(vma->vm_file), inode,
MMAP_CHECK, &pcr, &template, 0); current_cred(), secid, MAY_EXEC, MMAP_CHECK,
&pcr, &template, 0);
/* Is the mmap'ed file in policy? */ /* Is the mmap'ed file in policy? */
if (!(action & (IMA_MEASURE | IMA_APPRAISE_SUBMASK))) if (!(action & (IMA_MEASURE | IMA_APPRAISE_SUBMASK)))
...@@ -592,18 +593,21 @@ EXPORT_SYMBOL_GPL(ima_inode_hash); ...@@ -592,18 +593,21 @@ EXPORT_SYMBOL_GPL(ima_inode_hash);
/** /**
* ima_post_create_tmpfile - mark newly created tmpfile as new * ima_post_create_tmpfile - mark newly created tmpfile as new
* @mnt_userns: user namespace of the mount the inode was found from
* @file : newly created tmpfile * @file : newly created tmpfile
* *
* No measuring, appraising or auditing of newly created tmpfiles is needed. * No measuring, appraising or auditing of newly created tmpfiles is needed.
* Skip calling process_measurement(), but indicate which newly, created * Skip calling process_measurement(), but indicate which newly, created
* tmpfiles are in policy. * tmpfiles are in policy.
*/ */
void ima_post_create_tmpfile(struct inode *inode) void ima_post_create_tmpfile(struct user_namespace *mnt_userns,
struct inode *inode)
{ {
struct integrity_iint_cache *iint; struct integrity_iint_cache *iint;
int must_appraise; int must_appraise;
must_appraise = ima_must_appraise(inode, MAY_ACCESS, FILE_CHECK); must_appraise = ima_must_appraise(mnt_userns, inode, MAY_ACCESS,
FILE_CHECK);
if (!must_appraise) if (!must_appraise)
return; return;
...@@ -619,18 +623,21 @@ void ima_post_create_tmpfile(struct inode *inode) ...@@ -619,18 +623,21 @@ void ima_post_create_tmpfile(struct inode *inode)
/** /**
* ima_post_path_mknod - mark as a new inode * ima_post_path_mknod - mark as a new inode
* @mnt_userns: user namespace of the mount the inode was found from
* @dentry: newly created dentry * @dentry: newly created dentry
* *
* Mark files created via the mknodat syscall as new, so that the * Mark files created via the mknodat syscall as new, so that the
* file data can be written later. * file data can be written later.
*/ */
void ima_post_path_mknod(struct dentry *dentry) void ima_post_path_mknod(struct user_namespace *mnt_userns,
struct dentry *dentry)
{ {
struct integrity_iint_cache *iint; struct integrity_iint_cache *iint;
struct inode *inode = dentry->d_inode; struct inode *inode = dentry->d_inode;
int must_appraise; int must_appraise;
must_appraise = ima_must_appraise(inode, MAY_ACCESS, FILE_CHECK); must_appraise = ima_must_appraise(mnt_userns, inode, MAY_ACCESS,
FILE_CHECK);
if (!must_appraise) if (!must_appraise)
return; return;
...@@ -810,6 +817,7 @@ int ima_post_load_data(char *buf, loff_t size, ...@@ -810,6 +817,7 @@ int ima_post_load_data(char *buf, loff_t size,
/* /*
* process_buffer_measurement - Measure the buffer to ima log. * process_buffer_measurement - Measure the buffer to ima log.
* @mnt_userns: user namespace of the mount the inode was found from
* @inode: inode associated with the object being measured (NULL for KEY_CHECK) * @inode: inode associated with the object being measured (NULL for KEY_CHECK)
* @buf: pointer to the buffer that needs to be added to the log. * @buf: pointer to the buffer that needs to be added to the log.
* @size: size of buffer(in bytes). * @size: size of buffer(in bytes).
...@@ -820,7 +828,8 @@ int ima_post_load_data(char *buf, loff_t size, ...@@ -820,7 +828,8 @@ int ima_post_load_data(char *buf, loff_t size,
* *
* Based on policy, the buffer is measured into the ima log. * Based on policy, the buffer is measured into the ima log.
*/ */
void process_buffer_measurement(struct inode *inode, const void *buf, int size, void process_buffer_measurement(struct user_namespace *mnt_userns,
struct inode *inode, const void *buf, int size,
const char *eventname, enum ima_hooks func, const char *eventname, enum ima_hooks func,
int pcr, const char *keyring) int pcr, const char *keyring)
{ {
...@@ -860,8 +869,9 @@ void process_buffer_measurement(struct inode *inode, const void *buf, int size, ...@@ -860,8 +869,9 @@ void process_buffer_measurement(struct inode *inode, const void *buf, int size,
*/ */
if (func) { if (func) {
security_task_getsecid(current, &secid); security_task_getsecid(current, &secid);
action = ima_get_action(inode, current_cred(), secid, 0, func, action = ima_get_action(mnt_userns, inode, current_cred(),
&pcr, &template, keyring); secid, 0, func, &pcr, &template,
keyring);
if (!(action & IMA_MEASURE)) if (!(action & IMA_MEASURE))
return; return;
} }
...@@ -919,8 +929,9 @@ void ima_kexec_cmdline(int kernel_fd, const void *buf, int size) ...@@ -919,8 +929,9 @@ void ima_kexec_cmdline(int kernel_fd, const void *buf, int size)
if (!f.file) if (!f.file)
return; return;
process_buffer_measurement(file_inode(f.file), buf, size, process_buffer_measurement(file_mnt_user_ns(f.file), file_inode(f.file),
"kexec-cmdline", KEXEC_CMDLINE, 0, NULL); buf, size, "kexec-cmdline", KEXEC_CMDLINE, 0,
NULL);
fdput(f); fdput(f);
} }
......
...@@ -488,6 +488,7 @@ static bool ima_match_keyring(struct ima_rule_entry *rule, ...@@ -488,6 +488,7 @@ static bool ima_match_keyring(struct ima_rule_entry *rule,
/** /**
* ima_match_rules - determine whether an inode matches the policy rule. * ima_match_rules - determine whether an inode matches the policy rule.
* @rule: a pointer to a rule * @rule: a pointer to a rule
* @mnt_userns: user namespace of the mount the inode was found from
* @inode: a pointer to an inode * @inode: a pointer to an inode
* @cred: a pointer to a credentials structure for user validation * @cred: a pointer to a credentials structure for user validation
* @secid: the secid of the task to be validated * @secid: the secid of the task to be validated
...@@ -497,9 +498,10 @@ static bool ima_match_keyring(struct ima_rule_entry *rule, ...@@ -497,9 +498,10 @@ static bool ima_match_keyring(struct ima_rule_entry *rule,
* *
* Returns true on rule match, false on failure. * Returns true on rule match, false on failure.
*/ */
static bool ima_match_rules(struct ima_rule_entry *rule, struct inode *inode, static bool ima_match_rules(struct ima_rule_entry *rule,
const struct cred *cred, u32 secid, struct user_namespace *mnt_userns,
enum ima_hooks func, int mask, struct inode *inode, const struct cred *cred,
u32 secid, enum ima_hooks func, int mask,
const char *keyring) const char *keyring)
{ {
int i; int i;
...@@ -539,7 +541,7 @@ static bool ima_match_rules(struct ima_rule_entry *rule, struct inode *inode, ...@@ -539,7 +541,7 @@ static bool ima_match_rules(struct ima_rule_entry *rule, struct inode *inode,
} }
if ((rule->flags & IMA_FOWNER) && if ((rule->flags & IMA_FOWNER) &&
!rule->fowner_op(inode->i_uid, rule->fowner)) !rule->fowner_op(i_uid_into_mnt(mnt_userns, inode), rule->fowner))
return false; return false;
for (i = 0; i < MAX_LSM_RULES; i++) { for (i = 0; i < MAX_LSM_RULES; i++) {
int rc = 0; int rc = 0;
...@@ -602,6 +604,7 @@ static int get_subaction(struct ima_rule_entry *rule, enum ima_hooks func) ...@@ -602,6 +604,7 @@ static int get_subaction(struct ima_rule_entry *rule, enum ima_hooks func)
/** /**
* ima_match_policy - decision based on LSM and other conditions * ima_match_policy - decision based on LSM and other conditions
* @mnt_userns: user namespace of the mount the inode was found from
* @inode: pointer to an inode for which the policy decision is being made * @inode: pointer to an inode for which the policy decision is being made
* @cred: pointer to a credentials structure for which the policy decision is * @cred: pointer to a credentials structure for which the policy decision is
* being made * being made
...@@ -620,8 +623,9 @@ static int get_subaction(struct ima_rule_entry *rule, enum ima_hooks func) ...@@ -620,8 +623,9 @@ static int get_subaction(struct ima_rule_entry *rule, enum ima_hooks func)
* list when walking it. Reads are many orders of magnitude more numerous * list when walking it. Reads are many orders of magnitude more numerous
* than writes so ima_match_policy() is classical RCU candidate. * than writes so ima_match_policy() is classical RCU candidate.
*/ */
int ima_match_policy(struct inode *inode, const struct cred *cred, u32 secid, int ima_match_policy(struct user_namespace *mnt_userns, struct inode *inode,
enum ima_hooks func, int mask, int flags, int *pcr, const struct cred *cred, u32 secid, enum ima_hooks func,
int mask, int flags, int *pcr,
struct ima_template_desc **template_desc, struct ima_template_desc **template_desc,
const char *keyring) const char *keyring)
{ {
...@@ -637,8 +641,8 @@ int ima_match_policy(struct inode *inode, const struct cred *cred, u32 secid, ...@@ -637,8 +641,8 @@ int ima_match_policy(struct inode *inode, const struct cred *cred, u32 secid,
if (!(entry->action & actmask)) if (!(entry->action & actmask))
continue; continue;
if (!ima_match_rules(entry, inode, cred, secid, func, mask, if (!ima_match_rules(entry, mnt_userns, inode, cred, secid,
keyring)) func, mask, keyring))
continue; continue;
action |= entry->flags & IMA_ACTION_FLAGS; action |= entry->flags & IMA_ACTION_FLAGS;
......
...@@ -8,6 +8,7 @@ ...@@ -8,6 +8,7 @@
* Enables deferred processing of keys * Enables deferred processing of keys
*/ */
#include <linux/user_namespace.h>
#include <linux/workqueue.h> #include <linux/workqueue.h>
#include <keys/asymmetric-type.h> #include <keys/asymmetric-type.h>
#include "ima.h" #include "ima.h"
...@@ -158,7 +159,8 @@ void ima_process_queued_keys(void) ...@@ -158,7 +159,8 @@ void ima_process_queued_keys(void)
list_for_each_entry_safe(entry, tmp, &ima_keys, list) { list_for_each_entry_safe(entry, tmp, &ima_keys, list) {
if (!timer_expired) if (!timer_expired)
process_buffer_measurement(NULL, entry->payload, process_buffer_measurement(&init_user_ns, NULL,
entry->payload,
entry->payload_len, entry->payload_len,
entry->keyring_name, entry->keyring_name,
KEY_CHECK, 0, KEY_CHECK, 0,
......
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