diff options
| author | Dan Williams <dan.j.williams@intel.com> | 2009-09-08 17:55:54 -0700 | 
|---|---|---|
| committer | Dan Williams <dan.j.williams@intel.com> | 2009-09-08 17:55:54 -0700 | 
| commit | 9134d02bc0af4a8747d448d1f811ec5f8eb96df6 (patch) | |
| tree | 704c3e5dcc10f360815c4868a74711f82fb62e27 /kernel/irq/manage.c | |
| parent | bbb20089a3275a19e475dbc21320c3742e3ca423 (diff) | |
| parent | 80ffb3cceaefa405f2ecd46d66500ed8d53efe74 (diff) | |
Merge commit 'md/for-linus' into async-tx-next
Conflicts:
	drivers/md/raid5.c
Diffstat (limited to 'kernel/irq/manage.c')
| -rw-r--r-- | kernel/irq/manage.c | 55 | 
1 files changed, 49 insertions, 6 deletions
| diff --git a/kernel/irq/manage.c b/kernel/irq/manage.c index 50da6767290..61c679db468 100644 --- a/kernel/irq/manage.c +++ b/kernel/irq/manage.c @@ -80,14 +80,22 @@ int irq_can_set_affinity(unsigned int irq)  	return 1;  } -void -irq_set_thread_affinity(struct irq_desc *desc, const struct cpumask *cpumask) +/** + *	irq_set_thread_affinity - Notify irq threads to adjust affinity + *	@desc:		irq descriptor which has affitnity changed + * + *	We just set IRQTF_AFFINITY and delegate the affinity setting + *	to the interrupt thread itself. We can not call + *	set_cpus_allowed_ptr() here as we hold desc->lock and this + *	code can be called from hard interrupt context. + */ +void irq_set_thread_affinity(struct irq_desc *desc)  {  	struct irqaction *action = desc->action;  	while (action) {  		if (action->thread) -			set_cpus_allowed_ptr(action->thread, cpumask); +			set_bit(IRQTF_AFFINITY, &action->thread_flags);  		action = action->next;  	}  } @@ -112,7 +120,7 @@ int irq_set_affinity(unsigned int irq, const struct cpumask *cpumask)  	if (desc->status & IRQ_MOVE_PCNTXT) {  		if (!desc->chip->set_affinity(irq, cpumask)) {  			cpumask_copy(desc->affinity, cpumask); -			irq_set_thread_affinity(desc, cpumask); +			irq_set_thread_affinity(desc);  		}  	}  	else { @@ -122,7 +130,7 @@ int irq_set_affinity(unsigned int irq, const struct cpumask *cpumask)  #else  	if (!desc->chip->set_affinity(irq, cpumask)) {  		cpumask_copy(desc->affinity, cpumask); -		irq_set_thread_affinity(desc, cpumask); +		irq_set_thread_affinity(desc);  	}  #endif  	desc->status |= IRQ_AFFINITY_SET; @@ -176,7 +184,7 @@ int irq_select_affinity_usr(unsigned int irq)  	spin_lock_irqsave(&desc->lock, flags);  	ret = setup_affinity(irq, desc);  	if (!ret) -		irq_set_thread_affinity(desc, desc->affinity); +		irq_set_thread_affinity(desc);  	spin_unlock_irqrestore(&desc->lock, flags);  	return ret; @@ -443,6 +451,39 @@ static int irq_wait_for_interrupt(struct irqaction *action)  	return -1;  } +#ifdef CONFIG_SMP +/* + * Check whether we need to change the affinity of the interrupt thread. + */ +static void +irq_thread_check_affinity(struct irq_desc *desc, struct irqaction *action) +{ +	cpumask_var_t mask; + +	if (!test_and_clear_bit(IRQTF_AFFINITY, &action->thread_flags)) +		return; + +	/* +	 * In case we are out of memory we set IRQTF_AFFINITY again and +	 * try again next time +	 */ +	if (!alloc_cpumask_var(&mask, GFP_KERNEL)) { +		set_bit(IRQTF_AFFINITY, &action->thread_flags); +		return; +	} + +	spin_lock_irq(&desc->lock); +	cpumask_copy(mask, desc->affinity); +	spin_unlock_irq(&desc->lock); + +	set_cpus_allowed_ptr(current, mask); +	free_cpumask_var(mask); +} +#else +static inline void +irq_thread_check_affinity(struct irq_desc *desc, struct irqaction *action) { } +#endif +  /*   * Interrupt handler thread   */ @@ -458,6 +499,8 @@ static int irq_thread(void *data)  	while (!irq_wait_for_interrupt(action)) { +		irq_thread_check_affinity(desc, action); +  		atomic_inc(&desc->threads_active);  		spin_lock_irq(&desc->lock); | 
