/* * Copyright (C) 2016 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.server.wifi.util; import android.annotation.NonNull; import android.annotation.Nullable; import android.app.compat.CompatChanges; import android.net.InetAddresses; import android.net.IpConfiguration; import android.net.IpConfiguration.IpAssignment; import android.net.IpConfiguration.ProxySettings; import android.net.LinkAddress; import android.net.MacAddress; import android.net.ProxyInfo; import android.net.RouteInfo; import android.net.StaticIpConfiguration; import android.net.Uri; import android.net.wifi.OuiKeyedData; import android.net.wifi.ScanResult; import android.net.wifi.SecurityParams; import android.net.wifi.SoftApConfiguration; import android.net.wifi.WifiConfiguration; import android.net.wifi.WifiConfiguration.NetworkSelectionStatus; import android.net.wifi.WifiEnterpriseConfig; import android.net.wifi.WifiManager; import android.net.wifi.WifiMigration; import android.net.wifi.WifiSsid; import android.os.ParcelUuid; import android.os.PersistableBundle; import android.text.TextUtils; import android.util.Log; import android.util.Pair; import android.util.SparseIntArray; import com.android.modules.utils.build.SdkLevel; import org.xmlpull.v1.XmlPullParser; import org.xmlpull.v1.XmlPullParserException; import org.xmlpull.v1.XmlSerializer; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.net.Inet4Address; import java.net.InetAddress; import java.nio.charset.StandardCharsets; import java.util.ArrayList; import java.util.Arrays; import java.util.BitSet; import java.util.Collections; import java.util.HashMap; import java.util.List; import java.util.Locale; /** * Utils for manipulating XML data. This is essentially a wrapper over XmlUtils provided by core. * The utility provides methods to write/parse section headers and write/parse values. * This utility is designed for formatting the XML into the following format: * *
* * * ... * * * * ... * *
*
* * Note: These utility methods are meant to be used for: * 1. Backup/restore wifi network data to/from cloud. * 2. Persisting wifi network data to/from disk. */ public class XmlUtil { private static final String TAG = "WifiXmlUtil"; public static final String XML_TAG_VENDOR_DATA_LIST = "VendorDataList"; public static final String XML_TAG_OUI_KEYED_DATA = "OuiKeyedData"; public static final String XML_TAG_VENDOR_DATA_OUI = "VendorDataOui"; public static final String XML_TAG_PERSISTABLE_BUNDLE = "PersistableBundle"; /** * Ensure that the XML stream is at a start tag or the end of document. * * @throws XmlPullParserException if parsing errors occur. */ private static void gotoStartTag(XmlPullParser in) throws XmlPullParserException, IOException { int type = in.getEventType(); while (type != XmlPullParser.START_TAG && type != XmlPullParser.END_DOCUMENT) { type = in.next(); } } /** * Ensure that the XML stream is at an end tag or the end of document. * * @throws XmlPullParserException if parsing errors occur. */ private static void gotoEndTag(XmlPullParser in) throws XmlPullParserException, IOException { int type = in.getEventType(); while (type != XmlPullParser.END_TAG && type != XmlPullParser.END_DOCUMENT) { type = in.next(); } } /** * Start processing the XML stream at the document header. * * @param in XmlPullParser instance pointing to the XML stream. * @param headerName expected name for the start tag. * @throws XmlPullParserException if parsing errors occur. */ public static void gotoDocumentStart(XmlPullParser in, String headerName) throws XmlPullParserException, IOException { XmlUtilHelper.beginDocument(in, headerName); } /** * Move the XML stream to the next section header or indicate if there are no more sections. * The provided outerDepth is used to find sub sections within that depth. * * Use this to move across sections if the ordering of sections are variable. The returned name * can be used to decide what section is next. * * @param in XmlPullParser instance pointing to the XML stream. * @param headerName An array of one string, used to return the name of the next section. * @param outerDepth Find section within this depth. * @return {@code true} if a next section is found, {@code false} if there are no more sections. * @throws XmlPullParserException if parsing errors occur. */ public static boolean gotoNextSectionOrEnd( XmlPullParser in, String[] headerName, int outerDepth) throws XmlPullParserException, IOException { if (XmlUtilHelper.nextElementWithin(in, outerDepth)) { headerName[0] = in.getName(); return true; } return false; } /** * Move the XML stream to the next section header or indicate if there are no more sections. * If a section, exists ensure that the name matches the provided name. * The provided outerDepth is used to find sub sections within that depth. * * Use this to move across repeated sections until the end. * * @param in XmlPullParser instance pointing to the XML stream. * @param expectedName expected name for the section header. * @param outerDepth Find section within this depth. * @return {@code true} if a next section is found, {@code false} if there are no more sections. * @throws XmlPullParserException if the section header name does not match |expectedName|, * or if parsing errors occur. */ public static boolean gotoNextSectionWithNameOrEnd( XmlPullParser in, String expectedName, int outerDepth) throws XmlPullParserException, IOException { String[] headerName = new String[1]; if (gotoNextSectionOrEnd(in, headerName, outerDepth)) { if (headerName[0].equals(expectedName)) { return true; } throw new XmlPullParserException( "Next section name does not match expected name: " + expectedName); } return false; } /** * Move the XML stream to the next section header and ensure that the name matches the provided * name. * The provided outerDepth is used to find sub sections within that depth. * * Use this to move across sections if the ordering of sections are fixed. * * @param in XmlPullParser instance pointing to the XML stream. * @param expectedName expected name for the section header. * @param outerDepth Find section within this depth. * @throws XmlPullParserException if the section header name does not match |expectedName|, * there are no more sections or if parsing errors occur. */ public static void gotoNextSectionWithName( XmlPullParser in, String expectedName, int outerDepth) throws XmlPullParserException, IOException { if (!gotoNextSectionWithNameOrEnd(in, expectedName, outerDepth)) { throw new XmlPullParserException("Section not found. Expected: " + expectedName); } } /** * Checks if the stream is at the end of a section of values. This moves the stream to next tag * and checks if it finds an end tag at the specified depth. * * @param in XmlPullParser instance pointing to the XML stream. * @param sectionDepth depth of the start tag of this section. Used to match the end tag. * @return {@code true} if a end tag at the provided depth is found, {@code false} otherwise * @throws XmlPullParserException if parsing errors occur. */ public static boolean isNextSectionEnd(XmlPullParser in, int sectionDepth) throws XmlPullParserException, IOException { return !XmlUtilHelper.nextElementWithin(in, sectionDepth); } /** * Read the current value in the XML stream using core XmlUtils and stores the retrieved * value name in the string provided. This method reads the value contained in current start * tag. * Note: Because there could be genuine null values being read from the XML, this method raises * an exception to indicate errors. * * @param in XmlPullParser instance pointing to the XML stream. * @param valueName An array of one string, used to return the name attribute * of the value's tag. * @return value retrieved from the XML stream. * @throws XmlPullParserException if parsing errors occur. */ public static Object readCurrentValue(XmlPullParser in, String[] valueName) throws XmlPullParserException, IOException { Object value = XmlUtilHelper.readValueXml(in, valueName); // XmlUtils.readValue does not always move the stream to the end of the tag. So, move // it to the end tag before returning from here. gotoEndTag(in); return value; } /** * Read the next value in the XML stream using core XmlUtils and ensure that it matches the * provided name. This method moves the stream to the next start tag and reads the value * contained in it. * Note: Because there could be genuine null values being read from the XML, this method raises * an exception to indicate errors. * * @param in XmlPullParser instance pointing to the XML stream. * @return value retrieved from the XML stream. * @throws XmlPullParserException if the value read does not match |expectedName|, * or if parsing errors occur. */ public static Object readNextValueWithName(XmlPullParser in, String expectedName) throws XmlPullParserException, IOException { String[] valueName = new String[1]; XmlUtilHelper.nextElement(in); Object value = readCurrentValue(in, valueName); if (valueName[0].equals(expectedName)) { return value; } throw new XmlPullParserException( "Value not found. Expected: " + expectedName + ", but got: " + valueName[0]); } /** * Write the XML document start with the provided document header name. * * @param out XmlSerializer instance pointing to the XML stream. * @param headerName name for the start tag. */ public static void writeDocumentStart(XmlSerializer out, String headerName) throws IOException { out.startDocument(null, true); out.startTag(null, headerName); } /** * Write the XML document end with the provided document header name. * * @param out XmlSerializer instance pointing to the XML stream. * @param headerName name for the end tag. */ public static void writeDocumentEnd(XmlSerializer out, String headerName) throws IOException { out.endTag(null, headerName); out.endDocument(); } /** * Write a section start header tag with the provided section name. * * @param out XmlSerializer instance pointing to the XML stream. * @param headerName name for the start tag. */ public static void writeNextSectionStart(XmlSerializer out, String headerName) throws IOException { out.startTag(null, headerName); } /** * Write a section end header tag with the provided section name. * * @param out XmlSerializer instance pointing to the XML stream. * @param headerName name for the end tag. */ public static void writeNextSectionEnd(XmlSerializer out, String headerName) throws IOException { out.endTag(null, headerName); } /** * Write the value with the provided name in the XML stream using core XmlUtils. * * @param out XmlSerializer instance pointing to the XML stream. * @param name name of the value. * @param value value to be written. */ public static void writeNextValue(XmlSerializer out, String name, Object value) throws XmlPullParserException, IOException { XmlUtilHelper.writeValueXml(value, name, out); } /** * Utility class to serialize and deserialize {@link WifiConfiguration} object to XML & * vice versa. * This is used by both {@link com.android.server.wifi.WifiConfigStore} & * {@link com.android.server.wifi.WifiBackupRestore} modules. * The |writeConfigurationToXml| has 2 versions, one for backup and one for config store. * There is only 1 version of |parseXmlToConfiguration| for both backup & config store. * The parse method is written so that any element added/deleted in future revisions can * be easily handled. */ public static class WifiConfigurationXmlUtil { /** * List of XML tags corresponding to WifiConfiguration object elements. */ public static final String XML_TAG_SSID = "SSID"; public static final String XML_TAG_BSSID = "BSSID"; public static final String XML_TAG_CONFIG_KEY = "ConfigKey"; public static final String XML_TAG_PRE_SHARED_KEY = "PreSharedKey"; public static final String XML_TAG_WEP_KEYS = "WEPKeys"; public static final String XML_TAG_WEP_TX_KEY_INDEX = "WEPTxKeyIndex"; public static final String XML_TAG_HIDDEN_SSID = "HiddenSSID"; public static final String XML_TAG_REQUIRE_PMF = "RequirePMF"; public static final String XML_TAG_ALLOWED_KEY_MGMT = "AllowedKeyMgmt"; public static final String XML_TAG_ALLOWED_PROTOCOLS = "AllowedProtocols"; public static final String XML_TAG_ALLOWED_AUTH_ALGOS = "AllowedAuthAlgos"; public static final String XML_TAG_ALLOWED_GROUP_CIPHERS = "AllowedGroupCiphers"; public static final String XML_TAG_ALLOWED_PAIRWISE_CIPHERS = "AllowedPairwiseCiphers"; public static final String XML_TAG_ALLOWED_GROUP_MGMT_CIPHERS = "AllowedGroupMgmtCiphers"; public static final String XML_TAG_ALLOWED_SUITE_B_CIPHERS = "AllowedSuiteBCiphers"; public static final String XML_TAG_SHARED = "Shared"; public static final String XML_TAG_STATUS = "Status"; public static final String XML_TAG_FQDN = "FQDN"; public static final String XML_TAG_PROVIDER_FRIENDLY_NAME = "ProviderFriendlyName"; public static final String XML_TAG_LINKED_NETWORKS_LIST = "LinkedNetworksList"; public static final String XML_TAG_DEFAULT_GW_MAC_ADDRESS = "DefaultGwMacAddress"; public static final String XML_TAG_VALIDATED_INTERNET_ACCESS = "ValidatedInternetAccess"; public static final String XML_TAG_NO_INTERNET_ACCESS_EXPECTED = "NoInternetAccessExpected"; public static final String XML_TAG_METERED_HINT = "MeteredHint"; public static final String XML_TAG_METERED_OVERRIDE = "MeteredOverride"; public static final String XML_TAG_USE_EXTERNAL_SCORES = "UseExternalScores"; public static final String XML_TAG_CREATOR_UID = "CreatorUid"; public static final String XML_TAG_CREATOR_NAME = "CreatorName"; public static final String XML_TAG_LAST_UPDATE_UID = "LastUpdateUid"; public static final String XML_TAG_LAST_UPDATE_NAME = "LastUpdateName"; public static final String XML_TAG_LAST_CONNECT_UID = "LastConnectUid"; public static final String XML_TAG_IS_LEGACY_PASSPOINT_CONFIG = "IsLegacyPasspointConfig"; public static final String XML_TAG_ROAMING_CONSORTIUM_OIS = "RoamingConsortiumOIs"; public static final String XML_TAG_RANDOMIZED_MAC_ADDRESS = "RandomizedMacAddress"; public static final String XML_TAG_MAC_RANDOMIZATION_SETTING = "MacRandomizationSetting"; public static final String XML_TAG_SEND_DHCP_HOSTNAME = "SendDhcpHostname"; public static final String XML_TAG_CARRIER_ID = "CarrierId"; public static final String XML_TAG_SUBSCRIPTION_ID = "SubscriptionId"; public static final String XML_TAG_IS_AUTO_JOIN = "AutoJoinEnabled"; public static final String XML_TAG_PRIORITY = "Priority"; public static final String XML_TAG_DELETION_PRIORITY = "DeletionPriority"; public static final String XML_TAG_NUM_REBOOTS_SINCE_LAST_USE = "NumRebootsSinceLastUse"; public static final String XML_TAG_IS_TRUSTED = "Trusted"; public static final String XML_TAG_IS_OEM_PAID = "OemPaid"; public static final String XML_TAG_IS_OEM_PRIVATE = "OemPrivate"; public static final String XML_TAG_IS_CARRIER_MERGED = "CarrierMerged"; public static final String XML_TAG_SECURITY_PARAMS_LIST = "SecurityParamsList"; public static final String XML_TAG_SECURITY_PARAMS = "SecurityParams"; public static final String XML_TAG_SECURITY_TYPE = "SecurityType"; public static final String XML_TAG_IS_ENABLED = "IsEnabled"; public static final String XML_TAG_SAE_IS_H2E_ONLY_MODE = "SaeIsH2eOnlyMode"; public static final String XML_TAG_SAE_IS_PK_ONLY_MODE = "SaeIsPkOnlyMode"; public static final String XML_TAG_IS_ADDED_BY_AUTO_UPGRADE = "IsAddedByAutoUpgrade"; private static final String XML_TAG_IS_MOST_RECENTLY_CONNECTED = "IsMostRecentlyConnected"; private static final String XML_TAG_IS_RESTRICTED = "IsRestricted"; private static final String XML_TAG_SUBSCRIPTION_GROUP = "SubscriptionGroup"; public static final String XML_TAG_BSSID_ALLOW_LIST = "bssidAllowList"; public static final String XML_TAG_IS_REPEATER_ENABLED = "RepeaterEnabled"; public static final String XML_TAG_DPP_PRIVATE_EC_KEY = "DppPrivateEcKey"; public static final String XML_TAG_DPP_CONNECTOR = "DppConnector"; public static final String XML_TAG_DPP_CSIGN_KEY = "DppCSignKey"; public static final String XML_TAG_DPP_NET_ACCESS_KEY = "DppNetAccessKey"; public static final String XML_TAG_ENABLE_WIFI7 = "EnableWifi7"; /** * Write Wep Keys to the XML stream. * WepKeys array is initialized in WifiConfiguration constructor and all the elements * are set to null. User may choose to set any one of the key elements in WifiConfiguration. * XmlUtils serialization doesn't handle this array of nulls well . * So, write empty strings if the keys are not initialized and null if all * the elements are empty. */ private static void writeWepKeysToXml(XmlSerializer out, String[] wepKeys, @Nullable WifiConfigStoreEncryptionUtil encryptionUtil) throws XmlPullParserException, IOException { final int len = wepKeys == null ? 0 : wepKeys.length; String[] wepKeysToWrite = new String[len]; boolean hasWepKey = false; for (int i = 0; i < len; i++) { if (wepKeys[i] == null) { wepKeysToWrite[i] = new String(); } else { wepKeysToWrite[i] = wepKeys[i]; hasWepKey = true; } } if (!hasWepKey) { XmlUtil.writeNextValue(out, XML_TAG_WEP_KEYS, null); return; } if (encryptionUtil == null) { XmlUtil.writeNextValue(out, XML_TAG_WEP_KEYS, wepKeysToWrite); return; } EncryptedData[] encryptedDataArray = new EncryptedData[len]; for (int i = 0; i < len; i++) { if (wepKeys[i] == null) { encryptedDataArray[i] = new EncryptedData(new byte[0], new byte[0]); } else { encryptedDataArray[i] = encryptionUtil.encrypt(wepKeys[i].getBytes()); if (encryptedDataArray[i] == null) { // We silently fail encryption failures! Log.wtf(TAG, "Encryption of WEP keys failed"); // If any key encryption fails, we just fall back with unencrypted keys. XmlUtil.writeNextValue(out, XML_TAG_WEP_KEYS, wepKeysToWrite); return; } } } XmlUtil.writeNextSectionStart(out, XML_TAG_WEP_KEYS); for (int i = 0; i < len; i++) { XmlUtil.EncryptedDataXmlUtil.writeToXml(out, encryptedDataArray[i]); } XmlUtil.writeNextSectionEnd(out, XML_TAG_WEP_KEYS); } /** * Write preshared key to the XML stream. * * If encryptionUtil is null or if encryption fails for some reason, the pre-shared * key is stored in plaintext, else the encrypted psk is stored. */ private static void writePreSharedKeyToXml( XmlSerializer out, WifiConfiguration wifiConfig, @Nullable WifiConfigStoreEncryptionUtil encryptionUtil) throws XmlPullParserException, IOException { EncryptedData encryptedData = null; if (encryptionUtil != null && wifiConfig.preSharedKey != null) { if (wifiConfig.hasEncryptedPreSharedKey() && !wifiConfig.hasPreSharedKeyChanged()) { encryptedData = new EncryptedData(wifiConfig.getEncryptedPreSharedKey(), wifiConfig.getEncryptedPreSharedKeyIv()); } else { encryptedData = encryptionUtil.encrypt(wifiConfig.preSharedKey.getBytes()); if (encryptedData == null) { // We silently fail encryption failures! Log.wtf(TAG, "Encryption of preSharedKey failed"); } } } if (encryptedData != null) { writeNextSectionStart(out, XML_TAG_PRE_SHARED_KEY); EncryptedDataXmlUtil.writeToXml(out, encryptedData); wifiConfig.setEncryptedPreSharedKey(encryptedData.getEncryptedData(), encryptedData.getIv()); wifiConfig.setHasPreSharedKeyChanged(false); writeNextSectionEnd(out, XML_TAG_PRE_SHARED_KEY); } else { writeNextValue(out, XML_TAG_PRE_SHARED_KEY, wifiConfig.preSharedKey); } } private static void writeSecurityParamsListToXml( XmlSerializer out, WifiConfiguration configuration) throws XmlPullParserException, IOException { XmlUtil.writeNextSectionStart(out, XML_TAG_SECURITY_PARAMS_LIST); for (SecurityParams params: configuration.getSecurityParamsList()) { XmlUtil.writeNextSectionStart(out, XML_TAG_SECURITY_PARAMS); XmlUtil.writeNextValue( out, XML_TAG_SECURITY_TYPE, params.getSecurityType()); XmlUtil.writeNextValue( out, XML_TAG_IS_ENABLED, params.isEnabled()); XmlUtil.writeNextValue( out, XML_TAG_SAE_IS_H2E_ONLY_MODE, params.isSaeH2eOnlyMode()); XmlUtil.writeNextValue( out, XML_TAG_SAE_IS_PK_ONLY_MODE, params.isSaePkOnlyMode()); XmlUtil.writeNextValue( out, XML_TAG_IS_ADDED_BY_AUTO_UPGRADE, params.isAddedByAutoUpgrade()); XmlUtil.writeNextValue( out, XML_TAG_ALLOWED_SUITE_B_CIPHERS, params.getAllowedSuiteBCiphers().toByteArray()); XmlUtil.writeNextSectionEnd(out, XML_TAG_SECURITY_PARAMS); } XmlUtil.writeNextSectionEnd(out, XML_TAG_SECURITY_PARAMS_LIST); } private static void writeEncryptedBytesToXml( XmlSerializer out, @Nullable WifiConfigStoreEncryptionUtil encryptionUtil, String tag, byte[] data) throws XmlPullParserException, IOException { EncryptedData encryptedData = null; if (encryptionUtil != null) { encryptedData = encryptionUtil.encrypt(data); if (encryptedData == null && data != null && data.length != 0) { // We silently fail encryption failures! Log.wtf(TAG, "Encryption of " + tag + " failed"); } } if (encryptedData != null) { XmlUtil.writeNextSectionStart(out, tag); EncryptedDataXmlUtil.writeToXml(out, encryptedData); XmlUtil.writeNextSectionEnd(out, tag); } else { XmlUtil.writeNextValue(out, tag, data); } } /** * Write dpp configuration and connection keys to the XML stream. * * If encryptionUtil is null or if encryption fails for some reason, the dpp * keys are stored in plaintext, else the encrypted keys are stored. */ private static void writeDppConfigurationToXml( XmlSerializer out, WifiConfiguration configuration, @Nullable WifiConfigStoreEncryptionUtil encryptionUtil) throws XmlPullParserException, IOException { writeEncryptedBytesToXml(out, encryptionUtil, XML_TAG_DPP_PRIVATE_EC_KEY, configuration.getDppPrivateEcKey()); writeEncryptedBytesToXml(out, encryptionUtil, XML_TAG_DPP_CONNECTOR, configuration.getDppConnector()); writeEncryptedBytesToXml(out, encryptionUtil, XML_TAG_DPP_CSIGN_KEY, configuration.getDppCSignKey()); writeEncryptedBytesToXml(out, encryptionUtil, XML_TAG_DPP_NET_ACCESS_KEY, configuration.getDppNetAccessKey()); } /** * Write the Configuration data elements that are common for backup & config store to the * XML stream. * * @param out XmlSerializer instance pointing to the XML stream. * @param configuration WifiConfiguration object to be serialized. * @param encryptionUtil Instance of {@link EncryptedDataXmlUtil}. Backup/restore stores * keys unencrypted. */ public static void writeCommonElementsToXml( XmlSerializer out, WifiConfiguration configuration, @Nullable WifiConfigStoreEncryptionUtil encryptionUtil) throws XmlPullParserException, IOException { XmlUtil.writeNextValue(out, XML_TAG_CONFIG_KEY, configuration.getKey()); XmlUtil.writeNextValue(out, XML_TAG_SSID, configuration.SSID); writePreSharedKeyToXml(out, configuration, encryptionUtil); writeWepKeysToXml(out, configuration.wepKeys, encryptionUtil); XmlUtil.writeNextValue(out, XML_TAG_WEP_TX_KEY_INDEX, configuration.wepTxKeyIndex); XmlUtil.writeNextValue(out, XML_TAG_HIDDEN_SSID, configuration.hiddenSSID); XmlUtil.writeNextValue(out, XML_TAG_REQUIRE_PMF, configuration.requirePmf); XmlUtil.writeNextValue( out, XML_TAG_ALLOWED_KEY_MGMT, configuration.allowedKeyManagement.toByteArray()); XmlUtil.writeNextValue( out, XML_TAG_ALLOWED_PROTOCOLS, configuration.allowedProtocols.toByteArray()); XmlUtil.writeNextValue( out, XML_TAG_ALLOWED_AUTH_ALGOS, configuration.allowedAuthAlgorithms.toByteArray()); XmlUtil.writeNextValue( out, XML_TAG_ALLOWED_GROUP_CIPHERS, configuration.allowedGroupCiphers.toByteArray()); XmlUtil.writeNextValue( out, XML_TAG_ALLOWED_PAIRWISE_CIPHERS, configuration.allowedPairwiseCiphers.toByteArray()); XmlUtil.writeNextValue( out, XML_TAG_ALLOWED_GROUP_MGMT_CIPHERS, configuration.allowedGroupManagementCiphers.toByteArray()); XmlUtil.writeNextValue( out, XML_TAG_ALLOWED_SUITE_B_CIPHERS, configuration.allowedSuiteBCiphers.toByteArray()); XmlUtil.writeNextValue(out, XML_TAG_SHARED, configuration.shared); XmlUtil.writeNextValue(out, XML_TAG_IS_AUTO_JOIN, configuration.allowAutojoin); XmlUtil.writeNextValue(out, XML_TAG_PRIORITY, configuration.priority); XmlUtil.writeNextValue( out, XML_TAG_DELETION_PRIORITY, configuration.getDeletionPriority()); XmlUtil.writeNextValue( out, XML_TAG_NUM_REBOOTS_SINCE_LAST_USE, configuration.numRebootsSinceLastUse); XmlUtil.writeNextValue(out, XML_TAG_IS_REPEATER_ENABLED, configuration.isRepeaterEnabled()); XmlUtil.writeNextValue(out, XML_TAG_ENABLE_WIFI7, configuration.isWifi7Enabled()); writeSecurityParamsListToXml(out, configuration); XmlUtil.writeNextValue(out, XML_TAG_SEND_DHCP_HOSTNAME, configuration.isSendDhcpHostnameEnabled()); } /** * Write the Configuration data elements for backup from the provided Configuration to the * XML stream. * Note: This is a subset of the elements serialized for config store. * * @param out XmlSerializer instance pointing to the XML stream. * @param configuration WifiConfiguration object to be serialized. */ public static void writeToXmlForBackup(XmlSerializer out, WifiConfiguration configuration) throws XmlPullParserException, IOException { writeCommonElementsToXml(out, configuration, null); XmlUtil.writeNextValue(out, XML_TAG_METERED_OVERRIDE, configuration.meteredOverride); } /** * Write the Configuration data elements for config store from the provided Configuration * to the XML stream. * * @param out XmlSerializer instance pointing to the XML stream. * @param configuration WifiConfiguration object to be serialized. * @param encryptionUtil Instance of {@link EncryptedDataXmlUtil}. */ public static void writeToXmlForConfigStore( XmlSerializer out, WifiConfiguration configuration, @Nullable WifiConfigStoreEncryptionUtil encryptionUtil) throws XmlPullParserException, IOException { writeCommonElementsToXml(out, configuration, encryptionUtil); XmlUtil.writeNextValue(out, XML_TAG_IS_TRUSTED, configuration.trusted); XmlUtil.writeNextValue(out, XML_TAG_IS_RESTRICTED, configuration.restricted); XmlUtil.writeNextValue(out, XML_TAG_IS_OEM_PAID, configuration.oemPaid); XmlUtil.writeNextValue(out, XML_TAG_IS_OEM_PRIVATE, configuration.oemPrivate); XmlUtil.writeNextValue(out, XML_TAG_IS_CARRIER_MERGED, configuration.carrierMerged); XmlUtil.writeNextValue(out, XML_TAG_BSSID, configuration.BSSID); XmlUtil.writeNextValue(out, XML_TAG_STATUS, configuration.status); XmlUtil.writeNextValue(out, XML_TAG_FQDN, configuration.FQDN); XmlUtil.writeNextValue( out, XML_TAG_PROVIDER_FRIENDLY_NAME, configuration.providerFriendlyName); XmlUtil.writeNextValue( out, XML_TAG_LINKED_NETWORKS_LIST, configuration.linkedConfigurations); XmlUtil.writeNextValue( out, XML_TAG_DEFAULT_GW_MAC_ADDRESS, configuration.defaultGwMacAddress); XmlUtil.writeNextValue( out, XML_TAG_VALIDATED_INTERNET_ACCESS, configuration.validatedInternetAccess); XmlUtil.writeNextValue( out, XML_TAG_NO_INTERNET_ACCESS_EXPECTED, configuration.noInternetAccessExpected); XmlUtil.writeNextValue(out, XML_TAG_METERED_HINT, configuration.meteredHint); XmlUtil.writeNextValue(out, XML_TAG_METERED_OVERRIDE, configuration.meteredOverride); XmlUtil.writeNextValue( out, XML_TAG_USE_EXTERNAL_SCORES, configuration.useExternalScores); XmlUtil.writeNextValue(out, XML_TAG_CREATOR_UID, configuration.creatorUid); XmlUtil.writeNextValue(out, XML_TAG_CREATOR_NAME, configuration.creatorName); XmlUtil.writeNextValue(out, XML_TAG_LAST_UPDATE_UID, configuration.lastUpdateUid); XmlUtil.writeNextValue(out, XML_TAG_LAST_UPDATE_NAME, configuration.lastUpdateName); XmlUtil.writeNextValue(out, XML_TAG_LAST_CONNECT_UID, configuration.lastConnectUid); XmlUtil.writeNextValue( out, XML_TAG_IS_LEGACY_PASSPOINT_CONFIG, configuration.isLegacyPasspointConfig); XmlUtil.writeNextValue( out, XML_TAG_ROAMING_CONSORTIUM_OIS, configuration.roamingConsortiumIds); XmlUtil.writeNextValue(out, XML_TAG_RANDOMIZED_MAC_ADDRESS, configuration.getRandomizedMacAddress().toString()); XmlUtil.writeNextValue(out, XML_TAG_MAC_RANDOMIZATION_SETTING, configuration.macRandomizationSetting); XmlUtil.writeNextValue(out, XML_TAG_CARRIER_ID, configuration.carrierId); XmlUtil.writeNextValue(out, XML_TAG_IS_MOST_RECENTLY_CONNECTED, configuration.isMostRecentlyConnected); XmlUtil.writeNextValue(out, XML_TAG_SUBSCRIPTION_ID, configuration.subscriptionId); if (configuration.getSubscriptionGroup() != null) { XmlUtil.writeNextValue(out, XML_TAG_SUBSCRIPTION_GROUP, configuration.getSubscriptionGroup().toString()); } if (configuration.getBssidAllowlistInternal() != null) { XmlUtil.writeNextValue(out, XML_TAG_BSSID_ALLOW_LIST, covertMacAddressListToStringList(configuration .getBssidAllowlistInternal())); } writeDppConfigurationToXml(out, configuration, encryptionUtil); if (SdkLevel.isAtLeastV()) { writeVendorDataListToXml(out, configuration.getVendorData()); } } private static List covertMacAddressListToStringList(List macList) { List bssidList = new ArrayList<>(); for (MacAddress address : macList) { bssidList.add(address.toString()); } return bssidList; } private static List covertStringListToMacAddressList(List stringList) { List macAddressList = new ArrayList<>(); for (String address : stringList) { try { macAddressList.add(MacAddress.fromString(address)); } catch (Exception e) { Log.e(TAG, "Invalid BSSID String: " + address); } } return macAddressList; } /** * Populate wepKeys array elements only if they were non-empty in the backup data. * * @throws XmlPullParserException if parsing errors occur. */ private static void populateWepKeysFromXmlValue(Object value, String[] wepKeys) throws XmlPullParserException, IOException { String[] wepKeysInData = (String[]) value; if (wepKeysInData == null) { return; } if (wepKeysInData.length != wepKeys.length) { throw new XmlPullParserException( "Invalid Wep Keys length: " + wepKeysInData.length); } for (int i = 0; i < wepKeys.length; i++) { if (wepKeysInData[i].isEmpty()) { wepKeys[i] = null; } else { wepKeys[i] = wepKeysInData[i]; } } } private static String[] populateWepKeysFromXmlValue(XmlPullParser in, int outerTagDepth, @NonNull WifiConfigStoreEncryptionUtil encryptionUtil) throws XmlPullParserException, IOException { List wepKeyList = new ArrayList<>(); final List encryptedDataList = XmlUtil.EncryptedDataXmlUtil.parseListFromXml(in, outerTagDepth); EncryptedData emptyData = new EncryptedData(new byte[0], new byte[0]); for (int i = 0; i < encryptedDataList.size(); i++) { if (encryptedDataList.get(i).equals(emptyData)) { wepKeyList.add(null); continue; } byte[] passphraseBytes = encryptionUtil.decrypt(encryptedDataList.get(i)); if (passphraseBytes == null) { Log.wtf(TAG, "Decryption of passphraseBytes failed"); } else { wepKeyList.add(new String(passphraseBytes, StandardCharsets.UTF_8)); } } return wepKeyList.size() > 0 ? wepKeyList.toArray( new String[wepKeyList.size()]) : null; } private static SecurityParams parseSecurityParamsFromXml( XmlPullParser in, int outerTagDepth) throws XmlPullParserException, IOException { SecurityParams params = null; while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { String[] valueName = new String[1]; Object value = XmlUtil.readCurrentValue(in, valueName); String tagName = valueName[0]; if (tagName == null) { throw new XmlPullParserException("Missing value name"); } switch (tagName) { case WifiConfigurationXmlUtil.XML_TAG_SECURITY_TYPE: params = SecurityParams.createSecurityParamsBySecurityType((int) value); break; case WifiConfigurationXmlUtil.XML_TAG_IS_ENABLED: params.setEnabled((boolean) value); break; case WifiConfigurationXmlUtil.XML_TAG_SAE_IS_H2E_ONLY_MODE: if (null == params) { throw new XmlPullParserException("Missing security type."); } params.enableSaeH2eOnlyMode((boolean) value); break; case WifiConfigurationXmlUtil.XML_TAG_SAE_IS_PK_ONLY_MODE: if (null == params) { throw new XmlPullParserException("Missing security type."); } params.enableSaePkOnlyMode((boolean) value); break; case WifiConfigurationXmlUtil.XML_TAG_IS_ADDED_BY_AUTO_UPGRADE: if (null == params) { throw new XmlPullParserException("Missing security type."); } params.setIsAddedByAutoUpgrade((boolean) value); break; case WifiConfigurationXmlUtil.XML_TAG_ALLOWED_SUITE_B_CIPHERS: if (null == params) { throw new XmlPullParserException("Missing security type."); } byte[] suiteBCiphers = (byte[]) value; BitSet suiteBCiphersBitSet = BitSet.valueOf(suiteBCiphers); params.enableSuiteBCiphers( suiteBCiphersBitSet.get(WifiConfiguration.SuiteBCipher.ECDHE_ECDSA), suiteBCiphersBitSet.get(WifiConfiguration.SuiteBCipher.ECDHE_RSA)); break; } } return params; } private static byte[] readEncrytepdBytesFromXml( @Nullable WifiConfigStoreEncryptionUtil encryptionUtil, XmlPullParser in, int outerTagDepth) throws XmlPullParserException, IOException { if (encryptionUtil == null) { throw new XmlPullParserException( "Encrypted preSharedKey section not expected"); } EncryptedData encryptedData = EncryptedDataXmlUtil.parseFromXml(in, outerTagDepth + 1); return encryptionUtil.decrypt(encryptedData); } private static void parseSecurityParamsListFromXml( XmlPullParser in, int outerTagDepth, WifiConfiguration configuration) throws XmlPullParserException, IOException { List paramsList = new ArrayList<>(); while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { switch (in.getName()) { case WifiConfigurationXmlUtil.XML_TAG_SECURITY_PARAMS: SecurityParams params = parseSecurityParamsFromXml(in, outerTagDepth + 1); if (params != null) { paramsList.add(params); } break; } } if (!paramsList.isEmpty()) { configuration.setSecurityParams(paramsList); } } /** * Parses the configuration data elements from the provided XML stream to a * WifiConfiguration object. * Note: This is used for parsing both backup data and config store data. Looping through * the tags make it easy to add or remove elements in the future versions if needed. * * @param in XmlPullParser instance pointing to the XML stream. * @param outerTagDepth depth of the outer tag in the XML document. * @param shouldExpectEncryptedCredentials Whether to expect encrypted credentials or not. * @param encryptionUtil Instance of {@link EncryptedDataXmlUtil}. * @param fromSuggestion Is this WifiConfiguration created from a WifiNetworkSuggestion. * @return Pair if parsing is successful, * null otherwise. */ public static Pair parseFromXml( XmlPullParser in, int outerTagDepth, boolean shouldExpectEncryptedCredentials, @Nullable WifiConfigStoreEncryptionUtil encryptionUtil, boolean fromSuggestion) throws XmlPullParserException, IOException { WifiConfiguration configuration = new WifiConfiguration(); String configKeyInData = null; boolean macRandomizationSettingExists = false; boolean sendDhcpHostnameExists = false; byte[] dppConnector = null; byte[] dppCSign = null; byte[] dppNetAccessKey = null; // Loop through and parse out all the elements from the stream within this section. while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { if (in.getAttributeValue(null, "name") != null) { // Value elements. String[] valueName = new String[1]; Object value = XmlUtil.readCurrentValue(in, valueName); if (valueName[0] == null) { throw new XmlPullParserException("Missing value name"); } switch (valueName[0]) { case XML_TAG_CONFIG_KEY: configKeyInData = (String) value; break; case XML_TAG_SSID: configuration.SSID = (String) value; break; case XML_TAG_BSSID: configuration.BSSID = (String) value; break; case XML_TAG_PRE_SHARED_KEY: configuration.preSharedKey = (String) value; break; case XML_TAG_WEP_KEYS: populateWepKeysFromXmlValue(value, configuration.wepKeys); break; case XML_TAG_WEP_TX_KEY_INDEX: configuration.wepTxKeyIndex = (int) value; break; case XML_TAG_HIDDEN_SSID: configuration.hiddenSSID = (boolean) value; break; case XML_TAG_REQUIRE_PMF: configuration.requirePmf = (boolean) value; break; case XML_TAG_ALLOWED_KEY_MGMT: byte[] allowedKeyMgmt = (byte[]) value; configuration.allowedKeyManagement = BitSet.valueOf(allowedKeyMgmt); break; case XML_TAG_ALLOWED_PROTOCOLS: byte[] allowedProtocols = (byte[]) value; configuration.allowedProtocols = BitSet.valueOf(allowedProtocols); break; case XML_TAG_ALLOWED_AUTH_ALGOS: byte[] allowedAuthAlgorithms = (byte[]) value; configuration.allowedAuthAlgorithms = BitSet.valueOf( allowedAuthAlgorithms); break; case XML_TAG_ALLOWED_GROUP_CIPHERS: byte[] allowedGroupCiphers = (byte[]) value; configuration.allowedGroupCiphers = BitSet.valueOf(allowedGroupCiphers); break; case XML_TAG_ALLOWED_PAIRWISE_CIPHERS: byte[] allowedPairwiseCiphers = (byte[]) value; configuration.allowedPairwiseCiphers = BitSet.valueOf(allowedPairwiseCiphers); break; case XML_TAG_ALLOWED_GROUP_MGMT_CIPHERS: byte[] allowedGroupMgmtCiphers = (byte[]) value; configuration.allowedGroupManagementCiphers = BitSet.valueOf(allowedGroupMgmtCiphers); break; case XML_TAG_ALLOWED_SUITE_B_CIPHERS: byte[] allowedSuiteBCiphers = (byte[]) value; configuration.allowedSuiteBCiphers = BitSet.valueOf(allowedSuiteBCiphers); break; case XML_TAG_SHARED: configuration.shared = (boolean) value; break; case XML_TAG_STATUS: int status = (int) value; // Any network which was CURRENT before reboot needs // to be restored to ENABLED. if (status == WifiConfiguration.Status.CURRENT) { status = WifiConfiguration.Status.ENABLED; } configuration.status = status; break; case XML_TAG_FQDN: configuration.FQDN = (String) value; break; case XML_TAG_PROVIDER_FRIENDLY_NAME: configuration.providerFriendlyName = (String) value; break; case XML_TAG_LINKED_NETWORKS_LIST: configuration.linkedConfigurations = (HashMap) value; break; case XML_TAG_DEFAULT_GW_MAC_ADDRESS: configuration.defaultGwMacAddress = (String) value; break; case XML_TAG_VALIDATED_INTERNET_ACCESS: configuration.validatedInternetAccess = (boolean) value; break; case XML_TAG_NO_INTERNET_ACCESS_EXPECTED: configuration.noInternetAccessExpected = (boolean) value; break; case XML_TAG_METERED_HINT: configuration.meteredHint = (boolean) value; break; case XML_TAG_METERED_OVERRIDE: configuration.meteredOverride = (int) value; break; case XML_TAG_USE_EXTERNAL_SCORES: configuration.useExternalScores = (boolean) value; break; case XML_TAG_CREATOR_UID: configuration.creatorUid = (int) value; break; case XML_TAG_CREATOR_NAME: configuration.creatorName = (String) value; break; case XML_TAG_LAST_UPDATE_UID: configuration.lastUpdateUid = (int) value; break; case XML_TAG_LAST_UPDATE_NAME: configuration.lastUpdateName = (String) value; break; case XML_TAG_LAST_CONNECT_UID: configuration.lastConnectUid = (int) value; break; case XML_TAG_IS_LEGACY_PASSPOINT_CONFIG: configuration.isLegacyPasspointConfig = (boolean) value; break; case XML_TAG_ROAMING_CONSORTIUM_OIS: configuration.roamingConsortiumIds = (long[]) value; break; case XML_TAG_RANDOMIZED_MAC_ADDRESS: configuration.setRandomizedMacAddress( MacAddress.fromString((String) value)); break; case XML_TAG_MAC_RANDOMIZATION_SETTING: configuration.macRandomizationSetting = (int) value; macRandomizationSettingExists = true; break; case XML_TAG_SEND_DHCP_HOSTNAME: configuration.setSendDhcpHostnameEnabled((boolean) value); sendDhcpHostnameExists = true; break; case XML_TAG_CARRIER_ID: configuration.carrierId = (int) value; break; case XML_TAG_SUBSCRIPTION_ID: configuration.subscriptionId = (int) value; break; case XML_TAG_IS_AUTO_JOIN: configuration.allowAutojoin = (boolean) value; break; case XML_TAG_PRIORITY: configuration.priority = (int) value; break; case XML_TAG_DELETION_PRIORITY: configuration.setDeletionPriority((int) value); break; case XML_TAG_NUM_REBOOTS_SINCE_LAST_USE: configuration.numRebootsSinceLastUse = (int) value; break; case XML_TAG_IS_TRUSTED: configuration.trusted = (boolean) value; break; case XML_TAG_IS_OEM_PAID: configuration.oemPaid = (boolean) value; break; case XML_TAG_IS_OEM_PRIVATE: configuration.oemPrivate = (boolean) value; break; case XML_TAG_IS_MOST_RECENTLY_CONNECTED: configuration.isMostRecentlyConnected = (boolean) value; break; case XML_TAG_IS_CARRIER_MERGED: configuration.carrierMerged = (boolean) value; break; case XML_TAG_IS_RESTRICTED: configuration.restricted = (boolean) value; break; case XML_TAG_SUBSCRIPTION_GROUP: configuration.setSubscriptionGroup( ParcelUuid.fromString((String) value)); break; case XML_TAG_BSSID_ALLOW_LIST: configuration.setBssidAllowlist( covertStringListToMacAddressList((List) value)); break; case XML_TAG_IS_REPEATER_ENABLED: configuration.setRepeaterEnabled((boolean) value); break; case XML_TAG_DPP_PRIVATE_EC_KEY: configuration.setDppConfigurator((byte[]) value); break; case XML_TAG_DPP_CONNECTOR: dppConnector = (byte[]) value; break; case XML_TAG_DPP_CSIGN_KEY: dppCSign = (byte[]) value; break; case XML_TAG_DPP_NET_ACCESS_KEY: dppNetAccessKey = (byte[]) value; break; case XML_TAG_ENABLE_WIFI7: configuration.setWifi7Enabled((boolean) value); break; default: Log.w(TAG, "Ignoring unknown value name found: " + valueName[0]); break; } } else { String tagName = in.getName(); if (tagName == null) { throw new XmlPullParserException("Unexpected null tag found"); } switch (tagName) { case XML_TAG_PRE_SHARED_KEY: if (!shouldExpectEncryptedCredentials || encryptionUtil == null) { throw new XmlPullParserException( "Encrypted preSharedKey section not expected"); } EncryptedData encryptedData = EncryptedDataXmlUtil.parseFromXml(in, outerTagDepth + 1); byte[] preSharedKeyBytes = encryptionUtil.decrypt(encryptedData); if (preSharedKeyBytes == null) { Log.wtf(TAG, "Decryption of preSharedKey failed"); } else { configuration.preSharedKey = new String(preSharedKeyBytes); configuration.setEncryptedPreSharedKey( encryptedData.getEncryptedData(), encryptedData.getIv()); } break; case XML_TAG_WEP_KEYS: if (!shouldExpectEncryptedCredentials || encryptionUtil == null) { throw new XmlPullParserException( "Encrypted wepKeys section not expected"); } configuration.wepKeys = populateWepKeysFromXmlValue(in, outerTagDepth + 1, encryptionUtil); break; case XML_TAG_SECURITY_PARAMS_LIST: parseSecurityParamsListFromXml(in, outerTagDepth + 1, configuration); break; case XML_TAG_DPP_PRIVATE_EC_KEY: configuration.setDppConfigurator(readEncrytepdBytesFromXml( encryptionUtil, in, outerTagDepth)); break; case XML_TAG_DPP_CONNECTOR: dppConnector = readEncrytepdBytesFromXml(encryptionUtil, in, outerTagDepth); break; case XML_TAG_DPP_CSIGN_KEY: dppCSign = readEncrytepdBytesFromXml(encryptionUtil, in, outerTagDepth); break; case XML_TAG_DPP_NET_ACCESS_KEY: dppNetAccessKey = readEncrytepdBytesFromXml(encryptionUtil, in, outerTagDepth); break; case XML_TAG_VENDOR_DATA_LIST: if (SdkLevel.isAtLeastV()) { configuration.setVendorData( parseVendorDataListFromXml(in, outerTagDepth + 1)); } break; default: Log.w(TAG, "Ignoring unknown tag found: " + tagName); break; } } } if (!macRandomizationSettingExists) { configuration.macRandomizationSetting = WifiConfiguration.RANDOMIZATION_NONE; } if (configuration.macRandomizationSetting == WifiConfiguration.RANDOMIZATION_PERSISTENT && !fromSuggestion) { configuration.macRandomizationSetting = WifiConfiguration.RANDOMIZATION_AUTO; } if (!sendDhcpHostnameExists) { // Update legacy configs to send the DHCP hostname for secure networks only. configuration.setSendDhcpHostnameEnabled( !configuration.isSecurityType(WifiConfiguration.SECURITY_TYPE_OPEN) && !configuration.isSecurityType(WifiConfiguration.SECURITY_TYPE_OWE)); } configuration.convertLegacyFieldsToSecurityParamsIfNeeded(); configuration.setDppConnectionKeys(dppConnector, dppCSign, dppNetAccessKey); return Pair.create(configKeyInData, configuration); } } /** * Utility class to serialize and deseriaize {@link IpConfiguration} object to XML & vice versa. * This is used by both {@link com.android.server.wifi.WifiConfigStore} & * {@link com.android.server.wifi.WifiBackupRestore} modules. */ public static class IpConfigurationXmlUtil { /** * List of XML tags corresponding to IpConfiguration object elements. */ public static final String XML_TAG_IP_ASSIGNMENT = "IpAssignment"; public static final String XML_TAG_LINK_ADDRESS = "LinkAddress"; public static final String XML_TAG_LINK_PREFIX_LENGTH = "LinkPrefixLength"; public static final String XML_TAG_GATEWAY_ADDRESS = "GatewayAddress"; public static final String XML_TAG_DNS_SERVER_ADDRESSES = "DNSServers"; public static final String XML_TAG_PROXY_SETTINGS = "ProxySettings"; public static final String XML_TAG_PROXY_HOST = "ProxyHost"; public static final String XML_TAG_PROXY_PORT = "ProxyPort"; public static final String XML_TAG_PROXY_PAC_FILE = "ProxyPac"; public static final String XML_TAG_PROXY_EXCLUSION_LIST = "ProxyExclusionList"; private static List parseProxyExclusionListString( @Nullable String exclusionListString) { if (exclusionListString == null) { return Collections.emptyList(); } else { return Arrays.asList(exclusionListString.toLowerCase(Locale.ROOT).split(",")); } } private static String generateProxyExclusionListString(@NonNull String[] exclusionList) { return TextUtils.join(",", exclusionList); } /** * Write the static IP configuration data elements to XML stream. */ private static void writeStaticIpConfigurationToXml( XmlSerializer out, StaticIpConfiguration staticIpConfiguration) throws XmlPullParserException, IOException { if (staticIpConfiguration.getIpAddress() != null) { XmlUtil.writeNextValue( out, XML_TAG_LINK_ADDRESS, staticIpConfiguration.getIpAddress().getAddress().getHostAddress()); XmlUtil.writeNextValue( out, XML_TAG_LINK_PREFIX_LENGTH, staticIpConfiguration.getIpAddress().getPrefixLength()); } else { XmlUtil.writeNextValue( out, XML_TAG_LINK_ADDRESS, null); XmlUtil.writeNextValue( out, XML_TAG_LINK_PREFIX_LENGTH, null); } if (staticIpConfiguration.getGateway() != null) { XmlUtil.writeNextValue( out, XML_TAG_GATEWAY_ADDRESS, staticIpConfiguration.getGateway().getHostAddress()); } else { XmlUtil.writeNextValue( out, XML_TAG_GATEWAY_ADDRESS, null); } // Create a string array of DNS server addresses String[] dnsServers = new String[staticIpConfiguration.getDnsServers().size()]; int dnsServerIdx = 0; for (InetAddress inetAddr : staticIpConfiguration.getDnsServers()) { dnsServers[dnsServerIdx++] = inetAddr.getHostAddress(); } XmlUtil.writeNextValue( out, XML_TAG_DNS_SERVER_ADDRESSES, dnsServers); } /** * Write the IP configuration data elements from the provided Configuration to the XML * stream. * * @param out XmlSerializer instance pointing to the XML stream. * @param ipConfiguration IpConfiguration object to be serialized. */ public static void writeToXml(XmlSerializer out, IpConfiguration ipConfiguration) throws XmlPullParserException, IOException { // Write IP assignment settings XmlUtil.writeNextValue(out, XML_TAG_IP_ASSIGNMENT, ipConfiguration.getIpAssignment().toString()); switch (ipConfiguration.getIpAssignment()) { case STATIC: writeStaticIpConfigurationToXml( out, ipConfiguration.getStaticIpConfiguration()); break; case DHCP: case UNASSIGNED: break; default: Log.w(TAG, "Ignoring unknown ip assignment type: " + ipConfiguration.getIpAssignment()); break; } // Write proxy settings XmlUtil.writeNextValue( out, XML_TAG_PROXY_SETTINGS, ipConfiguration.getProxySettings().toString()); switch (ipConfiguration.getProxySettings()) { case STATIC: XmlUtil.writeNextValue( out, XML_TAG_PROXY_HOST, ipConfiguration.getHttpProxy().getHost()); XmlUtil.writeNextValue( out, XML_TAG_PROXY_PORT, ipConfiguration.getHttpProxy().getPort()); XmlUtil.writeNextValue( out, XML_TAG_PROXY_EXCLUSION_LIST, generateProxyExclusionListString( ipConfiguration.getHttpProxy().getExclusionList())); break; case PAC: XmlUtil.writeNextValue( out, XML_TAG_PROXY_PAC_FILE, ipConfiguration.getHttpProxy().getPacFileUrl().toString()); break; case NONE: case UNASSIGNED: break; default: Log.w(TAG, "Ignoring unknown proxy settings type: " + ipConfiguration.getProxySettings()); break; } } /** * Parse out the static IP configuration from the XML stream. */ private static StaticIpConfiguration parseStaticIpConfigurationFromXml(XmlPullParser in) throws XmlPullParserException, IOException { StaticIpConfiguration.Builder builder = new StaticIpConfiguration.Builder(); String linkAddressString = (String) XmlUtil.readNextValueWithName(in, XML_TAG_LINK_ADDRESS); Integer linkPrefixLength = (Integer) XmlUtil.readNextValueWithName(in, XML_TAG_LINK_PREFIX_LENGTH); if (linkAddressString != null && linkPrefixLength != null) { LinkAddress linkAddress = new LinkAddress( InetAddresses.parseNumericAddress(linkAddressString), linkPrefixLength); if (linkAddress.getAddress() instanceof Inet4Address) { builder.setIpAddress(linkAddress); } else { Log.w(TAG, "Non-IPv4 address: " + linkAddress); } } String gatewayAddressString = (String) XmlUtil.readNextValueWithName(in, XML_TAG_GATEWAY_ADDRESS); if (gatewayAddressString != null) { InetAddress gateway = InetAddresses.parseNumericAddress(gatewayAddressString); RouteInfo route = new RouteInfo(null, gateway, null, RouteInfo.RTN_UNICAST); if (route.isDefaultRoute() && route.getDestination().getAddress() instanceof Inet4Address) { builder.setGateway(gateway); } else { Log.w(TAG, "Non-IPv4 default route: " + route); } } String[] dnsServerAddressesString = (String[]) XmlUtil.readNextValueWithName(in, XML_TAG_DNS_SERVER_ADDRESSES); if (dnsServerAddressesString != null) { List dnsServerAddresses = new ArrayList<>(); for (String dnsServerAddressString : dnsServerAddressesString) { InetAddress dnsServerAddress = InetAddresses.parseNumericAddress(dnsServerAddressString); dnsServerAddresses.add(dnsServerAddress); } builder.setDnsServers(dnsServerAddresses); } return builder.build(); } /** * Parses the IP configuration data elements from the provided XML stream to an * IpConfiguration object. * * @param in XmlPullParser instance pointing to the XML stream. * @param outerTagDepth depth of the outer tag in the XML document. * @return IpConfiguration object if parsing is successful, null otherwise. */ public static IpConfiguration parseFromXml(XmlPullParser in, int outerTagDepth) throws XmlPullParserException, IOException { IpConfiguration ipConfiguration = new IpConfiguration(); // Parse out the IP assignment info first. String ipAssignmentString = (String) XmlUtil.readNextValueWithName(in, XML_TAG_IP_ASSIGNMENT); IpAssignment ipAssignment = IpAssignment.valueOf(ipAssignmentString); ipConfiguration.setIpAssignment(ipAssignment); switch (ipAssignment) { case STATIC: ipConfiguration.setStaticIpConfiguration(parseStaticIpConfigurationFromXml(in)); break; case DHCP: case UNASSIGNED: break; default: Log.w(TAG, "Ignoring unknown ip assignment type: " + ipAssignment); break; } // Parse out the proxy settings next. String proxySettingsString = (String) XmlUtil.readNextValueWithName(in, XML_TAG_PROXY_SETTINGS); ProxySettings proxySettings = ProxySettings.valueOf(proxySettingsString); ipConfiguration.setProxySettings(proxySettings); switch (proxySettings) { case STATIC: String proxyHost = (String) XmlUtil.readNextValueWithName(in, XML_TAG_PROXY_HOST); int proxyPort = (int) XmlUtil.readNextValueWithName(in, XML_TAG_PROXY_PORT); String proxyExclusionList = (String) XmlUtil.readNextValueWithName( in, XML_TAG_PROXY_EXCLUSION_LIST); ipConfiguration.setHttpProxy( ProxyInfo.buildDirectProxy( proxyHost, proxyPort, parseProxyExclusionListString(proxyExclusionList))); break; case PAC: String proxyPacFile = (String) XmlUtil.readNextValueWithName(in, XML_TAG_PROXY_PAC_FILE); ipConfiguration.setHttpProxy( ProxyInfo.buildPacProxy(Uri.parse(proxyPacFile))); break; case NONE: case UNASSIGNED: break; default: Log.w(TAG, "Ignoring unknown proxy settings type: " + proxySettings); break; } return ipConfiguration; } } /** * Utility class to serialize and deserialize {@link NetworkSelectionStatus} object to XML & * vice versa. This is used by {@link com.android.server.wifi.WifiConfigStore} module. */ public static class NetworkSelectionStatusXmlUtil { /** * List of XML tags corresponding to NetworkSelectionStatus object elements. */ public static final String XML_TAG_SELECTION_STATUS = "SelectionStatus"; public static final String XML_TAG_DISABLE_REASON = "DisableReason"; public static final String XML_TAG_CONNECT_CHOICE = "ConnectChoice"; public static final String XML_TAG_HAS_EVER_CONNECTED = "HasEverConnected"; public static final String XML_TAG_IS_CAPTIVE_PORTAL_NEVER_DETECTED = "CaptivePortalNeverDetected"; public static final String XML_TAG_HAS_EVER_VALIDATED_INTERNET_ACCESS = "HasEverValidatedInternetAccess"; public static final String XML_TAG_CONNECT_CHOICE_RSSI = "ConnectChoiceRssi"; /** * Write the NetworkSelectionStatus data elements from the provided status to the XML * stream. * * @param out XmlSerializer instance pointing to the XML stream. * @param selectionStatus NetworkSelectionStatus object to be serialized. */ public static void writeToXml(XmlSerializer out, NetworkSelectionStatus selectionStatus) throws XmlPullParserException, IOException { XmlUtil.writeNextValue( out, XML_TAG_SELECTION_STATUS, selectionStatus.getNetworkStatusString()); XmlUtil.writeNextValue( out, XML_TAG_DISABLE_REASON, selectionStatus.getNetworkSelectionDisableReasonString()); XmlUtil.writeNextValue(out, XML_TAG_CONNECT_CHOICE, selectionStatus.getConnectChoice()); XmlUtil.writeNextValue(out, XML_TAG_CONNECT_CHOICE_RSSI, selectionStatus.getConnectChoiceRssi()); XmlUtil.writeNextValue( out, XML_TAG_HAS_EVER_CONNECTED, selectionStatus.hasEverConnected()); XmlUtil.writeNextValue(out, XML_TAG_IS_CAPTIVE_PORTAL_NEVER_DETECTED, selectionStatus.hasNeverDetectedCaptivePortal()); XmlUtil.writeNextValue(out, XML_TAG_HAS_EVER_VALIDATED_INTERNET_ACCESS, selectionStatus.hasEverValidatedInternetAccess()); } /** * Parses the NetworkSelectionStatus data elements from the provided XML stream to a * NetworkSelectionStatus object. * * @param in XmlPullParser instance pointing to the XML stream. * @param outerTagDepth depth of the outer tag in the XML document. * @return NetworkSelectionStatus object if parsing is successful, null otherwise. */ public static NetworkSelectionStatus parseFromXml(XmlPullParser in, int outerTagDepth) throws XmlPullParserException, IOException { NetworkSelectionStatus selectionStatus = new NetworkSelectionStatus(); String statusString = ""; String disableReasonString = ""; // Initialize hasNeverDetectedCaptivePortal to "false" for upgrading legacy configs // which do not have the XML_TAG_IS_CAPTIVE_PORTAL_NEVER_DETECTED tag. selectionStatus.setHasNeverDetectedCaptivePortal(false); // Initialize hasEverValidatedInternetAccess to "true" for existing configs which don't // have any value stored. selectionStatus.setHasEverValidatedInternetAccess(true); // Loop through and parse out all the elements from the stream within this section. while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { String[] valueName = new String[1]; Object value = XmlUtil.readCurrentValue(in, valueName); if (valueName[0] == null) { throw new XmlPullParserException("Missing value name"); } switch (valueName[0]) { case XML_TAG_SELECTION_STATUS: statusString = (String) value; break; case XML_TAG_DISABLE_REASON: disableReasonString = (String) value; break; case XML_TAG_CONNECT_CHOICE: selectionStatus.setConnectChoice((String) value); break; case XML_TAG_CONNECT_CHOICE_RSSI: selectionStatus.setConnectChoiceRssi((int) value); break; case XML_TAG_HAS_EVER_CONNECTED: selectionStatus.setHasEverConnected((boolean) value); break; case XML_TAG_IS_CAPTIVE_PORTAL_NEVER_DETECTED: selectionStatus.setHasNeverDetectedCaptivePortal((boolean) value); break; case XML_TAG_HAS_EVER_VALIDATED_INTERNET_ACCESS: selectionStatus.setHasEverValidatedInternetAccess((boolean) value); break; default: Log.w(TAG, "Ignoring unknown value name found: " + valueName[0]); break; } } // Now figure out the network selection status codes from |selectionStatusString| & // |disableReasonString|. int status = Arrays.asList(NetworkSelectionStatus.QUALITY_NETWORK_SELECTION_STATUS) .indexOf(statusString); int disableReason = NetworkSelectionStatus.getDisableReasonByString(disableReasonString); // If either of the above codes are invalid or if the network was temporarily disabled // (blacklisted), restore the status as enabled. We don't want to persist blacklists // across reboots. if (status == -1 || disableReason == -1 || status == NetworkSelectionStatus.NETWORK_SELECTION_TEMPORARY_DISABLED) { status = NetworkSelectionStatus.NETWORK_SELECTION_ENABLED; disableReason = NetworkSelectionStatus.DISABLED_NONE; } selectionStatus.setNetworkSelectionStatus(status); selectionStatus.setNetworkSelectionDisableReason(disableReason); if (status == NetworkSelectionStatus.NETWORK_SELECTION_PERMANENTLY_DISABLED) { // Make the counter non-zero so that logging code works properly selectionStatus.setDisableReasonCounter(disableReason, 1); } return selectionStatus; } } /** * Utility class to serialize and deseriaize {@link WifiEnterpriseConfig} object to XML & * vice versa. This is used by {@link com.android.server.wifi.WifiConfigStore} module. */ public static class WifiEnterpriseConfigXmlUtil { /** * List of XML tags corresponding to WifiEnterpriseConfig object elements. */ public static final String XML_TAG_IDENTITY = "Identity"; public static final String XML_TAG_ANON_IDENTITY = "AnonIdentity"; public static final String XML_TAG_PASSWORD = "Password"; public static final String XML_TAG_CLIENT_CERT = "ClientCert"; public static final String XML_TAG_CA_CERT = "CaCert"; public static final String XML_TAG_SUBJECT_MATCH = "SubjectMatch"; public static final String XML_TAG_ENGINE = "Engine"; public static final String XML_TAG_ENGINE_ID = "EngineId"; public static final String XML_TAG_PRIVATE_KEY_ID = "PrivateKeyId"; public static final String XML_TAG_ALT_SUBJECT_MATCH = "AltSubjectMatch"; public static final String XML_TAG_DOM_SUFFIX_MATCH = "DomSuffixMatch"; public static final String XML_TAG_CA_PATH = "CaPath"; public static final String XML_TAG_EAP_METHOD = "EapMethod"; public static final String XML_TAG_PHASE2_METHOD = "Phase2Method"; public static final String XML_TAG_PLMN = "PLMN"; public static final String XML_TAG_REALM = "Realm"; public static final String XML_TAG_OCSP = "Ocsp"; public static final String XML_TAG_WAPI_CERT_SUITE = "WapiCertSuite"; public static final String XML_TAG_APP_INSTALLED_ROOT_CA_CERT = "AppInstalledRootCaCert"; public static final String XML_TAG_APP_INSTALLED_PRIVATE_KEY = "AppInstalledPrivateKey"; public static final String XML_TAG_KEYCHAIN_KEY_ALIAS = "KeyChainAlias"; public static final String XML_TAG_DECORATED_IDENTITY_PREFIX = "DecoratedIdentityPrefix"; public static final String XML_TAG_TRUST_ON_FIRST_USE = "TrustOnFirstUse"; public static final String XML_TAG_USER_APPROVE_NO_CA_CERT = "UserApproveNoCaCert"; public static final String XML_TAG_MINIMUM_TLS_VERSION = "MinimumTlsVersion"; public static final String XML_TAG_TOFU_DIALOG_STATE = "TofuDialogState"; public static final String XML_TAG_TOFU_CONNECTION_STATE = "TofuConnectionState"; /** * Write password key to the XML stream. * * If encryptionUtil is null or if encryption fails for some reason, the password is stored * in plaintext, else the encrypted psk is stored. */ private static void writePasswordToXml( XmlSerializer out, String password, @Nullable WifiConfigStoreEncryptionUtil encryptionUtil) throws XmlPullParserException, IOException { EncryptedData encryptedData = null; if (encryptionUtil != null) { if (password != null) { encryptedData = encryptionUtil.encrypt(password.getBytes()); if (encryptedData == null) { // We silently fail encryption failures! Log.wtf(TAG, "Encryption of password failed"); } } } if (encryptedData != null) { XmlUtil.writeNextSectionStart(out, XML_TAG_PASSWORD); EncryptedDataXmlUtil.writeToXml(out, encryptedData); XmlUtil.writeNextSectionEnd(out, XML_TAG_PASSWORD); } else { XmlUtil.writeNextValue(out, XML_TAG_PASSWORD, password); } } /** * Write the WifiEnterpriseConfig data elements from the provided config to the XML * stream. * * @param out XmlSerializer instance pointing to the XML stream. * @param enterpriseConfig WifiEnterpriseConfig object to be serialized. * @param encryptionUtil Instance of {@link EncryptedDataXmlUtil}. */ public static void writeToXml(XmlSerializer out, WifiEnterpriseConfig enterpriseConfig, @Nullable WifiConfigStoreEncryptionUtil encryptionUtil) throws XmlPullParserException, IOException { XmlUtil.writeNextValue(out, XML_TAG_IDENTITY, enterpriseConfig.getFieldValue(WifiEnterpriseConfig.IDENTITY_KEY)); XmlUtil.writeNextValue(out, XML_TAG_ANON_IDENTITY, enterpriseConfig.getFieldValue(WifiEnterpriseConfig.ANON_IDENTITY_KEY)); writePasswordToXml( out, enterpriseConfig.getFieldValue(WifiEnterpriseConfig.PASSWORD_KEY), encryptionUtil); XmlUtil.writeNextValue(out, XML_TAG_CLIENT_CERT, enterpriseConfig.getFieldValue(WifiEnterpriseConfig.CLIENT_CERT_KEY)); XmlUtil.writeNextValue(out, XML_TAG_CA_CERT, enterpriseConfig.getFieldValue(WifiEnterpriseConfig.CA_CERT_KEY)); XmlUtil.writeNextValue(out, XML_TAG_SUBJECT_MATCH, enterpriseConfig.getFieldValue(WifiEnterpriseConfig.SUBJECT_MATCH_KEY)); XmlUtil.writeNextValue(out, XML_TAG_ENGINE, enterpriseConfig.getFieldValue(WifiEnterpriseConfig.ENGINE_KEY)); XmlUtil.writeNextValue(out, XML_TAG_ENGINE_ID, enterpriseConfig.getFieldValue(WifiEnterpriseConfig.ENGINE_ID_KEY)); XmlUtil.writeNextValue(out, XML_TAG_PRIVATE_KEY_ID, enterpriseConfig.getFieldValue(WifiEnterpriseConfig.PRIVATE_KEY_ID_KEY)); XmlUtil.writeNextValue(out, XML_TAG_ALT_SUBJECT_MATCH, enterpriseConfig.getFieldValue(WifiEnterpriseConfig.ALTSUBJECT_MATCH_KEY)); XmlUtil.writeNextValue(out, XML_TAG_DOM_SUFFIX_MATCH, enterpriseConfig.getFieldValue(WifiEnterpriseConfig.DOM_SUFFIX_MATCH_KEY)); XmlUtil.writeNextValue(out, XML_TAG_CA_PATH, enterpriseConfig.getFieldValue(WifiEnterpriseConfig.CA_PATH_KEY)); XmlUtil.writeNextValue(out, XML_TAG_EAP_METHOD, enterpriseConfig.getEapMethod()); XmlUtil.writeNextValue(out, XML_TAG_PHASE2_METHOD, enterpriseConfig.getPhase2Method()); XmlUtil.writeNextValue(out, XML_TAG_PLMN, enterpriseConfig.getPlmn()); XmlUtil.writeNextValue(out, XML_TAG_REALM, enterpriseConfig.getRealm()); XmlUtil.writeNextValue(out, XML_TAG_OCSP, enterpriseConfig.getOcsp()); XmlUtil.writeNextValue(out, XML_TAG_WAPI_CERT_SUITE, enterpriseConfig.getWapiCertSuite()); XmlUtil.writeNextValue(out, XML_TAG_APP_INSTALLED_ROOT_CA_CERT, enterpriseConfig.isAppInstalledCaCert()); XmlUtil.writeNextValue(out, XML_TAG_APP_INSTALLED_PRIVATE_KEY, enterpriseConfig.isAppInstalledDeviceKeyAndCert()); XmlUtil.writeNextValue(out, XML_TAG_KEYCHAIN_KEY_ALIAS, enterpriseConfig.getClientKeyPairAliasInternal()); if (SdkLevel.isAtLeastS()) { XmlUtil.writeNextValue(out, XML_TAG_DECORATED_IDENTITY_PREFIX, enterpriseConfig.getDecoratedIdentityPrefix()); } XmlUtil.writeNextValue(out, XML_TAG_TRUST_ON_FIRST_USE, enterpriseConfig.isTrustOnFirstUseEnabled()); XmlUtil.writeNextValue(out, XML_TAG_USER_APPROVE_NO_CA_CERT, enterpriseConfig.isUserApproveNoCaCert()); XmlUtil.writeNextValue(out, XML_TAG_MINIMUM_TLS_VERSION, enterpriseConfig.getMinimumTlsVersion()); XmlUtil.writeNextValue(out, XML_TAG_TOFU_DIALOG_STATE, enterpriseConfig.getTofuDialogState()); XmlUtil.writeNextValue(out, XML_TAG_TOFU_CONNECTION_STATE, enterpriseConfig.getTofuConnectionState()); } /** * Parses the data elements from the provided XML stream to a WifiEnterpriseConfig object. * * @param in XmlPullParser instance pointing to the XML stream. * @param outerTagDepth depth of the outer tag in the XML document. * @param shouldExpectEncryptedCredentials Whether to expect encrypted credentials or not. * @param encryptionUtil Instance of {@link EncryptedDataXmlUtil}. * @return WifiEnterpriseConfig object if parsing is successful, null otherwise. */ public static WifiEnterpriseConfig parseFromXml(XmlPullParser in, int outerTagDepth, boolean shouldExpectEncryptedCredentials, @Nullable WifiConfigStoreEncryptionUtil encryptionUtil) throws XmlPullParserException, IOException { WifiEnterpriseConfig enterpriseConfig = new WifiEnterpriseConfig(); // Loop through and parse out all the elements from the stream within this section. while (XmlUtilHelper.nextElementWithin(in, outerTagDepth)) { if (in.getAttributeValue(null, "name") != null) { // Value elements. String[] valueName = new String[1]; Object value = XmlUtil.readCurrentValue(in, valueName); if (valueName[0] == null) { throw new XmlPullParserException("Missing value name"); } switch (valueName[0]) { case XML_TAG_IDENTITY: enterpriseConfig.setFieldValue( WifiEnterpriseConfig.IDENTITY_KEY, (String) value); break; case XML_TAG_ANON_IDENTITY: enterpriseConfig.setFieldValue( WifiEnterpriseConfig.ANON_IDENTITY_KEY, (String) value); break; case XML_TAG_PASSWORD: enterpriseConfig.setFieldValue( WifiEnterpriseConfig.PASSWORD_KEY, (String) value); if (shouldExpectEncryptedCredentials && !TextUtils.isEmpty(enterpriseConfig.getFieldValue( WifiEnterpriseConfig.PASSWORD_KEY))) { // Indicates that encryption of password failed when it was last // written. Log.e(TAG, "password value not expected"); } break; case XML_TAG_CLIENT_CERT: enterpriseConfig.setFieldValue( WifiEnterpriseConfig.CLIENT_CERT_KEY, (String) value); break; case XML_TAG_CA_CERT: enterpriseConfig.setFieldValue( WifiEnterpriseConfig.CA_CERT_KEY, (String) value); break; case XML_TAG_SUBJECT_MATCH: enterpriseConfig.setFieldValue( WifiEnterpriseConfig.SUBJECT_MATCH_KEY, (String) value); break; case XML_TAG_ENGINE: enterpriseConfig.setFieldValue( WifiEnterpriseConfig.ENGINE_KEY, (String) value); break; case XML_TAG_ENGINE_ID: enterpriseConfig.setFieldValue( WifiEnterpriseConfig.ENGINE_ID_KEY, (String) value); break; case XML_TAG_PRIVATE_KEY_ID: enterpriseConfig.setFieldValue( WifiEnterpriseConfig.PRIVATE_KEY_ID_KEY, (String) value); break; case XML_TAG_ALT_SUBJECT_MATCH: enterpriseConfig.setFieldValue( WifiEnterpriseConfig.ALTSUBJECT_MATCH_KEY, (String) value); break; case XML_TAG_DOM_SUFFIX_MATCH: enterpriseConfig.setFieldValue( WifiEnterpriseConfig.DOM_SUFFIX_MATCH_KEY, (String) value); break; case XML_TAG_CA_PATH: enterpriseConfig.setFieldValue( WifiEnterpriseConfig.CA_PATH_KEY, (String) value); break; case XML_TAG_OCSP: enterpriseConfig.setOcsp((int) value); break; case XML_TAG_EAP_METHOD: enterpriseConfig.setEapMethod((int) value); break; case XML_TAG_PHASE2_METHOD: enterpriseConfig.setPhase2Method((int) value); break; case XML_TAG_PLMN: enterpriseConfig.setPlmn((String) value); break; case XML_TAG_REALM: enterpriseConfig.setRealm((String) value); break; case XML_TAG_WAPI_CERT_SUITE: enterpriseConfig.setWapiCertSuite((String) value); break; case XML_TAG_APP_INSTALLED_ROOT_CA_CERT: enterpriseConfig.initIsAppInstalledCaCert((boolean) value); break; case XML_TAG_APP_INSTALLED_PRIVATE_KEY: enterpriseConfig.initIsAppInstalledDeviceKeyAndCert((boolean) value); break; case XML_TAG_KEYCHAIN_KEY_ALIAS: if (SdkLevel.isAtLeastS()) { enterpriseConfig.setClientKeyPairAlias((String) value); } break; case XML_TAG_DECORATED_IDENTITY_PREFIX: if (SdkLevel.isAtLeastS()) { enterpriseConfig.setDecoratedIdentityPrefix((String) value); } break; case XML_TAG_TRUST_ON_FIRST_USE: enterpriseConfig.enableTrustOnFirstUse((boolean) value); break; case XML_TAG_USER_APPROVE_NO_CA_CERT: enterpriseConfig.setUserApproveNoCaCert((boolean) value); break; case XML_TAG_MINIMUM_TLS_VERSION: enterpriseConfig.setMinimumTlsVersion((int) value); break; case XML_TAG_TOFU_DIALOG_STATE: enterpriseConfig.setTofuDialogState((int) value); break; case XML_TAG_TOFU_CONNECTION_STATE: enterpriseConfig.setTofuConnectionState((int) value); break; default: Log.w(TAG, "Ignoring unknown value name found: " + valueName[0]); break; } } else { String tagName = in.getName(); if (tagName == null) { throw new XmlPullParserException("Unexpected null tag found"); } switch (tagName) { case XML_TAG_PASSWORD: if (!shouldExpectEncryptedCredentials || encryptionUtil == null) { throw new XmlPullParserException( "encrypted password section not expected"); } EncryptedData encryptedData = EncryptedDataXmlUtil.parseFromXml(in, outerTagDepth + 1); byte[] passwordBytes = encryptionUtil.decrypt(encryptedData); if (passwordBytes == null) { Log.wtf(TAG, "Decryption of password failed"); } else { enterpriseConfig.setFieldValue( WifiEnterpriseConfig.PASSWORD_KEY, new String(passwordBytes)); } break; default: Log.w(TAG, "Ignoring unknown tag name found: " + tagName); break; } } } return enterpriseConfig; } } /** * Utility class to serialize and deseriaize {@link EncryptedData} object to XML & * vice versa. This is used by {@link com.android.server.wifi.WifiConfigStore} module. */ public static class EncryptedDataXmlUtil { /** * List of XML tags corresponding to EncryptedData object elements. */ private static final String XML_TAG_ENCRYPTED_DATA = "EncryptedData"; private static final String XML_TAG_IV = "IV"; /** * Write the NetworkSelectionStatus data elements from the provided status to the XML * stream. * * @param out XmlSerializer instance pointing to the XML stream. * @param encryptedData EncryptedData object to be serialized. */ public static void writeToXml(XmlSerializer out, EncryptedData encryptedData) throws XmlPullParserException, IOException { XmlUtil.writeNextValue( out, XML_TAG_ENCRYPTED_DATA, encryptedData.getEncryptedData()); XmlUtil.writeNextValue(out, XML_TAG_IV, encryptedData.getIv()); } /** * Parses the EncryptedData data elements from the provided XML stream to a * EncryptedData object. * * @param in XmlPullParser instance pointing to the XML stream. * @param outerTagDepth depth of the outer tag in the XML document. * @return EncryptedData object if parsing is successful, null otherwise. */ public static EncryptedData parseFromXml(XmlPullParser in, int outerTagDepth) throws XmlPullParserException, IOException { byte[] encryptedData = null; byte[] iv = null; // Loop through and parse out all the elements from the stream within this section. while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { String[] valueName = new String[1]; Object value = XmlUtil.readCurrentValue(in, valueName); if (valueName[0] == null) { throw new XmlPullParserException("Missing value name"); } switch (valueName[0]) { case XML_TAG_ENCRYPTED_DATA: encryptedData = (byte[]) value; break; case XML_TAG_IV: iv = (byte[]) value; break; default: Log.e(TAG, "Unknown value name found: " + valueName[0]); break; } } return new EncryptedData(encryptedData, iv); } /** * Parses the EncryptedData data elements arrays from the provided XML stream to a list of * EncryptedData object. * * @param in XmlPullParser instance pointing to the XML stream. * @param outerTagDepth depth of the outer tag in the XML document. * @return List of encryptedData object if parsing is successful, empty otherwise. */ public static @NonNull List parseListFromXml(XmlPullParser in, int outerTagDepth) throws XmlPullParserException, IOException { List encryptedDataList = new ArrayList<>(); // Loop through and parse out all the elements from the stream within this section. while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { if (in.getAttributeValue(null, "name") != null) { String[] valueName = new String[1]; Object value = XmlUtil.readCurrentValue(in, valueName); if (valueName[0] == null) { throw new XmlPullParserException("Missing value name"); } byte[] encryptedData; byte[] iv; if (XML_TAG_ENCRYPTED_DATA.equals(valueName[0])) { encryptedData = (byte[]) value; if (!XmlUtil.isNextSectionEnd(in, outerTagDepth) && in.getAttributeValue( null, "name") != null) { value = XmlUtil.readCurrentValue(in, valueName); if (valueName[0] == null) { throw new XmlPullParserException("Missing value name"); } if (XML_TAG_IV.equals(valueName[0])) { iv = (byte[]) value; encryptedDataList.add(new EncryptedData(encryptedData, iv)); } } } } } return encryptedDataList; } } public static boolean nextElementWithin(XmlPullParser parser, int outerDepth) throws IOException, XmlPullParserException { return XmlUtilHelper.nextElementWithin(parser, outerDepth); } /** * Utility class to serialize and deseriaize {@link SoftApConfiguration} object to XML * & vice versa. This is used by both {@link com.android.server.wifi.SoftApStore} modules. */ public static class SoftApConfigurationXmlUtil { /** * List of XML tags corresponding to SoftApConfiguration object elements. */ public static final String XML_TAG_CLIENT_MACADDRESS = "ClientMacAddress"; public static final String XML_TAG_BAND_CHANNEL = "BandChannel"; public static final String XML_TAG_SSID = "SSID"; // Use XML_TAG_WIFI_SSID instead public static final String XML_TAG_WIFI_SSID = "WifiSsid"; public static final String XML_TAG_BSSID = "Bssid"; public static final String XML_TAG_BAND = "Band"; public static final String XML_TAG_CHANNEL = "Channel"; public static final String XML_TAG_HIDDEN_SSID = "HiddenSSID"; public static final String XML_TAG_SECURITY_TYPE = "SecurityType"; public static final String XML_TAG_WPA2_PASSPHRASE = "Wpa2Passphrase"; public static final String XML_TAG_AP_BAND = "ApBand"; public static final String XML_TAG_PASSPHRASE = "Passphrase"; public static final String XML_TAG_MAX_NUMBER_OF_CLIENTS = "MaxNumberOfClients"; public static final String XML_TAG_AUTO_SHUTDOWN_ENABLED = "AutoShutdownEnabled"; public static final String XML_TAG_SHUTDOWN_TIMEOUT_MILLIS = "ShutdownTimeoutMillis"; public static final String XML_TAG_CLIENT_CONTROL_BY_USER = "ClientControlByUser"; public static final String XML_TAG_BLOCKED_CLIENT_LIST = "BlockedClientList"; public static final String XML_TAG_ALLOWED_CLIENT_LIST = "AllowedClientList"; public static final String XML_TAG_BRIDGED_MODE_OPPORTUNISTIC_SHUTDOWN_ENABLED = "BridgedModeOpportunisticShutdownEnabled"; public static final String XML_TAG_MAC_RAMDOMIZATION_SETTING = "MacRandomizationSetting"; public static final String XML_TAG_BAND_CHANNEL_MAP = "BandChannelMap"; public static final String XML_TAG_80211_AX_ENABLED = "80211axEnabled"; public static final String XML_TAG_80211_BE_ENABLED = "80211beEnabled"; public static final String XML_TAG_USER_CONFIGURATION = "UserConfiguration"; public static final String XML_TAG_BRIDGED_MODE_OPPORTUNISTIC_SHUTDOWN_TIMEOUT_MILLIS = "BridgedModeOpportunisticShutdownTimeoutMillis"; public static final String XML_TAG_VENDOR_ELEMENT = "VendorElement"; public static final String XML_TAG_VENDOR_ELEMENTS = "VendorElements"; public static final String XML_TAG_PERSISTENT_RANDOMIZED_MAC_ADDRESS = "PersistentRandomizedMacAddress"; /** * Parses the client list from the provided XML stream to a ArrayList object. * * @param in XmlPullParser instance pointing to the XML stream. * @param outerTagDepth depth of the outer tag in the XML document. * @return ArrayList object if parsing is successful, null otherwise. */ public static List parseClientListFromXml(XmlPullParser in, int outerTagDepth) throws XmlPullParserException, IOException, IllegalArgumentException { List clientList = new ArrayList<>(); // Loop through and parse out all the elements from the stream within this section. while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { String[] valueName = new String[1]; Object value = XmlUtil.readCurrentValue(in, valueName); if (valueName[0] == null) { throw new XmlPullParserException("Missing value name"); } switch (valueName[0]) { case XML_TAG_CLIENT_MACADDRESS: MacAddress client = MacAddress.fromString((String) value); clientList.add(client); break; default: Log.e(TAG, "Unknown value name found: " + valueName[0]); break; } } return clientList; } /** * Write the SoftApConfiguration client control list data elements * from the provided list to the XML stream. * * @param out XmlSerializer instance pointing to the XML stream. * @param clientList Client list object to be serialized. */ public static void writeClientListToXml(XmlSerializer out, List clientList) throws XmlPullParserException, IOException { for (MacAddress mac: clientList) { XmlUtil.writeNextValue(out, XML_TAG_CLIENT_MACADDRESS, mac.toString()); } } /** * Write the SoftApConfiguration vendor elements list information elements to the XML * * @param out XmlSerializer instance pointing to the XML stream * @param elements Vendor elements list */ public static void writeVendorElementsSetToXml( XmlSerializer out, List elements) throws XmlPullParserException, IOException { for (ScanResult.InformationElement e : elements) { XmlUtil.writeNextValue(out, XML_TAG_VENDOR_ELEMENT, InformationElementUtil.toHexString(e)); } } /** * Parses the vendor elements from the provided XML stream to HashSet object. * * @param in XmlPullParser instance pointing to the XML stream * @param outerTagDepth depth of the outer tag in the XML document * @return HashSet object if parsing is successful, empty set otherwise */ public static List parseVendorElementsFromXml( XmlPullParser in, int outerTagDepth) throws XmlPullParserException, IOException, IllegalArgumentException { List elements = new ArrayList<>(); while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { String[] valueName = new String[1]; Object value = XmlUtil.readCurrentValue(in, valueName); if (valueName[0] == null) { throw new XmlPullParserException("Missing value name"); } if (XML_TAG_VENDOR_ELEMENT.equals(valueName[0])) { ScanResult.InformationElement[] data = InformationElementUtil.parseInformationElements((String) value); elements.addAll(Arrays.asList(data)); } else { Log.e(TAG, "Unknown value name found: " + valueName[0]); } } return elements; } /** * Parses the band and channel from the provided XML stream to a SparseIntArray object. * * @param in XmlPullParser instance pointing to the XML stream. * @param outerTagDepth depth of the outer tag in the XML document. * @return SparseIntArray object if parsing is successful, null otherwise. */ public static SparseIntArray parseChannelsFromXml(XmlPullParser in, int outerTagDepth) throws XmlPullParserException, IOException, IllegalArgumentException { SparseIntArray channels = new SparseIntArray(); while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { int band = ApConfigUtil.INVALID_VALUE_FOR_BAND_OR_CHANNEL; int channel = ApConfigUtil.INVALID_VALUE_FOR_BAND_OR_CHANNEL; switch (in.getName()) { case XML_TAG_BAND_CHANNEL: while (!XmlUtil.isNextSectionEnd(in, outerTagDepth + 1)) { String[] valueName = new String[1]; Object value = XmlUtil.readCurrentValue(in, valueName); if (valueName[0] == null) { throw new XmlPullParserException("Missing value name"); } switch (valueName[0]) { case XML_TAG_BAND: band = (int) value; break; case XML_TAG_CHANNEL: channel = (int) value; break; default: Log.e(TAG, "Unknown value name found: " + valueName[0]); break; } } channels.put(band, channel); break; } } return channels; } /** * Write the SoftApConfiguration channels data elements * from the provided SparseIntArray to the XML stream. * * @param out XmlSerializer instance pointing to the XML stream. * @param channels SparseIntArray, which includes bands and channels, to be serialized. */ public static void writeChannelsToXml(XmlSerializer out, SparseIntArray channels) throws XmlPullParserException, IOException { for (int i = 0; i < channels.size(); i++) { XmlUtil.writeNextSectionStart(out, XML_TAG_BAND_CHANNEL); XmlUtil.writeNextValue(out, XML_TAG_BAND, channels.keyAt(i)); XmlUtil.writeNextValue(out, XML_TAG_CHANNEL, channels.valueAt(i)); XmlUtil.writeNextSectionEnd(out, XML_TAG_BAND_CHANNEL); } } /** * Write the SoftApConfiguration data elements to the XML stream. * * @param out XmlSerializer instance pointing to the XML stream. * @param softApConfig configuration of the Soft AP. */ public static void writeSoftApConfigurationToXml(@NonNull XmlSerializer out, @NonNull SoftApConfiguration softApConfig, WifiConfigStoreEncryptionUtil encryptionUtil) throws XmlPullParserException, IOException { if (softApConfig.getWifiSsid() != null) { XmlUtil.writeNextValue(out, XML_TAG_WIFI_SSID, softApConfig.getWifiSsid().toString()); } if (softApConfig.getBssid() != null) { XmlUtil.writeNextValue(out, XML_TAG_BSSID, softApConfig.getBssid().toString()); } if (!SdkLevel.isAtLeastS()) { // Band and channel change to store in Tag:BandChannelMap from S. XmlUtil.writeNextValue(out, XML_TAG_AP_BAND, softApConfig.getBand()); XmlUtil.writeNextValue(out, XML_TAG_CHANNEL, softApConfig.getChannel()); } XmlUtil.writeNextValue(out, XML_TAG_HIDDEN_SSID, softApConfig.isHiddenSsid()); XmlUtil.writeNextValue(out, XML_TAG_SECURITY_TYPE, softApConfig.getSecurityType()); if (!ApConfigUtil.isNonPasswordAP(softApConfig.getSecurityType())) { XmlUtil.writeSoftApPassphraseToXml(out, softApConfig.getPassphrase(), encryptionUtil); } XmlUtil.writeNextValue(out, XML_TAG_MAX_NUMBER_OF_CLIENTS, softApConfig.getMaxNumberOfClients()); XmlUtil.writeNextValue(out, XML_TAG_CLIENT_CONTROL_BY_USER, softApConfig.isClientControlByUserEnabled()); XmlUtil.writeNextValue(out, XML_TAG_AUTO_SHUTDOWN_ENABLED, softApConfig.isAutoShutdownEnabled()); XmlUtil.writeNextValue(out, XML_TAG_SHUTDOWN_TIMEOUT_MILLIS, softApConfig.getShutdownTimeoutMillis()); XmlUtil.writeNextSectionStart(out, XML_TAG_BLOCKED_CLIENT_LIST); XmlUtil.SoftApConfigurationXmlUtil.writeClientListToXml(out, softApConfig.getBlockedClientList()); XmlUtil.writeNextSectionEnd(out, XML_TAG_BLOCKED_CLIENT_LIST); XmlUtil.writeNextSectionStart(out, XML_TAG_ALLOWED_CLIENT_LIST); XmlUtil.SoftApConfigurationXmlUtil.writeClientListToXml(out, softApConfig.getAllowedClientList()); XmlUtil.writeNextSectionEnd(out, XML_TAG_ALLOWED_CLIENT_LIST); if (SdkLevel.isAtLeastS()) { XmlUtil.writeNextValue(out, XML_TAG_BRIDGED_MODE_OPPORTUNISTIC_SHUTDOWN_ENABLED, softApConfig.isBridgedModeOpportunisticShutdownEnabled()); XmlUtil.writeNextValue(out, XML_TAG_MAC_RAMDOMIZATION_SETTING, softApConfig.getMacRandomizationSetting()); XmlUtil.writeNextSectionStart(out, XML_TAG_BAND_CHANNEL_MAP); XmlUtil.SoftApConfigurationXmlUtil.writeChannelsToXml(out, softApConfig.getChannels()); XmlUtil.writeNextSectionEnd(out, XML_TAG_BAND_CHANNEL_MAP); XmlUtil.writeNextValue(out, XML_TAG_80211_AX_ENABLED, softApConfig.isIeee80211axEnabled()); XmlUtil.writeNextValue(out, XML_TAG_USER_CONFIGURATION, softApConfig.isUserConfiguration()); } if (SdkLevel.isAtLeastT()) { XmlUtil.writeNextValue(out, XML_TAG_BRIDGED_MODE_OPPORTUNISTIC_SHUTDOWN_TIMEOUT_MILLIS, softApConfig.getBridgedModeOpportunisticShutdownTimeoutMillisInternal()); XmlUtil.writeNextSectionStart(out, XML_TAG_VENDOR_ELEMENTS); XmlUtil.SoftApConfigurationXmlUtil.writeVendorElementsSetToXml(out, softApConfig.getVendorElementsInternal()); XmlUtil.writeNextSectionEnd(out, XML_TAG_VENDOR_ELEMENTS); XmlUtil.writeNextValue(out, XML_TAG_80211_BE_ENABLED, softApConfig.isIeee80211beEnabled()); if (softApConfig.getPersistentRandomizedMacAddress() != null) { XmlUtil.writeNextValue(out, XML_TAG_PERSISTENT_RANDOMIZED_MAC_ADDRESS, softApConfig.getPersistentRandomizedMacAddress().toString()); } } if (SdkLevel.isAtLeastV()) { writeVendorDataListToXml(out, softApConfig.getVendorData()); } } // End of writeSoftApConfigurationToXml /** * Returns configuration of the SoftAp from the XML stream. * * @param in XmlPullParser instance pointing to the XML stream. * @param outerTagDepth depth of the outer tag in the XML document. * @param settingsMigrationDataHolder the class instance of SettingsMigrationDataHolder */ @Nullable public static SoftApConfiguration parseFromXml(XmlPullParser in, int outerTagDepth, SettingsMigrationDataHolder settingsMigrationDataHolder, boolean shouldExpectEncryptedCredentials, @Nullable WifiConfigStoreEncryptionUtil encryptionUtil) throws XmlPullParserException, IOException { SoftApConfiguration.Builder softApConfigBuilder = new SoftApConfiguration.Builder(); int securityType = SoftApConfiguration.SECURITY_TYPE_OPEN; String passphrase = null; // SSID may be retrieved from the old encoding (XML_TAG_SSID) or the new encoding // (XML_TAG_WIFI_SSID). boolean hasSsid = false; String bssid = null; // Note that, during deserialization, we may read the old band encoding (XML_TAG_BAND) // or the new band encoding (XML_TAG_AP_BAND) that is used after the introduction of the // 6GHz band. If the old encoding is found, a conversion is done. int channel = -1; int apBand = -1; boolean hasBandChannelMap = false; List blockedList = new ArrayList<>(); List allowedList = new ArrayList<>(); boolean autoShutdownEnabledTagPresent = false; try { while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { if (in.getAttributeValue(null, "name") != null) { String[] valueName = new String[1]; Object value = XmlUtil.readCurrentValue(in, valueName); if (TextUtils.isEmpty(valueName[0])) { throw new XmlPullParserException("Missing value name"); } switch (valueName[0]) { case XML_TAG_SSID: hasSsid = true; softApConfigBuilder.setSsid((String) value); break; case XML_TAG_WIFI_SSID: hasSsid = true; final WifiSsid wifiSsid = WifiSsid.fromString((String) value); if (SdkLevel.isAtLeastT()) { softApConfigBuilder.setWifiSsid(wifiSsid); } else { // If the SSID is non-UTF-8, then use WifiManager.UNKNOWN_SSID. // This should not happen since non-UTF-8 SSIDs may only be set // with SoftApConfiguration#Builder#setWifiSsid(WifiSsid), // which is only available in T and above. final CharSequence utf8Ssid = wifiSsid.getUtf8Text(); softApConfigBuilder.setSsid(utf8Ssid != null ? utf8Ssid.toString() : WifiManager.UNKNOWN_SSID); } break; case XML_TAG_BSSID: bssid = (String) value; softApConfigBuilder.setBssid(MacAddress.fromString(bssid)); break; case XML_TAG_BAND: apBand = ApConfigUtil.convertWifiConfigBandToSoftApConfigBand( (int) value); break; case XML_TAG_AP_BAND: apBand = (int) value; break; case XML_TAG_CHANNEL: channel = (int) value; break; case XML_TAG_HIDDEN_SSID: softApConfigBuilder.setHiddenSsid((boolean) value); break; case XML_TAG_SECURITY_TYPE: securityType = (int) value; break; case XML_TAG_WPA2_PASSPHRASE: case XML_TAG_PASSPHRASE: passphrase = (String) value; break; case XML_TAG_MAX_NUMBER_OF_CLIENTS: softApConfigBuilder.setMaxNumberOfClients((int) value); break; case XML_TAG_AUTO_SHUTDOWN_ENABLED: softApConfigBuilder.setAutoShutdownEnabled((boolean) value); autoShutdownEnabledTagPresent = true; break; case XML_TAG_SHUTDOWN_TIMEOUT_MILLIS: long shutDownMillis = 0; if (value instanceof Integer) { shutDownMillis = Long.valueOf((int) value); } else if (value instanceof Long) { shutDownMillis = (long) value; } if (shutDownMillis == 0 && CompatChanges.isChangeEnabled( SoftApConfiguration.REMOVE_ZERO_FOR_TIMEOUT_SETTING)) { shutDownMillis = SoftApConfiguration.DEFAULT_TIMEOUT; } softApConfigBuilder.setShutdownTimeoutMillis(shutDownMillis); break; case XML_TAG_CLIENT_CONTROL_BY_USER: softApConfigBuilder.setClientControlByUserEnabled((boolean) value); break; case XML_TAG_BRIDGED_MODE_OPPORTUNISTIC_SHUTDOWN_ENABLED: if (SdkLevel.isAtLeastS()) { softApConfigBuilder.setBridgedModeOpportunisticShutdownEnabled( (boolean) value); } break; case XML_TAG_MAC_RAMDOMIZATION_SETTING: if (SdkLevel.isAtLeastS()) { softApConfigBuilder.setMacRandomizationSetting((int) value); } break; case XML_TAG_80211_AX_ENABLED: if (SdkLevel.isAtLeastS()) { softApConfigBuilder.setIeee80211axEnabled((boolean) value); } break; case XML_TAG_80211_BE_ENABLED: if (SdkLevel.isAtLeastT()) { softApConfigBuilder.setIeee80211beEnabled((boolean) value); } break; case XML_TAG_USER_CONFIGURATION: if (SdkLevel.isAtLeastS()) { softApConfigBuilder.setUserConfiguration((boolean) value); } break; case XML_TAG_BRIDGED_MODE_OPPORTUNISTIC_SHUTDOWN_TIMEOUT_MILLIS: if (SdkLevel.isAtLeastT()) { long bridgedTimeout = (long) value; bridgedTimeout = bridgedTimeout == 0 ? SoftApConfiguration.DEFAULT_TIMEOUT : bridgedTimeout; softApConfigBuilder .setBridgedModeOpportunisticShutdownTimeoutMillis( bridgedTimeout); } break; case XML_TAG_PERSISTENT_RANDOMIZED_MAC_ADDRESS: if (SdkLevel.isAtLeastT()) { softApConfigBuilder.setRandomizedMacAddress( MacAddress.fromString((String) value)); } break; default: Log.w(TAG, "Ignoring unknown value name " + valueName[0]); break; } } else { String tagName = in.getName(); List parseredList; if (tagName == null) { throw new XmlPullParserException("Unexpected null tag found"); } switch (tagName) { case XML_TAG_BLOCKED_CLIENT_LIST: parseredList = XmlUtil.SoftApConfigurationXmlUtil.parseClientListFromXml( in, outerTagDepth + 1); if (parseredList != null) { blockedList = new ArrayList<>(parseredList); } break; case XML_TAG_ALLOWED_CLIENT_LIST: parseredList = XmlUtil.SoftApConfigurationXmlUtil.parseClientListFromXml( in, outerTagDepth + 1); if (parseredList != null) { allowedList = new ArrayList<>(parseredList); } break; case XML_TAG_BAND_CHANNEL_MAP: if (SdkLevel.isAtLeastS()) { hasBandChannelMap = true; SparseIntArray channels = XmlUtil.SoftApConfigurationXmlUtil .parseChannelsFromXml(in, outerTagDepth + 1); softApConfigBuilder.setChannels(channels); } break; case XML_TAG_VENDOR_ELEMENTS: if (SdkLevel.isAtLeastT()) { softApConfigBuilder.setVendorElements( SoftApConfigurationXmlUtil.parseVendorElementsFromXml( in, outerTagDepth + 1)); } break; case XML_TAG_PASSPHRASE: passphrase = readSoftApPassphraseFromXml(in, outerTagDepth, shouldExpectEncryptedCredentials, encryptionUtil); break; case XML_TAG_VENDOR_DATA_LIST: if (SdkLevel.isAtLeastV()) { softApConfigBuilder.setVendorData( parseVendorDataListFromXml(in, outerTagDepth + 1)); } break; default: Log.w(TAG, "Ignoring unknown tag found: " + tagName); break; } } } softApConfigBuilder.setBlockedClientList(blockedList); softApConfigBuilder.setAllowedClientList(allowedList); if (!hasBandChannelMap) { // Set channel and band if (channel == 0) { softApConfigBuilder.setBand(apBand); } else { softApConfigBuilder.setChannel(channel, apBand); } } // We should at least have an SSID restored from store. if (!hasSsid) { Log.e(TAG, "Failed to parse SSID"); return null; } if (ApConfigUtil.isNonPasswordAP(securityType)) { softApConfigBuilder.setPassphrase(null, securityType); } else { softApConfigBuilder.setPassphrase(passphrase, securityType); } if (!autoShutdownEnabledTagPresent) { // Migrate data out of settings. WifiMigration.SettingsMigrationData migrationData = settingsMigrationDataHolder.retrieveData(); if (migrationData == null) { Log.e(TAG, "No migration data present"); } else { softApConfigBuilder.setAutoShutdownEnabled( migrationData.isSoftApTimeoutEnabled()); } } if (bssid != null && SdkLevel.isAtLeastS()) { // Force MAC randomization setting to none when BSSID is configured softApConfigBuilder.setMacRandomizationSetting( SoftApConfiguration.RANDOMIZATION_NONE); } } catch (IllegalArgumentException e) { Log.e(TAG, "Failed to parse configuration " + e); return null; } return softApConfigBuilder.build(); } // End of parseFromXml } private static void writeSoftApPassphraseToXml( XmlSerializer out, String passphrase, WifiConfigStoreEncryptionUtil encryptionUtil) throws XmlPullParserException, IOException { EncryptedData encryptedData = null; if (encryptionUtil != null && passphrase != null) { encryptedData = encryptionUtil.encrypt(passphrase.getBytes()); if (encryptedData == null) { // We silently fail encryption failures! Log.wtf(TAG, "Encryption of softAp passphrase failed"); } } if (encryptedData != null) { writeNextSectionStart(out, SoftApConfigurationXmlUtil.XML_TAG_PASSPHRASE); EncryptedDataXmlUtil.writeToXml(out, encryptedData); writeNextSectionEnd(out, SoftApConfigurationXmlUtil.XML_TAG_PASSPHRASE); } else { writeNextValue(out, SoftApConfigurationXmlUtil.XML_TAG_PASSPHRASE, passphrase); } } private static String readSoftApPassphraseFromXml(XmlPullParser in, int outerTagDepth, boolean shouldExpectEncryptedCredentials, WifiConfigStoreEncryptionUtil encryptionUtil) throws XmlPullParserException, IOException { if (!shouldExpectEncryptedCredentials || encryptionUtil == null) { throw new XmlPullParserException( "Encrypted passphraseBytes section not expected"); } EncryptedData encryptedData = XmlUtil.EncryptedDataXmlUtil.parseFromXml(in, outerTagDepth + 1); byte[] passphraseBytes = encryptionUtil.decrypt(encryptedData); if (passphraseBytes == null) { Log.wtf(TAG, "Decryption of passphraseBytes failed"); return null; } else { return new String(passphraseBytes); } } /** * Write the provided vendor data list to XML. * * @param out XmlSerializer instance pointing to the XML stream * @param vendorDataList Vendor data list */ private static void writeVendorDataListToXml( XmlSerializer out, List vendorDataList) throws XmlPullParserException, IOException { if (vendorDataList == null || vendorDataList.isEmpty()) { return; } XmlUtil.writeNextSectionStart(out, XML_TAG_VENDOR_DATA_LIST); for (OuiKeyedData data : vendorDataList) { writeOuiKeyedDataToXml(out, data); } XmlUtil.writeNextSectionEnd(out, XML_TAG_VENDOR_DATA_LIST); } private static void writeOuiKeyedDataToXml( XmlSerializer out, OuiKeyedData ouiKeyedData) throws XmlPullParserException, IOException { // PersistableBundle cannot be written directly to XML // Use byte[] as an intermediate data structure if (ouiKeyedData == null) return; byte[] bundleBytes; try { ByteArrayOutputStream outputStream = new ByteArrayOutputStream(); ouiKeyedData.getData().writeToStream(outputStream); bundleBytes = outputStream.toByteArray(); } catch (Exception e) { Log.e(TAG, "Unable to write PersistableBundle to byte[]"); return; } XmlUtil.writeNextSectionStart(out, XML_TAG_OUI_KEYED_DATA); XmlUtil.writeNextValue(out, XML_TAG_VENDOR_DATA_OUI, ouiKeyedData.getOui()); XmlUtil.writeNextValue(out, XML_TAG_PERSISTABLE_BUNDLE, bundleBytes); XmlUtil.writeNextSectionEnd(out, XML_TAG_OUI_KEYED_DATA); } /** * Parses the vendor data list from the provided XML stream . * * @param in XmlPullParser instance pointing to the XML stream * @param outerTagDepth depth of the outer tag in the XML document * @return List of OuiKeyedData if successful, empty list otherwise */ private static List parseVendorDataListFromXml( XmlPullParser in, int outerTagDepth) throws XmlPullParserException, IOException, IllegalArgumentException { List vendorDataList = new ArrayList<>(); while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { String tagName = in.getName(); if (tagName == null) { throw new XmlPullParserException("Unexpected null tag found"); } switch (tagName) { case XML_TAG_OUI_KEYED_DATA: OuiKeyedData data = parseOuiKeyedDataFromXml(in, outerTagDepth + 1); if (data != null) { vendorDataList.add(data); } break; default: Log.w(TAG, "Ignoring unknown tag found: " + tagName); break; } } return vendorDataList; } private static PersistableBundle readPersistableBundleFromBytes(byte[] bundleBytes) { try { ByteArrayInputStream inputStream = new ByteArrayInputStream(bundleBytes); return PersistableBundle.readFromStream(inputStream); } catch (Exception e) { Log.e(TAG, "Unable to read PersistableBundle from byte[]"); return null; } } private static OuiKeyedData parseOuiKeyedDataFromXml( XmlPullParser in, int outerTagDepth) throws XmlPullParserException, IOException, IllegalArgumentException { int oui = 0; PersistableBundle bundle = null; while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { String[] valueName = new String[1]; Object value = XmlUtil.readCurrentValue(in, valueName); if (valueName[0] == null) { throw new XmlPullParserException("Missing value name"); } switch (valueName[0]) { case XML_TAG_VENDOR_DATA_OUI: oui = (int) value; break; case XML_TAG_PERSISTABLE_BUNDLE: bundle = readPersistableBundleFromBytes((byte[]) value); break; default: Log.e(TAG, "Unknown value name found: " + valueName[0]); break; } } try { return new OuiKeyedData.Builder(oui, bundle).build(); } catch (Exception e) { Log.e(TAG, "Unable to build OuiKeyedData"); return null; } } }