1 /* 2 * Copyright (C) 2016 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 com.android.server.wifi; 18 19 import static com.android.server.wifi.util.NativeUtil.addEnclosingQuotes; 20 21 import android.content.pm.UserInfo; 22 import android.net.IpConfiguration; 23 import android.net.MacAddress; 24 import android.net.StaticIpConfiguration; 25 import android.net.wifi.WifiConfiguration; 26 import android.net.wifi.WifiEnterpriseConfig; 27 import android.net.wifi.WifiNetworkSpecifier; 28 import android.net.wifi.WifiScanner; 29 import android.os.PatternMatcher; 30 import android.os.UserHandle; 31 import android.text.TextUtils; 32 import android.util.Log; 33 import android.util.Pair; 34 35 import com.android.internal.annotations.VisibleForTesting; 36 import com.android.server.wifi.util.NativeUtil; 37 import com.android.server.wifi.util.TelephonyUtil; 38 39 import java.nio.charset.StandardCharsets; 40 import java.security.cert.X509Certificate; 41 import java.util.Arrays; 42 import java.util.BitSet; 43 import java.util.Comparator; 44 import java.util.List; 45 import java.util.Objects; 46 47 /** 48 * WifiConfiguration utility for any {@link android.net.wifi.WifiConfiguration} related operations. 49 * Currently contains: 50 * > Helper method to check if the WifiConfiguration object is visible to the provided users. 51 * > Helper methods to identify the encryption of a WifiConfiguration object. 52 */ 53 public class WifiConfigurationUtil { 54 private static final String TAG = "WifiConfigurationUtil"; 55 56 /** 57 * Constants used for validating external config objects. 58 */ 59 private static final int ENCLOSING_QUOTES_LEN = 2; 60 private static final int SSID_UTF_8_MIN_LEN = 1 + ENCLOSING_QUOTES_LEN; 61 private static final int SSID_UTF_8_MAX_LEN = 32 + ENCLOSING_QUOTES_LEN; 62 private static final int SSID_HEX_MIN_LEN = 2; 63 private static final int SSID_HEX_MAX_LEN = 64; 64 private static final int PSK_ASCII_MIN_LEN = 8 + ENCLOSING_QUOTES_LEN; 65 private static final int SAE_ASCII_MIN_LEN = 1 + ENCLOSING_QUOTES_LEN; 66 private static final int PSK_SAE_ASCII_MAX_LEN = 63 + ENCLOSING_QUOTES_LEN; 67 private static final int PSK_SAE_HEX_LEN = 64; 68 @VisibleForTesting 69 public static final String PASSWORD_MASK = "*"; 70 private static final String MATCH_EMPTY_SSID_PATTERN_PATH = ""; 71 private static final Pair<MacAddress, MacAddress> MATCH_NONE_BSSID_PATTERN = 72 new Pair(MacAddress.BROADCAST_ADDRESS, MacAddress.BROADCAST_ADDRESS); 73 private static final Pair<MacAddress, MacAddress> MATCH_ALL_BSSID_PATTERN = 74 new Pair(MacAddress.ALL_ZEROS_ADDRESS, MacAddress.ALL_ZEROS_ADDRESS); 75 76 /** 77 * Check whether a network configuration is visible to a user or any of its managed profiles. 78 * 79 * @param config the network configuration whose visibility should be checked 80 * @param profiles the user IDs of the user itself and all its managed profiles (can be obtained 81 * via {@link android.os.UserManager#getProfiles}) 82 * @return whether the network configuration is visible to the user or any of its managed 83 * profiles 84 */ isVisibleToAnyProfile(WifiConfiguration config, List<UserInfo> profiles)85 public static boolean isVisibleToAnyProfile(WifiConfiguration config, List<UserInfo> profiles) { 86 return (config.shared || doesUidBelongToAnyProfile(config.creatorUid, profiles)); 87 } 88 89 /** 90 * Check whether a uid belong to a user or any of its managed profiles. 91 * 92 * @param uid uid of the app. 93 * @param profiles the user IDs of the user itself and all its managed profiles (can be obtained 94 * via {@link android.os.UserManager#getProfiles}) 95 * @return whether the uid belongs to the user or any of its managed profiles. 96 */ doesUidBelongToAnyProfile(int uid, List<UserInfo> profiles)97 public static boolean doesUidBelongToAnyProfile(int uid, List<UserInfo> profiles) { 98 final int userId = UserHandle.getUserId(uid); 99 for (UserInfo profile : profiles) { 100 if (profile.id == userId) { 101 return true; 102 } 103 } 104 return false; 105 } 106 107 /** 108 * Checks if the provided |wepKeys| array contains any non-null value; 109 */ hasAnyValidWepKey(String[] wepKeys)110 public static boolean hasAnyValidWepKey(String[] wepKeys) { 111 for (int i = 0; i < wepKeys.length; i++) { 112 if (wepKeys[i] != null) { 113 return true; 114 } 115 } 116 return false; 117 } 118 119 /** 120 * Helper method to check if the provided |config| corresponds to a PSK network or not. 121 */ isConfigForPskNetwork(WifiConfiguration config)122 public static boolean isConfigForPskNetwork(WifiConfiguration config) { 123 return config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_PSK); 124 } 125 126 /** 127 * Helper method to check if the provided |config| corresponds to an SAE network or not. 128 */ isConfigForSaeNetwork(WifiConfiguration config)129 public static boolean isConfigForSaeNetwork(WifiConfiguration config) { 130 return config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.SAE); 131 } 132 133 /** 134 * Helper method to check if the provided |config| corresponds to an OWE network or not. 135 */ isConfigForOweNetwork(WifiConfiguration config)136 public static boolean isConfigForOweNetwork(WifiConfiguration config) { 137 return config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.OWE); 138 } 139 140 /** 141 * Helper method to check if the provided |config| corresponds to a EAP network or not. 142 */ isConfigForEapNetwork(WifiConfiguration config)143 public static boolean isConfigForEapNetwork(WifiConfiguration config) { 144 return (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_EAP) 145 || config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.IEEE8021X)); 146 } 147 148 /** 149 * Helper method to check if the provided |config| corresponds to a EAP Suite-B network or not. 150 */ isConfigForEapSuiteBNetwork(WifiConfiguration config)151 public static boolean isConfigForEapSuiteBNetwork(WifiConfiguration config) { 152 return config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.SUITE_B_192); 153 } 154 155 /** 156 * Helper method to check if the provided |config| corresponds to a WEP network or not. 157 */ isConfigForWepNetwork(WifiConfiguration config)158 public static boolean isConfigForWepNetwork(WifiConfiguration config) { 159 return (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.NONE) 160 && hasAnyValidWepKey(config.wepKeys)); 161 } 162 163 /** 164 * Helper method to check if the provided |config| corresponds to an open or enhanced 165 * open network, or not. 166 */ isConfigForOpenNetwork(WifiConfiguration config)167 public static boolean isConfigForOpenNetwork(WifiConfiguration config) { 168 return (!(isConfigForWepNetwork(config) || isConfigForPskNetwork(config) 169 || isConfigForEapNetwork(config) || isConfigForSaeNetwork(config) 170 || isConfigForEapSuiteBNetwork(config))); 171 } 172 173 /** 174 * Compare existing and new WifiConfiguration objects after a network update and return if 175 * IP parameters have changed or not. 176 * 177 * @param existingConfig Existing WifiConfiguration object corresponding to the network. 178 * @param newConfig New WifiConfiguration object corresponding to the network. 179 * @return true if IP parameters have changed, false otherwise. 180 */ hasIpChanged(WifiConfiguration existingConfig, WifiConfiguration newConfig)181 public static boolean hasIpChanged(WifiConfiguration existingConfig, 182 WifiConfiguration newConfig) { 183 if (existingConfig.getIpAssignment() != newConfig.getIpAssignment()) { 184 return true; 185 } 186 if (newConfig.getIpAssignment() == IpConfiguration.IpAssignment.STATIC) { 187 return !Objects.equals(existingConfig.getStaticIpConfiguration(), 188 newConfig.getStaticIpConfiguration()); 189 } 190 return false; 191 } 192 193 /** 194 * Compare existing and new WifiConfiguration objects after a network update and return if 195 * proxy parameters have changed or not. 196 * 197 * @param existingConfig Existing WifiConfiguration object corresponding to the network. 198 * @param newConfig New WifiConfiguration object corresponding to the network. 199 * @return true if proxy parameters have changed, false if no existing config and proxy settings 200 * are NONE, false otherwise. 201 */ hasProxyChanged(WifiConfiguration existingConfig, WifiConfiguration newConfig)202 public static boolean hasProxyChanged(WifiConfiguration existingConfig, 203 WifiConfiguration newConfig) { 204 if (existingConfig == null) { 205 return newConfig.getProxySettings() != IpConfiguration.ProxySettings.NONE; 206 } 207 if (newConfig.getProxySettings() != existingConfig.getProxySettings()) { 208 return true; 209 } 210 return !Objects.equals(existingConfig.getHttpProxy(), newConfig.getHttpProxy()); 211 } 212 213 /** 214 * Compare existing and new WifiConfiguration objects after a network update and return if 215 * MAC randomization setting has changed or not. 216 * @param existingConfig Existing WifiConfiguration object corresponding to the network. 217 * @param newConfig New WifiConfiguration object corresponding to the network. 218 * @return true if MAC randomization setting setting changed or the existing confiuration is 219 * null and the newConfig is setting macRandomizationSetting to the default value. 220 */ hasMacRandomizationSettingsChanged(WifiConfiguration existingConfig, WifiConfiguration newConfig)221 public static boolean hasMacRandomizationSettingsChanged(WifiConfiguration existingConfig, 222 WifiConfiguration newConfig) { 223 if (existingConfig == null) { 224 return newConfig.macRandomizationSetting != WifiConfiguration.RANDOMIZATION_PERSISTENT; 225 } 226 return newConfig.macRandomizationSetting != existingConfig.macRandomizationSetting; 227 } 228 229 /** 230 * Compare existing and new WifiEnterpriseConfig objects after a network update and return if 231 * credential parameters have changed or not. 232 * 233 * @param existingEnterpriseConfig Existing WifiConfiguration object corresponding to the 234 * network. 235 * @param newEnterpriseConfig New WifiConfiguration object corresponding to the network. 236 * @return true if credentials have changed, false otherwise. 237 */ 238 @VisibleForTesting hasEnterpriseConfigChanged(WifiEnterpriseConfig existingEnterpriseConfig, WifiEnterpriseConfig newEnterpriseConfig)239 public static boolean hasEnterpriseConfigChanged(WifiEnterpriseConfig existingEnterpriseConfig, 240 WifiEnterpriseConfig newEnterpriseConfig) { 241 if (existingEnterpriseConfig != null && newEnterpriseConfig != null) { 242 if (existingEnterpriseConfig.getEapMethod() != newEnterpriseConfig.getEapMethod()) { 243 return true; 244 } 245 if (existingEnterpriseConfig.getPhase2Method() 246 != newEnterpriseConfig.getPhase2Method()) { 247 return true; 248 } 249 if (!TextUtils.equals(existingEnterpriseConfig.getIdentity(), 250 newEnterpriseConfig.getIdentity())) { 251 return true; 252 } 253 if (!TelephonyUtil.isSimEapMethod(existingEnterpriseConfig.getEapMethod()) 254 && !TextUtils.equals(existingEnterpriseConfig.getAnonymousIdentity(), 255 newEnterpriseConfig.getAnonymousIdentity())) { 256 return true; 257 } 258 if (!TextUtils.equals(existingEnterpriseConfig.getPassword(), 259 newEnterpriseConfig.getPassword())) { 260 return true; 261 } 262 X509Certificate[] existingCaCerts = existingEnterpriseConfig.getCaCertificates(); 263 X509Certificate[] newCaCerts = newEnterpriseConfig.getCaCertificates(); 264 if (!Arrays.equals(existingCaCerts, newCaCerts)) { 265 return true; 266 } 267 } else { 268 // One of the configs may have an enterpriseConfig 269 if (existingEnterpriseConfig != null || newEnterpriseConfig != null) { 270 return true; 271 } 272 } 273 return false; 274 } 275 276 /** 277 * Compare existing and new WifiConfiguration objects after a network update and return if 278 * credential parameters have changed or not. 279 * 280 * @param existingConfig Existing WifiConfiguration object corresponding to the network. 281 * @param newConfig New WifiConfiguration object corresponding to the network. 282 * @return true if credentials have changed, false otherwise. 283 */ hasCredentialChanged(WifiConfiguration existingConfig, WifiConfiguration newConfig)284 public static boolean hasCredentialChanged(WifiConfiguration existingConfig, 285 WifiConfiguration newConfig) { 286 if (!Objects.equals(existingConfig.allowedKeyManagement, 287 newConfig.allowedKeyManagement)) { 288 return true; 289 } 290 if (!Objects.equals(existingConfig.allowedProtocols, newConfig.allowedProtocols)) { 291 return true; 292 } 293 if (!Objects.equals(existingConfig.allowedAuthAlgorithms, 294 newConfig.allowedAuthAlgorithms)) { 295 return true; 296 } 297 if (!Objects.equals(existingConfig.allowedPairwiseCiphers, 298 newConfig.allowedPairwiseCiphers)) { 299 return true; 300 } 301 if (!Objects.equals(existingConfig.allowedGroupCiphers, 302 newConfig.allowedGroupCiphers)) { 303 return true; 304 } 305 if (!Objects.equals(existingConfig.allowedGroupManagementCiphers, 306 newConfig.allowedGroupManagementCiphers)) { 307 return true; 308 } 309 if (!Objects.equals(existingConfig.allowedSuiteBCiphers, 310 newConfig.allowedSuiteBCiphers)) { 311 return true; 312 } 313 if (!Objects.equals(existingConfig.preSharedKey, newConfig.preSharedKey)) { 314 return true; 315 } 316 if (!Arrays.equals(existingConfig.wepKeys, newConfig.wepKeys)) { 317 return true; 318 } 319 if (existingConfig.wepTxKeyIndex != newConfig.wepTxKeyIndex) { 320 return true; 321 } 322 if (existingConfig.hiddenSSID != newConfig.hiddenSSID) { 323 return true; 324 } 325 if (existingConfig.requirePMF != newConfig.requirePMF) { 326 return true; 327 } 328 if (hasEnterpriseConfigChanged(existingConfig.enterpriseConfig, 329 newConfig.enterpriseConfig)) { 330 return true; 331 } 332 return false; 333 } 334 validateSsid(String ssid, boolean isAdd)335 private static boolean validateSsid(String ssid, boolean isAdd) { 336 if (isAdd) { 337 if (ssid == null) { 338 Log.e(TAG, "validateSsid : null string"); 339 return false; 340 } 341 } else { 342 if (ssid == null) { 343 // This is an update, so the SSID can be null if that is not being changed. 344 return true; 345 } 346 } 347 if (ssid.isEmpty()) { 348 Log.e(TAG, "validateSsid failed: empty string"); 349 return false; 350 } 351 if (ssid.startsWith("\"")) { 352 // UTF-8 SSID string 353 byte[] ssidBytes = ssid.getBytes(StandardCharsets.UTF_8); 354 if (ssidBytes.length < SSID_UTF_8_MIN_LEN) { 355 Log.e(TAG, "validateSsid failed: utf-8 ssid string size too small: " 356 + ssidBytes.length); 357 return false; 358 } 359 if (ssidBytes.length > SSID_UTF_8_MAX_LEN) { 360 Log.e(TAG, "validateSsid failed: utf-8 ssid string size too large: " 361 + ssidBytes.length); 362 return false; 363 } 364 } else { 365 // HEX SSID string 366 if (ssid.length() < SSID_HEX_MIN_LEN) { 367 Log.e(TAG, "validateSsid failed: hex string size too small: " + ssid.length()); 368 return false; 369 } 370 if (ssid.length() > SSID_HEX_MAX_LEN) { 371 Log.e(TAG, "validateSsid failed: hex string size too large: " + ssid.length()); 372 return false; 373 } 374 } 375 try { 376 NativeUtil.decodeSsid(ssid); 377 } catch (IllegalArgumentException e) { 378 Log.e(TAG, "validateSsid failed: malformed string: " + ssid); 379 return false; 380 } 381 return true; 382 } 383 validateBssid(MacAddress bssid)384 private static boolean validateBssid(MacAddress bssid) { 385 if (bssid == null) return true; 386 if (bssid.getAddressType() != MacAddress.TYPE_UNICAST) { 387 Log.e(TAG, "validateBssid failed: invalid bssid"); 388 return false; 389 } 390 return true; 391 } 392 validateBssid(String bssid)393 private static boolean validateBssid(String bssid) { 394 if (bssid == null) return true; 395 if (bssid.isEmpty()) { 396 Log.e(TAG, "validateBssid failed: empty string"); 397 return false; 398 } 399 MacAddress bssidMacAddress; 400 try { 401 bssidMacAddress = MacAddress.fromString(bssid); 402 } catch (IllegalArgumentException e) { 403 Log.e(TAG, "validateBssid failed: malformed string: " + bssid); 404 return false; 405 } 406 if (!validateBssid(bssidMacAddress)) { 407 return false; 408 } 409 return true; 410 } 411 validatePassword(String password, boolean isAdd, boolean isSae)412 private static boolean validatePassword(String password, boolean isAdd, boolean isSae) { 413 if (isAdd) { 414 if (password == null) { 415 Log.e(TAG, "validatePassword: null string"); 416 return false; 417 } 418 } else { 419 if (password == null) { 420 // This is an update, so the psk can be null if that is not being changed. 421 return true; 422 } else if (password.equals(PASSWORD_MASK)) { 423 // This is an update, so the app might have returned back the masked password, let 424 // it thru. WifiConfigManager will handle it. 425 return true; 426 } 427 } 428 if (password.isEmpty()) { 429 Log.e(TAG, "validatePassword failed: empty string"); 430 return false; 431 } 432 if (password.startsWith("\"")) { 433 // ASCII PSK string 434 byte[] passwordBytes = password.getBytes(StandardCharsets.US_ASCII); 435 int targetMinLength; 436 437 if (isSae) { 438 targetMinLength = SAE_ASCII_MIN_LEN; 439 } else { 440 targetMinLength = PSK_ASCII_MIN_LEN; 441 } 442 if (passwordBytes.length < targetMinLength) { 443 Log.e(TAG, "validatePassword failed: ASCII string size too small: " 444 + passwordBytes.length); 445 return false; 446 } 447 if (passwordBytes.length > PSK_SAE_ASCII_MAX_LEN) { 448 Log.e(TAG, "validatePassword failed: ASCII string size too large: " 449 + passwordBytes.length); 450 return false; 451 } 452 } else { 453 // HEX PSK string 454 if (password.length() != PSK_SAE_HEX_LEN) { 455 Log.e(TAG, "validatePassword failed: hex string size mismatch: " 456 + password.length()); 457 return false; 458 } 459 } 460 try { 461 NativeUtil.hexOrQuotedStringToBytes(password); 462 } catch (IllegalArgumentException e) { 463 Log.e(TAG, "validatePassword failed: malformed string: " + password); 464 return false; 465 } 466 return true; 467 } 468 validateBitSet(BitSet bitSet, int validValuesLength)469 private static boolean validateBitSet(BitSet bitSet, int validValuesLength) { 470 if (bitSet == null) return false; 471 BitSet clonedBitset = (BitSet) bitSet.clone(); 472 clonedBitset.clear(0, validValuesLength); 473 return clonedBitset.isEmpty(); 474 } 475 validateBitSets(WifiConfiguration config)476 private static boolean validateBitSets(WifiConfiguration config) { 477 // 1. Check |allowedKeyManagement|. 478 if (!validateBitSet(config.allowedKeyManagement, 479 WifiConfiguration.KeyMgmt.strings.length)) { 480 Log.e(TAG, "validateBitsets failed: invalid allowedKeyManagement bitset " 481 + config.allowedKeyManagement); 482 return false; 483 } 484 // 2. Check |allowedProtocols|. 485 if (!validateBitSet(config.allowedProtocols, 486 WifiConfiguration.Protocol.strings.length)) { 487 Log.e(TAG, "validateBitsets failed: invalid allowedProtocols bitset " 488 + config.allowedProtocols); 489 return false; 490 } 491 // 3. Check |allowedAuthAlgorithms|. 492 if (!validateBitSet(config.allowedAuthAlgorithms, 493 WifiConfiguration.AuthAlgorithm.strings.length)) { 494 Log.e(TAG, "validateBitsets failed: invalid allowedAuthAlgorithms bitset " 495 + config.allowedAuthAlgorithms); 496 return false; 497 } 498 // 4. Check |allowedGroupCiphers|. 499 if (!validateBitSet(config.allowedGroupCiphers, 500 WifiConfiguration.GroupCipher.strings.length)) { 501 Log.e(TAG, "validateBitsets failed: invalid allowedGroupCiphers bitset " 502 + config.allowedGroupCiphers); 503 return false; 504 } 505 // 5. Check |allowedPairwiseCiphers|. 506 if (!validateBitSet(config.allowedPairwiseCiphers, 507 WifiConfiguration.PairwiseCipher.strings.length)) { 508 Log.e(TAG, "validateBitsets failed: invalid allowedPairwiseCiphers bitset " 509 + config.allowedPairwiseCiphers); 510 return false; 511 } 512 return true; 513 } 514 validateKeyMgmt(BitSet keyMgmnt)515 private static boolean validateKeyMgmt(BitSet keyMgmnt) { 516 if (keyMgmnt.cardinality() > 1) { 517 if (keyMgmnt.cardinality() > 3) { 518 Log.e(TAG, "validateKeyMgmt failed: cardinality > 3"); 519 return false; 520 } 521 if (!keyMgmnt.get(WifiConfiguration.KeyMgmt.WPA_EAP)) { 522 Log.e(TAG, "validateKeyMgmt failed: not WPA_EAP"); 523 return false; 524 } 525 if (!keyMgmnt.get(WifiConfiguration.KeyMgmt.IEEE8021X) 526 && !keyMgmnt.get(WifiConfiguration.KeyMgmt.WPA_PSK)) { 527 Log.e(TAG, "validateKeyMgmt failed: not PSK or 8021X"); 528 return false; 529 } 530 if (keyMgmnt.cardinality() == 3 531 && !keyMgmnt.get(WifiConfiguration.KeyMgmt.SUITE_B_192)) { 532 Log.e(TAG, "validateKeyMgmt failed: not SUITE_B_192"); 533 return false; 534 } 535 } 536 return true; 537 } 538 validateIpConfiguration(IpConfiguration ipConfig)539 private static boolean validateIpConfiguration(IpConfiguration ipConfig) { 540 if (ipConfig == null) { 541 Log.e(TAG, "validateIpConfiguration failed: null IpConfiguration"); 542 return false; 543 } 544 if (ipConfig.getIpAssignment() == IpConfiguration.IpAssignment.STATIC) { 545 StaticIpConfiguration staticIpConfig = ipConfig.getStaticIpConfiguration(); 546 if (staticIpConfig == null) { 547 Log.e(TAG, "validateIpConfiguration failed: null StaticIpConfiguration"); 548 return false; 549 } 550 if (staticIpConfig.ipAddress == null) { 551 Log.e(TAG, "validateIpConfiguration failed: null static ip Address"); 552 return false; 553 } 554 } 555 return true; 556 } 557 558 /** 559 * Enums to specify if the provided config is being validated for add or update. 560 */ 561 public static final boolean VALIDATE_FOR_ADD = true; 562 public static final boolean VALIDATE_FOR_UPDATE = false; 563 564 /** 565 * Validate the configuration received from an external application. 566 * 567 * This method checks for the following parameters: 568 * 1. {@link WifiConfiguration#SSID} 569 * 2. {@link WifiConfiguration#BSSID} 570 * 3. {@link WifiConfiguration#preSharedKey} 571 * 4. {@link WifiConfiguration#allowedKeyManagement} 572 * 5. {@link WifiConfiguration#allowedProtocols} 573 * 6. {@link WifiConfiguration#allowedAuthAlgorithms} 574 * 7. {@link WifiConfiguration#allowedGroupCiphers} 575 * 8. {@link WifiConfiguration#allowedPairwiseCiphers} 576 * 9. {@link WifiConfiguration#getIpConfiguration()} 577 * 578 * @param config {@link WifiConfiguration} received from an external application. 579 * @param isAdd {@link #VALIDATE_FOR_ADD} to indicate a network config received for an add, 580 * {@link #VALIDATE_FOR_UPDATE} for a network config received for an update. 581 * These 2 cases need to be handled differently because the config received for an 582 * update could contain only the fields that are being changed. 583 * @return true if the parameters are valid, false otherwise. 584 */ validate(WifiConfiguration config, boolean isAdd)585 public static boolean validate(WifiConfiguration config, boolean isAdd) { 586 if (!validateSsid(config.SSID, isAdd)) { 587 return false; 588 } 589 if (!validateBssid(config.BSSID)) { 590 return false; 591 } 592 if (!validateBitSets(config)) { 593 return false; 594 } 595 if (!validateKeyMgmt(config.allowedKeyManagement)) { 596 return false; 597 } 598 if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_PSK) 599 && !validatePassword(config.preSharedKey, isAdd, false)) { 600 return false; 601 } 602 if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.OWE)) { 603 // PMF mandatory for OWE networks 604 if (!config.requirePMF) { 605 return false; 606 } 607 } 608 if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.SAE)) { 609 // PMF mandatory for WPA3-Personal networks 610 if (!config.requirePMF) { 611 return false; 612 } 613 if (!validatePassword(config.preSharedKey, isAdd, true)) { 614 return false; 615 } 616 } 617 if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.SUITE_B_192)) { 618 // PMF mandatory for WPA3-Enterprise networks 619 if (!config.requirePMF) { 620 return false; 621 } 622 } 623 if (!validateIpConfiguration(config.getIpConfiguration())) { 624 return false; 625 } 626 // TBD: Validate some enterprise params as well in the future here. 627 return true; 628 } 629 validateBssidPattern( Pair<MacAddress, MacAddress> bssidPatternMatcher)630 private static boolean validateBssidPattern( 631 Pair<MacAddress, MacAddress> bssidPatternMatcher) { 632 if (bssidPatternMatcher == null) return true; 633 MacAddress baseAddress = bssidPatternMatcher.first; 634 MacAddress mask = bssidPatternMatcher.second; 635 if (baseAddress.getAddressType() != MacAddress.TYPE_UNICAST) { 636 Log.e(TAG, "validateBssidPatternMatcher failed : invalid base address: " + baseAddress); 637 return false; 638 } 639 if (mask.equals(MacAddress.ALL_ZEROS_ADDRESS) 640 && !baseAddress.equals(MacAddress.ALL_ZEROS_ADDRESS)) { 641 Log.e(TAG, "validateBssidPatternMatcher failed : invalid mask/base: " + mask + "/" 642 + baseAddress); 643 return false; 644 } 645 // TBD: Can we do any more sanity checks? 646 return true; 647 } 648 649 // TODO(b/113878056): Some of this is duplicated in {@link WifiNetworkConfigBuilder}. 650 // Merge them somehow?. isValidNetworkSpecifier(WifiNetworkSpecifier specifier)651 private static boolean isValidNetworkSpecifier(WifiNetworkSpecifier specifier) { 652 PatternMatcher ssidPatternMatcher = specifier.ssidPatternMatcher; 653 Pair<MacAddress, MacAddress> bssidPatternMatcher = specifier.bssidPatternMatcher; 654 if (ssidPatternMatcher == null || bssidPatternMatcher == null) { 655 return false; 656 } 657 if (ssidPatternMatcher.getPath() == null || bssidPatternMatcher.first == null 658 || bssidPatternMatcher.second == null) { 659 return false; 660 } 661 return true; 662 } 663 isMatchNoneNetworkSpecifier(WifiNetworkSpecifier specifier)664 private static boolean isMatchNoneNetworkSpecifier(WifiNetworkSpecifier specifier) { 665 PatternMatcher ssidPatternMatcher = specifier.ssidPatternMatcher; 666 Pair<MacAddress, MacAddress> bssidPatternMatcher = specifier.bssidPatternMatcher; 667 if (ssidPatternMatcher.getType() != PatternMatcher.PATTERN_PREFIX 668 && ssidPatternMatcher.getPath().equals(MATCH_EMPTY_SSID_PATTERN_PATH)) { 669 return true; 670 } 671 if (bssidPatternMatcher.equals(MATCH_NONE_BSSID_PATTERN)) { 672 return true; 673 } 674 return false; 675 } 676 isMatchAllNetworkSpecifier(WifiNetworkSpecifier specifier)677 private static boolean isMatchAllNetworkSpecifier(WifiNetworkSpecifier specifier) { 678 PatternMatcher ssidPatternMatcher = specifier.ssidPatternMatcher; 679 Pair<MacAddress, MacAddress> bssidPatternMatcher = specifier.bssidPatternMatcher; 680 if (ssidPatternMatcher.match(MATCH_EMPTY_SSID_PATTERN_PATH) 681 && bssidPatternMatcher.equals(MATCH_ALL_BSSID_PATTERN)) { 682 return true; 683 } 684 return false; 685 } 686 687 /** 688 * Validate the configuration received from an external application inside 689 * {@link WifiNetworkSpecifier}. 690 * 691 * This method checks for the following parameters: 692 * 1. {@link WifiNetworkSpecifier#ssidPatternMatcher} 693 * 2. {@link WifiNetworkSpecifier#bssidPatternMatcher} 694 * 3. {@link WifiConfiguration#SSID} 695 * 4. {@link WifiConfiguration#BSSID} 696 * 5. {@link WifiConfiguration#preSharedKey} 697 * 6. {@link WifiConfiguration#allowedKeyManagement} 698 * 7. {@link WifiConfiguration#allowedProtocols} 699 * 8. {@link WifiConfiguration#allowedAuthAlgorithms} 700 * 9. {@link WifiConfiguration#allowedGroupCiphers} 701 * 10. {@link WifiConfiguration#allowedPairwiseCiphers} 702 * 11. {@link WifiConfiguration#getIpConfiguration()} 703 * 704 * @param specifier Instance of {@link WifiNetworkSpecifier}. 705 * @return true if the parameters are valid, false otherwise. 706 */ validateNetworkSpecifier(WifiNetworkSpecifier specifier)707 public static boolean validateNetworkSpecifier(WifiNetworkSpecifier specifier) { 708 if (!isValidNetworkSpecifier(specifier)) { 709 Log.e(TAG, "validateNetworkSpecifier failed : invalid network specifier"); 710 return false; 711 } 712 if (isMatchNoneNetworkSpecifier(specifier)) { 713 Log.e(TAG, "validateNetworkSpecifier failed : match-none specifier"); 714 return false; 715 } 716 if (isMatchAllNetworkSpecifier(specifier)) { 717 Log.e(TAG, "validateNetworkSpecifier failed : match-all specifier"); 718 return false; 719 } 720 WifiConfiguration config = specifier.wifiConfiguration; 721 if (specifier.ssidPatternMatcher.getType() == PatternMatcher.PATTERN_LITERAL) { 722 // For literal SSID matches, the value should satisfy SSID requirements. 723 // WifiConfiguration.SSID needs quotes around ASCII SSID. 724 if (!validateSsid(addEnclosingQuotes(specifier.ssidPatternMatcher.getPath()), true)) { 725 return false; 726 } 727 } else { 728 if (config.hiddenSSID) { 729 Log.e(TAG, "validateNetworkSpecifier failed : ssid pattern not supported " 730 + "for hidden networks"); 731 return false; 732 } 733 } 734 if (Objects.equals(specifier.bssidPatternMatcher.second, MacAddress.BROADCAST_ADDRESS)) { 735 // For literal BSSID matches, the value should satisfy MAC address requirements. 736 if (!validateBssid(specifier.bssidPatternMatcher.first)) { 737 return false; 738 } 739 } else { 740 if (!validateBssidPattern(specifier.bssidPatternMatcher)) { 741 return false; 742 } 743 } 744 if (!validateBitSets(config)) { 745 return false; 746 } 747 if (!validateKeyMgmt(config.allowedKeyManagement)) { 748 return false; 749 } 750 if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_PSK) 751 && !validatePassword(config.preSharedKey, true, false)) { 752 return false; 753 } 754 if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.OWE)) { 755 // PMF mandatory for OWE networks 756 if (!config.requirePMF) { 757 return false; 758 } 759 } 760 if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.SAE)) { 761 // PMF mandatory for WPA3-Personal networks 762 if (!config.requirePMF) { 763 return false; 764 } 765 if (!validatePassword(config.preSharedKey, true, true)) { 766 return false; 767 } 768 } 769 if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.SUITE_B_192)) { 770 // PMF mandatory for WPA3-Enterprise networks 771 if (!config.requirePMF) { 772 return false; 773 } 774 } 775 // TBD: Validate some enterprise params as well in the future here. 776 return true; 777 } 778 779 /** 780 * Check if the provided two networks are the same. 781 * Note: This does not check if network selection BSSID's are the same. 782 * 783 * @param config Configuration corresponding to a network. 784 * @param config1 Configuration corresponding to another network. 785 * 786 * @return true if |config| and |config1| are the same network. 787 * false otherwise. 788 */ isSameNetwork(WifiConfiguration config, WifiConfiguration config1)789 public static boolean isSameNetwork(WifiConfiguration config, WifiConfiguration config1) { 790 if (config == null && config1 == null) { 791 return true; 792 } 793 if (config == null || config1 == null) { 794 return false; 795 } 796 if (config.networkId != config1.networkId) { 797 return false; 798 } 799 if (!Objects.equals(config.SSID, config1.SSID)) { 800 return false; 801 } 802 if (WifiConfigurationUtil.hasCredentialChanged(config, config1)) { 803 return false; 804 } 805 return true; 806 } 807 808 /** 809 * Create a PnoNetwork object from the provided WifiConfiguration. 810 * 811 * @param config Configuration corresponding to the network. 812 * @return PnoNetwork object corresponding to the network. 813 */ createPnoNetwork( WifiConfiguration config)814 public static WifiScanner.PnoSettings.PnoNetwork createPnoNetwork( 815 WifiConfiguration config) { 816 WifiScanner.PnoSettings.PnoNetwork pnoNetwork = 817 new WifiScanner.PnoSettings.PnoNetwork(config.SSID); 818 if (config.hiddenSSID) { 819 pnoNetwork.flags |= WifiScanner.PnoSettings.PnoNetwork.FLAG_DIRECTED_SCAN; 820 } 821 pnoNetwork.flags |= WifiScanner.PnoSettings.PnoNetwork.FLAG_A_BAND; 822 pnoNetwork.flags |= WifiScanner.PnoSettings.PnoNetwork.FLAG_G_BAND; 823 if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_PSK)) { 824 pnoNetwork.authBitField |= WifiScanner.PnoSettings.PnoNetwork.AUTH_CODE_PSK; 825 } else if (config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.WPA_EAP) 826 || config.allowedKeyManagement.get(WifiConfiguration.KeyMgmt.IEEE8021X)) { 827 pnoNetwork.authBitField |= WifiScanner.PnoSettings.PnoNetwork.AUTH_CODE_EAPOL; 828 } else { 829 pnoNetwork.authBitField |= WifiScanner.PnoSettings.PnoNetwork.AUTH_CODE_OPEN; 830 } 831 return pnoNetwork; 832 } 833 834 835 /** 836 * General WifiConfiguration list sorting algorithm: 837 * 1, Place the fully enabled networks first. 838 * 2. Next place all the temporarily disabled networks. 839 * 3. Place the permanently disabled networks last (Permanently disabled networks are removed 840 * before WifiConfigManager uses this comparator today!). 841 * 842 * Among the networks with the same status, sort them in the order determined by the return of 843 * {@link #compareNetworksWithSameStatus(WifiConfiguration, WifiConfiguration)} method 844 * implementation. 845 */ 846 public abstract static class WifiConfigurationComparator implements 847 Comparator<WifiConfiguration> { 848 private static final int ENABLED_NETWORK_SCORE = 3; 849 private static final int TEMPORARY_DISABLED_NETWORK_SCORE = 2; 850 private static final int PERMANENTLY_DISABLED_NETWORK_SCORE = 1; 851 852 @Override compare(WifiConfiguration a, WifiConfiguration b)853 public int compare(WifiConfiguration a, WifiConfiguration b) { 854 int configAScore = getNetworkStatusScore(a); 855 int configBScore = getNetworkStatusScore(b); 856 if (configAScore == configBScore) { 857 return compareNetworksWithSameStatus(a, b); 858 } else { 859 return Integer.compare(configBScore, configAScore); 860 } 861 } 862 863 // This needs to be implemented by the connected/disconnected PNO list comparator. compareNetworksWithSameStatus(WifiConfiguration a, WifiConfiguration b)864 abstract int compareNetworksWithSameStatus(WifiConfiguration a, WifiConfiguration b); 865 866 /** 867 * Returns an integer representing a score for each configuration. The scores are assigned 868 * based on the status of the configuration. The scores are assigned according to the order: 869 * Fully enabled network > Temporarily disabled network > Permanently disabled network. 870 */ getNetworkStatusScore(WifiConfiguration config)871 private int getNetworkStatusScore(WifiConfiguration config) { 872 if (config.getNetworkSelectionStatus().isNetworkEnabled()) { 873 return ENABLED_NETWORK_SCORE; 874 } else if (config.getNetworkSelectionStatus().isNetworkTemporaryDisabled()) { 875 return TEMPORARY_DISABLED_NETWORK_SCORE; 876 } else { 877 return PERMANENTLY_DISABLED_NETWORK_SCORE; 878 } 879 } 880 } 881 } 882