diff options
author | Thomas Gleixner <tglx@linutronix.de> | 2008-02-13 03:20:43 -0500 |
---|---|---|
committer | Thomas Gleixner <tglx@linutronix.de> | 2008-02-14 16:08:30 -0500 |
commit | 5a7780e725d1bb4c3094fcc12f1c5c5faea1e988 (patch) | |
tree | 50fc5cde427f3854d0b84ba1037fef3fb4693e11 | |
parent | e760e716d47b48caf98da348368fd41b4a9b9e7e (diff) |
hrtimer: check relative timeouts for overflow
Various user space callers ask for relative timeouts. While we fixed
that overflow issue in hrtimer_start(), the sites which convert
relative user space values to absolute timeouts themself were uncovered.
Instead of putting overflow checks into each place add a function
which does the sanity checking and convert all affected callers to use
it.
Thanks to Frans Pop, who reported the problem and tested the fixes.
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Acked-by: Ingo Molnar <mingo@elte.hu>
Tested-by: Frans Pop <elendil@planet.nl>
-rw-r--r-- | include/linux/ktime.h | 2 | ||||
-rw-r--r-- | kernel/futex.c | 2 | ||||
-rw-r--r-- | kernel/futex_compat.c | 2 | ||||
-rw-r--r-- | kernel/hrtimer.c | 37 | ||||
-rw-r--r-- | kernel/posix-timers.c | 8 |
5 files changed, 29 insertions, 22 deletions
diff --git a/include/linux/ktime.h b/include/linux/ktime.h index 36c542b70c6d..2cd7fa73d1af 100644 --- a/include/linux/ktime.h +++ b/include/linux/ktime.h | |||
@@ -310,6 +310,8 @@ static inline ktime_t ktime_sub_us(const ktime_t kt, const u64 usec) | |||
310 | return ktime_sub_ns(kt, usec * 1000); | 310 | return ktime_sub_ns(kt, usec * 1000); |
311 | } | 311 | } |
312 | 312 | ||
313 | extern ktime_t ktime_add_safe(const ktime_t lhs, const ktime_t rhs); | ||
314 | |||
313 | /* | 315 | /* |
314 | * The resolution of the clocks. The resolution value is returned in | 316 | * The resolution of the clocks. The resolution value is returned in |
315 | * the clock_getres() system call to give application programmers an | 317 | * the clock_getres() system call to give application programmers an |
diff --git a/kernel/futex.c b/kernel/futex.c index a6baaec44b8f..221f2128a437 100644 --- a/kernel/futex.c +++ b/kernel/futex.c | |||
@@ -2116,7 +2116,7 @@ asmlinkage long sys_futex(u32 __user *uaddr, int op, u32 val, | |||
2116 | 2116 | ||
2117 | t = timespec_to_ktime(ts); | 2117 | t = timespec_to_ktime(ts); |
2118 | if (cmd == FUTEX_WAIT) | 2118 | if (cmd == FUTEX_WAIT) |
2119 | t = ktime_add(ktime_get(), t); | 2119 | t = ktime_add_safe(ktime_get(), t); |
2120 | tp = &t; | 2120 | tp = &t; |
2121 | } | 2121 | } |
2122 | /* | 2122 | /* |
diff --git a/kernel/futex_compat.c b/kernel/futex_compat.c index 133d558db452..7d5e4b016f39 100644 --- a/kernel/futex_compat.c +++ b/kernel/futex_compat.c | |||
@@ -176,7 +176,7 @@ asmlinkage long compat_sys_futex(u32 __user *uaddr, int op, u32 val, | |||
176 | 176 | ||
177 | t = timespec_to_ktime(ts); | 177 | t = timespec_to_ktime(ts); |
178 | if (cmd == FUTEX_WAIT) | 178 | if (cmd == FUTEX_WAIT) |
179 | t = ktime_add(ktime_get(), t); | 179 | t = ktime_add_safe(ktime_get(), t); |
180 | tp = &t; | 180 | tp = &t; |
181 | } | 181 | } |
182 | if (cmd == FUTEX_REQUEUE || cmd == FUTEX_CMP_REQUEUE) | 182 | if (cmd == FUTEX_REQUEUE || cmd == FUTEX_CMP_REQUEUE) |
diff --git a/kernel/hrtimer.c b/kernel/hrtimer.c index 3f4a57c7895d..c2893af9479e 100644 --- a/kernel/hrtimer.c +++ b/kernel/hrtimer.c | |||
@@ -326,6 +326,23 @@ u64 ktime_divns(const ktime_t kt, s64 div) | |||
326 | #endif /* BITS_PER_LONG >= 64 */ | 326 | #endif /* BITS_PER_LONG >= 64 */ |
327 | 327 | ||
328 | /* | 328 | /* |
329 | * Add two ktime values and do a safety check for overflow: | ||
330 | */ | ||
331 | ktime_t ktime_add_safe(const ktime_t lhs, const ktime_t rhs) | ||
332 | { | ||
333 | ktime_t res = ktime_add(lhs, rhs); | ||
334 | |||
335 | /* | ||
336 | * We use KTIME_SEC_MAX here, the maximum timeout which we can | ||
337 | * return to user space in a timespec: | ||
338 | */ | ||
339 | if (res.tv64 < 0 || res.tv64 < lhs.tv64 || res.tv64 < rhs.tv64) | ||
340 | res = ktime_set(KTIME_SEC_MAX, 0); | ||
341 | |||
342 | return res; | ||
343 | } | ||
344 | |||
345 | /* | ||
329 | * Check, whether the timer is on the callback pending list | 346 | * Check, whether the timer is on the callback pending list |
330 | */ | 347 | */ |
331 | static inline int hrtimer_cb_pending(const struct hrtimer *timer) | 348 | static inline int hrtimer_cb_pending(const struct hrtimer *timer) |
@@ -682,13 +699,7 @@ u64 hrtimer_forward(struct hrtimer *timer, ktime_t now, ktime_t interval) | |||
682 | */ | 699 | */ |
683 | orun++; | 700 | orun++; |
684 | } | 701 | } |
685 | timer->expires = ktime_add(timer->expires, interval); | 702 | timer->expires = ktime_add_safe(timer->expires, interval); |
686 | /* | ||
687 | * Make sure, that the result did not wrap with a very large | ||
688 | * interval. | ||
689 | */ | ||
690 | if (timer->expires.tv64 < 0) | ||
691 | timer->expires = ktime_set(KTIME_SEC_MAX, 0); | ||
692 | 703 | ||
693 | return orun; | 704 | return orun; |
694 | } | 705 | } |
@@ -839,7 +850,7 @@ hrtimer_start(struct hrtimer *timer, ktime_t tim, const enum hrtimer_mode mode) | |||
839 | new_base = switch_hrtimer_base(timer, base); | 850 | new_base = switch_hrtimer_base(timer, base); |
840 | 851 | ||
841 | if (mode == HRTIMER_MODE_REL) { | 852 | if (mode == HRTIMER_MODE_REL) { |
842 | tim = ktime_add(tim, new_base->get_time()); | 853 | tim = ktime_add_safe(tim, new_base->get_time()); |
843 | /* | 854 | /* |
844 | * CONFIG_TIME_LOW_RES is a temporary way for architectures | 855 | * CONFIG_TIME_LOW_RES is a temporary way for architectures |
845 | * to signal that they simply return xtime in | 856 | * to signal that they simply return xtime in |
@@ -848,16 +859,8 @@ hrtimer_start(struct hrtimer *timer, ktime_t tim, const enum hrtimer_mode mode) | |||
848 | * timeouts. This will go away with the GTOD framework. | 859 | * timeouts. This will go away with the GTOD framework. |
849 | */ | 860 | */ |
850 | #ifdef CONFIG_TIME_LOW_RES | 861 | #ifdef CONFIG_TIME_LOW_RES |
851 | tim = ktime_add(tim, base->resolution); | 862 | tim = ktime_add_safe(tim, base->resolution); |
852 | #endif | 863 | #endif |
853 | /* | ||
854 | * Careful here: User space might have asked for a | ||
855 | * very long sleep, so the add above might result in a | ||
856 | * negative number, which enqueues the timer in front | ||
857 | * of the queue. | ||
858 | */ | ||
859 | if (tim.tv64 < 0) | ||
860 | tim.tv64 = KTIME_MAX; | ||
861 | } | 864 | } |
862 | timer->expires = tim; | 865 | timer->expires = tim; |
863 | 866 | ||
diff --git a/kernel/posix-timers.c b/kernel/posix-timers.c index 022c9c3cee6f..a9b04203a66d 100644 --- a/kernel/posix-timers.c +++ b/kernel/posix-timers.c | |||
@@ -767,9 +767,11 @@ common_timer_set(struct k_itimer *timr, int flags, | |||
767 | /* SIGEV_NONE timers are not queued ! See common_timer_get */ | 767 | /* SIGEV_NONE timers are not queued ! See common_timer_get */ |
768 | if (((timr->it_sigev_notify & ~SIGEV_THREAD_ID) == SIGEV_NONE)) { | 768 | if (((timr->it_sigev_notify & ~SIGEV_THREAD_ID) == SIGEV_NONE)) { |
769 | /* Setup correct expiry time for relative timers */ | 769 | /* Setup correct expiry time for relative timers */ |
770 | if (mode == HRTIMER_MODE_REL) | 770 | if (mode == HRTIMER_MODE_REL) { |
771 | timer->expires = ktime_add(timer->expires, | 771 | timer->expires = |
772 | timer->base->get_time()); | 772 | ktime_add_safe(timer->expires, |
773 | timer->base->get_time()); | ||
774 | } | ||
773 | return 0; | 775 | return 0; |
774 | } | 776 | } |
775 | 777 | ||