1 /*
2  * Copyright (C) 2008 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.wifi;
18 
19 import android.annotation.IntDef;
20 import android.annotation.NonNull;
21 import android.annotation.SystemApi;
22 import android.annotation.UnsupportedAppUsage;
23 import android.content.pm.PackageManager;
24 import android.net.IpConfiguration;
25 import android.net.IpConfiguration.ProxySettings;
26 import android.net.MacAddress;
27 import android.net.NetworkSpecifier;
28 import android.net.ProxyInfo;
29 import android.net.StaticIpConfiguration;
30 import android.net.Uri;
31 import android.os.Build;
32 import android.os.Parcel;
33 import android.os.Parcelable;
34 import android.os.SystemClock;
35 import android.os.UserHandle;
36 import android.text.TextUtils;
37 import android.util.BackupUtils;
38 import android.util.Log;
39 import android.util.TimeUtils;
40 
41 import java.io.ByteArrayOutputStream;
42 import java.io.DataInputStream;
43 import java.io.DataOutputStream;
44 import java.io.IOException;
45 import java.lang.annotation.Retention;
46 import java.lang.annotation.RetentionPolicy;
47 import java.util.Arrays;
48 import java.util.BitSet;
49 import java.util.HashMap;
50 
51 /**
52  * A class representing a configured Wi-Fi network, including the
53  * security configuration.
54  *
55  * @deprecated Use {@link WifiNetworkSpecifier.Builder} to create {@link NetworkSpecifier} and
56  * {@link WifiNetworkSuggestion.Builder} to create {@link WifiNetworkSuggestion}. This will become a
57  * system use only object in the future.
58  */
59 @Deprecated
60 public class WifiConfiguration implements Parcelable {
61     private static final String TAG = "WifiConfiguration";
62     /**
63      * Current Version of the Backup Serializer.
64     */
65     private static final int BACKUP_VERSION = 3;
66     /** {@hide} */
67     public static final String ssidVarName = "ssid";
68     /** {@hide} */
69     public static final String bssidVarName = "bssid";
70     /** {@hide} */
71     public static final String pskVarName = "psk";
72     /** {@hide} */
73     @Deprecated
74     @UnsupportedAppUsage
75     public static final String[] wepKeyVarNames = { "wep_key0", "wep_key1", "wep_key2", "wep_key3" };
76     /** {@hide} */
77     @Deprecated
78     public static final String wepTxKeyIdxVarName = "wep_tx_keyidx";
79     /** {@hide} */
80     public static final String priorityVarName = "priority";
81     /** {@hide} */
82     public static final String hiddenSSIDVarName = "scan_ssid";
83     /** {@hide} */
84     public static final String pmfVarName = "ieee80211w";
85     /** {@hide} */
86     public static final String updateIdentiferVarName = "update_identifier";
87     /** {@hide} */
88     public static final int INVALID_NETWORK_ID = -1;
89     /** {@hide} */
90     public static final int LOCAL_ONLY_NETWORK_ID = -2;
91 
92     /** {@hide} */
93     private String mPasspointManagementObjectTree;
94     /** {@hide} */
95     private static final int MAXIMUM_RANDOM_MAC_GENERATION_RETRY = 3;
96 
97     /**
98      * Recognized key management schemes.
99      */
100     public static class KeyMgmt {
KeyMgmt()101         private KeyMgmt() { }
102 
103         /** WPA is not used; plaintext or static WEP could be used. */
104         public static final int NONE = 0;
105         /** WPA pre-shared key (requires {@code preSharedKey} to be specified). */
106         public static final int WPA_PSK = 1;
107         /** WPA using EAP authentication. Generally used with an external authentication server. */
108         public static final int WPA_EAP = 2;
109         /** IEEE 802.1X using EAP authentication and (optionally) dynamically
110          * generated WEP keys. */
111         public static final int IEEE8021X = 3;
112 
113         /** WPA2 pre-shared key for use with soft access point
114           * (requires {@code preSharedKey} to be specified).
115           * @hide
116           */
117         @SystemApi
118         public static final int WPA2_PSK = 4;
119         /**
120          * Hotspot 2.0 r2 OSEN:
121          * @hide
122          */
123         public static final int OSEN = 5;
124 
125         /**
126          * IEEE 802.11r Fast BSS Transition with PSK authentication.
127          * @hide
128          */
129         public static final int FT_PSK = 6;
130 
131         /**
132          * IEEE 802.11r Fast BSS Transition with EAP authentication.
133          * @hide
134          */
135         public static final int FT_EAP = 7;
136 
137         /**
138          * Simultaneous Authentication of Equals
139          */
140         public static final int SAE = 8;
141 
142         /**
143          * Opportunististic Wireless Encryption
144          */
145         public static final int OWE = 9;
146 
147         /**
148          * SUITE_B_192 192 bit level
149          */
150         public static final int SUITE_B_192 = 10;
151 
152         /**
153          * WPA pre-shared key with stronger SHA256-based algorithms.
154          * @hide
155          */
156         public static final int WPA_PSK_SHA256 = 11;
157 
158         /**
159          * WPA using EAP authentication with stronger SHA256-based algorithms.
160          * @hide
161          */
162         public static final int WPA_EAP_SHA256 = 12;
163 
164         public static final String varName = "key_mgmt";
165 
166         public static final String[] strings = { "NONE", "WPA_PSK", "WPA_EAP",
167                 "IEEE8021X", "WPA2_PSK", "OSEN", "FT_PSK", "FT_EAP",
168                 "SAE", "OWE", "SUITE_B_192", "WPA_PSK_SHA256", "WPA_EAP_SHA256" };
169     }
170 
171     /**
172      * Recognized security protocols.
173      */
174     public static class Protocol {
Protocol()175         private Protocol() { }
176 
177         /** WPA/IEEE 802.11i/D3.0
178          * @deprecated Due to security and performance limitations, use of WPA-1 networks
179          * is discouraged. WPA-2 (RSN) should be used instead. */
180         @Deprecated
181         public static final int WPA = 0;
182         /** RSN WPA2/WPA3/IEEE 802.11i */
183         public static final int RSN = 1;
184         /** HS2.0 r2 OSEN
185          * @hide
186          */
187         public static final int OSEN = 2;
188 
189         public static final String varName = "proto";
190 
191         public static final String[] strings = { "WPA", "RSN", "OSEN" };
192     }
193 
194     /**
195      * Recognized IEEE 802.11 authentication algorithms.
196      */
197     public static class AuthAlgorithm {
AuthAlgorithm()198         private AuthAlgorithm() { }
199 
200         /** Open System authentication (required for WPA/WPA2) */
201         public static final int OPEN = 0;
202         /** Shared Key authentication (requires static WEP keys)
203          * @deprecated Due to security and performance limitations, use of WEP networks
204          * is discouraged. */
205         @Deprecated
206         public static final int SHARED = 1;
207         /** LEAP/Network EAP (only used with LEAP) */
208         public static final int LEAP = 2;
209 
210         public static final String varName = "auth_alg";
211 
212         public static final String[] strings = { "OPEN", "SHARED", "LEAP" };
213     }
214 
215     /**
216      * Recognized pairwise ciphers for WPA.
217      */
218     public static class PairwiseCipher {
PairwiseCipher()219         private PairwiseCipher() { }
220 
221         /** Use only Group keys (deprecated) */
222         public static final int NONE = 0;
223         /** Temporal Key Integrity Protocol [IEEE 802.11i/D7.0]
224          * @deprecated Due to security and performance limitations, use of WPA-1 networks
225          * is discouraged. WPA-2 (RSN) should be used instead. */
226         @Deprecated
227         public static final int TKIP = 1;
228         /** AES in Counter mode with CBC-MAC [RFC 3610, IEEE 802.11i/D7.0] */
229         public static final int CCMP = 2;
230         /**
231          * AES in Galois/Counter Mode
232          */
233         public static final int GCMP_256 = 3;
234 
235         public static final String varName = "pairwise";
236 
237         public static final String[] strings = { "NONE", "TKIP", "CCMP", "GCMP_256" };
238     }
239 
240     /**
241      * Recognized group ciphers.
242      * <pre>
243      * CCMP = AES in Counter mode with CBC-MAC [RFC 3610, IEEE 802.11i/D7.0]
244      * TKIP = Temporal Key Integrity Protocol [IEEE 802.11i/D7.0]
245      * WEP104 = WEP (Wired Equivalent Privacy) with 104-bit key
246      * WEP40 = WEP (Wired Equivalent Privacy) with 40-bit key (original 802.11)
247      * GCMP_256 = AES in Galois/Counter Mode
248      * </pre>
249      */
250     public static class GroupCipher {
GroupCipher()251         private GroupCipher() { }
252 
253         /** WEP40 = WEP (Wired Equivalent Privacy) with 40-bit key (original 802.11)
254          * @deprecated Due to security and performance limitations, use of WEP networks
255          * is discouraged. */
256         @Deprecated
257         public static final int WEP40 = 0;
258         /** WEP104 = WEP (Wired Equivalent Privacy) with 104-bit key
259          * @deprecated Due to security and performance limitations, use of WEP networks
260          * is discouraged. */
261         @Deprecated
262         public static final int WEP104 = 1;
263         /** Temporal Key Integrity Protocol [IEEE 802.11i/D7.0] */
264         public static final int TKIP = 2;
265         /** AES in Counter mode with CBC-MAC [RFC 3610, IEEE 802.11i/D7.0] */
266         public static final int CCMP = 3;
267         /** Hotspot 2.0 r2 OSEN
268          * @hide
269          */
270         public static final int GTK_NOT_USED = 4;
271         /**
272          * AES in Galois/Counter Mode
273          */
274         public static final int GCMP_256 = 5;
275 
276         public static final String varName = "group";
277 
278         public static final String[] strings =
279                 { /* deprecated */ "WEP40", /* deprecated */ "WEP104",
280                         "TKIP", "CCMP", "GTK_NOT_USED", "GCMP_256" };
281     }
282 
283     /**
284      * Recognized group management ciphers.
285      * <pre>
286      * BIP_CMAC_256 = Cipher-based Message Authentication Code 256 bits
287      * BIP_GMAC_128 = Galois Message Authentication Code 128 bits
288      * BIP_GMAC_256 = Galois Message Authentication Code 256 bits
289      * </pre>
290      */
291     public static class GroupMgmtCipher {
GroupMgmtCipher()292         private GroupMgmtCipher() { }
293 
294         /** CMAC-256 = Cipher-based Message Authentication Code */
295         public static final int BIP_CMAC_256 = 0;
296 
297         /** GMAC-128 = Galois Message Authentication Code */
298         public static final int BIP_GMAC_128 = 1;
299 
300         /** GMAC-256 = Galois Message Authentication Code */
301         public static final int BIP_GMAC_256 = 2;
302 
303         private static final String varName = "groupMgmt";
304 
305         private static final String[] strings = { "BIP_CMAC_256",
306                 "BIP_GMAC_128", "BIP_GMAC_256"};
307     }
308 
309     /**
310      * Recognized suiteB ciphers.
311      * <pre>
312      * ECDHE_ECDSA
313      * ECDHE_RSA
314      * </pre>
315      * @hide
316      */
317     public static class SuiteBCipher {
SuiteBCipher()318         private SuiteBCipher() { }
319 
320         /** Diffie-Hellman with Elliptic Curve_ECDSA signature */
321         public static final int ECDHE_ECDSA = 0;
322 
323         /** Diffie-Hellman with_RSA signature */
324         public static final int ECDHE_RSA = 1;
325 
326         private static final String varName = "SuiteB";
327 
328         private static final String[] strings = { "ECDHE_ECDSA", "ECDHE_RSA" };
329     }
330 
331     /** Possible status of a network configuration. */
332     public static class Status {
Status()333         private Status() { }
334 
335         /** this is the network we are currently connected to */
336         public static final int CURRENT = 0;
337         /** supplicant will not attempt to use this network */
338         public static final int DISABLED = 1;
339         /** supplicant will consider this network available for association */
340         public static final int ENABLED = 2;
341 
342         public static final String[] strings = { "current", "disabled", "enabled" };
343     }
344 
345     /**
346      * Security types we support.
347      */
348     /** @hide */
349     public static final int SECURITY_TYPE_OPEN = 0;
350     /** @hide */
351     public static final int SECURITY_TYPE_WEP = 1;
352     /** @hide */
353     public static final int SECURITY_TYPE_PSK = 2;
354     /** @hide */
355     public static final int SECURITY_TYPE_EAP = 3;
356     /** @hide */
357     public static final int SECURITY_TYPE_SAE = 4;
358     /** @hide */
359     public static final int SECURITY_TYPE_EAP_SUITE_B = 5;
360     /** @hide */
361     public static final int SECURITY_TYPE_OWE = 6;
362 
363     /** @hide */
364     @Retention(RetentionPolicy.SOURCE)
365     @IntDef(prefix = { "SECURITY_TYPE_" }, value = {
366             SECURITY_TYPE_OPEN,
367             SECURITY_TYPE_WEP,
368             SECURITY_TYPE_PSK,
369             SECURITY_TYPE_EAP,
370             SECURITY_TYPE_SAE,
371             SECURITY_TYPE_EAP_SUITE_B,
372             SECURITY_TYPE_OWE
373     })
374     public @interface SecurityType {}
375 
376     /**
377      * @hide
378      * Set security params (sets the various bitsets exposed in WifiConfiguration).
379      *
380      * @param securityType One of the security types from {@link SecurityType}.
381      */
setSecurityParams(@ecurityType int securityType)382     public void setSecurityParams(@SecurityType int securityType) {
383         // Clear all the bitsets.
384         allowedKeyManagement.clear();
385         allowedProtocols.clear();
386         allowedAuthAlgorithms.clear();
387         allowedPairwiseCiphers.clear();
388         allowedGroupCiphers.clear();
389         allowedGroupManagementCiphers.clear();
390         allowedSuiteBCiphers.clear();
391 
392         switch (securityType) {
393             case SECURITY_TYPE_OPEN:
394                 allowedKeyManagement.set(WifiConfiguration.KeyMgmt.NONE);
395                 break;
396             case SECURITY_TYPE_WEP:
397                 allowedKeyManagement.set(WifiConfiguration.KeyMgmt.NONE);
398                 allowedAuthAlgorithms.set(WifiConfiguration.AuthAlgorithm.OPEN);
399                 allowedAuthAlgorithms.set(WifiConfiguration.AuthAlgorithm.SHARED);
400                 break;
401             case SECURITY_TYPE_PSK:
402                 allowedKeyManagement.set(WifiConfiguration.KeyMgmt.WPA_PSK);
403                 break;
404             case SECURITY_TYPE_EAP:
405                 allowedKeyManagement.set(WifiConfiguration.KeyMgmt.WPA_EAP);
406                 allowedKeyManagement.set(WifiConfiguration.KeyMgmt.IEEE8021X);
407                 break;
408             case SECURITY_TYPE_SAE:
409                 allowedKeyManagement.set(WifiConfiguration.KeyMgmt.SAE);
410                 requirePMF = true;
411                 break;
412             case SECURITY_TYPE_EAP_SUITE_B:
413                 allowedKeyManagement.set(WifiConfiguration.KeyMgmt.SUITE_B_192);
414                 allowedGroupCiphers.set(WifiConfiguration.GroupCipher.GCMP_256);
415                 allowedGroupManagementCiphers.set(WifiConfiguration.GroupMgmtCipher.BIP_GMAC_256);
416                 // Note: allowedSuiteBCiphers bitset will be set by the service once the
417                 // certificates are attached to this profile
418                 requirePMF = true;
419                 break;
420             case SECURITY_TYPE_OWE:
421                 allowedKeyManagement.set(WifiConfiguration.KeyMgmt.OWE);
422                 requirePMF = true;
423                 break;
424             default:
425                 throw new IllegalArgumentException("unknown security type " + securityType);
426         }
427     }
428 
429     /** @hide */
430     public static final int UNKNOWN_UID = -1;
431 
432     /**
433      * The ID number that the supplicant uses to identify this
434      * network configuration entry. This must be passed as an argument
435      * to most calls into the supplicant.
436      */
437     public int networkId;
438 
439     // Fixme We need remove this field to use only Quality network selection status only
440     /**
441      * The current status of this network configuration entry.
442      * @see Status
443      */
444     public int status;
445 
446     /**
447      * The network's SSID. Can either be a UTF-8 string,
448      * which must be enclosed in double quotation marks
449      * (e.g., {@code "MyNetwork"}), or a string of
450      * hex digits, which are not enclosed in quotes
451      * (e.g., {@code 01a243f405}).
452      */
453     public String SSID;
454 
455     /**
456      * When set, this network configuration entry should only be used when
457      * associating with the AP having the specified BSSID. The value is
458      * a string in the format of an Ethernet MAC address, e.g.,
459      * <code>XX:XX:XX:XX:XX:XX</code> where each <code>X</code> is a hex digit.
460      */
461     public String BSSID;
462 
463     /**
464      * 2GHz band.
465      * @hide
466      */
467     public static final int AP_BAND_2GHZ = 0;
468 
469     /**
470      * 5GHz band.
471      * @hide
472      */
473     public static final int AP_BAND_5GHZ = 1;
474 
475     /**
476      * Device is allowed to choose the optimal band (2Ghz or 5Ghz) based on device capability,
477      * operating country code and current radio conditions.
478      * @hide
479      */
480     public static final int AP_BAND_ANY = -1;
481 
482     /**
483      * The band which AP resides on
484      * -1:Any 0:2G 1:5G
485      * By default, 2G is chosen
486      * @hide
487      */
488     @UnsupportedAppUsage
489     public int apBand = AP_BAND_2GHZ;
490 
491     /**
492      * The channel which AP resides on,currently, US only
493      * 2G  1-11
494      * 5G  36,40,44,48,149,153,157,161,165
495      * 0 - find a random available channel according to the apBand
496      * @hide
497      */
498     @UnsupportedAppUsage
499     public int apChannel = 0;
500 
501     /**
502      * Pre-shared key for use with WPA-PSK. Either an ASCII string enclosed in
503      * double quotation marks (e.g., {@code "abcdefghij"} for PSK passphrase or
504      * a string of 64 hex digits for raw PSK.
505      * <p/>
506      * When the value of this key is read, the actual key is
507      * not returned, just a "*" if the key has a value, or the null
508      * string otherwise.
509      */
510     public String preSharedKey;
511 
512     /**
513      * Four WEP keys. For each of the four values, provide either an ASCII
514      * string enclosed in double quotation marks (e.g., {@code "abcdef"}),
515      * a string of hex digits (e.g., {@code 0102030405}), or an empty string
516      * (e.g., {@code ""}).
517      * <p/>
518      * When the value of one of these keys is read, the actual key is
519      * not returned, just a "*" if the key has a value, or the null
520      * string otherwise.
521      * @deprecated Due to security and performance limitations, use of WEP networks
522      * is discouraged.
523      */
524     @Deprecated
525     public String[] wepKeys;
526 
527     /** Default WEP key index, ranging from 0 to 3.
528      * @deprecated Due to security and performance limitations, use of WEP networks
529      * is discouraged. */
530     @Deprecated
531     public int wepTxKeyIndex;
532 
533     /**
534      * Priority determines the preference given to a network by {@code wpa_supplicant}
535      * when choosing an access point with which to associate.
536      * @deprecated This field does not exist anymore.
537      */
538     @Deprecated
539     public int priority;
540 
541     /**
542      * This is a network that does not broadcast its SSID, so an
543      * SSID-specific probe request must be used for scans.
544      */
545     public boolean hiddenSSID;
546 
547     /**
548      * This is a network that requries Protected Management Frames (PMF).
549      * @hide
550      */
551     public boolean requirePMF;
552 
553     /**
554      * Update identifier, for Passpoint network.
555      * @hide
556      */
557     public String updateIdentifier;
558 
559     /**
560      * The set of key management protocols supported by this configuration.
561      * See {@link KeyMgmt} for descriptions of the values.
562      * Defaults to WPA-PSK WPA-EAP.
563      */
564     @NonNull
565     public BitSet allowedKeyManagement;
566     /**
567      * The set of security protocols supported by this configuration.
568      * See {@link Protocol} for descriptions of the values.
569      * Defaults to WPA RSN.
570      */
571     @NonNull
572     public BitSet allowedProtocols;
573     /**
574      * The set of authentication protocols supported by this configuration.
575      * See {@link AuthAlgorithm} for descriptions of the values.
576      * Defaults to automatic selection.
577      */
578     @NonNull
579     public BitSet allowedAuthAlgorithms;
580     /**
581      * The set of pairwise ciphers for WPA supported by this configuration.
582      * See {@link PairwiseCipher} for descriptions of the values.
583      * Defaults to CCMP TKIP.
584      */
585     @NonNull
586     public BitSet allowedPairwiseCiphers;
587     /**
588      * The set of group ciphers supported by this configuration.
589      * See {@link GroupCipher} for descriptions of the values.
590      * Defaults to CCMP TKIP WEP104 WEP40.
591      */
592     @NonNull
593     public BitSet allowedGroupCiphers;
594     /**
595      * The set of group management ciphers supported by this configuration.
596      * See {@link GroupMgmtCipher} for descriptions of the values.
597      */
598     @NonNull
599     public BitSet allowedGroupManagementCiphers;
600     /**
601      * The set of SuiteB ciphers supported by this configuration.
602      * To be used for WPA3-Enterprise mode.
603      * See {@link SuiteBCipher} for descriptions of the values.
604      */
605     @NonNull
606     public BitSet allowedSuiteBCiphers;
607     /**
608      * The enterprise configuration details specifying the EAP method,
609      * certificates and other settings associated with the EAP.
610      */
611     public WifiEnterpriseConfig enterpriseConfig;
612 
613     /**
614      * Fully qualified domain name of a Passpoint configuration
615      */
616     public String FQDN;
617 
618     /**
619      * Name of Passpoint credential provider
620      */
621     public String providerFriendlyName;
622 
623     /**
624      * Flag indicating if this network is provided by a home Passpoint provider or a roaming
625      * Passpoint provider.  This flag will be {@code true} if this network is provided by
626      * a home Passpoint provider and {@code false} if is provided by a roaming Passpoint provider
627      * or is a non-Passpoint network.
628      */
629     public boolean isHomeProviderNetwork;
630 
631     /**
632      * Roaming Consortium Id list for Passpoint credential; identifies a set of networks where
633      * Passpoint credential will be considered valid
634      */
635     public long[] roamingConsortiumIds;
636 
637     /**
638      * @hide
639      * This network configuration is visible to and usable by other users on the
640      * same device.
641      */
642     @UnsupportedAppUsage
643     public boolean shared;
644 
645     /**
646      * @hide
647      */
648     @NonNull
649     @UnsupportedAppUsage
650     private IpConfiguration mIpConfiguration;
651 
652     /**
653      * @hide
654      * dhcp server MAC address if known
655      */
656     public String dhcpServer;
657 
658     /**
659      * @hide
660      * default Gateway MAC address if known
661      */
662     @UnsupportedAppUsage
663     public String defaultGwMacAddress;
664 
665     /**
666      * @hide
667      * last time we connected, this configuration had validated internet access
668      */
669     @UnsupportedAppUsage
670     public boolean validatedInternetAccess;
671 
672     /**
673      * @hide
674      * The number of beacon intervals between Delivery Traffic Indication Maps (DTIM)
675      * This value is populated from scan results that contain Beacon Frames, which are infrequent.
676      * The value is not guaranteed to be set or current (Although it SHOULDNT change once set)
677      * Valid values are from 1 - 255. Initialized here as 0, use this to check if set.
678      */
679     public int dtimInterval = 0;
680 
681     /**
682      * Flag indicating if this configuration represents a legacy Passpoint configuration
683      * (Release N or older).  This is used for migrating Passpoint configuration from N to O.
684      * This will no longer be needed after O.
685      * @hide
686      */
687     public boolean isLegacyPasspointConfig = false;
688     /**
689      * @hide
690      * Uid of app creating the configuration
691      */
692     @SystemApi
693     public int creatorUid;
694 
695     /**
696      * @hide
697      * Uid of last app issuing a connection related command
698      */
699     @UnsupportedAppUsage
700     public int lastConnectUid;
701 
702     /**
703      * @hide
704      * Uid of last app modifying the configuration
705      */
706     @SystemApi
707     public int lastUpdateUid;
708 
709     /**
710      * @hide
711      * Universal name for app creating the configuration
712      *    see {@link PackageManager#getNameForUid(int)}
713      */
714     @SystemApi
715     public String creatorName;
716 
717     /**
718      * @hide
719      * Universal name for app updating the configuration
720      *    see {@link PackageManager#getNameForUid(int)}
721      */
722     @SystemApi
723     public String lastUpdateName;
724 
725     /**
726      * @hide
727      * Status of user approval for connection
728      */
729     public int userApproved = USER_UNSPECIFIED;
730 
731     /** The Below RSSI thresholds are used to configure AutoJoin
732      *  - GOOD/LOW/BAD thresholds are used so as to calculate link score
733      *  - UNWANTED_SOFT are used by the blacklisting logic so as to handle
734      *  the unwanted network message coming from CS
735      *  - UNBLACKLIST thresholds are used so as to tweak the speed at which
736      *  the network is unblacklisted (i.e. if
737      *          it is seen with good RSSI, it is blacklisted faster)
738      *  - INITIAL_AUTOJOIN_ATTEMPT, used to determine how close from
739      *  the network we need to be before autojoin kicks in
740      */
741     /** @hide **/
742     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.P, trackingBug = 115609023)
743     public static int INVALID_RSSI = -127;
744 
745     // States for the userApproved field
746     /**
747      * @hide
748      * User hasn't specified if connection is okay
749      */
750     public static final int USER_UNSPECIFIED = 0;
751     /**
752      * @hide
753      * User has approved this for connection
754      */
755     public static final int USER_APPROVED = 1;
756     /**
757      * @hide
758      * User has banned this from connection
759      */
760     public static final int USER_BANNED = 2;
761     /**
762      * @hide
763      * Waiting for user input
764      */
765     public static final int USER_PENDING = 3;
766 
767     /**
768      * @hide
769      * Number of reports indicating no Internet Access
770      */
771     @UnsupportedAppUsage
772     public int numNoInternetAccessReports;
773 
774     /**
775      * @hide
776      * For debug: date at which the config was last updated
777      */
778     public String updateTime;
779 
780     /**
781      * @hide
782      * For debug: date at which the config was last updated
783      */
784     public String creationTime;
785 
786     /**
787      * @hide
788      * The WiFi configuration is considered to have no internet access for purpose of autojoining
789      * if there has been a report of it having no internet access, and, it never have had
790      * internet access in the past.
791      */
792     @SystemApi
hasNoInternetAccess()793     public boolean hasNoInternetAccess() {
794         return numNoInternetAccessReports > 0 && !validatedInternetAccess;
795     }
796 
797     /**
798      * The WiFi configuration is expected not to have Internet access (e.g., a wireless printer, a
799      * Chromecast hotspot, etc.). This will be set if the user explicitly confirms a connection to
800      * this configuration and selects "don't ask again".
801      * @hide
802      */
803     @UnsupportedAppUsage
804     public boolean noInternetAccessExpected;
805 
806     /**
807      * The WiFi configuration is expected not to have Internet access (e.g., a wireless printer, a
808      * Chromecast hotspot, etc.). This will be set if the user explicitly confirms a connection to
809      * this configuration and selects "don't ask again".
810      * @hide
811      */
812     @SystemApi
isNoInternetAccessExpected()813     public boolean isNoInternetAccessExpected() {
814         return noInternetAccessExpected;
815     }
816 
817     /**
818      * This Wifi configuration is expected for OSU(Online Sign Up) of Passpoint Release 2.
819      * @hide
820      */
821     public boolean osu;
822 
823     /**
824      * @hide
825      * Last time the system was connected to this configuration.
826      */
827     public long lastConnected;
828 
829     /**
830      * @hide
831      * Last time the system was disconnected to this configuration.
832      */
833     public long lastDisconnected;
834 
835     /**
836      * Set if the configuration was self added by the framework
837      * This boolean is cleared if we get a connect/save/ update or
838      * any wifiManager command that indicate the user interacted with the configuration
839      * since we will now consider that the configuration belong to him.
840      * @hide
841      */
842     @UnsupportedAppUsage
843     public boolean selfAdded;
844 
845     /**
846      * Set if the configuration was self added by the framework
847      * This boolean is set once and never cleared. It is used
848      * so as we never loose track of who created the
849      * configuration in the first place.
850      * @hide
851      */
852     public boolean didSelfAdd;
853 
854     /**
855      * Peer WifiConfiguration this WifiConfiguration was added for
856      * @hide
857      */
858     public String peerWifiConfiguration;
859 
860     /**
861      * @hide
862      * Indicate that a WifiConfiguration is temporary and should not be saved
863      * nor considered by AutoJoin.
864      */
865     public boolean ephemeral;
866 
867     /**
868      * @hide
869      * Indicate that a WifiConfiguration is temporary and should not be saved
870      * nor considered by AutoJoin.
871      */
872     @SystemApi
isEphemeral()873     public boolean isEphemeral() {
874       return ephemeral;
875     }
876 
877     /**
878      * Indicate whther the network is trusted or not. Networks are considered trusted
879      * if the user explicitly allowed this network connection.
880      * @hide
881      */
882     public boolean trusted;
883 
884     /**
885      * This Wifi configuration is created from a {@link WifiNetworkSuggestion}
886      * @hide
887      */
888     public boolean fromWifiNetworkSuggestion;
889 
890     /**
891      * This Wifi configuration is created from a {@link WifiNetworkSpecifier}
892      * @hide
893      */
894     public boolean fromWifiNetworkSpecifier;
895 
896     /**
897      * Indicates if the creator of this configuration has expressed that it
898      * should be considered metered.
899      *
900      * @see #isMetered(WifiConfiguration, WifiInfo)
901      * @hide
902      */
903     @SystemApi
904     public boolean meteredHint;
905 
906     /** {@hide} */
907     public static final int METERED_OVERRIDE_NONE = 0;
908     /** {@hide} */
909     public static final int METERED_OVERRIDE_METERED = 1;
910     /** {@hide} */
911     public static final int METERED_OVERRIDE_NOT_METERED = 2;
912 
913     /**
914      * Indicates if the end user has expressed an explicit opinion about the
915      * meteredness of this network, such as through the Settings app.
916      * <p>
917      * This should always override any values from {@link #meteredHint} or
918      * {@link WifiInfo#getMeteredHint()}.
919      *
920      * @see #isMetered(WifiConfiguration, WifiInfo)
921      * @hide
922      */
923     public int meteredOverride = METERED_OVERRIDE_NONE;
924 
925     /**
926      * Blend together all the various opinions to decide if the given network
927      * should be considered metered or not.
928      *
929      * @hide
930      */
isMetered(WifiConfiguration config, WifiInfo info)931     public static boolean isMetered(WifiConfiguration config, WifiInfo info) {
932         boolean metered = false;
933         if (info != null && info.getMeteredHint()) {
934             metered = true;
935         }
936         if (config != null && config.meteredHint) {
937             metered = true;
938         }
939         if (config != null
940                 && config.meteredOverride == WifiConfiguration.METERED_OVERRIDE_METERED) {
941             metered = true;
942         }
943         if (config != null
944                 && config.meteredOverride == WifiConfiguration.METERED_OVERRIDE_NOT_METERED) {
945             metered = false;
946         }
947         return metered;
948     }
949 
950     /**
951      * @hide
952      * Returns true if this WiFi config is for an open network.
953      */
isOpenNetwork()954     public boolean isOpenNetwork() {
955         final int cardinality = allowedKeyManagement.cardinality();
956         final boolean hasNoKeyMgmt = cardinality == 0
957                 || (cardinality == 1 && (allowedKeyManagement.get(KeyMgmt.NONE)
958                 || allowedKeyManagement.get(KeyMgmt.OWE)));
959 
960         boolean hasNoWepKeys = true;
961         if (wepKeys != null) {
962             for (int i = 0; i < wepKeys.length; i++) {
963                 if (wepKeys[i] != null) {
964                     hasNoWepKeys = false;
965                     break;
966                 }
967             }
968         }
969 
970         return hasNoKeyMgmt && hasNoWepKeys;
971     }
972 
973     /**
974      * @hide
975      * Setting this value will force scan results associated with this configuration to
976      * be included in the bucket of networks that are externally scored.
977      * If not set, associated scan results will be treated as legacy saved networks and
978      * will take precedence over networks in the scored category.
979      */
980     @SystemApi
981     public boolean useExternalScores;
982 
983     /**
984      * @hide
985      * Number of time the scorer overrode a the priority based choice, when comparing two
986      * WifiConfigurations, note that since comparing WifiConfiguration happens very often
987      * potentially at every scan, this number might become very large, even on an idle
988      * system.
989      */
990     @SystemApi
991     public int numScorerOverride;
992 
993     /**
994      * @hide
995      * Number of time the scorer overrode a the priority based choice, and the comparison
996      * triggered a network switch
997      */
998     @SystemApi
999     public int numScorerOverrideAndSwitchedNetwork;
1000 
1001     /**
1002      * @hide
1003      * Number of time we associated to this configuration.
1004      */
1005     @SystemApi
1006     public int numAssociation;
1007 
1008     /**
1009      * @hide
1010      * Use factory MAC when connecting to this network
1011      */
1012     public static final int RANDOMIZATION_NONE = 0;
1013     /**
1014      * @hide
1015      * Generate a randomized MAC once and reuse it for all connections to this network
1016      */
1017     public static final int RANDOMIZATION_PERSISTENT = 1;
1018 
1019     /**
1020      * @hide
1021      * Level of MAC randomization for this network
1022      */
1023     public int macRandomizationSetting = RANDOMIZATION_PERSISTENT;
1024 
1025     /**
1026      * @hide
1027      * Randomized MAC address to use with this particular network
1028      */
1029     @NonNull
1030     private MacAddress mRandomizedMacAddress;
1031 
1032     /**
1033      * @hide
1034      * Checks if the given MAC address can be used for Connected Mac Randomization
1035      * by verifying that it is non-null, unicast, locally assigned, and not default mac.
1036      * @param mac MacAddress to check
1037      * @return true if mac is good to use
1038      */
isValidMacAddressForRandomization(MacAddress mac)1039     public static boolean isValidMacAddressForRandomization(MacAddress mac) {
1040         return mac != null && !mac.isMulticastAddress() && mac.isLocallyAssigned()
1041                 && !MacAddress.fromString(WifiInfo.DEFAULT_MAC_ADDRESS).equals(mac);
1042     }
1043 
1044     /**
1045      * @hide
1046      * Returns Randomized MAC address to use with the network.
1047      * If it is not set/valid, creates a new randomized address.
1048      * If it can't generate a valid mac, returns the default MAC.
1049      */
getOrCreateRandomizedMacAddress()1050     public @NonNull MacAddress getOrCreateRandomizedMacAddress() {
1051         int randomMacGenerationCount = 0;
1052         while (!isValidMacAddressForRandomization(mRandomizedMacAddress)
1053                 && randomMacGenerationCount < MAXIMUM_RANDOM_MAC_GENERATION_RETRY) {
1054             mRandomizedMacAddress = MacAddress.createRandomUnicastAddress();
1055             randomMacGenerationCount++;
1056         }
1057 
1058         if (!isValidMacAddressForRandomization(mRandomizedMacAddress)) {
1059             mRandomizedMacAddress = MacAddress.fromString(WifiInfo.DEFAULT_MAC_ADDRESS);
1060         }
1061         return mRandomizedMacAddress;
1062     }
1063 
1064     /**
1065      * Returns MAC address set to be the local randomized MAC address.
1066      * Depending on user preference, the device may or may not use the returned MAC address for
1067      * connections to this network.
1068      * <p>
1069      * Information is restricted to Device Owner, Profile Owner, and Carrier apps
1070      * (which will only obtain addresses for configurations which they create). Other callers
1071      * will receive a default "02:00:00:00:00:00" MAC address.
1072      */
getRandomizedMacAddress()1073     public @NonNull MacAddress getRandomizedMacAddress() {
1074         return mRandomizedMacAddress;
1075     }
1076 
1077     /**
1078      * @hide
1079      * @param mac MacAddress to change into
1080      */
setRandomizedMacAddress(@onNull MacAddress mac)1081     public void setRandomizedMacAddress(@NonNull MacAddress mac) {
1082         if (mac == null) {
1083             Log.e(TAG, "setRandomizedMacAddress received null MacAddress.");
1084             return;
1085         }
1086         mRandomizedMacAddress = mac;
1087     }
1088 
1089     /** @hide
1090      * Boost given to RSSI on a home network for the purpose of calculating the score
1091      * This adds stickiness to home networks, as defined by:
1092      * - less than 4 known BSSIDs
1093      * - PSK only
1094      * - TODO: add a test to verify that all BSSIDs are behind same gateway
1095      ***/
1096     public static final int HOME_NETWORK_RSSI_BOOST = 5;
1097 
1098     /**
1099      * @hide
1100      * This class is used to contain all the information and API used for quality network selection
1101      */
1102     public static class NetworkSelectionStatus {
1103         /**
1104          * Quality Network Selection Status enable, temporary disabled, permanently disabled
1105          */
1106         /**
1107          * This network is allowed to join Quality Network Selection
1108          */
1109         public static final int NETWORK_SELECTION_ENABLED = 0;
1110         /**
1111          * network was temporary disabled. Can be re-enabled after a time period expire
1112          */
1113         public static final int NETWORK_SELECTION_TEMPORARY_DISABLED  = 1;
1114         /**
1115          * network was permanently disabled.
1116          */
1117         public static final int NETWORK_SELECTION_PERMANENTLY_DISABLED  = 2;
1118         /**
1119          * Maximum Network selection status
1120          */
1121         public static final int NETWORK_SELECTION_STATUS_MAX = 3;
1122 
1123         /**
1124          * Quality network selection status String (for debug purpose). Use Quality network
1125          * selection status value as index to extec the corresponding debug string
1126          */
1127         public static final String[] QUALITY_NETWORK_SELECTION_STATUS = {
1128                 "NETWORK_SELECTION_ENABLED",
1129                 "NETWORK_SELECTION_TEMPORARY_DISABLED",
1130                 "NETWORK_SELECTION_PERMANENTLY_DISABLED"};
1131 
1132         //Quality Network disabled reasons
1133         /**
1134          * Default value. Means not disabled
1135          */
1136         public static final int NETWORK_SELECTION_ENABLE = 0;
1137         /**
1138          * The starting index for network selection disabled reasons
1139          */
1140         public static final int NETWORK_SELECTION_DISABLED_STARTING_INDEX = 1;
1141         /**
1142          * @deprecated it is not used any more.
1143          * This network is disabled because higher layer (>2) network is bad
1144          */
1145         public static final int DISABLED_BAD_LINK = 1;
1146         /**
1147          * This network is disabled because multiple association rejects
1148          */
1149         public static final int DISABLED_ASSOCIATION_REJECTION = 2;
1150         /**
1151          * This network is disabled because multiple authentication failure
1152          */
1153         public static final int DISABLED_AUTHENTICATION_FAILURE = 3;
1154         /**
1155          * This network is disabled because multiple DHCP failure
1156          */
1157         public static final int DISABLED_DHCP_FAILURE = 4;
1158         /**
1159          * This network is disabled because of security network but no credentials
1160          */
1161         public static final int DISABLED_DNS_FAILURE = 5;
1162         /**
1163          * This network is temporarily disabled because it has no Internet access.
1164          */
1165         public static final int DISABLED_NO_INTERNET_TEMPORARY = 6;
1166         /**
1167          * This network is disabled because we started WPS
1168          */
1169         public static final int DISABLED_WPS_START = 7;
1170         /**
1171          * This network is disabled because EAP-TLS failure
1172          */
1173         public static final int DISABLED_TLS_VERSION_MISMATCH = 8;
1174         // Values above are for temporary disablement; values below are for permanent disablement.
1175         /**
1176          * This network is disabled due to absence of user credentials
1177          */
1178         public static final int DISABLED_AUTHENTICATION_NO_CREDENTIALS = 9;
1179         /**
1180          * This network is permanently disabled because it has no Internet access and user does not
1181          * want to stay connected.
1182          */
1183         public static final int DISABLED_NO_INTERNET_PERMANENT = 10;
1184         /**
1185          * This network is disabled due to WifiManager disable it explicitly
1186          */
1187         public static final int DISABLED_BY_WIFI_MANAGER = 11;
1188         /**
1189          * This network is disabled due to user switching
1190          */
1191         public static final int DISABLED_DUE_TO_USER_SWITCH = 12;
1192         /**
1193          * This network is disabled due to wrong password
1194          */
1195         public static final int DISABLED_BY_WRONG_PASSWORD = 13;
1196         /**
1197          * This network is disabled because service is not subscribed
1198          */
1199         public static final int DISABLED_AUTHENTICATION_NO_SUBSCRIPTION = 14;
1200         /**
1201          * This Maximum disable reason value
1202          */
1203         public static final int NETWORK_SELECTION_DISABLED_MAX = 15;
1204 
1205         /**
1206          * Quality network selection disable reason String (for debug purpose)
1207          */
1208         public static final String[] QUALITY_NETWORK_SELECTION_DISABLE_REASON = {
1209                 "NETWORK_SELECTION_ENABLE",
1210                 "NETWORK_SELECTION_DISABLED_BAD_LINK", // deprecated
1211                 "NETWORK_SELECTION_DISABLED_ASSOCIATION_REJECTION ",
1212                 "NETWORK_SELECTION_DISABLED_AUTHENTICATION_FAILURE",
1213                 "NETWORK_SELECTION_DISABLED_DHCP_FAILURE",
1214                 "NETWORK_SELECTION_DISABLED_DNS_FAILURE",
1215                 "NETWORK_SELECTION_DISABLED_NO_INTERNET_TEMPORARY",
1216                 "NETWORK_SELECTION_DISABLED_WPS_START",
1217                 "NETWORK_SELECTION_DISABLED_TLS_VERSION",
1218                 "NETWORK_SELECTION_DISABLED_AUTHENTICATION_NO_CREDENTIALS",
1219                 "NETWORK_SELECTION_DISABLED_NO_INTERNET_PERMANENT",
1220                 "NETWORK_SELECTION_DISABLED_BY_WIFI_MANAGER",
1221                 "NETWORK_SELECTION_DISABLED_BY_USER_SWITCH",
1222                 "NETWORK_SELECTION_DISABLED_BY_WRONG_PASSWORD",
1223                 "NETWORK_SELECTION_DISABLED_AUTHENTICATION_NO_SUBSCRIPTION"
1224         };
1225 
1226         /**
1227          * Invalid time stamp for network selection disable
1228          */
1229         public static final long INVALID_NETWORK_SELECTION_DISABLE_TIMESTAMP = -1L;
1230 
1231         /**
1232          *  This constant indicates the current configuration has connect choice set
1233          */
1234         private static final int CONNECT_CHOICE_EXISTS = 1;
1235 
1236         /**
1237          *  This constant indicates the current configuration does not have connect choice set
1238          */
1239         private static final int CONNECT_CHOICE_NOT_EXISTS = -1;
1240 
1241         // fields for QualityNetwork Selection
1242         /**
1243          * Network selection status, should be in one of three status: enable, temporaily disabled
1244          * or permanently disabled
1245          */
1246         private int mStatus;
1247 
1248         /**
1249          * Reason for disable this network
1250          */
1251         private int mNetworkSelectionDisableReason;
1252 
1253         /**
1254          * Last time we temporarily disabled the configuration
1255          */
1256         private long mTemporarilyDisabledTimestamp = INVALID_NETWORK_SELECTION_DISABLE_TIMESTAMP;
1257 
1258         /**
1259          * counter for each Network selection disable reason
1260          */
1261         private int[] mNetworkSeclectionDisableCounter = new int[NETWORK_SELECTION_DISABLED_MAX];
1262 
1263         /**
1264          * Connect Choice over this configuration
1265          *
1266          * When current wifi configuration is visible to the user but user explicitly choose to
1267          * connect to another network X, the another networks X's configure key will be stored here.
1268          * We will consider user has a preference of X over this network. And in the future,
1269          * network selection will always give X a higher preference over this configuration.
1270          * configKey is : "SSID"-WEP-WPA_PSK-WPA_EAP
1271          */
1272         private String mConnectChoice;
1273 
1274         /**
1275          * The system timestamp when we records the connectChoice. This value is obtained from
1276          * System.currentTimeMillis
1277          */
1278         private long mConnectChoiceTimestamp = INVALID_NETWORK_SELECTION_DISABLE_TIMESTAMP;
1279 
1280         /**
1281          * Used to cache the temporary candidate during the network selection procedure. It will be
1282          * kept updating once a new scan result has a higher score than current one
1283          */
1284         private ScanResult mCandidate;
1285 
1286         /**
1287          * Used to cache the score of the current temporary candidate during the network
1288          * selection procedure.
1289          */
1290         private int mCandidateScore;
1291 
1292         /**
1293          * Indicate whether this network is visible in latest Qualified Network Selection. This
1294          * means there is scan result found related to this Configuration and meet the minimum
1295          * requirement. The saved network need not join latest Qualified Network Selection. For
1296          * example, it is disabled. True means network is visible in latest Qualified Network
1297          * Selection and false means network is invisible
1298          */
1299         private boolean mSeenInLastQualifiedNetworkSelection;
1300 
1301         /**
1302          * Boolean indicating if we have ever successfully connected to this network.
1303          *
1304          * This value will be set to true upon a successful connection.
1305          * This value will be set to false if a previous value was not stored in the config or if
1306          * the credentials are updated (ex. a password change).
1307          */
1308         private boolean mHasEverConnected;
1309 
1310         /**
1311          * Boolean indicating whether {@link com.android.server.wifi.RecommendedNetworkEvaluator}
1312          * chose not to connect to this network in the last qualified network selection process.
1313          */
1314         private boolean mNotRecommended;
1315 
1316         /**
1317          * Set whether {@link com.android.server.wifi.RecommendedNetworkEvaluator} does not
1318          * recommend connecting to this network.
1319          */
setNotRecommended(boolean notRecommended)1320         public void setNotRecommended(boolean notRecommended) {
1321             mNotRecommended = notRecommended;
1322         }
1323 
1324         /**
1325          * Returns whether {@link com.android.server.wifi.RecommendedNetworkEvaluator} does not
1326          * recommend connecting to this network.
1327          */
isNotRecommended()1328         public boolean isNotRecommended() {
1329             return mNotRecommended;
1330         }
1331 
1332         /**
1333          * set whether this network is visible in latest Qualified Network Selection
1334          * @param seen value set to candidate
1335          */
setSeenInLastQualifiedNetworkSelection(boolean seen)1336         public void setSeenInLastQualifiedNetworkSelection(boolean seen) {
1337             mSeenInLastQualifiedNetworkSelection =  seen;
1338         }
1339 
1340         /**
1341          * get whether this network is visible in latest Qualified Network Selection
1342          * @return returns true -- network is visible in latest Qualified Network Selection
1343          *         false -- network is invisible in latest Qualified Network Selection
1344          */
getSeenInLastQualifiedNetworkSelection()1345         public boolean getSeenInLastQualifiedNetworkSelection() {
1346             return mSeenInLastQualifiedNetworkSelection;
1347         }
1348         /**
1349          * set the temporary candidate of current network selection procedure
1350          * @param scanCandidate {@link ScanResult} the candidate set to mCandidate
1351          */
setCandidate(ScanResult scanCandidate)1352         public void setCandidate(ScanResult scanCandidate) {
1353             mCandidate = scanCandidate;
1354         }
1355 
1356         /**
1357          * get the temporary candidate of current network selection procedure
1358          * @return  returns {@link ScanResult} temporary candidate of current network selection
1359          * procedure
1360          */
getCandidate()1361         public ScanResult getCandidate() {
1362             return mCandidate;
1363         }
1364 
1365         /**
1366          * set the score of the temporary candidate of current network selection procedure
1367          * @param score value set to mCandidateScore
1368          */
setCandidateScore(int score)1369         public void setCandidateScore(int score) {
1370             mCandidateScore = score;
1371         }
1372 
1373         /**
1374          * get the score of the temporary candidate of current network selection procedure
1375          * @return returns score of the temporary candidate of current network selection procedure
1376          */
getCandidateScore()1377         public int getCandidateScore() {
1378             return mCandidateScore;
1379         }
1380 
1381         /**
1382          * get user preferred choice over this configuration
1383          *@return returns configKey of user preferred choice over this configuration
1384          */
getConnectChoice()1385         public String getConnectChoice() {
1386             return mConnectChoice;
1387         }
1388 
1389         /**
1390          * set user preferred choice over this configuration
1391          * @param newConnectChoice, the configKey of user preferred choice over this configuration
1392          */
setConnectChoice(String newConnectChoice)1393         public void setConnectChoice(String newConnectChoice) {
1394             mConnectChoice = newConnectChoice;
1395         }
1396 
1397         /**
1398          * get the timeStamp when user select a choice over this configuration
1399          * @return returns when current connectChoice is set (time from System.currentTimeMillis)
1400          */
getConnectChoiceTimestamp()1401         public long getConnectChoiceTimestamp() {
1402             return mConnectChoiceTimestamp;
1403         }
1404 
1405         /**
1406          * set the timeStamp when user select a choice over this configuration
1407          * @param timeStamp, the timestamp set to connectChoiceTimestamp, expected timestamp should
1408          *        be obtained from System.currentTimeMillis
1409          */
setConnectChoiceTimestamp(long timeStamp)1410         public void setConnectChoiceTimestamp(long timeStamp) {
1411             mConnectChoiceTimestamp = timeStamp;
1412         }
1413 
1414         /**
1415          * get current Quality network selection status
1416          * @return returns current Quality network selection status in String (for debug purpose)
1417          */
getNetworkStatusString()1418         public String getNetworkStatusString() {
1419             return QUALITY_NETWORK_SELECTION_STATUS[mStatus];
1420         }
1421 
setHasEverConnected(boolean value)1422         public void setHasEverConnected(boolean value) {
1423             mHasEverConnected = value;
1424         }
1425 
getHasEverConnected()1426         public boolean getHasEverConnected() {
1427             return mHasEverConnected;
1428         }
1429 
NetworkSelectionStatus()1430         public NetworkSelectionStatus() {
1431             // previously stored configs will not have this parameter, so we default to false.
1432             mHasEverConnected = false;
1433         };
1434 
1435         /**
1436          * @param reason specific error reason
1437          * @return  corresponding network disable reason String (for debug purpose)
1438          */
getNetworkDisableReasonString(int reason)1439         public static String getNetworkDisableReasonString(int reason) {
1440             if (reason >= NETWORK_SELECTION_ENABLE && reason < NETWORK_SELECTION_DISABLED_MAX) {
1441                 return QUALITY_NETWORK_SELECTION_DISABLE_REASON[reason];
1442             } else {
1443                 return null;
1444             }
1445         }
1446         /**
1447          * get current network disable reason
1448          * @return current network disable reason in String (for debug purpose)
1449          */
getNetworkDisableReasonString()1450         public String getNetworkDisableReasonString() {
1451             return QUALITY_NETWORK_SELECTION_DISABLE_REASON[mNetworkSelectionDisableReason];
1452         }
1453 
1454         /**
1455          * get current network network selection status
1456          * @return return current network network selection status
1457          */
getNetworkSelectionStatus()1458         public int getNetworkSelectionStatus() {
1459             return mStatus;
1460         }
1461         /**
1462          * @return whether current network is enabled to join network selection
1463          */
isNetworkEnabled()1464         public boolean isNetworkEnabled() {
1465             return mStatus == NETWORK_SELECTION_ENABLED;
1466         }
1467 
1468         /**
1469          * @return whether current network is temporary disabled
1470          */
isNetworkTemporaryDisabled()1471         public boolean isNetworkTemporaryDisabled() {
1472             return mStatus == NETWORK_SELECTION_TEMPORARY_DISABLED;
1473         }
1474 
1475         /**
1476          * @return returns whether current network is permanently disabled
1477          */
isNetworkPermanentlyDisabled()1478         public boolean isNetworkPermanentlyDisabled() {
1479             return mStatus == NETWORK_SELECTION_PERMANENTLY_DISABLED;
1480         }
1481 
1482         /**
1483          * set current networ work selection status
1484          * @param status network selection status to set
1485          */
setNetworkSelectionStatus(int status)1486         public void setNetworkSelectionStatus(int status) {
1487             if (status >= 0 && status < NETWORK_SELECTION_STATUS_MAX) {
1488                 mStatus = status;
1489             }
1490         }
1491 
1492         /**
1493          * @return returns current network's disable reason
1494          */
getNetworkSelectionDisableReason()1495         public int getNetworkSelectionDisableReason() {
1496             return mNetworkSelectionDisableReason;
1497         }
1498 
1499         /**
1500          * set Network disable reason
1501          * @param  reason Network disable reason
1502          */
setNetworkSelectionDisableReason(int reason)1503         public void setNetworkSelectionDisableReason(int reason) {
1504             if (reason >= 0 && reason < NETWORK_SELECTION_DISABLED_MAX) {
1505                 mNetworkSelectionDisableReason = reason;
1506             } else {
1507                 throw new IllegalArgumentException("Illegal reason value: " + reason);
1508             }
1509         }
1510 
1511         /**
1512          * check whether network is disabled by this reason
1513          * @param reason a specific disable reason
1514          * @return true -- network is disabled for this reason
1515          *         false -- network is not disabled for this reason
1516          */
isDisabledByReason(int reason)1517         public boolean isDisabledByReason(int reason) {
1518             return mNetworkSelectionDisableReason == reason;
1519         }
1520 
1521         /**
1522          * @param timeStamp Set when current network is disabled in millisecond since January 1,
1523          * 1970 00:00:00.0 UTC
1524          */
setDisableTime(long timeStamp)1525         public void setDisableTime(long timeStamp) {
1526             mTemporarilyDisabledTimestamp = timeStamp;
1527         }
1528 
1529         /**
1530          * @return returns when current network is disabled in millisecond since January 1,
1531          * 1970 00:00:00.0 UTC
1532          */
getDisableTime()1533         public long getDisableTime() {
1534             return mTemporarilyDisabledTimestamp;
1535         }
1536 
1537         /**
1538          * get the disable counter of a specific reason
1539          * @param  reason specific failure reason
1540          * @exception throw IllegalArgumentException for illegal input
1541          * @return counter number for specific error reason.
1542          */
getDisableReasonCounter(int reason)1543         public int getDisableReasonCounter(int reason) {
1544             if (reason >= NETWORK_SELECTION_ENABLE && reason < NETWORK_SELECTION_DISABLED_MAX) {
1545                 return mNetworkSeclectionDisableCounter[reason];
1546             } else {
1547                 throw new IllegalArgumentException("Illegal reason value: " + reason);
1548             }
1549         }
1550 
1551         /**
1552          * set the counter of a specific failure reason
1553          * @param reason reason for disable error
1554          * @param value the counter value for this specific reason
1555          * @exception throw IllegalArgumentException for illegal input
1556          */
setDisableReasonCounter(int reason, int value)1557         public void setDisableReasonCounter(int reason, int value) {
1558             if (reason >= NETWORK_SELECTION_ENABLE && reason < NETWORK_SELECTION_DISABLED_MAX) {
1559                 mNetworkSeclectionDisableCounter[reason] = value;
1560             } else {
1561                 throw new IllegalArgumentException("Illegal reason value: " + reason);
1562             }
1563         }
1564 
1565         /**
1566          * increment the counter of a specific failure reason
1567          * @param reason a specific failure reason
1568          * @exception throw IllegalArgumentException for illegal input
1569          */
incrementDisableReasonCounter(int reason)1570         public void incrementDisableReasonCounter(int reason) {
1571             if (reason >= NETWORK_SELECTION_ENABLE  && reason < NETWORK_SELECTION_DISABLED_MAX) {
1572                 mNetworkSeclectionDisableCounter[reason]++;
1573             } else {
1574                 throw new IllegalArgumentException("Illegal reason value: " + reason);
1575             }
1576         }
1577 
1578         /**
1579          * clear the counter of a specific failure reason
1580          * @hide
1581          * @param reason a specific failure reason
1582          * @exception throw IllegalArgumentException for illegal input
1583          */
clearDisableReasonCounter(int reason)1584         public void clearDisableReasonCounter(int reason) {
1585             if (reason >= NETWORK_SELECTION_ENABLE && reason < NETWORK_SELECTION_DISABLED_MAX) {
1586                 mNetworkSeclectionDisableCounter[reason] = NETWORK_SELECTION_ENABLE;
1587             } else {
1588                 throw new IllegalArgumentException("Illegal reason value: " + reason);
1589             }
1590         }
1591 
1592         /**
1593          * clear all the failure reason counters
1594          */
clearDisableReasonCounter()1595         public void clearDisableReasonCounter() {
1596             Arrays.fill(mNetworkSeclectionDisableCounter, NETWORK_SELECTION_ENABLE);
1597         }
1598 
1599         /**
1600          * BSSID for connection to this network (through network selection procedure)
1601          */
1602         private String mNetworkSelectionBSSID;
1603 
1604         /**
1605          * get current network Selection BSSID
1606          * @return current network Selection BSSID
1607          */
getNetworkSelectionBSSID()1608         public String getNetworkSelectionBSSID() {
1609             return mNetworkSelectionBSSID;
1610         }
1611 
1612         /**
1613          * set network Selection BSSID
1614          * @param bssid The target BSSID for assocaition
1615          */
setNetworkSelectionBSSID(String bssid)1616         public void setNetworkSelectionBSSID(String bssid) {
1617             mNetworkSelectionBSSID = bssid;
1618         }
1619 
copy(NetworkSelectionStatus source)1620         public void copy(NetworkSelectionStatus source) {
1621             mStatus = source.mStatus;
1622             mNetworkSelectionDisableReason = source.mNetworkSelectionDisableReason;
1623             for (int index = NETWORK_SELECTION_ENABLE; index < NETWORK_SELECTION_DISABLED_MAX;
1624                     index++) {
1625                 mNetworkSeclectionDisableCounter[index] =
1626                         source.mNetworkSeclectionDisableCounter[index];
1627             }
1628             mTemporarilyDisabledTimestamp = source.mTemporarilyDisabledTimestamp;
1629             mNetworkSelectionBSSID = source.mNetworkSelectionBSSID;
1630             setSeenInLastQualifiedNetworkSelection(source.getSeenInLastQualifiedNetworkSelection());
1631             setCandidate(source.getCandidate());
1632             setCandidateScore(source.getCandidateScore());
1633             setConnectChoice(source.getConnectChoice());
1634             setConnectChoiceTimestamp(source.getConnectChoiceTimestamp());
1635             setHasEverConnected(source.getHasEverConnected());
1636             setNotRecommended(source.isNotRecommended());
1637         }
1638 
writeToParcel(Parcel dest)1639         public void writeToParcel(Parcel dest) {
1640             dest.writeInt(getNetworkSelectionStatus());
1641             dest.writeInt(getNetworkSelectionDisableReason());
1642             for (int index = NETWORK_SELECTION_ENABLE; index < NETWORK_SELECTION_DISABLED_MAX;
1643                     index++) {
1644                 dest.writeInt(getDisableReasonCounter(index));
1645             }
1646             dest.writeLong(getDisableTime());
1647             dest.writeString(getNetworkSelectionBSSID());
1648             if (getConnectChoice() != null) {
1649                 dest.writeInt(CONNECT_CHOICE_EXISTS);
1650                 dest.writeString(getConnectChoice());
1651                 dest.writeLong(getConnectChoiceTimestamp());
1652             } else {
1653                 dest.writeInt(CONNECT_CHOICE_NOT_EXISTS);
1654             }
1655             dest.writeInt(getHasEverConnected() ? 1 : 0);
1656             dest.writeInt(isNotRecommended() ? 1 : 0);
1657         }
1658 
readFromParcel(Parcel in)1659         public void readFromParcel(Parcel in) {
1660             setNetworkSelectionStatus(in.readInt());
1661             setNetworkSelectionDisableReason(in.readInt());
1662             for (int index = NETWORK_SELECTION_ENABLE; index < NETWORK_SELECTION_DISABLED_MAX;
1663                     index++) {
1664                 setDisableReasonCounter(index, in.readInt());
1665             }
1666             setDisableTime(in.readLong());
1667             setNetworkSelectionBSSID(in.readString());
1668             if (in.readInt() == CONNECT_CHOICE_EXISTS) {
1669                 setConnectChoice(in.readString());
1670                 setConnectChoiceTimestamp(in.readLong());
1671             } else {
1672                 setConnectChoice(null);
1673                 setConnectChoiceTimestamp(INVALID_NETWORK_SELECTION_DISABLE_TIMESTAMP);
1674             }
1675             setHasEverConnected(in.readInt() != 0);
1676             setNotRecommended(in.readInt() != 0);
1677         }
1678     }
1679 
1680     /**
1681      * @hide
1682      * network selection related member
1683      */
1684     private NetworkSelectionStatus mNetworkSelectionStatus = new NetworkSelectionStatus();
1685 
1686     /**
1687      * @hide
1688      * This class is intended to store extra failure reason information for the most recent
1689      * connection attempt, so that it may be surfaced to the settings UI
1690      */
1691     public static class RecentFailure {
1692 
1693         /**
1694          * No recent failure, or no specific reason given for the recent connection failure
1695          */
1696         public static final int NONE = 0;
1697         /**
1698          * Connection to this network recently failed due to Association Rejection Status 17
1699          * (AP is full)
1700          */
1701         public static final int STATUS_AP_UNABLE_TO_HANDLE_NEW_STA = 17;
1702         /**
1703          * Association Rejection Status code (NONE for success/non-association-rejection-fail)
1704          */
1705         private int mAssociationStatus = NONE;
1706 
1707         /**
1708          * @param status the association status code for the recent failure
1709          */
setAssociationStatus(int status)1710         public void setAssociationStatus(int status) {
1711             mAssociationStatus = status;
1712         }
1713         /**
1714          * Sets the RecentFailure to NONE
1715          */
clear()1716         public void clear() {
1717             mAssociationStatus = NONE;
1718         }
1719         /**
1720          * Get the recent failure code
1721          */
getAssociationStatus()1722         public int getAssociationStatus() {
1723             return mAssociationStatus;
1724         }
1725     }
1726 
1727     /**
1728      * @hide
1729      * RecentFailure member
1730      */
1731     final public RecentFailure recentFailure = new RecentFailure();
1732 
1733     /**
1734      * @hide
1735      * @return network selection status
1736      */
getNetworkSelectionStatus()1737     public NetworkSelectionStatus getNetworkSelectionStatus() {
1738         return mNetworkSelectionStatus;
1739     }
1740 
1741     /**
1742      * Set the network selection status
1743      * @hide
1744      */
setNetworkSelectionStatus(NetworkSelectionStatus status)1745     public void setNetworkSelectionStatus(NetworkSelectionStatus status) {
1746         mNetworkSelectionStatus = status;
1747     }
1748 
1749     /**
1750      * @hide
1751      * Linked Configurations: represent the set of Wificonfigurations that are equivalent
1752      * regarding roaming and auto-joining.
1753      * The linked configuration may or may not have same SSID, and may or may not have same
1754      * credentials.
1755      * For instance, linked configurations will have same defaultGwMacAddress or same dhcp server.
1756      */
1757     public HashMap<String, Integer>  linkedConfigurations;
1758 
WifiConfiguration()1759     public WifiConfiguration() {
1760         networkId = INVALID_NETWORK_ID;
1761         SSID = null;
1762         BSSID = null;
1763         FQDN = null;
1764         roamingConsortiumIds = new long[0];
1765         priority = 0;
1766         hiddenSSID = false;
1767         allowedKeyManagement = new BitSet();
1768         allowedProtocols = new BitSet();
1769         allowedAuthAlgorithms = new BitSet();
1770         allowedPairwiseCiphers = new BitSet();
1771         allowedGroupCiphers = new BitSet();
1772         allowedGroupManagementCiphers = new BitSet();
1773         allowedSuiteBCiphers = new BitSet();
1774         wepKeys = new String[4];
1775         for (int i = 0; i < wepKeys.length; i++) {
1776             wepKeys[i] = null;
1777         }
1778         enterpriseConfig = new WifiEnterpriseConfig();
1779         selfAdded = false;
1780         didSelfAdd = false;
1781         ephemeral = false;
1782         osu = false;
1783         trusted = true; // Networks are considered trusted by default.
1784         fromWifiNetworkSuggestion = false;
1785         fromWifiNetworkSpecifier = false;
1786         meteredHint = false;
1787         meteredOverride = METERED_OVERRIDE_NONE;
1788         useExternalScores = false;
1789         validatedInternetAccess = false;
1790         mIpConfiguration = new IpConfiguration();
1791         lastUpdateUid = -1;
1792         creatorUid = -1;
1793         shared = true;
1794         dtimInterval = 0;
1795         mRandomizedMacAddress = MacAddress.fromString(WifiInfo.DEFAULT_MAC_ADDRESS);
1796     }
1797 
1798     /**
1799      * Identify if this configuration represents a Passpoint network
1800      */
isPasspoint()1801     public boolean isPasspoint() {
1802         return !TextUtils.isEmpty(FQDN)
1803                 && !TextUtils.isEmpty(providerFriendlyName)
1804                 && enterpriseConfig != null
1805                 && enterpriseConfig.getEapMethod() != WifiEnterpriseConfig.Eap.NONE;
1806     }
1807 
1808     /**
1809      * Helper function, identify if a configuration is linked
1810      * @hide
1811      */
isLinked(WifiConfiguration config)1812     public boolean isLinked(WifiConfiguration config) {
1813         if (config != null) {
1814             if (config.linkedConfigurations != null && linkedConfigurations != null) {
1815                 if (config.linkedConfigurations.get(configKey()) != null
1816                         && linkedConfigurations.get(config.configKey()) != null) {
1817                     return true;
1818                 }
1819             }
1820         }
1821         return  false;
1822     }
1823 
1824     /**
1825      * Helper function, idenfity if a configuration should be treated as an enterprise network
1826      * @hide
1827      */
1828     @UnsupportedAppUsage
isEnterprise()1829     public boolean isEnterprise() {
1830         return (allowedKeyManagement.get(KeyMgmt.WPA_EAP)
1831                 || allowedKeyManagement.get(KeyMgmt.IEEE8021X)
1832                 || allowedKeyManagement.get(KeyMgmt.SUITE_B_192))
1833                 && enterpriseConfig != null
1834                 && enterpriseConfig.getEapMethod() != WifiEnterpriseConfig.Eap.NONE;
1835     }
1836 
1837     @Override
toString()1838     public String toString() {
1839         StringBuilder sbuf = new StringBuilder();
1840         if (this.status == WifiConfiguration.Status.CURRENT) {
1841             sbuf.append("* ");
1842         } else if (this.status == WifiConfiguration.Status.DISABLED) {
1843             sbuf.append("- DSBLE ");
1844         }
1845         sbuf.append("ID: ").append(this.networkId).append(" SSID: ").append(this.SSID).
1846                 append(" PROVIDER-NAME: ").append(this.providerFriendlyName).
1847                 append(" BSSID: ").append(this.BSSID).append(" FQDN: ").append(this.FQDN)
1848                 .append(" PRIO: ").append(this.priority)
1849                 .append(" HIDDEN: ").append(this.hiddenSSID)
1850                 .append(" PMF: ").append(this.requirePMF)
1851                 .append('\n');
1852 
1853 
1854         sbuf.append(" NetworkSelectionStatus ")
1855                 .append(mNetworkSelectionStatus.getNetworkStatusString() + "\n");
1856         if (mNetworkSelectionStatus.getNetworkSelectionDisableReason() > 0) {
1857             sbuf.append(" mNetworkSelectionDisableReason ")
1858                     .append(mNetworkSelectionStatus.getNetworkDisableReasonString() + "\n");
1859 
1860             for (int index = mNetworkSelectionStatus.NETWORK_SELECTION_ENABLE;
1861                     index < mNetworkSelectionStatus.NETWORK_SELECTION_DISABLED_MAX; index++) {
1862                 if (mNetworkSelectionStatus.getDisableReasonCounter(index) != 0) {
1863                     sbuf.append(NetworkSelectionStatus.getNetworkDisableReasonString(index)
1864                             + " counter:" + mNetworkSelectionStatus.getDisableReasonCounter(index)
1865                             + "\n");
1866                 }
1867             }
1868         }
1869         if (mNetworkSelectionStatus.getConnectChoice() != null) {
1870             sbuf.append(" connect choice: ").append(mNetworkSelectionStatus.getConnectChoice());
1871             sbuf.append(" connect choice set time: ")
1872                     .append(TimeUtils.logTimeOfDay(
1873                             mNetworkSelectionStatus.getConnectChoiceTimestamp()));
1874         }
1875         sbuf.append(" hasEverConnected: ")
1876                 .append(mNetworkSelectionStatus.getHasEverConnected()).append("\n");
1877 
1878         if (this.numAssociation > 0) {
1879             sbuf.append(" numAssociation ").append(this.numAssociation).append("\n");
1880         }
1881         if (this.numNoInternetAccessReports > 0) {
1882             sbuf.append(" numNoInternetAccessReports ");
1883             sbuf.append(this.numNoInternetAccessReports).append("\n");
1884         }
1885         if (this.updateTime != null) {
1886             sbuf.append(" update ").append(this.updateTime).append("\n");
1887         }
1888         if (this.creationTime != null) {
1889             sbuf.append(" creation ").append(this.creationTime).append("\n");
1890         }
1891         if (this.didSelfAdd) sbuf.append(" didSelfAdd");
1892         if (this.selfAdded) sbuf.append(" selfAdded");
1893         if (this.validatedInternetAccess) sbuf.append(" validatedInternetAccess");
1894         if (this.ephemeral) sbuf.append(" ephemeral");
1895         if (this.osu) sbuf.append(" osu");
1896         if (this.trusted) sbuf.append(" trusted");
1897         if (this.fromWifiNetworkSuggestion) sbuf.append(" fromWifiNetworkSuggestion");
1898         if (this.fromWifiNetworkSpecifier) sbuf.append(" fromWifiNetworkSpecifier");
1899         if (this.meteredHint) sbuf.append(" meteredHint");
1900         if (this.useExternalScores) sbuf.append(" useExternalScores");
1901         if (this.didSelfAdd || this.selfAdded || this.validatedInternetAccess
1902                 || this.ephemeral || this.trusted || this.fromWifiNetworkSuggestion
1903                 || this.fromWifiNetworkSpecifier || this.meteredHint || this.useExternalScores) {
1904             sbuf.append("\n");
1905         }
1906         if (this.meteredOverride != METERED_OVERRIDE_NONE) {
1907             sbuf.append(" meteredOverride ").append(meteredOverride).append("\n");
1908         }
1909         sbuf.append(" macRandomizationSetting: ").append(macRandomizationSetting).append("\n");
1910         sbuf.append(" mRandomizedMacAddress: ").append(mRandomizedMacAddress).append("\n");
1911         sbuf.append(" KeyMgmt:");
1912         for (int k = 0; k < this.allowedKeyManagement.size(); k++) {
1913             if (this.allowedKeyManagement.get(k)) {
1914                 sbuf.append(" ");
1915                 if (k < KeyMgmt.strings.length) {
1916                     sbuf.append(KeyMgmt.strings[k]);
1917                 } else {
1918                     sbuf.append("??");
1919                 }
1920             }
1921         }
1922         sbuf.append(" Protocols:");
1923         for (int p = 0; p < this.allowedProtocols.size(); p++) {
1924             if (this.allowedProtocols.get(p)) {
1925                 sbuf.append(" ");
1926                 if (p < Protocol.strings.length) {
1927                     sbuf.append(Protocol.strings[p]);
1928                 } else {
1929                     sbuf.append("??");
1930                 }
1931             }
1932         }
1933         sbuf.append('\n');
1934         sbuf.append(" AuthAlgorithms:");
1935         for (int a = 0; a < this.allowedAuthAlgorithms.size(); a++) {
1936             if (this.allowedAuthAlgorithms.get(a)) {
1937                 sbuf.append(" ");
1938                 if (a < AuthAlgorithm.strings.length) {
1939                     sbuf.append(AuthAlgorithm.strings[a]);
1940                 } else {
1941                     sbuf.append("??");
1942                 }
1943             }
1944         }
1945         sbuf.append('\n');
1946         sbuf.append(" PairwiseCiphers:");
1947         for (int pc = 0; pc < this.allowedPairwiseCiphers.size(); pc++) {
1948             if (this.allowedPairwiseCiphers.get(pc)) {
1949                 sbuf.append(" ");
1950                 if (pc < PairwiseCipher.strings.length) {
1951                     sbuf.append(PairwiseCipher.strings[pc]);
1952                 } else {
1953                     sbuf.append("??");
1954                 }
1955             }
1956         }
1957         sbuf.append('\n');
1958         sbuf.append(" GroupCiphers:");
1959         for (int gc = 0; gc < this.allowedGroupCiphers.size(); gc++) {
1960             if (this.allowedGroupCiphers.get(gc)) {
1961                 sbuf.append(" ");
1962                 if (gc < GroupCipher.strings.length) {
1963                     sbuf.append(GroupCipher.strings[gc]);
1964                 } else {
1965                     sbuf.append("??");
1966                 }
1967             }
1968         }
1969         sbuf.append('\n');
1970         sbuf.append(" GroupMgmtCiphers:");
1971         for (int gmc = 0; gmc < this.allowedGroupManagementCiphers.size(); gmc++) {
1972             if (this.allowedGroupManagementCiphers.get(gmc)) {
1973                 sbuf.append(" ");
1974                 if (gmc < GroupMgmtCipher.strings.length) {
1975                     sbuf.append(GroupMgmtCipher.strings[gmc]);
1976                 } else {
1977                     sbuf.append("??");
1978                 }
1979             }
1980         }
1981         sbuf.append('\n');
1982         sbuf.append(" SuiteBCiphers:");
1983         for (int sbc = 0; sbc < this.allowedSuiteBCiphers.size(); sbc++) {
1984             if (this.allowedSuiteBCiphers.get(sbc)) {
1985                 sbuf.append(" ");
1986                 if (sbc < SuiteBCipher.strings.length) {
1987                     sbuf.append(SuiteBCipher.strings[sbc]);
1988                 } else {
1989                     sbuf.append("??");
1990                 }
1991             }
1992         }
1993         sbuf.append('\n').append(" PSK/SAE: ");
1994         if (this.preSharedKey != null) {
1995             sbuf.append('*');
1996         }
1997 
1998         sbuf.append("\nEnterprise config:\n");
1999         sbuf.append(enterpriseConfig);
2000 
2001         sbuf.append("IP config:\n");
2002         sbuf.append(mIpConfiguration.toString());
2003 
2004         if (mNetworkSelectionStatus.getNetworkSelectionBSSID() != null) {
2005             sbuf.append(" networkSelectionBSSID="
2006                     + mNetworkSelectionStatus.getNetworkSelectionBSSID());
2007         }
2008         long now_ms = SystemClock.elapsedRealtime();
2009         if (mNetworkSelectionStatus.getDisableTime() != NetworkSelectionStatus
2010                 .INVALID_NETWORK_SELECTION_DISABLE_TIMESTAMP) {
2011             sbuf.append('\n');
2012             long diff = now_ms - mNetworkSelectionStatus.getDisableTime();
2013             if (diff <= 0) {
2014                 sbuf.append(" blackListed since <incorrect>");
2015             } else {
2016                 sbuf.append(" blackListed: ").append(Long.toString(diff / 1000)).append("sec ");
2017             }
2018         }
2019         if (creatorUid != 0) sbuf.append(" cuid=" + creatorUid);
2020         if (creatorName != null) sbuf.append(" cname=" + creatorName);
2021         if (lastUpdateUid != 0) sbuf.append(" luid=" + lastUpdateUid);
2022         if (lastUpdateName != null) sbuf.append(" lname=" + lastUpdateName);
2023         if (updateIdentifier != null) sbuf.append(" updateIdentifier=" + updateIdentifier);
2024         sbuf.append(" lcuid=" + lastConnectUid);
2025         sbuf.append(" userApproved=" + userApprovedAsString(userApproved));
2026         sbuf.append(" noInternetAccessExpected=" + noInternetAccessExpected);
2027         sbuf.append(" ");
2028 
2029         if (this.lastConnected != 0) {
2030             sbuf.append('\n');
2031             sbuf.append("lastConnected: ").append(TimeUtils.logTimeOfDay(this.lastConnected));
2032             sbuf.append(" ");
2033         }
2034         sbuf.append('\n');
2035         if (this.linkedConfigurations != null) {
2036             for (String key : this.linkedConfigurations.keySet()) {
2037                 sbuf.append(" linked: ").append(key);
2038                 sbuf.append('\n');
2039             }
2040         }
2041         sbuf.append("recentFailure: ").append("Association Rejection code: ")
2042                 .append(recentFailure.getAssociationStatus()).append("\n");
2043         return sbuf.toString();
2044     }
2045 
2046     /** {@hide} */
2047     @UnsupportedAppUsage
getPrintableSsid()2048     public String getPrintableSsid() {
2049         if (SSID == null) return "";
2050         final int length = SSID.length();
2051         if (length > 2 && (SSID.charAt(0) == '"') && SSID.charAt(length - 1) == '"') {
2052             return SSID.substring(1, length - 1);
2053         }
2054 
2055         /** The ascii-encoded string format is P"<ascii-encoded-string>"
2056          * The decoding is implemented in the supplicant for a newly configured
2057          * network.
2058          */
2059         if (length > 3 && (SSID.charAt(0) == 'P') && (SSID.charAt(1) == '"') &&
2060                 (SSID.charAt(length-1) == '"')) {
2061             WifiSsid wifiSsid = WifiSsid.createFromAsciiEncoded(
2062                     SSID.substring(2, length - 1));
2063             return wifiSsid.toString();
2064         }
2065         return SSID;
2066     }
2067 
2068     /** @hide **/
userApprovedAsString(int userApproved)2069     public static String userApprovedAsString(int userApproved) {
2070         switch (userApproved) {
2071             case USER_APPROVED:
2072                 return "USER_APPROVED";
2073             case USER_BANNED:
2074                 return "USER_BANNED";
2075             case USER_UNSPECIFIED:
2076                 return "USER_UNSPECIFIED";
2077             default:
2078                 return "INVALID";
2079         }
2080     }
2081 
2082     /**
2083      * Get an identifier for associating credentials with this config
2084      * @param current configuration contains values for additional fields
2085      *                that are not part of this configuration. Used
2086      *                when a config with some fields is passed by an application.
2087      * @throws IllegalStateException if config is invalid for key id generation
2088      * @hide
2089      */
getKeyIdForCredentials(WifiConfiguration current)2090     public String getKeyIdForCredentials(WifiConfiguration current) {
2091         String keyMgmt = "";
2092 
2093         try {
2094             // Get current config details for fields that are not initialized
2095             if (TextUtils.isEmpty(SSID)) SSID = current.SSID;
2096             if (allowedKeyManagement.cardinality() == 0) {
2097                 allowedKeyManagement = current.allowedKeyManagement;
2098             }
2099             if (allowedKeyManagement.get(KeyMgmt.WPA_EAP)) {
2100                 keyMgmt += KeyMgmt.strings[KeyMgmt.WPA_EAP];
2101             }
2102             if (allowedKeyManagement.get(KeyMgmt.OSEN)) {
2103                 keyMgmt += KeyMgmt.strings[KeyMgmt.OSEN];
2104             }
2105             if (allowedKeyManagement.get(KeyMgmt.IEEE8021X)) {
2106                 keyMgmt += KeyMgmt.strings[KeyMgmt.IEEE8021X];
2107             }
2108             if (allowedKeyManagement.get(KeyMgmt.SUITE_B_192)) {
2109                 keyMgmt += KeyMgmt.strings[KeyMgmt.SUITE_B_192];
2110             }
2111 
2112             if (TextUtils.isEmpty(keyMgmt)) {
2113                 throw new IllegalStateException("Not an EAP network");
2114             }
2115 
2116             return trimStringForKeyId(SSID) + "_" + keyMgmt + "_" +
2117                     trimStringForKeyId(enterpriseConfig.getKeyId(current != null ?
2118                             current.enterpriseConfig : null));
2119         } catch (NullPointerException e) {
2120             throw new IllegalStateException("Invalid config details");
2121         }
2122     }
2123 
trimStringForKeyId(String string)2124     private String trimStringForKeyId(String string) {
2125         // Remove quotes and spaces
2126         return string.replace("\"", "").replace(" ", "");
2127     }
2128 
readBitSet(Parcel src)2129     private static BitSet readBitSet(Parcel src) {
2130         int cardinality = src.readInt();
2131 
2132         BitSet set = new BitSet();
2133         for (int i = 0; i < cardinality; i++) {
2134             set.set(src.readInt());
2135         }
2136 
2137         return set;
2138     }
2139 
writeBitSet(Parcel dest, BitSet set)2140     private static void writeBitSet(Parcel dest, BitSet set) {
2141         int nextSetBit = -1;
2142 
2143         dest.writeInt(set.cardinality());
2144 
2145         while ((nextSetBit = set.nextSetBit(nextSetBit + 1)) != -1) {
2146             dest.writeInt(nextSetBit);
2147         }
2148     }
2149 
2150     /** @hide */
2151     @UnsupportedAppUsage
getAuthType()2152     public int getAuthType() {
2153         if (allowedKeyManagement.cardinality() > 1) {
2154             throw new IllegalStateException("More than one auth type set");
2155         }
2156         if (allowedKeyManagement.get(KeyMgmt.WPA_PSK)) {
2157             return KeyMgmt.WPA_PSK;
2158         } else if (allowedKeyManagement.get(KeyMgmt.WPA2_PSK)) {
2159             return KeyMgmt.WPA2_PSK;
2160         } else if (allowedKeyManagement.get(KeyMgmt.WPA_EAP)) {
2161             return KeyMgmt.WPA_EAP;
2162         } else if (allowedKeyManagement.get(KeyMgmt.IEEE8021X)) {
2163             return KeyMgmt.IEEE8021X;
2164         } else if (allowedKeyManagement.get(KeyMgmt.SAE)) {
2165             return KeyMgmt.SAE;
2166         } else if (allowedKeyManagement.get(KeyMgmt.OWE)) {
2167             return KeyMgmt.OWE;
2168         } else if (allowedKeyManagement.get(KeyMgmt.SUITE_B_192)) {
2169             return KeyMgmt.SUITE_B_192;
2170         }
2171         return KeyMgmt.NONE;
2172     }
2173 
2174     /* @hide
2175      * Cache the config key, this seems useful as a speed up since a lot of
2176      * lookups in the config store are done and based on this key.
2177      */
2178     String mCachedConfigKey;
2179 
2180     /** @hide
2181      *  return the string used to calculate the hash in WifiConfigStore
2182      *  and uniquely identify this WifiConfiguration
2183      */
configKey(boolean allowCached)2184     public String configKey(boolean allowCached) {
2185         String key;
2186         if (allowCached && mCachedConfigKey != null) {
2187             key = mCachedConfigKey;
2188         } else if (providerFriendlyName != null) {
2189             key = FQDN + KeyMgmt.strings[KeyMgmt.WPA_EAP];
2190             if (!shared) {
2191                 key += "-" + Integer.toString(UserHandle.getUserId(creatorUid));
2192             }
2193         } else {
2194             key = getSsidAndSecurityTypeString();
2195             if (!shared) {
2196                 key += "-" + Integer.toString(UserHandle.getUserId(creatorUid));
2197             }
2198             mCachedConfigKey = key;
2199         }
2200         return key;
2201     }
2202 
2203     /** @hide
2204      *  return the SSID + security type in String format.
2205      */
getSsidAndSecurityTypeString()2206     public String getSsidAndSecurityTypeString() {
2207         String key;
2208         if (allowedKeyManagement.get(KeyMgmt.WPA_PSK)) {
2209             key = SSID + KeyMgmt.strings[KeyMgmt.WPA_PSK];
2210         } else if (allowedKeyManagement.get(KeyMgmt.WPA_EAP)
2211                 || allowedKeyManagement.get(KeyMgmt.IEEE8021X)) {
2212             key = SSID + KeyMgmt.strings[KeyMgmt.WPA_EAP];
2213         } else if (wepKeys[0] != null) {
2214             key = SSID + "WEP";
2215         } else if (allowedKeyManagement.get(KeyMgmt.OWE)) {
2216             key = SSID + KeyMgmt.strings[KeyMgmt.OWE];
2217         } else if (allowedKeyManagement.get(KeyMgmt.SAE)) {
2218             key = SSID + KeyMgmt.strings[KeyMgmt.SAE];
2219         } else if (allowedKeyManagement.get(KeyMgmt.SUITE_B_192)) {
2220             key = SSID + KeyMgmt.strings[KeyMgmt.SUITE_B_192];
2221         } else {
2222             key = SSID + KeyMgmt.strings[KeyMgmt.NONE];
2223         }
2224         return key;
2225     }
2226 
2227     /** @hide
2228      * get configKey, force calculating the config string
2229      */
configKey()2230     public String configKey() {
2231         return configKey(false);
2232     }
2233 
2234     /** @hide */
2235     @UnsupportedAppUsage
getIpConfiguration()2236     public IpConfiguration getIpConfiguration() {
2237         return mIpConfiguration;
2238     }
2239 
2240     /** @hide */
2241     @UnsupportedAppUsage
setIpConfiguration(IpConfiguration ipConfiguration)2242     public void setIpConfiguration(IpConfiguration ipConfiguration) {
2243         if (ipConfiguration == null) ipConfiguration = new IpConfiguration();
2244         mIpConfiguration = ipConfiguration;
2245     }
2246 
2247     /** @hide */
2248     @UnsupportedAppUsage
getStaticIpConfiguration()2249     public StaticIpConfiguration getStaticIpConfiguration() {
2250         return mIpConfiguration.getStaticIpConfiguration();
2251     }
2252 
2253     /** @hide */
2254     @UnsupportedAppUsage
setStaticIpConfiguration(StaticIpConfiguration staticIpConfiguration)2255     public void setStaticIpConfiguration(StaticIpConfiguration staticIpConfiguration) {
2256         mIpConfiguration.setStaticIpConfiguration(staticIpConfiguration);
2257     }
2258 
2259     /** @hide */
2260     @UnsupportedAppUsage
getIpAssignment()2261     public IpConfiguration.IpAssignment getIpAssignment() {
2262         return mIpConfiguration.ipAssignment;
2263     }
2264 
2265     /** @hide */
2266     @UnsupportedAppUsage
setIpAssignment(IpConfiguration.IpAssignment ipAssignment)2267     public void setIpAssignment(IpConfiguration.IpAssignment ipAssignment) {
2268         mIpConfiguration.ipAssignment = ipAssignment;
2269     }
2270 
2271     /** @hide */
2272     @UnsupportedAppUsage
getProxySettings()2273     public IpConfiguration.ProxySettings getProxySettings() {
2274         return mIpConfiguration.proxySettings;
2275     }
2276 
2277     /** @hide */
2278     @UnsupportedAppUsage
setProxySettings(IpConfiguration.ProxySettings proxySettings)2279     public void setProxySettings(IpConfiguration.ProxySettings proxySettings) {
2280         mIpConfiguration.proxySettings = proxySettings;
2281     }
2282 
2283     /**
2284      * Returns the HTTP proxy used by this object.
2285      * @return a {@link ProxyInfo httpProxy} representing the proxy specified by this
2286      *                  WifiConfiguration, or {@code null} if no proxy is specified.
2287      */
getHttpProxy()2288     public ProxyInfo getHttpProxy() {
2289         if (mIpConfiguration.proxySettings == IpConfiguration.ProxySettings.NONE) {
2290             return null;
2291         }
2292         return new ProxyInfo(mIpConfiguration.httpProxy);
2293     }
2294 
2295     /**
2296      * Set the {@link ProxyInfo} for this WifiConfiguration. This method should only be used by a
2297      * device owner or profile owner. When other apps attempt to save a {@link WifiConfiguration}
2298      * with modified proxy settings, the methods {@link WifiManager#addNetwork} and
2299      * {@link WifiManager#updateNetwork} fail and return {@code -1}.
2300      *
2301      * @param httpProxy {@link ProxyInfo} representing the httpProxy to be used by this
2302      *                  WifiConfiguration. Setting this to {@code null} will explicitly set no
2303      *                  proxy, removing any proxy that was previously set.
2304      * @exception IllegalArgumentException for invalid httpProxy
2305      */
setHttpProxy(ProxyInfo httpProxy)2306     public void setHttpProxy(ProxyInfo httpProxy) {
2307         if (httpProxy == null) {
2308             mIpConfiguration.setProxySettings(IpConfiguration.ProxySettings.NONE);
2309             mIpConfiguration.setHttpProxy(null);
2310             return;
2311         }
2312         ProxyInfo httpProxyCopy;
2313         ProxySettings proxySettingCopy;
2314         if (!Uri.EMPTY.equals(httpProxy.getPacFileUrl())) {
2315             proxySettingCopy = IpConfiguration.ProxySettings.PAC;
2316             // Construct a new PAC URL Proxy
2317             httpProxyCopy = new ProxyInfo(httpProxy.getPacFileUrl(), httpProxy.getPort());
2318         } else {
2319             proxySettingCopy = IpConfiguration.ProxySettings.STATIC;
2320             // Construct a new HTTP Proxy
2321             httpProxyCopy = new ProxyInfo(httpProxy.getHost(), httpProxy.getPort(),
2322                     httpProxy.getExclusionListAsString());
2323         }
2324         if (!httpProxyCopy.isValid()) {
2325             throw new IllegalArgumentException("Invalid ProxyInfo: " + httpProxyCopy.toString());
2326         }
2327         mIpConfiguration.setProxySettings(proxySettingCopy);
2328         mIpConfiguration.setHttpProxy(httpProxyCopy);
2329     }
2330 
2331     /** @hide */
2332     @UnsupportedAppUsage
setProxy(ProxySettings settings, ProxyInfo proxy)2333     public void setProxy(ProxySettings settings, ProxyInfo proxy) {
2334         mIpConfiguration.proxySettings = settings;
2335         mIpConfiguration.httpProxy = proxy;
2336     }
2337 
2338     /** Implement the Parcelable interface {@hide} */
describeContents()2339     public int describeContents() {
2340         return 0;
2341     }
2342 
2343     /** @hide */
setPasspointManagementObjectTree(String passpointManagementObjectTree)2344     public void setPasspointManagementObjectTree(String passpointManagementObjectTree) {
2345         mPasspointManagementObjectTree = passpointManagementObjectTree;
2346     }
2347 
2348     /** @hide */
getMoTree()2349     public String getMoTree() {
2350         return mPasspointManagementObjectTree;
2351     }
2352 
2353     /** copy constructor {@hide} */
2354     @UnsupportedAppUsage
WifiConfiguration(WifiConfiguration source)2355     public WifiConfiguration(WifiConfiguration source) {
2356         if (source != null) {
2357             networkId = source.networkId;
2358             status = source.status;
2359             SSID = source.SSID;
2360             BSSID = source.BSSID;
2361             FQDN = source.FQDN;
2362             roamingConsortiumIds = source.roamingConsortiumIds.clone();
2363             providerFriendlyName = source.providerFriendlyName;
2364             isHomeProviderNetwork = source.isHomeProviderNetwork;
2365             preSharedKey = source.preSharedKey;
2366 
2367             mNetworkSelectionStatus.copy(source.getNetworkSelectionStatus());
2368             apBand = source.apBand;
2369             apChannel = source.apChannel;
2370 
2371             wepKeys = new String[4];
2372             for (int i = 0; i < wepKeys.length; i++) {
2373                 wepKeys[i] = source.wepKeys[i];
2374             }
2375 
2376             wepTxKeyIndex = source.wepTxKeyIndex;
2377             priority = source.priority;
2378             hiddenSSID = source.hiddenSSID;
2379             allowedKeyManagement   = (BitSet) source.allowedKeyManagement.clone();
2380             allowedProtocols       = (BitSet) source.allowedProtocols.clone();
2381             allowedAuthAlgorithms  = (BitSet) source.allowedAuthAlgorithms.clone();
2382             allowedPairwiseCiphers = (BitSet) source.allowedPairwiseCiphers.clone();
2383             allowedGroupCiphers    = (BitSet) source.allowedGroupCiphers.clone();
2384             allowedGroupManagementCiphers = (BitSet) source.allowedGroupManagementCiphers.clone();
2385             allowedSuiteBCiphers    = (BitSet) source.allowedSuiteBCiphers.clone();
2386             enterpriseConfig = new WifiEnterpriseConfig(source.enterpriseConfig);
2387 
2388             defaultGwMacAddress = source.defaultGwMacAddress;
2389 
2390             mIpConfiguration = new IpConfiguration(source.mIpConfiguration);
2391 
2392             if ((source.linkedConfigurations != null)
2393                     && (source.linkedConfigurations.size() > 0)) {
2394                 linkedConfigurations = new HashMap<String, Integer>();
2395                 linkedConfigurations.putAll(source.linkedConfigurations);
2396             }
2397             mCachedConfigKey = null; //force null configKey
2398             selfAdded = source.selfAdded;
2399             validatedInternetAccess = source.validatedInternetAccess;
2400             isLegacyPasspointConfig = source.isLegacyPasspointConfig;
2401             ephemeral = source.ephemeral;
2402             osu = source.osu;
2403             trusted = source.trusted;
2404             fromWifiNetworkSuggestion = source.fromWifiNetworkSuggestion;
2405             fromWifiNetworkSpecifier = source.fromWifiNetworkSpecifier;
2406             meteredHint = source.meteredHint;
2407             meteredOverride = source.meteredOverride;
2408             useExternalScores = source.useExternalScores;
2409 
2410             didSelfAdd = source.didSelfAdd;
2411             lastConnectUid = source.lastConnectUid;
2412             lastUpdateUid = source.lastUpdateUid;
2413             creatorUid = source.creatorUid;
2414             creatorName = source.creatorName;
2415             lastUpdateName = source.lastUpdateName;
2416             peerWifiConfiguration = source.peerWifiConfiguration;
2417 
2418             lastConnected = source.lastConnected;
2419             lastDisconnected = source.lastDisconnected;
2420             numScorerOverride = source.numScorerOverride;
2421             numScorerOverrideAndSwitchedNetwork = source.numScorerOverrideAndSwitchedNetwork;
2422             numAssociation = source.numAssociation;
2423             userApproved = source.userApproved;
2424             numNoInternetAccessReports = source.numNoInternetAccessReports;
2425             noInternetAccessExpected = source.noInternetAccessExpected;
2426             creationTime = source.creationTime;
2427             updateTime = source.updateTime;
2428             shared = source.shared;
2429             recentFailure.setAssociationStatus(source.recentFailure.getAssociationStatus());
2430             mRandomizedMacAddress = source.mRandomizedMacAddress;
2431             macRandomizationSetting = source.macRandomizationSetting;
2432             requirePMF = source.requirePMF;
2433             updateIdentifier = source.updateIdentifier;
2434         }
2435     }
2436 
2437     /** Implement the Parcelable interface {@hide} */
2438     @Override
writeToParcel(Parcel dest, int flags)2439     public void writeToParcel(Parcel dest, int flags) {
2440         dest.writeInt(networkId);
2441         dest.writeInt(status);
2442         mNetworkSelectionStatus.writeToParcel(dest);
2443         dest.writeString(SSID);
2444         dest.writeString(BSSID);
2445         dest.writeInt(apBand);
2446         dest.writeInt(apChannel);
2447         dest.writeString(FQDN);
2448         dest.writeString(providerFriendlyName);
2449         dest.writeInt(isHomeProviderNetwork ? 1 : 0);
2450         dest.writeInt(roamingConsortiumIds.length);
2451         for (long roamingConsortiumId : roamingConsortiumIds) {
2452             dest.writeLong(roamingConsortiumId);
2453         }
2454         dest.writeString(preSharedKey);
2455         for (String wepKey : wepKeys) {
2456             dest.writeString(wepKey);
2457         }
2458         dest.writeInt(wepTxKeyIndex);
2459         dest.writeInt(priority);
2460         dest.writeInt(hiddenSSID ? 1 : 0);
2461         dest.writeInt(requirePMF ? 1 : 0);
2462         dest.writeString(updateIdentifier);
2463 
2464         writeBitSet(dest, allowedKeyManagement);
2465         writeBitSet(dest, allowedProtocols);
2466         writeBitSet(dest, allowedAuthAlgorithms);
2467         writeBitSet(dest, allowedPairwiseCiphers);
2468         writeBitSet(dest, allowedGroupCiphers);
2469         writeBitSet(dest, allowedGroupManagementCiphers);
2470         writeBitSet(dest, allowedSuiteBCiphers);
2471 
2472         dest.writeParcelable(enterpriseConfig, flags);
2473 
2474         dest.writeParcelable(mIpConfiguration, flags);
2475         dest.writeString(dhcpServer);
2476         dest.writeString(defaultGwMacAddress);
2477         dest.writeInt(selfAdded ? 1 : 0);
2478         dest.writeInt(didSelfAdd ? 1 : 0);
2479         dest.writeInt(validatedInternetAccess ? 1 : 0);
2480         dest.writeInt(isLegacyPasspointConfig ? 1 : 0);
2481         dest.writeInt(ephemeral ? 1 : 0);
2482         dest.writeInt(trusted ? 1 : 0);
2483         dest.writeInt(fromWifiNetworkSuggestion ? 1 : 0);
2484         dest.writeInt(fromWifiNetworkSpecifier ? 1 : 0);
2485         dest.writeInt(meteredHint ? 1 : 0);
2486         dest.writeInt(meteredOverride);
2487         dest.writeInt(useExternalScores ? 1 : 0);
2488         dest.writeInt(creatorUid);
2489         dest.writeInt(lastConnectUid);
2490         dest.writeInt(lastUpdateUid);
2491         dest.writeString(creatorName);
2492         dest.writeString(lastUpdateName);
2493         dest.writeInt(numScorerOverride);
2494         dest.writeInt(numScorerOverrideAndSwitchedNetwork);
2495         dest.writeInt(numAssociation);
2496         dest.writeInt(userApproved);
2497         dest.writeInt(numNoInternetAccessReports);
2498         dest.writeInt(noInternetAccessExpected ? 1 : 0);
2499         dest.writeInt(shared ? 1 : 0);
2500         dest.writeString(mPasspointManagementObjectTree);
2501         dest.writeInt(recentFailure.getAssociationStatus());
2502         dest.writeParcelable(mRandomizedMacAddress, flags);
2503         dest.writeInt(macRandomizationSetting);
2504         dest.writeInt(osu ? 1 : 0);
2505     }
2506 
2507     /** Implement the Parcelable interface {@hide} */
2508     @UnsupportedAppUsage
2509     public static final @android.annotation.NonNull Creator<WifiConfiguration> CREATOR =
2510         new Creator<WifiConfiguration>() {
2511             public WifiConfiguration createFromParcel(Parcel in) {
2512                 WifiConfiguration config = new WifiConfiguration();
2513                 config.networkId = in.readInt();
2514                 config.status = in.readInt();
2515                 config.mNetworkSelectionStatus.readFromParcel(in);
2516                 config.SSID = in.readString();
2517                 config.BSSID = in.readString();
2518                 config.apBand = in.readInt();
2519                 config.apChannel = in.readInt();
2520                 config.FQDN = in.readString();
2521                 config.providerFriendlyName = in.readString();
2522                 config.isHomeProviderNetwork = in.readInt() != 0;
2523                 int numRoamingConsortiumIds = in.readInt();
2524                 config.roamingConsortiumIds = new long[numRoamingConsortiumIds];
2525                 for (int i = 0; i < numRoamingConsortiumIds; i++) {
2526                     config.roamingConsortiumIds[i] = in.readLong();
2527                 }
2528                 config.preSharedKey = in.readString();
2529                 for (int i = 0; i < config.wepKeys.length; i++) {
2530                     config.wepKeys[i] = in.readString();
2531                 }
2532                 config.wepTxKeyIndex = in.readInt();
2533                 config.priority = in.readInt();
2534                 config.hiddenSSID = in.readInt() != 0;
2535                 config.requirePMF = in.readInt() != 0;
2536                 config.updateIdentifier = in.readString();
2537 
2538                 config.allowedKeyManagement   = readBitSet(in);
2539                 config.allowedProtocols       = readBitSet(in);
2540                 config.allowedAuthAlgorithms  = readBitSet(in);
2541                 config.allowedPairwiseCiphers = readBitSet(in);
2542                 config.allowedGroupCiphers    = readBitSet(in);
2543                 config.allowedGroupManagementCiphers = readBitSet(in);
2544                 config.allowedSuiteBCiphers   = readBitSet(in);
2545 
2546                 config.enterpriseConfig = in.readParcelable(null);
2547                 config.setIpConfiguration(in.readParcelable(null));
2548                 config.dhcpServer = in.readString();
2549                 config.defaultGwMacAddress = in.readString();
2550                 config.selfAdded = in.readInt() != 0;
2551                 config.didSelfAdd = in.readInt() != 0;
2552                 config.validatedInternetAccess = in.readInt() != 0;
2553                 config.isLegacyPasspointConfig = in.readInt() != 0;
2554                 config.ephemeral = in.readInt() != 0;
2555                 config.trusted = in.readInt() != 0;
2556                 config.fromWifiNetworkSuggestion =  in.readInt() != 0;
2557                 config.fromWifiNetworkSpecifier =  in.readInt() != 0;
2558                 config.meteredHint = in.readInt() != 0;
2559                 config.meteredOverride = in.readInt();
2560                 config.useExternalScores = in.readInt() != 0;
2561                 config.creatorUid = in.readInt();
2562                 config.lastConnectUid = in.readInt();
2563                 config.lastUpdateUid = in.readInt();
2564                 config.creatorName = in.readString();
2565                 config.lastUpdateName = in.readString();
2566                 config.numScorerOverride = in.readInt();
2567                 config.numScorerOverrideAndSwitchedNetwork = in.readInt();
2568                 config.numAssociation = in.readInt();
2569                 config.userApproved = in.readInt();
2570                 config.numNoInternetAccessReports = in.readInt();
2571                 config.noInternetAccessExpected = in.readInt() != 0;
2572                 config.shared = in.readInt() != 0;
2573                 config.mPasspointManagementObjectTree = in.readString();
2574                 config.recentFailure.setAssociationStatus(in.readInt());
2575                 config.mRandomizedMacAddress = in.readParcelable(null);
2576                 config.macRandomizationSetting = in.readInt();
2577                 config.osu = in.readInt() != 0;
2578                 return config;
2579             }
2580 
2581             public WifiConfiguration[] newArray(int size) {
2582                 return new WifiConfiguration[size];
2583             }
2584         };
2585 
2586     /**
2587      * Serializes the object for backup
2588      * @hide
2589      */
getBytesForBackup()2590     public byte[] getBytesForBackup() throws IOException {
2591         ByteArrayOutputStream baos = new ByteArrayOutputStream();
2592         DataOutputStream out = new DataOutputStream(baos);
2593 
2594         out.writeInt(BACKUP_VERSION);
2595         BackupUtils.writeString(out, SSID);
2596         out.writeInt(apBand);
2597         out.writeInt(apChannel);
2598         BackupUtils.writeString(out, preSharedKey);
2599         out.writeInt(getAuthType());
2600         out.writeBoolean(hiddenSSID);
2601         return baos.toByteArray();
2602     }
2603 
2604     /**
2605      * Deserializes a byte array into the WiFiConfiguration Object
2606      * @hide
2607      */
getWifiConfigFromBackup(DataInputStream in)2608     public static WifiConfiguration getWifiConfigFromBackup(DataInputStream in) throws IOException,
2609             BackupUtils.BadVersionException {
2610         WifiConfiguration config = new WifiConfiguration();
2611         int version = in.readInt();
2612         if (version < 1 || version > BACKUP_VERSION) {
2613             throw new BackupUtils.BadVersionException("Unknown Backup Serialization Version");
2614         }
2615 
2616         if (version == 1) return null; // Version 1 is a bad dataset.
2617 
2618         config.SSID = BackupUtils.readString(in);
2619         config.apBand = in.readInt();
2620         config.apChannel = in.readInt();
2621         config.preSharedKey = BackupUtils.readString(in);
2622         config.allowedKeyManagement.set(in.readInt());
2623         if (version >= 3) {
2624             config.hiddenSSID = in.readBoolean();
2625         }
2626         return config;
2627     }
2628 }
2629