mirror of
https://github.com/followmsi/android_kernel_google_msm.git
synced 2024-11-06 23:17:41 +00:00
net: ping: Return EAFNOSUPPORT when appropriate.
1. For an IPv4 ping socket, ping_check_bind_addr does not check the family of the socket address that's passed in. Instead, make it behave like inet_bind, which enforces either that the address family is AF_INET, or that the family is AF_UNSPEC and the address is 0.0.0.0. 2. For an IPv6 ping socket, ping_check_bind_addr returns EINVAL if the socket family is not AF_INET6. Return EAFNOSUPPORT instead, for consistency with inet6_bind. 3. Make ping_v4_sendmsg and ping_v6_sendmsg return EAFNOSUPPORT instead of EINVAL if an incorrect socket address structure is passed in. 4. Make IPv6 ping sockets be IPv6-only. The code does not support IPv4, and it cannot easily be made to support IPv4 because the protocol numbers for ICMP and ICMPv6 are different. This makes connect(::ffff:192.0.2.1) fail with EAFNOSUPPORT instead of making the socket unusable. Among other things, this fixes an oops that can be triggered by: int s = socket(AF_INET, SOCK_DGRAM, IPPROTO_ICMP); struct sockaddr_in6 sin6 = { .sin6_family = AF_INET6, .sin6_addr = in6addr_any, }; bind(s, (struct sockaddr *) &sin6, sizeof(sin6)); [backport of net 9145736d4862145684009d6a72a6e61324a9439e] Change-Id: If06ca86d9f1e4593c0d6df174caca3487c57a241 Signed-off-by: Lorenzo Colitti <lorenzo@google.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
fd60bfcac9
commit
73c45b6b8f
2 changed files with 15 additions and 3 deletions
|
@ -255,6 +255,9 @@ int ping_init_sock(struct sock *sk)
|
|||
int i, j, count;
|
||||
int ret = 0;
|
||||
|
||||
if (sk->sk_family == AF_INET6)
|
||||
inet6_sk(sk)->ipv6only = 1;
|
||||
|
||||
inet_get_ping_group_range_net(net, range, range+1);
|
||||
if (range[0] <= group && group <= range[1])
|
||||
return 0;
|
||||
|
@ -302,6 +305,11 @@ int ping_check_bind_addr(struct sock *sk, struct inet_sock *isk,
|
|||
if (addr_len < sizeof(*addr))
|
||||
return -EINVAL;
|
||||
|
||||
if (addr->sin_family != AF_INET &&
|
||||
!(addr->sin_family == AF_UNSPEC &&
|
||||
addr->sin_addr.s_addr == htonl(INADDR_ANY)))
|
||||
return -EAFNOSUPPORT;
|
||||
|
||||
pr_debug("ping_check_bind_addr(sk=%p,addr=%pI4,port=%d)\n",
|
||||
sk, &addr->sin_addr.s_addr, ntohs(addr->sin_port));
|
||||
|
||||
|
@ -326,6 +334,9 @@ int ping_check_bind_addr(struct sock *sk, struct inet_sock *isk,
|
|||
if (addr_len < sizeof(*addr))
|
||||
return -EINVAL;
|
||||
|
||||
if (addr->sin6_family != AF_INET6)
|
||||
return -EAFNOSUPPORT;
|
||||
|
||||
pr_debug("ping_check_bind_addr(sk=%p,addr=%pI6c,port=%d)\n",
|
||||
sk, addr->sin6_addr.s6_addr, ntohs(addr->sin6_port));
|
||||
|
||||
|
@ -712,7 +723,7 @@ int ping_v4_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg,
|
|||
if (msg->msg_namelen < sizeof(*usin))
|
||||
return -EINVAL;
|
||||
if (usin->sin_family != AF_INET)
|
||||
return -EINVAL;
|
||||
return -EAFNOSUPPORT;
|
||||
daddr = usin->sin_addr.s_addr;
|
||||
/* no remote port */
|
||||
} else {
|
||||
|
|
|
@ -127,9 +127,10 @@ int ping_v6_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *msg,
|
|||
|
||||
if (msg->msg_name) {
|
||||
struct sockaddr_in6 *u = (struct sockaddr_in6 *) msg->msg_name;
|
||||
if (msg->msg_namelen < sizeof(struct sockaddr_in6) ||
|
||||
u->sin6_family != AF_INET6) {
|
||||
if (msg->msg_namelen < sizeof(*u))
|
||||
return -EINVAL;
|
||||
if (u->sin6_family != AF_INET6) {
|
||||
return -EAFNOSUPPORT;
|
||||
}
|
||||
if (sk->sk_bound_dev_if &&
|
||||
sk->sk_bound_dev_if != u->sin6_scope_id) {
|
||||
|
|
Loading…
Reference in a new issue