Skip to content

Commit f7cc889

Browse files
dcarattikuba-moo
authored andcommitted
mptcp: fix corrupt receiver key in MPC + data + checksum
using packetdrill it's possible to observe that the receiver key contains random values when clients transmit MP_CAPABLE with data and checksum (as specified in RFC8684 §3.1). Fix the layout of mptcp_out_options, to avoid using the skb extension copy when writing the MP_CAPABLE sub-option. Fixes: d7b2690 ("mptcp: shrink mptcp_out_options struct") Closes: multipath-tcp/mptcp_net-next#233 Reported-by: Poorva Sonparote <[email protected]> Signed-off-by: Davide Caratti <[email protected]> Signed-off-by: Mat Martineau <[email protected]> Link: https://lore.kernel.org/r/[email protected] Signed-off-by: Jakub Kicinski <[email protected]>
1 parent 27de809 commit f7cc889

File tree

2 files changed

+28
-15
lines changed

2 files changed

+28
-15
lines changed

include/net/mptcp.h

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,10 @@ struct mptcp_out_options {
6969
struct {
7070
u64 sndr_key;
7171
u64 rcvr_key;
72+
u64 data_seq;
73+
u32 subflow_seq;
74+
u16 data_len;
75+
__sum16 csum;
7276
};
7377
struct {
7478
struct mptcp_addr_info addr;

net/mptcp/options.c

Lines changed: 24 additions & 15 deletions
Original file line numberDiff line numberDiff line change
@@ -485,11 +485,11 @@ static bool mptcp_established_options_mp(struct sock *sk, struct sk_buff *skb,
485485
mpext = mptcp_get_ext(skb);
486486
data_len = mpext ? mpext->data_len : 0;
487487

488-
/* we will check ext_copy.data_len in mptcp_write_options() to
488+
/* we will check ops->data_len in mptcp_write_options() to
489489
* discriminate between TCPOLEN_MPTCP_MPC_ACK_DATA and
490490
* TCPOLEN_MPTCP_MPC_ACK
491491
*/
492-
opts->ext_copy.data_len = data_len;
492+
opts->data_len = data_len;
493493
opts->suboptions = OPTION_MPTCP_MPC_ACK;
494494
opts->sndr_key = subflow->local_key;
495495
opts->rcvr_key = subflow->remote_key;
@@ -505,9 +505,9 @@ static bool mptcp_established_options_mp(struct sock *sk, struct sk_buff *skb,
505505
len = TCPOLEN_MPTCP_MPC_ACK_DATA;
506506
if (opts->csum_reqd) {
507507
/* we need to propagate more info to csum the pseudo hdr */
508-
opts->ext_copy.data_seq = mpext->data_seq;
509-
opts->ext_copy.subflow_seq = mpext->subflow_seq;
510-
opts->ext_copy.csum = mpext->csum;
508+
opts->data_seq = mpext->data_seq;
509+
opts->subflow_seq = mpext->subflow_seq;
510+
opts->csum = mpext->csum;
511511
len += TCPOLEN_MPTCP_DSS_CHECKSUM;
512512
}
513513
*size = ALIGN(len, 4);
@@ -1227,7 +1227,7 @@ static void mptcp_set_rwin(const struct tcp_sock *tp)
12271227
WRITE_ONCE(msk->rcv_wnd_sent, ack_seq);
12281228
}
12291229

1230-
static u16 mptcp_make_csum(const struct mptcp_ext *mpext)
1230+
static u16 __mptcp_make_csum(u64 data_seq, u32 subflow_seq, u16 data_len, __sum16 sum)
12311231
{
12321232
struct csum_pseudo_header header;
12331233
__wsum csum;
@@ -1237,15 +1237,21 @@ static u16 mptcp_make_csum(const struct mptcp_ext *mpext)
12371237
* always the 64-bit value, irrespective of what length is used in the
12381238
* DSS option itself.
12391239
*/
1240-
header.data_seq = cpu_to_be64(mpext->data_seq);
1241-
header.subflow_seq = htonl(mpext->subflow_seq);
1242-
header.data_len = htons(mpext->data_len);
1240+
header.data_seq = cpu_to_be64(data_seq);
1241+
header.subflow_seq = htonl(subflow_seq);
1242+
header.data_len = htons(data_len);
12431243
header.csum = 0;
12441244

1245-
csum = csum_partial(&header, sizeof(header), ~csum_unfold(mpext->csum));
1245+
csum = csum_partial(&header, sizeof(header), ~csum_unfold(sum));
12461246
return (__force u16)csum_fold(csum);
12471247
}
12481248

1249+
static u16 mptcp_make_csum(const struct mptcp_ext *mpext)
1250+
{
1251+
return __mptcp_make_csum(mpext->data_seq, mpext->subflow_seq, mpext->data_len,
1252+
mpext->csum);
1253+
}
1254+
12491255
void mptcp_write_options(__be32 *ptr, const struct tcp_sock *tp,
12501256
struct mptcp_out_options *opts)
12511257
{
@@ -1337,7 +1343,7 @@ void mptcp_write_options(__be32 *ptr, const struct tcp_sock *tp,
13371343
len = TCPOLEN_MPTCP_MPC_SYN;
13381344
} else if (OPTION_MPTCP_MPC_SYNACK & opts->suboptions) {
13391345
len = TCPOLEN_MPTCP_MPC_SYNACK;
1340-
} else if (opts->ext_copy.data_len) {
1346+
} else if (opts->data_len) {
13411347
len = TCPOLEN_MPTCP_MPC_ACK_DATA;
13421348
if (opts->csum_reqd)
13431349
len += TCPOLEN_MPTCP_DSS_CHECKSUM;
@@ -1366,14 +1372,17 @@ void mptcp_write_options(__be32 *ptr, const struct tcp_sock *tp,
13661372

13671373
put_unaligned_be64(opts->rcvr_key, ptr);
13681374
ptr += 2;
1369-
if (!opts->ext_copy.data_len)
1375+
if (!opts->data_len)
13701376
goto mp_capable_done;
13711377

13721378
if (opts->csum_reqd) {
1373-
put_unaligned_be32(opts->ext_copy.data_len << 16 |
1374-
mptcp_make_csum(&opts->ext_copy), ptr);
1379+
put_unaligned_be32(opts->data_len << 16 |
1380+
__mptcp_make_csum(opts->data_seq,
1381+
opts->subflow_seq,
1382+
opts->data_len,
1383+
opts->csum), ptr);
13751384
} else {
1376-
put_unaligned_be32(opts->ext_copy.data_len << 16 |
1385+
put_unaligned_be32(opts->data_len << 16 |
13771386
TCPOPT_NOP << 8 | TCPOPT_NOP, ptr);
13781387
}
13791388
ptr += 1;

0 commit comments

Comments
 (0)