From patchwork Tue Dec 7 06:02:03 2021 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Arne Schwabe X-Patchwork-Id: 2135 Return-Path: Delivered-To: patchwork@openvpn.net Delivered-To: patchwork@openvpn.net Received: from director14.mail.ord1d.rsapps.net ([172.31.255.6]) by backend41.mail.ord1d.rsapps.net with LMTP id OOhnItCTr2EZUwAAqwncew (envelope-from ) for ; Tue, 07 Dec 2021 12:03:12 -0500 Received: from proxy16.mail.iad3b.rsapps.net ([172.31.255.6]) by director14.mail.ord1d.rsapps.net with LMTP id 4IUYBNGTr2FiaQAAeJ7fFg (envelope-from ) for ; Tue, 07 Dec 2021 12:03:13 -0500 Received: from smtp34.gate.iad3b ([172.31.255.6]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) by proxy16.mail.iad3b.rsapps.net with LMTPS id wGfMONCTr2HQQQAAPj+4aA (envelope-from ) for ; Tue, 07 Dec 2021 12:03:12 -0500 X-Spam-Threshold: 95 X-Spam-Score: 0 X-Spam-Flag: NO X-Virus-Scanned: OK X-Orig-To: openvpnslackdevel@openvpn.net X-Originating-Ip: [216.105.38.7] Authentication-Results: smtp34.gate.iad3b.rsapps.net; iprev=pass policy.iprev="216.105.38.7"; spf=pass smtp.mailfrom="openvpn-devel-bounces@lists.sourceforge.net" smtp.helo="lists.sourceforge.net"; dkim=fail (signature verification failed) header.d=sourceforge.net; dkim=fail (signature verification failed) header.d=sf.net; dmarc=none (p=nil; dis=none) header.from=rfc2549.org X-Suspicious-Flag: YES X-Classification-ID: 8eeb9896-577f-11ec-a204-5254005e8ddb-1-1 Received: from [216.105.38.7] ([216.105.38.7:60438] helo=lists.sourceforge.net) by smtp34.gate.iad3b.rsapps.net (envelope-from ) (ecelerity 4.2.38.62370 r(:)) with ESMTPS (cipher=DHE-RSA-AES256-GCM-SHA384) id 20/71-02284-FC39FA16; Tue, 07 Dec 2021 12:03:11 -0500 Received: from [127.0.0.1] (helo=sfs-ml-2.v29.lw.sourceforge.com) by sfs-ml-2.v29.lw.sourceforge.com with esmtp (Exim 4.94.2) (envelope-from ) id 1mudrD-0000th-Ld; Tue, 07 Dec 2021 17:02:23 +0000 Received: from [172.30.20.202] (helo=mx.sourceforge.net) by sfs-ml-2.v29.lw.sourceforge.com with esmtps (TLS1.2) tls TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 (Exim 4.94.2) (envelope-from ) id 1mudrB-0000t5-9g for openvpn-devel@lists.sourceforge.net; Tue, 07 Dec 2021 17:02:21 +0000 DKIM-Signature: v=1; a=rsa-sha256; q=dns/txt; c=relaxed/relaxed; d=sourceforge.net; s=x; h=Content-Transfer-Encoding:MIME-Version:References: In-Reply-To:Message-Id:Date:Subject:To:From:Sender:Reply-To:Cc:Content-Type: Content-ID:Content-Description:Resent-Date:Resent-From:Resent-Sender: Resent-To:Resent-Cc:Resent-Message-ID:List-Id:List-Help:List-Unsubscribe: List-Subscribe:List-Post:List-Owner:List-Archive; bh=cM+3kzRjSDDCMdC3Hi0TW8Vxq74RX0D35oxrtezLRb0=; b=HmEXrH62zMZiqvHv2lDFqm2q7U qG40hbtsoyeE5hKTiDzuIVgs4YqFxXB9Oe33kDTZQYBLcyobRaZ1FBVdV95OzL+gtpHRtgZoayxcE 1pXfpASs5WdKhKxg3anXSd1ZrAiQawWBABi8UugCwVkrDOHEuanc+tI4gIJQ7Rcfl3U8=; DKIM-Signature: v=1; a=rsa-sha256; q=dns/txt; c=relaxed/relaxed; d=sf.net; s=x ; h=Content-Transfer-Encoding:MIME-Version:References:In-Reply-To:Message-Id: Date:Subject:To:From:Sender:Reply-To:Cc:Content-Type:Content-ID: Content-Description:Resent-Date:Resent-From:Resent-Sender:Resent-To:Resent-Cc :Resent-Message-ID:List-Id:List-Help:List-Unsubscribe:List-Subscribe: List-Post:List-Owner:List-Archive; bh=cM+3kzRjSDDCMdC3Hi0TW8Vxq74RX0D35oxrtezLRb0=; b=d603nKgQIPYu1B3MB3Nvqnps7g CMFvXHSs/n4cL6s7xwFWslppbwXMkRuKwZjq24cfDdOoc18Z+Eof1PTYJQRxqv29SRHwQoKLuLjmX a5KlqoGjNSYUA7NwFbmL6h/cW9wPUTzRC1LnQwqLYGS1Hm69HZZF3MyBGX6jBMfUH+X8=; Received: from mail.blinkt.de ([192.26.174.232]) by sfi-mx-1.v28.lw.sourceforge.com with esmtps (TLSv1.2:ECDHE-RSA-AES256-GCM-SHA384:256) (Exim 4.92.3) id 1mudr9-007aKF-8c for openvpn-devel@lists.sourceforge.net; Tue, 07 Dec 2021 17:02:21 +0000 Received: from kamera.blinkt.de ([2001:638:502:390:20c:29ff:fec8:535c]) by mail.blinkt.de with smtp (Exim 4.94.2 (FreeBSD)) (envelope-from ) id 1mudr2-000IeE-9g for openvpn-devel@lists.sourceforge.net; Tue, 07 Dec 2021 18:02:12 +0100 Received: (nullmailer pid 3275924 invoked by uid 10006); Tue, 07 Dec 2021 17:02:12 -0000 From: Arne Schwabe To: openvpn-devel@lists.sourceforge.net Date: Tue, 7 Dec 2021 18:02:03 +0100 Message-Id: <20211207170211.3275837-14-arne@rfc2549.org> X-Mailer: git-send-email 2.25.1 In-Reply-To: <20211207170211.3275837-1-arne@rfc2549.org> References: <20211207170211.3275837-1-arne@rfc2549.org> MIME-Version: 1.0 X-Spam-Report: Spam detection software, running on the system "util-spamd-1.v13.lw.sourceforge.com", has NOT identified this incoming email as spam. The original message has been attached to this so you can view it or label similar future email. If you have any questions, see the administrator of that system for details. Content preview: The current mssfix parameter is a bit as it needs manual calculation of the allowable packet size and also the resulting MSS value does not take into account if IPv4 or IPv6 is used on the outer tunne [...] Content analysis details: (0.3 points, 6.0 required) pts rule name description ---- ---------------------- -------------------------------------------------- 0.2 HEADER_FROM_DIFFERENT_DOMAINS From and EnvelopeFrom 2nd level mail domains are different 0.0 SPF_HELO_NONE SPF: HELO does not publish an SPF Record 0.0 SPF_NONE SPF: sender does not publish an SPF Record X-Headers-End: 1mudr9-007aKF-8c Subject: [Openvpn-devel] [PATCH 13/21] Implement optional mtu parameter for mssfix X-BeenThere: openvpn-devel@lists.sourceforge.net X-Mailman-Version: 2.1.21 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: openvpn-devel-bounces@lists.sourceforge.net X-getmail-retrieved-from-mailbox: Inbox The current mssfix parameter is a bit as it needs manual calculation of the allowable packet size and also the resulting MSS value does not take into account if IPv4 or IPv6 is used on the outer tunnel. The mtu parameter fixes both of these problem by dynamically including the real overhead. The syntax and naming of the parater is chosen for compatiblity with OpenVPN3. Signed-off-by: Arne Schwabe --- Changes.rst | 6 ++++++ doc/man-sections/link-options.rst | 22 ++++++++++++++-------- src/openvpn/init.c | 7 ++++--- src/openvpn/mss.c | 29 ++++++++++++++++++++++++++++- src/openvpn/mss.h | 3 ++- src/openvpn/multi.c | 3 ++- src/openvpn/options.c | 13 +++++++++++-- src/openvpn/options.h | 5 ++++- src/openvpn/ssl.c | 11 +++++++---- src/openvpn/ssl.h | 5 ++++- 10 files changed, 82 insertions(+), 22 deletions(-) diff --git a/Changes.rst b/Changes.rst index b7d7f2054..cf6a2f86d 100644 --- a/Changes.rst +++ b/Changes.rst @@ -62,6 +62,12 @@ Optional ciphers in ``--data-ciphers`` Ciphers in ``--data-ciphers`` can now be prefixed with a ``?`` to mark those as optional and only use them if the SSL library supports them. + +Improved ``--mssfix`` calculation + The ``--mssfix`` option now allows an optional :code:`mtu` parameter to specify + that different overhead for IPv4/IPv6 should taken into account and the resulting + size is specified as the total size of the VPN packets including IP and UDP headers. + Deprecated features ------------------- ``inetd`` has been removed diff --git a/doc/man-sections/link-options.rst b/doc/man-sections/link-options.rst index b1ae4e75a..f41c0c4f1 100644 --- a/doc/man-sections/link-options.rst +++ b/doc/man-sections/link-options.rst @@ -110,19 +110,25 @@ the local and the remote host. (:code:`p2p`). OpenVPN 2.0 introduces a new mode (:code:`server`) which implements a multi-client server capability. ---mssfix max +--mssfix max [mtu] Announce to TCP sessions running over the tunnel that they should limit their send packet sizes such that after OpenVPN has encapsulated them, the resulting UDP packet size that OpenVPN sends to its peer will not exceed ``max`` bytes. The default value is :code:`1450`. - The ``max`` parameter is interpreted in the same way as the - ``--link-mtu`` parameter, i.e. the UDP packet size after encapsulation - overhead has been added in, but not including the UDP header itself. - Resulting packet would be at most 28 bytes larger for IPv4 and 48 bytes - for IPv6 (20/40 bytes for IP header and 8 bytes for UDP header). Default - value of 1450 allows IPv4 packets to be transmitted over a link with MTU - 1473 or higher without IP level fragmentation. + If the :code:`mtu` parameter is specified the ``max`` value is interpreted + as the resulting packet size of VPN packets including the IP and UDP header. + Use :code:`0` as max to use the default of :code:1450. Support for the + :code:`mtu` parameter was added with OpenVPN version 2.6.0. + + If the :code:`mtu` parameter is not specified, the ``max`` parameter + is interpreted in the same way as the ``--link-mtu`` parameter, i.e. + the UDP packet size after encapsulation overhead has been added in, but + not including the UDP header itself. Resulting packet would be at most 28 + bytes larger for IPv4 and 48 bytes for IPv6 (20/40 bytes for IP header and + 8 bytes for UDP header). Default value of 1450 allows IPv4 packets to be + transmitted over a link with MTU 1473 or higher without IP level + fragmentation. The ``--mssfix`` option only makes sense when you are using the UDP protocol for OpenVPN peer-to-peer communication, i.e. ``--proto udp``. diff --git a/src/openvpn/init.c b/src/openvpn/init.c index 0287dda35..b3653971c 100644 --- a/src/openvpn/init.c +++ b/src/openvpn/init.c @@ -2207,7 +2207,7 @@ do_deferred_p2p_ncp(struct context *c) #endif if (!tls_session_update_crypto_params(session, &c->options, &c->c2.frame, - frame_fragment)) + frame_fragment, get_link_socket_info(c))) { msg(D_TLS_ERRORS, "ERROR: failed to set crypto cipher"); return false; @@ -2322,7 +2322,7 @@ do_deferred_options(struct context *c, const unsigned int found) struct tls_session *session = &c->c2.tls_multi->session[TM_ACTIVE]; if (!tls_session_update_crypto_params(session, &c->options, &c->c2.frame, - frame_fragment)) + frame_fragment, get_link_socket_info(c))) { msg(D_TLS_ERRORS, "OPTIONS ERROR: failed to import crypto options"); return false; @@ -4238,7 +4238,8 @@ init_instance(struct context *c, const struct env_set *env, const unsigned int f #endif /* initialize dynamic MTU variable */ - frame_calculate_mssfix(&c->c2.frame, &c->c1.ks.key_type, &c->options); + frame_calculate_mssfix(&c->c2.frame, &c->c1.ks.key_type, &c->options, + get_link_socket_info(c)); /* bind the TCP/UDP socket */ if (c->mode == CM_P2P || c->mode == CM_TOP || c->mode == CM_CHILD_TCP) diff --git a/src/openvpn/mss.c b/src/openvpn/mss.c index e4311c42a..b1f10d53e 100644 --- a/src/openvpn/mss.c +++ b/src/openvpn/mss.c @@ -207,9 +207,30 @@ mss_fixup_dowork(struct buffer *buf, uint16_t maxmss) } } +static unsigned int +get_ip_encap_overhead(const struct options *options, + const struct link_socket_info *lsi) +{ + /* Add the overhead of the encapsulating IP packets */ + sa_family_t af; + if (lsi->lsa) + { + af = lsi->lsa->actual.dest.addr.sa.sa_family; + } + else + { + /* In the early init before the connection is established or we + * are in listen mode we can only make an educated guess + * from the af of the connection entry */ + af = options->ce.af; + } + return datagram_overhead(af, lsi->proto); +} + void frame_calculate_mssfix(struct frame *frame, struct key_type *kt, - const struct options *options) + const struct options *options, + struct link_socket_info *lsi) { if (options->ce.mssfix == 0) { @@ -236,6 +257,12 @@ frame_calculate_mssfix(struct frame *frame, struct key_type *kt, * * (RFC 879, section 7). */ + if (options->ce.mssfix_encap) + { + /* Add the overhead of the encapsulating IP packets */ + overhead += get_ip_encap_overhead(options, lsi); + } + /* Add 20 bytes for the IPv4 header and TCP header of the payload, * the mssfix routes will add 20 extra if payload is IPv6 */ overhead += 20 + 20; diff --git a/src/openvpn/mss.h b/src/openvpn/mss.h index 856f4c4e3..eecc79948 100644 --- a/src/openvpn/mss.h +++ b/src/openvpn/mss.h @@ -37,6 +37,7 @@ void mss_fixup_dowork(struct buffer *buf, uint16_t maxmss); /** Set the --mssfix option. */ void frame_calculate_mssfix(struct frame *frame, struct key_type *kt, - const struct options *options); + const struct options *options, + struct link_socket_info *lsi); #endif diff --git a/src/openvpn/multi.c b/src/openvpn/multi.c index e5ffebff2..67b7114ad 100644 --- a/src/openvpn/multi.c +++ b/src/openvpn/multi.c @@ -2286,7 +2286,8 @@ multi_client_generate_tls_keys(struct context *c) #endif struct tls_session *session = &c->c2.tls_multi->session[TM_ACTIVE]; if (!tls_session_update_crypto_params(session, &c->options, - &c->c2.frame, frame_fragment)) + &c->c2.frame, frame_fragment, + get_link_socket_info(c))) { msg(D_TLS_ERRORS, "TLS Error: initializing data channel failed"); register_signal(c, SIGUSR1, "process-push-msg-failed"); diff --git a/src/openvpn/options.c b/src/openvpn/options.c index 6dd573adb..8997fa988 100644 --- a/src/openvpn/options.c +++ b/src/openvpn/options.c @@ -6771,18 +6771,27 @@ add_option(struct options *options, VERIFY_PERMISSION(OPT_P_GENERAL); script_security_set(atoi(p[1])); } - else if (streq(p[0], "mssfix") && !p[2]) + else if (streq(p[0], "mssfix") && !p[3]) { VERIFY_PERMISSION(OPT_P_GENERAL|OPT_P_CONNECTION); if (p[1]) { options->ce.mssfix = positive_atoi(p[1]); } - else + + if (!p[1] || options->ce.mssfix == 0) { options->ce.mssfix_default = true; } + if (p[2] && streq(p[2], "mtu")) + { + options->ce.mssfix_encap = true; + } + else if (p[2]) + { + msg(msglevel, "Unknown parameter to --mssfix: %s", p[2]); + } } else if (streq(p[0], "disable-occ") && !p[1]) { diff --git a/src/openvpn/options.h b/src/openvpn/options.h index d4f41cd71..557edab9b 100644 --- a/src/openvpn/options.h +++ b/src/openvpn/options.h @@ -126,7 +126,10 @@ struct connection_entry int fragment; /* internal fragmentation size */ int mssfix; /* Upper bound on TCP MSS */ - bool mssfix_default; /* true if --mssfix was supplied without a parameter */ + bool mssfix_default; /* true if --mssfix was supplied without a parameter + * or 0 was specified as MTU */ + bool mssfix_encap; /* true if --mssfix had the "mtu" parameter to include + * overhead from IP and TCP/UDP encapsulation */ int explicit_exit_notification; /* Explicitly tell peer when we are exiting via OCC_EXIT or [RESTART] message */ diff --git a/src/openvpn/ssl.c b/src/openvpn/ssl.c index 41981d220..c5b085646 100644 --- a/src/openvpn/ssl.c +++ b/src/openvpn/ssl.c @@ -1883,7 +1883,8 @@ cleanup: bool tls_session_update_crypto_params_do_work(struct tls_session *session, struct options* options, struct frame *frame, - struct frame *frame_fragment) + struct frame *frame_fragment, + struct link_socket_info *lsi) { if (session->key[KS_PRIMARY].crypto_options.key_ctx_bi.initialized) { @@ -1913,7 +1914,7 @@ tls_session_update_crypto_params_do_work(struct tls_session *session, options->replay, packet_id_long_form); frame_finalize(frame, options->ce.link_mtu_defined, options->ce.link_mtu, options->ce.tun_mtu_defined, options->ce.tun_mtu); - frame_calculate_mssfix(frame, &session->opt->key_type, options); + frame_calculate_mssfix(frame, &session->opt->key_type, options, lsi); frame_print(frame, D_MTU_INFO, "Data Channel MTU parms"); /* @@ -1938,7 +1939,8 @@ tls_session_update_crypto_params_do_work(struct tls_session *session, bool tls_session_update_crypto_params(struct tls_session *session, struct options *options, struct frame *frame, - struct frame *frame_fragment) + struct frame *frame_fragment, + struct link_socket_info *lsi) { bool cipher_allowed_as_fallback = options->enable_ncp_fallback @@ -1957,7 +1959,8 @@ tls_session_update_crypto_params(struct tls_session *session, /* Import crypto settings that might be set by pull/push */ session->opt->crypto_flags |= options->data_channel_crypto_flags; - return tls_session_update_crypto_params_do_work(session, options, frame, frame_fragment); + return tls_session_update_crypto_params_do_work(session, options, frame, + frame_fragment, lsi); } diff --git a/src/openvpn/ssl.h b/src/openvpn/ssl.h index b14453fe2..e566acd81 100644 --- a/src/openvpn/ssl.h +++ b/src/openvpn/ssl.h @@ -508,13 +508,16 @@ void tls_update_remote_addr(struct tls_multi *multi, * @param frame The frame options for this session (frame overhead is * adjusted based on the selected cipher/auth). * @param frame_fragment The fragment frame options. + * @param lsi link socket info to adjust MTU related options + * depending on the current protocol * * @return true if updating succeeded or keys are already generated, false otherwise. */ bool tls_session_update_crypto_params(struct tls_session *session, struct options *options, struct frame *frame, - struct frame *frame_fragment); + struct frame *frame_fragment, + struct link_socket_info *lsi); /* * inline functions