diff options
Diffstat (limited to 'kernel/workqueue.c')
| -rw-r--r-- | kernel/workqueue.c | 45 | 
1 files changed, 40 insertions, 5 deletions
| diff --git a/kernel/workqueue.c b/kernel/workqueue.c index 4aa9f5bc6b2..f02c4a4a0c3 100644 --- a/kernel/workqueue.c +++ b/kernel/workqueue.c @@ -272,6 +272,15 @@ static cpumask_var_t *wq_numa_possible_cpumask;  static bool wq_disable_numa;  module_param_named(disable_numa, wq_disable_numa, bool, 0444); +/* see the comment above the definition of WQ_POWER_EFFICIENT */ +#ifdef CONFIG_WQ_POWER_EFFICIENT_DEFAULT +static bool wq_power_efficient = true; +#else +static bool wq_power_efficient; +#endif + +module_param_named(power_efficient, wq_power_efficient, bool, 0444); +  static bool wq_numa_enabled;		/* unbound NUMA affinity enabled */  /* buf for wq_update_unbound_numa_attrs(), protected by CPU hotplug exclusion */ @@ -296,7 +305,7 @@ static DEFINE_HASHTABLE(unbound_pool_hash, UNBOUND_POOL_HASH_ORDER);  static struct workqueue_attrs *unbound_std_wq_attrs[NR_STD_WORKER_POOLS];  struct workqueue_struct *system_wq __read_mostly; -EXPORT_SYMBOL_GPL(system_wq); +EXPORT_SYMBOL(system_wq);  struct workqueue_struct *system_highpri_wq __read_mostly;  EXPORT_SYMBOL_GPL(system_highpri_wq);  struct workqueue_struct *system_long_wq __read_mostly; @@ -305,6 +314,10 @@ struct workqueue_struct *system_unbound_wq __read_mostly;  EXPORT_SYMBOL_GPL(system_unbound_wq);  struct workqueue_struct *system_freezable_wq __read_mostly;  EXPORT_SYMBOL_GPL(system_freezable_wq); +struct workqueue_struct *system_power_efficient_wq __read_mostly; +EXPORT_SYMBOL_GPL(system_power_efficient_wq); +struct workqueue_struct *system_freezable_power_efficient_wq __read_mostly; +EXPORT_SYMBOL_GPL(system_freezable_power_efficient_wq);  static int worker_thread(void *__worker);  static void copy_workqueue_attrs(struct workqueue_attrs *to, @@ -1411,7 +1424,7 @@ bool queue_work_on(int cpu, struct workqueue_struct *wq,  	local_irq_restore(flags);  	return ret;  } -EXPORT_SYMBOL_GPL(queue_work_on); +EXPORT_SYMBOL(queue_work_on);  void delayed_work_timer_fn(unsigned long __data)  { @@ -1485,7 +1498,7 @@ bool queue_delayed_work_on(int cpu, struct workqueue_struct *wq,  	local_irq_restore(flags);  	return ret;  } -EXPORT_SYMBOL_GPL(queue_delayed_work_on); +EXPORT_SYMBOL(queue_delayed_work_on);  /**   * mod_delayed_work_on - modify delay of or queue a delayed work on specific CPU @@ -2059,6 +2072,7 @@ static bool manage_workers(struct worker *worker)  	if (unlikely(!mutex_trylock(&pool->manager_mutex))) {  		spin_unlock_irq(&pool->lock);  		mutex_lock(&pool->manager_mutex); +		spin_lock_irq(&pool->lock);  		ret = true;  	} @@ -4085,6 +4099,10 @@ struct workqueue_struct *__alloc_workqueue_key(const char *fmt,  	struct workqueue_struct *wq;  	struct pool_workqueue *pwq; +	/* see the comment above the definition of WQ_POWER_EFFICIENT */ +	if ((flags & WQ_POWER_EFFICIENT) && wq_power_efficient) +		flags |= WQ_UNBOUND; +  	/* allocate wq and format name */  	if (flags & WQ_UNBOUND)  		tbl_size = wq_numa_tbl_len * sizeof(wq->numa_pwq_tbl[0]); @@ -4311,6 +4329,12 @@ bool current_is_workqueue_rescuer(void)   * no synchronization around this function and the test result is   * unreliable and only useful as advisory hints or for debugging.   * + * If @cpu is WORK_CPU_UNBOUND, the test is performed on the local CPU. + * Note that both per-cpu and unbound workqueues may be associated with + * multiple pool_workqueues which have separate congested states.  A + * workqueue being congested on one CPU doesn't mean the workqueue is also + * contested on other CPUs / NUMA nodes. + *   * RETURNS:   * %true if congested, %false otherwise.   */ @@ -4321,6 +4345,9 @@ bool workqueue_congested(int cpu, struct workqueue_struct *wq)  	rcu_read_lock_sched(); +	if (cpu == WORK_CPU_UNBOUND) +		cpu = smp_processor_id(); +  	if (!(wq->flags & WQ_UNBOUND))  		pwq = per_cpu_ptr(wq->cpu_pwqs, cpu);  	else @@ -4895,7 +4922,8 @@ static void __init wq_numa_init(void)  	BUG_ON(!tbl);  	for_each_node(node) -		BUG_ON(!alloc_cpumask_var_node(&tbl[node], GFP_KERNEL, node)); +		BUG_ON(!alloc_cpumask_var_node(&tbl[node], GFP_KERNEL, +				node_online(node) ? node : NUMA_NO_NODE));  	for_each_possible_cpu(cpu) {  		node = cpu_to_node(cpu); @@ -4974,8 +5002,15 @@ static int __init init_workqueues(void)  					    WQ_UNBOUND_MAX_ACTIVE);  	system_freezable_wq = alloc_workqueue("events_freezable",  					      WQ_FREEZABLE, 0); +	system_power_efficient_wq = alloc_workqueue("events_power_efficient", +					      WQ_POWER_EFFICIENT, 0); +	system_freezable_power_efficient_wq = alloc_workqueue("events_freezable_power_efficient", +					      WQ_FREEZABLE | WQ_POWER_EFFICIENT, +					      0);  	BUG_ON(!system_wq || !system_highpri_wq || !system_long_wq || -	       !system_unbound_wq || !system_freezable_wq); +	       !system_unbound_wq || !system_freezable_wq || +	       !system_power_efficient_wq || +	       !system_freezable_power_efficient_wq);  	return 0;  }  early_initcall(init_workqueues); | 
