diff options
| author | Ingo Molnar <mingo@kernel.org> | 2020-05-28 07:58:12 +0200 | 
|---|---|---|
| committer | Ingo Molnar <mingo@kernel.org> | 2020-05-28 07:58:12 +0200 | 
| commit | 0bffedbce90818228f554651baf8d7c75f2876d8 (patch) | |
| tree | 96101208e0200c70c6688737ac6596bdd0ed2950 /net/mptcp/options.c | |
| parent | c50c75e9b87946499a62bffc021e95c87a1d57cd (diff) | |
| parent | 9cb1fd0efd195590b828b9b865421ad345a4a145 (diff) | |
Merge tag 'v5.7-rc7' into perf/core, to pick up fixes
Signed-off-by: Ingo Molnar <mingo@kernel.org>
Diffstat (limited to 'net/mptcp/options.c')
| -rw-r--r-- | net/mptcp/options.c | 105 | 
1 files changed, 46 insertions, 59 deletions
| diff --git a/net/mptcp/options.c b/net/mptcp/options.c index faf57585b892..7793b6011fa7 100644 --- a/net/mptcp/options.c +++ b/net/mptcp/options.c @@ -7,6 +7,7 @@  #define pr_fmt(fmt) "MPTCP: " fmt  #include <linux/kernel.h> +#include <crypto/sha.h>  #include <net/tcp.h>  #include <net/mptcp.h>  #include "protocol.h" @@ -16,10 +17,10 @@ static bool mptcp_cap_flag_sha256(u8 flags)  	return (flags & MPTCP_CAP_FLAG_MASK) == MPTCP_CAP_HMAC_SHA256;  } -void mptcp_parse_option(const struct sk_buff *skb, const unsigned char *ptr, -			int opsize, struct tcp_options_received *opt_rx) +static void mptcp_parse_option(const struct sk_buff *skb, +			       const unsigned char *ptr, int opsize, +			       struct mptcp_options_received *mp_opt)  { -	struct mptcp_options_received *mp_opt = &opt_rx->mptcp;  	u8 subtype = *ptr >> 4;  	int expected_opsize;  	u8 version; @@ -283,12 +284,20 @@ void mptcp_parse_option(const struct sk_buff *skb, const unsigned char *ptr,  }  void mptcp_get_options(const struct sk_buff *skb, -		       struct tcp_options_received *opt_rx) +		       struct mptcp_options_received *mp_opt)  { -	const unsigned char *ptr;  	const struct tcphdr *th = tcp_hdr(skb); -	int length = (th->doff * 4) - sizeof(struct tcphdr); +	const unsigned char *ptr; +	int length; + +	/* initialize option status */ +	mp_opt->mp_capable = 0; +	mp_opt->mp_join = 0; +	mp_opt->add_addr = 0; +	mp_opt->rm_addr = 0; +	mp_opt->dss = 0; +	length = (th->doff * 4) - sizeof(struct tcphdr);  	ptr = (const unsigned char *)(th + 1);  	while (length > 0) { @@ -308,7 +317,7 @@ void mptcp_get_options(const struct sk_buff *skb,  			if (opsize > length)  				return;	/* don't parse partial options */  			if (opcode == TCPOPT_MPTCP) -				mptcp_parse_option(skb, ptr, opsize, opt_rx); +				mptcp_parse_option(skb, ptr, opsize, mp_opt);  			ptr += opsize - 2;  			length -= opsize;  		} @@ -344,28 +353,6 @@ bool mptcp_syn_options(struct sock *sk, const struct sk_buff *skb,  	return false;  } -void mptcp_rcv_synsent(struct sock *sk) -{ -	struct mptcp_subflow_context *subflow = mptcp_subflow_ctx(sk); -	struct tcp_sock *tp = tcp_sk(sk); - -	if (subflow->request_mptcp && tp->rx_opt.mptcp.mp_capable) { -		subflow->mp_capable = 1; -		subflow->can_ack = 1; -		subflow->remote_key = tp->rx_opt.mptcp.sndr_key; -		pr_debug("subflow=%p, remote_key=%llu", subflow, -			 subflow->remote_key); -	} else if (subflow->request_join && tp->rx_opt.mptcp.mp_join) { -		subflow->mp_join = 1; -		subflow->thmac = tp->rx_opt.mptcp.thmac; -		subflow->remote_nonce = tp->rx_opt.mptcp.nonce; -		pr_debug("subflow=%p, thmac=%llu, remote_nonce=%u", subflow, -			 subflow->thmac, subflow->remote_nonce); -	} else if (subflow->request_mptcp) { -		tcp_sk(sk)->is_mptcp = 0; -	} -} -  /* MP_JOIN client subflow must wait for 4th ack before sending any data:   * TCP can't schedule delack timer before the subflow is fully established.   * MPTCP uses the delack timer to do 3rd ack retransmissions @@ -549,7 +536,7 @@ static bool mptcp_established_options_dss(struct sock *sk, struct sk_buff *skb,  static u64 add_addr_generate_hmac(u64 key1, u64 key2, u8 addr_id,  				  struct in_addr *addr)  { -	u8 hmac[MPTCP_ADDR_HMAC_LEN]; +	u8 hmac[SHA256_DIGEST_SIZE];  	u8 msg[7];  	msg[0] = addr_id; @@ -559,14 +546,14 @@ static u64 add_addr_generate_hmac(u64 key1, u64 key2, u8 addr_id,  	mptcp_crypto_hmac_sha(key1, key2, msg, 7, hmac); -	return get_unaligned_be64(hmac); +	return get_unaligned_be64(&hmac[SHA256_DIGEST_SIZE - sizeof(u64)]);  }  #if IS_ENABLED(CONFIG_MPTCP_IPV6)  static u64 add_addr6_generate_hmac(u64 key1, u64 key2, u8 addr_id,  				   struct in6_addr *addr)  { -	u8 hmac[MPTCP_ADDR_HMAC_LEN]; +	u8 hmac[SHA256_DIGEST_SIZE];  	u8 msg[19];  	msg[0] = addr_id; @@ -576,7 +563,7 @@ static u64 add_addr6_generate_hmac(u64 key1, u64 key2, u8 addr_id,  	mptcp_crypto_hmac_sha(key1, key2, msg, 19, hmac); -	return get_unaligned_be64(hmac); +	return get_unaligned_be64(&hmac[SHA256_DIGEST_SIZE - sizeof(u64)]);  }  #endif @@ -709,7 +696,7 @@ static bool check_fully_established(struct mptcp_sock *msk, struct sock *sk,  	if (TCP_SKB_CB(skb)->seq != subflow->ssn_offset + 1)  		return subflow->mp_capable; -	if (mp_opt->use_ack) { +	if (mp_opt->dss && mp_opt->use_ack) {  		/* subflows are fully established as soon as we get any  		 * additional ack.  		 */ @@ -717,8 +704,6 @@ static bool check_fully_established(struct mptcp_sock *msk, struct sock *sk,  		goto fully_established;  	} -	WARN_ON_ONCE(subflow->can_ack); -  	/* If the first established packet does not contain MP_CAPABLE + data  	 * then fallback to TCP  	 */ @@ -728,6 +713,8 @@ static bool check_fully_established(struct mptcp_sock *msk, struct sock *sk,  		return false;  	} +	if (unlikely(!READ_ONCE(msk->pm.server_side))) +		pr_warn_once("bogus mpc option on established client sk");  	subflow->fully_established = 1;  	subflow->remote_key = mp_opt->sndr_key;  	subflow->can_ack = 1; @@ -819,41 +806,41 @@ void mptcp_incoming_options(struct sock *sk, struct sk_buff *skb,  {  	struct mptcp_subflow_context *subflow = mptcp_subflow_ctx(sk);  	struct mptcp_sock *msk = mptcp_sk(subflow->conn); -	struct mptcp_options_received *mp_opt; +	struct mptcp_options_received mp_opt;  	struct mptcp_ext *mpext; -	mp_opt = &opt_rx->mptcp; -	if (!check_fully_established(msk, sk, subflow, skb, mp_opt)) +	mptcp_get_options(skb, &mp_opt); +	if (!check_fully_established(msk, sk, subflow, skb, &mp_opt))  		return; -	if (mp_opt->add_addr && add_addr_hmac_valid(msk, mp_opt)) { +	if (mp_opt.add_addr && add_addr_hmac_valid(msk, &mp_opt)) {  		struct mptcp_addr_info addr; -		addr.port = htons(mp_opt->port); -		addr.id = mp_opt->addr_id; -		if (mp_opt->family == MPTCP_ADDR_IPVERSION_4) { +		addr.port = htons(mp_opt.port); +		addr.id = mp_opt.addr_id; +		if (mp_opt.family == MPTCP_ADDR_IPVERSION_4) {  			addr.family = AF_INET; -			addr.addr = mp_opt->addr; +			addr.addr = mp_opt.addr;  		}  #if IS_ENABLED(CONFIG_MPTCP_IPV6) -		else if (mp_opt->family == MPTCP_ADDR_IPVERSION_6) { +		else if (mp_opt.family == MPTCP_ADDR_IPVERSION_6) {  			addr.family = AF_INET6; -			addr.addr6 = mp_opt->addr6; +			addr.addr6 = mp_opt.addr6;  		}  #endif -		if (!mp_opt->echo) +		if (!mp_opt.echo)  			mptcp_pm_add_addr_received(msk, &addr); -		mp_opt->add_addr = 0; +		mp_opt.add_addr = 0;  	} -	if (!mp_opt->dss) +	if (!mp_opt.dss)  		return;  	/* we can't wait for recvmsg() to update the ack_seq, otherwise  	 * monodirectional flows will stuck  	 */ -	if (mp_opt->use_ack) -		update_una(msk, mp_opt); +	if (mp_opt.use_ack) +		update_una(msk, &mp_opt);  	mpext = skb_ext_add(skb, SKB_EXT_MPTCP);  	if (!mpext) @@ -861,8 +848,8 @@ void mptcp_incoming_options(struct sock *sk, struct sk_buff *skb,  	memset(mpext, 0, sizeof(*mpext)); -	if (mp_opt->use_map) { -		if (mp_opt->mpc_map) { +	if (mp_opt.use_map) { +		if (mp_opt.mpc_map) {  			/* this is an MP_CAPABLE carrying MPTCP data  			 * we know this map the first chunk of data  			 */ @@ -872,16 +859,16 @@ void mptcp_incoming_options(struct sock *sk, struct sk_buff *skb,  			mpext->subflow_seq = 1;  			mpext->dsn64 = 1;  			mpext->mpc_map = 1; +			mpext->data_fin = 0;  		} else { -			mpext->data_seq = mp_opt->data_seq; -			mpext->subflow_seq = mp_opt->subflow_seq; -			mpext->dsn64 = mp_opt->dsn64; +			mpext->data_seq = mp_opt.data_seq; +			mpext->subflow_seq = mp_opt.subflow_seq; +			mpext->dsn64 = mp_opt.dsn64; +			mpext->data_fin = mp_opt.data_fin;  		} -		mpext->data_len = mp_opt->data_len; +		mpext->data_len = mp_opt.data_len;  		mpext->use_map = 1;  	} - -	mpext->data_fin = mp_opt->data_fin;  }  void mptcp_write_options(__be32 *ptr, struct mptcp_out_options *opts) | 
