1 /* 2 * Copyright (C) 2011 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 17 package android.net; 18 19 import android.annotation.IntDef; 20 import android.annotation.NonNull; 21 import android.annotation.Nullable; 22 import android.annotation.SystemApi; 23 import android.compat.annotation.UnsupportedAppUsage; 24 import android.os.Build; 25 import android.os.Parcel; 26 import android.os.Parcelable; 27 28 import com.android.net.module.util.NetUtils; 29 import com.android.net.module.util.NetworkStackConstants; 30 31 import java.lang.annotation.Retention; 32 import java.lang.annotation.RetentionPolicy; 33 import java.net.Inet4Address; 34 import java.net.Inet6Address; 35 import java.net.InetAddress; 36 import java.net.UnknownHostException; 37 import java.util.Collection; 38 import java.util.Objects; 39 40 /** 41 * Represents a network route. 42 * <p> 43 * This is used both to describe static network configuration and live network 44 * configuration information. 45 * 46 * A route contains three pieces of information: 47 * <ul> 48 * <li>a destination {@link IpPrefix} specifying the network destinations covered by this route. 49 * If this is {@code null} it indicates a default route of the address family (IPv4 or IPv6) 50 * implied by the gateway IP address. 51 * <li>a gateway {@link InetAddress} indicating the next hop to use. If this is {@code null} it 52 * indicates a directly-connected route. 53 * <li>an interface (which may be unspecified). 54 * </ul> 55 * Either the destination or the gateway may be {@code null}, but not both. If the 56 * destination and gateway are both specified, they must be of the same address family 57 * (IPv4 or IPv6). 58 */ 59 public final class RouteInfo implements Parcelable { 60 /** @hide */ 61 @IntDef(value = { 62 RTN_UNICAST, 63 RTN_UNREACHABLE, 64 RTN_THROW, 65 }) 66 @Retention(RetentionPolicy.SOURCE) 67 public @interface RouteType {} 68 69 /** 70 * The IP destination address for this route. 71 */ 72 @NonNull 73 private final IpPrefix mDestination; 74 75 /** 76 * The gateway address for this route. 77 */ 78 @UnsupportedAppUsage 79 @Nullable 80 private final InetAddress mGateway; 81 82 /** 83 * The interface for this route. 84 */ 85 @Nullable 86 private final String mInterface; 87 88 89 /** Unicast route. @hide */ 90 @SystemApi 91 public static final int RTN_UNICAST = 1; 92 93 /** Unreachable route. @hide */ 94 @SystemApi 95 public static final int RTN_UNREACHABLE = 7; 96 97 /** Throw route. @hide */ 98 @SystemApi 99 public static final int RTN_THROW = 9; 100 101 /** 102 * The type of this route; one of the RTN_xxx constants above. 103 */ 104 private final int mType; 105 106 /** 107 * The maximum transmission unit size for this route. 108 */ 109 private final int mMtu; 110 111 // Derived data members. 112 // TODO: remove these. 113 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023) 114 private final boolean mIsHost; 115 private final boolean mHasGateway; 116 117 /** 118 * Constructs a RouteInfo object. 119 * 120 * If destination is null, then gateway must be specified and the 121 * constructed route is either the IPv4 default route <code>0.0.0.0</code> 122 * if the gateway is an instance of {@link Inet4Address}, or the IPv6 default 123 * route <code>::/0</code> if gateway is an instance of 124 * {@link Inet6Address}. 125 * <p> 126 * destination and gateway may not both be null. 127 * 128 * @param destination the destination prefix 129 * @param gateway the IP address to route packets through 130 * @param iface the interface name to send packets on 131 * @param type the type of this route 132 * 133 * @hide 134 */ 135 @SystemApi 136 public RouteInfo(@Nullable IpPrefix destination, @Nullable InetAddress gateway, 137 @Nullable String iface, @RouteType int type) { 138 this(destination, gateway, iface, type, 0); 139 } 140 141 /** 142 * Constructs a RouteInfo object. 143 * 144 * If destination is null, then gateway must be specified and the 145 * constructed route is either the IPv4 default route <code>0.0.0.0</code> 146 * if the gateway is an instance of {@link Inet4Address}, or the IPv6 default 147 * route <code>::/0</code> if gateway is an instance of 148 * {@link Inet6Address}. 149 * <p> 150 * destination and gateway may not both be null. 151 * 152 * @param destination the destination prefix 153 * @param gateway the IP address to route packets through 154 * @param iface the interface name to send packets on 155 * @param type the type of this route 156 * @param mtu the maximum transmission unit size for this route 157 * 158 * @hide 159 */ 160 @SystemApi 161 public RouteInfo(@Nullable IpPrefix destination, @Nullable InetAddress gateway, 162 @Nullable String iface, @RouteType int type, int mtu) { 163 switch (type) { 164 case RTN_UNICAST: 165 case RTN_UNREACHABLE: 166 case RTN_THROW: 167 // TODO: It would be nice to ensure that route types that don't have nexthops or 168 // interfaces, such as unreachable or throw, can't be created if an interface or 169 // a gateway is specified. This is a bit too complicated to do at the moment 170 // because: 171 // 172 // - LinkProperties sets the interface on routes added to it, and modifies the 173 // interfaces of all the routes when its interface name changes. 174 // - Even when the gateway is null, we store a non-null gateway here. 175 // 176 // For now, we just rely on the code that sets routes to do things properly. 177 break; 178 default: 179 throw new IllegalArgumentException("Unknown route type " + type); 180 } 181 182 if (destination == null) { 183 if (gateway != null) { 184 if (gateway instanceof Inet4Address) { 185 destination = new IpPrefix(NetworkStackConstants.IPV4_ADDR_ANY, 0); 186 } else { 187 destination = new IpPrefix(NetworkStackConstants.IPV6_ADDR_ANY, 0); 188 } 189 } else { 190 // no destination, no gateway. invalid. 191 throw new IllegalArgumentException("Invalid arguments passed in: " + gateway + "," + 192 destination); 193 } 194 } 195 // TODO: set mGateway to null if there is no gateway. This is more correct, saves space, and 196 // matches the documented behaviour. Before we can do this we need to fix all callers (e.g., 197 // ConnectivityService) to stop doing things like r.getGateway().equals(), ... . 198 if (gateway == null) { 199 if (destination.getAddress() instanceof Inet4Address) { 200 gateway = NetworkStackConstants.IPV4_ADDR_ANY; 201 } else { 202 gateway = NetworkStackConstants.IPV6_ADDR_ANY; 203 } 204 } 205 mHasGateway = (!gateway.isAnyLocalAddress()); 206 207 if ((destination.getAddress() instanceof Inet4Address 208 && !(gateway instanceof Inet4Address)) 209 || (destination.getAddress() instanceof Inet6Address 210 && !(gateway instanceof Inet6Address))) { 211 throw new IllegalArgumentException("address family mismatch in RouteInfo constructor"); 212 } 213 mDestination = destination; // IpPrefix objects are immutable. 214 mGateway = gateway; // InetAddress objects are immutable. 215 mInterface = iface; // Strings are immutable. 216 mType = type; 217 mIsHost = isHost(); 218 mMtu = mtu; 219 } 220 221 /** 222 * Constructs a {@code RouteInfo} object. 223 * 224 * If destination is null, then gateway must be specified and the 225 * constructed route is either the IPv4 default route <code>0.0.0.0</code> 226 * if the gateway is an instance of {@link Inet4Address}, or the IPv6 default 227 * route <code>::/0</code> if gateway is an instance of {@link Inet6Address}. 228 * <p> 229 * Destination and gateway may not both be null. 230 * 231 * @param destination the destination address and prefix in an {@link IpPrefix} 232 * @param gateway the {@link InetAddress} to route packets through 233 * @param iface the interface name to send packets on 234 * 235 * @hide 236 */ 237 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) 238 public RouteInfo(@Nullable IpPrefix destination, @Nullable InetAddress gateway, 239 @Nullable String iface) { 240 this(destination, gateway, iface, RTN_UNICAST); 241 } 242 243 /** 244 * @hide 245 */ 246 @UnsupportedAppUsage 247 public RouteInfo(@Nullable LinkAddress destination, @Nullable InetAddress gateway, 248 @Nullable String iface) { 249 this(destination == null ? null : 250 new IpPrefix(destination.getAddress(), destination.getPrefixLength()), 251 gateway, iface); 252 } 253 254 /** 255 * Constructs a {@code RouteInfo} object. 256 * 257 * If destination is null, then gateway must be specified and the 258 * constructed route is either the IPv4 default route <code>0.0.0.0</code> 259 * if the gateway is an instance of {@link Inet4Address}, or the IPv6 default 260 * route <code>::/0</code> if gateway is an instance of {@link Inet6Address}. 261 * <p> 262 * Destination and gateway may not both be null. 263 * 264 * @param destination the destination address and prefix in an {@link IpPrefix} 265 * @param gateway the {@link InetAddress} to route packets through 266 * 267 * @hide 268 */ 269 public RouteInfo(@Nullable IpPrefix destination, @Nullable InetAddress gateway) { 270 this(destination, gateway, null); 271 } 272 273 /** 274 * @hide 275 * 276 * TODO: Remove this. 277 */ 278 @UnsupportedAppUsage 279 public RouteInfo(@Nullable LinkAddress destination, @Nullable InetAddress gateway) { 280 this(destination, gateway, null); 281 } 282 283 /** 284 * Constructs a default {@code RouteInfo} object. 285 * 286 * @param gateway the {@link InetAddress} to route packets through 287 * 288 * @hide 289 */ 290 @UnsupportedAppUsage 291 public RouteInfo(@NonNull InetAddress gateway) { 292 this((IpPrefix) null, gateway, null); 293 } 294 295 /** 296 * Constructs a {@code RouteInfo} object representing a direct connected subnet. 297 * 298 * @param destination the {@link IpPrefix} describing the address and prefix 299 * length of the subnet. 300 * 301 * @hide 302 */ 303 public RouteInfo(@NonNull IpPrefix destination) { 304 this(destination, null, null); 305 } 306 307 /** 308 * @hide 309 */ 310 public RouteInfo(@NonNull LinkAddress destination) { 311 this(destination, null, null); 312 } 313 314 /** 315 * @hide 316 */ 317 public RouteInfo(@NonNull IpPrefix destination, @RouteType int type) { 318 this(destination, null, null, type); 319 } 320 321 /** 322 * @hide 323 */ 324 public static RouteInfo makeHostRoute(@NonNull InetAddress host, @Nullable String iface) { 325 return makeHostRoute(host, null, iface); 326 } 327 328 /** 329 * @hide 330 */ 331 public static RouteInfo makeHostRoute(@Nullable InetAddress host, @Nullable InetAddress gateway, 332 @Nullable String iface) { 333 if (host == null) return null; 334 335 if (host instanceof Inet4Address) { 336 return new RouteInfo(new IpPrefix(host, 32), gateway, iface); 337 } else { 338 return new RouteInfo(new IpPrefix(host, 128), gateway, iface); 339 } 340 } 341 342 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023) 343 private boolean isHost() { 344 return (mDestination.getAddress() instanceof Inet4Address && 345 mDestination.getPrefixLength() == 32) || 346 (mDestination.getAddress() instanceof Inet6Address && 347 mDestination.getPrefixLength() == 128); 348 } 349 350 /** 351 * Retrieves the destination address and prefix length in the form of an {@link IpPrefix}. 352 * 353 * @return {@link IpPrefix} specifying the destination. This is never {@code null}. 354 */ 355 @NonNull 356 public IpPrefix getDestination() { 357 return mDestination; 358 } 359 360 /** 361 * TODO: Convert callers to use IpPrefix and then remove. 362 * @hide 363 */ 364 @NonNull 365 public LinkAddress getDestinationLinkAddress() { 366 return new LinkAddress(mDestination.getAddress(), mDestination.getPrefixLength()); 367 } 368 369 /** 370 * Retrieves the gateway or next hop {@link InetAddress} for this route. 371 * 372 * @return {@link InetAddress} specifying the gateway or next hop. This may be 373 * {@code null} for a directly-connected route." 374 */ 375 @Nullable 376 public InetAddress getGateway() { 377 return mGateway; 378 } 379 380 /** 381 * Retrieves the interface used for this route if specified, else {@code null}. 382 * 383 * @return The name of the interface used for this route. 384 */ 385 @Nullable 386 public String getInterface() { 387 return mInterface; 388 } 389 390 /** 391 * Retrieves the type of this route. 392 * 393 * @return The type of this route; one of the {@code RTN_xxx} constants defined in this class. 394 * 395 * @hide 396 */ 397 @SystemApi 398 @RouteType 399 public int getType() { 400 return mType; 401 } 402 403 /** 404 * Retrieves the MTU size for this route. 405 * 406 * @return The MTU size, or 0 if it has not been set. 407 * @hide 408 */ 409 @SystemApi 410 public int getMtu() { 411 return mMtu; 412 } 413 414 /** 415 * Indicates if this route is a default route (ie, has no destination specified). 416 * 417 * @return {@code true} if the destination has a prefix length of 0. 418 */ 419 public boolean isDefaultRoute() { 420 return mType == RTN_UNICAST && mDestination.getPrefixLength() == 0; 421 } 422 423 /** 424 * Indicates if this route is an unreachable default route. 425 * 426 * @return {@code true} if it's an unreachable route with prefix length of 0. 427 * @hide 428 */ 429 private boolean isUnreachableDefaultRoute() { 430 return mType == RTN_UNREACHABLE && mDestination.getPrefixLength() == 0; 431 } 432 433 /** 434 * Indicates if this route is an IPv4 default route. 435 * @hide 436 */ 437 public boolean isIPv4Default() { 438 return isDefaultRoute() && mDestination.getAddress() instanceof Inet4Address; 439 } 440 441 /** 442 * Indicates if this route is an IPv4 unreachable default route. 443 * @hide 444 */ 445 public boolean isIPv4UnreachableDefault() { 446 return isUnreachableDefaultRoute() && mDestination.getAddress() instanceof Inet4Address; 447 } 448 449 /** 450 * Indicates if this route is an IPv6 default route. 451 * @hide 452 */ 453 public boolean isIPv6Default() { 454 return isDefaultRoute() && mDestination.getAddress() instanceof Inet6Address; 455 } 456 457 /** 458 * Indicates if this route is an IPv6 unreachable default route. 459 * @hide 460 */ 461 public boolean isIPv6UnreachableDefault() { 462 return isUnreachableDefaultRoute() && mDestination.getAddress() instanceof Inet6Address; 463 } 464 465 /** 466 * Indicates if this route is a host route (ie, matches only a single host address). 467 * 468 * @return {@code true} if the destination has a prefix length of 32 or 128 for IPv4 or IPv6, 469 * respectively. 470 * @hide 471 */ 472 public boolean isHostRoute() { 473 return mIsHost; 474 } 475 476 /** 477 * Indicates if this route has a next hop ({@code true}) or is directly-connected 478 * ({@code false}). 479 * 480 * @return {@code true} if a gateway is specified 481 */ 482 public boolean hasGateway() { 483 return mHasGateway; 484 } 485 486 /** 487 * Determines whether the destination and prefix of this route includes the specified 488 * address. 489 * 490 * @param destination A {@link InetAddress} to test to see if it would match this route. 491 * @return {@code true} if the destination and prefix length cover the given address. 492 */ 493 public boolean matches(InetAddress destination) { 494 return mDestination.contains(destination); 495 } 496 497 /** 498 * Find the route from a Collection of routes that best matches a given address. 499 * May return null if no routes are applicable. 500 * @param routes a Collection of RouteInfos to chose from 501 * @param dest the InetAddress your trying to get to 502 * @return the RouteInfo from the Collection that best fits the given address 503 * 504 * @hide 505 */ 506 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) 507 @Nullable 508 public static RouteInfo selectBestRoute(Collection<RouteInfo> routes, InetAddress dest) { 509 return NetUtils.selectBestRoute(routes, dest); 510 } 511 512 /** 513 * Returns a human-readable description of this object. 514 */ 515 public String toString() { 516 String val = ""; 517 if (mDestination != null) val = mDestination.toString(); 518 if (mType == RTN_UNREACHABLE) { 519 val += " unreachable"; 520 } else if (mType == RTN_THROW) { 521 val += " throw"; 522 } else { 523 val += " ->"; 524 if (mGateway != null) val += " " + mGateway.getHostAddress(); 525 if (mInterface != null) val += " " + mInterface; 526 if (mType != RTN_UNICAST) { 527 val += " unknown type " + mType; 528 } 529 } 530 val += " mtu " + mMtu; 531 return val; 532 } 533 534 /** 535 * Compares this RouteInfo object against the specified object and indicates if they are equal. 536 * @return {@code true} if the objects are equal, {@code false} otherwise. 537 */ 538 public boolean equals(@Nullable Object obj) { 539 if (this == obj) return true; 540 541 if (!(obj instanceof RouteInfo)) return false; 542 543 RouteInfo target = (RouteInfo) obj; 544 545 return Objects.equals(mDestination, target.getDestination()) && 546 Objects.equals(mGateway, target.getGateway()) && 547 Objects.equals(mInterface, target.getInterface()) && 548 mType == target.getType() && mMtu == target.getMtu(); 549 } 550 551 /** 552 * A helper class that contains the destination, the gateway and the interface in a 553 * {@code RouteInfo}, used by {@link ConnectivityService#updateRoutes} or 554 * {@link LinkProperties#addRoute} to calculate the list to be updated. 555 * {@code RouteInfo} objects with different interfaces are treated as different routes because 556 * *usually* on Android different interfaces use different routing tables, and moving a route 557 * to a new routing table never constitutes an update, but is always a remove and an add. 558 * 559 * @hide 560 */ 561 public static class RouteKey { 562 @NonNull private final IpPrefix mDestination; 563 @Nullable private final InetAddress mGateway; 564 @Nullable private final String mInterface; 565 566 RouteKey(@NonNull IpPrefix destination, @Nullable InetAddress gateway, 567 @Nullable String iface) { 568 mDestination = destination; 569 mGateway = gateway; 570 mInterface = iface; 571 } 572 573 @Override 574 public boolean equals(@Nullable Object o) { 575 if (!(o instanceof RouteKey)) { 576 return false; 577 } 578 RouteKey p = (RouteKey) o; 579 // No need to do anything special for scoped addresses. Inet6Address#equals does not 580 // consider the scope ID, but the netd route IPCs (e.g., INetd#networkAddRouteParcel) 581 // and the kernel ignore scoped addresses both in the prefix and in the nexthop and only 582 // look at RTA_OIF. 583 return Objects.equals(p.mDestination, mDestination) 584 && Objects.equals(p.mGateway, mGateway) 585 && Objects.equals(p.mInterface, mInterface); 586 } 587 588 @Override 589 public int hashCode() { 590 return Objects.hash(mDestination, mGateway, mInterface); 591 } 592 } 593 594 /** 595 * Get {@code RouteKey} of this {@code RouteInfo}. 596 * @return a {@code RouteKey} object. 597 * 598 * @hide 599 */ 600 @NonNull 601 public RouteKey getRouteKey() { 602 return new RouteKey(mDestination, mGateway, mInterface); 603 } 604 605 /** 606 * Returns a hashcode for this <code>RouteInfo</code> object. 607 */ 608 public int hashCode() { 609 return (mDestination.hashCode() * 41) 610 + (mGateway == null ? 0 :mGateway.hashCode() * 47) 611 + (mInterface == null ? 0 :mInterface.hashCode() * 67) 612 + (mType * 71) + (mMtu * 89); 613 } 614 615 /** 616 * Implement the Parcelable interface 617 */ 618 public int describeContents() { 619 return 0; 620 } 621 622 /** 623 * Implement the Parcelable interface 624 */ 625 public void writeToParcel(Parcel dest, int flags) { 626 dest.writeParcelable(mDestination, flags); 627 byte[] gatewayBytes = (mGateway == null) ? null : mGateway.getAddress(); 628 dest.writeByteArray(gatewayBytes); 629 dest.writeString(mInterface); 630 dest.writeInt(mType); 631 dest.writeInt(mMtu); 632 } 633 634 /** 635 * Implement the Parcelable interface. 636 */ 637 public static final @android.annotation.NonNull Creator<RouteInfo> CREATOR = 638 new Creator<RouteInfo>() { 639 public RouteInfo createFromParcel(Parcel in) { 640 IpPrefix dest = in.readParcelable(null); 641 642 InetAddress gateway = null; 643 byte[] addr = in.createByteArray(); 644 try { 645 gateway = InetAddress.getByAddress(addr); 646 } catch (UnknownHostException e) {} 647 648 String iface = in.readString(); 649 int type = in.readInt(); 650 int mtu = in.readInt(); 651 652 return new RouteInfo(dest, gateway, iface, type, mtu); 653 } 654 655 public RouteInfo[] newArray(int size) { 656 return new RouteInfo[size]; 657 } 658 }; 659 } 660