diff options
| -rw-r--r-- | arch/x86/include/asm/preempt.h | 1 | ||||
| -rw-r--r-- | kernel/context_tracking.c | 40 | ||||
| -rw-r--r-- | kernel/sched/core.c | 41 | 
3 files changed, 42 insertions, 40 deletions
| diff --git a/arch/x86/include/asm/preempt.h b/arch/x86/include/asm/preempt.h index 7024c12f7bfe..400873450e33 100644 --- a/arch/x86/include/asm/preempt.h +++ b/arch/x86/include/asm/preempt.h @@ -105,6 +105,7 @@ static __always_inline bool should_resched(void)  # ifdef CONFIG_CONTEXT_TRACKING      extern asmlinkage void ___preempt_schedule_context(void);  #   define __preempt_schedule_context() asm ("call ___preempt_schedule_context") +    extern asmlinkage void preempt_schedule_context(void);  # endif  #endif diff --git a/kernel/context_tracking.c b/kernel/context_tracking.c index 5664985c46a0..937ecdfdf258 100644 --- a/kernel/context_tracking.c +++ b/kernel/context_tracking.c @@ -107,46 +107,6 @@ void context_tracking_user_enter(void)  }  NOKPROBE_SYMBOL(context_tracking_user_enter); -#ifdef CONFIG_PREEMPT -/** - * preempt_schedule_context - preempt_schedule called by tracing - * - * The tracing infrastructure uses preempt_enable_notrace to prevent - * recursion and tracing preempt enabling caused by the tracing - * infrastructure itself. But as tracing can happen in areas coming - * from userspace or just about to enter userspace, a preempt enable - * can occur before user_exit() is called. This will cause the scheduler - * to be called when the system is still in usermode. - * - * To prevent this, the preempt_enable_notrace will use this function - * instead of preempt_schedule() to exit user context if needed before - * calling the scheduler. - */ -asmlinkage __visible void __sched notrace preempt_schedule_context(void) -{ -	enum ctx_state prev_ctx; - -	if (likely(!preemptible())) -		return; - -	/* -	 * Need to disable preemption in case user_exit() is traced -	 * and the tracer calls preempt_enable_notrace() causing -	 * an infinite recursion. -	 */ -	preempt_disable_notrace(); -	prev_ctx = exception_enter(); -	preempt_enable_no_resched_notrace(); - -	preempt_schedule(); - -	preempt_disable_notrace(); -	exception_exit(prev_ctx); -	preempt_enable_notrace(); -} -EXPORT_SYMBOL_GPL(preempt_schedule_context); -#endif /* CONFIG_PREEMPT */ -  /**   * context_tracking_user_exit - Inform the context tracking that the CPU is   *                              exiting userspace mode and entering the kernel. diff --git a/kernel/sched/core.c b/kernel/sched/core.c index dde8adb7d0c0..240157c13ddc 100644 --- a/kernel/sched/core.c +++ b/kernel/sched/core.c @@ -2951,6 +2951,47 @@ asmlinkage __visible void __sched notrace preempt_schedule(void)  }  NOKPROBE_SYMBOL(preempt_schedule);  EXPORT_SYMBOL(preempt_schedule); + +#ifdef CONFIG_CONTEXT_TRACKING +/** + * preempt_schedule_context - preempt_schedule called by tracing + * + * The tracing infrastructure uses preempt_enable_notrace to prevent + * recursion and tracing preempt enabling caused by the tracing + * infrastructure itself. But as tracing can happen in areas coming + * from userspace or just about to enter userspace, a preempt enable + * can occur before user_exit() is called. This will cause the scheduler + * to be called when the system is still in usermode. + * + * To prevent this, the preempt_enable_notrace will use this function + * instead of preempt_schedule() to exit user context if needed before + * calling the scheduler. + */ +asmlinkage __visible void __sched notrace preempt_schedule_context(void) +{ +	enum ctx_state prev_ctx; + +	if (likely(!preemptible())) +		return; + +	do { +		__preempt_count_add(PREEMPT_ACTIVE); +		/* +		 * Needs preempt disabled in case user_exit() is traced +		 * and the tracer calls preempt_enable_notrace() causing +		 * an infinite recursion. +		 */ +		prev_ctx = exception_enter(); +		__schedule(); +		exception_exit(prev_ctx); + +		__preempt_count_sub(PREEMPT_ACTIVE); +		barrier(); +	} while (need_resched()); +} +EXPORT_SYMBOL_GPL(preempt_schedule_context); +#endif /* CONFIG_CONTEXT_TRACKING */ +  #endif /* CONFIG_PREEMPT */  /* | 
