* Re: [RFC] event about group change
[not found] ` <4F6C1201.9030300-Sze3O3UU22JBDgjK7y7TUQ@public.gmane.org>
@ 2012-03-23 6:04 ` Alexander Nikiforov
0 siblings, 0 replies; 3+ messages in thread
From: Alexander Nikiforov @ 2012-03-23 6:04 UTC (permalink / raw)
To: cgroups-u79uwXL29TY76Z2rM5mHXA
Cc: a.nikiforov-Sze3O3UU22JBDgjK7y7TUQ, Kyungmin Park,
Dmitry Solodkiy
[-- Attachment #1: Type: text/plain, Size: 1 bytes --]
[-- Attachment #2: fork-exit.patch --]
[-- Type: text/x-patch, Size: 3645 bytes --]
diff --git a/include/linux/cgroup.h b/include/linux/cgroup.h
index e9b6021..86b0031 100644
--- a/include/linux/cgroup.h
+++ b/include/linux/cgroup.h
@@ -192,6 +192,11 @@ struct cgroup_pidlist {
struct rw_semaphore mutex;
};
+struct fe_eventfd_list {
+ struct list_head list;
+ struct eventfd_ctx *eventfd;
+};
+
struct cgroup {
unsigned long flags; /* "unsigned long" so bitops work */
@@ -243,6 +248,10 @@ struct cgroup {
/* List of events which userspace want to receive */
struct list_head event_list;
spinlock_t event_list_lock;
+
+ /* fork-exit event */
+ struct list_head fe_notify;
+ spinlock_t fe_list_lock;
};
/*
diff --git a/kernel/cgroup.c b/kernel/cgroup.c
index a5d3b53..6c5dda9 100644
--- a/kernel/cgroup.c
+++ b/kernel/cgroup.c
@@ -1336,6 +1336,8 @@ static void init_cgroup_housekeeping(struct cgroup *cgrp)
INIT_LIST_HEAD(&cgrp->css_sets);
INIT_LIST_HEAD(&cgrp->release_list);
INIT_LIST_HEAD(&cgrp->pidlists);
+ INIT_LIST_HEAD(&cgrp->fe_notify);
+ spin_lock_init(&cgrp->fe_list_lock);
mutex_init(&cgrp->pidlist_mutex);
INIT_LIST_HEAD(&cgrp->event_list);
spin_lock_init(&cgrp->event_list_lock);
@@ -3659,6 +3661,38 @@ static int cgroup_clone_children_write(struct cgroup *cgrp,
return 0;
}
+static int tasks_register_event(struct cgroup *cgrp,
+ struct cftype *cft, struct eventfd_ctx *eventfd, const char *args)
+{
+ struct fe_eventfd_list *ev;
+
+ ev = kmalloc(sizeof(*ev), GFP_KERNEL);
+ if(!ev)
+ return -ENOMEM;
+
+ spin_lock(&cgrp->fe_list_lock);
+ ev->eventfd = eventfd;
+ list_add(&ev->list, &cgrp->fe_notify);
+ spin_unlock(&cgrp->fe_list_lock);
+
+ return 0;
+}
+
+static void tasks_unregister_event(struct cgroup *cgrp,
+ struct cftype *cft, struct eventfd_ctx *eventfd)
+{
+ struct fe_eventfd_list *ev, *tmp;
+
+ spin_lock(&cgrp->fe_list_lock);
+ list_for_each_entry_safe(ev, tmp, &cgrp->fe_notify, list) {
+ if (ev->eventfd == eventfd) {
+ list_del(&ev->list);
+ kfree(ev);
+ }
+ }
+ spin_unlock(&cgrp->fe_list_lock);
+}
+
/*
* for the common functions, 'private' gives the type of file
*/
@@ -3670,6 +3704,8 @@ static struct cftype files[] = {
.open = cgroup_tasks_open,
.write_u64 = cgroup_tasks_write,
.release = cgroup_pidlist_release,
+ .register_event = tasks_register_event,
+ .unregister_event = tasks_unregister_event,
.mode = S_IRUGO | S_IWUSR,
},
{
@@ -4558,6 +4594,22 @@ void cgroup_fork(struct task_struct *child)
child->cgroups = current->cgroups;
get_css_set(child->cgroups);
INIT_LIST_HEAD(&child->cg_list);
+
+ struct cgroupfs_root *root;
+
+ /* send event to the userspace */
+ mutex_lock(&cgroup_mutex);
+ for_each_active_root(root) {
+ struct cgroup *cgrp;
+ struct fe_eventfd_list *ev;
+
+ cgrp = task_cgroup_from_root(child, root);
+
+ list_for_each_entry(ev, &cgrp->fe_notify, list) {
+ eventfd_signal(ev->eventfd, 1);
+ }
+ }
+ mutex_unlock(&cgroup_mutex);
}
/**
@@ -4653,6 +4705,7 @@ void cgroup_exit(struct task_struct *tsk, int run_callbacks)
{
struct css_set *cg;
int i;
+ struct cgroupfs_root *root;
/*
* Unlink from the css_set task list if necessary.
@@ -4666,6 +4719,20 @@ void cgroup_exit(struct task_struct *tsk, int run_callbacks)
write_unlock(&css_set_lock);
}
+ /* send event to the userspace */
+ mutex_lock(&cgroup_mutex);
+ for_each_active_root(root) {
+ struct cgroup *cgrp;
+ struct fe_eventfd_list *ev;
+
+ cgrp = task_cgroup_from_root(tsk, root);
+
+ list_for_each_entry(ev, &cgrp->fe_notify, list) {
+ eventfd_signal(ev->eventfd, 1);
+ }
+ }
+ mutex_unlock(&cgroup_mutex);
+
/* Reassign the task to the init_css_set. */
task_lock(tsk);
cg = tsk->cgroups;
^ permalink raw reply related [flat|nested] 3+ messages in thread