提交 c4b92fc1 编写于 作者: E Eric W. Biederman 提交者: Linus Torvalds

[PATCH] pid: implement signal functions that take a struct pid *

Currently the signal functions all either take a task or a pid_t argument.
This patch implements variants that take a struct pid *.  After all of the
users have been update it is my intention to remove the variants that take a
pid_t as using pid_t can be more work (an extra hash table lookup) and
difficult to get right in the presence of multiple pid namespaces.

There are two kinds of functions introduced in this patch.  The are the
general use functions kill_pgrp and kill_pid which take a priv argument that
is ultimately used to create the appropriate siginfo information, Then there
are _kill_pgrp_info, kill_pgrp_info, kill_pid_info the internal implementation
helpers that take an explicit siginfo.

The distinction is made because filling out an explcit siginfo is tricky, and
will be even more tricky when pid namespaces are introduced.
Signed-off-by: NEric W. Biederman <ebiederm@xmission.com>
Signed-off-by: NAndrew Morton <akpm@osdl.org>
Signed-off-by: NLinus Torvalds <torvalds@osdl.org>
上级 558cb325
...@@ -1267,6 +1267,11 @@ extern int send_sig_info(int, struct siginfo *, struct task_struct *); ...@@ -1267,6 +1267,11 @@ extern int send_sig_info(int, struct siginfo *, struct task_struct *);
extern int send_group_sig_info(int, struct siginfo *, struct task_struct *); extern int send_group_sig_info(int, struct siginfo *, struct task_struct *);
extern int force_sigsegv(int, struct task_struct *); extern int force_sigsegv(int, struct task_struct *);
extern int force_sig_info(int, struct siginfo *, struct task_struct *); extern int force_sig_info(int, struct siginfo *, struct task_struct *);
extern int __kill_pgrp_info(int sig, struct siginfo *info, struct pid *pgrp);
extern int kill_pgrp_info(int sig, struct siginfo *info, struct pid *pgrp);
extern int kill_pid_info(int sig, struct siginfo *info, struct pid *pid);
extern int kill_pgrp(struct pid *pid, int sig, int priv);
extern int kill_pid(struct pid *pid, int sig, int priv);
extern int __kill_pg_info(int sig, struct siginfo *info, pid_t pgrp); extern int __kill_pg_info(int sig, struct siginfo *info, pid_t pgrp);
extern int kill_pg_info(int, struct siginfo *, pid_t); extern int kill_pg_info(int, struct siginfo *, pid_t);
extern int kill_proc_info(int, struct siginfo *, pid_t); extern int kill_proc_info(int, struct siginfo *, pid_t);
......
...@@ -1055,28 +1055,44 @@ int group_send_sig_info(int sig, struct siginfo *info, struct task_struct *p) ...@@ -1055,28 +1055,44 @@ int group_send_sig_info(int sig, struct siginfo *info, struct task_struct *p)
} }
/* /*
* kill_pg_info() sends a signal to a process group: this is what the tty * kill_pgrp_info() sends a signal to a process group: this is what the tty
* control characters do (^C, ^Z etc) * control characters do (^C, ^Z etc)
*/ */
int __kill_pg_info(int sig, struct siginfo *info, pid_t pgrp) int __kill_pgrp_info(int sig, struct siginfo *info, struct pid *pgrp)
{ {
struct task_struct *p = NULL; struct task_struct *p = NULL;
int retval, success; int retval, success;
if (pgrp <= 0)
return -EINVAL;
success = 0; success = 0;
retval = -ESRCH; retval = -ESRCH;
do_each_task_pid(pgrp, PIDTYPE_PGID, p) { do_each_pid_task(pgrp, PIDTYPE_PGID, p) {
int err = group_send_sig_info(sig, info, p); int err = group_send_sig_info(sig, info, p);
success |= !err; success |= !err;
retval = err; retval = err;
} while_each_task_pid(pgrp, PIDTYPE_PGID, p); } while_each_pid_task(pgrp, PIDTYPE_PGID, p);
return success ? 0 : retval; return success ? 0 : retval;
} }
int kill_pgrp_info(int sig, struct siginfo *info, struct pid *pgrp)
{
int retval;
read_lock(&tasklist_lock);
retval = __kill_pgrp_info(sig, info, pgrp);
read_unlock(&tasklist_lock);
return retval;
}
int __kill_pg_info(int sig, struct siginfo *info, pid_t pgrp)
{
if (pgrp <= 0)
return -EINVAL;
return __kill_pgrp_info(sig, info, find_pid(pgrp));
}
int int
kill_pg_info(int sig, struct siginfo *info, pid_t pgrp) kill_pg_info(int sig, struct siginfo *info, pid_t pgrp)
{ {
...@@ -1089,8 +1105,7 @@ kill_pg_info(int sig, struct siginfo *info, pid_t pgrp) ...@@ -1089,8 +1105,7 @@ kill_pg_info(int sig, struct siginfo *info, pid_t pgrp)
return retval; return retval;
} }
int int kill_pid_info(int sig, struct siginfo *info, struct pid *pid)
kill_proc_info(int sig, struct siginfo *info, pid_t pid)
{ {
int error; int error;
int acquired_tasklist_lock = 0; int acquired_tasklist_lock = 0;
...@@ -1101,7 +1116,7 @@ kill_proc_info(int sig, struct siginfo *info, pid_t pid) ...@@ -1101,7 +1116,7 @@ kill_proc_info(int sig, struct siginfo *info, pid_t pid)
read_lock(&tasklist_lock); read_lock(&tasklist_lock);
acquired_tasklist_lock = 1; acquired_tasklist_lock = 1;
} }
p = find_task_by_pid(pid); p = pid_task(pid, PIDTYPE_PID);
error = -ESRCH; error = -ESRCH;
if (p) if (p)
error = group_send_sig_info(sig, info, p); error = group_send_sig_info(sig, info, p);
...@@ -1111,6 +1126,16 @@ kill_proc_info(int sig, struct siginfo *info, pid_t pid) ...@@ -1111,6 +1126,16 @@ kill_proc_info(int sig, struct siginfo *info, pid_t pid)
return error; return error;
} }
int
kill_proc_info(int sig, struct siginfo *info, pid_t pid)
{
int error;
rcu_read_lock();
error = kill_pid_info(sig, info, find_pid(pid));
rcu_read_unlock();
return error;
}
/* like kill_proc_info(), but doesn't use uid/euid of "current" */ /* like kill_proc_info(), but doesn't use uid/euid of "current" */
int kill_proc_info_as_uid(int sig, struct siginfo *info, pid_t pid, int kill_proc_info_as_uid(int sig, struct siginfo *info, pid_t pid,
uid_t uid, uid_t euid, u32 secid) uid_t uid, uid_t euid, u32 secid)
...@@ -1264,6 +1289,18 @@ force_sigsegv(int sig, struct task_struct *p) ...@@ -1264,6 +1289,18 @@ force_sigsegv(int sig, struct task_struct *p)
return 0; return 0;
} }
int kill_pgrp(struct pid *pid, int sig, int priv)
{
return kill_pgrp_info(sig, __si_special(priv), pid);
}
EXPORT_SYMBOL(kill_pgrp);
int kill_pid(struct pid *pid, int sig, int priv)
{
return kill_pid_info(sig, __si_special(priv), pid);
}
EXPORT_SYMBOL(kill_pid);
int int
kill_pg(pid_t pgrp, int sig, int priv) kill_pg(pid_t pgrp, int sig, int priv)
{ {
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册