1 /*
2  * Copyright (C) 2016 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 package android.telephony;
17 
18 import android.annotation.SystemApi;
19 import android.os.Parcel;
20 import android.os.Parcelable;
21 
22 import java.util.ArrayList;
23 import java.util.Arrays;
24 import java.util.List;
25 
26 /**
27  * Parcelable class to store Telephony histogram.
28  * @hide
29  */
30 @SystemApi
31 public final class TelephonyHistogram implements Parcelable {
32     // Type of Telephony histogram Eg: RIL histogram will have all timing data associated with
33     // RIL calls. Similarly we can have any other Telephony histogram.
34     private final int mCategory;
35 
36     // Unique Id identifying a sample within particular category of histogram
37     private final int mId;
38 
39     // Min time taken in ms
40     private int mMinTimeMs;
41 
42     // Max time taken in ms
43     private int mMaxTimeMs;
44 
45     // Average time taken in ms
46     private int mAverageTimeMs;
47 
48     // Total count of samples
49     private int mSampleCount;
50 
51     // Array storing time taken for first #RANGE_CALCULATION_COUNT samples of histogram.
52     private int[] mInitialTimings;
53 
54     // Total number of time ranges expected (must be greater than 1)
55     private final int mBucketCount;
56 
57     // Array storing endpoints of range buckets. Calculated based on values of minTime & maxTime
58     // after totalTimeCount is #RANGE_CALCULATION_COUNT.
59     private final int[] mBucketEndPoints;
60 
61     // Array storing counts for each time range starting from smallest value range
62     private final int[] mBucketCounters;
63 
64     /**
65      * Constant for Telephony category
66      */
67     public static final int TELEPHONY_CATEGORY_RIL = 1;
68 
69     // Count of Histogram samples after which time buckets are created.
70     private static final int RANGE_CALCULATION_COUNT = 10;
71 
72 
73     // Constant used to indicate #initialTimings is null while parceling
74     private static final int ABSENT = 0;
75 
76     // Constant used to indicate #initialTimings is not null while parceling
77     private static final int PRESENT = 1;
78 
79     // Throws exception if #totalBuckets is not greater than one.
TelephonyHistogram(int category, int id, int bucketCount)80     public TelephonyHistogram (int category, int id, int bucketCount) {
81         if (bucketCount <= 1) {
82             throw new IllegalArgumentException("Invalid number of buckets");
83         }
84         mCategory = category;
85         mId = id;
86         mMinTimeMs = Integer.MAX_VALUE;
87         mMaxTimeMs = 0;
88         mAverageTimeMs = 0;
89         mSampleCount = 0;
90         mInitialTimings = new int[RANGE_CALCULATION_COUNT];
91         mBucketCount = bucketCount;
92         mBucketEndPoints = new int[bucketCount - 1];
93         mBucketCounters = new int[bucketCount];
94     }
95 
TelephonyHistogram(TelephonyHistogram th)96     public TelephonyHistogram(TelephonyHistogram th) {
97         mCategory = th.getCategory();
98         mId = th.getId();
99         mMinTimeMs = th.getMinTime();
100         mMaxTimeMs = th.getMaxTime();
101         mAverageTimeMs = th.getAverageTime();
102         mSampleCount = th.getSampleCount();
103         mInitialTimings = th.getInitialTimings();
104         mBucketCount = th.getBucketCount();
105         mBucketEndPoints = th.getBucketEndPoints();
106         mBucketCounters = th.getBucketCounters();
107     }
108 
getCategory()109     public int getCategory() {
110         return mCategory;
111     }
112 
getId()113     public int getId() {
114         return mId;
115     }
116 
getMinTime()117     public int getMinTime() {
118         return mMinTimeMs;
119     }
120 
getMaxTime()121     public int getMaxTime() {
122         return mMaxTimeMs;
123     }
124 
getAverageTime()125     public int getAverageTime() {
126         return mAverageTimeMs;
127     }
128 
getSampleCount()129     public int getSampleCount () {
130         return mSampleCount;
131     }
132 
getInitialTimings()133     private int[] getInitialTimings() {
134         return mInitialTimings;
135     }
136 
getBucketCount()137     public int getBucketCount() {
138         return mBucketCount;
139     }
140 
getBucketEndPoints()141     public int[] getBucketEndPoints() {
142         if (mSampleCount > 1 && mSampleCount < 10) {
143             int[] tempEndPoints = new int[mBucketCount - 1];
144             calculateBucketEndPoints(tempEndPoints);
145             return tempEndPoints;
146         } else {
147             return getDeepCopyOfArray(mBucketEndPoints);
148         }
149     }
150 
getBucketCounters()151     public int[] getBucketCounters() {
152         if (mSampleCount > 1 && mSampleCount < 10) {
153             int[] tempEndPoints = new int[mBucketCount - 1];
154             int[] tempBucketCounters = new int[mBucketCount];
155             calculateBucketEndPoints(tempEndPoints);
156             for (int j = 0; j < mSampleCount; j++) {
157                 addToBucketCounter(tempEndPoints, tempBucketCounters, mInitialTimings[j]);
158             }
159             return tempBucketCounters;
160         } else {
161             return getDeepCopyOfArray(mBucketCounters);
162         }
163     }
164 
getDeepCopyOfArray(int[] array)165     private int[] getDeepCopyOfArray(int[] array) {
166         int[] clone = new int[array.length];
167         System.arraycopy(array, 0, clone, 0, array.length);
168         return clone;
169     }
170 
addToBucketCounter(int[] bucketEndPoints, int[] bucketCounters, int time)171     private void addToBucketCounter(int[] bucketEndPoints, int[] bucketCounters, int time) {
172         int i;
173         for (i = 0; i < bucketEndPoints.length; i++) {
174             if (time <= bucketEndPoints[i]) {
175                 bucketCounters[i]++;
176                 return;
177             }
178         }
179         bucketCounters[i]++;
180     }
181 
calculateBucketEndPoints(int[] bucketEndPoints)182     private void calculateBucketEndPoints(int[] bucketEndPoints) {
183         for (int i = 1; i < mBucketCount; i++) {
184             int endPt = mMinTimeMs + (i * (mMaxTimeMs - mMinTimeMs)) / mBucketCount;
185             bucketEndPoints[i - 1] = endPt;
186         }
187     }
188 
189     // Add new value of time taken
190     // This function updates minTime, maxTime, averageTime & totalTimeCount every time it is
191     // called. initialTimings[] is updated if totalTimeCount <= #RANGE_CALCULATION_COUNT. When
192     // totalTimeCount = RANGE_CALCULATION_COUNT, based on the min, max time & the number of buckets
193     // expected, bucketEndPoints[] would be calculated. Then bucketCounters[] would be filled up
194     // using values stored in initialTimings[]. Thereafter bucketCounters[] will always be updated.
addTimeTaken(int time)195     public void addTimeTaken(int time) {
196         // Initialize all fields if its first entry or if integer overflow is going to occur while
197         // trying to calculate averageTime
198         if (mSampleCount == 0 || (mSampleCount == Integer.MAX_VALUE)) {
199             if (mSampleCount == 0) {
200                 mMinTimeMs = time;
201                 mMaxTimeMs = time;
202                 mAverageTimeMs = time;
203             } else {
204                 mInitialTimings = new int[RANGE_CALCULATION_COUNT];
205             }
206             mSampleCount = 1;
207             Arrays.fill(mInitialTimings, 0);
208             mInitialTimings[0] = time;
209             Arrays.fill(mBucketEndPoints, 0);
210             Arrays.fill(mBucketCounters, 0);
211         } else {
212             if (time < mMinTimeMs) {
213                 mMinTimeMs = time;
214             }
215             if (time > mMaxTimeMs) {
216                 mMaxTimeMs = time;
217             }
218             long totalTime = ((long)mAverageTimeMs) * mSampleCount + time;
219             mAverageTimeMs = (int)(totalTime/++mSampleCount);
220 
221             if (mSampleCount < RANGE_CALCULATION_COUNT) {
222                 mInitialTimings[mSampleCount - 1] = time;
223             } else if (mSampleCount == RANGE_CALCULATION_COUNT) {
224                 mInitialTimings[mSampleCount - 1] = time;
225 
226                 // Calculate bucket endpoints based on bucketCount expected
227                 calculateBucketEndPoints(mBucketEndPoints);
228 
229                 // Use values stored in initialTimings[] to update bucketCounters
230                 for (int j = 0; j < RANGE_CALCULATION_COUNT; j++) {
231                     addToBucketCounter(mBucketEndPoints, mBucketCounters, mInitialTimings[j]);
232                 }
233                 mInitialTimings = null;
234             } else {
235                 addToBucketCounter(mBucketEndPoints, mBucketCounters, time);
236             }
237 
238         }
239     }
240 
241     public String toString() {
242         String basic = " Histogram id = " + mId + " Time(ms): min = " + mMinTimeMs + " max = "
243                 + mMaxTimeMs + " avg = " + mAverageTimeMs + " Count = " + mSampleCount;
244         if (mSampleCount < RANGE_CALCULATION_COUNT) {
245             return basic;
246         } else {
247             StringBuffer intervals = new StringBuffer(" Interval Endpoints:");
248             for (int i = 0; i < mBucketEndPoints.length; i++) {
249                 intervals.append(" " + mBucketEndPoints[i]);
250             }
251             intervals.append(" Interval counters:");
252             for (int i = 0; i < mBucketCounters.length; i++) {
253                 intervals.append(" " + mBucketCounters[i]);
254             }
255             return basic + intervals;
256         }
257     }
258 
259     public static final Parcelable.Creator<TelephonyHistogram> CREATOR =
260             new Parcelable.Creator<TelephonyHistogram> () {
261 
262                 @Override
263                 public TelephonyHistogram createFromParcel(Parcel in) {
264                     return new TelephonyHistogram(in);
265                 }
266 
267                 @Override
268                 public TelephonyHistogram[] newArray(int size) {
269                     return new TelephonyHistogram[size];
270                 }
271             };
272 
273     public TelephonyHistogram(Parcel in) {
274         mCategory = in.readInt();
275         mId = in.readInt();
276         mMinTimeMs = in.readInt();
277         mMaxTimeMs = in.readInt();
278         mAverageTimeMs = in.readInt();
279         mSampleCount = in.readInt();
280         if (in.readInt() == PRESENT) {
281             mInitialTimings = new int[RANGE_CALCULATION_COUNT];
282             in.readIntArray(mInitialTimings);
283         }
284         mBucketCount = in.readInt();
285         mBucketEndPoints = new int[mBucketCount - 1];
286         in.readIntArray(mBucketEndPoints);
287         mBucketCounters = new int[mBucketCount];
288         in.readIntArray(mBucketCounters);
289     }
290 
291     public void writeToParcel(Parcel out, int flags) {
292         out.writeInt(mCategory);
293         out.writeInt(mId);
294         out.writeInt(mMinTimeMs);
295         out.writeInt(mMaxTimeMs);
296         out.writeInt(mAverageTimeMs);
297         out.writeInt(mSampleCount);
298         if (mInitialTimings == null) {
299             out.writeInt(ABSENT);
300         } else {
301             out.writeInt(PRESENT);
302             out.writeIntArray(mInitialTimings);
303         }
304         out.writeInt(mBucketCount);
305         out.writeIntArray(mBucketEndPoints);
306         out.writeIntArray(mBucketCounters);
307     }
308 
309     @Override
310     public int describeContents() {
311         return 0;
312     }
313 }
314