blob: 69ac1e71f50ef1136dbbf2bbb3fbc8924cd338d2 [file] [log] [blame]
Jeff Sharkey9a13f362011-04-26 16:25:36 -07001/*
2 * Copyright (C) 2011 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
17package android.net;
18
19import android.os.Parcel;
20import android.os.Parcelable;
21import android.os.SystemClock;
Jeff Sharkey47eb1022011-08-25 17:48:52 -070022import android.util.Log;
Jeff Sharkey61ee0bb2011-05-29 22:50:42 -070023import android.util.SparseBooleanArray;
Jeff Sharkey9a13f362011-04-26 16:25:36 -070024
Jeff Sharkeya63ba592011-07-19 23:47:12 -070025import com.android.internal.util.Objects;
26
Jeff Sharkey9a13f362011-04-26 16:25:36 -070027import java.io.CharArrayWriter;
28import java.io.PrintWriter;
Jeff Sharkey4a971222011-06-11 22:16:55 -070029import java.util.Arrays;
Jeff Sharkey75279902011-05-24 18:39:45 -070030import java.util.HashSet;
Jeff Sharkey9a13f362011-04-26 16:25:36 -070031
32/**
Jeff Sharkey75279902011-05-24 18:39:45 -070033 * Collection of active network statistics. Can contain summary details across
34 * all interfaces, or details with per-UID granularity. Internally stores data
35 * as a large table, closely matching {@code /proc/} data format. This structure
36 * optimizes for rapid in-memory comparison, but consider using
37 * {@link NetworkStatsHistory} when persisting.
Jeff Sharkey9a13f362011-04-26 16:25:36 -070038 *
39 * @hide
40 */
41public class NetworkStats implements Parcelable {
Jeff Sharkey47eb1022011-08-25 17:48:52 -070042 private static final String TAG = "NetworkStats";
43
Jeff Sharkey75279902011-05-24 18:39:45 -070044 /** {@link #iface} value when interface details unavailable. */
Jeff Sharkey9a13f362011-04-26 16:25:36 -070045 public static final String IFACE_ALL = null;
Jeff Sharkey75279902011-05-24 18:39:45 -070046 /** {@link #uid} value when UID details unavailable. */
47 public static final int UID_ALL = -1;
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -070048 /** {@link #set} value when all sets combined. */
49 public static final int SET_ALL = -1;
50 /** {@link #set} value where background data is accounted. */
51 public static final int SET_DEFAULT = 0;
52 /** {@link #set} value where foreground data is accounted. */
53 public static final int SET_FOREGROUND = 1;
54 /** {@link #tag} value for total data across all tags. */
Jeff Sharkey1b5a2a92011-06-18 18:34:16 -070055 public static final int TAG_NONE = 0;
Jeff Sharkeyeedcb952011-05-17 14:55:15 -070056
Jeff Sharkey9a13f362011-04-26 16:25:36 -070057 /**
58 * {@link SystemClock#elapsedRealtime()} timestamp when this data was
59 * generated.
60 */
Jeff Sharkeyd37948f2011-07-12 13:57:00 -070061 private final long elapsedRealtime;
62 private int size;
63 private String[] iface;
64 private int[] uid;
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -070065 private int[] set;
Jeff Sharkeyd37948f2011-07-12 13:57:00 -070066 private int[] tag;
67 private long[] rxBytes;
68 private long[] rxPackets;
69 private long[] txBytes;
70 private long[] txPackets;
Jeff Sharkey63d27a92011-08-03 17:04:22 -070071 private long[] operations;
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -070072
73 public static class Entry {
74 public String iface;
75 public int uid;
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -070076 public int set;
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -070077 public int tag;
78 public long rxBytes;
79 public long rxPackets;
80 public long txBytes;
81 public long txPackets;
Jeff Sharkey63d27a92011-08-03 17:04:22 -070082 public long operations;
Jeff Sharkeyd37948f2011-07-12 13:57:00 -070083
84 public Entry() {
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -070085 this(IFACE_ALL, UID_ALL, SET_DEFAULT, TAG_NONE, 0L, 0L, 0L, 0L, 0L);
Jeff Sharkey63d27a92011-08-03 17:04:22 -070086 }
87
88 public Entry(long rxBytes, long rxPackets, long txBytes, long txPackets, long operations) {
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -070089 this(IFACE_ALL, UID_ALL, SET_DEFAULT, TAG_NONE, rxBytes, rxPackets, txBytes, txPackets,
90 operations);
Jeff Sharkeyd37948f2011-07-12 13:57:00 -070091 }
92
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -070093 public Entry(String iface, int uid, int set, int tag, long rxBytes, long rxPackets,
94 long txBytes, long txPackets, long operations) {
Jeff Sharkeyd37948f2011-07-12 13:57:00 -070095 this.iface = iface;
96 this.uid = uid;
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -070097 this.set = set;
Jeff Sharkeyd37948f2011-07-12 13:57:00 -070098 this.tag = tag;
99 this.rxBytes = rxBytes;
100 this.rxPackets = rxPackets;
101 this.txBytes = txBytes;
102 this.txPackets = txPackets;
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700103 this.operations = operations;
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700104 }
Jeff Sharkeyb3d59572011-09-07 17:20:27 -0700105
106 @Override
107 public String toString() {
108 final StringBuilder builder = new StringBuilder();
109 builder.append("iface=").append(iface);
110 builder.append(" uid=").append(uid);
111 builder.append(" set=").append(setToString(set));
112 builder.append(" tag=").append(tagToString(tag));
113 builder.append(" rxBytes=").append(rxBytes);
114 builder.append(" rxPackets=").append(rxPackets);
115 builder.append(" txBytes=").append(txBytes);
116 builder.append(" txPackets=").append(txPackets);
117 builder.append(" operations=").append(operations);
118 return builder.toString();
119 }
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700120 }
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700121
Jeff Sharkey4a971222011-06-11 22:16:55 -0700122 public NetworkStats(long elapsedRealtime, int initialSize) {
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700123 this.elapsedRealtime = elapsedRealtime;
Jeff Sharkey4a971222011-06-11 22:16:55 -0700124 this.size = 0;
125 this.iface = new String[initialSize];
126 this.uid = new int[initialSize];
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700127 this.set = new int[initialSize];
Jeff Sharkey1b5a2a92011-06-18 18:34:16 -0700128 this.tag = new int[initialSize];
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700129 this.rxBytes = new long[initialSize];
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700130 this.rxPackets = new long[initialSize];
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700131 this.txBytes = new long[initialSize];
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700132 this.txPackets = new long[initialSize];
Jeff Sharkey63d27a92011-08-03 17:04:22 -0700133 this.operations = new long[initialSize];
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700134 }
135
136 public NetworkStats(Parcel parcel) {
137 elapsedRealtime = parcel.readLong();
Jeff Sharkey4a971222011-06-11 22:16:55 -0700138 size = parcel.readInt();
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700139 iface = parcel.createStringArray();
140 uid = parcel.createIntArray();
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700141 set = parcel.createIntArray();
Jeff Sharkey1b5a2a92011-06-18 18:34:16 -0700142 tag = parcel.createIntArray();
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700143 rxBytes = parcel.createLongArray();
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700144 rxPackets = parcel.createLongArray();
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700145 txBytes = parcel.createLongArray();
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700146 txPackets = parcel.createLongArray();
Jeff Sharkey63d27a92011-08-03 17:04:22 -0700147 operations = parcel.createLongArray();
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700148 }
149
150 /** {@inheritDoc} */
151 public void writeToParcel(Parcel dest, int flags) {
152 dest.writeLong(elapsedRealtime);
153 dest.writeInt(size);
154 dest.writeStringArray(iface);
155 dest.writeIntArray(uid);
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700156 dest.writeIntArray(set);
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700157 dest.writeIntArray(tag);
158 dest.writeLongArray(rxBytes);
159 dest.writeLongArray(rxPackets);
160 dest.writeLongArray(txBytes);
161 dest.writeLongArray(txPackets);
Jeff Sharkey63d27a92011-08-03 17:04:22 -0700162 dest.writeLongArray(operations);
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700163 }
164
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700165 // @VisibleForTesting
166 public NetworkStats addIfaceValues(
167 String iface, long rxBytes, long rxPackets, long txBytes, long txPackets) {
168 return addValues(
169 iface, UID_ALL, SET_DEFAULT, TAG_NONE, rxBytes, rxPackets, txBytes, txPackets, 0L);
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700170 }
171
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700172 // @VisibleForTesting
173 public NetworkStats addValues(String iface, int uid, int set, int tag, long rxBytes,
174 long rxPackets, long txBytes, long txPackets, long operations) {
175 return addValues(new Entry(
176 iface, uid, set, tag, rxBytes, rxPackets, txBytes, txPackets, operations));
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700177 }
178
179 /**
180 * Add new stats entry, copying from given {@link Entry}. The {@link Entry}
181 * object can be recycled across multiple calls.
182 */
183 public NetworkStats addValues(Entry entry) {
Jeff Sharkey4a971222011-06-11 22:16:55 -0700184 if (size >= this.iface.length) {
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700185 final int newLength = Math.max(iface.length, 10) * 3 / 2;
186 iface = Arrays.copyOf(iface, newLength);
187 uid = Arrays.copyOf(uid, newLength);
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700188 set = Arrays.copyOf(set, newLength);
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700189 tag = Arrays.copyOf(tag, newLength);
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700190 rxBytes = Arrays.copyOf(rxBytes, newLength);
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700191 rxPackets = Arrays.copyOf(rxPackets, newLength);
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700192 txBytes = Arrays.copyOf(txBytes, newLength);
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700193 txPackets = Arrays.copyOf(txPackets, newLength);
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700194 operations = Arrays.copyOf(operations, newLength);
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700195 }
196
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700197 iface[size] = entry.iface;
198 uid[size] = entry.uid;
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700199 set[size] = entry.set;
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700200 tag[size] = entry.tag;
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700201 rxBytes[size] = entry.rxBytes;
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700202 rxPackets[size] = entry.rxPackets;
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700203 txBytes[size] = entry.txBytes;
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700204 txPackets[size] = entry.txPackets;
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700205 operations[size] = entry.operations;
Jeff Sharkey4a971222011-06-11 22:16:55 -0700206 size++;
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700207
Jeff Sharkey4a971222011-06-11 22:16:55 -0700208 return this;
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700209 }
210
Jeff Sharkey1b5a2a92011-06-18 18:34:16 -0700211 /**
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700212 * Return specific stats entry.
213 */
214 public Entry getValues(int i, Entry recycle) {
215 final Entry entry = recycle != null ? recycle : new Entry();
216 entry.iface = iface[i];
217 entry.uid = uid[i];
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700218 entry.set = set[i];
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700219 entry.tag = tag[i];
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700220 entry.rxBytes = rxBytes[i];
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700221 entry.rxPackets = rxPackets[i];
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700222 entry.txBytes = txBytes[i];
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700223 entry.txPackets = txPackets[i];
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700224 entry.operations = operations[i];
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700225 return entry;
226 }
227
228 public long getElapsedRealtime() {
229 return elapsedRealtime;
230 }
231
Jeff Sharkey1059c3c2011-10-04 16:54:49 -0700232 /**
233 * Return age of this {@link NetworkStats} object with respect to
234 * {@link SystemClock#elapsedRealtime()}.
235 */
236 public long getElapsedRealtimeAge() {
237 return SystemClock.elapsedRealtime() - elapsedRealtime;
238 }
239
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700240 public int size() {
241 return size;
242 }
243
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700244 // @VisibleForTesting
245 public int internalSize() {
246 return iface.length;
247 }
248
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700249 @Deprecated
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700250 public NetworkStats combineValues(String iface, int uid, int tag, long rxBytes, long rxPackets,
Jeff Sharkey63d27a92011-08-03 17:04:22 -0700251 long txBytes, long txPackets, long operations) {
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700252 return combineValues(
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700253 iface, uid, SET_DEFAULT, tag, rxBytes, rxPackets, txBytes, txPackets, operations);
254 }
255
256 public NetworkStats combineValues(String iface, int uid, int set, int tag, long rxBytes,
257 long rxPackets, long txBytes, long txPackets, long operations) {
258 return combineValues(new Entry(
259 iface, uid, set, tag, rxBytes, rxPackets, txBytes, txPackets, operations));
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700260 }
261
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700262 /**
Jeff Sharkey1b5a2a92011-06-18 18:34:16 -0700263 * Combine given values with an existing row, or create a new row if
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700264 * {@link #findIndex(String, int, int, int)} is unable to find match. Can
265 * also be used to subtract values from existing rows.
Jeff Sharkey1b5a2a92011-06-18 18:34:16 -0700266 */
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700267 public NetworkStats combineValues(Entry entry) {
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700268 final int i = findIndex(entry.iface, entry.uid, entry.set, entry.tag);
Jeff Sharkey1b5a2a92011-06-18 18:34:16 -0700269 if (i == -1) {
270 // only create new entry when positive contribution
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700271 addValues(entry);
Jeff Sharkey1b5a2a92011-06-18 18:34:16 -0700272 } else {
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700273 rxBytes[i] += entry.rxBytes;
274 rxPackets[i] += entry.rxPackets;
275 txBytes[i] += entry.txBytes;
276 txPackets[i] += entry.txPackets;
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700277 operations[i] += entry.operations;
Jeff Sharkey1b5a2a92011-06-18 18:34:16 -0700278 }
279 return this;
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700280 }
281
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700282 /**
Jeff Sharkey905b5892011-09-30 15:19:49 -0700283 * Combine all values from another {@link NetworkStats} into this object.
284 */
285 public void combineAllValues(NetworkStats another) {
286 NetworkStats.Entry entry = null;
287 for (int i = 0; i < another.size; i++) {
288 entry = another.getValues(i, entry);
289 combineValues(entry);
290 }
291 }
292
293 /**
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700294 * Find first stats index that matches the requested parameters.
295 */
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700296 public int findIndex(String iface, int uid, int set, int tag) {
Jeff Sharkey4a971222011-06-11 22:16:55 -0700297 for (int i = 0; i < size; i++) {
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700298 if (Objects.equal(iface, this.iface[i]) && uid == this.uid[i] && set == this.set[i]
299 && tag == this.tag[i]) {
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700300 return i;
301 }
302 }
303 return -1;
304 }
305
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700306 /**
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700307 * Splice in {@link #operations} from the given {@link NetworkStats} based
308 * on matching {@link #uid} and {@link #tag} rows. Ignores {@link #iface},
309 * since operation counts are at data layer.
310 */
311 public void spliceOperationsFrom(NetworkStats stats) {
312 for (int i = 0; i < size; i++) {
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700313 final int j = stats.findIndex(IFACE_ALL, uid[i], set[i], tag[i]);
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700314 if (j == -1) {
315 operations[i] = 0;
316 } else {
317 operations[i] = stats.operations[j];
318 }
319 }
320 }
321
322 /**
Jeff Sharkey75279902011-05-24 18:39:45 -0700323 * Return list of unique interfaces known by this data structure.
324 */
Jeff Sharkey61ee0bb2011-05-29 22:50:42 -0700325 public String[] getUniqueIfaces() {
Jeff Sharkey75279902011-05-24 18:39:45 -0700326 final HashSet<String> ifaces = new HashSet<String>();
327 for (String iface : this.iface) {
328 if (iface != IFACE_ALL) {
329 ifaces.add(iface);
330 }
331 }
332 return ifaces.toArray(new String[ifaces.size()]);
333 }
334
335 /**
Jeff Sharkey61ee0bb2011-05-29 22:50:42 -0700336 * Return list of unique UIDs known by this data structure.
337 */
338 public int[] getUniqueUids() {
339 final SparseBooleanArray uids = new SparseBooleanArray();
340 for (int uid : this.uid) {
341 uids.put(uid, true);
342 }
343
344 final int size = uids.size();
345 final int[] result = new int[size];
346 for (int i = 0; i < size; i++) {
347 result[i] = uids.keyAt(i);
348 }
349 return result;
350 }
351
352 /**
Jeff Sharkey8e9992a2011-08-23 18:37:23 -0700353 * Return total bytes represented by this snapshot object, usually used when
354 * checking if a {@link #subtract(NetworkStats)} delta passes a threshold.
355 */
356 public long getTotalBytes() {
Jeff Sharkey07b0dd92011-09-01 13:06:19 -0700357 final Entry entry = getTotal(null);
358 return entry.rxBytes + entry.txBytes;
359 }
360
361 /**
362 * Return total of all fields represented by this snapshot object.
363 */
364 public Entry getTotal(Entry recycle) {
Jeff Sharkey1059c3c2011-10-04 16:54:49 -0700365 return getTotal(recycle, null, UID_ALL);
366 }
367
368 /**
369 * Return total of all fields represented by this snapshot object matching
370 * the requested {@link #uid}.
371 */
372 public Entry getTotal(Entry recycle, int limitUid) {
373 return getTotal(recycle, null, limitUid);
374 }
375
376 /**
377 * Return total of all fields represented by this snapshot object matching
378 * the requested {@link #iface}.
379 */
380 public Entry getTotal(Entry recycle, HashSet<String> limitIface) {
381 return getTotal(recycle, limitIface, UID_ALL);
382 }
383
384 /**
385 * Return total of all fields represented by this snapshot object matching
386 * the requested {@link #iface} and {@link #uid}.
387 *
388 * @param limitIface Set of {@link #iface} to include in total; or {@code
389 * null} to include all ifaces.
390 */
391 private Entry getTotal(Entry recycle, HashSet<String> limitIface, int limitUid) {
Jeff Sharkey07b0dd92011-09-01 13:06:19 -0700392 final Entry entry = recycle != null ? recycle : new Entry();
393
394 entry.iface = IFACE_ALL;
Jeff Sharkey1059c3c2011-10-04 16:54:49 -0700395 entry.uid = limitUid;
Jeff Sharkey07b0dd92011-09-01 13:06:19 -0700396 entry.set = SET_ALL;
397 entry.tag = TAG_NONE;
398 entry.rxBytes = 0;
399 entry.rxPackets = 0;
400 entry.txBytes = 0;
401 entry.txPackets = 0;
Jeff Sharkey1059c3c2011-10-04 16:54:49 -0700402 entry.operations = 0;
Jeff Sharkey07b0dd92011-09-01 13:06:19 -0700403
Jeff Sharkey8e9992a2011-08-23 18:37:23 -0700404 for (int i = 0; i < size; i++) {
Jeff Sharkey1059c3c2011-10-04 16:54:49 -0700405 final boolean matchesUid = (limitUid == UID_ALL) || (limitUid == uid[i]);
406 final boolean matchesIface = (limitIface == null) || (limitIface.contains(iface[i]));
Jeff Sharkey8e9992a2011-08-23 18:37:23 -0700407
Jeff Sharkey1059c3c2011-10-04 16:54:49 -0700408 if (matchesUid && matchesIface) {
409 // skip specific tags, since already counted in TAG_NONE
410 if (tag[i] != TAG_NONE) continue;
411
412 entry.rxBytes += rxBytes[i];
413 entry.rxPackets += rxPackets[i];
414 entry.txBytes += txBytes[i];
415 entry.txPackets += txPackets[i];
416 entry.operations += operations[i];
417 }
Jeff Sharkey8e9992a2011-08-23 18:37:23 -0700418 }
Jeff Sharkey07b0dd92011-09-01 13:06:19 -0700419 return entry;
Jeff Sharkey8e9992a2011-08-23 18:37:23 -0700420 }
421
422 /**
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700423 * Subtract the given {@link NetworkStats}, effectively leaving the delta
424 * between two snapshots in time. Assumes that statistics rows collect over
425 * time, and that none of them have disappeared.
Jeff Sharkey75279902011-05-24 18:39:45 -0700426 *
Jeff Sharkey3f391352011-06-05 17:42:53 -0700427 * @throws IllegalArgumentException when given {@link NetworkStats} is
428 * non-monotonic.
429 */
430 public NetworkStats subtract(NetworkStats value) {
431 return subtract(value, true, false);
432 }
433
434 /**
435 * Subtract the given {@link NetworkStats}, effectively leaving the delta
436 * between two snapshots in time. Assumes that statistics rows collect over
437 * time, and that none of them have disappeared.
438 * <p>
439 * Instead of throwing when counters are non-monotonic, this variant clamps
440 * results to never be negative.
441 */
442 public NetworkStats subtractClamped(NetworkStats value) {
443 return subtract(value, false, true);
444 }
445
446 /**
447 * Subtract the given {@link NetworkStats}, effectively leaving the delta
448 * between two snapshots in time. Assumes that statistics rows collect over
449 * time, and that none of them have disappeared.
450 *
Jeff Sharkey75279902011-05-24 18:39:45 -0700451 * @param enforceMonotonic Validate that incoming value is strictly
452 * monotonic compared to this object.
Jeff Sharkey3f391352011-06-05 17:42:53 -0700453 * @param clampNegative Instead of throwing like {@code enforceMonotonic},
454 * clamp resulting counters at 0 to prevent negative values.
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700455 */
Jeff Sharkey3f391352011-06-05 17:42:53 -0700456 private NetworkStats subtract(
457 NetworkStats value, boolean enforceMonotonic, boolean clampNegative) {
Jeff Sharkey75279902011-05-24 18:39:45 -0700458 final long deltaRealtime = this.elapsedRealtime - value.elapsedRealtime;
459 if (enforceMonotonic && deltaRealtime < 0) {
460 throw new IllegalArgumentException("found non-monotonic realtime");
461 }
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700462
Jeff Sharkey75279902011-05-24 18:39:45 -0700463 // result will have our rows, and elapsed time between snapshots
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700464 final Entry entry = new Entry();
Jeff Sharkey4a971222011-06-11 22:16:55 -0700465 final NetworkStats result = new NetworkStats(deltaRealtime, size);
466 for (int i = 0; i < size; i++) {
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700467 entry.iface = iface[i];
468 entry.uid = uid[i];
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700469 entry.set = set[i];
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700470 entry.tag = tag[i];
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700471
472 // find remote row that matches, and subtract
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700473 final int j = value.findIndex(entry.iface, entry.uid, entry.set, entry.tag);
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700474 if (j == -1) {
475 // newly appearing row, return entire value
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700476 entry.rxBytes = rxBytes[i];
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700477 entry.rxPackets = rxPackets[i];
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700478 entry.txBytes = txBytes[i];
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700479 entry.txPackets = txPackets[i];
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700480 entry.operations = operations[i];
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700481 } else {
482 // existing row, subtract remote value
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700483 entry.rxBytes = rxBytes[i] - value.rxBytes[j];
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700484 entry.rxPackets = rxPackets[i] - value.rxPackets[j];
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700485 entry.txBytes = txBytes[i] - value.txBytes[j];
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700486 entry.txPackets = txPackets[i] - value.txPackets[j];
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700487 entry.operations = operations[i] - value.operations[j];
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700488 if (enforceMonotonic
489 && (entry.rxBytes < 0 || entry.rxPackets < 0 || entry.txBytes < 0
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700490 || entry.txPackets < 0 || entry.operations < 0)) {
Jeff Sharkey47eb1022011-08-25 17:48:52 -0700491 Log.v(TAG, "lhs=" + this);
492 Log.v(TAG, "rhs=" + value);
493 throw new IllegalArgumentException(
494 "found non-monotonic values at lhs[" + i + "] - rhs[" + j + "]");
Jeff Sharkey75279902011-05-24 18:39:45 -0700495 }
Jeff Sharkey3f391352011-06-05 17:42:53 -0700496 if (clampNegative) {
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700497 entry.rxBytes = Math.max(0, entry.rxBytes);
498 entry.rxPackets = Math.max(0, entry.rxPackets);
499 entry.txBytes = Math.max(0, entry.txBytes);
500 entry.txPackets = Math.max(0, entry.txPackets);
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700501 entry.operations = Math.max(0, entry.operations);
Jeff Sharkey3f391352011-06-05 17:42:53 -0700502 }
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700503 }
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700504
505 result.addValues(entry);
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700506 }
507
Jeff Sharkey4a971222011-06-11 22:16:55 -0700508 return result;
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700509 }
510
Jeff Sharkey905b5892011-09-30 15:19:49 -0700511 /**
512 * Return total statistics grouped by {@link #iface}; doesn't mutate the
513 * original structure.
514 */
515 public NetworkStats groupedByIface() {
516 final NetworkStats stats = new NetworkStats(elapsedRealtime, 10);
517
518 final Entry entry = new Entry();
519 entry.uid = UID_ALL;
520 entry.set = SET_ALL;
521 entry.tag = TAG_NONE;
522 entry.operations = 0L;
523
524 for (int i = 0; i < size; i++) {
525 // skip specific tags, since already counted in TAG_NONE
526 if (tag[i] != TAG_NONE) continue;
527
528 entry.iface = iface[i];
529 entry.rxBytes = rxBytes[i];
530 entry.rxPackets = rxPackets[i];
531 entry.txBytes = txBytes[i];
532 entry.txPackets = txPackets[i];
533 stats.combineValues(entry);
534 }
535
536 return stats;
537 }
538
Jeff Sharkey1059c3c2011-10-04 16:54:49 -0700539 /**
540 * Return total statistics grouped by {@link #uid}; doesn't mutate the
541 * original structure.
542 */
543 public NetworkStats groupedByUid() {
544 final NetworkStats stats = new NetworkStats(elapsedRealtime, 10);
545
546 final Entry entry = new Entry();
547 entry.iface = IFACE_ALL;
548 entry.set = SET_ALL;
549 entry.tag = TAG_NONE;
550
551 for (int i = 0; i < size; i++) {
552 // skip specific tags, since already counted in TAG_NONE
553 if (tag[i] != TAG_NONE) continue;
554
555 entry.uid = uid[i];
556 entry.rxBytes = rxBytes[i];
557 entry.rxPackets = rxPackets[i];
558 entry.txBytes = txBytes[i];
559 entry.txPackets = txPackets[i];
560 entry.operations = operations[i];
561 stats.combineValues(entry);
562 }
563
564 return stats;
565 }
566
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700567 public void dump(String prefix, PrintWriter pw) {
568 pw.print(prefix);
569 pw.print("NetworkStats: elapsedRealtime="); pw.println(elapsedRealtime);
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700570 for (int i = 0; i < size; i++) {
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700571 pw.print(prefix);
572 pw.print(" iface="); pw.print(iface[i]);
573 pw.print(" uid="); pw.print(uid[i]);
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700574 pw.print(" set="); pw.print(setToString(set[i]));
575 pw.print(" tag="); pw.print(tagToString(tag[i]));
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700576 pw.print(" rxBytes="); pw.print(rxBytes[i]);
Jeff Sharkeyfd8be3e2011-07-11 14:36:15 -0700577 pw.print(" rxPackets="); pw.print(rxPackets[i]);
Jeff Sharkeyd37948f2011-07-12 13:57:00 -0700578 pw.print(" txBytes="); pw.print(txBytes[i]);
Jeff Sharkeya63ba592011-07-19 23:47:12 -0700579 pw.print(" txPackets="); pw.print(txPackets[i]);
580 pw.print(" operations="); pw.println(operations[i]);
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700581 }
582 }
583
Jeff Sharkeyb5d55e32011-08-10 17:53:27 -0700584 /**
585 * Return text description of {@link #set} value.
586 */
587 public static String setToString(int set) {
588 switch (set) {
589 case SET_ALL:
590 return "ALL";
591 case SET_DEFAULT:
592 return "DEFAULT";
593 case SET_FOREGROUND:
594 return "FOREGROUND";
595 default:
596 return "UNKNOWN";
597 }
598 }
599
600 /**
601 * Return text description of {@link #tag} value.
602 */
603 public static String tagToString(int tag) {
604 return "0x" + Integer.toHexString(tag);
605 }
606
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700607 @Override
608 public String toString() {
609 final CharArrayWriter writer = new CharArrayWriter();
610 dump("", new PrintWriter(writer));
611 return writer.toString();
612 }
613
614 /** {@inheritDoc} */
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700615 public int describeContents() {
616 return 0;
617 }
618
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700619 public static final Creator<NetworkStats> CREATOR = new Creator<NetworkStats>() {
620 public NetworkStats createFromParcel(Parcel in) {
621 return new NetworkStats(in);
622 }
623
624 public NetworkStats[] newArray(int size) {
625 return new NetworkStats[size];
626 }
627 };
628}