Commit 49e3cf44 authored by Tejun Heo's avatar Tejun Heo

workqueue: replace for_each_pwq_cpu() with for_each_pwq()

Introduce for_each_pwq() which iterates all pool_workqueues of a
workqueue using the recently added workqueue->pwqs list and replace
for_each_pwq_cpu() usages with it.

This is primarily to remove the single unbound CPU assumption from pwq
iteration for the scheduled unbound pools with custom attributes
support which would introduce multiple unbound pwqs per workqueue;
however, it also simplifies iterator users.

Note that pwq->pool initialization is moved to alloc_and_link_pwqs()
as that now is the only place which is explicitly handling the two pwq
types.

This patch doesn't introduce any visible behavior changes.
Signed-off-by: default avatarTejun Heo <tj@kernel.org>
Reviewed-by: default avatarLai Jiangshan <laijs@cn.fujitsu.com>
parent 30cdf249
...@@ -273,12 +273,6 @@ static inline int __next_wq_cpu(int cpu, const struct cpumask *mask, ...@@ -273,12 +273,6 @@ static inline int __next_wq_cpu(int cpu, const struct cpumask *mask,
return WORK_CPU_END; return WORK_CPU_END;
} }
static inline int __next_pwq_cpu(int cpu, const struct cpumask *mask,
struct workqueue_struct *wq)
{
return __next_wq_cpu(cpu, mask, !(wq->flags & WQ_UNBOUND) ? 1 : 2);
}
/* /*
* CPU iterators * CPU iterators
* *
...@@ -289,8 +283,6 @@ static inline int __next_pwq_cpu(int cpu, const struct cpumask *mask, ...@@ -289,8 +283,6 @@ static inline int __next_pwq_cpu(int cpu, const struct cpumask *mask,
* *
* for_each_wq_cpu() : possible CPUs + WORK_CPU_UNBOUND * for_each_wq_cpu() : possible CPUs + WORK_CPU_UNBOUND
* for_each_online_wq_cpu() : online CPUs + WORK_CPU_UNBOUND * for_each_online_wq_cpu() : online CPUs + WORK_CPU_UNBOUND
* for_each_pwq_cpu() : possible CPUs for bound workqueues,
* WORK_CPU_UNBOUND for unbound workqueues
*/ */
#define for_each_wq_cpu(cpu) \ #define for_each_wq_cpu(cpu) \
for ((cpu) = __next_wq_cpu(-1, cpu_possible_mask, 3); \ for ((cpu) = __next_wq_cpu(-1, cpu_possible_mask, 3); \
...@@ -302,10 +294,13 @@ static inline int __next_pwq_cpu(int cpu, const struct cpumask *mask, ...@@ -302,10 +294,13 @@ static inline int __next_pwq_cpu(int cpu, const struct cpumask *mask,
(cpu) < WORK_CPU_END; \ (cpu) < WORK_CPU_END; \
(cpu) = __next_wq_cpu((cpu), cpu_online_mask, 3)) (cpu) = __next_wq_cpu((cpu), cpu_online_mask, 3))
#define for_each_pwq_cpu(cpu, wq) \ /**
for ((cpu) = __next_pwq_cpu(-1, cpu_possible_mask, (wq)); \ * for_each_pwq - iterate through all pool_workqueues of the specified workqueue
(cpu) < WORK_CPU_END; \ * @pwq: iteration cursor
(cpu) = __next_pwq_cpu((cpu), cpu_possible_mask, (wq))) * @wq: the target workqueue
*/
#define for_each_pwq(pwq, wq) \
list_for_each_entry((pwq), &(wq)->pwqs, pwqs_node)
#ifdef CONFIG_DEBUG_OBJECTS_WORK #ifdef CONFIG_DEBUG_OBJECTS_WORK
...@@ -2505,15 +2500,14 @@ static bool flush_workqueue_prep_pwqs(struct workqueue_struct *wq, ...@@ -2505,15 +2500,14 @@ static bool flush_workqueue_prep_pwqs(struct workqueue_struct *wq,
int flush_color, int work_color) int flush_color, int work_color)
{ {
bool wait = false; bool wait = false;
unsigned int cpu; struct pool_workqueue *pwq;
if (flush_color >= 0) { if (flush_color >= 0) {
WARN_ON_ONCE(atomic_read(&wq->nr_pwqs_to_flush)); WARN_ON_ONCE(atomic_read(&wq->nr_pwqs_to_flush));
atomic_set(&wq->nr_pwqs_to_flush, 1); atomic_set(&wq->nr_pwqs_to_flush, 1);
} }
for_each_pwq_cpu(cpu, wq) { for_each_pwq(pwq, wq) {
struct pool_workqueue *pwq = get_pwq(cpu, wq);
struct worker_pool *pool = pwq->pool; struct worker_pool *pool = pwq->pool;
spin_lock_irq(&pool->lock); spin_lock_irq(&pool->lock);
...@@ -2712,7 +2706,7 @@ EXPORT_SYMBOL_GPL(flush_workqueue); ...@@ -2712,7 +2706,7 @@ EXPORT_SYMBOL_GPL(flush_workqueue);
void drain_workqueue(struct workqueue_struct *wq) void drain_workqueue(struct workqueue_struct *wq)
{ {
unsigned int flush_cnt = 0; unsigned int flush_cnt = 0;
unsigned int cpu; struct pool_workqueue *pwq;
/* /*
* __queue_work() needs to test whether there are drainers, is much * __queue_work() needs to test whether there are drainers, is much
...@@ -2726,8 +2720,7 @@ void drain_workqueue(struct workqueue_struct *wq) ...@@ -2726,8 +2720,7 @@ void drain_workqueue(struct workqueue_struct *wq)
reflush: reflush:
flush_workqueue(wq); flush_workqueue(wq);
for_each_pwq_cpu(cpu, wq) { for_each_pwq(pwq, wq) {
struct pool_workqueue *pwq = get_pwq(cpu, wq);
bool drained; bool drained;
spin_lock_irq(&pwq->pool->lock); spin_lock_irq(&pwq->pool->lock);
...@@ -3100,6 +3093,7 @@ int keventd_up(void) ...@@ -3100,6 +3093,7 @@ int keventd_up(void)
static int alloc_and_link_pwqs(struct workqueue_struct *wq) static int alloc_and_link_pwqs(struct workqueue_struct *wq)
{ {
bool highpri = wq->flags & WQ_HIGHPRI;
int cpu; int cpu;
if (!(wq->flags & WQ_UNBOUND)) { if (!(wq->flags & WQ_UNBOUND)) {
...@@ -3110,6 +3104,7 @@ static int alloc_and_link_pwqs(struct workqueue_struct *wq) ...@@ -3110,6 +3104,7 @@ static int alloc_and_link_pwqs(struct workqueue_struct *wq)
for_each_possible_cpu(cpu) { for_each_possible_cpu(cpu) {
struct pool_workqueue *pwq = get_pwq(cpu, wq); struct pool_workqueue *pwq = get_pwq(cpu, wq);
pwq->pool = get_std_worker_pool(cpu, highpri);
list_add_tail(&pwq->pwqs_node, &wq->pwqs); list_add_tail(&pwq->pwqs_node, &wq->pwqs);
} }
} else { } else {
...@@ -3120,6 +3115,7 @@ static int alloc_and_link_pwqs(struct workqueue_struct *wq) ...@@ -3120,6 +3115,7 @@ static int alloc_and_link_pwqs(struct workqueue_struct *wq)
return -ENOMEM; return -ENOMEM;
wq->pool_wq.single = pwq; wq->pool_wq.single = pwq;
pwq->pool = get_std_worker_pool(WORK_CPU_UNBOUND, highpri);
list_add_tail(&pwq->pwqs_node, &wq->pwqs); list_add_tail(&pwq->pwqs_node, &wq->pwqs);
} }
...@@ -3154,7 +3150,7 @@ struct workqueue_struct *__alloc_workqueue_key(const char *fmt, ...@@ -3154,7 +3150,7 @@ struct workqueue_struct *__alloc_workqueue_key(const char *fmt,
{ {
va_list args, args1; va_list args, args1;
struct workqueue_struct *wq; struct workqueue_struct *wq;
unsigned int cpu; struct pool_workqueue *pwq;
size_t namelen; size_t namelen;
/* determine namelen, allocate wq and format name */ /* determine namelen, allocate wq and format name */
...@@ -3195,11 +3191,8 @@ struct workqueue_struct *__alloc_workqueue_key(const char *fmt, ...@@ -3195,11 +3191,8 @@ struct workqueue_struct *__alloc_workqueue_key(const char *fmt,
if (alloc_and_link_pwqs(wq) < 0) if (alloc_and_link_pwqs(wq) < 0)
goto err; goto err;
for_each_pwq_cpu(cpu, wq) { for_each_pwq(pwq, wq) {
struct pool_workqueue *pwq = get_pwq(cpu, wq);
BUG_ON((unsigned long)pwq & WORK_STRUCT_FLAG_MASK); BUG_ON((unsigned long)pwq & WORK_STRUCT_FLAG_MASK);
pwq->pool = get_std_worker_pool(cpu, flags & WQ_HIGHPRI);
pwq->wq = wq; pwq->wq = wq;
pwq->flush_color = -1; pwq->flush_color = -1;
pwq->max_active = max_active; pwq->max_active = max_active;
...@@ -3234,8 +3227,8 @@ struct workqueue_struct *__alloc_workqueue_key(const char *fmt, ...@@ -3234,8 +3227,8 @@ struct workqueue_struct *__alloc_workqueue_key(const char *fmt,
spin_lock_irq(&workqueue_lock); spin_lock_irq(&workqueue_lock);
if (workqueue_freezing && wq->flags & WQ_FREEZABLE) if (workqueue_freezing && wq->flags & WQ_FREEZABLE)
for_each_pwq_cpu(cpu, wq) for_each_pwq(pwq, wq)
get_pwq(cpu, wq)->max_active = 0; pwq->max_active = 0;
list_add(&wq->list, &workqueues); list_add(&wq->list, &workqueues);
...@@ -3261,14 +3254,13 @@ EXPORT_SYMBOL_GPL(__alloc_workqueue_key); ...@@ -3261,14 +3254,13 @@ EXPORT_SYMBOL_GPL(__alloc_workqueue_key);
*/ */
void destroy_workqueue(struct workqueue_struct *wq) void destroy_workqueue(struct workqueue_struct *wq)
{ {
unsigned int cpu; struct pool_workqueue *pwq;
/* drain it before proceeding with destruction */ /* drain it before proceeding with destruction */
drain_workqueue(wq); drain_workqueue(wq);
/* sanity checks */ /* sanity checks */
for_each_pwq_cpu(cpu, wq) { for_each_pwq(pwq, wq) {
struct pool_workqueue *pwq = get_pwq(cpu, wq);
int i; int i;
for (i = 0; i < WORK_NR_COLORS; i++) for (i = 0; i < WORK_NR_COLORS; i++)
...@@ -3330,7 +3322,7 @@ static void pwq_set_max_active(struct pool_workqueue *pwq, int max_active) ...@@ -3330,7 +3322,7 @@ static void pwq_set_max_active(struct pool_workqueue *pwq, int max_active)
*/ */
void workqueue_set_max_active(struct workqueue_struct *wq, int max_active) void workqueue_set_max_active(struct workqueue_struct *wq, int max_active)
{ {
unsigned int cpu; struct pool_workqueue *pwq;
max_active = wq_clamp_max_active(max_active, wq->flags, wq->name); max_active = wq_clamp_max_active(max_active, wq->flags, wq->name);
...@@ -3338,8 +3330,7 @@ void workqueue_set_max_active(struct workqueue_struct *wq, int max_active) ...@@ -3338,8 +3330,7 @@ void workqueue_set_max_active(struct workqueue_struct *wq, int max_active)
wq->saved_max_active = max_active; wq->saved_max_active = max_active;
for_each_pwq_cpu(cpu, wq) { for_each_pwq(pwq, wq) {
struct pool_workqueue *pwq = get_pwq(cpu, wq);
struct worker_pool *pool = pwq->pool; struct worker_pool *pool = pwq->pool;
spin_lock(&pool->lock); spin_lock(&pool->lock);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment