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