userns: Convert taskstats to handle the user and pid namespaces.
- Explicitly limit exit task stat broadcast to the initial user and pid namespaces, as it is already limited to the initial network namespace. - For broadcast task stats explicitly generate all of the idenitiers in terms of the initial user namespace and the initial pid namespace. - For request stats report them in terms of the current user namespace and the current pid namespace. Netlink messages are delivered syncrhonously to the kernel allowing us to get the user namespace and the pid namespace from the current task. - Pass the namespaces for representing pids and uids and gids into bacct_add_task. Cc: Balbir Singh <bsingharora@gmail.com> Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
This commit is contained in:
parent
cca080d9b6
commit
4bd6e32ace
@ -10,9 +10,13 @@
|
|||||||
#include <linux/taskstats.h>
|
#include <linux/taskstats.h>
|
||||||
|
|
||||||
#ifdef CONFIG_TASKSTATS
|
#ifdef CONFIG_TASKSTATS
|
||||||
extern void bacct_add_tsk(struct taskstats *stats, struct task_struct *tsk);
|
extern void bacct_add_tsk(struct user_namespace *user_ns,
|
||||||
|
struct pid_namespace *pid_ns,
|
||||||
|
struct taskstats *stats, struct task_struct *tsk);
|
||||||
#else
|
#else
|
||||||
static inline void bacct_add_tsk(struct taskstats *stats, struct task_struct *tsk)
|
static inline void bacct_add_tsk(struct user_namespace *user_ns,
|
||||||
|
struct pid_namespace *pid_ns,
|
||||||
|
struct taskstats *stats, struct task_struct *tsk)
|
||||||
{}
|
{}
|
||||||
#endif /* CONFIG_TASKSTATS */
|
#endif /* CONFIG_TASKSTATS */
|
||||||
|
|
||||||
|
@ -927,7 +927,6 @@ config UIDGID_CONVERTED
|
|||||||
# Features
|
# Features
|
||||||
depends on IMA = n
|
depends on IMA = n
|
||||||
depends on EVM = n
|
depends on EVM = n
|
||||||
depends on TASKSTATS = n
|
|
||||||
depends on TRACING = n
|
depends on TRACING = n
|
||||||
depends on FS_POSIX_ACL = n
|
depends on FS_POSIX_ACL = n
|
||||||
depends on QUOTA = n
|
depends on QUOTA = n
|
||||||
|
@ -27,6 +27,7 @@
|
|||||||
#include <linux/cgroup.h>
|
#include <linux/cgroup.h>
|
||||||
#include <linux/fs.h>
|
#include <linux/fs.h>
|
||||||
#include <linux/file.h>
|
#include <linux/file.h>
|
||||||
|
#include <linux/pid_namespace.h>
|
||||||
#include <net/genetlink.h>
|
#include <net/genetlink.h>
|
||||||
#include <linux/atomic.h>
|
#include <linux/atomic.h>
|
||||||
|
|
||||||
@ -174,7 +175,9 @@ static void send_cpu_listeners(struct sk_buff *skb,
|
|||||||
up_write(&listeners->sem);
|
up_write(&listeners->sem);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void fill_stats(struct task_struct *tsk, struct taskstats *stats)
|
static void fill_stats(struct user_namespace *user_ns,
|
||||||
|
struct pid_namespace *pid_ns,
|
||||||
|
struct task_struct *tsk, struct taskstats *stats)
|
||||||
{
|
{
|
||||||
memset(stats, 0, sizeof(*stats));
|
memset(stats, 0, sizeof(*stats));
|
||||||
/*
|
/*
|
||||||
@ -190,7 +193,7 @@ static void fill_stats(struct task_struct *tsk, struct taskstats *stats)
|
|||||||
stats->version = TASKSTATS_VERSION;
|
stats->version = TASKSTATS_VERSION;
|
||||||
stats->nvcsw = tsk->nvcsw;
|
stats->nvcsw = tsk->nvcsw;
|
||||||
stats->nivcsw = tsk->nivcsw;
|
stats->nivcsw = tsk->nivcsw;
|
||||||
bacct_add_tsk(stats, tsk);
|
bacct_add_tsk(user_ns, pid_ns, stats, tsk);
|
||||||
|
|
||||||
/* fill in extended acct fields */
|
/* fill in extended acct fields */
|
||||||
xacct_add_tsk(stats, tsk);
|
xacct_add_tsk(stats, tsk);
|
||||||
@ -207,7 +210,7 @@ static int fill_stats_for_pid(pid_t pid, struct taskstats *stats)
|
|||||||
rcu_read_unlock();
|
rcu_read_unlock();
|
||||||
if (!tsk)
|
if (!tsk)
|
||||||
return -ESRCH;
|
return -ESRCH;
|
||||||
fill_stats(tsk, stats);
|
fill_stats(current_user_ns(), task_active_pid_ns(current), tsk, stats);
|
||||||
put_task_struct(tsk);
|
put_task_struct(tsk);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -291,6 +294,12 @@ static int add_del_listener(pid_t pid, const struct cpumask *mask, int isadd)
|
|||||||
if (!cpumask_subset(mask, cpu_possible_mask))
|
if (!cpumask_subset(mask, cpu_possible_mask))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (current_user_ns() != &init_user_ns)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (task_active_pid_ns(current) != &init_pid_ns)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
if (isadd == REGISTER) {
|
if (isadd == REGISTER) {
|
||||||
for_each_cpu(cpu, mask) {
|
for_each_cpu(cpu, mask) {
|
||||||
s = kmalloc_node(sizeof(struct listener),
|
s = kmalloc_node(sizeof(struct listener),
|
||||||
@ -631,11 +640,12 @@ void taskstats_exit(struct task_struct *tsk, int group_dead)
|
|||||||
if (rc < 0)
|
if (rc < 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
stats = mk_reply(rep_skb, TASKSTATS_TYPE_PID, tsk->pid);
|
stats = mk_reply(rep_skb, TASKSTATS_TYPE_PID,
|
||||||
|
task_pid_nr_ns(tsk, &init_pid_ns));
|
||||||
if (!stats)
|
if (!stats)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
fill_stats(tsk, stats);
|
fill_stats(&init_user_ns, &init_pid_ns, tsk, stats);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Doesn't matter if tsk is the leader or the last group member leaving
|
* Doesn't matter if tsk is the leader or the last group member leaving
|
||||||
@ -643,7 +653,8 @@ void taskstats_exit(struct task_struct *tsk, int group_dead)
|
|||||||
if (!is_thread_group || !group_dead)
|
if (!is_thread_group || !group_dead)
|
||||||
goto send;
|
goto send;
|
||||||
|
|
||||||
stats = mk_reply(rep_skb, TASKSTATS_TYPE_TGID, tsk->tgid);
|
stats = mk_reply(rep_skb, TASKSTATS_TYPE_TGID,
|
||||||
|
task_tgid_nr_ns(tsk, &init_pid_ns));
|
||||||
if (!stats)
|
if (!stats)
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -26,7 +26,9 @@
|
|||||||
/*
|
/*
|
||||||
* fill in basic accounting fields
|
* fill in basic accounting fields
|
||||||
*/
|
*/
|
||||||
void bacct_add_tsk(struct taskstats *stats, struct task_struct *tsk)
|
void bacct_add_tsk(struct user_namespace *user_ns,
|
||||||
|
struct pid_namespace *pid_ns,
|
||||||
|
struct taskstats *stats, struct task_struct *tsk)
|
||||||
{
|
{
|
||||||
const struct cred *tcred;
|
const struct cred *tcred;
|
||||||
struct timespec uptime, ts;
|
struct timespec uptime, ts;
|
||||||
@ -55,13 +57,13 @@ void bacct_add_tsk(struct taskstats *stats, struct task_struct *tsk)
|
|||||||
stats->ac_flag |= AXSIG;
|
stats->ac_flag |= AXSIG;
|
||||||
stats->ac_nice = task_nice(tsk);
|
stats->ac_nice = task_nice(tsk);
|
||||||
stats->ac_sched = tsk->policy;
|
stats->ac_sched = tsk->policy;
|
||||||
stats->ac_pid = tsk->pid;
|
stats->ac_pid = task_pid_nr_ns(tsk, pid_ns);
|
||||||
rcu_read_lock();
|
rcu_read_lock();
|
||||||
tcred = __task_cred(tsk);
|
tcred = __task_cred(tsk);
|
||||||
stats->ac_uid = tcred->uid;
|
stats->ac_uid = from_kuid_munged(user_ns, tcred->uid);
|
||||||
stats->ac_gid = tcred->gid;
|
stats->ac_gid = from_kgid_munged(user_ns, tcred->gid);
|
||||||
stats->ac_ppid = pid_alive(tsk) ?
|
stats->ac_ppid = pid_alive(tsk) ?
|
||||||
rcu_dereference(tsk->real_parent)->tgid : 0;
|
task_tgid_nr_ns(rcu_dereference(tsk->real_parent), pid_ns) : 0;
|
||||||
rcu_read_unlock();
|
rcu_read_unlock();
|
||||||
stats->ac_utime = cputime_to_usecs(tsk->utime);
|
stats->ac_utime = cputime_to_usecs(tsk->utime);
|
||||||
stats->ac_stime = cputime_to_usecs(tsk->stime);
|
stats->ac_stime = cputime_to_usecs(tsk->stime);
|
||||||
|
Loading…
Reference in New Issue
Block a user