diff options
Diffstat (limited to 'include/linux/ktime.h')
-rw-r--r-- | include/linux/ktime.h | 27 |
1 files changed, 21 insertions, 6 deletions
diff --git a/include/linux/ktime.h b/include/linux/ktime.h index 5fc3d1083071..2b6a204bd8d4 100644 --- a/include/linux/ktime.h +++ b/include/linux/ktime.h | |||
@@ -166,19 +166,34 @@ static inline bool ktime_before(const ktime_t cmp1, const ktime_t cmp2) | |||
166 | } | 166 | } |
167 | 167 | ||
168 | #if BITS_PER_LONG < 64 | 168 | #if BITS_PER_LONG < 64 |
169 | extern u64 __ktime_divns(const ktime_t kt, s64 div); | 169 | extern s64 __ktime_divns(const ktime_t kt, s64 div); |
170 | static inline u64 ktime_divns(const ktime_t kt, s64 div) | 170 | static inline s64 ktime_divns(const ktime_t kt, s64 div) |
171 | { | 171 | { |
172 | /* | ||
173 | * Negative divisors could cause an inf loop, | ||
174 | * so bug out here. | ||
175 | */ | ||
176 | BUG_ON(div < 0); | ||
172 | if (__builtin_constant_p(div) && !(div >> 32)) { | 177 | if (__builtin_constant_p(div) && !(div >> 32)) { |
173 | u64 ns = kt.tv64; | 178 | s64 ns = kt.tv64; |
174 | do_div(ns, div); | 179 | u64 tmp = ns < 0 ? -ns : ns; |
175 | return ns; | 180 | |
181 | do_div(tmp, div); | ||
182 | return ns < 0 ? -tmp : tmp; | ||
176 | } else { | 183 | } else { |
177 | return __ktime_divns(kt, div); | 184 | return __ktime_divns(kt, div); |
178 | } | 185 | } |
179 | } | 186 | } |
180 | #else /* BITS_PER_LONG < 64 */ | 187 | #else /* BITS_PER_LONG < 64 */ |
181 | # define ktime_divns(kt, div) (u64)((kt).tv64 / (div)) | 188 | static inline s64 ktime_divns(const ktime_t kt, s64 div) |
189 | { | ||
190 | /* | ||
191 | * 32-bit implementation cannot handle negative divisors, | ||
192 | * so catch them on 64bit as well. | ||
193 | */ | ||
194 | WARN_ON(div < 0); | ||
195 | return kt.tv64 / div; | ||
196 | } | ||
182 | #endif | 197 | #endif |
183 | 198 | ||
184 | static inline s64 ktime_to_us(const ktime_t kt) | 199 | static inline s64 ktime_to_us(const ktime_t kt) |