1 /* 2 * Copyright (C) 2014 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.cts.verifier.sensors; 18 19 import java.util.Timer; 20 import java.util.TimerTask; 21 import java.util.concurrent.CountDownLatch; 22 import java.util.concurrent.TimeUnit; 23 24 import com.android.cts.verifier.R; 25 import com.android.cts.verifier.sensors.base.SensorCtsVerifierTestActivity; 26 import com.android.cts.verifier.sensors.helpers.SensorTestScreenManipulator; 27 28 import android.app.AlarmManager; 29 import android.app.PendingIntent; 30 import android.content.BroadcastReceiver; 31 import android.content.Context; 32 import android.content.Intent; 33 import android.content.IntentFilter; 34 import android.hardware.cts.helpers.SensorNotSupportedException; 35 import android.hardware.cts.helpers.SuspendStateMonitor; 36 import android.hardware.cts.helpers.TestSensorEnvironment; 37 import android.hardware.Sensor; 38 import android.hardware.SensorManager; 39 import android.hardware.TriggerEvent; 40 import android.hardware.TriggerEventListener; 41 import android.os.Handler; 42 import android.os.Looper; 43 import android.os.PowerManager; 44 import android.os.PowerManager.WakeLock; 45 import android.os.SystemClock; 46 import androidx.localbroadcastmanager.content.LocalBroadcastManager; 47 import android.util.Log; 48 49 import junit.framework.Assert; 50 51 /** 52 * Test cases for Significant Motion sensor. 53 * They use walking motion to change the location and trigger Significant Motion. 54 */ 55 public class SignificantMotionTestActivity extends SensorCtsVerifierTestActivity { SignificantMotionTestActivity()56 public SignificantMotionTestActivity() { 57 super(SignificantMotionTestActivity.class, true); 58 } 59 60 // acceptable time difference between event time and system time 61 private static final long MAX_ACCEPTABLE_EVENT_TIME_DELAY_NANOS = 62 TimeUnit.MILLISECONDS.toNanos(500); 63 64 // acceptable time difference between event time and AP wake up time. 65 private static final long MAX_ACCEPTABLE_DELAY_EVENT_AP_WAKE_UP_NS = 66 TimeUnit.MILLISECONDS.toNanos(2000); 67 68 // time to wait for SMD after the device has gone into suspend. Even after 69 // 45 secs if SMD does not trigger, the test will fail. 70 private static final long ALARM_WAKE_TIME_DELAY_MS = TimeUnit.SECONDS.toMillis(45); 71 72 // time for the test to wait for a trigger 73 private static final int TRIGGER_MAX_DELAY_SECONDS = 30; 74 private static final int VIBRATE_DURATION_MILLIS = 10000; 75 76 private static final int EVENT_VALUES_LENGTH = 1; 77 private static final float EXPECTED_EVENT_VALUE = 1.0f; 78 private static String ACTION_ALARM = "SignificantMotionTestActivity.ACTION_ALARM"; 79 80 private SensorManager mSensorManager; 81 private Sensor mSensorSignificantMotion; 82 private TriggerVerifier mVerifier; 83 private SensorTestScreenManipulator mScreenManipulator; 84 private WakeLock mPartialWakeLock; 85 86 /** 87 * Test cases. 88 */ 89 @SuppressWarnings("unused") testTrigger()90 public String testTrigger() throws Throwable { 91 return runTest( 92 R.string.snsr_significant_motion_test_trigger, 93 true /* isMotionExpected */, 94 false /* cancelEventNotification */, 95 false /* vibrate */); 96 } 97 98 @SuppressWarnings("unused") testNotTriggerAfterCancel()99 public String testNotTriggerAfterCancel() throws Throwable { 100 return runTest( 101 R.string.snsr_significant_motion_test_cancel, 102 false /* isMotionExpected */, 103 true /* cancelEventNotification */, 104 false /* vibrate */); 105 } 106 107 /** 108 * Verifies that Significant Motion is not trigger by the vibrator motion. 109 */ 110 @SuppressWarnings("unused") testVibratorDoesNotTrigger()111 public String testVibratorDoesNotTrigger() throws Throwable { 112 return runTest( 113 R.string.snsr_significant_motion_test_vibration, 114 false /* isMotionExpected */, 115 false /* cancelEventNotification */, 116 true /* vibrate */); 117 } 118 119 /** 120 * Verifies that the natural motion of keeping the device in hand does not change the location. 121 * It ensures that Significant Motion will not trigger in that scenario. 122 */ 123 @SuppressWarnings("unused") testInHandDoesNotTrigger()124 public String testInHandDoesNotTrigger() throws Throwable { 125 return runTest( 126 R.string.snsr_significant_motion_test_in_hand, 127 false /* isMotionExpected */, 128 false /* cancelEventNotification */, 129 false /* vibrate */); 130 } 131 132 @SuppressWarnings("unused") testSittingDoesNotTrigger()133 public String testSittingDoesNotTrigger() throws Throwable { 134 return runTest( 135 R.string.snsr_significant_motion_test_sitting, 136 false /* isMotionExpected */, 137 false /* cancelEventNotification */, 138 false /* vibrate */); 139 } 140 141 @SuppressWarnings("unused") testTriggerDeactivation()142 public String testTriggerDeactivation() throws Throwable { 143 144 setFirstExecutionInstruction(R.string.snsr_significant_motion_test_deactivation); 145 146 TriggerVerifier verifier = new TriggerVerifier(); 147 mSensorManager.requestTriggerSensor(verifier, mSensorSignificantMotion); 148 getTestLogger().logWaitForSound(); 149 150 String result; 151 try { 152 mPartialWakeLock.acquire(); 153 154 // wait for the first event to trigger 155 verifier.verifyEventTriggered(); 156 157 // wait for a second event not to trigger 158 result = verifier.verifyEventNotTriggered(); 159 } finally { 160 playSound(); 161 mScreenManipulator.turnScreenOn(); 162 mPartialWakeLock.release(); 163 } 164 return result; 165 } 166 167 public static class AlarmReceiver extends BroadcastReceiver { 168 @Override onReceive(Context context, Intent intent)169 public void onReceive(Context context, Intent intent) { 170 Intent alarm_intent = new Intent(context, SignificantMotionTestActivity.class); 171 alarm_intent.setAction(SignificantMotionTestActivity.ACTION_ALARM); 172 LocalBroadcastManager.getInstance(context).sendBroadcastSync(alarm_intent); 173 } 174 } 175 176 public BroadcastReceiver myBroadCastReceiver = new BroadcastReceiver() { 177 @Override 178 public void onReceive(Context context, Intent intent) { 179 mVerifier.releaseLatch(); 180 mScreenManipulator.turnScreenOn(); 181 try { 182 playSound(); 183 } catch (InterruptedException e) { 184 // Ignore ... 185 } 186 } 187 }; 188 189 @SuppressWarnings("unused") testAPWakeUpOnSMDTrigger()190 public String testAPWakeUpOnSMDTrigger() throws Throwable { 191 192 setFirstExecutionInstruction(R.string.snsr_significant_motion_ap_suspend); 193 194 mVerifier = new TriggerVerifier(); 195 mSensorManager.requestTriggerSensor(mVerifier, mSensorSignificantMotion); 196 long testStartTimeNs = SystemClock.elapsedRealtimeNanos(); 197 Handler handler = new Handler(Looper.getMainLooper()); 198 SuspendStateMonitor suspendStateMonitor = new SuspendStateMonitor(); 199 200 Intent intent = new Intent(this, AlarmReceiver.class); 201 PendingIntent pendingIntent = PendingIntent.getBroadcast(this, 0, intent, PendingIntent.FLAG_MUTABLE_UNAUDITED); 202 203 AlarmManager am = (AlarmManager) getSystemService(ALARM_SERVICE); 204 am.setExact(AlarmManager.ELAPSED_REALTIME_WAKEUP, 205 SystemClock.elapsedRealtime() + ALARM_WAKE_TIME_DELAY_MS, pendingIntent); 206 try { 207 // Wait for the first event to trigger. Device is expected to go into suspend here. 208 mVerifier.verifyEventTriggered(); 209 long eventTimeStampNs = mVerifier.getTimeStampForTriggerEvent(); 210 long endTimeNs = SystemClock.elapsedRealtimeNanos(); 211 long lastWakeupTimeNs = TimeUnit.MILLISECONDS.toNanos( 212 suspendStateMonitor.getLastWakeUpTime()); 213 Assert.assertTrue(getString(R.string.snsr_device_did_not_go_into_suspend), 214 testStartTimeNs < lastWakeupTimeNs && lastWakeupTimeNs < endTimeNs); 215 long timestampDelta = Math.abs(lastWakeupTimeNs - eventTimeStampNs); 216 Assert.assertTrue( 217 String.format(getString(R.string.snsr_device_did_not_wake_up_at_trigger), 218 TimeUnit.NANOSECONDS.toMillis(lastWakeupTimeNs), 219 TimeUnit.NANOSECONDS.toMillis(eventTimeStampNs)), 220 timestampDelta < MAX_ACCEPTABLE_DELAY_EVENT_AP_WAKE_UP_NS); 221 } finally { 222 am.cancel(pendingIntent); 223 suspendStateMonitor.cancel(); 224 mScreenManipulator.turnScreenOn(); 225 playSound(); 226 } 227 return null; 228 } 229 230 /** 231 * @param instructionsResId Instruction to be shown to testers 232 * @param isMotionExpected Should the device detect significant motion event 233 * for this test? 234 * @param cancelEventNotification If TRUE, motion notifications will be 235 * requested first and request will be cancelled 236 * @param vibrate If TRUE, vibration will be concurrent with the test 237 * @throws Throwable 238 */ 239 private String runTest( 240 int instructionsResId, 241 boolean isMotionExpected, 242 boolean cancelEventNotification, 243 boolean vibrate) throws Throwable { 244 245 setFirstExecutionInstruction(instructionsResId); 246 247 if (vibrate) { 248 vibrate(VIBRATE_DURATION_MILLIS); 249 } 250 251 TriggerVerifier verifier = new TriggerVerifier(); 252 boolean success = mSensorManager.requestTriggerSensor(verifier, mSensorSignificantMotion); 253 Assert.assertTrue( 254 getString(R.string.snsr_significant_motion_registration, success), 255 success); 256 if (cancelEventNotification) { 257 Assert.assertTrue( 258 getString(R.string.snsr_significant_motion_cancelation), 259 mSensorManager.cancelTriggerSensor(verifier, mSensorSignificantMotion)); 260 } 261 getTestLogger().logWaitForSound(); 262 263 String result; 264 try { 265 mPartialWakeLock.acquire(); 266 267 if (isMotionExpected) { 268 result = verifier.verifyEventTriggered(); 269 } else { 270 result = verifier.verifyEventNotTriggered(); 271 } 272 } finally { 273 mSensorManager.cancelTriggerSensor(verifier, mSensorSignificantMotion); 274 275 // notify user test finished 276 playSound(); 277 mScreenManipulator.turnScreenOn(); 278 mPartialWakeLock.release(); 279 } 280 return result; 281 } 282 283 @Override 284 protected void activitySetUp() { 285 mSensorManager = (SensorManager) getApplicationContext() 286 .getSystemService(Context.SENSOR_SERVICE); 287 mSensorSignificantMotion = mSensorManager.getDefaultSensor(Sensor.TYPE_SIGNIFICANT_MOTION); 288 if (mSensorSignificantMotion == null) { 289 throw new SensorNotSupportedException(Sensor.TYPE_SIGNIFICANT_MOTION); 290 } 291 292 mScreenManipulator = new SensorTestScreenManipulator(this); 293 try { 294 mScreenManipulator.initialize(this); 295 } catch (InterruptedException e) { 296 } 297 PowerManager pm = (PowerManager)getSystemService(Context.POWER_SERVICE); 298 mPartialWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "SignificantMotionTestActivity"); 299 LocalBroadcastManager.getInstance(this).registerReceiver(myBroadCastReceiver, 300 new IntentFilter(ACTION_ALARM)); 301 } 302 303 @Override 304 protected void activityCleanUp() { 305 if (mScreenManipulator != null) { 306 // after this screen does not have to be on constantly 307 mScreenManipulator.releaseScreenOn(); 308 } 309 if (mPartialWakeLock != null && mPartialWakeLock.isHeld()) { 310 mPartialWakeLock.release(); 311 } 312 LocalBroadcastManager.getInstance(this).unregisterReceiver(myBroadCastReceiver); 313 } 314 315 @Override 316 protected void onDestroy() { 317 super.onDestroy(); 318 if (mScreenManipulator != null){ 319 mScreenManipulator.close(); 320 } 321 } 322 323 /** 324 * Helper Trigger listener for testing. 325 * It cannot be reused. 326 */ 327 private class TriggerVerifier extends TriggerEventListener { 328 private volatile CountDownLatch mCountDownLatch; 329 private volatile TriggerEventRegistry mEventRegistry; 330 private volatile long mTimestampForTriggeredEvent = 0; 331 332 // TODO: refactor out if needed 333 private class TriggerEventRegistry { 334 public final TriggerEvent triggerEvent; 335 public final long realtimeTimestampNanos; 336 337 public TriggerEventRegistry(TriggerEvent event, long realtimeTimestampNanos) { 338 this.triggerEvent = event; 339 this.realtimeTimestampNanos = realtimeTimestampNanos; 340 } 341 } 342 343 public void onTrigger(TriggerEvent event) { 344 long elapsedRealtimeNanos = SystemClock.elapsedRealtimeNanos(); 345 mEventRegistry = new TriggerEventRegistry(event, elapsedRealtimeNanos); 346 mCountDownLatch.countDown(); 347 } 348 349 public void releaseLatch() { 350 if (mCountDownLatch != null) { 351 mCountDownLatch.countDown(); 352 } 353 } 354 355 public long getTimeStampForTriggerEvent() { 356 return mTimestampForTriggeredEvent; 357 } 358 359 public String verifyEventTriggered() throws Throwable { 360 TriggerEventRegistry registry = awaitForEvent(); 361 362 // verify an event arrived, and it is indeed a Significant Motion event 363 TriggerEvent event = registry.triggerEvent; 364 String eventArrivalMessage = 365 getString(R.string.snsr_significant_motion_event_arrival, event != null); 366 Assert.assertNotNull(eventArrivalMessage, event); 367 368 int eventType = event.sensor.getType(); 369 String eventTypeMessage = getString( 370 R.string.snsr_significant_motion_event_type, 371 Sensor.TYPE_SIGNIFICANT_MOTION, 372 eventType); 373 Assert.assertEquals(eventTypeMessage, Sensor.TYPE_SIGNIFICANT_MOTION, eventType); 374 375 String sensorName = event.sensor.getName(); 376 int valuesLength = event.values.length; 377 String valuesLengthMessage = getString( 378 R.string.snsr_event_length, 379 EVENT_VALUES_LENGTH, 380 valuesLength, 381 sensorName); 382 Assert.assertEquals(valuesLengthMessage, EVENT_VALUES_LENGTH, valuesLength); 383 384 float value = event.values[0]; 385 String valuesMessage = getString( 386 R.string.snsr_event_value, 387 EXPECTED_EVENT_VALUE, 388 value, 389 sensorName); 390 Assert.assertEquals(valuesMessage, EXPECTED_EVENT_VALUE, value); 391 392 long deltaThreshold = MAX_ACCEPTABLE_EVENT_TIME_DELAY_NANOS 393 + TestSensorEnvironment.getSensorMaxDetectionLatencyNs(event.sensor); 394 return assertTimestampSynchronization( 395 event.timestamp, 396 registry.realtimeTimestampNanos, 397 deltaThreshold, 398 sensorName); 399 } 400 401 public String verifyEventNotTriggered() throws Throwable { 402 TriggerEventRegistry registry = awaitForEvent(); 403 404 TriggerEvent event = registry.triggerEvent; 405 String eventMessage = 406 getString(R.string.snsr_significant_motion_event_unexpected, event != null); 407 Assert.assertNull(eventMessage, event); 408 return eventMessage; 409 } 410 411 private TriggerEventRegistry awaitForEvent() throws InterruptedException { 412 mCountDownLatch = new CountDownLatch(1); 413 mCountDownLatch.await(TRIGGER_MAX_DELAY_SECONDS, TimeUnit.SECONDS); 414 TriggerEventRegistry registry = mEventRegistry; 415 416 // Save the last timestamp when the event triggered. 417 if (mEventRegistry != null && mEventRegistry.triggerEvent != null) { 418 mTimestampForTriggeredEvent = mEventRegistry.triggerEvent.timestamp; 419 } 420 421 mEventRegistry = null; 422 return registry != null ? registry : new TriggerEventRegistry(null, 0); 423 } 424 } 425 } 426