Commit 5ed67f05 authored by Pavel Emelyanov's avatar Pavel Emelyanov Committed by Thomas Gleixner

posix timers: Allocate timer id per process (v2)

Currently kernel generates IDs for posix timers in a global manner --
there's a kernel-wide IDR tree from which IDs are created. This makes
it impossible to recreate a timer with a desired ID (in particular
this is done by the CRIU checkpoint-restore project) -- since these
IDs are global it may happen, that at the time we recreate a timer, the
ID we want for it is already busy by some other timer.

In order to address this, replace the IDR tree with a global hash
table for timers and makes timer IDs unique per signal_struct (to
which timers are linked anyway). With this, two timers belonging to
different processes may have equal IDs and we can recreate either of
them with the ID we want.
Signed-off-by: default avatarPavel Emelyanov <xemul@parallels.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Michael Kerrisk <mtk.manpages@gmail.com>
Cc: Matthew Helsley <matt.helsley@gmail.com>
Link: http://lkml.kernel.org/r/513D9FF5.9010004@parallels.comSigned-off-by: default avatarThomas Gleixner <tglx@linutronix.de>
parent 4e8f8b34
...@@ -55,6 +55,7 @@ struct cpu_timer_list { ...@@ -55,6 +55,7 @@ struct cpu_timer_list {
/* POSIX.1b interval timer structure. */ /* POSIX.1b interval timer structure. */
struct k_itimer { struct k_itimer {
struct list_head list; /* free/ allocate list */ struct list_head list; /* free/ allocate list */
struct hlist_node t_hash;
spinlock_t it_lock; spinlock_t it_lock;
clockid_t it_clock; /* which timer type */ clockid_t it_clock; /* which timer type */
timer_t it_id; /* timer id */ timer_t it_id; /* timer id */
......
...@@ -526,6 +526,7 @@ struct signal_struct { ...@@ -526,6 +526,7 @@ struct signal_struct {
unsigned int has_child_subreaper:1; unsigned int has_child_subreaper:1;
/* POSIX.1b Interval Timers */ /* POSIX.1b Interval Timers */
int posix_timer_id;
struct list_head posix_timers; struct list_head posix_timers;
/* ITIMER_REAL timer for the process */ /* ITIMER_REAL timer for the process */
......
...@@ -40,38 +40,31 @@ ...@@ -40,38 +40,31 @@
#include <linux/list.h> #include <linux/list.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/compiler.h> #include <linux/compiler.h>
#include <linux/idr.h> #include <linux/hash.h>
#include <linux/posix-clock.h> #include <linux/posix-clock.h>
#include <linux/posix-timers.h> #include <linux/posix-timers.h>
#include <linux/syscalls.h> #include <linux/syscalls.h>
#include <linux/wait.h> #include <linux/wait.h>
#include <linux/workqueue.h> #include <linux/workqueue.h>
#include <linux/export.h> #include <linux/export.h>
#include <linux/hashtable.h>
/* /*
* Management arrays for POSIX timers. Timers are kept in slab memory * Management arrays for POSIX timers. Timers are now kept in static hash table
* Timer ids are allocated by an external routine that keeps track of the * with 512 entries.
* id and the timer. The external interface is: * Timer ids are allocated by local routine, which selects proper hash head by
* * key, constructed from current->signal address and per signal struct counter.
* void *idr_find(struct idr *idp, int id); to find timer_id <id> * This keeps timer ids unique per process, but now they can intersect between
* int idr_get_new(struct idr *idp, void *ptr); to get a new id and * processes.
* related it to <ptr>
* void idr_remove(struct idr *idp, int id); to release <id>
* void idr_init(struct idr *idp); to initialize <idp>
* which we supply.
* The idr_get_new *may* call slab for more memory so it must not be
* called under a spin lock. Likewise idr_remore may release memory
* (but it may be ok to do this under a lock...).
* idr_find is just a memory look up and is quite fast. A -1 return
* indicates that the requested id does not exist.
*/ */
/* /*
* Lets keep our timers in a slab cache :-) * Lets keep our timers in a slab cache :-)
*/ */
static struct kmem_cache *posix_timers_cache; static struct kmem_cache *posix_timers_cache;
static struct idr posix_timers_id;
static DEFINE_SPINLOCK(idr_lock); static DEFINE_HASHTABLE(posix_timers_hashtable, 9);
static DEFINE_SPINLOCK(hash_lock);
/* /*
* we assume that the new SIGEV_THREAD_ID shares no bits with the other * we assume that the new SIGEV_THREAD_ID shares no bits with the other
...@@ -152,6 +145,57 @@ static struct k_itimer *__lock_timer(timer_t timer_id, unsigned long *flags); ...@@ -152,6 +145,57 @@ static struct k_itimer *__lock_timer(timer_t timer_id, unsigned long *flags);
__timr; \ __timr; \
}) })
static int hash(struct signal_struct *sig, unsigned int nr)
{
return hash_32(hash32_ptr(sig) ^ nr, HASH_BITS(posix_timers_hashtable));
}
static struct k_itimer *__posix_timers_find(struct hlist_head *head,
struct signal_struct *sig,
timer_t id)
{
struct hlist_node *node;
struct k_itimer *timer;
hlist_for_each_entry_rcu(timer, head, t_hash) {
if ((timer->it_signal == sig) && (timer->it_id == id))
return timer;
}
return NULL;
}
static struct k_itimer *posix_timer_by_id(timer_t id)
{
struct signal_struct *sig = current->signal;
struct hlist_head *head = &posix_timers_hashtable[hash(sig, id)];
return __posix_timers_find(head, sig, id);
}
static int posix_timer_add(struct k_itimer *timer)
{
struct signal_struct *sig = current->signal;
int first_free_id = sig->posix_timer_id;
struct hlist_head *head;
int ret = -ENOENT;
do {
spin_lock(&hash_lock);
head = &posix_timers_hashtable[hash(sig, sig->posix_timer_id)];
if (!__posix_timers_find(head, sig, sig->posix_timer_id)) {
hlist_add_head_rcu(&timer->t_hash, head);
ret = sig->posix_timer_id;
}
if (++sig->posix_timer_id < 0)
sig->posix_timer_id = 0;
if ((sig->posix_timer_id == first_free_id) && (ret == -ENOENT))
/* Loop over all possible ids completed */
ret = -EAGAIN;
spin_unlock(&hash_lock);
} while (ret == -ENOENT);
return ret;
}
static inline void unlock_timer(struct k_itimer *timr, unsigned long flags) static inline void unlock_timer(struct k_itimer *timr, unsigned long flags)
{ {
spin_unlock_irqrestore(&timr->it_lock, flags); spin_unlock_irqrestore(&timr->it_lock, flags);
...@@ -298,7 +342,6 @@ static __init int init_posix_timers(void) ...@@ -298,7 +342,6 @@ static __init int init_posix_timers(void)
posix_timers_cache = kmem_cache_create("posix_timers_cache", posix_timers_cache = kmem_cache_create("posix_timers_cache",
sizeof (struct k_itimer), 0, SLAB_PANIC, sizeof (struct k_itimer), 0, SLAB_PANIC,
NULL); NULL);
idr_init(&posix_timers_id);
return 0; return 0;
} }
...@@ -520,9 +563,9 @@ static void release_posix_timer(struct k_itimer *tmr, int it_id_set) ...@@ -520,9 +563,9 @@ static void release_posix_timer(struct k_itimer *tmr, int it_id_set)
{ {
if (it_id_set) { if (it_id_set) {
unsigned long flags; unsigned long flags;
spin_lock_irqsave(&idr_lock, flags); spin_lock_irqsave(&hash_lock, flags);
idr_remove(&posix_timers_id, tmr->it_id); hlist_del_rcu(&tmr->t_hash);
spin_unlock_irqrestore(&idr_lock, flags); spin_unlock_irqrestore(&hash_lock, flags);
} }
put_pid(tmr->it_pid); put_pid(tmr->it_pid);
sigqueue_free(tmr->sigq); sigqueue_free(tmr->sigq);
...@@ -568,22 +611,11 @@ SYSCALL_DEFINE3(timer_create, const clockid_t, which_clock, ...@@ -568,22 +611,11 @@ SYSCALL_DEFINE3(timer_create, const clockid_t, which_clock,
return -EAGAIN; return -EAGAIN;
spin_lock_init(&new_timer->it_lock); spin_lock_init(&new_timer->it_lock);
new_timer_id = posix_timer_add(new_timer);
idr_preload(GFP_KERNEL); if (new_timer_id < 0) {
spin_lock_irq(&idr_lock); error = new_timer_id;
error = idr_alloc(&posix_timers_id, new_timer, 0, 0, GFP_NOWAIT);
spin_unlock_irq(&idr_lock);
idr_preload_end();
if (error < 0) {
/*
* Weird looking, but we return EAGAIN if the IDR is
* full (proper POSIX return value for this)
*/
if (error == -ENOSPC)
error = -EAGAIN;
goto out; goto out;
} }
new_timer_id = error;
it_id_set = IT_ID_SET; it_id_set = IT_ID_SET;
new_timer->it_id = (timer_t) new_timer_id; new_timer->it_id = (timer_t) new_timer_id;
...@@ -661,7 +693,7 @@ static struct k_itimer *__lock_timer(timer_t timer_id, unsigned long *flags) ...@@ -661,7 +693,7 @@ static struct k_itimer *__lock_timer(timer_t timer_id, unsigned long *flags)
return NULL; return NULL;
rcu_read_lock(); rcu_read_lock();
timr = idr_find(&posix_timers_id, (int)timer_id); timr = posix_timer_by_id(timer_id);
if (timr) { if (timr) {
spin_lock_irqsave(&timr->it_lock, *flags); spin_lock_irqsave(&timr->it_lock, *flags);
if (timr->it_signal == current->signal) { if (timr->it_signal == current->signal) {
......
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