diff options
Diffstat (limited to 'kernel/timer.c')
| -rw-r--r-- | kernel/timer.c | 54 |
1 files changed, 45 insertions, 9 deletions
diff --git a/kernel/timer.c b/kernel/timer.c index ee305c8d4e18..97bf05baade7 100644 --- a/kernel/timer.c +++ b/kernel/timer.c | |||
| @@ -90,8 +90,13 @@ static DEFINE_PER_CPU(struct tvec_base *, tvec_bases) = &boot_tvec_bases; | |||
| 90 | 90 | ||
| 91 | /* | 91 | /* |
| 92 | * Note that all tvec_bases are 2 byte aligned and lower bit of | 92 | * Note that all tvec_bases are 2 byte aligned and lower bit of |
| 93 | * base in timer_list is guaranteed to be zero. Use the LSB for | 93 | * base in timer_list is guaranteed to be zero. Use the LSB to |
| 94 | * the new flag to indicate whether the timer is deferrable | 94 | * indicate whether the timer is deferrable. |
| 95 | * | ||
| 96 | * A deferrable timer will work normally when the system is busy, but | ||
| 97 | * will not cause a CPU to come out of idle just to service it; instead, | ||
| 98 | * the timer will be serviced when the CPU eventually wakes up with a | ||
| 99 | * subsequent non-deferrable timer. | ||
| 95 | */ | 100 | */ |
| 96 | #define TBASE_DEFERRABLE_FLAG (0x1) | 101 | #define TBASE_DEFERRABLE_FLAG (0x1) |
| 97 | 102 | ||
| @@ -321,6 +326,7 @@ EXPORT_SYMBOL_GPL(round_jiffies_up_relative); | |||
| 321 | 326 | ||
| 322 | /** | 327 | /** |
| 323 | * set_timer_slack - set the allowed slack for a timer | 328 | * set_timer_slack - set the allowed slack for a timer |
| 329 | * @timer: the timer to be modified | ||
| 324 | * @slack_hz: the amount of time (in jiffies) allowed for rounding | 330 | * @slack_hz: the amount of time (in jiffies) allowed for rounding |
| 325 | * | 331 | * |
| 326 | * Set the amount of time, in jiffies, that a certain timer has | 332 | * Set the amount of time, in jiffies, that a certain timer has |
| @@ -577,6 +583,19 @@ static void __init_timer(struct timer_list *timer, | |||
| 577 | lockdep_init_map(&timer->lockdep_map, name, key, 0); | 583 | lockdep_init_map(&timer->lockdep_map, name, key, 0); |
| 578 | } | 584 | } |
| 579 | 585 | ||
| 586 | void setup_deferrable_timer_on_stack_key(struct timer_list *timer, | ||
| 587 | const char *name, | ||
| 588 | struct lock_class_key *key, | ||
| 589 | void (*function)(unsigned long), | ||
| 590 | unsigned long data) | ||
| 591 | { | ||
| 592 | timer->function = function; | ||
| 593 | timer->data = data; | ||
| 594 | init_timer_on_stack_key(timer, name, key); | ||
| 595 | timer_set_deferrable(timer); | ||
| 596 | } | ||
| 597 | EXPORT_SYMBOL_GPL(setup_deferrable_timer_on_stack_key); | ||
| 598 | |||
| 580 | /** | 599 | /** |
| 581 | * init_timer_key - initialize a timer | 600 | * init_timer_key - initialize a timer |
| 582 | * @timer: the timer to be initialized | 601 | * @timer: the timer to be initialized |
| @@ -679,12 +698,8 @@ __mod_timer(struct timer_list *timer, unsigned long expires, | |||
| 679 | cpu = smp_processor_id(); | 698 | cpu = smp_processor_id(); |
| 680 | 699 | ||
| 681 | #if defined(CONFIG_NO_HZ) && defined(CONFIG_SMP) | 700 | #if defined(CONFIG_NO_HZ) && defined(CONFIG_SMP) |
| 682 | if (!pinned && get_sysctl_timer_migration() && idle_cpu(cpu)) { | 701 | if (!pinned && get_sysctl_timer_migration() && idle_cpu(cpu)) |
| 683 | int preferred_cpu = get_nohz_load_balancer(); | 702 | cpu = get_nohz_timer_target(); |
| 684 | |||
| 685 | if (preferred_cpu >= 0) | ||
| 686 | cpu = preferred_cpu; | ||
| 687 | } | ||
| 688 | #endif | 703 | #endif |
| 689 | new_base = per_cpu(tvec_bases, cpu); | 704 | new_base = per_cpu(tvec_bases, cpu); |
| 690 | 705 | ||
| @@ -1289,7 +1304,6 @@ void run_local_timers(void) | |||
| 1289 | { | 1304 | { |
| 1290 | hrtimer_run_queues(); | 1305 | hrtimer_run_queues(); |
| 1291 | raise_softirq(TIMER_SOFTIRQ); | 1306 | raise_softirq(TIMER_SOFTIRQ); |
| 1292 | softlockup_tick(); | ||
| 1293 | } | 1307 | } |
| 1294 | 1308 | ||
| 1295 | /* | 1309 | /* |
| @@ -1750,3 +1764,25 @@ unsigned long msleep_interruptible(unsigned int msecs) | |||
| 1750 | } | 1764 | } |
| 1751 | 1765 | ||
| 1752 | EXPORT_SYMBOL(msleep_interruptible); | 1766 | EXPORT_SYMBOL(msleep_interruptible); |
| 1767 | |||
| 1768 | static int __sched do_usleep_range(unsigned long min, unsigned long max) | ||
| 1769 | { | ||
| 1770 | ktime_t kmin; | ||
| 1771 | unsigned long delta; | ||
| 1772 | |||
| 1773 | kmin = ktime_set(0, min * NSEC_PER_USEC); | ||
| 1774 | delta = (max - min) * NSEC_PER_USEC; | ||
| 1775 | return schedule_hrtimeout_range(&kmin, delta, HRTIMER_MODE_REL); | ||
| 1776 | } | ||
| 1777 | |||
| 1778 | /** | ||
| 1779 | * usleep_range - Drop in replacement for udelay where wakeup is flexible | ||
| 1780 | * @min: Minimum time in usecs to sleep | ||
| 1781 | * @max: Maximum time in usecs to sleep | ||
| 1782 | */ | ||
| 1783 | void usleep_range(unsigned long min, unsigned long max) | ||
| 1784 | { | ||
| 1785 | __set_current_state(TASK_UNINTERRUPTIBLE); | ||
| 1786 | do_usleep_range(min, max); | ||
| 1787 | } | ||
| 1788 | EXPORT_SYMBOL(usleep_range); | ||
