diff options
Diffstat (limited to 'net/ipv4/tcp.c')
| -rw-r--r-- | net/ipv4/tcp.c | 17 | 
1 files changed, 10 insertions, 7 deletions
| diff --git a/net/ipv4/tcp.c b/net/ipv4/tcp.c index 810cc164f795..6f0caf9a866d 100644 --- a/net/ipv4/tcp.c +++ b/net/ipv4/tcp.c @@ -2691,6 +2691,9 @@ int tcp_disconnect(struct sock *sk, int flags)  	tp->window_clamp = 0;  	tp->delivered = 0;  	tp->delivered_ce = 0; +	if (icsk->icsk_ca_ops->release) +		icsk->icsk_ca_ops->release(sk); +	memset(icsk->icsk_ca_priv, 0, sizeof(icsk->icsk_ca_priv));  	tcp_set_ca_state(sk, TCP_CA_Open);  	tp->is_sack_reneg = 0;  	tcp_clear_retrans(tp); @@ -3246,10 +3249,7 @@ static int do_tcp_setsockopt(struct sock *sk, int level,  #ifdef CONFIG_TCP_MD5SIG  	case TCP_MD5SIG:  	case TCP_MD5SIG_EXT: -		if ((1 << sk->sk_state) & (TCPF_CLOSE | TCPF_LISTEN)) -			err = tp->af_specific->md5_parse(sk, optname, optval, optlen); -		else -			err = -EINVAL; +		err = tp->af_specific->md5_parse(sk, optname, optval, optlen);  		break;  #endif  	case TCP_USER_TIMEOUT: @@ -4033,11 +4033,14 @@ EXPORT_SYMBOL(tcp_md5_hash_skb_data);  int tcp_md5_hash_key(struct tcp_md5sig_pool *hp, const struct tcp_md5sig_key *key)  { +	u8 keylen = READ_ONCE(key->keylen); /* paired with WRITE_ONCE() in tcp_md5_do_add */  	struct scatterlist sg; -	sg_init_one(&sg, key->key, key->keylen); -	ahash_request_set_crypt(hp->md5_req, &sg, NULL, key->keylen); -	return crypto_ahash_update(hp->md5_req); +	sg_init_one(&sg, key->key, keylen); +	ahash_request_set_crypt(hp->md5_req, &sg, NULL, keylen); + +	/* We use data_race() because tcp_md5_do_add() might change key->key under us */ +	return data_race(crypto_ahash_update(hp->md5_req));  }  EXPORT_SYMBOL(tcp_md5_hash_key); |