diff options
Diffstat (limited to 'kernel/sched/wait.c')
| -rw-r--r-- | kernel/sched/wait.c | 22 | 
1 files changed, 20 insertions, 2 deletions
| diff --git a/kernel/sched/wait.c b/kernel/sched/wait.c index a6f00012c72e..20f27e2cf7ae 100644 --- a/kernel/sched/wait.c +++ b/kernel/sched/wait.c @@ -41,13 +41,31 @@ void add_wait_queue_priority(struct wait_queue_head *wq_head, struct wait_queue_  {  	unsigned long flags; -	wq_entry->flags |= WQ_FLAG_EXCLUSIVE | WQ_FLAG_PRIORITY; +	wq_entry->flags |= WQ_FLAG_PRIORITY;  	spin_lock_irqsave(&wq_head->lock, flags);  	__add_wait_queue(wq_head, wq_entry);  	spin_unlock_irqrestore(&wq_head->lock, flags);  }  EXPORT_SYMBOL_GPL(add_wait_queue_priority); +int add_wait_queue_priority_exclusive(struct wait_queue_head *wq_head, +				      struct wait_queue_entry *wq_entry) +{ +	struct list_head *head = &wq_head->head; + +	wq_entry->flags |= WQ_FLAG_EXCLUSIVE | WQ_FLAG_PRIORITY; + +	guard(spinlock_irqsave)(&wq_head->lock); + +	if (!list_empty(head) && +	    (list_first_entry(head, typeof(*wq_entry), entry)->flags & WQ_FLAG_PRIORITY)) +		return -EBUSY; + +	list_add(&wq_entry->entry, head); +	return 0; +} +EXPORT_SYMBOL_GPL(add_wait_queue_priority_exclusive); +  void remove_wait_queue(struct wait_queue_head *wq_head, struct wait_queue_entry *wq_entry)  {  	unsigned long flags; @@ -65,7 +83,7 @@ EXPORT_SYMBOL(remove_wait_queue);   * the non-exclusive tasks. Normally, exclusive tasks will be at the end of   * the list and any non-exclusive tasks will be woken first. A priority task   * may be at the head of the list, and can consume the event without any other - * tasks being woken. + * tasks being woken if it's also an exclusive task.   *   * There are circumstances in which we can try to wake a task which has already   * started to run but is not in state TASK_RUNNING. try_to_wake_up() returns | 
