// SPDX-License-Identifier: GPL-2.0 #ifndef _LINUX_HRTIMER_REARM_H #define _LINUX_HRTIMER_REARM_H #ifdef CONFIG_HRTIMER_REARM_DEFERRED #include void __hrtimer_rearm_deferred(void); /* * This is purely CPU local, so check the TIF bit first to avoid the overhead of * the atomic test_and_clear_bit() operation for the common case where the bit * is not set. */ static __always_inline bool hrtimer_test_and_clear_rearm_deferred_tif(unsigned long tif_work) { lockdep_assert_irqs_disabled(); if (unlikely(tif_work & _TIF_HRTIMER_REARM)) { clear_thread_flag(TIF_HRTIMER_REARM); return true; } return false; } #define TIF_REARM_MASK (_TIF_NEED_RESCHED | _TIF_NEED_RESCHED_LAZY | _TIF_HRTIMER_REARM) /* Invoked from the exit to user before invoking exit_to_user_mode_loop() */ static __always_inline bool hrtimer_rearm_deferred_user_irq(unsigned long *tif_work, const unsigned long tif_mask) { /* Help the compiler to optimize the function out for syscall returns */ if (!(tif_mask & _TIF_HRTIMER_REARM)) return false; /* * Rearm the timer if none of the resched flags is set before going into * the loop which re-enables interrupts. */ if (unlikely((*tif_work & TIF_REARM_MASK) == _TIF_HRTIMER_REARM)) { clear_thread_flag(TIF_HRTIMER_REARM); __hrtimer_rearm_deferred(); /* Don't go into the loop if HRTIMER_REARM was the only flag */ *tif_work &= ~TIF_HRTIMER_REARM; return !*tif_work; } return false; } /* Invoked from the time slice extension decision function */ static __always_inline void hrtimer_rearm_deferred_tif(unsigned long tif_work) { if (hrtimer_test_and_clear_rearm_deferred_tif(tif_work)) __hrtimer_rearm_deferred(); } /* * This is to be called on all irqentry_exit() paths that will enable * interrupts. */ static __always_inline void hrtimer_rearm_deferred(void) { hrtimer_rearm_deferred_tif(read_thread_flags()); } /* * Invoked from the scheduler on entry to __schedule() so it can defer * rearming after the load balancing callbacks which might change hrtick. */ static __always_inline bool hrtimer_test_and_clear_rearm_deferred(void) { return hrtimer_test_and_clear_rearm_deferred_tif(read_thread_flags()); } #else /* CONFIG_HRTIMER_REARM_DEFERRED */ static __always_inline void __hrtimer_rearm_deferred(void) { } static __always_inline void hrtimer_rearm_deferred(void) { } static __always_inline void hrtimer_rearm_deferred_tif(unsigned long tif_work) { } static __always_inline bool hrtimer_rearm_deferred_user_irq(unsigned long *tif_work, const unsigned long tif_mask) { return false; } static __always_inline bool hrtimer_test_and_clear_rearm_deferred(void) { return false; } #endif /* !CONFIG_HRTIMER_REARM_DEFERRED */ #endif