/* * Copyright (C) 2017 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.dialer.strictmode; import android.app.Application; import android.content.Context; import android.os.Looper; import android.os.StrictMode; import android.os.StrictMode.ThreadPolicy; import android.preference.PreferenceManager; import android.support.annotation.AnyThread; import android.support.v4.os.UserManagerCompat; import com.android.dialer.buildtype.BuildType; import com.android.dialer.function.Supplier; import com.android.dialer.storage.StorageComponent; /** Utilities for enforcing strict-mode in an app. */ public final class StrictModeUtils { private static final ThreadPolicy THREAD_NO_PENALTY = new StrictMode.ThreadPolicy.Builder().permitAll().build(); /** * Convenience method for disabling and enabling the thread policy death penalty using lambdas. * *

For example: * *

* Value foo = StrictModeUtils.bypass(() -> doDiskAccessOnMainThreadReturningValue()); * * *

The thread policy is only mutated if this is called from the main thread. */ @AnyThread public static T bypass(Supplier supplier) { if (isStrictModeAllowed() && onMainThread()) { ThreadPolicy originalPolicy = StrictMode.getThreadPolicy(); StrictMode.setThreadPolicy(THREAD_NO_PENALTY); try { return supplier.get(); } finally { StrictMode.setThreadPolicy(originalPolicy); } } return supplier.get(); } /** * Convenience method for disabling and enabling the thread policy death penalty using lambdas. * *

For example: * *

* StrictModeUtils.bypass(() -> doDiskAccessOnMainThread()); * * *

The thread policy is only mutated if this is called from the main thread. */ @AnyThread public static void bypass(Runnable runnable) { if (isStrictModeAllowed() && onMainThread()) { ThreadPolicy originalPolicy = StrictMode.getThreadPolicy(); StrictMode.setThreadPolicy(THREAD_NO_PENALTY); try { runnable.run(); } finally { StrictMode.setThreadPolicy(originalPolicy); } } else { runnable.run(); } } public static boolean isStrictModeAllowed() { return BuildType.get() == BuildType.BUGFOOD; } private static boolean onMainThread() { return Looper.getMainLooper().equals(Looper.myLooper()); } /** * We frequently access shared preferences on the main thread, which causes strict mode * violations. When strict mode is allowed, warm up the shared preferences so that later uses of * shared preferences access the in-memory versions and we don't have to bypass strict mode at * every point in the application where shared preferences are accessed. */ public static void warmupSharedPrefs(Application application) { // From credential-encrypted (CE) storage, i.e.: // /data/data/com.android.dialer/shared_prefs if (UserManagerCompat.isUserUnlocked(application)) { // _preferences.xml PreferenceManager.getDefaultSharedPreferences(application); // .xml application.getSharedPreferences(application.getPackageName(), Context.MODE_PRIVATE); } // From device-encrypted (DE) storage, i.e.: // /data/user_de/0/com.android.dialer/shared_prefs/ // _preferences.xml StorageComponent.get(application).unencryptedSharedPrefs(); } private StrictModeUtils() {} }