summaryrefslogtreecommitdiff
path: root/net-misc/iputils/files/iputils-20190709-ping-try-next-addrinfo-on-connect-failure.patch
blob: a308fb7c9f63fe341466d98ea89e087f72434e07 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
From: Benjamin Poirier <benjamin.poirier@gmail.com>
Date: Wed, 25 Dec 2019 13:33:12 +0900
Subject: ping: try next addrinfo on connect failure

On hosts that have routing rules matching on the outgoing interface [1],
getaddrinfo() may return results sorted in a suboptimal order because it is
not aware of the network interface passed to ping via the "-I" option.  In
particular, address reachability detection may fail and getaddrinfo() will
return ipv6 results first, even though the only routes available are ipv4.

Improve user experience by trying next addrinfo entry if we encounter a
failure at connect() time because of missing or unreachable routes.

[1] For example, on switches running Cumulus Linux, the default VRF is used
for front ports and a "mgmt" VRF is used for the management interface, which
also handles all DNS traffic.  (VRFs apply different routing rules based on
the iif/oif, ie.  influenced by SO_BINDTODEVICE.) In the default vrf, it's
possible to ping an ipv4 address via the mgmt vrf by specifying "-I mgmt".
However, that will fail if the target host is specified by name, has a AAAA
record and there is no ipv6 route to it.

Since libc commit 5ddb5bf5fb, getaddrinfo() does a udp connect to result
addresses to check if there is a route to them.  This is to implement
RFC3484 §6 Rule 1 ("Avoid unusable destinations") which is part of the
algorithm to order results.  getaddrinfo() is unaware of ping's "-I" option
and tries to connect its socket via the default vrf, which has no ipv6 route
to the target host (and, in fact, no ipv4 route either).  Following this
failure, getaddrinfo() returns results ordered according to
/etc/gai.conf (Rule 6) - by default, ipv6 first.

ping tries only the first entry returned by getaddrinfo() and fails to
connect to it because there is no ipv6 route to the host, even in the mgmt
vrf.  However, if getaddrinfo() had ordered the ipv4 result first or ping
had tried the next addrinfo entry (the ipv4 one), ping could connect a udp
socket to it and later successfully exchange icmp messages with it.

Example:

    cumulus@act-5812-10:~$ ip vrf list
    Name              Table
    -----------------------
    mgmt              1001
    cumulus@act-5812-10:~$ ip vrf identify
    cumulus@act-5812-10:~$ # --> default vrf
    cumulus@act-5812-10:~$
    cumulus@act-5812-10:~$ ip rule
    99:     from all to 10.230.0.53 ipproto udp dport 53 lookup mgmt
    99:     from all to 10.20.249.1 ipproto udp dport 53 lookup mgmt
    1000:   from all lookup [l3mdev-table]
    32765:  from all lookup local
    32766:  from all lookup main
    32767:  from all lookup default

    cumulus@act-5812-10:~$ ip route

    cumulus@act-5812-10:~$ ip -6 route
    ::1 dev lo proto kernel metric 256 pref medium

    cumulus@act-5812-10:~$ ip route show vrf mgmt
    default via 10.230.130.1 dev eth0
    unreachable default metric 4278198272
    10.230.130.0/24 dev eth0 proto kernel scope link src 10.230.130.211
    127.0.0.0/8 dev mgmt proto kernel scope link src 127.0.0.1

    cumulus@act-5812-10:~$ ip -6 route show vrf mgmt
    ::1 dev mgmt proto kernel metric 256 pref medium
    anycast fe80:: dev eth0 proto kernel metric 0 pref medium
    fe80::/64 dev eth0 proto kernel metric 256 pref medium
    ff00::/8 dev eth0 metric 256 pref medium
    unreachable default dev lo metric 4278198272 pref medium

    cumulus@act-5812-10:~$ host google.com
    google.com has address 172.217.0.46
    google.com has IPv6 address 2607:f8b0:4005:802::200e
    google.com mail is handled by 30 alt2.aspmx.l.google.com.
    google.com mail is handled by 40 alt3.aspmx.l.google.com.
    google.com mail is handled by 20 alt1.aspmx.l.google.com.
    google.com mail is handled by 10 aspmx.l.google.com.
    google.com mail is handled by 50 alt4.aspmx.l.google.com.

Success with numeric address

    cumulus@act-5812-10:~$ ping -n -c1 -I mgmt 172.217.0.46
    ping: Warning: source address might be selected on device other than mgmt.
    PING 172.217.0.46 (172.217.0.46) from 10.230.130.211 mgmt: 56(84) bytes of data.
    64 bytes from 172.217.0.46: icmp_seq=1 ttl=51 time=4.68 ms

    --- 172.217.0.46 ping statistics ---
    1 packets transmitted, 1 received, 0% packet loss, time 0ms
    rtt min/avg/max/mdev = 4.675/4.675/4.675/0.000 ms

Failure with host by name

    cumulus@act-5812-10:~$ ping -n -c1 -I mgmt google.com
    connect: No route to host

Success when running in the mgmt vrf because getaddrinfo()'s address
reachability test is effective and ipv4 result(s) are ordered first.

    cumulus@act-5812-10:~$ ip vrf exec mgmt ping -n -c1 google.com
    PING google.com (172.217.0.46) 56(84) bytes of data.
    64 bytes from 172.217.0.46: icmp_seq=1 ttl=51 time=4.65 ms

    --- google.com ping statistics ---
    1 packets transmitted, 1 received, 0% packet loss, time 0ms
    rtt min/avg/max/mdev = 4.650/4.650/4.650/0.000 ms

For demonstration purposes, the following configuration allows one to
reproduce a similar problem.  Starting from a host with a vanilla
configuration, default ipv4 route using eth0, no ipv6 global routes:

    root@vsid:~# ip route
    default via 192.168.15.1 dev eth0
    192.168.15.0/24 dev eth0 proto kernel scope link src 192.168.15.100

    root@vsid:~# ip -6 route
    ::1 dev lo proto kernel metric 256 pref medium
    fe80::/64 dev eth0 proto kernel metric 256 pref medium

    root@vsid:~# ip rou flush table main

    root@vsid:~# ip rou add table 1 192.168.15.0/24 dev eth0

    root@vsid:~# ip rou add table 1 default via 192.168.15.1

    root@vsid:~# ip rule
    0:    from all lookup local
    32766:  from all lookup main
    32767:  from all lookup default
    root@vsid:~# ip rule add pref 1 to 192.168.15.1 ipproto udp dport 53 lookup 1
    root@vsid:~# ip rule add pref 2 oif eth0 lookup 1
    root@vsid:~# ping -c1 -I eth0 google.com

    ping: connect: Network is unreachable

With the current patch

    root@vsid:~# /src/iputils/builddir/ping/ping -c1 -I eth0 google.com
    PING  (172.217.174.110) from 192.168.15.100 eth0: 56(84) bytes of data.
    64 bytes from nrt12s28-in-f14.1e100.net (172.217.174.110): icmp_seq=1 ttl=53 time=11.3 ms

    ---  ping statistics ---
    1 packets transmitted, 1 received, 0% packet loss, time 0ms
    rtt min/avg/max/mdev = 11.313/11.313/11.313/0.000 ms

Signed-off-by: Benjamin Poirier <bpoirier@cumulusnetworks.com>
Origin: https://github.com/iputils/iputils/commit/2705c8248281fbb8efaa5326ab1d0ed0a670bd3d
Bug-Debian: https://bugs.debian.org/947921
See-also: https://github.com/iputils/iputils/pull/244
---
 ping.c         | 3 +++
 ping6_common.c | 7 ++++++-
 2 files changed, 9 insertions(+), 1 deletion(-)

diff --git a/ping.c b/ping.c
index 34653be..013c4e6 100644
--- a/ping.c
+++ b/ping.c
@@ -628,6 +628,9 @@ int ping4_run(int argc, char **argv, struct addrinfo *ai, socket_st *sock)
 					error(2, errno, _("cannot set broadcasting"));
 				if (connect(probe_fd, (struct sockaddr *)&dst, sizeof(dst)) == -1)
 					error(2, errno, "connect");
+			} else if ((errno == EHOSTUNREACH || errno == ENETUNREACH) && ai->ai_next) {
+				close(probe_fd);
+				return -1;
 			} else
 				error(2, errno, "connect");
 		}
diff --git a/ping6_common.c b/ping6_common.c
index 6cc5404..bc1030b 100644
--- a/ping6_common.c
+++ b/ping6_common.c
@@ -651,8 +651,13 @@ int ping6_run(int argc, char **argv, struct addrinfo *ai, struct socket_st *sock
 			firsthop.sin6_family = AF_INET6;
 
 		firsthop.sin6_port = htons(1025);
-		if (connect(probe_fd, (struct sockaddr *)&firsthop, sizeof(firsthop)) == -1)
+		if (connect(probe_fd, (struct sockaddr *)&firsthop, sizeof(firsthop)) == -1) {
+			if ((errno == EHOSTUNREACH || errno == ENETUNREACH) && ai->ai_next) {
+				close(probe_fd);
+				return -1;
+			}
 			error(2, errno, "connect");
+		}
 		alen = sizeof source6;
 		if (getsockname(probe_fd, (struct sockaddr *)&source6, &alen) == -1)
 			error(2, errno, "getsockname");
-- 
2.25.0.rc2