1 /*
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 
17 package android.net;
18 
19 import android.annotation.NonNull;
20 import android.annotation.Nullable;
21 import android.annotation.SystemApi;
22 import android.compat.annotation.UnsupportedAppUsage;
23 import android.os.Build;
24 import android.os.Parcel;
25 import android.os.Parcelable;
26 import android.text.TextUtils;
27 
28 import com.android.net.module.util.LinkPropertiesUtils;
29 
30 import java.net.Inet4Address;
31 import java.net.Inet6Address;
32 import java.net.InetAddress;
33 import java.net.UnknownHostException;
34 import java.util.ArrayList;
35 import java.util.Collection;
36 import java.util.Collections;
37 import java.util.Hashtable;
38 import java.util.List;
39 import java.util.Objects;
40 import java.util.StringJoiner;
41 
42 /**
43  * Describes the properties of a network link.
44  *
45  * A link represents a connection to a network.
46  * It may have multiple addresses and multiple gateways,
47  * multiple dns servers but only one http proxy and one
48  * network interface.
49  *
50  * Note that this is just a holder of data.  Modifying it
51  * does not affect live networks.
52  *
53  */
54 public final class LinkProperties implements Parcelable {
55     // The interface described by the network link.
56     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
57     private String mIfaceName;
58     private final ArrayList<LinkAddress> mLinkAddresses = new ArrayList<>();
59     private final ArrayList<InetAddress> mDnses = new ArrayList<>();
60     // PCSCF addresses are addresses of SIP proxies that only exist for the IMS core service.
61     private final ArrayList<InetAddress> mPcscfs = new ArrayList<InetAddress>();
62     private final ArrayList<InetAddress> mValidatedPrivateDnses = new ArrayList<>();
63     private boolean mUsePrivateDns;
64     private String mPrivateDnsServerName;
65     private String mDomains;
66     private ArrayList<RouteInfo> mRoutes = new ArrayList<>();
67     private Inet4Address mDhcpServerAddress;
68     private ProxyInfo mHttpProxy;
69     private int mMtu;
70     // in the format "rmem_min,rmem_def,rmem_max,wmem_min,wmem_def,wmem_max"
71     private String mTcpBufferSizes;
72     private IpPrefix mNat64Prefix;
73     private boolean mWakeOnLanSupported;
74     private Uri mCaptivePortalApiUrl;
75     private CaptivePortalData mCaptivePortalData;
76 
77     /**
78      * Indicates whether parceling should preserve fields that are set based on permissions of
79      * the process receiving the {@link LinkProperties}.
80      */
81     private final transient boolean mParcelSensitiveFields;
82 
83     private static final int MIN_MTU    = 68;
84 
85     private static final int MIN_MTU_V6 = 1280;
86 
87     private static final int MAX_MTU    = 10000;
88 
89     private static final int INET6_ADDR_LENGTH = 16;
90 
91     // Stores the properties of links that are "stacked" above this link.
92     // Indexed by interface name to allow modification and to prevent duplicates being added.
93     private Hashtable<String, LinkProperties> mStackedLinks = new Hashtable<>();
94 
95     /**
96      * @hide
97      */
98     @UnsupportedAppUsage(implicitMember =
99             "values()[Landroid/net/LinkProperties$ProvisioningChange;")
100     public enum ProvisioningChange {
101         @UnsupportedAppUsage
102         STILL_NOT_PROVISIONED,
103         @UnsupportedAppUsage
104         LOST_PROVISIONING,
105         @UnsupportedAppUsage
106         GAINED_PROVISIONING,
107         @UnsupportedAppUsage
108         STILL_PROVISIONED,
109     }
110 
111     /**
112      * Compare the provisioning states of two LinkProperties instances.
113      *
114      * @hide
115      */
116     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553)
117     public static ProvisioningChange compareProvisioning(
118             LinkProperties before, LinkProperties after) {
119         if (before.isProvisioned() && after.isProvisioned()) {
120             // On dual-stack networks, DHCPv4 renewals can occasionally fail.
121             // When this happens, IPv6-reachable services continue to function
122             // normally but IPv4-only services (naturally) fail.
123             //
124             // When an application using an IPv4-only service reports a bad
125             // network condition to the framework, attempts to re-validate
126             // the network succeed (since we support IPv6-only networks) and
127             // nothing is changed.
128             //
129             // For users, this is confusing and unexpected behaviour, and is
130             // not necessarily easy to diagnose.  Therefore, we treat changing
131             // from a dual-stack network to an IPv6-only network equivalent to
132             // a total loss of provisioning.
133             //
134             // For one such example of this, see b/18867306.
135             //
136             // Additionally, losing IPv6 provisioning can result in TCP
137             // connections getting stuck until timeouts fire and other
138             // baffling failures. Therefore, loss of either IPv4 or IPv6 on a
139             // previously dual-stack network is deemed a lost of provisioning.
140             if ((before.isIpv4Provisioned() && !after.isIpv4Provisioned())
141                     || (before.isIpv6Provisioned() && !after.isIpv6Provisioned())) {
142                 return ProvisioningChange.LOST_PROVISIONING;
143             }
144             return ProvisioningChange.STILL_PROVISIONED;
145         } else if (before.isProvisioned() && !after.isProvisioned()) {
146             return ProvisioningChange.LOST_PROVISIONING;
147         } else if (!before.isProvisioned() && after.isProvisioned()) {
148             return ProvisioningChange.GAINED_PROVISIONING;
149         } else {  // !before.isProvisioned() && !after.isProvisioned()
150             return ProvisioningChange.STILL_NOT_PROVISIONED;
151         }
152     }
153 
154     /**
155      * Constructs a new {@code LinkProperties} with default values.
156      */
157     public LinkProperties() {
158         mParcelSensitiveFields = false;
159     }
160 
161     /**
162      * @hide
163      */
164     @SystemApi
165     public LinkProperties(@Nullable LinkProperties source) {
166         this(source, false /* parcelSensitiveFields */);
167     }
168 
169     /**
170      * Create a copy of a {@link LinkProperties} that may preserve fields that were set
171      * based on the permissions of the process that originally received it.
172      *
173      * <p>By default {@link LinkProperties} does not preserve such fields during parceling, as
174      * they should not be shared outside of the process that receives them without appropriate
175      * checks.
176      * @param parcelSensitiveFields Whether the sensitive fields should be kept when parceling
177      * @hide
178      */
179     @SystemApi
180     public LinkProperties(@Nullable LinkProperties source, boolean parcelSensitiveFields) {
181         mParcelSensitiveFields = parcelSensitiveFields;
182         if (source == null) return;
183         mIfaceName = source.mIfaceName;
184         mLinkAddresses.addAll(source.mLinkAddresses);
185         mDnses.addAll(source.mDnses);
186         mValidatedPrivateDnses.addAll(source.mValidatedPrivateDnses);
187         mUsePrivateDns = source.mUsePrivateDns;
188         mPrivateDnsServerName = source.mPrivateDnsServerName;
189         mPcscfs.addAll(source.mPcscfs);
190         mDomains = source.mDomains;
191         mRoutes.addAll(source.mRoutes);
192         mHttpProxy = (source.mHttpProxy == null) ? null : new ProxyInfo(source.mHttpProxy);
193         for (LinkProperties l: source.mStackedLinks.values()) {
194             addStackedLink(l);
195         }
196         setMtu(source.mMtu);
197         setDhcpServerAddress(source.getDhcpServerAddress());
198         mTcpBufferSizes = source.mTcpBufferSizes;
199         mNat64Prefix = source.mNat64Prefix;
200         mWakeOnLanSupported = source.mWakeOnLanSupported;
201         mCaptivePortalApiUrl = source.mCaptivePortalApiUrl;
202         mCaptivePortalData = source.mCaptivePortalData;
203     }
204 
205     /**
206      * Sets the interface name for this link.  All {@link RouteInfo} already set for this
207      * will have their interface changed to match this new value.
208      *
209      * @param iface The name of the network interface used for this link.
210      */
211     public void setInterfaceName(@Nullable String iface) {
212         mIfaceName = iface;
213         ArrayList<RouteInfo> newRoutes = new ArrayList<>(mRoutes.size());
214         for (RouteInfo route : mRoutes) {
215             newRoutes.add(routeWithInterface(route));
216         }
217         mRoutes = newRoutes;
218     }
219 
220     /**
221      * Gets the interface name for this link.  May be {@code null} if not set.
222      *
223      * @return The interface name set for this link or {@code null}.
224      */
225     public @Nullable String getInterfaceName() {
226         return mIfaceName;
227     }
228 
229     /**
230      * @hide
231      */
232     @SystemApi
233     public @NonNull List<String> getAllInterfaceNames() {
234         List<String> interfaceNames = new ArrayList<>(mStackedLinks.size() + 1);
235         if (mIfaceName != null) interfaceNames.add(mIfaceName);
236         for (LinkProperties stacked: mStackedLinks.values()) {
237             interfaceNames.addAll(stacked.getAllInterfaceNames());
238         }
239         return interfaceNames;
240     }
241 
242     /**
243      * Returns all the addresses on this link.  We often think of a link having a single address,
244      * however, particularly with Ipv6 several addresses are typical.  Note that the
245      * {@code LinkProperties} actually contains {@link LinkAddress} objects which also include
246      * prefix lengths for each address.  This is a simplified utility alternative to
247      * {@link LinkProperties#getLinkAddresses}.
248      *
249      * @return An unmodifiable {@link List} of {@link InetAddress} for this link.
250      * @hide
251      */
252     @SystemApi
253     public @NonNull List<InetAddress> getAddresses() {
254         final List<InetAddress> addresses = new ArrayList<>();
255         for (LinkAddress linkAddress : mLinkAddresses) {
256             addresses.add(linkAddress.getAddress());
257         }
258         return Collections.unmodifiableList(addresses);
259     }
260 
261     /**
262      * Returns all the addresses on this link and all the links stacked above it.
263      * @hide
264      */
265     @UnsupportedAppUsage
266     public @NonNull List<InetAddress> getAllAddresses() {
267         List<InetAddress> addresses = new ArrayList<>();
268         for (LinkAddress linkAddress : mLinkAddresses) {
269             addresses.add(linkAddress.getAddress());
270         }
271         for (LinkProperties stacked: mStackedLinks.values()) {
272             addresses.addAll(stacked.getAllAddresses());
273         }
274         return addresses;
275     }
276 
277     private int findLinkAddressIndex(LinkAddress address) {
278         for (int i = 0; i < mLinkAddresses.size(); i++) {
279             if (mLinkAddresses.get(i).isSameAddressAs(address)) {
280                 return i;
281             }
282         }
283         return -1;
284     }
285 
286     /**
287      * Adds a {@link LinkAddress} to this {@code LinkProperties} if a {@link LinkAddress} of the
288      * same address/prefix does not already exist.  If it does exist it is replaced.
289      * @param address The {@code LinkAddress} to add.
290      * @return true if {@code address} was added or updated, false otherwise.
291      * @hide
292      */
293     @SystemApi
294     public boolean addLinkAddress(@NonNull LinkAddress address) {
295         if (address == null) {
296             return false;
297         }
298         int i = findLinkAddressIndex(address);
299         if (i < 0) {
300             // Address was not present. Add it.
301             mLinkAddresses.add(address);
302             return true;
303         } else if (mLinkAddresses.get(i).equals(address)) {
304             // Address was present and has same properties. Do nothing.
305             return false;
306         } else {
307             // Address was present and has different properties. Update it.
308             mLinkAddresses.set(i, address);
309             return true;
310         }
311     }
312 
313     /**
314      * Removes a {@link LinkAddress} from this {@code LinkProperties}.  Specifically, matches
315      * and {@link LinkAddress} with the same address and prefix.
316      *
317      * @param toRemove A {@link LinkAddress} specifying the address to remove.
318      * @return true if the address was removed, false if it did not exist.
319      * @hide
320      */
321     @SystemApi
322     public boolean removeLinkAddress(@NonNull LinkAddress toRemove) {
323         int i = findLinkAddressIndex(toRemove);
324         if (i >= 0) {
325             mLinkAddresses.remove(i);
326             return true;
327         }
328         return false;
329     }
330 
331     /**
332      * Returns all the {@link LinkAddress} on this link.  Typically a link will have
333      * one IPv4 address and one or more IPv6 addresses.
334      *
335      * @return An unmodifiable {@link List} of {@link LinkAddress} for this link.
336      */
337     public @NonNull List<LinkAddress> getLinkAddresses() {
338         return Collections.unmodifiableList(mLinkAddresses);
339     }
340 
341     /**
342      * Returns all the addresses on this link and all the links stacked above it.
343      * @hide
344      */
345     @SystemApi
346     public @NonNull List<LinkAddress> getAllLinkAddresses() {
347         List<LinkAddress> addresses = new ArrayList<>(mLinkAddresses);
348         for (LinkProperties stacked: mStackedLinks.values()) {
349             addresses.addAll(stacked.getAllLinkAddresses());
350         }
351         return addresses;
352     }
353 
354     /**
355      * Replaces the {@link LinkAddress} in this {@code LinkProperties} with
356      * the given {@link Collection} of {@link LinkAddress}.
357      *
358      * @param addresses The {@link Collection} of {@link LinkAddress} to set in this
359      *                  object.
360      */
361     public void setLinkAddresses(@NonNull Collection<LinkAddress> addresses) {
362         mLinkAddresses.clear();
363         for (LinkAddress address: addresses) {
364             addLinkAddress(address);
365         }
366     }
367 
368     /**
369      * Adds the given {@link InetAddress} to the list of DNS servers, if not present.
370      *
371      * @param dnsServer The {@link InetAddress} to add to the list of DNS servers.
372      * @return true if the DNS server was added, false if it was already present.
373      * @hide
374      */
375     @SystemApi
376     public boolean addDnsServer(@NonNull InetAddress dnsServer) {
377         if (dnsServer != null && !mDnses.contains(dnsServer)) {
378             mDnses.add(dnsServer);
379             return true;
380         }
381         return false;
382     }
383 
384     /**
385      * Removes the given {@link InetAddress} from the list of DNS servers.
386      *
387      * @param dnsServer The {@link InetAddress} to remove from the list of DNS servers.
388      * @return true if the DNS server was removed, false if it did not exist.
389      * @hide
390      */
391     @SystemApi
392     public boolean removeDnsServer(@NonNull InetAddress dnsServer) {
393         return mDnses.remove(dnsServer);
394     }
395 
396     /**
397      * Replaces the DNS servers in this {@code LinkProperties} with
398      * the given {@link Collection} of {@link InetAddress} objects.
399      *
400      * @param dnsServers The {@link Collection} of DNS servers to set in this object.
401      */
402     public void setDnsServers(@NonNull Collection<InetAddress> dnsServers) {
403         mDnses.clear();
404         for (InetAddress dnsServer: dnsServers) {
405             addDnsServer(dnsServer);
406         }
407     }
408 
409     /**
410      * Returns all the {@link InetAddress} for DNS servers on this link.
411      *
412      * @return An unmodifiable {@link List} of {@link InetAddress} for DNS servers on
413      *         this link.
414      */
415     public @NonNull List<InetAddress> getDnsServers() {
416         return Collections.unmodifiableList(mDnses);
417     }
418 
419     /**
420      * Set whether private DNS is currently in use on this network.
421      *
422      * @param usePrivateDns The private DNS state.
423      * @hide
424      */
425     @SystemApi
426     public void setUsePrivateDns(boolean usePrivateDns) {
427         mUsePrivateDns = usePrivateDns;
428     }
429 
430     /**
431      * Returns whether private DNS is currently in use on this network. When
432      * private DNS is in use, applications must not send unencrypted DNS
433      * queries as doing so could reveal private user information. Furthermore,
434      * if private DNS is in use and {@link #getPrivateDnsServerName} is not
435      * {@code null}, DNS queries must be sent to the specified DNS server.
436      *
437      * @return {@code true} if private DNS is in use, {@code false} otherwise.
438      */
439     public boolean isPrivateDnsActive() {
440         return mUsePrivateDns;
441     }
442 
443     /**
444      * Set the name of the private DNS server to which private DNS queries
445      * should be sent when in strict mode. This value should be {@code null}
446      * when private DNS is off or in opportunistic mode.
447      *
448      * @param privateDnsServerName The private DNS server name.
449      * @hide
450      */
451     @SystemApi
452     public void setPrivateDnsServerName(@Nullable String privateDnsServerName) {
453         mPrivateDnsServerName = privateDnsServerName;
454     }
455 
456     /**
457      * Set DHCP server address.
458      *
459      * @param serverAddress the server address to set.
460      */
461     public void setDhcpServerAddress(@Nullable Inet4Address serverAddress) {
462         mDhcpServerAddress = serverAddress;
463     }
464 
465      /**
466      * Get DHCP server address
467      *
468      * @return The current DHCP server address.
469      */
470     public @Nullable Inet4Address getDhcpServerAddress() {
471         return mDhcpServerAddress;
472     }
473 
474     /**
475      * Returns the private DNS server name that is in use. If not {@code null},
476      * private DNS is in strict mode. In this mode, applications should ensure
477      * that all DNS queries are encrypted and sent to this hostname and that
478      * queries are only sent if the hostname's certificate is valid. If
479      * {@code null} and {@link #isPrivateDnsActive} is {@code true}, private
480      * DNS is in opportunistic mode, and applications should ensure that DNS
481      * queries are encrypted and sent to a DNS server returned by
482      * {@link #getDnsServers}. System DNS will handle each of these cases
483      * correctly, but applications implementing their own DNS lookups must make
484      * sure to follow these requirements.
485      *
486      * @return The private DNS server name.
487      */
488     public @Nullable String getPrivateDnsServerName() {
489         return mPrivateDnsServerName;
490     }
491 
492     /**
493      * Adds the given {@link InetAddress} to the list of validated private DNS servers,
494      * if not present. This is distinct from the server name in that these are actually
495      * resolved addresses.
496      *
497      * @param dnsServer The {@link InetAddress} to add to the list of validated private DNS servers.
498      * @return true if the DNS server was added, false if it was already present.
499      * @hide
500      */
501     public boolean addValidatedPrivateDnsServer(@NonNull InetAddress dnsServer) {
502         if (dnsServer != null && !mValidatedPrivateDnses.contains(dnsServer)) {
503             mValidatedPrivateDnses.add(dnsServer);
504             return true;
505         }
506         return false;
507     }
508 
509     /**
510      * Removes the given {@link InetAddress} from the list of validated private DNS servers.
511      *
512      * @param dnsServer The {@link InetAddress} to remove from the list of validated private DNS
513      *        servers.
514      * @return true if the DNS server was removed, false if it did not exist.
515      * @hide
516      */
517     public boolean removeValidatedPrivateDnsServer(@NonNull InetAddress dnsServer) {
518         return mValidatedPrivateDnses.remove(dnsServer);
519     }
520 
521     /**
522      * Replaces the validated private DNS servers in this {@code LinkProperties} with
523      * the given {@link Collection} of {@link InetAddress} objects.
524      *
525      * @param dnsServers The {@link Collection} of validated private DNS servers to set in this
526      *        object.
527      * @hide
528      */
529     @SystemApi
530     public void setValidatedPrivateDnsServers(@NonNull Collection<InetAddress> dnsServers) {
531         mValidatedPrivateDnses.clear();
532         for (InetAddress dnsServer: dnsServers) {
533             addValidatedPrivateDnsServer(dnsServer);
534         }
535     }
536 
537     /**
538      * Returns all the {@link InetAddress} for validated private DNS servers on this link.
539      * These are resolved from the private DNS server name.
540      *
541      * @return An unmodifiable {@link List} of {@link InetAddress} for validated private
542      *         DNS servers on this link.
543      * @hide
544      */
545     @SystemApi
546     public @NonNull List<InetAddress> getValidatedPrivateDnsServers() {
547         return Collections.unmodifiableList(mValidatedPrivateDnses);
548     }
549 
550     /**
551      * Adds the given {@link InetAddress} to the list of PCSCF servers, if not present.
552      *
553      * @param pcscfServer The {@link InetAddress} to add to the list of PCSCF servers.
554      * @return true if the PCSCF server was added, false otherwise.
555      * @hide
556      */
557     @SystemApi
558     public boolean addPcscfServer(@NonNull InetAddress pcscfServer) {
559         if (pcscfServer != null && !mPcscfs.contains(pcscfServer)) {
560             mPcscfs.add(pcscfServer);
561             return true;
562         }
563         return false;
564     }
565 
566     /**
567      * Removes the given {@link InetAddress} from the list of PCSCF servers.
568      *
569      * @param pcscfServer The {@link InetAddress} to remove from the list of PCSCF servers.
570      * @return true if the PCSCF server was removed, false otherwise.
571      * @hide
572      */
573     public boolean removePcscfServer(@NonNull InetAddress pcscfServer) {
574         return mPcscfs.remove(pcscfServer);
575     }
576 
577     /**
578      * Replaces the PCSCF servers in this {@code LinkProperties} with
579      * the given {@link Collection} of {@link InetAddress} objects.
580      *
581      * @param pcscfServers The {@link Collection} of PCSCF servers to set in this object.
582      * @hide
583      */
584     @SystemApi
585     public void setPcscfServers(@NonNull Collection<InetAddress> pcscfServers) {
586         mPcscfs.clear();
587         for (InetAddress pcscfServer: pcscfServers) {
588             addPcscfServer(pcscfServer);
589         }
590     }
591 
592     /**
593      * Returns all the {@link InetAddress} for PCSCF servers on this link.
594      *
595      * @return An unmodifiable {@link List} of {@link InetAddress} for PCSCF servers on
596      *         this link.
597      * @hide
598      */
599     @SystemApi
600     public @NonNull List<InetAddress> getPcscfServers() {
601         return Collections.unmodifiableList(mPcscfs);
602     }
603 
604     /**
605      * Sets the DNS domain search path used on this link.
606      *
607      * @param domains A {@link String} listing in priority order the comma separated
608      *                domains to search when resolving host names on this link.
609      */
610     public void setDomains(@Nullable String domains) {
611         mDomains = domains;
612     }
613 
614     /**
615      * Get the DNS domains search path set for this link. May be {@code null} if not set.
616      *
617      * @return A {@link String} containing the comma separated domains to search when resolving host
618      *         names on this link or {@code null}.
619      */
620     public @Nullable String getDomains() {
621         return mDomains;
622     }
623 
624     /**
625      * Sets the Maximum Transmission Unit size to use on this link.  This should not be used
626      * unless the system default (1500) is incorrect.  Values less than 68 or greater than
627      * 10000 will be ignored.
628      *
629      * @param mtu The MTU to use for this link.
630      */
631     public void setMtu(int mtu) {
632         mMtu = mtu;
633     }
634 
635     /**
636      * Gets any non-default MTU size set for this link.  Note that if the default is being used
637      * this will return 0.
638      *
639      * @return The mtu value set for this link.
640      */
641     public int getMtu() {
642         return mMtu;
643     }
644 
645     /**
646      * Sets the tcp buffers sizes to be used when this link is the system default.
647      * Should be of the form "rmem_min,rmem_def,rmem_max,wmem_min,wmem_def,wmem_max".
648      *
649      * @param tcpBufferSizes The tcp buffers sizes to use.
650      *
651      * @hide
652      */
653     @SystemApi
654     public void setTcpBufferSizes(@Nullable String tcpBufferSizes) {
655         mTcpBufferSizes = tcpBufferSizes;
656     }
657 
658     /**
659      * Gets the tcp buffer sizes. May be {@code null} if not set.
660      *
661      * @return the tcp buffer sizes to use when this link is the system default or {@code null}.
662      *
663      * @hide
664      */
665     @SystemApi
666     public @Nullable String getTcpBufferSizes() {
667         return mTcpBufferSizes;
668     }
669 
670     private RouteInfo routeWithInterface(RouteInfo route) {
671         return new RouteInfo(
672             route.getDestination(),
673             route.getGateway(),
674             mIfaceName,
675             route.getType(),
676             route.getMtu());
677     }
678 
679     private int findRouteIndexByRouteKey(RouteInfo route) {
680         for (int i = 0; i < mRoutes.size(); i++) {
681             if (mRoutes.get(i).getRouteKey().equals(route.getRouteKey())) {
682                 return i;
683             }
684         }
685         return -1;
686     }
687 
688     /**
689      * Adds a {@link RouteInfo} to this {@code LinkProperties}. If there is a {@link RouteInfo}
690      * with the same destination, gateway and interface with different properties
691      * (e.g., different MTU), it will be updated. If the {@link RouteInfo} had an
692      * interface name set and that differs from the interface set for this
693      * {@code LinkProperties} an {@link IllegalArgumentException} will be thrown.
694      * The proper course is to add either un-named or properly named {@link RouteInfo}.
695      *
696      * @param route A {@link RouteInfo} to add to this object.
697      * @return {@code true} was added or updated, false otherwise.
698      */
699     public boolean addRoute(@NonNull RouteInfo route) {
700         String routeIface = route.getInterface();
701         if (routeIface != null && !routeIface.equals(mIfaceName)) {
702             throw new IllegalArgumentException(
703                     "Route added with non-matching interface: " + routeIface
704                             + " vs. " + mIfaceName);
705         }
706         route = routeWithInterface(route);
707 
708         int i = findRouteIndexByRouteKey(route);
709         if (i == -1) {
710             // Route was not present. Add it.
711             mRoutes.add(route);
712             return true;
713         } else if (mRoutes.get(i).equals(route)) {
714             // Route was present and has same properties. Do nothing.
715             return false;
716         } else {
717             // Route was present and has different properties. Update it.
718             mRoutes.set(i, route);
719             return true;
720         }
721     }
722 
723     /**
724      * Removes a {@link RouteInfo} from this {@code LinkProperties}, if present. The route must
725      * specify an interface and the interface must match the interface of this
726      * {@code LinkProperties}, or it will not be removed.
727      *
728      * @param route A {@link RouteInfo} specifying the route to remove.
729      * @return {@code true} if the route was removed, {@code false} if it was not present.
730      *
731      * @hide
732      */
733     @SystemApi
734     public boolean removeRoute(@NonNull RouteInfo route) {
735         return Objects.equals(mIfaceName, route.getInterface()) && mRoutes.remove(route);
736     }
737 
738     /**
739      * Returns all the {@link RouteInfo} set on this link.
740      *
741      * @return An unmodifiable {@link List} of {@link RouteInfo} for this link.
742      */
743     public @NonNull List<RouteInfo> getRoutes() {
744         return Collections.unmodifiableList(mRoutes);
745     }
746 
747     /**
748      * Make sure this LinkProperties instance contains routes that cover the local subnet
749      * of its link addresses. Add any route that is missing.
750      * @hide
751      */
752     public void ensureDirectlyConnectedRoutes() {
753         for (LinkAddress addr : mLinkAddresses) {
754             addRoute(new RouteInfo(addr, null, mIfaceName));
755         }
756     }
757 
758     /**
759      * Returns all the routes on this link and all the links stacked above it.
760      * @hide
761      */
762     @SystemApi
763     public @NonNull List<RouteInfo> getAllRoutes() {
764         List<RouteInfo> routes = new ArrayList<>(mRoutes);
765         for (LinkProperties stacked: mStackedLinks.values()) {
766             routes.addAll(stacked.getAllRoutes());
767         }
768         return routes;
769     }
770 
771     /**
772      * Sets the recommended {@link ProxyInfo} to use on this link, or {@code null} for none.
773      * Note that Http Proxies are only a hint - the system recommends their use, but it does
774      * not enforce it and applications may ignore them.
775      *
776      * @param proxy A {@link ProxyInfo} defining the HTTP Proxy to use on this link.
777      */
778     public void setHttpProxy(@Nullable ProxyInfo proxy) {
779         mHttpProxy = proxy;
780     }
781 
782     /**
783      * Gets the recommended {@link ProxyInfo} (or {@code null}) set on this link.
784      *
785      * @return The {@link ProxyInfo} set on this link or {@code null}.
786      */
787     public @Nullable ProxyInfo getHttpProxy() {
788         return mHttpProxy;
789     }
790 
791     /**
792      * Returns the NAT64 prefix in use on this link, if any.
793      *
794      * @return the NAT64 prefix or {@code null}.
795      */
796     public @Nullable IpPrefix getNat64Prefix() {
797         return mNat64Prefix;
798     }
799 
800     /**
801      * Sets the NAT64 prefix in use on this link.
802      *
803      * Currently, only 96-bit prefixes (i.e., where the 32-bit IPv4 address is at the end of the
804      * 128-bit IPv6 address) are supported or {@code null} for no prefix.
805      *
806      * @param prefix the NAT64 prefix.
807      */
808     public void setNat64Prefix(@Nullable IpPrefix prefix) {
809         if (prefix != null && prefix.getPrefixLength() != 96) {
810             throw new IllegalArgumentException("Only 96-bit prefixes are supported: " + prefix);
811         }
812         mNat64Prefix = prefix;  // IpPrefix objects are immutable.
813     }
814 
815     /**
816      * Adds a stacked link.
817      *
818      * If there is already a stacked link with the same interface name as link,
819      * that link is replaced with link. Otherwise, link is added to the list
820      * of stacked links.
821      *
822      * @param link The link to add.
823      * @return true if the link was stacked, false otherwise.
824      * @hide
825      */
826     @UnsupportedAppUsage
827     public boolean addStackedLink(@NonNull LinkProperties link) {
828         if (link.getInterfaceName() != null) {
829             mStackedLinks.put(link.getInterfaceName(), link);
830             return true;
831         }
832         return false;
833     }
834 
835     /**
836      * Removes a stacked link.
837      *
838      * If there is a stacked link with the given interface name, it is
839      * removed. Otherwise, nothing changes.
840      *
841      * @param iface The interface name of the link to remove.
842      * @return true if the link was removed, false otherwise.
843      * @hide
844      */
845     public boolean removeStackedLink(@NonNull String iface) {
846         LinkProperties removed = mStackedLinks.remove(iface);
847         return removed != null;
848     }
849 
850     /**
851      * Returns all the links stacked on top of this link.
852      * @hide
853      */
854     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553)
855     public @NonNull List<LinkProperties> getStackedLinks() {
856         if (mStackedLinks.isEmpty()) {
857             return Collections.emptyList();
858         }
859         final List<LinkProperties> stacked = new ArrayList<>();
860         for (LinkProperties link : mStackedLinks.values()) {
861             stacked.add(new LinkProperties(link));
862         }
863         return Collections.unmodifiableList(stacked);
864     }
865 
866     /**
867      * Clears this object to its initial state.
868      */
869     public void clear() {
870         if (mParcelSensitiveFields) {
871             throw new UnsupportedOperationException(
872                     "Cannot clear LinkProperties when parcelSensitiveFields is set");
873         }
874 
875         mIfaceName = null;
876         mLinkAddresses.clear();
877         mDnses.clear();
878         mUsePrivateDns = false;
879         mPrivateDnsServerName = null;
880         mPcscfs.clear();
881         mDomains = null;
882         mRoutes.clear();
883         mHttpProxy = null;
884         mStackedLinks.clear();
885         mMtu = 0;
886         mDhcpServerAddress = null;
887         mTcpBufferSizes = null;
888         mNat64Prefix = null;
889         mWakeOnLanSupported = false;
890         mCaptivePortalApiUrl = null;
891         mCaptivePortalData = null;
892     }
893 
894     /**
895      * Implement the Parcelable interface
896      */
897     public int describeContents() {
898         return 0;
899     }
900 
901     @Override
902     public String toString() {
903         // Space as a separator, so no need for spaces at start/end of the individual fragments.
904         final StringJoiner resultJoiner = new StringJoiner(" ", "{", "}");
905 
906         if (mIfaceName != null) {
907             resultJoiner.add("InterfaceName:");
908             resultJoiner.add(mIfaceName);
909         }
910 
911         resultJoiner.add("LinkAddresses: [");
912         if (!mLinkAddresses.isEmpty()) {
913             resultJoiner.add(TextUtils.join(",", mLinkAddresses));
914         }
915         resultJoiner.add("]");
916 
917         resultJoiner.add("DnsAddresses: [");
918         if (!mDnses.isEmpty()) {
919             resultJoiner.add(TextUtils.join(",", mDnses));
920         }
921         resultJoiner.add("]");
922 
923         if (mUsePrivateDns) {
924             resultJoiner.add("UsePrivateDns: true");
925         }
926 
927         if (mPrivateDnsServerName != null) {
928             resultJoiner.add("PrivateDnsServerName:");
929             resultJoiner.add(mPrivateDnsServerName);
930         }
931 
932         if (!mPcscfs.isEmpty()) {
933             resultJoiner.add("PcscfAddresses: [");
934             resultJoiner.add(TextUtils.join(",", mPcscfs));
935             resultJoiner.add("]");
936         }
937 
938         if (!mValidatedPrivateDnses.isEmpty()) {
939             final StringJoiner validatedPrivateDnsesJoiner =
940                     new StringJoiner(",", "ValidatedPrivateDnsAddresses: [", "]");
941             for (final InetAddress addr : mValidatedPrivateDnses) {
942                 validatedPrivateDnsesJoiner.add(addr.getHostAddress());
943             }
944             resultJoiner.add(validatedPrivateDnsesJoiner.toString());
945         }
946 
947         resultJoiner.add("Domains:");
948         resultJoiner.add(mDomains);
949 
950         resultJoiner.add("MTU:");
951         resultJoiner.add(Integer.toString(mMtu));
952 
953         if (mWakeOnLanSupported) {
954             resultJoiner.add("WakeOnLanSupported: true");
955         }
956 
957         if (mDhcpServerAddress != null) {
958             resultJoiner.add("ServerAddress:");
959             resultJoiner.add(mDhcpServerAddress.toString());
960         }
961 
962         if (mCaptivePortalApiUrl != null) {
963             resultJoiner.add("CaptivePortalApiUrl: " + mCaptivePortalApiUrl);
964         }
965 
966         if (mCaptivePortalData != null) {
967             resultJoiner.add("CaptivePortalData: " + mCaptivePortalData);
968         }
969 
970         if (mTcpBufferSizes != null) {
971             resultJoiner.add("TcpBufferSizes:");
972             resultJoiner.add(mTcpBufferSizes);
973         }
974 
975         resultJoiner.add("Routes: [");
976         if (!mRoutes.isEmpty()) {
977             resultJoiner.add(TextUtils.join(",", mRoutes));
978         }
979         resultJoiner.add("]");
980 
981         if (mHttpProxy != null) {
982             resultJoiner.add("HttpProxy:");
983             resultJoiner.add(mHttpProxy.toString());
984         }
985 
986         if (mNat64Prefix != null) {
987             resultJoiner.add("Nat64Prefix:");
988             resultJoiner.add(mNat64Prefix.toString());
989         }
990 
991         final Collection<LinkProperties> stackedLinksValues = mStackedLinks.values();
992         if (!stackedLinksValues.isEmpty()) {
993             final StringJoiner stackedLinksJoiner = new StringJoiner(",", "Stacked: [", "]");
994             for (final LinkProperties lp : stackedLinksValues) {
995                 stackedLinksJoiner.add("[ " + lp + " ]");
996             }
997             resultJoiner.add(stackedLinksJoiner.toString());
998         }
999 
1000         return resultJoiner.toString();
1001     }
1002 
1003     /**
1004      * Returns true if this link has an IPv4 address.
1005      *
1006      * @return {@code true} if there is an IPv4 address, {@code false} otherwise.
1007      * @hide
1008      */
1009     @SystemApi
1010     public boolean hasIpv4Address() {
1011         for (LinkAddress address : mLinkAddresses) {
1012             if (address.getAddress() instanceof Inet4Address) {
1013                 return true;
1014             }
1015         }
1016         return false;
1017     }
1018 
1019     /**
1020      * For backward compatibility.
1021      * This was annotated with @UnsupportedAppUsage in P, so we can't remove the method completely
1022      * just yet.
1023      * @return {@code true} if there is an IPv4 address, {@code false} otherwise.
1024      * @hide
1025      */
1026     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P)
1027     public boolean hasIPv4Address() {
1028         return hasIpv4Address();
1029     }
1030 
1031     /**
1032      * Returns true if this link or any of its stacked interfaces has an IPv4 address.
1033      *
1034      * @return {@code true} if there is an IPv4 address, {@code false} otherwise.
1035      */
1036     private boolean hasIpv4AddressOnInterface(String iface) {
1037         // mIfaceName can be null.
1038         return (Objects.equals(iface, mIfaceName) && hasIpv4Address())
1039                 || (iface != null && mStackedLinks.containsKey(iface)
1040                         && mStackedLinks.get(iface).hasIpv4Address());
1041     }
1042 
1043     /**
1044      * Returns true if this link has a global preferred IPv6 address.
1045      *
1046      * @return {@code true} if there is a global preferred IPv6 address, {@code false} otherwise.
1047      * @hide
1048      */
1049     @SystemApi
1050     public boolean hasGlobalIpv6Address() {
1051         for (LinkAddress address : mLinkAddresses) {
1052           if (address.getAddress() instanceof Inet6Address && address.isGlobalPreferred()) {
1053             return true;
1054           }
1055         }
1056         return false;
1057     }
1058 
1059     /**
1060      * Returns true if this link has an IPv4 unreachable default route.
1061      *
1062      * @return {@code true} if there is an IPv4 unreachable default route, {@code false} otherwise.
1063      * @hide
1064      */
1065     public boolean hasIpv4UnreachableDefaultRoute() {
1066         for (RouteInfo r : mRoutes) {
1067             if (r.isIPv4UnreachableDefault()) {
1068                 return true;
1069             }
1070         }
1071         return false;
1072     }
1073 
1074     /**
1075      * For backward compatibility.
1076      * This was annotated with @UnsupportedAppUsage in P, so we can't remove the method completely
1077      * just yet.
1078      * @return {@code true} if there is a global preferred IPv6 address, {@code false} otherwise.
1079      * @hide
1080      */
1081     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P)
1082     public boolean hasGlobalIPv6Address() {
1083         return hasGlobalIpv6Address();
1084     }
1085 
1086     /**
1087      * Returns true if this link has an IPv4 default route.
1088      *
1089      * @return {@code true} if there is an IPv4 default route, {@code false} otherwise.
1090      * @hide
1091      */
1092     @SystemApi
1093     public boolean hasIpv4DefaultRoute() {
1094         for (RouteInfo r : mRoutes) {
1095             if (r.isIPv4Default()) {
1096                 return true;
1097             }
1098         }
1099         return false;
1100     }
1101 
1102     /**
1103      * Returns true if this link has an IPv6 unreachable default route.
1104      *
1105      * @return {@code true} if there is an IPv6 unreachable default route, {@code false} otherwise.
1106      * @hide
1107      */
1108     public boolean hasIpv6UnreachableDefaultRoute() {
1109         for (RouteInfo r : mRoutes) {
1110             if (r.isIPv6UnreachableDefault()) {
1111                 return true;
1112             }
1113         }
1114         return false;
1115     }
1116 
1117     /**
1118      * For backward compatibility.
1119      * This was annotated with @UnsupportedAppUsage in P, so we can't remove the method completely
1120      * just yet.
1121      * @return {@code true} if there is an IPv4 default route, {@code false} otherwise.
1122      * @hide
1123      */
1124     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P)
1125     public boolean hasIPv4DefaultRoute() {
1126         return hasIpv4DefaultRoute();
1127     }
1128 
1129     /**
1130      * Returns true if this link has an IPv6 default route.
1131      *
1132      * @return {@code true} if there is an IPv6 default route, {@code false} otherwise.
1133      * @hide
1134      */
1135     @SystemApi
1136     public boolean hasIpv6DefaultRoute() {
1137         for (RouteInfo r : mRoutes) {
1138             if (r.isIPv6Default()) {
1139                 return true;
1140             }
1141         }
1142         return false;
1143     }
1144 
1145     /**
1146      * For backward compatibility.
1147      * This was annotated with @UnsupportedAppUsage in P, so we can't remove the method completely
1148      * just yet.
1149      * @return {@code true} if there is an IPv6 default route, {@code false} otherwise.
1150      * @hide
1151      */
1152     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P)
1153     public boolean hasIPv6DefaultRoute() {
1154         return hasIpv6DefaultRoute();
1155     }
1156 
1157     /**
1158      * Returns true if this link has an IPv4 DNS server.
1159      *
1160      * @return {@code true} if there is an IPv4 DNS server, {@code false} otherwise.
1161      * @hide
1162      */
1163     @SystemApi
1164     public boolean hasIpv4DnsServer() {
1165         for (InetAddress ia : mDnses) {
1166             if (ia instanceof Inet4Address) {
1167                 return true;
1168             }
1169         }
1170         return false;
1171     }
1172 
1173     /**
1174      * For backward compatibility.
1175      * This was annotated with @UnsupportedAppUsage in P, so we can't remove the method completely
1176      * just yet.
1177      * @return {@code true} if there is an IPv4 DNS server, {@code false} otherwise.
1178      * @hide
1179      */
1180     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P)
1181     public boolean hasIPv4DnsServer() {
1182         return hasIpv4DnsServer();
1183     }
1184 
1185     /**
1186      * Returns true if this link has an IPv6 DNS server.
1187      *
1188      * @return {@code true} if there is an IPv6 DNS server, {@code false} otherwise.
1189      * @hide
1190      */
1191     @SystemApi
1192     public boolean hasIpv6DnsServer() {
1193         for (InetAddress ia : mDnses) {
1194             if (ia instanceof Inet6Address) {
1195                 return true;
1196             }
1197         }
1198         return false;
1199     }
1200 
1201     /**
1202      * For backward compatibility.
1203      * This was annotated with @UnsupportedAppUsage in P, so we can't remove the method completely
1204      * just yet.
1205      * @return {@code true} if there is an IPv6 DNS server, {@code false} otherwise.
1206      * @hide
1207      */
1208     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P)
1209     public boolean hasIPv6DnsServer() {
1210         return hasIpv6DnsServer();
1211     }
1212 
1213     /**
1214      * Returns true if this link has an IPv4 PCSCF server.
1215      *
1216      * @return {@code true} if there is an IPv4 PCSCF server, {@code false} otherwise.
1217      * @hide
1218      */
1219     public boolean hasIpv4PcscfServer() {
1220         for (InetAddress ia : mPcscfs) {
1221           if (ia instanceof Inet4Address) {
1222             return true;
1223           }
1224         }
1225         return false;
1226     }
1227 
1228     /**
1229      * Returns true if this link has an IPv6 PCSCF server.
1230      *
1231      * @return {@code true} if there is an IPv6 PCSCF server, {@code false} otherwise.
1232      * @hide
1233      */
1234     public boolean hasIpv6PcscfServer() {
1235         for (InetAddress ia : mPcscfs) {
1236           if (ia instanceof Inet6Address) {
1237             return true;
1238           }
1239         }
1240         return false;
1241     }
1242 
1243     /**
1244      * Returns true if this link is provisioned for global IPv4 connectivity.
1245      * This requires an IP address, default route, and DNS server.
1246      *
1247      * @return {@code true} if the link is provisioned, {@code false} otherwise.
1248      * @hide
1249      */
1250     @SystemApi
1251     public boolean isIpv4Provisioned() {
1252         return (hasIpv4Address()
1253                 && hasIpv4DefaultRoute()
1254                 && hasIpv4DnsServer());
1255     }
1256 
1257     /**
1258      * Returns true if this link is provisioned for global IPv6 connectivity.
1259      * This requires an IP address, default route, and DNS server.
1260      *
1261      * @return {@code true} if the link is provisioned, {@code false} otherwise.
1262      * @hide
1263      */
1264     @SystemApi
1265     public boolean isIpv6Provisioned() {
1266         return (hasGlobalIpv6Address()
1267                 && hasIpv6DefaultRoute()
1268                 && hasIpv6DnsServer());
1269     }
1270 
1271     /**
1272      * For backward compatibility.
1273      * This was annotated with @UnsupportedAppUsage in P, so we can't remove the method completely
1274      * just yet.
1275      * @return {@code true} if the link is provisioned, {@code false} otherwise.
1276      * @hide
1277      */
1278     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P)
1279     public boolean isIPv6Provisioned() {
1280         return isIpv6Provisioned();
1281     }
1282 
1283 
1284     /**
1285      * Returns true if this link is provisioned for global connectivity,
1286      * for at least one Internet Protocol family.
1287      *
1288      * @return {@code true} if the link is provisioned, {@code false} otherwise.
1289      * @hide
1290      */
1291     @SystemApi
1292     public boolean isProvisioned() {
1293         return (isIpv4Provisioned() || isIpv6Provisioned());
1294     }
1295 
1296     /**
1297      * Evaluate whether the {@link InetAddress} is considered reachable.
1298      *
1299      * @return {@code true} if the given {@link InetAddress} is considered reachable,
1300      *         {@code false} otherwise.
1301      * @hide
1302      */
1303     @SystemApi
1304     public boolean isReachable(@NonNull InetAddress ip) {
1305         final List<RouteInfo> allRoutes = getAllRoutes();
1306         // If we don't have a route to this IP address, it's not reachable.
1307         final RouteInfo bestRoute = RouteInfo.selectBestRoute(allRoutes, ip);
1308         if (bestRoute == null) {
1309             return false;
1310         }
1311 
1312         // TODO: better source address evaluation for destination addresses.
1313 
1314         if (ip instanceof Inet4Address) {
1315             // For IPv4, it suffices for now to simply have any address.
1316             return hasIpv4AddressOnInterface(bestRoute.getInterface());
1317         } else if (ip instanceof Inet6Address) {
1318             if (ip.isLinkLocalAddress()) {
1319                 // For now, just make sure link-local destinations have
1320                 // scopedIds set, since transmits will generally fail otherwise.
1321                 // TODO: verify it matches the ifindex of one of the interfaces.
1322                 return (((Inet6Address)ip).getScopeId() != 0);
1323             }  else {
1324                 // For non-link-local destinations check that either the best route
1325                 // is directly connected or that some global preferred address exists.
1326                 // TODO: reconsider all cases (disconnected ULA networks, ...).
1327                 return (!bestRoute.hasGateway() || hasGlobalIpv6Address());
1328             }
1329         }
1330 
1331         return false;
1332     }
1333 
1334     /**
1335      * Compares this {@code LinkProperties} interface name against the target
1336      *
1337      * @param target LinkProperties to compare.
1338      * @return {@code true} if both are identical, {@code false} otherwise.
1339      * @hide
1340      */
1341     @UnsupportedAppUsage
1342     public boolean isIdenticalInterfaceName(@NonNull LinkProperties target) {
1343         return LinkPropertiesUtils.isIdenticalInterfaceName(target, this);
1344     }
1345 
1346     /**
1347      * Compares this {@code LinkProperties} DHCP server address against the target
1348      *
1349      * @param target LinkProperties to compare.
1350      * @return {@code true} if both are identical, {@code false} otherwise.
1351      * @hide
1352      */
1353     public boolean isIdenticalDhcpServerAddress(@NonNull LinkProperties target) {
1354         return Objects.equals(mDhcpServerAddress, target.mDhcpServerAddress);
1355     }
1356 
1357     /**
1358      * Compares this {@code LinkProperties} interface addresses against the target
1359      *
1360      * @param target LinkProperties to compare.
1361      * @return {@code true} if both are identical, {@code false} otherwise.
1362      * @hide
1363      */
1364     @UnsupportedAppUsage
1365     public boolean isIdenticalAddresses(@NonNull LinkProperties target) {
1366         return LinkPropertiesUtils.isIdenticalAddresses(target, this);
1367     }
1368 
1369     /**
1370      * Compares this {@code LinkProperties} DNS addresses against the target
1371      *
1372      * @param target LinkProperties to compare.
1373      * @return {@code true} if both are identical, {@code false} otherwise.
1374      * @hide
1375      */
1376     @UnsupportedAppUsage
1377     public boolean isIdenticalDnses(@NonNull LinkProperties target) {
1378         return LinkPropertiesUtils.isIdenticalDnses(target, this);
1379     }
1380 
1381     /**
1382      * Compares this {@code LinkProperties} private DNS settings against the
1383      * target.
1384      *
1385      * @param target LinkProperties to compare.
1386      * @return {@code true} if both are identical, {@code false} otherwise.
1387      * @hide
1388      */
1389     public boolean isIdenticalPrivateDns(@NonNull LinkProperties target) {
1390         return (isPrivateDnsActive() == target.isPrivateDnsActive()
1391                 && TextUtils.equals(getPrivateDnsServerName(),
1392                 target.getPrivateDnsServerName()));
1393     }
1394 
1395     /**
1396      * Compares this {@code LinkProperties} validated private DNS addresses against
1397      * the target
1398      *
1399      * @param target LinkProperties to compare.
1400      * @return {@code true} if both are identical, {@code false} otherwise.
1401      * @hide
1402      */
1403     public boolean isIdenticalValidatedPrivateDnses(@NonNull LinkProperties target) {
1404         Collection<InetAddress> targetDnses = target.getValidatedPrivateDnsServers();
1405         return (mValidatedPrivateDnses.size() == targetDnses.size())
1406                 ? mValidatedPrivateDnses.containsAll(targetDnses) : false;
1407     }
1408 
1409     /**
1410      * Compares this {@code LinkProperties} PCSCF addresses against the target
1411      *
1412      * @param target LinkProperties to compare.
1413      * @return {@code true} if both are identical, {@code false} otherwise.
1414      * @hide
1415      */
1416     public boolean isIdenticalPcscfs(@NonNull LinkProperties target) {
1417         Collection<InetAddress> targetPcscfs = target.getPcscfServers();
1418         return (mPcscfs.size() == targetPcscfs.size()) ?
1419                     mPcscfs.containsAll(targetPcscfs) : false;
1420     }
1421 
1422     /**
1423      * Compares this {@code LinkProperties} Routes against the target
1424      *
1425      * @param target LinkProperties to compare.
1426      * @return {@code true} if both are identical, {@code false} otherwise.
1427      * @hide
1428      */
1429     @UnsupportedAppUsage
1430     public boolean isIdenticalRoutes(@NonNull LinkProperties target) {
1431         return LinkPropertiesUtils.isIdenticalRoutes(target, this);
1432     }
1433 
1434     /**
1435      * Compares this {@code LinkProperties} HttpProxy against the target
1436      *
1437      * @param target LinkProperties to compare.
1438      * @return {@code true} if both are identical, {@code false} otherwise.
1439      * @hide
1440      */
1441     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
1442     public boolean isIdenticalHttpProxy(@NonNull LinkProperties target) {
1443         return LinkPropertiesUtils.isIdenticalHttpProxy(target, this);
1444     }
1445 
1446     /**
1447      * Compares this {@code LinkProperties} stacked links against the target
1448      *
1449      * @param target LinkProperties to compare.
1450      * @return {@code true} if both are identical, {@code false} otherwise.
1451      * @hide
1452      */
1453     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553)
1454     public boolean isIdenticalStackedLinks(@NonNull LinkProperties target) {
1455         if (!mStackedLinks.keySet().equals(target.mStackedLinks.keySet())) {
1456             return false;
1457         }
1458         for (LinkProperties stacked : mStackedLinks.values()) {
1459             // Hashtable values can never be null.
1460             String iface = stacked.getInterfaceName();
1461             if (!stacked.equals(target.mStackedLinks.get(iface))) {
1462                 return false;
1463             }
1464         }
1465         return true;
1466     }
1467 
1468     /**
1469      * Compares this {@code LinkProperties} MTU against the target
1470      *
1471      * @param target LinkProperties to compare.
1472      * @return {@code true} if both are identical, {@code false} otherwise.
1473      * @hide
1474      */
1475     public boolean isIdenticalMtu(@NonNull LinkProperties target) {
1476         return getMtu() == target.getMtu();
1477     }
1478 
1479     /**
1480      * Compares this {@code LinkProperties} Tcp buffer sizes against the target.
1481      *
1482      * @param target LinkProperties to compare.
1483      * @return {@code true} if both are identical, {@code false} otherwise.
1484      * @hide
1485      */
1486     public boolean isIdenticalTcpBufferSizes(@NonNull LinkProperties target) {
1487         return Objects.equals(mTcpBufferSizes, target.mTcpBufferSizes);
1488     }
1489 
1490     /**
1491      * Compares this {@code LinkProperties} NAT64 prefix against the target.
1492      *
1493      * @param target LinkProperties to compare.
1494      * @return {@code true} if both are identical, {@code false} otherwise.
1495      * @hide
1496      */
1497     public boolean isIdenticalNat64Prefix(@NonNull LinkProperties target) {
1498         return Objects.equals(mNat64Prefix, target.mNat64Prefix);
1499     }
1500 
1501     /**
1502      * Compares this {@code LinkProperties} WakeOnLan supported against the target.
1503      *
1504      * @param target LinkProperties to compare.
1505      * @return {@code true} if both are identical, {@code false} otherwise.
1506      * @hide
1507      */
1508     public boolean isIdenticalWakeOnLan(LinkProperties target) {
1509         return isWakeOnLanSupported() == target.isWakeOnLanSupported();
1510     }
1511 
1512     /**
1513      * Compares this {@code LinkProperties}'s CaptivePortalApiUrl against the target.
1514      *
1515      * @param target LinkProperties to compare.
1516      * @return {@code true} if both are identical, {@code false} otherwise.
1517      * @hide
1518      */
1519     public boolean isIdenticalCaptivePortalApiUrl(LinkProperties target) {
1520         return Objects.equals(mCaptivePortalApiUrl, target.mCaptivePortalApiUrl);
1521     }
1522 
1523     /**
1524      * Compares this {@code LinkProperties}'s CaptivePortalData against the target.
1525      *
1526      * @param target LinkProperties to compare.
1527      * @return {@code true} if both are identical, {@code false} otherwise.
1528      * @hide
1529      */
1530     public boolean isIdenticalCaptivePortalData(LinkProperties target) {
1531         return Objects.equals(mCaptivePortalData, target.mCaptivePortalData);
1532     }
1533 
1534     /**
1535      * Set whether the network interface supports WakeOnLAN
1536      *
1537      * @param supported WakeOnLAN supported value
1538      *
1539      * @hide
1540      */
1541     public void setWakeOnLanSupported(boolean supported) {
1542         mWakeOnLanSupported = supported;
1543     }
1544 
1545     /**
1546      * Returns whether the network interface supports WakeOnLAN
1547      *
1548      * @return {@code true} if interface supports WakeOnLAN, {@code false} otherwise.
1549      */
1550     public boolean isWakeOnLanSupported() {
1551         return mWakeOnLanSupported;
1552     }
1553 
1554     /**
1555      * Set the URL of the captive portal API endpoint to get more information about the network.
1556      * @hide
1557      */
1558     @SystemApi
1559     public void setCaptivePortalApiUrl(@Nullable Uri url) {
1560         mCaptivePortalApiUrl = url;
1561     }
1562 
1563     /**
1564      * Get the URL of the captive portal API endpoint to get more information about the network.
1565      *
1566      * <p>This is null unless the application has
1567      * {@link android.Manifest.permission.NETWORK_SETTINGS} or
1568      * {@link NetworkStack#PERMISSION_MAINLINE_NETWORK_STACK} permissions, and the network provided
1569      * the URL.
1570      * @hide
1571      */
1572     @SystemApi
1573     @Nullable
1574     public Uri getCaptivePortalApiUrl() {
1575         return mCaptivePortalApiUrl;
1576     }
1577 
1578     /**
1579      * Set the CaptivePortalData obtained from the captive portal API (RFC7710bis).
1580      * @hide
1581      */
1582     @SystemApi
1583     public void setCaptivePortalData(@Nullable CaptivePortalData data) {
1584         mCaptivePortalData = data;
1585     }
1586 
1587     /**
1588      * Get the CaptivePortalData obtained from the captive portal API (RFC7710bis).
1589      *
1590      * <p>This is null unless the application has
1591      * {@link android.Manifest.permission.NETWORK_SETTINGS} or
1592      * {@link NetworkStack#PERMISSION_MAINLINE_NETWORK_STACK} permissions.
1593      * @hide
1594      */
1595     @SystemApi
1596     @Nullable
1597     public CaptivePortalData getCaptivePortalData() {
1598         return mCaptivePortalData;
1599     }
1600 
1601     /**
1602      * Compares this {@code LinkProperties} instance against the target
1603      * LinkProperties in {@code obj}. Two LinkPropertieses are equal if
1604      * all their fields are equal in values.
1605      *
1606      * For collection fields, such as mDnses, containsAll() is used to check
1607      * if two collections contains the same elements, independent of order.
1608      * There are two thoughts regarding containsAll()
1609      * 1. Duplicated elements. eg, (A, B, B) and (A, A, B) are equal.
1610      * 2. Worst case performance is O(n^2).
1611      *
1612      * @param obj the object to be tested for equality.
1613      * @return {@code true} if both objects are equal, {@code false} otherwise.
1614      */
1615     @Override
1616     public boolean equals(@Nullable Object obj) {
1617         if (this == obj) return true;
1618 
1619         if (!(obj instanceof LinkProperties)) return false;
1620 
1621         LinkProperties target = (LinkProperties) obj;
1622         /*
1623          * This method does not check that stacked interfaces are equal, because
1624          * stacked interfaces are not so much a property of the link as a
1625          * description of connections between links.
1626          */
1627         return isIdenticalInterfaceName(target)
1628                 && isIdenticalAddresses(target)
1629                 && isIdenticalDhcpServerAddress(target)
1630                 && isIdenticalDnses(target)
1631                 && isIdenticalPrivateDns(target)
1632                 && isIdenticalValidatedPrivateDnses(target)
1633                 && isIdenticalPcscfs(target)
1634                 && isIdenticalRoutes(target)
1635                 && isIdenticalHttpProxy(target)
1636                 && isIdenticalStackedLinks(target)
1637                 && isIdenticalMtu(target)
1638                 && isIdenticalTcpBufferSizes(target)
1639                 && isIdenticalNat64Prefix(target)
1640                 && isIdenticalWakeOnLan(target)
1641                 && isIdenticalCaptivePortalApiUrl(target)
1642                 && isIdenticalCaptivePortalData(target);
1643     }
1644 
1645     /**
1646      * Generate hashcode based on significant fields
1647      *
1648      * Equal objects must produce the same hash code, while unequal objects
1649      * may have the same hash codes.
1650      */
1651     @Override
1652     public int hashCode() {
1653         return ((null == mIfaceName) ? 0 : mIfaceName.hashCode()
1654                 + mLinkAddresses.size() * 31
1655                 + mDnses.size() * 37
1656                 + mValidatedPrivateDnses.size() * 61
1657                 + ((null == mDomains) ? 0 : mDomains.hashCode())
1658                 + mRoutes.size() * 41
1659                 + ((null == mHttpProxy) ? 0 : mHttpProxy.hashCode())
1660                 + mStackedLinks.hashCode() * 47)
1661                 + mMtu * 51
1662                 + ((null == mTcpBufferSizes) ? 0 : mTcpBufferSizes.hashCode())
1663                 + (mUsePrivateDns ? 57 : 0)
1664                 + ((null == mDhcpServerAddress) ? 0 : mDhcpServerAddress.hashCode())
1665                 + mPcscfs.size() * 67
1666                 + ((null == mPrivateDnsServerName) ? 0 : mPrivateDnsServerName.hashCode())
1667                 + Objects.hash(mNat64Prefix)
1668                 + (mWakeOnLanSupported ? 71 : 0)
1669                 + Objects.hash(mCaptivePortalApiUrl, mCaptivePortalData);
1670     }
1671 
1672     /**
1673      * Implement the Parcelable interface.
1674      */
1675     public void writeToParcel(Parcel dest, int flags) {
1676         dest.writeString(getInterfaceName());
1677         dest.writeInt(mLinkAddresses.size());
1678         for (LinkAddress linkAddress : mLinkAddresses) {
1679             dest.writeParcelable(linkAddress, flags);
1680         }
1681 
1682         writeAddresses(dest, mDnses);
1683         writeAddresses(dest, mValidatedPrivateDnses);
1684         dest.writeBoolean(mUsePrivateDns);
1685         dest.writeString(mPrivateDnsServerName);
1686         writeAddresses(dest, mPcscfs);
1687         dest.writeString(mDomains);
1688         writeAddress(dest, mDhcpServerAddress);
1689         dest.writeInt(mMtu);
1690         dest.writeString(mTcpBufferSizes);
1691         dest.writeInt(mRoutes.size());
1692         for (RouteInfo route : mRoutes) {
1693             dest.writeParcelable(route, flags);
1694         }
1695 
1696         if (mHttpProxy != null) {
1697             dest.writeByte((byte)1);
1698             dest.writeParcelable(mHttpProxy, flags);
1699         } else {
1700             dest.writeByte((byte)0);
1701         }
1702         dest.writeParcelable(mNat64Prefix, 0);
1703 
1704         ArrayList<LinkProperties> stackedLinks = new ArrayList<>(mStackedLinks.values());
1705         dest.writeList(stackedLinks);
1706 
1707         dest.writeBoolean(mWakeOnLanSupported);
1708         dest.writeParcelable(mParcelSensitiveFields ? mCaptivePortalApiUrl : null, 0);
1709         dest.writeParcelable(mParcelSensitiveFields ? mCaptivePortalData : null, 0);
1710     }
1711 
1712     private static void writeAddresses(@NonNull Parcel dest, @NonNull List<InetAddress> list) {
1713         dest.writeInt(list.size());
1714         for (InetAddress d : list) {
1715             writeAddress(dest, d);
1716         }
1717     }
1718 
1719     private static void writeAddress(@NonNull Parcel dest, @Nullable InetAddress addr) {
1720         byte[] addressBytes = (addr == null ? null : addr.getAddress());
1721         dest.writeByteArray(addressBytes);
1722         if (addr instanceof Inet6Address) {
1723             final Inet6Address v6Addr = (Inet6Address) addr;
1724             final boolean hasScopeId = v6Addr.getScopeId() != 0;
1725             dest.writeBoolean(hasScopeId);
1726             if (hasScopeId) dest.writeInt(v6Addr.getScopeId());
1727         }
1728     }
1729 
1730     @Nullable
1731     private static InetAddress readAddress(@NonNull Parcel p) throws UnknownHostException {
1732         final byte[] addr = p.createByteArray();
1733         if (addr == null) return null;
1734 
1735         if (addr.length == INET6_ADDR_LENGTH) {
1736             final boolean hasScopeId = p.readBoolean();
1737             final int scopeId = hasScopeId ? p.readInt() : 0;
1738             return Inet6Address.getByAddress(null /* host */, addr, scopeId);
1739         }
1740 
1741         return InetAddress.getByAddress(addr);
1742     }
1743 
1744     /**
1745      * Implement the Parcelable interface.
1746      */
1747     public static final @android.annotation.NonNull Creator<LinkProperties> CREATOR =
1748         new Creator<LinkProperties>() {
1749             public LinkProperties createFromParcel(Parcel in) {
1750                 LinkProperties netProp = new LinkProperties();
1751 
1752                 String iface = in.readString();
1753                 if (iface != null) {
1754                     netProp.setInterfaceName(iface);
1755                 }
1756                 int addressCount = in.readInt();
1757                 for (int i = 0; i < addressCount; i++) {
1758                     netProp.addLinkAddress(in.readParcelable(null));
1759                 }
1760                 addressCount = in.readInt();
1761                 for (int i = 0; i < addressCount; i++) {
1762                     try {
1763                         netProp.addDnsServer(readAddress(in));
1764                     } catch (UnknownHostException e) { }
1765                 }
1766                 addressCount = in.readInt();
1767                 for (int i = 0; i < addressCount; i++) {
1768                     try {
1769                         netProp.addValidatedPrivateDnsServer(readAddress(in));
1770                     } catch (UnknownHostException e) { }
1771                 }
1772                 netProp.setUsePrivateDns(in.readBoolean());
1773                 netProp.setPrivateDnsServerName(in.readString());
1774                 addressCount = in.readInt();
1775                 for (int i = 0; i < addressCount; i++) {
1776                     try {
1777                         netProp.addPcscfServer(readAddress(in));
1778                     } catch (UnknownHostException e) { }
1779                 }
1780                 netProp.setDomains(in.readString());
1781                 try {
1782                     netProp.setDhcpServerAddress((Inet4Address) InetAddress
1783                             .getByAddress(in.createByteArray()));
1784                 } catch (UnknownHostException e) { }
1785                 netProp.setMtu(in.readInt());
1786                 netProp.setTcpBufferSizes(in.readString());
1787                 addressCount = in.readInt();
1788                 for (int i = 0; i < addressCount; i++) {
1789                     netProp.addRoute(in.readParcelable(null));
1790                 }
1791                 if (in.readByte() == 1) {
1792                     netProp.setHttpProxy(in.readParcelable(null));
1793                 }
1794                 netProp.setNat64Prefix(in.readParcelable(null));
1795                 ArrayList<LinkProperties> stackedLinks = new ArrayList<LinkProperties>();
1796                 in.readList(stackedLinks, LinkProperties.class.getClassLoader());
1797                 for (LinkProperties stackedLink: stackedLinks) {
1798                     netProp.addStackedLink(stackedLink);
1799                 }
1800                 netProp.setWakeOnLanSupported(in.readBoolean());
1801 
1802                 netProp.setCaptivePortalApiUrl(in.readParcelable(null));
1803                 netProp.setCaptivePortalData(in.readParcelable(null));
1804                 return netProp;
1805             }
1806 
1807             public LinkProperties[] newArray(int size) {
1808                 return new LinkProperties[size];
1809             }
1810         };
1811 
1812     /**
1813      * Check the valid MTU range based on IPv4 or IPv6.
1814      * @hide
1815      */
1816     public static boolean isValidMtu(int mtu, boolean ipv6) {
1817         if (ipv6) {
1818             return mtu >= MIN_MTU_V6 && mtu <= MAX_MTU;
1819         } else {
1820             return mtu >= MIN_MTU && mtu <= MAX_MTU;
1821         }
1822     }
1823 }
1824