diff options
| author | Arnd Bergmann <arnd@arndb.de> | 2012-03-15 16:11:36 +0000 | 
|---|---|---|
| committer | Arnd Bergmann <arnd@arndb.de> | 2012-03-15 16:11:40 +0000 | 
| commit | 86a30bece9ad4cc91c393a829a7b128291e0fb65 (patch) | |
| tree | 6cb3e6ad413d74118535f77436056c8d3cfae0eb /kernel/irq/manage.c | |
| parent | 243d58ec5792299fa212d05a4113c0ebac2df6a3 (diff) | |
| parent | a323f66439c04d1c3ae4dc20cc2d44d52ee43c9f (diff) | |
Merge branch 'fixes-non-critical' of git://github.com/hzhuang1/linux into next/maintainers
* 'fixes-non-critical' of git://github.com/hzhuang1/linux:
  MAINTAINERS: update MAINTAINERS email entry
  MAINTAINERS: update maintainer entry for pxa/hx4700
(update to v3.3-rc7)
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Diffstat (limited to 'kernel/irq/manage.c')
| -rw-r--r-- | kernel/irq/manage.c | 44 | 
1 files changed, 38 insertions, 6 deletions
diff --git a/kernel/irq/manage.c b/kernel/irq/manage.c index 32313c08444..0f0d4704ddd 100644 --- a/kernel/irq/manage.c +++ b/kernel/irq/manage.c @@ -985,6 +985,11 @@ __setup_irq(unsigned int irq, struct irq_desc *desc, struct irqaction *new)  		/* add new interrupt at end of irq queue */  		do { +			/* +			 * Or all existing action->thread_mask bits, +			 * so we can find the next zero bit for this +			 * new action. +			 */  			thread_mask |= old->thread_mask;  			old_ptr = &old->next;  			old = *old_ptr; @@ -993,14 +998,41 @@ __setup_irq(unsigned int irq, struct irq_desc *desc, struct irqaction *new)  	}  	/* -	 * Setup the thread mask for this irqaction. Unlikely to have -	 * 32 resp 64 irqs sharing one line, but who knows. +	 * Setup the thread mask for this irqaction for ONESHOT. For +	 * !ONESHOT irqs the thread mask is 0 so we can avoid a +	 * conditional in irq_wake_thread().  	 */ -	if (new->flags & IRQF_ONESHOT && thread_mask == ~0UL) { -		ret = -EBUSY; -		goto out_mask; +	if (new->flags & IRQF_ONESHOT) { +		/* +		 * Unlikely to have 32 resp 64 irqs sharing one line, +		 * but who knows. +		 */ +		if (thread_mask == ~0UL) { +			ret = -EBUSY; +			goto out_mask; +		} +		/* +		 * The thread_mask for the action is or'ed to +		 * desc->thread_active to indicate that the +		 * IRQF_ONESHOT thread handler has been woken, but not +		 * yet finished. The bit is cleared when a thread +		 * completes. When all threads of a shared interrupt +		 * line have completed desc->threads_active becomes +		 * zero and the interrupt line is unmasked. See +		 * handle.c:irq_wake_thread() for further information. +		 * +		 * If no thread is woken by primary (hard irq context) +		 * interrupt handlers, then desc->threads_active is +		 * also checked for zero to unmask the irq line in the +		 * affected hard irq flow handlers +		 * (handle_[fasteoi|level]_irq). +		 * +		 * The new action gets the first zero bit of +		 * thread_mask assigned. See the loop above which or's +		 * all existing action->thread_mask bits. +		 */ +		new->thread_mask = 1 << ffz(thread_mask);  	} -	new->thread_mask = 1 << ffz(thread_mask);  	if (!shared) {  		init_waitqueue_head(&desc->wait_for_threads);  | 
