mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-07 22:42:04 +00:00
ip: zero sockaddr returned on error queue
The sockaddr is returned in IP(V6)_RECVERR as part of errhdr. That structure is defined and allocated on the stack as struct { struct sock_extended_err ee; struct sockaddr_in(6) offender; } errhdr; The second part is only initialized for certain SO_EE_ORIGIN values. Always initialize it completely. An MTU exceeded error on a SOCK_RAW/IPPROTO_RAW is one example that would return uninitialized bytes. Signed-off-by: Willem de Bruijn <willemb@google.com> ---- Also verified that there is no padding between errhdr.ee and errhdr.offender that could leak additional kernel data. Acked-by: Eric Dumazet <edumazet@google.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
4315ef8d8b
commit
f812116b17
@ -461,17 +461,13 @@ int ip_recv_error(struct sock *sk, struct msghdr *msg, int len, int *addr_len)
|
|||||||
|
|
||||||
memcpy(&errhdr.ee, &serr->ee, sizeof(struct sock_extended_err));
|
memcpy(&errhdr.ee, &serr->ee, sizeof(struct sock_extended_err));
|
||||||
sin = &errhdr.offender;
|
sin = &errhdr.offender;
|
||||||
sin->sin_family = AF_UNSPEC;
|
memset(sin, 0, sizeof(*sin));
|
||||||
|
|
||||||
if (serr->ee.ee_origin == SO_EE_ORIGIN_ICMP ||
|
if (serr->ee.ee_origin == SO_EE_ORIGIN_ICMP ||
|
||||||
ipv4_pktinfo_prepare_errqueue(sk, skb, serr->ee.ee_origin)) {
|
ipv4_pktinfo_prepare_errqueue(sk, skb, serr->ee.ee_origin)) {
|
||||||
struct inet_sock *inet = inet_sk(sk);
|
|
||||||
|
|
||||||
sin->sin_family = AF_INET;
|
sin->sin_family = AF_INET;
|
||||||
sin->sin_addr.s_addr = ip_hdr(skb)->saddr;
|
sin->sin_addr.s_addr = ip_hdr(skb)->saddr;
|
||||||
sin->sin_port = 0;
|
if (inet_sk(sk)->cmsg_flags)
|
||||||
memset(&sin->sin_zero, 0, sizeof(sin->sin_zero));
|
|
||||||
if (inet->cmsg_flags)
|
|
||||||
ip_cmsg_recv(msg, skb);
|
ip_cmsg_recv(msg, skb);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -393,11 +393,10 @@ int ipv6_recv_error(struct sock *sk, struct msghdr *msg, int len, int *addr_len)
|
|||||||
|
|
||||||
memcpy(&errhdr.ee, &serr->ee, sizeof(struct sock_extended_err));
|
memcpy(&errhdr.ee, &serr->ee, sizeof(struct sock_extended_err));
|
||||||
sin = &errhdr.offender;
|
sin = &errhdr.offender;
|
||||||
sin->sin6_family = AF_UNSPEC;
|
memset(sin, 0, sizeof(*sin));
|
||||||
|
|
||||||
if (serr->ee.ee_origin != SO_EE_ORIGIN_LOCAL) {
|
if (serr->ee.ee_origin != SO_EE_ORIGIN_LOCAL) {
|
||||||
sin->sin6_family = AF_INET6;
|
sin->sin6_family = AF_INET6;
|
||||||
sin->sin6_flowinfo = 0;
|
|
||||||
sin->sin6_port = 0;
|
|
||||||
if (np->rxopt.all) {
|
if (np->rxopt.all) {
|
||||||
if (serr->ee.ee_origin != SO_EE_ORIGIN_ICMP &&
|
if (serr->ee.ee_origin != SO_EE_ORIGIN_ICMP &&
|
||||||
serr->ee.ee_origin != SO_EE_ORIGIN_ICMP6)
|
serr->ee.ee_origin != SO_EE_ORIGIN_ICMP6)
|
||||||
@ -412,12 +411,9 @@ int ipv6_recv_error(struct sock *sk, struct msghdr *msg, int len, int *addr_len)
|
|||||||
ipv6_iface_scope_id(&sin->sin6_addr,
|
ipv6_iface_scope_id(&sin->sin6_addr,
|
||||||
IP6CB(skb)->iif);
|
IP6CB(skb)->iif);
|
||||||
} else {
|
} else {
|
||||||
struct inet_sock *inet = inet_sk(sk);
|
|
||||||
|
|
||||||
ipv6_addr_set_v4mapped(ip_hdr(skb)->saddr,
|
ipv6_addr_set_v4mapped(ip_hdr(skb)->saddr,
|
||||||
&sin->sin6_addr);
|
&sin->sin6_addr);
|
||||||
sin->sin6_scope_id = 0;
|
if (inet_sk(sk)->cmsg_flags)
|
||||||
if (inet->cmsg_flags)
|
|
||||||
ip_cmsg_recv(msg, skb);
|
ip_cmsg_recv(msg, skb);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user