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 17 package com.android.server.power.stats; 18 19 import android.os.BatteryStats; 20 import android.os.Parcel; 21 import android.util.Log; 22 23 import androidx.test.filters.SmallTest; 24 25 import junit.framework.Assert; 26 import junit.framework.TestCase; 27 28 import org.mockito.Mockito; 29 30 import java.io.PrintWriter; 31 import java.io.StringWriter; 32 33 /** 34 * Provides test cases for android.os.BatteryStats. 35 */ 36 public class BatteryStatsTimeBaseTest extends TestCase { 37 private static final String TAG = "BatteryStatsTimeBaseTest"; 38 39 static class TestTimeBase extends BatteryStatsImpl.TimeBase { 40 populate(long uptime, long realtime, boolean running, long pastUptime, long uptimeStart, long pastRealtime, long realtimeStart, long unpluggedUptime, long unpluggedRealtime)41 public void populate(long uptime, long realtime, boolean running, long pastUptime, 42 long uptimeStart, long pastRealtime, long realtimeStart, 43 long unpluggedUptime, long unpluggedRealtime) { 44 mUptimeUs = uptime; 45 mRealtimeUs = realtime; 46 mRunning = running; 47 mPastUptimeUs = pastUptime; 48 mUptimeStartUs = uptimeStart; 49 mPastRealtimeUs = pastRealtime; 50 mRealtimeStartUs = realtimeStart; 51 mUnpluggedUptimeUs = unpluggedUptime; 52 mUnpluggedRealtimeUs = unpluggedRealtime; 53 } 54 verify(long uptime, long realtime, boolean running, long pastUptime, long uptimeStart, long pastRealtime, long realtimeStart, long unpluggedUptime, long unpluggedRealtime)55 public void verify(long uptime, long realtime, boolean running, long pastUptime, 56 long uptimeStart, long pastRealtime, long realtimeStart, 57 long unpluggedUptime, long unpluggedRealtime) { 58 Assert.assertEquals(uptime, mUptimeUs); 59 Assert.assertEquals(realtime, mRealtimeUs); 60 Assert.assertEquals(running, mRunning); 61 Assert.assertEquals(pastUptime, mPastUptimeUs); 62 Assert.assertEquals(uptimeStart, mUptimeStartUs); 63 Assert.assertEquals(pastRealtime, mPastRealtimeUs); 64 Assert.assertEquals(realtimeStart, mRealtimeStartUs); 65 Assert.assertEquals(unpluggedUptime, mUnpluggedUptimeUs); 66 Assert.assertEquals(unpluggedRealtime, mUnpluggedRealtimeUs); 67 } 68 } 69 70 /** 71 * Test the observers and the setRunning call. 72 */ 73 @SmallTest testRunning()74 public void testRunning() throws Exception { 75 TestTimeBase tb = new TestTimeBase(); 76 77 // Toggle running once, to accumulate past uptime and past realtime 78 // so the test values aren't 0. 79 tb.setRunning(true, 100, 10000); 80 tb.setRunning(false, 200, 11000); 81 Assert.assertEquals(100, tb.getUptimeStart()); 82 Assert.assertEquals(10000, tb.getRealtimeStart()); 83 84 // Create some observers 85 BatteryStatsImpl.TimeBaseObs observer1 = Mockito.mock(BatteryStatsImpl.TimeBaseObs.class); 86 BatteryStatsImpl.TimeBaseObs observer2 = Mockito.mock(BatteryStatsImpl.TimeBaseObs.class); 87 BatteryStatsImpl.TimeBaseObs observer3 = Mockito.mock(BatteryStatsImpl.TimeBaseObs.class); 88 89 // Add them 90 tb.add(observer1); 91 tb.add(observer2); 92 tb.add(observer3); 93 Assert.assertTrue(tb.hasObserver(observer1)); 94 Assert.assertTrue(tb.hasObserver(observer2)); 95 Assert.assertTrue(tb.hasObserver(observer3)); 96 97 // Remove one 98 tb.remove(observer3); 99 Assert.assertTrue(tb.hasObserver(observer1)); 100 Assert.assertTrue(tb.hasObserver(observer2)); 101 Assert.assertFalse(tb.hasObserver(observer3)); 102 103 // Start running, make sure we get a started call on the two active observers 104 // and not the third. 105 tb.setRunning(true, 250, 14000); 106 107 Assert.assertTrue(tb.isRunning()); 108 109 if (false) { 110 Log.d(TAG, "mUptimeStart=" + tb.getUptimeStart() 111 + " mRealtimeStart=" + tb.getRealtimeStart() 112 + " mUptime=" + tb.getUptime(250) 113 + " mRealtime=" + tb.getRealtime(14000) 114 + " isRunning=" + tb.isRunning()); 115 } 116 117 Assert.assertEquals(250, tb.getUptimeStart()); 118 Assert.assertEquals(14000, tb.getRealtimeStart()); 119 Assert.assertEquals(100, tb.getUptime(250)); 120 Assert.assertEquals(1000, tb.getRealtime(14000)); 121 122 Mockito.verify(observer1).onTimeStarted(14000, 100, 1000); 123 Mockito.verify(observer1, Mockito.never()).onTimeStopped(-1, -1, -1); 124 Mockito.verifyNoMoreInteractions(observer1); 125 Mockito.verify(observer2).onTimeStarted(14000, 100, 1000); 126 Mockito.verify(observer2, Mockito.never()).onTimeStopped(-1, -1, -1); 127 Mockito.verifyNoMoreInteractions(observer2); 128 129 Mockito.reset(observer1); 130 Mockito.reset(observer2); 131 Mockito.reset(observer3); 132 133 // Advance the "timer" and make sure the getters account for the current time passed in 134 Assert.assertEquals(400, tb.getUptime(550)); 135 Assert.assertEquals(1555, tb.getRealtime(14555)); 136 137 // Stop running, make sure we get a stopped call on the two active observers 138 // and not the third. 139 tb.setRunning(false, 402, 14002); 140 141 Assert.assertFalse(tb.isRunning()); 142 143 if (false) { 144 Log.d(TAG, "mUptimeStart=" + tb.getUptimeStart() 145 + " mRealtimeStart=" + tb.getRealtimeStart() 146 + " mUptime=" + tb.getUptime(250) 147 + " mRealtime=" + tb.getRealtime(14000) 148 + " isRunning=" + tb.isRunning()); 149 } 150 151 Assert.assertEquals(252, tb.getUptime(402)); 152 Assert.assertEquals(1002, tb.getRealtime(14002)); 153 154 Mockito.verify(observer1).onTimeStopped(14002, 252, 1002); 155 Mockito.verify(observer1, Mockito.never()).onTimeStopped(-1, -1, -1); 156 Mockito.verifyNoMoreInteractions(observer1); 157 Mockito.verify(observer2).onTimeStopped(14002, 252, 1002); 158 Mockito.verify(observer2, Mockito.never()).onTimeStopped(-1, -1, -1); 159 Mockito.verifyNoMoreInteractions(observer2); 160 161 // Advance the "timer" and make sure the getters account for the current time passed in 162 // is the same as the time when running went to false. 163 Assert.assertEquals(252, tb.getUptime(600)); 164 Assert.assertEquals(1002, tb.getRealtime(17000)); 165 } 166 167 /** 168 * Test that reset while running updates the plugged and unplugged times 169 */ 170 @SmallTest testResetWhileRunning()171 public void testResetWhileRunning() throws Exception { 172 TestTimeBase tb = new TestTimeBase(); 173 tb.populate(100, 200, true, 300, 400, 500, 600, 700, 800); 174 175 tb.reset(666, 6666); 176 177 // Not sure if this is a bug: reset while running does not 178 // reset mPastUptime, but while it is running it does. 179 tb.verify(100, 200, true, 300, 666, 500, 6666, 300, 500); 180 } 181 182 /** 183 * Test that reset while running updates the plugged and unplugged times 184 */ 185 @SmallTest testResetWhileNotRunning()186 public void testResetWhileNotRunning() throws Exception { 187 TestTimeBase tb = new TestTimeBase(); 188 tb.populate(100, 200, false, 300, 400, 500, 600, 700, 800); 189 190 tb.reset(666, 6666); 191 192 tb.verify(100, 200, false, 0, 400, 0, 600, 700, 800); 193 } 194 195 /** 196 * Test init 197 */ 198 @SmallTest testInit()199 public void testInit() throws Exception { 200 TestTimeBase tb = new TestTimeBase(); 201 tb.populate(100, 200, false, 300, 400, 500, 600, 700, 800); 202 203 tb.init(666, 6666); 204 205 tb.verify(0, 0, false, 0, 666, 0, 6666, 0, 0); 206 } 207 208 /** 209 * Test writeToParcel and readFromParcel 210 */ 211 @SmallTest testParcellingWhileRunning()212 public void testParcellingWhileRunning() throws Exception { 213 TestTimeBase tb1 = new TestTimeBase(); 214 215 tb1.populate(100, 200, true, 300, 400, 500, 600, 700, 800); 216 217 Parcel parcel = Parcel.obtain(); 218 tb1.writeToParcel(parcel, 666, 6666); 219 220 parcel.setDataPosition(0); 221 222 TestTimeBase tb2 = new TestTimeBase(); 223 tb2.readFromParcel(parcel); 224 225 // Running is not preserved across parceling 226 tb2.verify(100, 200, false, 300+666-400, 400, 500+6666-600, 600, 700, 800); 227 } 228 229 /** 230 * Test writeToParcel and readFromParcel 231 */ 232 @SmallTest testParcellingWhileNotRunning()233 public void testParcellingWhileNotRunning() throws Exception { 234 TestTimeBase tb1 = new TestTimeBase(); 235 236 tb1.populate(100, 200, false, 300, 400, 500, 600, 700, 800); 237 238 Parcel parcel = Parcel.obtain(); 239 tb1.writeToParcel(parcel, 666, 6666); 240 241 parcel.setDataPosition(0); 242 243 TestTimeBase tb2 = new TestTimeBase(); 244 tb2.readFromParcel(parcel); 245 246 tb2.verify(100, 200, false, 300, 400, 500, 600, 700, 800); 247 } 248 249 /** 250 * Test writeSummaryToParcel and readSummaryFromParcel 251 */ 252 @SmallTest testSummary()253 public void testSummary() throws Exception { 254 TestTimeBase tb1 = new TestTimeBase(); 255 256 tb1.populate(100, 200, true, 300, 400, 500, 600, 700, 800); 257 258 Parcel parcel = Parcel.obtain(); 259 tb1.writeSummaryToParcel(parcel, 666, 6666); 260 261 parcel.setDataPosition(0); 262 263 TestTimeBase tb2 = new TestTimeBase(); 264 265 // readSummaryFromParcel doesn't affect the other fields. 266 // Not sure if this is deliberate 267 tb2.populate(1, 2, true, 3, 4, 5, 6, 7, 8); 268 269 tb2.readSummaryFromParcel(parcel); 270 271 tb2.verify(666, 6766, true, 3, 4, 5, 6, 7, 8); 272 } 273 274 /** 275 * Test computeUptime 276 */ 277 @SmallTest testComputeUptime()278 public void testComputeUptime() throws Exception { 279 TestTimeBase tb = new TestTimeBase(); 280 281 tb.populate(100, 200, true, 300, 400, 500, 600, 50, 60); 282 283 Assert.assertEquals(100+300+666-400, 284 tb.computeUptime(666, BatteryStats.STATS_SINCE_CHARGED)); 285 } 286 287 /** 288 * Test computeUptime 289 */ 290 @SmallTest testComputeRealtime()291 public void testComputeRealtime() throws Exception { 292 TestTimeBase tb = new TestTimeBase(); 293 294 tb.populate(100, 200, true, 300, 400, 500, 600, 50, 60); 295 296 Assert.assertEquals(200+500+6666-600, 297 tb.computeRealtime(6666, BatteryStats.STATS_SINCE_CHARGED)); 298 } 299 300 /** 301 * Test dump 302 */ 303 @SmallTest testDump()304 public void testDump() throws Exception { 305 TestTimeBase tb = new TestTimeBase(); 306 307 tb.populate(100, 200, true, 300, 400, 500, 600, 50, 60); 308 309 StringWriter sw = new StringWriter(); 310 PrintWriter pw = new PrintWriter(sw); 311 312 tb.dump(pw, "+++++ "); 313 314 pw.close(); 315 316 // note the spaces at the ends of the lines which come from formatTimeMs. 317 final String CORRECT = "+++++ mRunning=true\n" 318 + "+++++ mUptime=0ms \n" 319 + "+++++ mRealtime=0ms \n" 320 + "+++++ mPastUptime=0ms mUptimeStart=0ms mUnpluggedUptime=0ms \n" 321 + "+++++ mPastRealtime=0ms mRealtimeStart=0ms mUnpluggedRealtime=0ms \n"; 322 323 Assert.assertEquals(CORRECT, sw.toString()); 324 } 325 326 } 327 328