1 /* 2 * Copyright (C) 2020 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package android.net.wifi; 18 19 import static android.os.Environment.getDataMiscCeDirectory; 20 import static android.os.Environment.getDataMiscDirectory; 21 22 import android.annotation.IntDef; 23 import android.annotation.NonNull; 24 import android.annotation.Nullable; 25 import android.annotation.SystemApi; 26 import android.content.Context; 27 import android.os.Parcel; 28 import android.os.Parcelable; 29 import android.os.UserHandle; 30 import android.provider.Settings; 31 import android.util.AtomicFile; 32 import android.util.SparseArray; 33 34 import java.io.File; 35 import java.io.FileNotFoundException; 36 import java.io.InputStream; 37 import java.lang.annotation.Retention; 38 import java.lang.annotation.RetentionPolicy; 39 import java.util.Objects; 40 41 /** 42 * Class used to provide one time hooks for existing OEM devices to migrate their config store 43 * data and other settings to the wifi apex. 44 * @hide 45 */ 46 @SystemApi 47 public final class WifiMigration { 48 /** 49 * Directory to read the wifi config store files from under. 50 */ 51 private static final String LEGACY_WIFI_STORE_DIRECTORY_NAME = "wifi"; 52 /** 53 * Config store file for general shared store file. 54 * AOSP Path on Android 10: /data/misc/wifi/WifiConfigStore.xml 55 */ 56 public static final int STORE_FILE_SHARED_GENERAL = 0; 57 /** 58 * Config store file for softap shared store file. 59 * AOSP Path on Android 10: /data/misc/wifi/softap.conf 60 */ 61 public static final int STORE_FILE_SHARED_SOFTAP = 1; 62 /** 63 * Config store file for general user store file. 64 * AOSP Path on Android 10: /data/misc_ce/<userId>/wifi/WifiConfigStore.xml 65 */ 66 public static final int STORE_FILE_USER_GENERAL = 2; 67 /** 68 * Config store file for network suggestions user store file. 69 * AOSP Path on Android 10: /data/misc_ce/<userId>/wifi/WifiConfigStoreNetworkSuggestions.xml 70 */ 71 public static final int STORE_FILE_USER_NETWORK_SUGGESTIONS = 3; 72 73 /** @hide */ 74 @IntDef(prefix = { "STORE_FILE_SHARED_" }, value = { 75 STORE_FILE_SHARED_GENERAL, 76 STORE_FILE_SHARED_SOFTAP, 77 }) 78 @Retention(RetentionPolicy.SOURCE) 79 public @interface SharedStoreFileId { } 80 81 /** @hide */ 82 @IntDef(prefix = { "STORE_FILE_USER_" }, value = { 83 STORE_FILE_USER_GENERAL, 84 STORE_FILE_USER_NETWORK_SUGGESTIONS 85 }) 86 @Retention(RetentionPolicy.SOURCE) 87 public @interface UserStoreFileId { } 88 89 /** 90 * Mapping of Store file Id to Store file names. 91 * 92 * NOTE: This is the default path for the files on AOSP devices. If the OEM has modified 93 * the path or renamed the files, please edit this appropriately. 94 */ 95 private static final SparseArray<String> STORE_ID_TO_FILE_NAME = 96 new SparseArray<String>() {{ 97 put(STORE_FILE_SHARED_GENERAL, "WifiConfigStore.xml"); 98 put(STORE_FILE_SHARED_SOFTAP, "WifiConfigStoreSoftAp.xml"); 99 put(STORE_FILE_USER_GENERAL, "WifiConfigStore.xml"); 100 put(STORE_FILE_USER_NETWORK_SUGGESTIONS, "WifiConfigStoreNetworkSuggestions.xml"); 101 }}; 102 103 /** 104 * Pre-apex wifi shared folder. 105 */ getLegacyWifiSharedDirectory()106 private static File getLegacyWifiSharedDirectory() { 107 return new File(getDataMiscDirectory(), LEGACY_WIFI_STORE_DIRECTORY_NAME); 108 } 109 110 /** 111 * Pre-apex wifi user folder. 112 */ getLegacyWifiUserDirectory(int userId)113 private static File getLegacyWifiUserDirectory(int userId) { 114 return new File(getDataMiscCeDirectory(userId), LEGACY_WIFI_STORE_DIRECTORY_NAME); 115 } 116 117 /** 118 * Legacy files were stored as AtomicFile. So, always use AtomicFile to operate on it to ensure 119 * data integrity. 120 */ getSharedAtomicFile(@haredStoreFileId int storeFileId)121 private static AtomicFile getSharedAtomicFile(@SharedStoreFileId int storeFileId) { 122 return new AtomicFile(new File( 123 getLegacyWifiSharedDirectory(), 124 STORE_ID_TO_FILE_NAME.get(storeFileId))); 125 } 126 127 /** 128 * Legacy files were stored as AtomicFile. So, always use AtomicFile to operate on it to ensure 129 * data integrity. 130 */ getUserAtomicFile(@serStoreFileId int storeFileId, int userId)131 private static AtomicFile getUserAtomicFile(@UserStoreFileId int storeFileId, int userId) { 132 return new AtomicFile(new File( 133 getLegacyWifiUserDirectory(userId), 134 STORE_ID_TO_FILE_NAME.get(storeFileId))); 135 } 136 WifiMigration()137 private WifiMigration() { } 138 139 /** 140 * Load data from legacy shared wifi config store file. 141 * <p> 142 * Expected AOSP format is available in the sample files under {@code 143 * frameworks/base/wifi/non-updatable/migration_samples/}. 144 * </p> 145 * <p> 146 * Note: 147 * <li>OEMs need to change the implementation of 148 * {@link #convertAndRetrieveSharedConfigStoreFile(int)} only if their existing config store 149 * format or file locations differs from the vanilla AOSP implementation.</li> 150 * <li>The wifi apex will invoke 151 * {@link #convertAndRetrieveSharedConfigStoreFile(int)} 152 * method on every bootup, it is the responsibility of the OEM implementation to ensure that 153 * they perform the necessary in place conversion of their config store file to conform to the 154 * AOSP format. The OEM should ensure that the method should only return the 155 * {@link InputStream} stream for the data to be migrated only on the first bootup.</li> 156 * <li>Once the migration is done, the apex will invoke 157 * {@link #removeSharedConfigStoreFile(int)} to delete the store file.</li> 158 * <li>The only relevant invocation of {@link #convertAndRetrieveSharedConfigStoreFile(int)} 159 * occurs when a previously released device upgrades to the wifi apex from an OEM 160 * implementation of the wifi stack. 161 * <li>Ensure that the legacy file paths are accessible to the wifi module (sepolicy rules, file 162 * permissions, etc). Since the wifi service continues to run inside system_server process, this 163 * method will be called from the same context (so ideally the file should still be accessible). 164 * </li> 165 * 166 * @param storeFileId Identifier for the config store file. One of 167 * {@link #STORE_FILE_SHARED_GENERAL} or {@link #STORE_FILE_SHARED_GENERAL} 168 * @return Instance of {@link InputStream} for migrating data, null if no migration is 169 * necessary. 170 * @throws IllegalArgumentException on invalid storeFileId. 171 */ 172 @Nullable convertAndRetrieveSharedConfigStoreFile( @haredStoreFileId int storeFileId)173 public static InputStream convertAndRetrieveSharedConfigStoreFile( 174 @SharedStoreFileId int storeFileId) { 175 if (storeFileId != STORE_FILE_SHARED_GENERAL && storeFileId != STORE_FILE_SHARED_SOFTAP) { 176 throw new IllegalArgumentException("Invalid shared store file id"); 177 } 178 try { 179 // OEMs should do conversions necessary here before returning the stream. 180 return getSharedAtomicFile(storeFileId).openRead(); 181 } catch (FileNotFoundException e) { 182 // Special handling for softap.conf. 183 // Note: OEM devices upgrading from Q -> R will only have the softap.conf file. 184 // Test devices running previous R builds however may have already migrated to the 185 // XML format. So, check for that above before falling back to check for legacy file. 186 if (storeFileId == STORE_FILE_SHARED_SOFTAP) { 187 return SoftApConfToXmlMigrationUtil.convert(); 188 } 189 return null; 190 } 191 } 192 193 /** 194 * Remove the legacy shared wifi config store file. 195 * 196 * @param storeFileId Identifier for the config store file. One of 197 * {@link #STORE_FILE_SHARED_GENERAL} or {@link #STORE_FILE_SHARED_GENERAL} 198 * @throws IllegalArgumentException on invalid storeFileId. 199 */ removeSharedConfigStoreFile(@haredStoreFileId int storeFileId)200 public static void removeSharedConfigStoreFile(@SharedStoreFileId int storeFileId) { 201 if (storeFileId != STORE_FILE_SHARED_GENERAL && storeFileId != STORE_FILE_SHARED_SOFTAP) { 202 throw new IllegalArgumentException("Invalid shared store file id"); 203 } 204 AtomicFile file = getSharedAtomicFile(storeFileId); 205 if (file.exists()) { 206 file.delete(); 207 return; 208 } 209 // Special handling for softap.conf. 210 // Note: OEM devices upgrading from Q -> R will only have the softap.conf file. 211 // Test devices running previous R builds however may have already migrated to the 212 // XML format. So, check for that above before falling back to check for legacy file. 213 if (storeFileId == STORE_FILE_SHARED_SOFTAP) { 214 SoftApConfToXmlMigrationUtil.remove(); 215 } 216 } 217 218 /** 219 * Load data from legacy user wifi config store file. 220 * <p> 221 * Expected AOSP format is available in the sample files under {@code 222 * frameworks/base/wifi/non-updatable/migration_samples/}. 223 * </p> 224 * <p> 225 * Note: 226 * <li>OEMs need to change the implementation of 227 * {@link #convertAndRetrieveUserConfigStoreFile(int, UserHandle)} only if their existing config 228 * store format or file locations differs from the vanilla AOSP implementation.</li> 229 * <li>The wifi apex will invoke 230 * {@link #convertAndRetrieveUserConfigStoreFile(int, UserHandle)} 231 * method on every bootup, it is the responsibility of the OEM implementation to ensure that 232 * they perform the necessary in place conversion of their config store file to conform to the 233 * AOSP format. The OEM should ensure that the method should only return the 234 * {@link InputStream} stream for the data to be migrated only on the first bootup.</li> 235 * <li>Once the migration is done, the apex will invoke 236 * {@link #removeUserConfigStoreFile(int, UserHandle)} to delete the store file.</li> 237 * <li>The only relevant invocation of 238 * {@link #convertAndRetrieveUserConfigStoreFile(int, UserHandle)} occurs when a previously 239 * released device upgrades to the wifi apex from an OEM implementation of the wifi 240 * stack. 241 * </li> 242 * <li>Ensure that the legacy file paths are accessible to the wifi module (sepolicy rules, file 243 * permissions, etc). Since the wifi service continues to run inside system_server process, this 244 * method will be called from the same context (so ideally the file should still be accessible). 245 * </li> 246 * 247 * @param storeFileId Identifier for the config store file. One of 248 * {@link #STORE_FILE_USER_GENERAL} or {@link #STORE_FILE_USER_NETWORK_SUGGESTIONS} 249 * @param userHandle User handle. 250 * @return Instance of {@link InputStream} for migrating data, null if no migration is 251 * necessary. 252 * @throws IllegalArgumentException on invalid storeFileId or userHandle. 253 */ 254 @Nullable convertAndRetrieveUserConfigStoreFile( @serStoreFileId int storeFileId, @NonNull UserHandle userHandle)255 public static InputStream convertAndRetrieveUserConfigStoreFile( 256 @UserStoreFileId int storeFileId, @NonNull UserHandle userHandle) { 257 if (storeFileId != STORE_FILE_USER_GENERAL 258 && storeFileId != STORE_FILE_USER_NETWORK_SUGGESTIONS) { 259 throw new IllegalArgumentException("Invalid user store file id"); 260 } 261 Objects.requireNonNull(userHandle); 262 try { 263 // OEMs should do conversions necessary here before returning the stream. 264 return getUserAtomicFile(storeFileId, userHandle.getIdentifier()).openRead(); 265 } catch (FileNotFoundException e) { 266 return null; 267 } 268 } 269 270 /** 271 * Remove the legacy user wifi config store file. 272 * 273 * @param storeFileId Identifier for the config store file. One of 274 * {@link #STORE_FILE_USER_GENERAL} or {@link #STORE_FILE_USER_NETWORK_SUGGESTIONS} 275 * @param userHandle User handle. 276 * @throws IllegalArgumentException on invalid storeFileId or userHandle. 277 */ removeUserConfigStoreFile( @serStoreFileId int storeFileId, @NonNull UserHandle userHandle)278 public static void removeUserConfigStoreFile( 279 @UserStoreFileId int storeFileId, @NonNull UserHandle userHandle) { 280 if (storeFileId != STORE_FILE_USER_GENERAL 281 && storeFileId != STORE_FILE_USER_NETWORK_SUGGESTIONS) { 282 throw new IllegalArgumentException("Invalid user store file id"); 283 } 284 Objects.requireNonNull(userHandle); 285 AtomicFile file = getUserAtomicFile(storeFileId, userHandle.getIdentifier()); 286 if (file.exists()) { 287 file.delete(); 288 } 289 } 290 291 /** 292 * Container for all the wifi settings data to migrate. 293 */ 294 public static final class SettingsMigrationData implements Parcelable { 295 private final boolean mScanAlwaysAvailable; 296 private final boolean mP2pFactoryResetPending; 297 private final String mP2pDeviceName; 298 private final boolean mSoftApTimeoutEnabled; 299 private final boolean mWakeupEnabled; 300 private final boolean mScanThrottleEnabled; 301 private final boolean mVerboseLoggingEnabled; 302 SettingsMigrationData(boolean scanAlwaysAvailable, boolean p2pFactoryResetPending, @Nullable String p2pDeviceName, boolean softApTimeoutEnabled, boolean wakeupEnabled, boolean scanThrottleEnabled, boolean verboseLoggingEnabled)303 private SettingsMigrationData(boolean scanAlwaysAvailable, boolean p2pFactoryResetPending, 304 @Nullable String p2pDeviceName, boolean softApTimeoutEnabled, boolean wakeupEnabled, 305 boolean scanThrottleEnabled, boolean verboseLoggingEnabled) { 306 mScanAlwaysAvailable = scanAlwaysAvailable; 307 mP2pFactoryResetPending = p2pFactoryResetPending; 308 mP2pDeviceName = p2pDeviceName; 309 mSoftApTimeoutEnabled = softApTimeoutEnabled; 310 mWakeupEnabled = wakeupEnabled; 311 mScanThrottleEnabled = scanThrottleEnabled; 312 mVerboseLoggingEnabled = verboseLoggingEnabled; 313 } 314 315 public static final @NonNull Parcelable.Creator<SettingsMigrationData> CREATOR = 316 new Parcelable.Creator<SettingsMigrationData>() { 317 @Override 318 public SettingsMigrationData createFromParcel(Parcel in) { 319 boolean scanAlwaysAvailable = in.readBoolean(); 320 boolean p2pFactoryResetPending = in.readBoolean(); 321 String p2pDeviceName = in.readString(); 322 boolean softApTimeoutEnabled = in.readBoolean(); 323 boolean wakeupEnabled = in.readBoolean(); 324 boolean scanThrottleEnabled = in.readBoolean(); 325 boolean verboseLoggingEnabled = in.readBoolean(); 326 return new SettingsMigrationData( 327 scanAlwaysAvailable, p2pFactoryResetPending, 328 p2pDeviceName, softApTimeoutEnabled, wakeupEnabled, 329 scanThrottleEnabled, verboseLoggingEnabled); 330 } 331 332 @Override 333 public SettingsMigrationData[] newArray(int size) { 334 return new SettingsMigrationData[size]; 335 } 336 }; 337 338 @Override describeContents()339 public int describeContents() { 340 return 0; 341 } 342 343 @Override writeToParcel(@onNull Parcel dest, int flags)344 public void writeToParcel(@NonNull Parcel dest, int flags) { 345 dest.writeBoolean(mScanAlwaysAvailable); 346 dest.writeBoolean(mP2pFactoryResetPending); 347 dest.writeString(mP2pDeviceName); 348 dest.writeBoolean(mSoftApTimeoutEnabled); 349 dest.writeBoolean(mWakeupEnabled); 350 dest.writeBoolean(mScanThrottleEnabled); 351 dest.writeBoolean(mVerboseLoggingEnabled); 352 } 353 354 /** 355 * @return True if scans are allowed even when wifi is toggled off, false otherwise. 356 */ isScanAlwaysAvailable()357 public boolean isScanAlwaysAvailable() { 358 return mScanAlwaysAvailable; 359 } 360 361 /** 362 * @return indicate whether factory reset request is pending. 363 */ isP2pFactoryResetPending()364 public boolean isP2pFactoryResetPending() { 365 return mP2pFactoryResetPending; 366 } 367 368 /** 369 * @return the Wi-Fi peer-to-peer device name 370 */ getP2pDeviceName()371 public @Nullable String getP2pDeviceName() { 372 return mP2pDeviceName; 373 } 374 375 /** 376 * @return Whether soft AP will shut down after a timeout period when no devices are 377 * connected. 378 */ isSoftApTimeoutEnabled()379 public boolean isSoftApTimeoutEnabled() { 380 return mSoftApTimeoutEnabled; 381 } 382 383 /** 384 * @return whether Wi-Fi Wakeup feature is enabled. 385 */ isWakeUpEnabled()386 public boolean isWakeUpEnabled() { 387 return mWakeupEnabled; 388 } 389 390 /** 391 * @return Whether wifi scan throttle is enabled or not. 392 */ isScanThrottleEnabled()393 public boolean isScanThrottleEnabled() { 394 return mScanThrottleEnabled; 395 } 396 397 /** 398 * @return Whether to enable verbose logging in Wi-Fi. 399 */ isVerboseLoggingEnabled()400 public boolean isVerboseLoggingEnabled() { 401 return mVerboseLoggingEnabled; 402 } 403 404 /** 405 * Builder to create instance of {@link SettingsMigrationData}. 406 */ 407 public static final class Builder { 408 private boolean mScanAlwaysAvailable; 409 private boolean mP2pFactoryResetPending; 410 private String mP2pDeviceName; 411 private boolean mSoftApTimeoutEnabled; 412 private boolean mWakeupEnabled; 413 private boolean mScanThrottleEnabled; 414 private boolean mVerboseLoggingEnabled; 415 Builder()416 public Builder() { 417 } 418 419 /** 420 * Setting to allow scans even when wifi is toggled off. 421 * 422 * @param available true if available, false otherwise. 423 * @return Instance of {@link Builder} to enable chaining of the builder method. 424 */ setScanAlwaysAvailable(boolean available)425 public @NonNull Builder setScanAlwaysAvailable(boolean available) { 426 mScanAlwaysAvailable = available; 427 return this; 428 } 429 430 /** 431 * Indicate whether factory reset request is pending. 432 * 433 * @param pending true if pending, false otherwise. 434 * @return Instance of {@link Builder} to enable chaining of the builder method. 435 */ setP2pFactoryResetPending(boolean pending)436 public @NonNull Builder setP2pFactoryResetPending(boolean pending) { 437 mP2pFactoryResetPending = pending; 438 return this; 439 } 440 441 /** 442 * The Wi-Fi peer-to-peer device name 443 * 444 * @param name Name if set, null otherwise. 445 * @return Instance of {@link Builder} to enable chaining of the builder method. 446 */ setP2pDeviceName(@ullable String name)447 public @NonNull Builder setP2pDeviceName(@Nullable String name) { 448 mP2pDeviceName = name; 449 return this; 450 } 451 452 /** 453 * Whether soft AP will shut down after a timeout period when no devices are connected. 454 * 455 * @param enabled true if enabled, false otherwise. 456 * @return Instance of {@link Builder} to enable chaining of the builder method. 457 */ setSoftApTimeoutEnabled(boolean enabled)458 public @NonNull Builder setSoftApTimeoutEnabled(boolean enabled) { 459 mSoftApTimeoutEnabled = enabled; 460 return this; 461 } 462 463 /** 464 * Value to specify if Wi-Fi Wakeup feature is enabled. 465 * 466 * @param enabled true if enabled, false otherwise. 467 * @return Instance of {@link Builder} to enable chaining of the builder method. 468 */ setWakeUpEnabled(boolean enabled)469 public @NonNull Builder setWakeUpEnabled(boolean enabled) { 470 mWakeupEnabled = enabled; 471 return this; 472 } 473 474 /** 475 * Whether wifi scan throttle is enabled or not. 476 * 477 * @param enabled true if enabled, false otherwise. 478 * @return Instance of {@link Builder} to enable chaining of the builder method. 479 */ setScanThrottleEnabled(boolean enabled)480 public @NonNull Builder setScanThrottleEnabled(boolean enabled) { 481 mScanThrottleEnabled = enabled; 482 return this; 483 } 484 485 /** 486 * Setting to enable verbose logging in Wi-Fi. 487 * 488 * @param enabled true if enabled, false otherwise. 489 * @return Instance of {@link Builder} to enable chaining of the builder method. 490 */ setVerboseLoggingEnabled(boolean enabled)491 public @NonNull Builder setVerboseLoggingEnabled(boolean enabled) { 492 mVerboseLoggingEnabled = enabled; 493 return this; 494 } 495 496 /** 497 * Build an instance of {@link SettingsMigrationData}. 498 * 499 * @return Instance of {@link SettingsMigrationData}. 500 */ build()501 public @NonNull SettingsMigrationData build() { 502 return new SettingsMigrationData(mScanAlwaysAvailable, mP2pFactoryResetPending, 503 mP2pDeviceName, mSoftApTimeoutEnabled, mWakeupEnabled, mScanThrottleEnabled, 504 mVerboseLoggingEnabled); 505 } 506 } 507 } 508 509 /** 510 * Load data from Settings.Global values. 511 * 512 * <p> 513 * Note: 514 * <li> This is method is invoked once on the first bootup. OEM can safely delete these settings 515 * once the migration is complete. The first & only relevant invocation of 516 * {@link #loadFromSettings(Context)} ()} occurs when a previously released 517 * device upgrades to the wifi apex from an OEM implementation of the wifi stack. 518 * </li> 519 * 520 * @param context Context to use for loading the settings provider. 521 * @return Instance of {@link SettingsMigrationData} for migrating data. 522 */ 523 @NonNull loadFromSettings(@onNull Context context)524 public static SettingsMigrationData loadFromSettings(@NonNull Context context) { 525 if (Settings.Global.getInt( 526 context.getContentResolver(), Settings.Global.WIFI_MIGRATION_COMPLETED, 0) == 1) { 527 // migration already complete, ignore. 528 return null; 529 } 530 SettingsMigrationData data = new SettingsMigrationData.Builder() 531 .setScanAlwaysAvailable( 532 Settings.Global.getInt(context.getContentResolver(), 533 Settings.Global.WIFI_SCAN_ALWAYS_AVAILABLE, 0) == 1) 534 .setP2pFactoryResetPending( 535 Settings.Global.getInt(context.getContentResolver(), 536 Settings.Global.WIFI_P2P_PENDING_FACTORY_RESET, 0) == 1) 537 .setP2pDeviceName( 538 Settings.Global.getString(context.getContentResolver(), 539 Settings.Global.WIFI_P2P_DEVICE_NAME)) 540 .setSoftApTimeoutEnabled( 541 Settings.Global.getInt(context.getContentResolver(), 542 Settings.Global.SOFT_AP_TIMEOUT_ENABLED, 1) == 1) 543 .setWakeUpEnabled( 544 Settings.Global.getInt(context.getContentResolver(), 545 Settings.Global.WIFI_WAKEUP_ENABLED, 0) == 1) 546 .setScanThrottleEnabled( 547 Settings.Global.getInt(context.getContentResolver(), 548 Settings.Global.WIFI_SCAN_THROTTLE_ENABLED, 1) == 1) 549 .setVerboseLoggingEnabled( 550 Settings.Global.getInt(context.getContentResolver(), 551 Settings.Global.WIFI_VERBOSE_LOGGING_ENABLED, 0) == 1) 552 .build(); 553 Settings.Global.putInt( 554 context.getContentResolver(), Settings.Global.WIFI_MIGRATION_COMPLETED, 1); 555 return data; 556 557 } 558 } 559