Default thread pool from AsyncTask can queue up to a certain number of tasks. We will use a custom pool with enough queue size to load app data. Fixes: 30355247 Change-Id: I5ad4c340191c785463011c1698d1d4625aba44ec
441 lines
17 KiB
Java
441 lines
17 KiB
Java
/*
|
|
* 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.settings.datausage;
|
|
|
|
import android.app.LoaderManager;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.content.Loader;
|
|
import android.content.pm.ApplicationInfo;
|
|
import android.content.pm.PackageManager;
|
|
import android.content.pm.UserInfo;
|
|
import android.graphics.drawable.Drawable;
|
|
import android.net.INetworkStatsSession;
|
|
import android.net.NetworkPolicy;
|
|
import android.net.NetworkStatsHistory;
|
|
import android.net.NetworkTemplate;
|
|
import android.net.TrafficStats;
|
|
import android.os.AsyncTask;
|
|
import android.os.Bundle;
|
|
import android.os.Process;
|
|
import android.os.RemoteException;
|
|
import android.os.UserHandle;
|
|
import android.os.UserManager;
|
|
import android.support.v14.preference.SwitchPreference;
|
|
import android.support.v7.preference.Preference;
|
|
import android.support.v7.preference.PreferenceCategory;
|
|
import android.text.format.Formatter;
|
|
import android.util.ArraySet;
|
|
import android.util.Log;
|
|
import android.view.View;
|
|
import android.widget.AdapterView;
|
|
import com.android.internal.logging.MetricsProto.MetricsEvent;
|
|
import com.android.settings.AppHeader;
|
|
import com.android.settings.R;
|
|
import com.android.settings.applications.AppInfoBase;
|
|
import com.android.settingslib.AppItem;
|
|
import com.android.settingslib.Utils;
|
|
import com.android.settingslib.net.ChartData;
|
|
import com.android.settingslib.net.ChartDataLoader;
|
|
import com.android.settingslib.net.UidDetailProvider;
|
|
|
|
import java.util.concurrent.BlockingQueue;
|
|
import java.util.concurrent.LinkedBlockingQueue;
|
|
import java.util.concurrent.RejectedExecutionException;
|
|
import java.util.concurrent.ThreadPoolExecutor;
|
|
import java.util.concurrent.TimeUnit;
|
|
|
|
import static android.net.NetworkPolicyManager.POLICY_REJECT_METERED_BACKGROUND;
|
|
|
|
public class AppDataUsage extends DataUsageBase implements Preference.OnPreferenceChangeListener,
|
|
DataSaverBackend.Listener {
|
|
|
|
private static final String TAG = "AppDataUsage";
|
|
|
|
public static final String ARG_APP_ITEM = "app_item";
|
|
public static final String ARG_NETWORK_TEMPLATE = "network_template";
|
|
|
|
private static final String KEY_TOTAL_USAGE = "total_usage";
|
|
private static final String KEY_FOREGROUND_USAGE = "foreground_usage";
|
|
private static final String KEY_BACKGROUND_USAGE = "background_usage";
|
|
private static final String KEY_APP_SETTINGS = "app_settings";
|
|
private static final String KEY_RESTRICT_BACKGROUND = "restrict_background";
|
|
private static final String KEY_APP_LIST = "app_list";
|
|
private static final String KEY_CYCLE = "cycle";
|
|
private static final String KEY_UNRESTRICTED_DATA = "unrestricted_data_saver";
|
|
|
|
private static final int LOADER_CHART_DATA = 2;
|
|
|
|
private final ArraySet<String> mPackages = new ArraySet<>();
|
|
private Preference mTotalUsage;
|
|
private Preference mForegroundUsage;
|
|
private Preference mBackgroundUsage;
|
|
private Preference mAppSettings;
|
|
private SwitchPreference mRestrictBackground;
|
|
private PreferenceCategory mAppList;
|
|
|
|
private Drawable mIcon;
|
|
private CharSequence mLabel;
|
|
private String mPackageName;
|
|
private INetworkStatsSession mStatsSession;
|
|
private CycleAdapter mCycleAdapter;
|
|
|
|
private long mStart;
|
|
private long mEnd;
|
|
private ChartData mChartData;
|
|
private NetworkTemplate mTemplate;
|
|
private NetworkPolicy mPolicy;
|
|
private AppItem mAppItem;
|
|
private Intent mAppSettingsIntent;
|
|
private SpinnerPreference mCycle;
|
|
private SwitchPreference mUnrestrictedData;
|
|
private DataSaverBackend mDataSaverBackend;
|
|
|
|
// Parameters to construct an efficient ThreadPoolExecutor
|
|
private static final int CPU_COUNT = Runtime.getRuntime().availableProcessors();
|
|
private static final int CORE_POOL_SIZE = Math.max(2, Math.min(CPU_COUNT - 1, 4));
|
|
private static final int MAXIMUM_POOL_SIZE = CPU_COUNT * 2 + 1;
|
|
private static final int KEEP_ALIVE_SECONDS = 30;
|
|
|
|
@Override
|
|
public void onCreate(Bundle icicle) {
|
|
super.onCreate(icicle);
|
|
final Bundle args = getArguments();
|
|
|
|
try {
|
|
mStatsSession = services.mStatsService.openSession();
|
|
} catch (RemoteException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
|
|
mAppItem = (args != null) ? (AppItem) args.getParcelable(ARG_APP_ITEM) : null;
|
|
mTemplate = (args != null) ? (NetworkTemplate) args.getParcelable(ARG_NETWORK_TEMPLATE)
|
|
: null;
|
|
if (mTemplate == null) {
|
|
Context context = getContext();
|
|
mTemplate = DataUsageSummary.getDefaultTemplate(context,
|
|
DataUsageSummary.getDefaultSubscriptionId(context));
|
|
}
|
|
if (mAppItem == null) {
|
|
int uid = (args != null) ? args.getInt(AppInfoBase.ARG_PACKAGE_UID, -1)
|
|
: getActivity().getIntent().getIntExtra(AppInfoBase.ARG_PACKAGE_UID, -1);
|
|
if (uid == -1) {
|
|
// TODO: Log error.
|
|
getActivity().finish();
|
|
} else {
|
|
addUid(uid);
|
|
mAppItem = new AppItem(uid);
|
|
mAppItem.addUid(uid);
|
|
}
|
|
} else {
|
|
for (int i = 0; i < mAppItem.uids.size(); i++) {
|
|
addUid(mAppItem.uids.keyAt(i));
|
|
}
|
|
}
|
|
addPreferencesFromResource(R.xml.app_data_usage);
|
|
|
|
mTotalUsage = findPreference(KEY_TOTAL_USAGE);
|
|
mForegroundUsage = findPreference(KEY_FOREGROUND_USAGE);
|
|
mBackgroundUsage = findPreference(KEY_BACKGROUND_USAGE);
|
|
|
|
mCycle = (SpinnerPreference) findPreference(KEY_CYCLE);
|
|
mCycleAdapter = new CycleAdapter(getContext(), mCycle, mCycleListener, false);
|
|
|
|
if (mAppItem.key > 0) {
|
|
if (mPackages.size() != 0) {
|
|
PackageManager pm = getPackageManager();
|
|
try {
|
|
ApplicationInfo info = pm.getApplicationInfo(mPackages.valueAt(0), 0);
|
|
mIcon = info.loadIcon(pm);
|
|
mLabel = info.loadLabel(pm);
|
|
mPackageName = info.packageName;
|
|
} catch (PackageManager.NameNotFoundException e) {
|
|
}
|
|
}
|
|
if (!UserHandle.isApp(mAppItem.key)) {
|
|
removePreference(KEY_UNRESTRICTED_DATA);
|
|
removePreference(KEY_RESTRICT_BACKGROUND);
|
|
} else {
|
|
mRestrictBackground = (SwitchPreference) findPreference(KEY_RESTRICT_BACKGROUND);
|
|
mRestrictBackground.setOnPreferenceChangeListener(this);
|
|
mUnrestrictedData = (SwitchPreference) findPreference(KEY_UNRESTRICTED_DATA);
|
|
mUnrestrictedData.setOnPreferenceChangeListener(this);
|
|
}
|
|
mDataSaverBackend = new DataSaverBackend(getContext());
|
|
mAppSettings = findPreference(KEY_APP_SETTINGS);
|
|
|
|
mAppSettingsIntent = new Intent(Intent.ACTION_MANAGE_NETWORK_USAGE);
|
|
mAppSettingsIntent.addCategory(Intent.CATEGORY_DEFAULT);
|
|
|
|
PackageManager pm = getPackageManager();
|
|
boolean matchFound = false;
|
|
for (String packageName : mPackages) {
|
|
mAppSettingsIntent.setPackage(packageName);
|
|
if (pm.resolveActivity(mAppSettingsIntent, 0) != null) {
|
|
matchFound = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!matchFound) {
|
|
removePreference(KEY_APP_SETTINGS);
|
|
mAppSettings = null;
|
|
}
|
|
|
|
if (mPackages.size() > 1) {
|
|
mAppList = (PreferenceCategory) findPreference(KEY_APP_LIST);
|
|
final int packageSize = mPackages.size();
|
|
final BlockingQueue<Runnable> workQueue = new LinkedBlockingQueue<>(packageSize);
|
|
final ThreadPoolExecutor executor = new ThreadPoolExecutor(CORE_POOL_SIZE,
|
|
MAXIMUM_POOL_SIZE, KEEP_ALIVE_SECONDS, TimeUnit.SECONDS, workQueue);
|
|
for (int i = 1; i < mPackages.size(); i++) {
|
|
final AppPrefLoader loader = new AppPrefLoader();
|
|
loader.executeOnExecutor(executor, mPackages.valueAt(i));
|
|
}
|
|
} else {
|
|
removePreference(KEY_APP_LIST);
|
|
}
|
|
} else {
|
|
if (mAppItem.key == TrafficStats.UID_REMOVED) {
|
|
mLabel = getContext().getString(R.string.data_usage_uninstalled_apps_users);
|
|
} else if (mAppItem.key == TrafficStats.UID_TETHERING) {
|
|
mLabel = getContext().getString(R.string.tether_settings_title_all);
|
|
} else {
|
|
final int userId = UidDetailProvider.getUserIdForKey(mAppItem.key);
|
|
final UserManager um = UserManager.get(getActivity());
|
|
final UserInfo info = um.getUserInfo(userId);
|
|
final PackageManager pm = getPackageManager();
|
|
mIcon = Utils.getUserIcon(getActivity(), um, info);
|
|
mLabel = Utils.getUserLabel(getActivity(), info);
|
|
mPackageName = getActivity().getPackageName();
|
|
}
|
|
removePreference(KEY_UNRESTRICTED_DATA);
|
|
removePreference(KEY_APP_SETTINGS);
|
|
removePreference(KEY_RESTRICT_BACKGROUND);
|
|
removePreference(KEY_APP_LIST);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onDestroy() {
|
|
TrafficStats.closeQuietly(mStatsSession);
|
|
super.onDestroy();
|
|
}
|
|
|
|
@Override
|
|
public void onResume() {
|
|
super.onResume();
|
|
if (mDataSaverBackend != null) {
|
|
mDataSaverBackend.addListener(this);
|
|
}
|
|
mPolicy = services.mPolicyEditor.getPolicy(mTemplate);
|
|
getLoaderManager().restartLoader(LOADER_CHART_DATA,
|
|
ChartDataLoader.buildArgs(mTemplate, mAppItem), mChartDataCallbacks);
|
|
updatePrefs();
|
|
}
|
|
|
|
@Override
|
|
public void onPause() {
|
|
super.onPause();
|
|
if (mDataSaverBackend != null) {
|
|
mDataSaverBackend.remListener(this);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean onPreferenceChange(Preference preference, Object newValue) {
|
|
if (preference == mRestrictBackground) {
|
|
mDataSaverBackend.setIsBlacklisted(mAppItem.key, mPackageName, !(Boolean) newValue);
|
|
return true;
|
|
} else if (preference == mUnrestrictedData) {
|
|
mDataSaverBackend.setIsWhitelisted(mAppItem.key, mPackageName, (Boolean) newValue);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public boolean onPreferenceTreeClick(Preference preference) {
|
|
if (preference == mAppSettings) {
|
|
// TODO: target towards entire UID instead of just first package
|
|
getActivity().startActivityAsUser(mAppSettingsIntent, new UserHandle(
|
|
UserHandle.getUserId(mAppItem.key)));
|
|
return true;
|
|
}
|
|
return super.onPreferenceTreeClick(preference);
|
|
}
|
|
|
|
private void updatePrefs() {
|
|
updatePrefs(getAppRestrictBackground(), getUnrestrictData());
|
|
}
|
|
|
|
private void updatePrefs(boolean restrictBackground, boolean unrestrictData) {
|
|
if (mRestrictBackground != null) {
|
|
mRestrictBackground.setChecked(!restrictBackground);
|
|
}
|
|
if (mUnrestrictedData != null) {
|
|
if (restrictBackground) {
|
|
mUnrestrictedData.setVisible(false);
|
|
} else {
|
|
mUnrestrictedData.setVisible(true);
|
|
mUnrestrictedData.setChecked(unrestrictData);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void addUid(int uid) {
|
|
String[] packages = getPackageManager().getPackagesForUid(uid);
|
|
if (packages != null) {
|
|
for (int i = 0; i < packages.length; i++) {
|
|
mPackages.add(packages[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void bindData() {
|
|
final long backgroundBytes, foregroundBytes;
|
|
if (mChartData == null || mStart == 0) {
|
|
backgroundBytes = foregroundBytes = 0;
|
|
mCycle.setVisible(false);
|
|
} else {
|
|
mCycle.setVisible(true);
|
|
final long now = System.currentTimeMillis();
|
|
NetworkStatsHistory.Entry entry = null;
|
|
entry = mChartData.detailDefault.getValues(mStart, mEnd, now, entry);
|
|
backgroundBytes = entry.rxBytes + entry.txBytes;
|
|
entry = mChartData.detailForeground.getValues(mStart, mEnd, now, entry);
|
|
foregroundBytes = entry.rxBytes + entry.txBytes;
|
|
}
|
|
final long totalBytes = backgroundBytes + foregroundBytes;
|
|
final Context context = getContext();
|
|
|
|
mTotalUsage.setSummary(Formatter.formatFileSize(context, totalBytes));
|
|
mForegroundUsage.setSummary(Formatter.formatFileSize(context, foregroundBytes));
|
|
mBackgroundUsage.setSummary(Formatter.formatFileSize(context, backgroundBytes));
|
|
}
|
|
|
|
private boolean getAppRestrictBackground() {
|
|
final int uid = mAppItem.key;
|
|
final int uidPolicy = services.mPolicyManager.getUidPolicy(uid);
|
|
return (uidPolicy & POLICY_REJECT_METERED_BACKGROUND) != 0;
|
|
}
|
|
|
|
private boolean getUnrestrictData() {
|
|
if (mDataSaverBackend != null) {
|
|
return mDataSaverBackend.isWhitelisted(mAppItem.key);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public void onViewCreated(View view, Bundle savedInstanceState) {
|
|
super.onViewCreated(view, savedInstanceState);
|
|
|
|
View header = setPinnedHeaderView(R.layout.app_header);
|
|
String pkg = mPackages.size() != 0 ? mPackages.valueAt(0) : null;
|
|
int uid = 0;
|
|
try {
|
|
uid = pkg != null ? getPackageManager().getPackageUid(pkg, 0) : 0;
|
|
} catch (PackageManager.NameNotFoundException e) {
|
|
}
|
|
AppHeader.setupHeaderView(getActivity(), mIcon, mLabel,
|
|
pkg, uid, AppHeader.includeAppInfo(this), 0, header, null);
|
|
}
|
|
|
|
@Override
|
|
protected int getMetricsCategory() {
|
|
return MetricsEvent.APP_DATA_USAGE;
|
|
}
|
|
|
|
private AdapterView.OnItemSelectedListener mCycleListener =
|
|
new AdapterView.OnItemSelectedListener() {
|
|
@Override
|
|
public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
|
|
final CycleAdapter.CycleItem cycle = (CycleAdapter.CycleItem) mCycle.getSelectedItem();
|
|
|
|
mStart = cycle.start;
|
|
mEnd = cycle.end;
|
|
bindData();
|
|
}
|
|
|
|
@Override
|
|
public void onNothingSelected(AdapterView<?> parent) {
|
|
// ignored
|
|
}
|
|
};
|
|
|
|
private final LoaderManager.LoaderCallbacks<ChartData> mChartDataCallbacks =
|
|
new LoaderManager.LoaderCallbacks<ChartData>() {
|
|
@Override
|
|
public Loader<ChartData> onCreateLoader(int id, Bundle args) {
|
|
return new ChartDataLoader(getActivity(), mStatsSession, args);
|
|
}
|
|
|
|
@Override
|
|
public void onLoadFinished(Loader<ChartData> loader, ChartData data) {
|
|
mChartData = data;
|
|
mCycleAdapter.updateCycleList(mPolicy, mChartData);
|
|
bindData();
|
|
}
|
|
|
|
@Override
|
|
public void onLoaderReset(Loader<ChartData> loader) {
|
|
}
|
|
};
|
|
|
|
private class AppPrefLoader extends AsyncTask<String, Void, Preference> {
|
|
@Override
|
|
protected Preference doInBackground(String... params) {
|
|
PackageManager pm = getPackageManager();
|
|
String pkg = params[0];
|
|
try {
|
|
ApplicationInfo info = pm.getApplicationInfo(pkg, 0);
|
|
Preference preference = new Preference(getPrefContext());
|
|
preference.setIcon(info.loadIcon(pm));
|
|
preference.setTitle(info.loadLabel(pm));
|
|
preference.setSelectable(false);
|
|
return preference;
|
|
} catch (PackageManager.NameNotFoundException e) {
|
|
}
|
|
return null;
|
|
}
|
|
|
|
@Override
|
|
protected void onPostExecute(Preference pref) {
|
|
if (pref != null && mAppList != null) {
|
|
mAppList.addPreference(pref);
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onDataSaverChanged(boolean isDataSaving) {
|
|
|
|
}
|
|
|
|
@Override
|
|
public void onWhitelistStatusChanged(int uid, boolean isWhitelisted) {
|
|
if (mAppItem.uids.get(uid, false)) {
|
|
updatePrefs(getAppRestrictBackground(), isWhitelisted);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onBlacklistStatusChanged(int uid, boolean isBlacklisted) {
|
|
if (mAppItem.uids.get(uid, false)) {
|
|
updatePrefs(isBlacklisted, getUnrestrictData());
|
|
}
|
|
}
|
|
}
|