blob: 71689f91972641f4c32e2d178d30113bfbb393f8 [file] [log] [blame]
Remi NGUYEN VAN678277c2021-05-11 13:37:06 +00001/*
2 * Copyright (C) 2010 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package android.net;
18
19import static android.net.RouteInfo.RTN_UNREACHABLE;
20
21import static com.android.testutils.MiscAsserts.assertEqualBothWays;
22import static com.android.testutils.MiscAsserts.assertFieldCountEquals;
23import static com.android.testutils.MiscAsserts.assertNotEqualEitherWay;
24import static com.android.testutils.ParcelUtils.assertParcelingIsLossless;
25
26import static org.junit.Assert.assertEquals;
27import static org.junit.Assert.assertFalse;
28import static org.junit.Assert.assertNotEquals;
29import static org.junit.Assert.assertNull;
30import static org.junit.Assert.assertTrue;
31import static org.junit.Assert.fail;
32
33import android.os.Build;
34
35import androidx.core.os.BuildCompat;
36import androidx.test.filters.SmallTest;
37import androidx.test.runner.AndroidJUnit4;
38
39import com.android.testutils.DevSdkIgnoreRule;
40import com.android.testutils.DevSdkIgnoreRule.IgnoreAfter;
41import com.android.testutils.DevSdkIgnoreRule.IgnoreUpTo;
42
43import org.junit.Rule;
44import org.junit.Test;
45import org.junit.runner.RunWith;
46
47import java.net.Inet4Address;
48import java.net.Inet6Address;
49import java.net.InetAddress;
50
51@RunWith(AndroidJUnit4.class)
52@SmallTest
53public class RouteInfoTest {
54 @Rule
55 public final DevSdkIgnoreRule ignoreRule = new DevSdkIgnoreRule();
56
57 private static final int INVALID_ROUTE_TYPE = -1;
58
59 private InetAddress Address(String addr) {
60 return InetAddresses.parseNumericAddress(addr);
61 }
62
63 private IpPrefix Prefix(String prefix) {
64 return new IpPrefix(prefix);
65 }
66
67 private static boolean isAtLeastR() {
68 // BuildCompat.isAtLeastR is documented to return false on release SDKs (including R)
69 return Build.VERSION.SDK_INT > Build.VERSION_CODES.Q || BuildCompat.isAtLeastR();
70 }
71
72 @Test
73 public void testConstructor() {
74 RouteInfo r;
75 // Invalid input.
76 try {
77 r = new RouteInfo((IpPrefix) null, null, "rmnet0");
78 fail("Expected RuntimeException: destination and gateway null");
79 } catch (RuntimeException e) { }
80
81 try {
82 r = new RouteInfo(Prefix("2001:db8:ace::/49"), Address("2001:db8::1"), "rmnet0",
83 INVALID_ROUTE_TYPE);
84 fail("Invalid route type should cause exception");
85 } catch (IllegalArgumentException e) { }
86
87 try {
88 r = new RouteInfo(Prefix("2001:db8:ace::/49"), Address("192.0.2.1"), "rmnet0",
89 RTN_UNREACHABLE);
90 fail("Address family mismatch should cause exception");
91 } catch (IllegalArgumentException e) { }
92
93 try {
94 r = new RouteInfo(Prefix("0.0.0.0/0"), Address("2001:db8::1"), "rmnet0",
95 RTN_UNREACHABLE);
96 fail("Address family mismatch should cause exception");
97 } catch (IllegalArgumentException e) { }
98
99 // Null destination is default route.
100 r = new RouteInfo((IpPrefix) null, Address("2001:db8::1"), null);
101 assertEquals(Prefix("::/0"), r.getDestination());
102 assertEquals(Address("2001:db8::1"), r.getGateway());
103 assertNull(r.getInterface());
104
105 r = new RouteInfo((IpPrefix) null, Address("192.0.2.1"), "wlan0");
106 assertEquals(Prefix("0.0.0.0/0"), r.getDestination());
107 assertEquals(Address("192.0.2.1"), r.getGateway());
108 assertEquals("wlan0", r.getInterface());
109
110 // Null gateway sets gateway to unspecified address (why?).
111 r = new RouteInfo(Prefix("2001:db8:beef:cafe::/48"), null, "lo");
112 assertEquals(Prefix("2001:db8:beef::/48"), r.getDestination());
113 assertEquals(Address("::"), r.getGateway());
114 assertEquals("lo", r.getInterface());
115
116 r = new RouteInfo(Prefix("192.0.2.5/24"), null);
117 assertEquals(Prefix("192.0.2.0/24"), r.getDestination());
118 assertEquals(Address("0.0.0.0"), r.getGateway());
119 assertNull(r.getInterface());
120 }
121
122 @Test
123 public void testMatches() {
124 class PatchedRouteInfo {
125 private final RouteInfo mRouteInfo;
126
127 public PatchedRouteInfo(IpPrefix destination, InetAddress gateway, String iface) {
128 mRouteInfo = new RouteInfo(destination, gateway, iface);
129 }
130
131 public boolean matches(InetAddress destination) {
132 return mRouteInfo.matches(destination);
133 }
134 }
135
136 PatchedRouteInfo r;
137
138 r = new PatchedRouteInfo(Prefix("2001:db8:f00::ace:d00d/127"), null, "rmnet0");
139 assertTrue(r.matches(Address("2001:db8:f00::ace:d00c")));
140 assertTrue(r.matches(Address("2001:db8:f00::ace:d00d")));
141 assertFalse(r.matches(Address("2001:db8:f00::ace:d00e")));
142 assertFalse(r.matches(Address("2001:db8:f00::bad:d00d")));
143 assertFalse(r.matches(Address("2001:4868:4860::8888")));
144 assertFalse(r.matches(Address("8.8.8.8")));
145
146 r = new PatchedRouteInfo(Prefix("192.0.2.0/23"), null, "wlan0");
147 assertTrue(r.matches(Address("192.0.2.43")));
148 assertTrue(r.matches(Address("192.0.3.21")));
149 assertFalse(r.matches(Address("192.0.0.21")));
150 assertFalse(r.matches(Address("8.8.8.8")));
151
152 PatchedRouteInfo ipv6Default = new PatchedRouteInfo(Prefix("::/0"), null, "rmnet0");
153 assertTrue(ipv6Default.matches(Address("2001:db8::f00")));
154 assertFalse(ipv6Default.matches(Address("192.0.2.1")));
155
156 PatchedRouteInfo ipv4Default = new PatchedRouteInfo(Prefix("0.0.0.0/0"), null, "rmnet0");
157 assertTrue(ipv4Default.matches(Address("255.255.255.255")));
158 assertTrue(ipv4Default.matches(Address("192.0.2.1")));
159 assertFalse(ipv4Default.matches(Address("2001:db8::f00")));
160 }
161
162 @Test
163 public void testEquals() {
164 // IPv4
165 RouteInfo r1 = new RouteInfo(Prefix("2001:db8:ace::/48"), Address("2001:db8::1"), "wlan0");
166 RouteInfo r2 = new RouteInfo(Prefix("2001:db8:ace::/48"), Address("2001:db8::1"), "wlan0");
167 assertEqualBothWays(r1, r2);
168
169 RouteInfo r3 = new RouteInfo(Prefix("2001:db8:ace::/49"), Address("2001:db8::1"), "wlan0");
170 RouteInfo r4 = new RouteInfo(Prefix("2001:db8:ace::/48"), Address("2001:db8::2"), "wlan0");
171 RouteInfo r5 = new RouteInfo(Prefix("2001:db8:ace::/48"), Address("2001:db8::1"), "rmnet0");
172 assertNotEqualEitherWay(r1, r3);
173 assertNotEqualEitherWay(r1, r4);
174 assertNotEqualEitherWay(r1, r5);
175
176 // IPv6
177 r1 = new RouteInfo(Prefix("192.0.2.0/25"), Address("192.0.2.1"), "wlan0");
178 r2 = new RouteInfo(Prefix("192.0.2.0/25"), Address("192.0.2.1"), "wlan0");
179 assertEqualBothWays(r1, r2);
180
181 r3 = new RouteInfo(Prefix("192.0.2.0/24"), Address("192.0.2.1"), "wlan0");
182 r4 = new RouteInfo(Prefix("192.0.2.0/25"), Address("192.0.2.2"), "wlan0");
183 r5 = new RouteInfo(Prefix("192.0.2.0/25"), Address("192.0.2.1"), "rmnet0");
184 assertNotEqualEitherWay(r1, r3);
185 assertNotEqualEitherWay(r1, r4);
186 assertNotEqualEitherWay(r1, r5);
187
188 // Interfaces (but not destinations or gateways) can be null.
189 r1 = new RouteInfo(Prefix("192.0.2.0/25"), Address("192.0.2.1"), null);
190 r2 = new RouteInfo(Prefix("192.0.2.0/25"), Address("192.0.2.1"), null);
191 r3 = new RouteInfo(Prefix("192.0.2.0/24"), Address("192.0.2.1"), "wlan0");
192 assertEqualBothWays(r1, r2);
193 assertNotEqualEitherWay(r1, r3);
194 }
195
196 @Test
197 public void testHostAndDefaultRoutes() {
198 RouteInfo r;
199
200 r = new RouteInfo(Prefix("0.0.0.0/0"), Address("0.0.0.0"), "wlan0");
201 assertFalse(r.isHostRoute());
202 assertTrue(r.isDefaultRoute());
203 assertTrue(r.isIPv4Default());
204 assertFalse(r.isIPv6Default());
205 if (isAtLeastR()) {
206 assertFalse(r.isIPv4UnreachableDefault());
207 assertFalse(r.isIPv6UnreachableDefault());
208 }
209
210 r = new RouteInfo(Prefix("::/0"), Address("::"), "wlan0");
211 assertFalse(r.isHostRoute());
212 assertTrue(r.isDefaultRoute());
213 assertFalse(r.isIPv4Default());
214 assertTrue(r.isIPv6Default());
215 if (isAtLeastR()) {
216 assertFalse(r.isIPv4UnreachableDefault());
217 assertFalse(r.isIPv6UnreachableDefault());
218 }
219
220 r = new RouteInfo(Prefix("192.0.2.0/24"), null, "wlan0");
221 assertFalse(r.isHostRoute());
222 assertFalse(r.isDefaultRoute());
223 assertFalse(r.isIPv4Default());
224 assertFalse(r.isIPv6Default());
225 if (isAtLeastR()) {
226 assertFalse(r.isIPv4UnreachableDefault());
227 assertFalse(r.isIPv6UnreachableDefault());
228 }
229
230 r = new RouteInfo(Prefix("2001:db8::/48"), null, "wlan0");
231 assertFalse(r.isHostRoute());
232 assertFalse(r.isDefaultRoute());
233 assertFalse(r.isIPv4Default());
234 assertFalse(r.isIPv6Default());
235 if (isAtLeastR()) {
236 assertFalse(r.isIPv4UnreachableDefault());
237 assertFalse(r.isIPv6UnreachableDefault());
238 }
239
240 r = new RouteInfo(Prefix("192.0.2.0/32"), Address("0.0.0.0"), "wlan0");
241 assertTrue(r.isHostRoute());
242 assertFalse(r.isDefaultRoute());
243 assertFalse(r.isIPv4Default());
244 assertFalse(r.isIPv6Default());
245 if (isAtLeastR()) {
246 assertFalse(r.isIPv4UnreachableDefault());
247 assertFalse(r.isIPv6UnreachableDefault());
248 }
249
250 r = new RouteInfo(Prefix("2001:db8::/128"), Address("::"), "wlan0");
251 assertTrue(r.isHostRoute());
252 assertFalse(r.isDefaultRoute());
253 assertFalse(r.isIPv4Default());
254 assertFalse(r.isIPv6Default());
255 if (isAtLeastR()) {
256 assertFalse(r.isIPv4UnreachableDefault());
257 assertFalse(r.isIPv6UnreachableDefault());
258 }
259
260 r = new RouteInfo(Prefix("192.0.2.0/32"), null, "wlan0");
261 assertTrue(r.isHostRoute());
262 assertFalse(r.isDefaultRoute());
263 assertFalse(r.isIPv4Default());
264 assertFalse(r.isIPv6Default());
265 if (isAtLeastR()) {
266 assertFalse(r.isIPv4UnreachableDefault());
267 assertFalse(r.isIPv6UnreachableDefault());
268 }
269
270 r = new RouteInfo(Prefix("2001:db8::/128"), null, "wlan0");
271 assertTrue(r.isHostRoute());
272 assertFalse(r.isDefaultRoute());
273 assertFalse(r.isIPv4Default());
274 assertFalse(r.isIPv6Default());
275 if (isAtLeastR()) {
276 assertFalse(r.isIPv4UnreachableDefault());
277 assertFalse(r.isIPv6UnreachableDefault());
278 }
279
280 r = new RouteInfo(Prefix("::/128"), Address("fe80::"), "wlan0");
281 assertTrue(r.isHostRoute());
282 assertFalse(r.isDefaultRoute());
283 assertFalse(r.isIPv4Default());
284 assertFalse(r.isIPv6Default());
285 if (isAtLeastR()) {
286 assertFalse(r.isIPv4UnreachableDefault());
287 assertFalse(r.isIPv6UnreachableDefault());
288 }
289
290 r = new RouteInfo(Prefix("0.0.0.0/32"), Address("192.0.2.1"), "wlan0");
291 assertTrue(r.isHostRoute());
292 assertFalse(r.isDefaultRoute());
293 assertFalse(r.isIPv4Default());
294 assertFalse(r.isIPv6Default());
295 if (isAtLeastR()) {
296 assertFalse(r.isIPv4UnreachableDefault());
297 assertFalse(r.isIPv6UnreachableDefault());
298 }
299
300 r = new RouteInfo(Prefix("0.0.0.0/32"), Address("192.0.2.1"), "wlan0");
301 assertTrue(r.isHostRoute());
302 assertFalse(r.isDefaultRoute());
303 assertFalse(r.isIPv4Default());
304 assertFalse(r.isIPv6Default());
305 if (isAtLeastR()) {
306 assertFalse(r.isIPv4UnreachableDefault());
307 assertFalse(r.isIPv6UnreachableDefault());
308 }
309
310 r = new RouteInfo(new IpPrefix(Inet4Address.ANY, 0), RTN_UNREACHABLE);
311 assertFalse(r.isHostRoute());
312 assertFalse(r.isDefaultRoute());
313 assertFalse(r.isIPv4Default());
314 assertFalse(r.isIPv6Default());
315 if (isAtLeastR()) {
316 assertTrue(r.isIPv4UnreachableDefault());
317 assertFalse(r.isIPv6UnreachableDefault());
318 }
319
320 r = new RouteInfo(new IpPrefix(Inet6Address.ANY, 0), RTN_UNREACHABLE);
321 assertFalse(r.isHostRoute());
322 assertFalse(r.isDefaultRoute());
323 assertFalse(r.isIPv4Default());
324 assertFalse(r.isIPv6Default());
325 if (isAtLeastR()) {
326 assertFalse(r.isIPv4UnreachableDefault());
327 assertTrue(r.isIPv6UnreachableDefault());
328 }
329 }
330
331 @Test
332 public void testTruncation() {
333 LinkAddress l;
334 RouteInfo r;
335
336 l = new LinkAddress("192.0.2.5/30");
337 r = new RouteInfo(l, Address("192.0.2.1"), "wlan0");
338 assertEquals("192.0.2.4", r.getDestination().getAddress().getHostAddress());
339
340 l = new LinkAddress("2001:db8:1:f::5/63");
341 r = new RouteInfo(l, Address("2001:db8:5::1"), "wlan0");
342 assertEquals("2001:db8:1:e::", r.getDestination().getAddress().getHostAddress());
343 }
344
345 // Make sure that creating routes to multicast addresses doesn't throw an exception. Even though
346 // there's nothing we can do with them, we don't want to crash if, e.g., someone calls
347 // requestRouteToHostAddress("230.0.0.0", MOBILE_HIPRI);
348 @Test
349 public void testMulticastRoute() {
350 RouteInfo r;
351 r = new RouteInfo(Prefix("230.0.0.0/32"), Address("192.0.2.1"), "wlan0");
352 r = new RouteInfo(Prefix("ff02::1/128"), Address("2001:db8::1"), "wlan0");
353 // No exceptions? Good.
354 }
355
356 @Test
357 public void testParceling() {
358 RouteInfo r;
359 r = new RouteInfo(Prefix("192.0.2.0/24"), Address("192.0.2.1"), null);
360 assertParcelingIsLossless(r);
361 r = new RouteInfo(Prefix("192.0.2.0/24"), null, "wlan0");
362 assertParcelingIsLossless(r);
363 r = new RouteInfo(Prefix("192.0.2.0/24"), Address("192.0.2.1"), "wlan0", RTN_UNREACHABLE);
364 assertParcelingIsLossless(r);
365 }
366
367 @Test @IgnoreUpTo(Build.VERSION_CODES.Q)
368 public void testMtuParceling() {
369 final RouteInfo r = new RouteInfo(Prefix("ff02::1/128"), Address("2001:db8::"), "testiface",
370 RTN_UNREACHABLE, 1450 /* mtu */);
371 assertParcelingIsLossless(r);
372 }
373
374 @Test @IgnoreAfter(Build.VERSION_CODES.Q)
375 public void testFieldCount_Q() {
376 assertFieldCountEquals(6, RouteInfo.class);
377 }
378
379 @Test @IgnoreUpTo(Build.VERSION_CODES.Q)
380 public void testFieldCount() {
381 // Make sure any new field is covered by the above parceling tests when changing this number
382 assertFieldCountEquals(7, RouteInfo.class);
383 }
384
385 @Test @IgnoreUpTo(Build.VERSION_CODES.Q)
386 public void testMtu() {
387 RouteInfo r;
388 r = new RouteInfo(Prefix("0.0.0.0/0"), Address("0.0.0.0"), "wlan0",
389 RouteInfo.RTN_UNICAST, 1500);
390 assertEquals(1500, r.getMtu());
391
392 r = new RouteInfo(Prefix("0.0.0.0/0"), Address("0.0.0.0"), "wlan0");
393 assertEquals(0, r.getMtu());
394 }
395
396 @Test @IgnoreUpTo(Build.VERSION_CODES.Q)
397 public void testRouteKey() {
398 RouteInfo.RouteKey k1, k2;
399 // Only prefix, null gateway and null interface
400 k1 = new RouteInfo(Prefix("2001:db8::/128"), null).getRouteKey();
401 k2 = new RouteInfo(Prefix("2001:db8::/128"), null).getRouteKey();
402 assertEquals(k1, k2);
403 assertEquals(k1.hashCode(), k2.hashCode());
404
405 // With prefix, gateway and interface. Type and MTU does not affect RouteKey equality
406 k1 = new RouteInfo(Prefix("192.0.2.0/24"), Address("192.0.2.1"), "wlan0",
407 RTN_UNREACHABLE, 1450).getRouteKey();
408 k2 = new RouteInfo(Prefix("192.0.2.0/24"), Address("192.0.2.1"), "wlan0",
409 RouteInfo.RTN_UNICAST, 1400).getRouteKey();
410 assertEquals(k1, k2);
411 assertEquals(k1.hashCode(), k2.hashCode());
412
413 // Different scope IDs are ignored by the kernel, so we consider them equal here too.
414 k1 = new RouteInfo(Prefix("2001:db8::/64"), Address("fe80::1%1"), "wlan0").getRouteKey();
415 k2 = new RouteInfo(Prefix("2001:db8::/64"), Address("fe80::1%2"), "wlan0").getRouteKey();
416 assertEquals(k1, k2);
417 assertEquals(k1.hashCode(), k2.hashCode());
418
419 // Different prefix
420 k1 = new RouteInfo(Prefix("192.0.2.0/24"), null).getRouteKey();
421 k2 = new RouteInfo(Prefix("192.0.3.0/24"), null).getRouteKey();
422 assertNotEquals(k1, k2);
423
424 // Different gateway
425 k1 = new RouteInfo(Prefix("ff02::1/128"), Address("2001:db8::1"), null).getRouteKey();
426 k2 = new RouteInfo(Prefix("ff02::1/128"), Address("2001:db8::2"), null).getRouteKey();
427 assertNotEquals(k1, k2);
428
429 // Different interface
430 k1 = new RouteInfo(Prefix("ff02::1/128"), null, "tun0").getRouteKey();
431 k2 = new RouteInfo(Prefix("ff02::1/128"), null, "tun1").getRouteKey();
432 assertNotEquals(k1, k2);
433 }
434}