Commit 006ebb40 authored by Stephen Smalley's avatar Stephen Smalley Committed by James Morris

Security: split proc ptrace checking into read vs. attach

Enable security modules to distinguish reading of process state via
proc from full ptrace access by renaming ptrace_may_attach to
ptrace_may_access and adding a mode argument indicating whether only
read access or full attach access is requested.  This allows security
modules to permit access to reading process state without granting
full ptrace access.  The base DAC/capability checking remains unchanged.

Read access to /proc/pid/mem continues to apply a full ptrace attach
check since check_mem_permission() already requires the current task
to already be ptracing the target.  The other ptrace checks within
proc for elements like environ, maps, and fds are changed to pass the
read mode instead of attach.

In the SELinux case, we model such reading of process state as a
reading of a proc file labeled with the target process' label.  This
enables SELinux policy to permit such reading of process state without
permitting control or manipulation of the target process, as there are
a number of cases where programs probe for such information via proc
but do not need to be able to control the target (e.g. procps,
lsof, PolicyKit, ConsoleKit).  At present we have to choose between
allowing full ptrace in policy (more permissive than required/desired)
or breaking functionality (or in some cases just silencing the denials
via dontaudit rules but this can hide genuine attacks).

This version of the patch incorporates comments from Casey Schaufler
(change/replace existing ptrace_may_attach interface, pass access
mode), and Chris Wright (provide greater consistency in the checking).

Note that like their predecessors __ptrace_may_attach and
ptrace_may_attach, the __ptrace_may_access and ptrace_may_access
interfaces use different return value conventions from each other (0
or -errno vs. 1 or 0).  I retained this difference to avoid any
changes to the caller logic but made the difference clearer by
changing the latter interface to return a bool rather than an int and
by adding a comment about it to ptrace.h for any future callers.
Signed-off-by: default avatarStephen Smalley <sds@tycho.nsa.gov>
Acked-by: default avatarChris Wright <chrisw@sous-sol.org>
Signed-off-by: default avatarJames Morris <jmorris@namei.org>
parent feb2a5b8
...@@ -233,7 +233,7 @@ static int check_mem_permission(struct task_struct *task) ...@@ -233,7 +233,7 @@ static int check_mem_permission(struct task_struct *task)
*/ */
if (task->parent == current && (task->ptrace & PT_PTRACED) && if (task->parent == current && (task->ptrace & PT_PTRACED) &&
task_is_stopped_or_traced(task) && task_is_stopped_or_traced(task) &&
ptrace_may_attach(task)) ptrace_may_access(task, PTRACE_MODE_ATTACH))
return 0; return 0;
/* /*
...@@ -251,7 +251,8 @@ struct mm_struct *mm_for_maps(struct task_struct *task) ...@@ -251,7 +251,8 @@ struct mm_struct *mm_for_maps(struct task_struct *task)
task_lock(task); task_lock(task);
if (task->mm != mm) if (task->mm != mm)
goto out; goto out;
if (task->mm != current->mm && __ptrace_may_attach(task) < 0) if (task->mm != current->mm &&
__ptrace_may_access(task, PTRACE_MODE_READ) < 0)
goto out; goto out;
task_unlock(task); task_unlock(task);
return mm; return mm;
...@@ -518,7 +519,7 @@ static int proc_fd_access_allowed(struct inode *inode) ...@@ -518,7 +519,7 @@ static int proc_fd_access_allowed(struct inode *inode)
*/ */
task = get_proc_task(inode); task = get_proc_task(inode);
if (task) { if (task) {
allowed = ptrace_may_attach(task); allowed = ptrace_may_access(task, PTRACE_MODE_READ);
put_task_struct(task); put_task_struct(task);
} }
return allowed; return allowed;
...@@ -904,7 +905,7 @@ static ssize_t environ_read(struct file *file, char __user *buf, ...@@ -904,7 +905,7 @@ static ssize_t environ_read(struct file *file, char __user *buf,
if (!task) if (!task)
goto out_no_task; goto out_no_task;
if (!ptrace_may_attach(task)) if (!ptrace_may_access(task, PTRACE_MODE_READ))
goto out; goto out;
ret = -ENOMEM; ret = -ENOMEM;
......
...@@ -210,7 +210,7 @@ static int show_map(struct seq_file *m, void *v) ...@@ -210,7 +210,7 @@ static int show_map(struct seq_file *m, void *v)
dev_t dev = 0; dev_t dev = 0;
int len; int len;
if (maps_protect && !ptrace_may_attach(task)) if (maps_protect && !ptrace_may_access(task, PTRACE_MODE_READ))
return -EACCES; return -EACCES;
if (file) { if (file) {
...@@ -646,7 +646,7 @@ static ssize_t pagemap_read(struct file *file, char __user *buf, ...@@ -646,7 +646,7 @@ static ssize_t pagemap_read(struct file *file, char __user *buf,
goto out; goto out;
ret = -EACCES; ret = -EACCES;
if (!ptrace_may_attach(task)) if (!ptrace_may_access(task, PTRACE_MODE_READ))
goto out_task; goto out_task;
ret = -EINVAL; ret = -EINVAL;
...@@ -747,7 +747,7 @@ static int show_numa_map_checked(struct seq_file *m, void *v) ...@@ -747,7 +747,7 @@ static int show_numa_map_checked(struct seq_file *m, void *v)
struct proc_maps_private *priv = m->private; struct proc_maps_private *priv = m->private;
struct task_struct *task = priv->task; struct task_struct *task = priv->task;
if (maps_protect && !ptrace_may_attach(task)) if (maps_protect && !ptrace_may_access(task, PTRACE_MODE_READ))
return -EACCES; return -EACCES;
return show_numa_map(m, v); return show_numa_map(m, v);
......
...@@ -113,7 +113,7 @@ static int show_map(struct seq_file *m, void *_vml) ...@@ -113,7 +113,7 @@ static int show_map(struct seq_file *m, void *_vml)
struct proc_maps_private *priv = m->private; struct proc_maps_private *priv = m->private;
struct task_struct *task = priv->task; struct task_struct *task = priv->task;
if (maps_protect && !ptrace_may_attach(task)) if (maps_protect && !ptrace_may_access(task, PTRACE_MODE_READ))
return -EACCES; return -EACCES;
return nommu_vma_show(m, vml->vma); return nommu_vma_show(m, vml->vma);
......
...@@ -95,8 +95,12 @@ extern void __ptrace_link(struct task_struct *child, ...@@ -95,8 +95,12 @@ extern void __ptrace_link(struct task_struct *child,
struct task_struct *new_parent); struct task_struct *new_parent);
extern void __ptrace_unlink(struct task_struct *child); extern void __ptrace_unlink(struct task_struct *child);
extern void ptrace_untrace(struct task_struct *child); extern void ptrace_untrace(struct task_struct *child);
extern int ptrace_may_attach(struct task_struct *task); #define PTRACE_MODE_READ 1
extern int __ptrace_may_attach(struct task_struct *task); #define PTRACE_MODE_ATTACH 2
/* Returns 0 on success, -errno on denial. */
extern int __ptrace_may_access(struct task_struct *task, unsigned int mode);
/* Returns true on success, false on denial. */
extern bool ptrace_may_access(struct task_struct *task, unsigned int mode);
static inline int ptrace_reparented(struct task_struct *child) static inline int ptrace_reparented(struct task_struct *child)
{ {
......
...@@ -46,7 +46,8 @@ struct audit_krule; ...@@ -46,7 +46,8 @@ struct audit_krule;
*/ */
extern int cap_capable(struct task_struct *tsk, int cap); extern int cap_capable(struct task_struct *tsk, int cap);
extern int cap_settime(struct timespec *ts, struct timezone *tz); extern int cap_settime(struct timespec *ts, struct timezone *tz);
extern int cap_ptrace(struct task_struct *parent, struct task_struct *child); extern int cap_ptrace(struct task_struct *parent, struct task_struct *child,
unsigned int mode);
extern int cap_capget(struct task_struct *target, kernel_cap_t *effective, kernel_cap_t *inheritable, kernel_cap_t *permitted); extern int cap_capget(struct task_struct *target, kernel_cap_t *effective, kernel_cap_t *inheritable, kernel_cap_t *permitted);
extern int cap_capset_check(struct task_struct *target, kernel_cap_t *effective, kernel_cap_t *inheritable, kernel_cap_t *permitted); extern int cap_capset_check(struct task_struct *target, kernel_cap_t *effective, kernel_cap_t *inheritable, kernel_cap_t *permitted);
extern void cap_capset_set(struct task_struct *target, kernel_cap_t *effective, kernel_cap_t *inheritable, kernel_cap_t *permitted); extern void cap_capset_set(struct task_struct *target, kernel_cap_t *effective, kernel_cap_t *inheritable, kernel_cap_t *permitted);
...@@ -1170,6 +1171,7 @@ static inline void security_free_mnt_opts(struct security_mnt_opts *opts) ...@@ -1170,6 +1171,7 @@ static inline void security_free_mnt_opts(struct security_mnt_opts *opts)
* attributes would be changed by the execve. * attributes would be changed by the execve.
* @parent contains the task_struct structure for parent process. * @parent contains the task_struct structure for parent process.
* @child contains the task_struct structure for child process. * @child contains the task_struct structure for child process.
* @mode contains the PTRACE_MODE flags indicating the form of access.
* Return 0 if permission is granted. * Return 0 if permission is granted.
* @capget: * @capget:
* Get the @effective, @inheritable, and @permitted capability sets for * Get the @effective, @inheritable, and @permitted capability sets for
...@@ -1295,7 +1297,8 @@ static inline void security_free_mnt_opts(struct security_mnt_opts *opts) ...@@ -1295,7 +1297,8 @@ static inline void security_free_mnt_opts(struct security_mnt_opts *opts)
struct security_operations { struct security_operations {
char name[SECURITY_NAME_MAX + 1]; char name[SECURITY_NAME_MAX + 1];
int (*ptrace) (struct task_struct *parent, struct task_struct *child); int (*ptrace) (struct task_struct *parent, struct task_struct *child,
unsigned int mode);
int (*capget) (struct task_struct *target, int (*capget) (struct task_struct *target,
kernel_cap_t *effective, kernel_cap_t *effective,
kernel_cap_t *inheritable, kernel_cap_t *permitted); kernel_cap_t *inheritable, kernel_cap_t *permitted);
...@@ -1573,7 +1576,8 @@ extern struct dentry *securityfs_create_dir(const char *name, struct dentry *par ...@@ -1573,7 +1576,8 @@ extern struct dentry *securityfs_create_dir(const char *name, struct dentry *par
extern void securityfs_remove(struct dentry *dentry); extern void securityfs_remove(struct dentry *dentry);
/* Security operations */ /* Security operations */
int security_ptrace(struct task_struct *parent, struct task_struct *child); int security_ptrace(struct task_struct *parent, struct task_struct *child,
unsigned int mode);
int security_capget(struct task_struct *target, int security_capget(struct task_struct *target,
kernel_cap_t *effective, kernel_cap_t *effective,
kernel_cap_t *inheritable, kernel_cap_t *inheritable,
...@@ -1755,9 +1759,11 @@ static inline int security_init(void) ...@@ -1755,9 +1759,11 @@ static inline int security_init(void)
return 0; return 0;
} }
static inline int security_ptrace(struct task_struct *parent, struct task_struct *child) static inline int security_ptrace(struct task_struct *parent,
struct task_struct *child,
unsigned int mode)
{ {
return cap_ptrace(parent, child); return cap_ptrace(parent, child, mode);
} }
static inline int security_capget(struct task_struct *target, static inline int security_capget(struct task_struct *target,
......
...@@ -121,7 +121,7 @@ int ptrace_check_attach(struct task_struct *child, int kill) ...@@ -121,7 +121,7 @@ int ptrace_check_attach(struct task_struct *child, int kill)
return ret; return ret;
} }
int __ptrace_may_attach(struct task_struct *task) int __ptrace_may_access(struct task_struct *task, unsigned int mode)
{ {
/* May we inspect the given task? /* May we inspect the given task?
* This check is used both for attaching with ptrace * This check is used both for attaching with ptrace
...@@ -148,16 +148,16 @@ int __ptrace_may_attach(struct task_struct *task) ...@@ -148,16 +148,16 @@ int __ptrace_may_attach(struct task_struct *task)
if (!dumpable && !capable(CAP_SYS_PTRACE)) if (!dumpable && !capable(CAP_SYS_PTRACE))
return -EPERM; return -EPERM;
return security_ptrace(current, task); return security_ptrace(current, task, mode);
} }
int ptrace_may_attach(struct task_struct *task) bool ptrace_may_access(struct task_struct *task, unsigned int mode)
{ {
int err; int err;
task_lock(task); task_lock(task);
err = __ptrace_may_attach(task); err = __ptrace_may_access(task, mode);
task_unlock(task); task_unlock(task);
return !err; return (!err ? true : false);
} }
int ptrace_attach(struct task_struct *task) int ptrace_attach(struct task_struct *task)
...@@ -195,7 +195,7 @@ int ptrace_attach(struct task_struct *task) ...@@ -195,7 +195,7 @@ int ptrace_attach(struct task_struct *task)
/* the same process cannot be attached many times */ /* the same process cannot be attached many times */
if (task->ptrace & PT_PTRACED) if (task->ptrace & PT_PTRACED)
goto bad; goto bad;
retval = __ptrace_may_attach(task); retval = __ptrace_may_access(task, PTRACE_MODE_ATTACH);
if (retval) if (retval)
goto bad; goto bad;
...@@ -494,7 +494,8 @@ int ptrace_traceme(void) ...@@ -494,7 +494,8 @@ int ptrace_traceme(void)
*/ */
task_lock(current); task_lock(current);
if (!(current->ptrace & PT_PTRACED)) { if (!(current->ptrace & PT_PTRACED)) {
ret = security_ptrace(current->parent, current); ret = security_ptrace(current->parent, current,
PTRACE_MODE_ATTACH);
/* /*
* Set the ptrace bit in the process ptrace flags. * Set the ptrace bit in the process ptrace flags.
*/ */
......
...@@ -63,7 +63,8 @@ int cap_settime(struct timespec *ts, struct timezone *tz) ...@@ -63,7 +63,8 @@ int cap_settime(struct timespec *ts, struct timezone *tz)
return 0; return 0;
} }
int cap_ptrace (struct task_struct *parent, struct task_struct *child) int cap_ptrace (struct task_struct *parent, struct task_struct *child,
unsigned int mode)
{ {
/* Derived from arch/i386/kernel/ptrace.c:sys_ptrace. */ /* Derived from arch/i386/kernel/ptrace.c:sys_ptrace. */
if (!cap_issubset(child->cap_permitted, parent->cap_permitted) && if (!cap_issubset(child->cap_permitted, parent->cap_permitted) &&
......
...@@ -30,7 +30,8 @@ ...@@ -30,7 +30,8 @@
#include <linux/prctl.h> #include <linux/prctl.h>
#include <linux/securebits.h> #include <linux/securebits.h>
static int dummy_ptrace (struct task_struct *parent, struct task_struct *child) static int dummy_ptrace (struct task_struct *parent, struct task_struct *child,
unsigned int mode)
{ {
return 0; return 0;
} }
......
...@@ -161,9 +161,10 @@ int mod_reg_security(const char *name, struct security_operations *ops) ...@@ -161,9 +161,10 @@ int mod_reg_security(const char *name, struct security_operations *ops)
/* Security operations */ /* Security operations */
int security_ptrace(struct task_struct *parent, struct task_struct *child) int security_ptrace(struct task_struct *parent, struct task_struct *child,
unsigned int mode)
{ {
return security_ops->ptrace(parent, child); return security_ops->ptrace(parent, child, mode);
} }
int security_capget(struct task_struct *target, int security_capget(struct task_struct *target,
......
...@@ -1686,14 +1686,23 @@ static inline u32 file_to_av(struct file *file) ...@@ -1686,14 +1686,23 @@ static inline u32 file_to_av(struct file *file)
/* Hook functions begin here. */ /* Hook functions begin here. */
static int selinux_ptrace(struct task_struct *parent, struct task_struct *child) static int selinux_ptrace(struct task_struct *parent,
struct task_struct *child,
unsigned int mode)
{ {
int rc; int rc;
rc = secondary_ops->ptrace(parent, child); rc = secondary_ops->ptrace(parent, child, mode);
if (rc) if (rc)
return rc; return rc;
if (mode == PTRACE_MODE_READ) {
struct task_security_struct *tsec = parent->security;
struct task_security_struct *csec = child->security;
return avc_has_perm(tsec->sid, csec->sid,
SECCLASS_FILE, FILE__READ, NULL);
}
return task_has_perm(parent, child, PROCESS__PTRACE); return task_has_perm(parent, child, PROCESS__PTRACE);
} }
......
...@@ -95,11 +95,12 @@ struct inode_smack *new_inode_smack(char *smack) ...@@ -95,11 +95,12 @@ struct inode_smack *new_inode_smack(char *smack)
* *
* Do the capability checks, and require read and write. * Do the capability checks, and require read and write.
*/ */
static int smack_ptrace(struct task_struct *ptp, struct task_struct *ctp) static int smack_ptrace(struct task_struct *ptp, struct task_struct *ctp,
unsigned int mode)
{ {
int rc; int rc;
rc = cap_ptrace(ptp, ctp); rc = cap_ptrace(ptp, ctp, mode);
if (rc != 0) if (rc != 0)
return rc; return rc;
......
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