Skip to content

Commit 20500c7

Browse files
dhowellsthangqn-ampere
authored andcommitted
rxrpc: Fix ICMP/ICMP6 error handling
[ Upstream commit ac56a0b ] Because rxrpc pretends to be a tunnel on top of a UDP/UDP6 socket, allowing it to siphon off UDP packets early in the handling of received UDP packets thereby avoiding the packet going through the UDP receive queue, it doesn't get ICMP packets through the UDP ->sk_error_report() callback. In fact, it doesn't appear that there's any usable option for getting hold of ICMP packets. Fix this by adding a new UDP encap hook to distribute error messages for UDP tunnels. If the hook is set, then the tunnel driver will be able to see ICMP packets. The hook provides the offset into the packet of the UDP header of the original packet that caused the notification. An alternative would be to call the ->error_handler() hook - but that requires that the skbuff be cloned (as ip_icmp_error() or ipv6_cmp_error() do, though isn't really necessary or desirable in rxrpc's case is we want to parse them there and then, not queue them). Changes ======= ver #3) - Fixed an uninitialised variable. ver #2) - Fixed some missing CONFIG_AF_RXRPC_IPV6 conditionals. Fixes: 5271953 ("rxrpc: Use the UDP encap_rcv hook") Signed-off-by: David Howells <dhowells@redhat.com> Signed-off-by: Sasha Levin <sashal@kernel.org>
1 parent 6ed5440 commit 20500c7

File tree

8 files changed

+270
-38
lines changed

8 files changed

+270
-38
lines changed

include/linux/udp.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,7 @@ struct udp_sock {
7575
* For encapsulation sockets.
7676
*/
7777
int (*encap_rcv)(struct sock *sk, struct sk_buff *skb);
78+
void (*encap_err_rcv)(struct sock *sk, struct sk_buff *skb, unsigned int udp_offset);
7879
int (*encap_err_lookup)(struct sock *sk, struct sk_buff *skb);
7980
void (*encap_destroy)(struct sock *sk);
8081

include/net/udp_tunnel.h

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -67,6 +67,9 @@ static inline int udp_sock_create(struct net *net,
6767
typedef int (*udp_tunnel_encap_rcv_t)(struct sock *sk, struct sk_buff *skb);
6868
typedef int (*udp_tunnel_encap_err_lookup_t)(struct sock *sk,
6969
struct sk_buff *skb);
70+
typedef void (*udp_tunnel_encap_err_rcv_t)(struct sock *sk,
71+
struct sk_buff *skb,
72+
unsigned int udp_offset);
7073
typedef void (*udp_tunnel_encap_destroy_t)(struct sock *sk);
7174
typedef struct sk_buff *(*udp_tunnel_gro_receive_t)(struct sock *sk,
7275
struct list_head *head,
@@ -80,6 +83,7 @@ struct udp_tunnel_sock_cfg {
8083
__u8 encap_type;
8184
udp_tunnel_encap_rcv_t encap_rcv;
8285
udp_tunnel_encap_err_lookup_t encap_err_lookup;
86+
udp_tunnel_encap_err_rcv_t encap_err_rcv;
8387
udp_tunnel_encap_destroy_t encap_destroy;
8488
udp_tunnel_gro_receive_t gro_receive;
8589
udp_tunnel_gro_complete_t gro_complete;

net/ipv4/udp.c

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -781,6 +781,8 @@ int __udp4_lib_err(struct sk_buff *skb, u32 info, struct udp_table *udptable)
781781
*/
782782
if (tunnel) {
783783
/* ...not for tunnels though: we don't have a sending socket */
784+
if (udp_sk(sk)->encap_err_rcv)
785+
udp_sk(sk)->encap_err_rcv(sk, skb, iph->ihl << 2);
784786
goto out;
785787
}
786788
if (!inet->recverr) {

net/ipv4/udp_tunnel_core.c

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,7 @@ void setup_udp_tunnel_sock(struct net *net, struct socket *sock,
7575

7676
udp_sk(sk)->encap_type = cfg->encap_type;
7777
udp_sk(sk)->encap_rcv = cfg->encap_rcv;
78+
udp_sk(sk)->encap_err_rcv = cfg->encap_err_rcv;
7879
udp_sk(sk)->encap_err_lookup = cfg->encap_err_lookup;
7980
udp_sk(sk)->encap_destroy = cfg->encap_destroy;
8081
udp_sk(sk)->gro_receive = cfg->gro_receive;

net/ipv6/udp.c

Lines changed: 4 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -614,8 +614,11 @@ int __udp6_lib_err(struct sk_buff *skb, struct inet6_skb_parm *opt,
614614
}
615615

616616
/* Tunnels don't have an application socket: don't pass errors back */
617-
if (tunnel)
617+
if (tunnel) {
618+
if (udp_sk(sk)->encap_err_rcv)
619+
udp_sk(sk)->encap_err_rcv(sk, skb, offset);
618620
goto out;
621+
}
619622

620623
if (!np->recverr) {
621624
if (!harderr || sk->sk_state != TCP_ESTABLISHED)

net/rxrpc/ar-internal.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -990,6 +990,7 @@ void rxrpc_send_keepalive(struct rxrpc_peer *);
990990
/*
991991
* peer_event.c
992992
*/
993+
void rxrpc_encap_err_rcv(struct sock *sk, struct sk_buff *skb, unsigned int udp_offset);
993994
void rxrpc_error_report(struct sock *);
994995
void rxrpc_peer_keepalive_worker(struct work_struct *);
995996

net/rxrpc/local_object.c

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -137,6 +137,7 @@ static int rxrpc_open_socket(struct rxrpc_local *local, struct net *net)
137137

138138
tuncfg.encap_type = UDP_ENCAP_RXRPC;
139139
tuncfg.encap_rcv = rxrpc_input_packet;
140+
tuncfg.encap_err_rcv = rxrpc_encap_err_rcv;
140141
tuncfg.sk_user_data = local;
141142
setup_udp_tunnel_sock(net, local->socket, &tuncfg);
142143

0 commit comments

Comments
 (0)