Cluster apps by user in Data Usage.

When displaying apps that have used data, cluster all app usage
together regardless of profile/user.  Always persist policy rules
using primary UID (the UID under the default user).

Bug: 6140462
Change-Id: Ia00bb42b26987553926f4027583dbe03b3bafba1
This commit is contained in:
Jeff Sharkey
2012-03-21 17:09:07 -07:00
parent 686d1ee97a
commit ef6e1ff728
2 changed files with 94 additions and 73 deletions

View File

@@ -76,9 +76,12 @@ import android.net.NetworkTemplate;
import android.os.AsyncTask; import android.os.AsyncTask;
import android.os.Bundle; import android.os.Bundle;
import android.os.INetworkManagementService; import android.os.INetworkManagementService;
import android.os.Parcel;
import android.os.Parcelable;
import android.os.RemoteException; import android.os.RemoteException;
import android.os.ServiceManager; import android.os.ServiceManager;
import android.os.SystemProperties; import android.os.SystemProperties;
import android.os.UserId;
import android.preference.Preference; import android.preference.Preference;
import android.provider.Settings; import android.provider.Settings;
import android.telephony.TelephonyManager; import android.telephony.TelephonyManager;
@@ -88,6 +91,7 @@ import android.text.format.Formatter;
import android.text.format.Time; import android.text.format.Time;
import android.util.Log; import android.util.Log;
import android.util.SparseArray; import android.util.SparseArray;
import android.util.SparseBooleanArray;
import android.view.LayoutInflater; import android.view.LayoutInflater;
import android.view.Menu; import android.view.Menu;
import android.view.MenuInflater; import android.view.MenuInflater;
@@ -133,7 +137,6 @@ import com.android.settings.widget.PieChartView;
import com.google.android.collect.Lists; import com.google.android.collect.Lists;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections; import java.util.Collections;
import java.util.List; import java.util.List;
import java.util.Locale; import java.util.Locale;
@@ -230,7 +233,7 @@ public class DataUsageSummary extends Fragment {
private NetworkTemplate mTemplate; private NetworkTemplate mTemplate;
private ChartData mChartData; private ChartData mChartData;
private int[] mAppDetailUids = null; private AppItem mCurrentApp = null;
private Intent mAppSettingsIntent; private Intent mAppSettingsIntent;
@@ -684,7 +687,7 @@ public class DataUsageSummary extends Fragment {
// TODO: consider chaining two loaders together instead of reloading // TODO: consider chaining two loaders together instead of reloading
// network history when showing app detail. // network history when showing app detail.
getLoaderManager().restartLoader(LOADER_CHART_DATA, getLoaderManager().restartLoader(LOADER_CHART_DATA,
ChartDataLoader.buildArgs(mTemplate, mAppDetailUids), mChartDataCallbacks); ChartDataLoader.buildArgs(mTemplate, mCurrentApp), mChartDataCallbacks);
// detail mode can change visible menus, invalidate // detail mode can change visible menus, invalidate
getActivity().invalidateOptionsMenu(); getActivity().invalidateOptionsMenu();
@@ -693,15 +696,11 @@ public class DataUsageSummary extends Fragment {
} }
private boolean isAppDetailMode() { private boolean isAppDetailMode() {
return mAppDetailUids != null; return mCurrentApp != null;
}
private int getAppDetailPrimaryUid() {
return mAppDetailUids[0];
} }
/** /**
* Update UID details panels to match {@link #mAppDetailUids}, showing or * Update UID details panels to match {@link #mCurrentApp}, showing or
* hiding them depending on {@link #isAppDetailMode()}. * hiding them depending on {@link #isAppDetailMode()}.
*/ */
private void updateAppDetail() { private void updateAppDetail() {
@@ -725,8 +724,8 @@ public class DataUsageSummary extends Fragment {
mChart.bindNetworkPolicy(null); mChart.bindNetworkPolicy(null);
// show icon and all labels appearing under this app // show icon and all labels appearing under this app
final int primaryUid = getAppDetailPrimaryUid(); final int appId = mCurrentApp.appId;
final UidDetail detail = mUidDetailProvider.getUidDetail(primaryUid, true); final UidDetail detail = mUidDetailProvider.getUidDetail(appId, true);
mAppIcon.setImageDrawable(detail.icon); mAppIcon.setImageDrawable(detail.icon);
mAppTitles.removeAllViews(); mAppTitles.removeAllViews();
@@ -740,7 +739,7 @@ public class DataUsageSummary extends Fragment {
// enable settings button when package provides it // enable settings button when package provides it
// TODO: target torwards entire UID instead of just first package // TODO: target torwards entire UID instead of just first package
final String[] packageNames = pm.getPackagesForUid(primaryUid); final String[] packageNames = pm.getPackagesForUid(appId);
if (packageNames != null && packageNames.length > 0) { if (packageNames != null && packageNames.length > 0) {
mAppSettingsIntent = new Intent(Intent.ACTION_MANAGE_NETWORK_USAGE); mAppSettingsIntent = new Intent(Intent.ACTION_MANAGE_NETWORK_USAGE);
mAppSettingsIntent.setPackage(packageNames[0]); mAppSettingsIntent.setPackage(packageNames[0]);
@@ -756,8 +755,7 @@ public class DataUsageSummary extends Fragment {
updateDetailData(); updateDetailData();
if (NetworkPolicyManager.isUidValidForPolicy(context, primaryUid) if (UserId.isApp(appId) && !getRestrictBackground() && isBandwidthControlEnabled()
&& !getRestrictBackground() && isBandwidthControlEnabled()
&& hasMobileRadio(context)) { && hasMobileRadio(context)) {
setPreferenceTitle(mAppRestrictView, R.string.data_usage_app_restrict_background); setPreferenceTitle(mAppRestrictView, R.string.data_usage_app_restrict_background);
setPreferenceSummary(mAppRestrictView, setPreferenceSummary(mAppRestrictView,
@@ -851,10 +849,10 @@ public class DataUsageSummary extends Fragment {
} }
private boolean getAppRestrictBackground() { private boolean getAppRestrictBackground() {
final int primaryUid = getAppDetailPrimaryUid(); final int appId = mCurrentApp.appId;
final int uidPolicy; final int uidPolicy;
try { try {
uidPolicy = mPolicyService.getUidPolicy(primaryUid); uidPolicy = mPolicyService.getAppPolicy(appId);
} catch (RemoteException e) { } catch (RemoteException e) {
// since we can't do much without policy, we bail hard. // since we can't do much without policy, we bail hard.
throw new RuntimeException("problem reading network policy", e); throw new RuntimeException("problem reading network policy", e);
@@ -865,9 +863,9 @@ public class DataUsageSummary extends Fragment {
private void setAppRestrictBackground(boolean restrictBackground) { private void setAppRestrictBackground(boolean restrictBackground) {
if (LOGD) Log.d(TAG, "setAppRestrictBackground()"); if (LOGD) Log.d(TAG, "setAppRestrictBackground()");
final int primaryUid = getAppDetailPrimaryUid(); final int appId = mCurrentApp.appId;
try { try {
mPolicyService.setUidPolicy(primaryUid, mPolicyService.setAppPolicy(appId,
restrictBackground ? POLICY_REJECT_METERED_BACKGROUND : POLICY_NONE); restrictBackground ? POLICY_REJECT_METERED_BACKGROUND : POLICY_NONE);
} catch (RemoteException e) { } catch (RemoteException e) {
throw new RuntimeException("unable to save policy", e); throw new RuntimeException("unable to save policy", e);
@@ -1050,9 +1048,9 @@ public class DataUsageSummary extends Fragment {
/** {@inheritDoc} */ /** {@inheritDoc} */
public void onItemClick(AdapterView<?> parent, View view, int position, long id) { public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
final Context context = view.getContext(); final Context context = view.getContext();
final AppUsageItem app = (AppUsageItem) parent.getItemAtPosition(position); final AppItem app = (AppItem) parent.getItemAtPosition(position);
final UidDetail detail = mUidDetailProvider.getUidDetail(app.uids[0], true); final UidDetail detail = mUidDetailProvider.getUidDetail(app.appId, true);
AppDetailsFragment.show(DataUsageSummary.this, app.uids, detail.label); AppDetailsFragment.show(DataUsageSummary.this, app, detail.label);
} }
}; };
@@ -1362,25 +1360,53 @@ public class DataUsageSummary extends Fragment {
} }
} }
private static class AppUsageItem implements Comparable<AppUsageItem> { public static class AppItem implements Comparable<AppItem>, Parcelable {
public int[] uids; public final int appId;
public SparseBooleanArray uids = new SparseBooleanArray();
public long total; public long total;
public AppUsageItem(int uid) { public AppItem(int appId) {
uids = new int[] { uid }; this.appId = appId;
}
public AppItem(Parcel parcel) {
appId = parcel.readInt();
uids = parcel.readSparseBooleanArray();
total = parcel.readLong();
} }
public void addUid(int uid) { public void addUid(int uid) {
if (contains(uids, uid)) return; uids.put(uid, true);
final int length = uids.length;
uids = Arrays.copyOf(uids, length + 1);
uids[length] = uid;
} }
/** {@inheritDoc} */ @Override
public int compareTo(AppUsageItem another) { public void writeToParcel(Parcel dest, int flags) {
dest.writeInt(appId);
dest.writeSparseBooleanArray(uids);
dest.writeLong(total);
}
@Override
public int describeContents() {
return 0;
}
@Override
public int compareTo(AppItem another) {
return Long.compare(another.total, total); return Long.compare(another.total, total);
} }
public static final Creator<AppItem> CREATOR = new Creator<AppItem>() {
@Override
public AppItem createFromParcel(Parcel in) {
return new AppItem(in);
}
@Override
public AppItem[] newArray(int size) {
return new AppItem[size];
}
};
} }
/** /**
@@ -1390,7 +1416,7 @@ public class DataUsageSummary extends Fragment {
private final UidDetailProvider mProvider; private final UidDetailProvider mProvider;
private final int mInsetSide; private final int mInsetSide;
private ArrayList<AppUsageItem> mItems = Lists.newArrayList(); private ArrayList<AppItem> mItems = Lists.newArrayList();
private long mLargest; private long mLargest;
public DataUsageAdapter(UidDetailProvider provider, int insetSide) { public DataUsageAdapter(UidDetailProvider provider, int insetSide) {
@@ -1404,29 +1430,29 @@ public class DataUsageSummary extends Fragment {
public void bindStats(NetworkStats stats) { public void bindStats(NetworkStats stats) {
mItems.clear(); mItems.clear();
final AppUsageItem systemItem = new AppUsageItem(android.os.Process.SYSTEM_UID); final AppItem systemItem = new AppItem(android.os.Process.SYSTEM_UID);
final SparseArray<AppUsageItem> knownUids = new SparseArray<AppUsageItem>(); final SparseArray<AppItem> knownUids = new SparseArray<AppItem>();
NetworkStats.Entry entry = null; NetworkStats.Entry entry = null;
final int size = stats != null ? stats.size() : 0; final int size = stats != null ? stats.size() : 0;
for (int i = 0; i < size; i++) { for (int i = 0; i < size; i++) {
entry = stats.getValues(i, entry); entry = stats.getValues(i, entry);
final int uid = entry.uid; final boolean isApp = UserId.isApp(entry.uid);
final boolean isApp = uid >= android.os.Process.FIRST_APPLICATION_UID final int appId = isApp ? UserId.getAppId(entry.uid) : entry.uid;
&& uid <= android.os.Process.LAST_APPLICATION_UID; if (isApp || appId == UID_REMOVED || appId == UID_TETHERING) {
if (isApp || uid == UID_REMOVED || uid == UID_TETHERING) { AppItem item = knownUids.get(appId);
AppUsageItem item = knownUids.get(uid);
if (item == null) { if (item == null) {
item = new AppUsageItem(uid); item = new AppItem(appId);
knownUids.put(uid, item); knownUids.put(appId, item);
mItems.add(item); mItems.add(item);
} }
item.total += entry.rxBytes + entry.txBytes; item.total += entry.rxBytes + entry.txBytes;
item.addUid(entry.uid);
} else { } else {
systemItem.total += entry.rxBytes + entry.txBytes; systemItem.total += entry.rxBytes + entry.txBytes;
systemItem.addUid(uid); systemItem.addUid(entry.uid);
} }
} }
@@ -1451,7 +1477,7 @@ public class DataUsageSummary extends Fragment {
@Override @Override
public long getItemId(int position) { public long getItemId(int position) {
return mItems.get(position).uids[0]; return mItems.get(position).appId;
} }
@Override @Override
@@ -1472,7 +1498,7 @@ public class DataUsageSummary extends Fragment {
android.R.id.progress); android.R.id.progress);
// kick off async load of app details // kick off async load of app details
final AppUsageItem item = mItems.get(position); final AppItem item = mItems.get(position);
UidDetailTask.bindView(mProvider, item, convertView); UidDetailTask.bindView(mProvider, item, convertView);
text1.setText(Formatter.formatFileSize(context, item.total)); text1.setText(Formatter.formatFileSize(context, item.total));
@@ -1489,13 +1515,13 @@ public class DataUsageSummary extends Fragment {
* {@link DataUsageSummary}. * {@link DataUsageSummary}.
*/ */
public static class AppDetailsFragment extends Fragment { public static class AppDetailsFragment extends Fragment {
private static final String EXTRA_UIDS = "uids"; private static final String EXTRA_APP = "app";
public static void show(DataUsageSummary parent, int[] uids, CharSequence label) { public static void show(DataUsageSummary parent, AppItem app, CharSequence label) {
if (!parent.isAdded()) return; if (!parent.isAdded()) return;
final Bundle args = new Bundle(); final Bundle args = new Bundle();
args.putIntArray(EXTRA_UIDS, uids); args.putParcelable(EXTRA_APP, app);
final AppDetailsFragment fragment = new AppDetailsFragment(); final AppDetailsFragment fragment = new AppDetailsFragment();
fragment.setArguments(args); fragment.setArguments(args);
@@ -1511,7 +1537,7 @@ public class DataUsageSummary extends Fragment {
public void onStart() { public void onStart() {
super.onStart(); super.onStart();
final DataUsageSummary target = (DataUsageSummary) getTargetFragment(); final DataUsageSummary target = (DataUsageSummary) getTargetFragment();
target.mAppDetailUids = getArguments().getIntArray(EXTRA_UIDS); target.mCurrentApp = getArguments().getParcelable(EXTRA_APP);
target.updateBody(); target.updateBody();
} }
@@ -1519,7 +1545,7 @@ public class DataUsageSummary extends Fragment {
public void onStop() { public void onStop() {
super.onStop(); super.onStop();
final DataUsageSummary target = (DataUsageSummary) getTargetFragment(); final DataUsageSummary target = (DataUsageSummary) getTargetFragment();
target.mAppDetailUids = null; target.mCurrentApp = null;
target.updateBody(); target.updateBody();
} }
} }
@@ -1967,23 +1993,23 @@ public class DataUsageSummary extends Fragment {
*/ */
private static class UidDetailTask extends AsyncTask<Void, Void, UidDetail> { private static class UidDetailTask extends AsyncTask<Void, Void, UidDetail> {
private final UidDetailProvider mProvider; private final UidDetailProvider mProvider;
private final AppUsageItem mItem; private final AppItem mItem;
private final View mTarget; private final View mTarget;
private UidDetailTask(UidDetailProvider provider, AppUsageItem item, View target) { private UidDetailTask(UidDetailProvider provider, AppItem item, View target) {
mProvider = checkNotNull(provider); mProvider = checkNotNull(provider);
mItem = checkNotNull(item); mItem = checkNotNull(item);
mTarget = checkNotNull(target); mTarget = checkNotNull(target);
} }
public static void bindView( public static void bindView(
UidDetailProvider provider, AppUsageItem item, View target) { UidDetailProvider provider, AppItem item, View target) {
final UidDetailTask existing = (UidDetailTask) target.getTag(); final UidDetailTask existing = (UidDetailTask) target.getTag();
if (existing != null) { if (existing != null) {
existing.cancel(false); existing.cancel(false);
} }
final UidDetail cachedDetail = provider.getUidDetail(item.uids[0], false); final UidDetail cachedDetail = provider.getUidDetail(item.appId, false);
if (cachedDetail != null) { if (cachedDetail != null) {
bindView(cachedDetail, target); bindView(cachedDetail, target);
} else { } else {
@@ -2012,7 +2038,7 @@ public class DataUsageSummary extends Fragment {
@Override @Override
protected UidDetail doInBackground(Void... params) { protected UidDetail doInBackground(Void... params) {
return mProvider.getUidDetail(mItem.uids[0], true); return mProvider.getUidDetail(mItem.appId, true);
} }
@Override @Override
@@ -2188,13 +2214,4 @@ public class DataUsageSummary extends Fragment {
summary.setVisibility(View.VISIBLE); summary.setVisibility(View.VISIBLE);
summary.setText(string); summary.setText(string);
} }
private static boolean contains(int[] haystack, int needle) {
for (int value : haystack) {
if (value == needle) {
return true;
}
}
return false;
}
} }

View File

@@ -30,25 +30,27 @@ import android.net.NetworkTemplate;
import android.os.Bundle; import android.os.Bundle;
import android.os.RemoteException; import android.os.RemoteException;
import com.android.settings.DataUsageSummary.AppItem;
/** /**
* Loader for historical chart data for both network and UID details. * Loader for historical chart data for both network and UID details.
*/ */
public class ChartDataLoader extends AsyncTaskLoader<ChartData> { public class ChartDataLoader extends AsyncTaskLoader<ChartData> {
private static final String KEY_TEMPLATE = "template"; private static final String KEY_TEMPLATE = "template";
private static final String KEY_UIDS = "uids"; private static final String KEY_APP = "app";
private static final String KEY_FIELDS = "fields"; private static final String KEY_FIELDS = "fields";
private final INetworkStatsService mStatsService; private final INetworkStatsService mStatsService;
private final Bundle mArgs; private final Bundle mArgs;
public static Bundle buildArgs(NetworkTemplate template, int[] uids) { public static Bundle buildArgs(NetworkTemplate template, AppItem app) {
return buildArgs(template, uids, FIELD_RX_BYTES | FIELD_TX_BYTES); return buildArgs(template, app, FIELD_RX_BYTES | FIELD_TX_BYTES);
} }
public static Bundle buildArgs(NetworkTemplate template, int[] uids, int fields) { public static Bundle buildArgs(NetworkTemplate template, AppItem app, int fields) {
final Bundle args = new Bundle(); final Bundle args = new Bundle();
args.putParcelable(KEY_TEMPLATE, template); args.putParcelable(KEY_TEMPLATE, template);
args.putIntArray(KEY_UIDS, uids); args.putParcelable(KEY_APP, app);
args.putInt(KEY_FIELDS, fields); args.putInt(KEY_FIELDS, fields);
return args; return args;
} }
@@ -68,11 +70,11 @@ public class ChartDataLoader extends AsyncTaskLoader<ChartData> {
@Override @Override
public ChartData loadInBackground() { public ChartData loadInBackground() {
final NetworkTemplate template = mArgs.getParcelable(KEY_TEMPLATE); final NetworkTemplate template = mArgs.getParcelable(KEY_TEMPLATE);
final int[] uids = mArgs.getIntArray(KEY_UIDS); final AppItem app = mArgs.getParcelable(KEY_APP);
final int fields = mArgs.getInt(KEY_FIELDS); final int fields = mArgs.getInt(KEY_FIELDS);
try { try {
return loadInBackground(template, uids, fields); return loadInBackground(template, app, fields);
} catch (RemoteException e) { } catch (RemoteException e) {
// since we can't do much without history, and we don't want to // since we can't do much without history, and we don't want to
// leave with half-baked UI, we bail hard. // leave with half-baked UI, we bail hard.
@@ -80,17 +82,19 @@ public class ChartDataLoader extends AsyncTaskLoader<ChartData> {
} }
} }
private ChartData loadInBackground(NetworkTemplate template, int[] uids, int fields) private ChartData loadInBackground(NetworkTemplate template, AppItem app, int fields)
throws RemoteException { throws RemoteException {
final ChartData data = new ChartData(); final ChartData data = new ChartData();
data.network = mStatsService.getHistoryForNetwork(template, fields); data.network = mStatsService.getHistoryForNetwork(template, fields);
if (uids != null) { if (app != null) {
data.detailDefault = null; data.detailDefault = null;
data.detailForeground = null; data.detailForeground = null;
// load stats for current uid and template // load stats for current uid and template
for (int uid : uids) { final int size = app.uids.size();
for (int i = 0; i < size; i++) {
final int uid = app.uids.keyAt(i);
data.detailDefault = collectHistoryForUid( data.detailDefault = collectHistoryForUid(
template, uid, SET_DEFAULT, data.detailDefault); template, uid, SET_DEFAULT, data.detailDefault);
data.detailForeground = collectHistoryForUid( data.detailForeground = collectHistoryForUid(