diff options
Diffstat (limited to 'kernel/workqueue.c')
| -rw-r--r-- | kernel/workqueue.c | 34 | 
1 files changed, 18 insertions, 16 deletions
| diff --git a/kernel/workqueue.c b/kernel/workqueue.c index ca71582fcfab..c579dbab2e36 100644 --- a/kernel/workqueue.c +++ b/kernel/workqueue.c @@ -1458,13 +1458,13 @@ static void __queue_delayed_work(int cpu, struct workqueue_struct *wq,  	timer_stats_timer_set_start_info(&dwork->timer);  	dwork->wq = wq; +	/* timer isn't guaranteed to run in this cpu, record earlier */ +	if (cpu == WORK_CPU_UNBOUND) +		cpu = raw_smp_processor_id();  	dwork->cpu = cpu;  	timer->expires = jiffies + delay; -	if (unlikely(cpu != WORK_CPU_UNBOUND)) -		add_timer_on(timer, cpu); -	else -		add_timer(timer); +	add_timer_on(timer, cpu);  }  /** @@ -3199,6 +3199,7 @@ static struct worker_pool *get_unbound_pool(const struct workqueue_attrs *attrs)  	u32 hash = wqattrs_hash(attrs);  	struct worker_pool *pool;  	int node; +	int target_node = NUMA_NO_NODE;  	lockdep_assert_held(&wq_pool_mutex); @@ -3210,13 +3211,25 @@ static struct worker_pool *get_unbound_pool(const struct workqueue_attrs *attrs)  		}  	} +	/* if cpumask is contained inside a NUMA node, we belong to that node */ +	if (wq_numa_enabled) { +		for_each_node(node) { +			if (cpumask_subset(attrs->cpumask, +					   wq_numa_possible_cpumask[node])) { +				target_node = node; +				break; +			} +		} +	} +  	/* nope, create a new one */ -	pool = kzalloc(sizeof(*pool), GFP_KERNEL); +	pool = kzalloc_node(sizeof(*pool), GFP_KERNEL, target_node);  	if (!pool || init_worker_pool(pool) < 0)  		goto fail;  	lockdep_set_subclass(&pool->lock, 1);	/* see put_pwq() */  	copy_workqueue_attrs(pool->attrs, attrs); +	pool->node = target_node;  	/*  	 * no_numa isn't a worker_pool attribute, always clear it.  See @@ -3224,17 +3237,6 @@ static struct worker_pool *get_unbound_pool(const struct workqueue_attrs *attrs)  	 */  	pool->attrs->no_numa = false; -	/* if cpumask is contained inside a NUMA node, we belong to that node */ -	if (wq_numa_enabled) { -		for_each_node(node) { -			if (cpumask_subset(pool->attrs->cpumask, -					   wq_numa_possible_cpumask[node])) { -				pool->node = node; -				break; -			} -		} -	} -  	if (worker_pool_assign_id(pool) < 0)  		goto fail; | 
