1 /*
2  * Copyright (C) 2017 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 android.net.metrics;
18 
19 import android.net.MacAddress;
20 import android.os.Process;
21 import android.os.SystemClock;
22 import android.util.SparseIntArray;
23 
24 import java.util.StringJoiner;
25 
26 /**
27  * An event logged per interface and that aggregates WakeupEvents for that interface.
28  * {@hide}
29  */
30 public class WakeupStats {
31 
32     private static final int NO_UID = -1;
33 
34     public final long creationTimeMs = SystemClock.elapsedRealtime();
35     public final String iface;
36 
37     public long totalWakeups = 0;
38     public long rootWakeups = 0;
39     public long systemWakeups = 0;
40     public long nonApplicationWakeups = 0;
41     public long applicationWakeups = 0;
42     public long noUidWakeups = 0;
43     public long durationSec = 0;
44 
45     public long l2UnicastCount = 0;
46     public long l2MulticastCount = 0;
47     public long l2BroadcastCount = 0;
48 
49     public final SparseIntArray ethertypes = new SparseIntArray();
50     public final SparseIntArray ipNextHeaders = new SparseIntArray();
51 
WakeupStats(String iface)52     public WakeupStats(String iface) {
53         this.iface = iface;
54     }
55 
56     /** Update durationSec with current time. */
updateDuration()57     public void updateDuration() {
58         durationSec = (SystemClock.elapsedRealtime() - creationTimeMs) / 1000;
59     }
60 
61     /** Update wakeup counters for the given WakeupEvent. */
countEvent(WakeupEvent ev)62     public void countEvent(WakeupEvent ev) {
63         totalWakeups++;
64         switch (ev.uid) {
65             case Process.ROOT_UID:
66                 rootWakeups++;
67                 break;
68             case Process.SYSTEM_UID:
69                 systemWakeups++;
70                 break;
71             case NO_UID:
72                 noUidWakeups++;
73                 break;
74             default:
75                 if (ev.uid >= Process.FIRST_APPLICATION_UID) {
76                     applicationWakeups++;
77                 } else {
78                     nonApplicationWakeups++;
79                 }
80                 break;
81         }
82 
83         if (ev.dstHwAddr != null) {
84             switch (ev.dstHwAddr.getAddressType()) {
85                 case MacAddress.TYPE_UNICAST:
86                     l2UnicastCount++;
87                     break;
88                 case MacAddress.TYPE_MULTICAST:
89                     l2MulticastCount++;
90                     break;
91                 case MacAddress.TYPE_BROADCAST:
92                     l2BroadcastCount++;
93                     break;
94                 default:
95                     break;
96             }
97         }
98 
99         increment(ethertypes, ev.ethertype);
100         if (ev.ipNextHeader >= 0) {
101             increment(ipNextHeaders, ev.ipNextHeader);
102         }
103     }
104 
105     @Override
toString()106     public String toString() {
107         updateDuration();
108         StringJoiner j = new StringJoiner(", ", "WakeupStats(", ")");
109         j.add(iface);
110         j.add("" + durationSec + "s");
111         j.add("total: " + totalWakeups);
112         j.add("root: " + rootWakeups);
113         j.add("system: " + systemWakeups);
114         j.add("apps: " + applicationWakeups);
115         j.add("non-apps: " + nonApplicationWakeups);
116         j.add("no uid: " + noUidWakeups);
117         j.add(String.format("l2 unicast/multicast/broadcast: %d/%d/%d",
118                 l2UnicastCount, l2MulticastCount, l2BroadcastCount));
119         for (int i = 0; i < ethertypes.size(); i++) {
120             int eth = ethertypes.keyAt(i);
121             int count = ethertypes.valueAt(i);
122             j.add(String.format("ethertype 0x%x: %d", eth, count));
123         }
124         for (int i = 0; i < ipNextHeaders.size(); i++) {
125             int proto = ipNextHeaders.keyAt(i);
126             int count = ipNextHeaders.valueAt(i);
127             j.add(String.format("ipNxtHdr %d: %d", proto, count));
128         }
129         return j.toString();
130     }
131 
increment(SparseIntArray counters, int key)132     private static void increment(SparseIntArray counters, int key) {
133         int newcount = counters.get(key, 0) + 1;
134         counters.put(key, newcount);
135     }
136 }
137