diff options
Diffstat (limited to 'kernel/softirq.c')
| -rw-r--r-- | kernel/softirq.c | 92 | 
1 files changed, 49 insertions, 43 deletions
| diff --git a/kernel/softirq.c b/kernel/softirq.c index 11025ccc06dd..8a1e6e104892 100644 --- a/kernel/softirq.c +++ b/kernel/softirq.c @@ -89,7 +89,7 @@ static void wakeup_softirqd(void)   * where hardirqs are disabled legitimately:   */  #ifdef CONFIG_TRACE_IRQFLAGS -static void __local_bh_disable(unsigned long ip, unsigned int cnt) +void __local_bh_disable_ip(unsigned long ip, unsigned int cnt)  {  	unsigned long flags; @@ -107,33 +107,21 @@ static void __local_bh_disable(unsigned long ip, unsigned int cnt)  	/*  	 * Were softirqs turned off above:  	 */ -	if (softirq_count() == cnt) +	if (softirq_count() == (cnt & SOFTIRQ_MASK))  		trace_softirqs_off(ip);  	raw_local_irq_restore(flags);  	if (preempt_count() == cnt)  		trace_preempt_off(CALLER_ADDR0, get_parent_ip(CALLER_ADDR1));  } -#else /* !CONFIG_TRACE_IRQFLAGS */ -static inline void __local_bh_disable(unsigned long ip, unsigned int cnt) -{ -	preempt_count_add(cnt); -	barrier(); -} +EXPORT_SYMBOL(__local_bh_disable_ip);  #endif /* CONFIG_TRACE_IRQFLAGS */ -void local_bh_disable(void) -{ -	__local_bh_disable(_RET_IP_, SOFTIRQ_DISABLE_OFFSET); -} - -EXPORT_SYMBOL(local_bh_disable); -  static void __local_bh_enable(unsigned int cnt)  {  	WARN_ON_ONCE(!irqs_disabled()); -	if (softirq_count() == cnt) +	if (softirq_count() == (cnt & SOFTIRQ_MASK))  		trace_softirqs_on(_RET_IP_);  	preempt_count_sub(cnt);  } @@ -151,7 +139,7 @@ void _local_bh_enable(void)  EXPORT_SYMBOL(_local_bh_enable); -static inline void _local_bh_enable_ip(unsigned long ip) +void __local_bh_enable_ip(unsigned long ip, unsigned int cnt)  {  	WARN_ON_ONCE(in_irq() || irqs_disabled());  #ifdef CONFIG_TRACE_IRQFLAGS @@ -166,7 +154,7 @@ static inline void _local_bh_enable_ip(unsigned long ip)  	 * Keep preemption disabled until we are done with  	 * softirq processing:   	 */ -	preempt_count_sub(SOFTIRQ_DISABLE_OFFSET - 1); +	preempt_count_sub(cnt - 1);  	if (unlikely(!in_interrupt() && local_softirq_pending())) {  		/* @@ -182,18 +170,7 @@ static inline void _local_bh_enable_ip(unsigned long ip)  #endif  	preempt_check_resched();  } - -void local_bh_enable(void) -{ -	_local_bh_enable_ip(_RET_IP_); -} -EXPORT_SYMBOL(local_bh_enable); - -void local_bh_enable_ip(unsigned long ip) -{ -	_local_bh_enable_ip(ip); -} -EXPORT_SYMBOL(local_bh_enable_ip); +EXPORT_SYMBOL(__local_bh_enable_ip);  /*   * We restart softirq processing for at most MAX_SOFTIRQ_RESTART times, @@ -211,14 +188,48 @@ EXPORT_SYMBOL(local_bh_enable_ip);  #define MAX_SOFTIRQ_TIME  msecs_to_jiffies(2)  #define MAX_SOFTIRQ_RESTART 10 +#ifdef CONFIG_TRACE_IRQFLAGS +/* + * When we run softirqs from irq_exit() and thus on the hardirq stack we need + * to keep the lockdep irq context tracking as tight as possible in order to + * not miss-qualify lock contexts and miss possible deadlocks. + */ + +static inline bool lockdep_softirq_start(void) +{ +	bool in_hardirq = false; + +	if (trace_hardirq_context(current)) { +		in_hardirq = true; +		trace_hardirq_exit(); +	} + +	lockdep_softirq_enter(); + +	return in_hardirq; +} + +static inline void lockdep_softirq_end(bool in_hardirq) +{ +	lockdep_softirq_exit(); + +	if (in_hardirq) +		trace_hardirq_enter(); +} +#else +static inline bool lockdep_softirq_start(void) { return false; } +static inline void lockdep_softirq_end(bool in_hardirq) { } +#endif +  asmlinkage void __do_softirq(void)  { -	struct softirq_action *h; -	__u32 pending;  	unsigned long end = jiffies + MAX_SOFTIRQ_TIME; -	int cpu;  	unsigned long old_flags = current->flags;  	int max_restart = MAX_SOFTIRQ_RESTART; +	struct softirq_action *h; +	bool in_hardirq; +	__u32 pending; +	int cpu;  	/*  	 * Mask out PF_MEMALLOC s current task context is borrowed for the @@ -230,8 +241,8 @@ asmlinkage void __do_softirq(void)  	pending = local_softirq_pending();  	account_irq_enter_time(current); -	__local_bh_disable(_RET_IP_, SOFTIRQ_OFFSET); -	lockdep_softirq_enter(); +	__local_bh_disable_ip(_RET_IP_, SOFTIRQ_OFFSET); +	in_hardirq = lockdep_softirq_start();  	cpu = smp_processor_id();  restart: @@ -278,16 +289,13 @@ restart:  		wakeup_softirqd();  	} -	lockdep_softirq_exit(); - +	lockdep_softirq_end(in_hardirq);  	account_irq_exit_time(current);  	__local_bh_enable(SOFTIRQ_OFFSET);  	WARN_ON_ONCE(in_interrupt());  	tsk_restore_flags(current, old_flags, PF_MEMALLOC);  } - -  asmlinkage void do_softirq(void)  {  	__u32 pending; @@ -311,8 +319,6 @@ asmlinkage void do_softirq(void)   */  void irq_enter(void)  { -	int cpu = smp_processor_id(); -  	rcu_irq_enter();  	if (is_idle_task(current) && !in_interrupt()) {  		/* @@ -320,7 +326,7 @@ void irq_enter(void)  		 * here, as softirq will be serviced on return from interrupt.  		 */  		local_bh_disable(); -		tick_check_idle(cpu); +		tick_check_idle();  		_local_bh_enable();  	} @@ -375,13 +381,13 @@ void irq_exit(void)  #endif  	account_irq_exit_time(current); -	trace_hardirq_exit();  	preempt_count_sub(HARDIRQ_OFFSET);  	if (!in_interrupt() && local_softirq_pending())  		invoke_softirq();  	tick_irq_exit();  	rcu_irq_exit(); +	trace_hardirq_exit(); /* must be last! */  }  /* | 
