diff options
Diffstat (limited to 'net/unix')
| -rw-r--r-- | net/unix/af_unix.c | 37 | ||||
| -rw-r--r-- | net/unix/garbage.c | 9 | 
2 files changed, 36 insertions, 10 deletions
| diff --git a/net/unix/af_unix.c b/net/unix/af_unix.c index 5e695a9a609c..142f56770b77 100644 --- a/net/unix/af_unix.c +++ b/net/unix/af_unix.c @@ -2613,10 +2613,24 @@ static struct sk_buff *manage_oob(struct sk_buff *skb, struct sock *sk,  {  	struct unix_sock *u = unix_sk(sk); -	if (!unix_skb_len(skb) && !(flags & MSG_PEEK)) { -		skb_unlink(skb, &sk->sk_receive_queue); -		consume_skb(skb); -		skb = NULL; +	if (!unix_skb_len(skb)) { +		struct sk_buff *unlinked_skb = NULL; + +		spin_lock(&sk->sk_receive_queue.lock); + +		if (copied && (!u->oob_skb || skb == u->oob_skb)) { +			skb = NULL; +		} else if (flags & MSG_PEEK) { +			skb = skb_peek_next(skb, &sk->sk_receive_queue); +		} else { +			unlinked_skb = skb; +			skb = skb_peek_next(skb, &sk->sk_receive_queue); +			__skb_unlink(unlinked_skb, &sk->sk_receive_queue); +		} + +		spin_unlock(&sk->sk_receive_queue.lock); + +		consume_skb(unlinked_skb);  	} else {  		struct sk_buff *unlinked_skb = NULL; @@ -3093,12 +3107,23 @@ static int unix_ioctl(struct socket *sock, unsigned int cmd, unsigned long arg)  #if IS_ENABLED(CONFIG_AF_UNIX_OOB)  	case SIOCATMARK:  		{ +			struct unix_sock *u = unix_sk(sk);  			struct sk_buff *skb;  			int answ = 0; +			mutex_lock(&u->iolock); +  			skb = skb_peek(&sk->sk_receive_queue); -			if (skb && skb == READ_ONCE(unix_sk(sk)->oob_skb)) -				answ = 1; +			if (skb) { +				struct sk_buff *oob_skb = READ_ONCE(u->oob_skb); + +				if (skb == oob_skb || +				    (!oob_skb && !unix_skb_len(skb))) +					answ = 1; +			} + +			mutex_unlock(&u->iolock); +  			err = put_user(answ, (int __user *)arg);  		}  		break; diff --git a/net/unix/garbage.c b/net/unix/garbage.c index dfe94a90ece4..23efb78fe9ef 100644 --- a/net/unix/garbage.c +++ b/net/unix/garbage.c @@ -476,6 +476,7 @@ prev_vertex:  	}  	if (vertex->index == vertex->scc_index) { +		struct unix_vertex *v;  		struct list_head scc;  		bool scc_dead = true; @@ -486,15 +487,15 @@ prev_vertex:  		 */  		__list_cut_position(&scc, &vertex_stack, &vertex->scc_entry); -		list_for_each_entry_reverse(vertex, &scc, scc_entry) { +		list_for_each_entry_reverse(v, &scc, scc_entry) {  			/* Don't restart DFS from this vertex in unix_walk_scc(). */ -			list_move_tail(&vertex->entry, &unix_visited_vertices); +			list_move_tail(&v->entry, &unix_visited_vertices);  			/* Mark vertex as off-stack. */ -			vertex->index = unix_vertex_grouped_index; +			v->index = unix_vertex_grouped_index;  			if (scc_dead) -				scc_dead = unix_vertex_dead(vertex); +				scc_dead = unix_vertex_dead(v);  		}  		if (scc_dead) |