1 /* 2 * Copyright (C) 2019 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.shared; 18 19 import static android.net.shared.ParcelableUtil.fromParcelableArray; 20 import static android.net.shared.ParcelableUtil.toParcelableArray; 21 import static android.text.TextUtils.join; 22 23 import android.net.InetAddresses; 24 import android.net.InitialConfigurationParcelable; 25 import android.net.IpPrefix; 26 import android.net.LinkAddress; 27 import android.net.RouteInfo; 28 29 import java.net.Inet4Address; 30 import java.net.InetAddress; 31 import java.util.Arrays; 32 import java.util.HashSet; 33 import java.util.List; 34 import java.util.Set; 35 import java.util.function.Predicate; 36 37 /** @hide */ 38 public class InitialConfiguration { 39 public final Set<LinkAddress> ipAddresses = new HashSet<>(); 40 public final Set<IpPrefix> directlyConnectedRoutes = new HashSet<>(); 41 public final Set<InetAddress> dnsServers = new HashSet<>(); 42 43 private static final int RFC6177_MIN_PREFIX_LENGTH = 48; 44 private static final int RFC7421_PREFIX_LENGTH = 64; 45 46 public static final InetAddress INET6_ANY = InetAddresses.parseNumericAddress("::"); 47 48 /** 49 * Create a InitialConfiguration that is a copy of the specified configuration. 50 */ copy(InitialConfiguration config)51 public static InitialConfiguration copy(InitialConfiguration config) { 52 if (config == null) { 53 return null; 54 } 55 InitialConfiguration configCopy = new InitialConfiguration(); 56 configCopy.ipAddresses.addAll(config.ipAddresses); 57 configCopy.directlyConnectedRoutes.addAll(config.directlyConnectedRoutes); 58 configCopy.dnsServers.addAll(config.dnsServers); 59 return configCopy; 60 } 61 62 @Override toString()63 public String toString() { 64 return String.format( 65 "InitialConfiguration(IPs: {%s}, prefixes: {%s}, DNS: {%s})", 66 join(", ", ipAddresses), join(", ", directlyConnectedRoutes), 67 join(", ", dnsServers)); 68 } 69 70 /** 71 * Tests whether the contents of this IpConfiguration represent a valid configuration. 72 */ isValid()73 public boolean isValid() { 74 if (ipAddresses.isEmpty()) { 75 return false; 76 } 77 78 // For every IP address, there must be at least one prefix containing that address. 79 for (LinkAddress addr : ipAddresses) { 80 if (!any(directlyConnectedRoutes, (p) -> p.contains(addr.getAddress()))) { 81 return false; 82 } 83 } 84 // For every dns server, there must be at least one prefix containing that address. 85 for (InetAddress addr : dnsServers) { 86 if (!any(directlyConnectedRoutes, (p) -> p.contains(addr))) { 87 return false; 88 } 89 } 90 // All IPv6 LinkAddresses have an RFC7421-suitable prefix length 91 // (read: compliant with RFC4291#section2.5.4). 92 if (any(ipAddresses, not(InitialConfiguration::isPrefixLengthCompliant))) { 93 return false; 94 } 95 // If directlyConnectedRoutes contains an IPv6 default route 96 // then ipAddresses MUST contain at least one non-ULA GUA. 97 if (any(directlyConnectedRoutes, InitialConfiguration::isIPv6DefaultRoute) 98 && all(ipAddresses, not(InitialConfiguration::isIPv6GUA))) { 99 return false; 100 } 101 // The prefix length of routes in directlyConnectedRoutes be within reasonable 102 // bounds for IPv6: /48-/64 just as we’d accept in RIOs. 103 if (any(directlyConnectedRoutes, not(InitialConfiguration::isPrefixLengthCompliant))) { 104 return false; 105 } 106 // There no more than one IPv4 address 107 if (ipAddresses.stream().filter(InitialConfiguration::isIPv4).count() > 1) { 108 return false; 109 } 110 111 return true; 112 } 113 114 /** 115 * @return true if the given list of addressess and routes satisfies provisioning for this 116 * InitialConfiguration. LinkAddresses and RouteInfo objects are not compared with equality 117 * because addresses and routes seen by Netlink will contain additional fields like flags, 118 * interfaces, and so on. If this InitialConfiguration has no IP address specified, the 119 * provisioning check always fails. 120 * 121 * If the given list of routes is null, only addresses are taken into considerations. 122 */ isProvisionedBy(List<LinkAddress> addresses, List<RouteInfo> routes)123 public boolean isProvisionedBy(List<LinkAddress> addresses, List<RouteInfo> routes) { 124 if (ipAddresses.isEmpty()) { 125 return false; 126 } 127 128 for (LinkAddress addr : ipAddresses) { 129 if (!any(addresses, (addrSeen) -> addr.isSameAddressAs(addrSeen))) { 130 return false; 131 } 132 } 133 134 if (routes != null) { 135 for (IpPrefix prefix : directlyConnectedRoutes) { 136 if (!any(routes, (routeSeen) -> isDirectlyConnectedRoute(routeSeen, prefix))) { 137 return false; 138 } 139 } 140 } 141 142 return true; 143 } 144 145 /** 146 * Convert this configuration to a {@link InitialConfigurationParcelable}. 147 */ toStableParcelable()148 public InitialConfigurationParcelable toStableParcelable() { 149 final InitialConfigurationParcelable p = new InitialConfigurationParcelable(); 150 p.ipAddresses = ipAddresses.toArray(new LinkAddress[0]); 151 p.directlyConnectedRoutes = directlyConnectedRoutes.toArray(new IpPrefix[0]); 152 p.dnsServers = toParcelableArray( 153 dnsServers, IpConfigurationParcelableUtil::parcelAddress, String.class); 154 return p; 155 } 156 157 /** 158 * Create an instance of {@link InitialConfiguration} based on the contents of the specified 159 * {@link InitialConfigurationParcelable}. 160 */ fromStableParcelable(InitialConfigurationParcelable p)161 public static InitialConfiguration fromStableParcelable(InitialConfigurationParcelable p) { 162 if (p == null) return null; 163 final InitialConfiguration config = new InitialConfiguration(); 164 config.ipAddresses.addAll(Arrays.asList(p.ipAddresses)); 165 config.directlyConnectedRoutes.addAll(Arrays.asList(p.directlyConnectedRoutes)); 166 config.dnsServers.addAll( 167 fromParcelableArray(p.dnsServers, IpConfigurationParcelableUtil::unparcelAddress)); 168 return config; 169 } 170 171 @Override equals(Object obj)172 public boolean equals(Object obj) { 173 if (!(obj instanceof InitialConfiguration)) return false; 174 final InitialConfiguration other = (InitialConfiguration) obj; 175 return ipAddresses.equals(other.ipAddresses) 176 && directlyConnectedRoutes.equals(other.directlyConnectedRoutes) 177 && dnsServers.equals(other.dnsServers); 178 } 179 isDirectlyConnectedRoute(RouteInfo route, IpPrefix prefix)180 private static boolean isDirectlyConnectedRoute(RouteInfo route, IpPrefix prefix) { 181 return !route.hasGateway() 182 && route.getType() == RouteInfo.RTN_UNICAST 183 && prefix.equals(route.getDestination()); 184 } 185 isPrefixLengthCompliant(LinkAddress addr)186 private static boolean isPrefixLengthCompliant(LinkAddress addr) { 187 return isIPv4(addr) || isCompliantIPv6PrefixLength(addr.getPrefixLength()); 188 } 189 isPrefixLengthCompliant(IpPrefix prefix)190 private static boolean isPrefixLengthCompliant(IpPrefix prefix) { 191 return isIPv4(prefix) || isCompliantIPv6PrefixLength(prefix.getPrefixLength()); 192 } 193 isCompliantIPv6PrefixLength(int prefixLength)194 private static boolean isCompliantIPv6PrefixLength(int prefixLength) { 195 return (RFC6177_MIN_PREFIX_LENGTH <= prefixLength) 196 && (prefixLength <= RFC7421_PREFIX_LENGTH); 197 } 198 isIPv4(IpPrefix prefix)199 private static boolean isIPv4(IpPrefix prefix) { 200 return prefix.getAddress() instanceof Inet4Address; 201 } 202 isIPv4(LinkAddress addr)203 private static boolean isIPv4(LinkAddress addr) { 204 return addr.getAddress() instanceof Inet4Address; 205 } 206 isIPv6DefaultRoute(IpPrefix prefix)207 private static boolean isIPv6DefaultRoute(IpPrefix prefix) { 208 return prefix.getAddress().equals(INET6_ANY); 209 } 210 isIPv6GUA(LinkAddress addr)211 private static boolean isIPv6GUA(LinkAddress addr) { 212 return addr.isIpv6() && addr.isGlobalPreferred(); 213 } 214 215 // TODO: extract out into CollectionUtils. 216 217 /** 218 * Indicate whether any element of the specified iterable verifies the specified predicate. 219 */ any(Iterable<T> coll, Predicate<T> fn)220 public static <T> boolean any(Iterable<T> coll, Predicate<T> fn) { 221 for (T t : coll) { 222 if (fn.test(t)) { 223 return true; 224 } 225 } 226 return false; 227 } 228 229 /** 230 * Indicate whether all elements of the specified iterable verifies the specified predicate. 231 */ all(Iterable<T> coll, Predicate<T> fn)232 public static <T> boolean all(Iterable<T> coll, Predicate<T> fn) { 233 return !any(coll, not(fn)); 234 } 235 236 /** 237 * Create a predicate that returns the opposite value of the specified predicate. 238 */ not(Predicate<T> fn)239 public static <T> Predicate<T> not(Predicate<T> fn) { 240 return (t) -> !fn.test(t); 241 } 242 } 243