diff options
Diffstat (limited to 'include/linux/sched.h')
| -rw-r--r-- | include/linux/sched.h | 29 | 
1 files changed, 17 insertions, 12 deletions
| diff --git a/include/linux/sched.h b/include/linux/sched.h index 2b272382673d..f8188b833350 100644 --- a/include/linux/sched.h +++ b/include/linux/sched.h @@ -2152,6 +2152,8 @@ static inline struct mutex *__get_task_blocked_on(struct task_struct *p)  static inline void __set_task_blocked_on(struct task_struct *p, struct mutex *m)  { +	struct mutex *blocked_on = READ_ONCE(p->blocked_on); +  	WARN_ON_ONCE(!m);  	/* The task should only be setting itself as blocked */  	WARN_ON_ONCE(p != current); @@ -2162,8 +2164,8 @@ static inline void __set_task_blocked_on(struct task_struct *p, struct mutex *m)  	 * with a different mutex. Note, setting it to the same  	 * lock repeatedly is ok.  	 */ -	WARN_ON_ONCE(p->blocked_on && p->blocked_on != m); -	p->blocked_on = m; +	WARN_ON_ONCE(blocked_on && blocked_on != m); +	WRITE_ONCE(p->blocked_on, m);  }  static inline void set_task_blocked_on(struct task_struct *p, struct mutex *m) @@ -2174,16 +2176,19 @@ static inline void set_task_blocked_on(struct task_struct *p, struct mutex *m)  static inline void __clear_task_blocked_on(struct task_struct *p, struct mutex *m)  { -	WARN_ON_ONCE(!m); -	/* Currently we serialize blocked_on under the mutex::wait_lock */ -	lockdep_assert_held_once(&m->wait_lock); -	/* -	 * There may be cases where we re-clear already cleared -	 * blocked_on relationships, but make sure we are not -	 * clearing the relationship with a different lock. -	 */ -	WARN_ON_ONCE(m && p->blocked_on && p->blocked_on != m); -	p->blocked_on = NULL; +	if (m) { +		struct mutex *blocked_on = READ_ONCE(p->blocked_on); + +		/* Currently we serialize blocked_on under the mutex::wait_lock */ +		lockdep_assert_held_once(&m->wait_lock); +		/* +		 * There may be cases where we re-clear already cleared +		 * blocked_on relationships, but make sure we are not +		 * clearing the relationship with a different lock. +		 */ +		WARN_ON_ONCE(blocked_on && blocked_on != m); +	} +	WRITE_ONCE(p->blocked_on, NULL);  }  static inline void clear_task_blocked_on(struct task_struct *p, struct mutex *m) | 
