git: 4bca16a4fb52 - stable/13 - netinet: allow UDP tunnels to be removed

From: Kristof Provost <kp_at_FreeBSD.org>
Date: Mon, 28 Feb 2022 15:39:08 UTC
The branch stable/13 has been updated by kp:

URL: https://cgit.FreeBSD.org/src/commit/?id=4bca16a4fb521e7918be1d2d517aae32b8602faa

commit 4bca16a4fb521e7918be1d2d517aae32b8602faa
Author:     Kristof Provost <kp@FreeBSD.org>
AuthorDate: 2022-02-15 10:49:39 +0000
Commit:     Kristof Provost <kp@FreeBSD.org>
CommitDate: 2022-02-28 15:38:05 +0000

    netinet: allow UDP tunnels to be removed
    
    udp_set_kernel_tunneling() rejects new callbacks if one is already set.
    Allow callbacks to be cleared. The use case for this is OpenVPN DCO,
    where the socket is opened by userspace and then adopted by the kernel
    to run the tunnel. If the DCO interface is removed but userspace does
    not close the socket (something the kernel cannot prevent) the installed
    callbacks could be called with an invalidated context.
    
    Allow new functions to be set, but only if they're NULL (i.e. allow the
    callback functions to be cleared).
    
    Reviewed by:    tuexen
    MFC after:      3 weeks
    Sponsored by:   Rubicon Communications, LLC ("Netgate")
    Differential Revision:  https://reviews.freebsd.org/D34288
    
    (cherry picked from commit 995cba5a0c9659e623b910429222ac2831a2ecca)
---
 sys/netinet/udp_usrreq.c | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/sys/netinet/udp_usrreq.c b/sys/netinet/udp_usrreq.c
index 8ed7cefe6623..cb7d10f68c48 100644
--- a/sys/netinet/udp_usrreq.c
+++ b/sys/netinet/udp_usrreq.c
@@ -1604,8 +1604,8 @@ udp_set_kernel_tunneling(struct socket *so, udp_tun_func_t f, udp_tun_icmp_t i,
 	KASSERT(inp != NULL, ("udp_set_kernel_tunneling: inp == NULL"));
 	INP_WLOCK(inp);
 	up = intoudpcb(inp);
-	if ((up->u_tun_func != NULL) ||
-	    (up->u_icmp_func != NULL)) {
+	if ((f != NULL || i != NULL) && ((up->u_tun_func != NULL) ||
+	    (up->u_icmp_func != NULL))) {
 		INP_WUNLOCK(inp);
 		return (EBUSY);
 	}