Commit a3bbfb37 authored by Christoph Lameter's avatar Christoph Lameter Committed by Linus Torvalds

[PATCH] Posix layer <-> clock driver API fix

This is needed for an mmtimer driver update that we are currently working
on.  The mmtimer driver provides CLOCK_SGI_CYCLE via clock_gettime and
clock_settime.

With this api fix one will be able to use timer_create, timer_settime and
friends from userspace to schedule and receive signals via timer interrupts
of mmtimer.

Changelog
	* Clean up timer api for drivers that use register_posix_clock. Drivers
	  will then be able to use posix timers to schedule interrupts.
	* Change API for posix_clocks[].timer_create to only pass one pointer
	  to a k_itimer structure that is now allocated and managed by the
	  posix layer in the same way as for the other posix timer
	  functions.
	* Isolate a posix_timer_event(timr) function in posix-timers.c that may
	  be called by the interrupt routine of a timer to signal that the
	  scheduled event has taken place.
Signed-off-by: default avatarChristoph Lameter <clameter@sgi.com>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 3a9e3908
...@@ -33,8 +33,7 @@ struct k_clock { ...@@ -33,8 +33,7 @@ struct k_clock {
struct k_clock_abs *abs_struct; struct k_clock_abs *abs_struct;
int (*clock_set) (struct timespec * tp); int (*clock_set) (struct timespec * tp);
int (*clock_get) (struct timespec * tp); int (*clock_get) (struct timespec * tp);
int (*timer_create) (int which_clock, struct sigevent __user *timer_event_spec, int (*timer_create) (struct k_itimer *timer);
timer_t __user * created_timer_id);
int (*nsleep) (int which_clock, int flags, int (*nsleep) (int which_clock, int flags,
struct timespec * t); struct timespec * t);
int (*timer_set) (struct k_itimer * timr, int flags, int (*timer_set) (struct k_itimer * timr, int flags,
...@@ -48,13 +47,13 @@ struct k_clock { ...@@ -48,13 +47,13 @@ struct k_clock {
void register_posix_clock(int clock_id, struct k_clock *new_clock); void register_posix_clock(int clock_id, struct k_clock *new_clock);
/* Error handlers for timer_create, nanosleep and settime */ /* Error handlers for timer_create, nanosleep and settime */
int do_posix_clock_notimer_create(int which_clock, int do_posix_clock_notimer_create(struct k_itimer *timer);
struct sigevent __user *time_event_spec,
timer_t __user *created_timer_id);
int do_posix_clock_nonanosleep(int which_clock, int flags, struct timespec * t); int do_posix_clock_nonanosleep(int which_clock, int flags, struct timespec * t);
int do_posix_clock_nosettime(struct timespec *tp); int do_posix_clock_nosettime(struct timespec *tp);
/* function to call to trigger timer event */
int posix_timer_event(struct k_itimer *timr, int si_private);
struct now_struct { struct now_struct {
unsigned long jiffies; unsigned long jiffies;
}; };
......
...@@ -384,32 +384,10 @@ void do_schedule_next_timer(struct siginfo *info) ...@@ -384,32 +384,10 @@ void do_schedule_next_timer(struct siginfo *info)
unlock_timer(timr, flags); unlock_timer(timr, flags);
} }
/* int posix_timer_event(struct k_itimer *timr,int si_private)
* Notify the task and set up the timer for the next expiration (if
* applicable). This function requires that the k_itimer structure
* it_lock is taken. This code will requeue the timer only if we get
* either an error return or a flag (ret > 0) from send_seg_info
* indicating that the signal was either not queued or was queued
* without an info block. In this case, we will not get a call back to
* do_schedule_next_timer() so we do it here. This should be rare...
* An interesting problem can occur if, while a signal, and thus a call
* back is pending, the timer is rearmed, i.e. stopped and restarted.
* We then need to sort out the call back and do the right thing. What
* we do is to put a counter in the info block and match it with the
* timers copy on the call back. If they don't match, we just ignore
* the call back. The counter is local to the timer and we use odd to
* indicate a call back is pending. Note that we do allow the timer to
* be deleted while a signal is pending. The standard says we can
* allow that signal to be delivered, and we do.
*/
static void timer_notify_task(struct k_itimer *timr)
{ {
int ret;
memset(&timr->sigq->info, 0, sizeof(siginfo_t)); memset(&timr->sigq->info, 0, sizeof(siginfo_t));
timr->sigq->info.si_sys_private = si_private;
/* /*
* Send signal to the process that owns this timer. * Send signal to the process that owns this timer.
...@@ -424,12 +402,6 @@ static void timer_notify_task(struct k_itimer *timr) ...@@ -424,12 +402,6 @@ static void timer_notify_task(struct k_itimer *timr)
timr->sigq->info.si_code = SI_TIMER; timr->sigq->info.si_code = SI_TIMER;
timr->sigq->info.si_tid = timr->it_id; timr->sigq->info.si_tid = timr->it_id;
timr->sigq->info.si_value = timr->it_sigev_value; timr->sigq->info.si_value = timr->it_sigev_value;
if (timr->it_incr)
timr->sigq->info.si_sys_private = ++timr->it_requeue_pending;
else {
remove_from_abslist(timr);
}
if (timr->it_sigev_notify & SIGEV_THREAD_ID) { if (timr->it_sigev_notify & SIGEV_THREAD_ID) {
if (unlikely(timr->it_process->flags & PF_EXITING)) { if (unlikely(timr->it_process->flags & PF_EXITING)) {
timr->it_sigev_notify = SIGEV_SIGNAL; timr->it_sigev_notify = SIGEV_SIGNAL;
...@@ -437,28 +409,20 @@ static void timer_notify_task(struct k_itimer *timr) ...@@ -437,28 +409,20 @@ static void timer_notify_task(struct k_itimer *timr)
timr->it_process = timr->it_process->group_leader; timr->it_process = timr->it_process->group_leader;
goto group; goto group;
} }
ret = send_sigqueue(timr->it_sigev_signo, timr->sigq, return send_sigqueue(timr->it_sigev_signo, timr->sigq,
timr->it_process); timr->it_process);
} }
else { else {
group: group:
ret = send_group_sigqueue(timr->it_sigev_signo, timr->sigq, return send_group_sigqueue(timr->it_sigev_signo, timr->sigq,
timr->it_process); timr->it_process);
} }
if (ret) {
/*
* signal was not sent because of sig_ignor
* we will not get a call back to restart it AND
* it should be restarted.
*/
schedule_next_timer(timr);
}
} }
/* /*
* This function gets called when a POSIX.1b interval timer expires. It * This function gets called when a POSIX.1b interval timer expires. It
* is used as a callback from the kernel internal timer. The * is used as a callback from the kernel internal timer. The
* run_timer_list code ALWAYS calls with interrutps on. * run_timer_list code ALWAYS calls with interrupts on.
* This code is for CLOCK_REALTIME* and CLOCK_MONOTONIC* timers. * This code is for CLOCK_REALTIME* and CLOCK_MONOTONIC* timers.
*/ */
...@@ -501,8 +465,23 @@ static void posix_timer_fn(unsigned long __data) ...@@ -501,8 +465,23 @@ static void posix_timer_fn(unsigned long __data)
spin_unlock(&abs_list.lock); spin_unlock(&abs_list.lock);
} }
if (do_notify) if (do_notify) {
timer_notify_task(timr); int si_private=0;
if (timr->it_incr)
si_private = ++timr->it_requeue_pending;
else {
remove_from_abslist(timr);
}
if (posix_timer_event(timr, si_private))
/*
* signal was not sent because of sig_ignor
* we will not get a call back to restart it AND
* it should be restarted.
*/
schedule_next_timer(timr);
}
unlock_timer(timr, flags); /* hold thru abs lock to keep irq off */ unlock_timer(timr, flags); /* hold thru abs lock to keep irq off */
} }
...@@ -585,10 +564,6 @@ sys_timer_create(clockid_t which_clock, ...@@ -585,10 +564,6 @@ sys_timer_create(clockid_t which_clock,
!posix_clocks[which_clock].res) !posix_clocks[which_clock].res)
return -EINVAL; return -EINVAL;
if (posix_clocks[which_clock].timer_create)
return posix_clocks[which_clock].timer_create(which_clock,
timer_event_spec, created_timer_id);
new_timer = alloc_posix_timer(); new_timer = alloc_posix_timer();
if (unlikely(!new_timer)) if (unlikely(!new_timer))
return -EAGAIN; return -EAGAIN;
...@@ -620,11 +595,17 @@ sys_timer_create(clockid_t which_clock, ...@@ -620,11 +595,17 @@ sys_timer_create(clockid_t which_clock,
new_timer->it_clock = which_clock; new_timer->it_clock = which_clock;
new_timer->it_incr = 0; new_timer->it_incr = 0;
new_timer->it_overrun = -1; new_timer->it_overrun = -1;
init_timer(&new_timer->it_timer); if (posix_clocks[which_clock].timer_create) {
new_timer->it_timer.expires = 0; error = posix_clocks[which_clock].timer_create(new_timer);
new_timer->it_timer.data = (unsigned long) new_timer; if (error)
new_timer->it_timer.function = posix_timer_fn; goto out;
set_timer_inactive(new_timer); } else {
init_timer(&new_timer->it_timer);
new_timer->it_timer.expires = 0;
new_timer->it_timer.data = (unsigned long) new_timer;
new_timer->it_timer.function = posix_timer_fn;
set_timer_inactive(new_timer);
}
/* /*
* return the timer_id now. The next step is hard to * return the timer_id now. The next step is hard to
...@@ -1239,9 +1220,7 @@ int do_posix_clock_nosettime(struct timespec *tp) ...@@ -1239,9 +1220,7 @@ int do_posix_clock_nosettime(struct timespec *tp)
return -EINVAL; return -EINVAL;
} }
int do_posix_clock_notimer_create(int which_clock, int do_posix_clock_notimer_create(struct k_itimer *timer) {
struct sigevent __user *timer_event_spec,
timer_t __user *created_timer_id) {
return -EINVAL; return -EINVAL;
} }
......
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