blob: 0f207bc9536e3c2ff91a4520861593a20e57312d [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;
22
23import java.io.CharArrayWriter;
24import java.io.PrintWriter;
25
26/**
27 * Collection of network statistics. Can contain summary details across all
28 * interfaces, or details with per-UID granularity. Designed to parcel quickly
29 * across process boundaries.
30 *
31 * @hide
32 */
33public class NetworkStats implements Parcelable {
34 /** {@link #iface} value when entry is summarized over all interfaces. */
35 public static final String IFACE_ALL = null;
36 /** {@link #uid} value when entry is summarized over all UIDs. */
37 public static final int UID_ALL = 0;
38
Jeff Sharkeyeedcb952011-05-17 14:55:15 -070039 // NOTE: data should only be accounted for once in this structure; if data
40 // is broken out, the summarized version should not be included.
41
Jeff Sharkey9a13f362011-04-26 16:25:36 -070042 /**
43 * {@link SystemClock#elapsedRealtime()} timestamp when this data was
44 * generated.
45 */
46 public final long elapsedRealtime;
47 public final String[] iface;
48 public final int[] uid;
49 public final long[] rx;
50 public final long[] tx;
51
52 // TODO: add fg/bg stats and tag granularity
53
54 private NetworkStats(long elapsedRealtime, String[] iface, int[] uid, long[] rx, long[] tx) {
55 this.elapsedRealtime = elapsedRealtime;
56 this.iface = iface;
57 this.uid = uid;
58 this.rx = rx;
59 this.tx = tx;
60 }
61
62 public NetworkStats(Parcel parcel) {
63 elapsedRealtime = parcel.readLong();
64 iface = parcel.createStringArray();
65 uid = parcel.createIntArray();
66 rx = parcel.createLongArray();
67 tx = parcel.createLongArray();
68 }
69
70 public static class Builder {
71 private long mElapsedRealtime;
72 private final String[] mIface;
73 private final int[] mUid;
74 private final long[] mRx;
75 private final long[] mTx;
76
77 private int mIndex = 0;
78
79 public Builder(long elapsedRealtime, int size) {
80 mElapsedRealtime = elapsedRealtime;
81 mIface = new String[size];
82 mUid = new int[size];
83 mRx = new long[size];
84 mTx = new long[size];
85 }
86
Jeff Sharkeyeedcb952011-05-17 14:55:15 -070087 public Builder addEntry(String iface, int uid, long rx, long tx) {
Jeff Sharkey9a13f362011-04-26 16:25:36 -070088 mIface[mIndex] = iface;
89 mUid[mIndex] = uid;
90 mRx[mIndex] = rx;
91 mTx[mIndex] = tx;
92 mIndex++;
Jeff Sharkeyeedcb952011-05-17 14:55:15 -070093 return this;
Jeff Sharkey9a13f362011-04-26 16:25:36 -070094 }
95
96 public NetworkStats build() {
97 if (mIndex != mIface.length) {
98 throw new IllegalArgumentException("unexpected number of entries");
99 }
100 return new NetworkStats(mElapsedRealtime, mIface, mUid, mRx, mTx);
101 }
102 }
103
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700104 public int length() {
105 // length is identical for all fields
106 return iface.length;
107 }
108
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700109 /**
110 * Find first stats index that matches the requested parameters.
111 */
112 public int findIndex(String iface, int uid) {
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700113 final int length = length();
114 for (int i = 0; i < length; i++) {
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700115 if (equal(iface, this.iface[i]) && uid == this.uid[i]) {
116 return i;
117 }
118 }
119 return -1;
120 }
121
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700122 /**
123 * Subtract the given {@link NetworkStats}, effectively leaving the delta
124 * between two snapshots in time. Assumes that statistics rows collect over
125 * time, and that none of them have disappeared.
126 */
127 public NetworkStats subtract(NetworkStats value) {
128 // result will have our rows, but no meaningful timestamp
129 final int length = length();
130 final NetworkStats.Builder result = new NetworkStats.Builder(-1, length);
131
132 for (int i = 0; i < length; i++) {
133 final String iface = this.iface[i];
134 final int uid = this.uid[i];
135
136 // find remote row that matches, and subtract
137 final int j = value.findIndex(iface, uid);
138 if (j == -1) {
139 // newly appearing row, return entire value
140 result.addEntry(iface, uid, this.rx[i], this.tx[i]);
141 } else {
142 // existing row, subtract remote value
143 final long rx = this.rx[i] - value.rx[j];
144 final long tx = this.tx[i] - value.tx[j];
145 result.addEntry(iface, uid, rx, tx);
146 }
147 }
148
149 return result.build();
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700150 }
151
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700152 private static boolean equal(Object a, Object b) {
153 return a == b || (a != null && a.equals(b));
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700154 }
155
156 public void dump(String prefix, PrintWriter pw) {
157 pw.print(prefix);
158 pw.print("NetworkStats: elapsedRealtime="); pw.println(elapsedRealtime);
159 for (int i = 0; i < iface.length; i++) {
160 pw.print(prefix);
161 pw.print(" iface="); pw.print(iface[i]);
162 pw.print(" uid="); pw.print(uid[i]);
163 pw.print(" rx="); pw.print(rx[i]);
164 pw.print(" tx="); pw.println(tx[i]);
165 }
166 }
167
168 @Override
169 public String toString() {
170 final CharArrayWriter writer = new CharArrayWriter();
171 dump("", new PrintWriter(writer));
172 return writer.toString();
173 }
174
175 /** {@inheritDoc} */
Jeff Sharkeyeedcb952011-05-17 14:55:15 -0700176 public int describeContents() {
177 return 0;
178 }
179
180 /** {@inheritDoc} */
Jeff Sharkey9a13f362011-04-26 16:25:36 -0700181 public void writeToParcel(Parcel dest, int flags) {
182 dest.writeLong(elapsedRealtime);
183 dest.writeStringArray(iface);
184 dest.writeIntArray(uid);
185 dest.writeLongArray(rx);
186 dest.writeLongArray(tx);
187 }
188
189 public static final Creator<NetworkStats> CREATOR = new Creator<NetworkStats>() {
190 public NetworkStats createFromParcel(Parcel in) {
191 return new NetworkStats(in);
192 }
193
194 public NetworkStats[] newArray(int size) {
195 return new NetworkStats[size];
196 }
197 };
198}