1 /*
2  * Copyright (C) 2012 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 com.android.server.net;
18 
19 import static android.net.NetworkStats.DEFAULT_NETWORK_NO;
20 import static android.net.NetworkStats.DEFAULT_NETWORK_YES;
21 import static android.net.NetworkStats.IFACE_ALL;
22 import static android.net.NetworkStats.METERED_NO;
23 import static android.net.NetworkStats.METERED_YES;
24 import static android.net.NetworkStats.ROAMING_NO;
25 import static android.net.NetworkStats.ROAMING_YES;
26 import static android.net.NetworkStats.SET_ALL;
27 import static android.net.NetworkStats.SET_DEFAULT;
28 import static android.net.NetworkStats.TAG_NONE;
29 import static android.net.NetworkStats.UID_ALL;
30 import static android.net.TrafficStats.UID_REMOVED;
31 import static android.text.format.DateUtils.WEEK_IN_MILLIS;
32 
33 import static com.android.server.net.NetworkStatsService.TAG;
34 
35 import android.net.NetworkIdentity;
36 import android.net.NetworkStats;
37 import android.net.NetworkStatsHistory;
38 import android.net.NetworkTemplate;
39 import android.net.TrafficStats;
40 import android.os.Binder;
41 import android.service.NetworkStatsCollectionKeyProto;
42 import android.service.NetworkStatsCollectionProto;
43 import android.service.NetworkStatsCollectionStatsProto;
44 import android.telephony.SubscriptionPlan;
45 import android.text.format.DateUtils;
46 import android.util.ArrayMap;
47 import android.util.AtomicFile;
48 import android.util.IntArray;
49 import android.util.MathUtils;
50 import android.util.Range;
51 import android.util.Slog;
52 import android.util.proto.ProtoOutputStream;
53 
54 import com.android.internal.annotations.VisibleForTesting;
55 import com.android.internal.util.ArrayUtils;
56 import com.android.internal.util.FileRotator;
57 import com.android.internal.util.IndentingPrintWriter;
58 
59 import libcore.io.IoUtils;
60 
61 import com.google.android.collect.Lists;
62 import com.google.android.collect.Maps;
63 
64 import java.io.BufferedInputStream;
65 import java.io.DataInputStream;
66 import java.io.DataOutputStream;
67 import java.io.File;
68 import java.io.FileNotFoundException;
69 import java.io.IOException;
70 import java.io.InputStream;
71 import java.io.PrintWriter;
72 import java.net.ProtocolException;
73 import java.time.ZonedDateTime;
74 import java.util.ArrayList;
75 import java.util.Collections;
76 import java.util.HashMap;
77 import java.util.Iterator;
78 import java.util.Objects;
79 
80 /**
81  * Collection of {@link NetworkStatsHistory}, stored based on combined key of
82  * {@link NetworkIdentitySet}, UID, set, and tag. Knows how to persist itself.
83  */
84 public class NetworkStatsCollection implements FileRotator.Reader {
85     /** File header magic number: "ANET" */
86     private static final int FILE_MAGIC = 0x414E4554;
87 
88     private static final int VERSION_NETWORK_INIT = 1;
89 
90     private static final int VERSION_UID_INIT = 1;
91     private static final int VERSION_UID_WITH_IDENT = 2;
92     private static final int VERSION_UID_WITH_TAG = 3;
93     private static final int VERSION_UID_WITH_SET = 4;
94 
95     private static final int VERSION_UNIFIED_INIT = 16;
96 
97     private ArrayMap<Key, NetworkStatsHistory> mStats = new ArrayMap<>();
98 
99     private final long mBucketDuration;
100 
101     private long mStartMillis;
102     private long mEndMillis;
103     private long mTotalBytes;
104     private boolean mDirty;
105 
NetworkStatsCollection(long bucketDuration)106     public NetworkStatsCollection(long bucketDuration) {
107         mBucketDuration = bucketDuration;
108         reset();
109     }
110 
clear()111     public void clear() {
112         reset();
113     }
114 
reset()115     public void reset() {
116         mStats.clear();
117         mStartMillis = Long.MAX_VALUE;
118         mEndMillis = Long.MIN_VALUE;
119         mTotalBytes = 0;
120         mDirty = false;
121     }
122 
getStartMillis()123     public long getStartMillis() {
124         return mStartMillis;
125     }
126 
127     /**
128      * Return first atomic bucket in this collection, which is more conservative
129      * than {@link #mStartMillis}.
130      */
getFirstAtomicBucketMillis()131     public long getFirstAtomicBucketMillis() {
132         if (mStartMillis == Long.MAX_VALUE) {
133             return Long.MAX_VALUE;
134         } else {
135             return mStartMillis + mBucketDuration;
136         }
137     }
138 
getEndMillis()139     public long getEndMillis() {
140         return mEndMillis;
141     }
142 
getTotalBytes()143     public long getTotalBytes() {
144         return mTotalBytes;
145     }
146 
isDirty()147     public boolean isDirty() {
148         return mDirty;
149     }
150 
clearDirty()151     public void clearDirty() {
152         mDirty = false;
153     }
154 
isEmpty()155     public boolean isEmpty() {
156         return mStartMillis == Long.MAX_VALUE && mEndMillis == Long.MIN_VALUE;
157     }
158 
159     @VisibleForTesting
roundUp(long time)160     public long roundUp(long time) {
161         if (time == Long.MIN_VALUE || time == Long.MAX_VALUE
162                 || time == SubscriptionPlan.TIME_UNKNOWN) {
163             return time;
164         } else {
165             final long mod = time % mBucketDuration;
166             if (mod > 0) {
167                 time -= mod;
168                 time += mBucketDuration;
169             }
170             return time;
171         }
172     }
173 
174     @VisibleForTesting
roundDown(long time)175     public long roundDown(long time) {
176         if (time == Long.MIN_VALUE || time == Long.MAX_VALUE
177                 || time == SubscriptionPlan.TIME_UNKNOWN) {
178             return time;
179         } else {
180             final long mod = time % mBucketDuration;
181             if (mod > 0) {
182                 time -= mod;
183             }
184             return time;
185         }
186     }
187 
188     /**
189      * Safely multiple a value by a rational.
190      * <p>
191      * Internally it uses integer-based math whenever possible, but switches
192      * over to double-based math if values would overflow.
193      */
194     @VisibleForTesting
multiplySafe(long value, long num, long den)195     public static long multiplySafe(long value, long num, long den) {
196         if (den == 0) den = 1;
197         long x = value;
198         long y = num;
199 
200         // Logic shamelessly borrowed from Math.multiplyExact()
201         long r = x * y;
202         long ax = Math.abs(x);
203         long ay = Math.abs(y);
204         if (((ax | ay) >>> 31 != 0)) {
205             // Some bits greater than 2^31 that might cause overflow
206             // Check the result using the divide operator
207             // and check for the special case of Long.MIN_VALUE * -1
208             if (((y != 0) && (r / y != x)) ||
209                     (x == Long.MIN_VALUE && y == -1)) {
210                 // Use double math to avoid overflowing
211                 return (long) (((double) num / den) * value);
212             }
213         }
214         return r / den;
215     }
216 
getRelevantUids(@etworkStatsAccess.Level int accessLevel)217     public int[] getRelevantUids(@NetworkStatsAccess.Level int accessLevel) {
218         return getRelevantUids(accessLevel, Binder.getCallingUid());
219     }
220 
getRelevantUids(@etworkStatsAccess.Level int accessLevel, final int callerUid)221     public int[] getRelevantUids(@NetworkStatsAccess.Level int accessLevel,
222                 final int callerUid) {
223         IntArray uids = new IntArray();
224         for (int i = 0; i < mStats.size(); i++) {
225             final Key key = mStats.keyAt(i);
226             if (NetworkStatsAccess.isAccessibleToUser(key.uid, callerUid, accessLevel)) {
227                 int j = uids.binarySearch(key.uid);
228 
229                 if (j < 0) {
230                     j = ~j;
231                     uids.add(j, key.uid);
232                 }
233             }
234         }
235         return uids.toArray();
236     }
237 
238     /**
239      * Combine all {@link NetworkStatsHistory} in this collection which match
240      * the requested parameters.
241      */
getHistory(NetworkTemplate template, SubscriptionPlan augmentPlan, int uid, int set, int tag, int fields, long start, long end, @NetworkStatsAccess.Level int accessLevel, int callerUid)242     public NetworkStatsHistory getHistory(NetworkTemplate template, SubscriptionPlan augmentPlan,
243             int uid, int set, int tag, int fields, long start, long end,
244             @NetworkStatsAccess.Level int accessLevel, int callerUid) {
245         if (!NetworkStatsAccess.isAccessibleToUser(uid, callerUid, accessLevel)) {
246             throw new SecurityException("Network stats history of uid " + uid
247                     + " is forbidden for caller " + callerUid);
248         }
249 
250         // 180 days of history should be enough for anyone; if we end up needing
251         // more, we'll dynamically grow the history object.
252         final int bucketEstimate = (int) MathUtils.constrain(((end - start) / mBucketDuration), 0,
253                 (180 * DateUtils.DAY_IN_MILLIS) / mBucketDuration);
254         final NetworkStatsHistory combined = new NetworkStatsHistory(
255                 mBucketDuration, bucketEstimate, fields);
256 
257         // shortcut when we know stats will be empty
258         if (start == end) return combined;
259 
260         // Figure out the window of time that we should be augmenting (if any)
261         long augmentStart = SubscriptionPlan.TIME_UNKNOWN;
262         long augmentEnd = (augmentPlan != null) ? augmentPlan.getDataUsageTime()
263                 : SubscriptionPlan.TIME_UNKNOWN;
264         // And if augmenting, we might need to collect more data to adjust with
265         long collectStart = start;
266         long collectEnd = end;
267 
268         if (augmentEnd != SubscriptionPlan.TIME_UNKNOWN) {
269             final Iterator<Range<ZonedDateTime>> it = augmentPlan.cycleIterator();
270             while (it.hasNext()) {
271                 final Range<ZonedDateTime> cycle = it.next();
272                 final long cycleStart = cycle.getLower().toInstant().toEpochMilli();
273                 final long cycleEnd = cycle.getUpper().toInstant().toEpochMilli();
274                 if (cycleStart <= augmentEnd && augmentEnd < cycleEnd) {
275                     augmentStart = cycleStart;
276                     collectStart = Long.min(collectStart, augmentStart);
277                     collectEnd = Long.max(collectEnd, augmentEnd);
278                     break;
279                 }
280             }
281         }
282 
283         if (augmentStart != SubscriptionPlan.TIME_UNKNOWN) {
284             // Shrink augmentation window so we don't risk undercounting.
285             augmentStart = roundUp(augmentStart);
286             augmentEnd = roundDown(augmentEnd);
287             // Grow collection window so we get all the stats needed.
288             collectStart = roundDown(collectStart);
289             collectEnd = roundUp(collectEnd);
290         }
291 
292         for (int i = 0; i < mStats.size(); i++) {
293             final Key key = mStats.keyAt(i);
294             if (key.uid == uid && NetworkStats.setMatches(set, key.set) && key.tag == tag
295                     && templateMatches(template, key.ident)) {
296                 final NetworkStatsHistory value = mStats.valueAt(i);
297                 combined.recordHistory(value, collectStart, collectEnd);
298             }
299         }
300 
301         if (augmentStart != SubscriptionPlan.TIME_UNKNOWN) {
302             final NetworkStatsHistory.Entry entry = combined.getValues(
303                     augmentStart, augmentEnd, null);
304 
305             // If we don't have any recorded data for this time period, give
306             // ourselves something to scale with.
307             if (entry.rxBytes == 0 || entry.txBytes == 0) {
308                 combined.recordData(augmentStart, augmentEnd,
309                         new NetworkStats.Entry(1, 0, 1, 0, 0));
310                 combined.getValues(augmentStart, augmentEnd, entry);
311             }
312 
313             final long rawBytes = entry.rxBytes + entry.txBytes;
314             final long rawRxBytes = entry.rxBytes;
315             final long rawTxBytes = entry.txBytes;
316             final long targetBytes = augmentPlan.getDataUsageBytes();
317             final long targetRxBytes = multiplySafe(targetBytes, rawRxBytes, rawBytes);
318             final long targetTxBytes = multiplySafe(targetBytes, rawTxBytes, rawBytes);
319 
320             // Scale all matching buckets to reach anchor target
321             final long beforeTotal = combined.getTotalBytes();
322             for (int i = 0; i < combined.size(); i++) {
323                 combined.getValues(i, entry);
324                 if (entry.bucketStart >= augmentStart
325                         && entry.bucketStart + entry.bucketDuration <= augmentEnd) {
326                     entry.rxBytes = multiplySafe(targetRxBytes, entry.rxBytes, rawRxBytes);
327                     entry.txBytes = multiplySafe(targetTxBytes, entry.txBytes, rawTxBytes);
328                     // We purposefully clear out packet counters to indicate
329                     // that this data has been augmented.
330                     entry.rxPackets = 0;
331                     entry.txPackets = 0;
332                     combined.setValues(i, entry);
333                 }
334             }
335 
336             final long deltaTotal = combined.getTotalBytes() - beforeTotal;
337             if (deltaTotal != 0) {
338                 Slog.d(TAG, "Augmented network usage by " + deltaTotal + " bytes");
339             }
340 
341             // Finally we can slice data as originally requested
342             final NetworkStatsHistory sliced = new NetworkStatsHistory(
343                     mBucketDuration, bucketEstimate, fields);
344             sliced.recordHistory(combined, start, end);
345             return sliced;
346         } else {
347             return combined;
348         }
349     }
350 
351     /**
352      * Summarize all {@link NetworkStatsHistory} in this collection which match
353      * the requested parameters.
354      */
getSummary(NetworkTemplate template, long start, long end, @NetworkStatsAccess.Level int accessLevel, int callerUid)355     public NetworkStats getSummary(NetworkTemplate template, long start, long end,
356             @NetworkStatsAccess.Level int accessLevel, int callerUid) {
357         final long now = System.currentTimeMillis();
358 
359         final NetworkStats stats = new NetworkStats(end - start, 24);
360 
361         // shortcut when we know stats will be empty
362         if (start == end) return stats;
363 
364         final NetworkStats.Entry entry = new NetworkStats.Entry();
365         NetworkStatsHistory.Entry historyEntry = null;
366 
367         for (int i = 0; i < mStats.size(); i++) {
368             final Key key = mStats.keyAt(i);
369             if (templateMatches(template, key.ident)
370                     && NetworkStatsAccess.isAccessibleToUser(key.uid, callerUid, accessLevel)
371                     && key.set < NetworkStats.SET_DEBUG_START) {
372                 final NetworkStatsHistory value = mStats.valueAt(i);
373                 historyEntry = value.getValues(start, end, now, historyEntry);
374 
375                 entry.iface = IFACE_ALL;
376                 entry.uid = key.uid;
377                 entry.set = key.set;
378                 entry.tag = key.tag;
379                 entry.defaultNetwork = key.ident.areAllMembersOnDefaultNetwork() ?
380                         DEFAULT_NETWORK_YES : DEFAULT_NETWORK_NO;
381                 entry.metered = key.ident.isAnyMemberMetered() ? METERED_YES : METERED_NO;
382                 entry.roaming = key.ident.isAnyMemberRoaming() ? ROAMING_YES : ROAMING_NO;
383                 entry.rxBytes = historyEntry.rxBytes;
384                 entry.rxPackets = historyEntry.rxPackets;
385                 entry.txBytes = historyEntry.txBytes;
386                 entry.txPackets = historyEntry.txPackets;
387                 entry.operations = historyEntry.operations;
388 
389                 if (!entry.isEmpty()) {
390                     stats.combineValues(entry);
391                 }
392             }
393         }
394 
395         return stats;
396     }
397 
398     /**
399      * Record given {@link android.net.NetworkStats.Entry} into this collection.
400      */
recordData(NetworkIdentitySet ident, int uid, int set, int tag, long start, long end, NetworkStats.Entry entry)401     public void recordData(NetworkIdentitySet ident, int uid, int set, int tag, long start,
402             long end, NetworkStats.Entry entry) {
403         final NetworkStatsHistory history = findOrCreateHistory(ident, uid, set, tag);
404         history.recordData(start, end, entry);
405         noteRecordedHistory(history.getStart(), history.getEnd(), entry.rxBytes + entry.txBytes);
406     }
407 
408     /**
409      * Record given {@link NetworkStatsHistory} into this collection.
410      */
recordHistory(Key key, NetworkStatsHistory history)411     private void recordHistory(Key key, NetworkStatsHistory history) {
412         if (history.size() == 0) return;
413         noteRecordedHistory(history.getStart(), history.getEnd(), history.getTotalBytes());
414 
415         NetworkStatsHistory target = mStats.get(key);
416         if (target == null) {
417             target = new NetworkStatsHistory(history.getBucketDuration());
418             mStats.put(key, target);
419         }
420         target.recordEntireHistory(history);
421     }
422 
423     /**
424      * Record all {@link NetworkStatsHistory} contained in the given collection
425      * into this collection.
426      */
recordCollection(NetworkStatsCollection another)427     public void recordCollection(NetworkStatsCollection another) {
428         for (int i = 0; i < another.mStats.size(); i++) {
429             final Key key = another.mStats.keyAt(i);
430             final NetworkStatsHistory value = another.mStats.valueAt(i);
431             recordHistory(key, value);
432         }
433     }
434 
findOrCreateHistory( NetworkIdentitySet ident, int uid, int set, int tag)435     private NetworkStatsHistory findOrCreateHistory(
436             NetworkIdentitySet ident, int uid, int set, int tag) {
437         final Key key = new Key(ident, uid, set, tag);
438         final NetworkStatsHistory existing = mStats.get(key);
439 
440         // update when no existing, or when bucket duration changed
441         NetworkStatsHistory updated = null;
442         if (existing == null) {
443             updated = new NetworkStatsHistory(mBucketDuration, 10);
444         } else if (existing.getBucketDuration() != mBucketDuration) {
445             updated = new NetworkStatsHistory(existing, mBucketDuration);
446         }
447 
448         if (updated != null) {
449             mStats.put(key, updated);
450             return updated;
451         } else {
452             return existing;
453         }
454     }
455 
456     @Override
read(InputStream in)457     public void read(InputStream in) throws IOException {
458         read(new DataInputStream(in));
459     }
460 
read(DataInputStream in)461     public void read(DataInputStream in) throws IOException {
462         // verify file magic header intact
463         final int magic = in.readInt();
464         if (magic != FILE_MAGIC) {
465             throw new ProtocolException("unexpected magic: " + magic);
466         }
467 
468         final int version = in.readInt();
469         switch (version) {
470             case VERSION_UNIFIED_INIT: {
471                 // uid := size *(NetworkIdentitySet size *(uid set tag NetworkStatsHistory))
472                 final int identSize = in.readInt();
473                 for (int i = 0; i < identSize; i++) {
474                     final NetworkIdentitySet ident = new NetworkIdentitySet(in);
475 
476                     final int size = in.readInt();
477                     for (int j = 0; j < size; j++) {
478                         final int uid = in.readInt();
479                         final int set = in.readInt();
480                         final int tag = in.readInt();
481 
482                         final Key key = new Key(ident, uid, set, tag);
483                         final NetworkStatsHistory history = new NetworkStatsHistory(in);
484                         recordHistory(key, history);
485                     }
486                 }
487                 break;
488             }
489             default: {
490                 throw new ProtocolException("unexpected version: " + version);
491             }
492         }
493     }
494 
write(DataOutputStream out)495     public void write(DataOutputStream out) throws IOException {
496         // cluster key lists grouped by ident
497         final HashMap<NetworkIdentitySet, ArrayList<Key>> keysByIdent = Maps.newHashMap();
498         for (Key key : mStats.keySet()) {
499             ArrayList<Key> keys = keysByIdent.get(key.ident);
500             if (keys == null) {
501                 keys = Lists.newArrayList();
502                 keysByIdent.put(key.ident, keys);
503             }
504             keys.add(key);
505         }
506 
507         out.writeInt(FILE_MAGIC);
508         out.writeInt(VERSION_UNIFIED_INIT);
509 
510         out.writeInt(keysByIdent.size());
511         for (NetworkIdentitySet ident : keysByIdent.keySet()) {
512             final ArrayList<Key> keys = keysByIdent.get(ident);
513             ident.writeToStream(out);
514 
515             out.writeInt(keys.size());
516             for (Key key : keys) {
517                 final NetworkStatsHistory history = mStats.get(key);
518                 out.writeInt(key.uid);
519                 out.writeInt(key.set);
520                 out.writeInt(key.tag);
521                 history.writeToStream(out);
522             }
523         }
524 
525         out.flush();
526     }
527 
528     @Deprecated
readLegacyNetwork(File file)529     public void readLegacyNetwork(File file) throws IOException {
530         final AtomicFile inputFile = new AtomicFile(file);
531 
532         DataInputStream in = null;
533         try {
534             in = new DataInputStream(new BufferedInputStream(inputFile.openRead()));
535 
536             // verify file magic header intact
537             final int magic = in.readInt();
538             if (magic != FILE_MAGIC) {
539                 throw new ProtocolException("unexpected magic: " + magic);
540             }
541 
542             final int version = in.readInt();
543             switch (version) {
544                 case VERSION_NETWORK_INIT: {
545                     // network := size *(NetworkIdentitySet NetworkStatsHistory)
546                     final int size = in.readInt();
547                     for (int i = 0; i < size; i++) {
548                         final NetworkIdentitySet ident = new NetworkIdentitySet(in);
549                         final NetworkStatsHistory history = new NetworkStatsHistory(in);
550 
551                         final Key key = new Key(ident, UID_ALL, SET_ALL, TAG_NONE);
552                         recordHistory(key, history);
553                     }
554                     break;
555                 }
556                 default: {
557                     throw new ProtocolException("unexpected version: " + version);
558                 }
559             }
560         } catch (FileNotFoundException e) {
561             // missing stats is okay, probably first boot
562         } finally {
563             IoUtils.closeQuietly(in);
564         }
565     }
566 
567     @Deprecated
readLegacyUid(File file, boolean onlyTags)568     public void readLegacyUid(File file, boolean onlyTags) throws IOException {
569         final AtomicFile inputFile = new AtomicFile(file);
570 
571         DataInputStream in = null;
572         try {
573             in = new DataInputStream(new BufferedInputStream(inputFile.openRead()));
574 
575             // verify file magic header intact
576             final int magic = in.readInt();
577             if (magic != FILE_MAGIC) {
578                 throw new ProtocolException("unexpected magic: " + magic);
579             }
580 
581             final int version = in.readInt();
582             switch (version) {
583                 case VERSION_UID_INIT: {
584                     // uid := size *(UID NetworkStatsHistory)
585 
586                     // drop this data version, since we don't have a good
587                     // mapping into NetworkIdentitySet.
588                     break;
589                 }
590                 case VERSION_UID_WITH_IDENT: {
591                     // uid := size *(NetworkIdentitySet size *(UID NetworkStatsHistory))
592 
593                     // drop this data version, since this version only existed
594                     // for a short time.
595                     break;
596                 }
597                 case VERSION_UID_WITH_TAG:
598                 case VERSION_UID_WITH_SET: {
599                     // uid := size *(NetworkIdentitySet size *(uid set tag NetworkStatsHistory))
600                     final int identSize = in.readInt();
601                     for (int i = 0; i < identSize; i++) {
602                         final NetworkIdentitySet ident = new NetworkIdentitySet(in);
603 
604                         final int size = in.readInt();
605                         for (int j = 0; j < size; j++) {
606                             final int uid = in.readInt();
607                             final int set = (version >= VERSION_UID_WITH_SET) ? in.readInt()
608                                     : SET_DEFAULT;
609                             final int tag = in.readInt();
610 
611                             final Key key = new Key(ident, uid, set, tag);
612                             final NetworkStatsHistory history = new NetworkStatsHistory(in);
613 
614                             if ((tag == TAG_NONE) != onlyTags) {
615                                 recordHistory(key, history);
616                             }
617                         }
618                     }
619                     break;
620                 }
621                 default: {
622                     throw new ProtocolException("unexpected version: " + version);
623                 }
624             }
625         } catch (FileNotFoundException e) {
626             // missing stats is okay, probably first boot
627         } finally {
628             IoUtils.closeQuietly(in);
629         }
630     }
631 
632     /**
633      * Remove any {@link NetworkStatsHistory} attributed to the requested UID,
634      * moving any {@link NetworkStats#TAG_NONE} series to
635      * {@link TrafficStats#UID_REMOVED}.
636      */
removeUids(int[] uids)637     public void removeUids(int[] uids) {
638         final ArrayList<Key> knownKeys = Lists.newArrayList();
639         knownKeys.addAll(mStats.keySet());
640 
641         // migrate all UID stats into special "removed" bucket
642         for (Key key : knownKeys) {
643             if (ArrayUtils.contains(uids, key.uid)) {
644                 // only migrate combined TAG_NONE history
645                 if (key.tag == TAG_NONE) {
646                     final NetworkStatsHistory uidHistory = mStats.get(key);
647                     final NetworkStatsHistory removedHistory = findOrCreateHistory(
648                             key.ident, UID_REMOVED, SET_DEFAULT, TAG_NONE);
649                     removedHistory.recordEntireHistory(uidHistory);
650                 }
651                 mStats.remove(key);
652                 mDirty = true;
653             }
654         }
655     }
656 
noteRecordedHistory(long startMillis, long endMillis, long totalBytes)657     private void noteRecordedHistory(long startMillis, long endMillis, long totalBytes) {
658         if (startMillis < mStartMillis) mStartMillis = startMillis;
659         if (endMillis > mEndMillis) mEndMillis = endMillis;
660         mTotalBytes += totalBytes;
661         mDirty = true;
662     }
663 
estimateBuckets()664     private int estimateBuckets() {
665         return (int) (Math.min(mEndMillis - mStartMillis, WEEK_IN_MILLIS * 5)
666                 / mBucketDuration);
667     }
668 
getSortedKeys()669     private ArrayList<Key> getSortedKeys() {
670         final ArrayList<Key> keys = Lists.newArrayList();
671         keys.addAll(mStats.keySet());
672         Collections.sort(keys);
673         return keys;
674     }
675 
dump(IndentingPrintWriter pw)676     public void dump(IndentingPrintWriter pw) {
677         for (Key key : getSortedKeys()) {
678             pw.print("ident="); pw.print(key.ident.toString());
679             pw.print(" uid="); pw.print(key.uid);
680             pw.print(" set="); pw.print(NetworkStats.setToString(key.set));
681             pw.print(" tag="); pw.println(NetworkStats.tagToString(key.tag));
682 
683             final NetworkStatsHistory history = mStats.get(key);
684             pw.increaseIndent();
685             history.dump(pw, true);
686             pw.decreaseIndent();
687         }
688     }
689 
writeToProto(ProtoOutputStream proto, long tag)690     public void writeToProto(ProtoOutputStream proto, long tag) {
691         final long start = proto.start(tag);
692 
693         for (Key key : getSortedKeys()) {
694             final long startStats = proto.start(NetworkStatsCollectionProto.STATS);
695 
696             // Key
697             final long startKey = proto.start(NetworkStatsCollectionStatsProto.KEY);
698             key.ident.writeToProto(proto, NetworkStatsCollectionKeyProto.IDENTITY);
699             proto.write(NetworkStatsCollectionKeyProto.UID, key.uid);
700             proto.write(NetworkStatsCollectionKeyProto.SET, key.set);
701             proto.write(NetworkStatsCollectionKeyProto.TAG, key.tag);
702             proto.end(startKey);
703 
704             // Value
705             final NetworkStatsHistory history = mStats.get(key);
706             history.writeToProto(proto, NetworkStatsCollectionStatsProto.HISTORY);
707             proto.end(startStats);
708         }
709 
710         proto.end(start);
711     }
712 
dumpCheckin(PrintWriter pw, long start, long end)713     public void dumpCheckin(PrintWriter pw, long start, long end) {
714         dumpCheckin(pw, start, end, NetworkTemplate.buildTemplateMobileWildcard(), "cell");
715         dumpCheckin(pw, start, end, NetworkTemplate.buildTemplateWifiWildcard(), "wifi");
716         dumpCheckin(pw, start, end, NetworkTemplate.buildTemplateEthernet(), "eth");
717         dumpCheckin(pw, start, end, NetworkTemplate.buildTemplateBluetooth(), "bt");
718     }
719 
720     /**
721      * Dump all contained stats that match requested parameters, but group
722      * together all matching {@link NetworkTemplate} under a single prefix.
723      */
dumpCheckin(PrintWriter pw, long start, long end, NetworkTemplate groupTemplate, String groupPrefix)724     private void dumpCheckin(PrintWriter pw, long start, long end, NetworkTemplate groupTemplate,
725             String groupPrefix) {
726         final ArrayMap<Key, NetworkStatsHistory> grouped = new ArrayMap<>();
727 
728         // Walk through all history, grouping by matching network templates
729         for (int i = 0; i < mStats.size(); i++) {
730             final Key key = mStats.keyAt(i);
731             final NetworkStatsHistory value = mStats.valueAt(i);
732 
733             if (!templateMatches(groupTemplate, key.ident)) continue;
734             if (key.set >= NetworkStats.SET_DEBUG_START) continue;
735 
736             final Key groupKey = new Key(null, key.uid, key.set, key.tag);
737             NetworkStatsHistory groupHistory = grouped.get(groupKey);
738             if (groupHistory == null) {
739                 groupHistory = new NetworkStatsHistory(value.getBucketDuration());
740                 grouped.put(groupKey, groupHistory);
741             }
742             groupHistory.recordHistory(value, start, end);
743         }
744 
745         for (int i = 0; i < grouped.size(); i++) {
746             final Key key = grouped.keyAt(i);
747             final NetworkStatsHistory value = grouped.valueAt(i);
748 
749             if (value.size() == 0) continue;
750 
751             pw.print("c,");
752             pw.print(groupPrefix); pw.print(',');
753             pw.print(key.uid); pw.print(',');
754             pw.print(NetworkStats.setToCheckinString(key.set)); pw.print(',');
755             pw.print(key.tag);
756             pw.println();
757 
758             value.dumpCheckin(pw);
759         }
760     }
761 
762     /**
763      * Test if given {@link NetworkTemplate} matches any {@link NetworkIdentity}
764      * in the given {@link NetworkIdentitySet}.
765      */
templateMatches(NetworkTemplate template, NetworkIdentitySet identSet)766     private static boolean templateMatches(NetworkTemplate template, NetworkIdentitySet identSet) {
767         for (NetworkIdentity ident : identSet) {
768             if (template.matches(ident)) {
769                 return true;
770             }
771         }
772         return false;
773     }
774 
775     private static class Key implements Comparable<Key> {
776         public final NetworkIdentitySet ident;
777         public final int uid;
778         public final int set;
779         public final int tag;
780 
781         private final int hashCode;
782 
Key(NetworkIdentitySet ident, int uid, int set, int tag)783         public Key(NetworkIdentitySet ident, int uid, int set, int tag) {
784             this.ident = ident;
785             this.uid = uid;
786             this.set = set;
787             this.tag = tag;
788             hashCode = Objects.hash(ident, uid, set, tag);
789         }
790 
791         @Override
hashCode()792         public int hashCode() {
793             return hashCode;
794         }
795 
796         @Override
equals(Object obj)797         public boolean equals(Object obj) {
798             if (obj instanceof Key) {
799                 final Key key = (Key) obj;
800                 return uid == key.uid && set == key.set && tag == key.tag
801                         && Objects.equals(ident, key.ident);
802             }
803             return false;
804         }
805 
806         @Override
compareTo(Key another)807         public int compareTo(Key another) {
808             int res = 0;
809             if (ident != null && another.ident != null) {
810                 res = ident.compareTo(another.ident);
811             }
812             if (res == 0) {
813                 res = Integer.compare(uid, another.uid);
814             }
815             if (res == 0) {
816                 res = Integer.compare(set, another.set);
817             }
818             if (res == 0) {
819                 res = Integer.compare(tag, another.tag);
820             }
821             return res;
822         }
823     }
824 }
825