1 /* 2 * Copyright (C) 2008 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.hardware; 18 19 import android.annotation.SystemApi; 20 import android.annotation.SystemService; 21 import android.compat.annotation.UnsupportedAppUsage; 22 import android.content.Context; 23 import android.os.Build; 24 import android.os.Handler; 25 import android.os.MemoryFile; 26 import android.util.Log; 27 import android.util.SparseArray; 28 29 import java.util.ArrayList; 30 import java.util.Collections; 31 import java.util.List; 32 33 /** 34 * <p> 35 * SensorManager lets you access the device's {@link android.hardware.Sensor 36 * sensors}. 37 * </p> 38 * <p> 39 * Always make sure to disable sensors you don't need, especially when your 40 * activity is paused. Failing to do so can drain the battery in just a few 41 * hours. Note that the system will <i>not</i> disable sensors automatically when 42 * the screen turns off. 43 * </p> 44 * <p class="note"> 45 * Note: Don't use this mechanism with a Trigger Sensor, have a look 46 * at {@link TriggerEventListener}. {@link Sensor#TYPE_SIGNIFICANT_MOTION} 47 * is an example of a trigger sensor. 48 * </p> 49 * <p> 50 * In order to access sensor data at high sampling rates (i.e. greater than 200 Hz 51 * for {@link SensorEventListener} and greater than {@link SensorDirectChannel#RATE_NORMAL} 52 * for {@link SensorDirectChannel}), apps must declare 53 * the {@link android.Manifest.permission#HIGH_SAMPLING_RATE_SENSORS} permission 54 * in their AndroidManifest.xml file. 55 * </p> 56 * <pre class="prettyprint"> 57 * public class SensorActivity extends Activity implements SensorEventListener { 58 * private final SensorManager mSensorManager; 59 * private final Sensor mAccelerometer; 60 * 61 * public SensorActivity() { 62 * mSensorManager = (SensorManager)getSystemService(SENSOR_SERVICE); 63 * mAccelerometer = mSensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER); 64 * } 65 * 66 * protected void onResume() { 67 * super.onResume(); 68 * mSensorManager.registerListener(this, mAccelerometer, SensorManager.SENSOR_DELAY_NORMAL); 69 * } 70 * 71 * protected void onPause() { 72 * super.onPause(); 73 * mSensorManager.unregisterListener(this); 74 * } 75 * 76 * public void onAccuracyChanged(Sensor sensor, int accuracy) { 77 * } 78 * 79 * public void onSensorChanged(SensorEvent event) { 80 * } 81 * } 82 * </pre> 83 * 84 * @see SensorEventListener 85 * @see SensorEvent 86 * @see Sensor 87 * 88 */ 89 @SystemService(Context.SENSOR_SERVICE) 90 public abstract class SensorManager { 91 /** @hide */ 92 protected static final String TAG = "SensorManager"; 93 94 private static final float[] sTempMatrix = new float[16]; 95 96 // Cached lists of sensors by type. Guarded by mSensorListByType. 97 private final SparseArray<List<Sensor>> mSensorListByType = 98 new SparseArray<List<Sensor>>(); 99 100 // Legacy sensor manager implementation. Guarded by mSensorListByType during initialization. 101 private LegacySensorManager mLegacySensorManager; 102 103 /* NOTE: sensor IDs must be a power of 2 */ 104 105 /** 106 * A constant describing an orientation sensor. See 107 * {@link android.hardware.SensorListener SensorListener} for more details. 108 * 109 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 110 */ 111 @Deprecated 112 public static final int SENSOR_ORIENTATION = 1 << 0; 113 114 /** 115 * A constant describing an accelerometer. See 116 * {@link android.hardware.SensorListener SensorListener} for more details. 117 * 118 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 119 */ 120 @Deprecated 121 public static final int SENSOR_ACCELEROMETER = 1 << 1; 122 123 /** 124 * A constant describing a temperature sensor See 125 * {@link android.hardware.SensorListener SensorListener} for more details. 126 * 127 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 128 */ 129 @Deprecated 130 public static final int SENSOR_TEMPERATURE = 1 << 2; 131 132 /** 133 * A constant describing a magnetic sensor See 134 * {@link android.hardware.SensorListener SensorListener} for more details. 135 * 136 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 137 */ 138 @Deprecated 139 public static final int SENSOR_MAGNETIC_FIELD = 1 << 3; 140 141 /** 142 * A constant describing an ambient light sensor See 143 * {@link android.hardware.SensorListener SensorListener} for more details. 144 * 145 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 146 */ 147 @Deprecated 148 public static final int SENSOR_LIGHT = 1 << 4; 149 150 /** 151 * A constant describing a proximity sensor See 152 * {@link android.hardware.SensorListener SensorListener} for more details. 153 * 154 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 155 */ 156 @Deprecated 157 public static final int SENSOR_PROXIMITY = 1 << 5; 158 159 /** 160 * A constant describing a Tricorder See 161 * {@link android.hardware.SensorListener SensorListener} for more details. 162 * 163 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 164 */ 165 @Deprecated 166 public static final int SENSOR_TRICORDER = 1 << 6; 167 168 /** 169 * A constant describing an orientation sensor. See 170 * {@link android.hardware.SensorListener SensorListener} for more details. 171 * 172 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 173 */ 174 @Deprecated 175 public static final int SENSOR_ORIENTATION_RAW = 1 << 7; 176 177 /** 178 * A constant that includes all sensors 179 * 180 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 181 */ 182 @Deprecated 183 public static final int SENSOR_ALL = 0x7F; 184 185 /** 186 * Smallest sensor ID 187 * 188 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 189 */ 190 @Deprecated 191 public static final int SENSOR_MIN = SENSOR_ORIENTATION; 192 193 /** 194 * Largest sensor ID 195 * 196 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 197 */ 198 @Deprecated 199 public static final int SENSOR_MAX = ((SENSOR_ALL + 1) >> 1); 200 201 202 /** 203 * Index of the X value in the array returned by 204 * {@link android.hardware.SensorListener#onSensorChanged} 205 * 206 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 207 */ 208 @Deprecated 209 public static final int DATA_X = 0; 210 211 /** 212 * Index of the Y value in the array returned by 213 * {@link android.hardware.SensorListener#onSensorChanged} 214 * 215 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 216 */ 217 @Deprecated 218 public static final int DATA_Y = 1; 219 220 /** 221 * Index of the Z value in the array returned by 222 * {@link android.hardware.SensorListener#onSensorChanged} 223 * 224 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 225 */ 226 @Deprecated 227 public static final int DATA_Z = 2; 228 229 /** 230 * Offset to the untransformed values in the array returned by 231 * {@link android.hardware.SensorListener#onSensorChanged} 232 * 233 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 234 */ 235 @Deprecated 236 public static final int RAW_DATA_INDEX = 3; 237 238 /** 239 * Index of the untransformed X value in the array returned by 240 * {@link android.hardware.SensorListener#onSensorChanged} 241 * 242 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 243 */ 244 @Deprecated 245 public static final int RAW_DATA_X = 3; 246 247 /** 248 * Index of the untransformed Y value in the array returned by 249 * {@link android.hardware.SensorListener#onSensorChanged} 250 * 251 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 252 */ 253 @Deprecated 254 public static final int RAW_DATA_Y = 4; 255 256 /** 257 * Index of the untransformed Z value in the array returned by 258 * {@link android.hardware.SensorListener#onSensorChanged} 259 * 260 * @deprecated use {@link android.hardware.Sensor Sensor} instead. 261 */ 262 @Deprecated 263 public static final int RAW_DATA_Z = 5; 264 265 /** Standard gravity (g) on Earth. This value is equivalent to 1G */ 266 public static final float STANDARD_GRAVITY = 9.80665f; 267 268 /** Sun's gravity in SI units (m/s^2) */ 269 public static final float GRAVITY_SUN = 275.0f; 270 /** Mercury's gravity in SI units (m/s^2) */ 271 public static final float GRAVITY_MERCURY = 3.70f; 272 /** Venus' gravity in SI units (m/s^2) */ 273 public static final float GRAVITY_VENUS = 8.87f; 274 /** Earth's gravity in SI units (m/s^2) */ 275 public static final float GRAVITY_EARTH = 9.80665f; 276 /** The Moon's gravity in SI units (m/s^2) */ 277 public static final float GRAVITY_MOON = 1.6f; 278 /** Mars' gravity in SI units (m/s^2) */ 279 public static final float GRAVITY_MARS = 3.71f; 280 /** Jupiter's gravity in SI units (m/s^2) */ 281 public static final float GRAVITY_JUPITER = 23.12f; 282 /** Saturn's gravity in SI units (m/s^2) */ 283 public static final float GRAVITY_SATURN = 8.96f; 284 /** Uranus' gravity in SI units (m/s^2) */ 285 public static final float GRAVITY_URANUS = 8.69f; 286 /** Neptune's gravity in SI units (m/s^2) */ 287 public static final float GRAVITY_NEPTUNE = 11.0f; 288 /** Pluto's gravity in SI units (m/s^2) */ 289 public static final float GRAVITY_PLUTO = 0.6f; 290 /** Gravity (estimate) on the first Death Star in Empire units (m/s^2) */ 291 public static final float GRAVITY_DEATH_STAR_I = 0.000000353036145f; 292 /** Gravity on the island */ 293 public static final float GRAVITY_THE_ISLAND = 4.815162342f; 294 295 296 /** Maximum magnetic field on Earth's surface */ 297 public static final float MAGNETIC_FIELD_EARTH_MAX = 60.0f; 298 /** Minimum magnetic field on Earth's surface */ 299 public static final float MAGNETIC_FIELD_EARTH_MIN = 30.0f; 300 301 302 /** Standard atmosphere, or average sea-level pressure in hPa (millibar) */ 303 public static final float PRESSURE_STANDARD_ATMOSPHERE = 1013.25f; 304 305 306 /** Maximum luminance of sunlight in lux */ 307 public static final float LIGHT_SUNLIGHT_MAX = 120000.0f; 308 /** luminance of sunlight in lux */ 309 public static final float LIGHT_SUNLIGHT = 110000.0f; 310 /** luminance in shade in lux */ 311 public static final float LIGHT_SHADE = 20000.0f; 312 /** luminance under an overcast sky in lux */ 313 public static final float LIGHT_OVERCAST = 10000.0f; 314 /** luminance at sunrise in lux */ 315 public static final float LIGHT_SUNRISE = 400.0f; 316 /** luminance under a cloudy sky in lux */ 317 public static final float LIGHT_CLOUDY = 100.0f; 318 /** luminance at night with full moon in lux */ 319 public static final float LIGHT_FULLMOON = 0.25f; 320 /** luminance at night with no moon in lux*/ 321 public static final float LIGHT_NO_MOON = 0.001f; 322 323 324 /** get sensor data as fast as possible */ 325 public static final int SENSOR_DELAY_FASTEST = 0; 326 /** rate suitable for games */ 327 public static final int SENSOR_DELAY_GAME = 1; 328 /** rate suitable for the user interface */ 329 public static final int SENSOR_DELAY_UI = 2; 330 /** rate (default) suitable for screen orientation changes */ 331 public static final int SENSOR_DELAY_NORMAL = 3; 332 333 334 /** 335 * The values returned by this sensor cannot be trusted because the sensor 336 * had no contact with what it was measuring (for example, the heart rate 337 * monitor is not in contact with the user). 338 */ 339 public static final int SENSOR_STATUS_NO_CONTACT = -1; 340 341 /** 342 * The values returned by this sensor cannot be trusted, calibration is 343 * needed or the environment doesn't allow readings 344 */ 345 public static final int SENSOR_STATUS_UNRELIABLE = 0; 346 347 /** 348 * This sensor is reporting data with low accuracy, calibration with the 349 * environment is needed 350 */ 351 public static final int SENSOR_STATUS_ACCURACY_LOW = 1; 352 353 /** 354 * This sensor is reporting data with an average level of accuracy, 355 * calibration with the environment may improve the readings 356 */ 357 public static final int SENSOR_STATUS_ACCURACY_MEDIUM = 2; 358 359 /** This sensor is reporting data with maximum accuracy */ 360 public static final int SENSOR_STATUS_ACCURACY_HIGH = 3; 361 362 /** see {@link #remapCoordinateSystem} */ 363 public static final int AXIS_X = 1; 364 /** see {@link #remapCoordinateSystem} */ 365 public static final int AXIS_Y = 2; 366 /** see {@link #remapCoordinateSystem} */ 367 public static final int AXIS_Z = 3; 368 /** see {@link #remapCoordinateSystem} */ 369 public static final int AXIS_MINUS_X = AXIS_X | 0x80; 370 /** see {@link #remapCoordinateSystem} */ 371 public static final int AXIS_MINUS_Y = AXIS_Y | 0x80; 372 /** see {@link #remapCoordinateSystem} */ 373 public static final int AXIS_MINUS_Z = AXIS_Z | 0x80; 374 375 376 /** 377 * {@hide} 378 */ 379 @UnsupportedAppUsage SensorManager()380 public SensorManager() { 381 } 382 383 /** 384 * Gets the full list of sensors that are available. 385 * @hide 386 */ getFullSensorList()387 protected abstract List<Sensor> getFullSensorList(); 388 389 /** 390 * Gets the full list of dynamic sensors that are available. 391 * @hide 392 */ getFullDynamicSensorList()393 protected abstract List<Sensor> getFullDynamicSensorList(); 394 395 /** 396 * @return available sensors. 397 * @deprecated This method is deprecated, use 398 * {@link SensorManager#getSensorList(int)} instead 399 */ 400 @Deprecated getSensors()401 public int getSensors() { 402 return getLegacySensorManager().getSensors(); 403 } 404 405 /** 406 * Use this method to get the list of available sensors of a certain type. 407 * Make multiple calls to get sensors of different types or use 408 * {@link android.hardware.Sensor#TYPE_ALL Sensor.TYPE_ALL} to get all the 409 * sensors. Note that the {@link android.hardware.Sensor#getName()} is 410 * expected to yield a value that is unique across any sensors that return 411 * the same value for {@link android.hardware.Sensor#getType()}. 412 * 413 * <p class="note"> 414 * NOTE: Both wake-up and non wake-up sensors matching the given type are 415 * returned. Check {@link Sensor#isWakeUpSensor()} to know the wake-up properties 416 * of the returned {@link Sensor}. 417 * </p> 418 * 419 * @param type 420 * of sensors requested 421 * 422 * @return a list of sensors matching the asked type. 423 * 424 * @see #getDefaultSensor(int) 425 * @see Sensor 426 */ getSensorList(int type)427 public List<Sensor> getSensorList(int type) { 428 // cache the returned lists the first time 429 List<Sensor> list; 430 final List<Sensor> fullList = getFullSensorList(); 431 synchronized (mSensorListByType) { 432 list = mSensorListByType.get(type); 433 if (list == null) { 434 if (type == Sensor.TYPE_ALL) { 435 list = fullList; 436 } else { 437 list = new ArrayList<Sensor>(); 438 for (Sensor i : fullList) { 439 if (i.getType() == type) { 440 list.add(i); 441 } 442 } 443 } 444 list = Collections.unmodifiableList(list); 445 mSensorListByType.append(type, list); 446 } 447 } 448 return list; 449 } 450 451 /** 452 * Use this method to get a list of available dynamic sensors of a certain type. 453 * Make multiple calls to get sensors of different types or use 454 * {@link android.hardware.Sensor#TYPE_ALL Sensor.TYPE_ALL} to get all dynamic sensors. 455 * 456 * <p class="note"> 457 * NOTE: Both wake-up and non wake-up sensors matching the given type are 458 * returned. Check {@link Sensor#isWakeUpSensor()} to know the wake-up properties 459 * of the returned {@link Sensor}. 460 * </p> 461 * 462 * @param type of sensors requested 463 * 464 * @return a list of dynamic sensors matching the requested type. 465 * 466 * @see Sensor 467 */ getDynamicSensorList(int type)468 public List<Sensor> getDynamicSensorList(int type) { 469 // cache the returned lists the first time 470 final List<Sensor> fullList = getFullDynamicSensorList(); 471 if (type == Sensor.TYPE_ALL) { 472 return Collections.unmodifiableList(fullList); 473 } else { 474 List<Sensor> list = new ArrayList(); 475 for (Sensor i : fullList) { 476 if (i.getType() == type) { 477 list.add(i); 478 } 479 } 480 return Collections.unmodifiableList(list); 481 } 482 } 483 484 /** 485 * Use this method to get the default sensor for a given type. Note that the 486 * returned sensor could be a composite sensor, and its data could be 487 * averaged or filtered. If you need to access the raw sensors use 488 * {@link SensorManager#getSensorList(int) getSensorList}. 489 * 490 * @param type 491 * of sensors requested 492 * 493 * @return the default sensor matching the requested type if one exists and the application 494 * has the necessary permissions, or null otherwise. 495 * 496 * @see #getSensorList(int) 497 * @see Sensor 498 */ getDefaultSensor(int type)499 public Sensor getDefaultSensor(int type) { 500 // TODO: need to be smarter, for now, just return the 1st sensor 501 List<Sensor> l = getSensorList(type); 502 boolean wakeUpSensor = false; 503 // For the following sensor types, return a wake-up sensor. These types are by default 504 // defined as wake-up sensors. For the rest of the SDK defined sensor types return a 505 // non_wake-up version. 506 if (type == Sensor.TYPE_PROXIMITY || type == Sensor.TYPE_SIGNIFICANT_MOTION 507 || type == Sensor.TYPE_TILT_DETECTOR || type == Sensor.TYPE_WAKE_GESTURE 508 || type == Sensor.TYPE_GLANCE_GESTURE || type == Sensor.TYPE_PICK_UP_GESTURE 509 || type == Sensor.TYPE_LOW_LATENCY_OFFBODY_DETECT 510 || type == Sensor.TYPE_WRIST_TILT_GESTURE 511 || type == Sensor.TYPE_DYNAMIC_SENSOR_META || type == Sensor.TYPE_HINGE_ANGLE) { 512 wakeUpSensor = true; 513 } 514 515 for (Sensor sensor : l) { 516 if (sensor.isWakeUpSensor() == wakeUpSensor) return sensor; 517 } 518 return null; 519 } 520 521 /** 522 * Return a Sensor with the given type and wakeUp properties. If multiple sensors of this 523 * type exist, any one of them may be returned. 524 * <p> 525 * For example, 526 * <ul> 527 * <li>getDefaultSensor({@link Sensor#TYPE_ACCELEROMETER}, true) returns a wake-up 528 * accelerometer sensor if it exists. </li> 529 * <li>getDefaultSensor({@link Sensor#TYPE_PROXIMITY}, false) returns a non wake-up 530 * proximity sensor if it exists. </li> 531 * <li>getDefaultSensor({@link Sensor#TYPE_PROXIMITY}, true) returns a wake-up proximity 532 * sensor which is the same as the Sensor returned by {@link #getDefaultSensor(int)}. </li> 533 * </ul> 534 * </p> 535 * <p class="note"> 536 * Note: Sensors like {@link Sensor#TYPE_PROXIMITY} and {@link Sensor#TYPE_SIGNIFICANT_MOTION} 537 * are declared as wake-up sensors by default. 538 * </p> 539 * @param type 540 * type of sensor requested 541 * @param wakeUp 542 * flag to indicate whether the Sensor is a wake-up or non wake-up sensor. 543 * @return the default sensor matching the requested type and wakeUp properties if one exists 544 * and the application has the necessary permissions, or null otherwise. 545 * @see Sensor#isWakeUpSensor() 546 */ getDefaultSensor(int type, boolean wakeUp)547 public Sensor getDefaultSensor(int type, boolean wakeUp) { 548 List<Sensor> l = getSensorList(type); 549 for (Sensor sensor : l) { 550 if (sensor.isWakeUpSensor() == wakeUp) { 551 return sensor; 552 } 553 } 554 return null; 555 } 556 557 /** 558 * Registers a listener for given sensors. 559 * 560 * @deprecated This method is deprecated, use 561 * {@link SensorManager#registerListener(SensorEventListener, Sensor, int)} 562 * instead. 563 * 564 * @param listener 565 * sensor listener object 566 * 567 * @param sensors 568 * a bit masks of the sensors to register to 569 * 570 * @return <code>true</code> if the sensor is supported and successfully 571 * enabled 572 */ 573 @Deprecated registerListener(SensorListener listener, int sensors)574 public boolean registerListener(SensorListener listener, int sensors) { 575 return registerListener(listener, sensors, SENSOR_DELAY_NORMAL); 576 } 577 578 /** 579 * Registers a SensorListener for given sensors. 580 * 581 * @deprecated This method is deprecated, use 582 * {@link SensorManager#registerListener(SensorEventListener, Sensor, int)} 583 * instead. 584 * 585 * @param listener 586 * sensor listener object 587 * 588 * @param sensors 589 * a bit masks of the sensors to register to 590 * 591 * @param rate 592 * rate of events. This is only a hint to the system. events may be 593 * received faster or slower than the specified rate. Usually events 594 * are received faster. The value must be one of 595 * {@link #SENSOR_DELAY_NORMAL}, {@link #SENSOR_DELAY_UI}, 596 * {@link #SENSOR_DELAY_GAME}, or {@link #SENSOR_DELAY_FASTEST}. 597 * 598 * @return <code>true</code> if the sensor is supported and successfully 599 * enabled 600 */ 601 @Deprecated registerListener(SensorListener listener, int sensors, int rate)602 public boolean registerListener(SensorListener listener, int sensors, int rate) { 603 return getLegacySensorManager().registerListener(listener, sensors, rate); 604 } 605 606 /** 607 * Unregisters a listener for all sensors. 608 * 609 * @deprecated This method is deprecated, use 610 * {@link SensorManager#unregisterListener(SensorEventListener)} 611 * instead. 612 * 613 * @param listener 614 * a SensorListener object 615 */ 616 @Deprecated unregisterListener(SensorListener listener)617 public void unregisterListener(SensorListener listener) { 618 unregisterListener(listener, SENSOR_ALL | SENSOR_ORIENTATION_RAW); 619 } 620 621 /** 622 * Unregisters a listener for the sensors with which it is registered. 623 * 624 * @deprecated This method is deprecated, use 625 * {@link SensorManager#unregisterListener(SensorEventListener, Sensor)} 626 * instead. 627 * 628 * @param listener 629 * a SensorListener object 630 * 631 * @param sensors 632 * a bit masks of the sensors to unregister from 633 */ 634 @Deprecated unregisterListener(SensorListener listener, int sensors)635 public void unregisterListener(SensorListener listener, int sensors) { 636 getLegacySensorManager().unregisterListener(listener, sensors); 637 } 638 639 /** 640 * Unregisters a listener for the sensors with which it is registered. 641 * 642 * <p class="note"> 643 * Note: Don't use this method with a one shot trigger sensor such as 644 * {@link Sensor#TYPE_SIGNIFICANT_MOTION}. 645 * Use {@link #cancelTriggerSensor(TriggerEventListener, Sensor)} instead. 646 * </p> 647 * 648 * @param listener 649 * a SensorEventListener object 650 * 651 * @param sensor 652 * the sensor to unregister from 653 * 654 * @see #unregisterListener(SensorEventListener) 655 * @see #registerListener(SensorEventListener, Sensor, int) 656 */ unregisterListener(SensorEventListener listener, Sensor sensor)657 public void unregisterListener(SensorEventListener listener, Sensor sensor) { 658 if (listener == null || sensor == null) { 659 return; 660 } 661 662 unregisterListenerImpl(listener, sensor); 663 } 664 665 /** 666 * Unregisters a listener for all sensors. 667 * 668 * @param listener 669 * a SensorListener object 670 * 671 * @see #unregisterListener(SensorEventListener, Sensor) 672 * @see #registerListener(SensorEventListener, Sensor, int) 673 * 674 */ unregisterListener(SensorEventListener listener)675 public void unregisterListener(SensorEventListener listener) { 676 if (listener == null) { 677 return; 678 } 679 680 unregisterListenerImpl(listener, null); 681 } 682 683 /** @hide */ unregisterListenerImpl(SensorEventListener listener, Sensor sensor)684 protected abstract void unregisterListenerImpl(SensorEventListener listener, Sensor sensor); 685 686 /** 687 * Registers a {@link android.hardware.SensorEventListener SensorEventListener} for the given 688 * sensor at the given sampling frequency. 689 * <p> 690 * The events will be delivered to the provided {@code SensorEventListener} as soon as they are 691 * available. To reduce the power consumption, applications can use 692 * {@link #registerListener(SensorEventListener, Sensor, int, int)} instead and specify a 693 * positive non-zero maximum reporting latency. 694 * </p> 695 * <p> 696 * In the case of non-wake-up sensors, the events are only delivered while the Application 697 * Processor (AP) is not in suspend mode. See {@link Sensor#isWakeUpSensor()} for more details. 698 * To ensure delivery of events from non-wake-up sensors even when the screen is OFF, the 699 * application registering to the sensor must hold a partial wake-lock to keep the AP awake, 700 * otherwise some events might be lost while the AP is asleep. Note that although events might 701 * be lost while the AP is asleep, the sensor will still consume power if it is not explicitly 702 * deactivated by the application. Applications must unregister their {@code 703 * SensorEventListener}s in their activity's {@code onPause()} method to avoid consuming power 704 * while the device is inactive. See {@link #registerListener(SensorEventListener, Sensor, int, 705 * int)} for more details on hardware FIFO (queueing) capabilities and when some sensor events 706 * might be lost. 707 * </p> 708 * <p> 709 * In the case of wake-up sensors, each event generated by the sensor will cause the AP to 710 * wake-up, ensuring that each event can be delivered. Because of this, registering to a wake-up 711 * sensor has very significant power implications. Call {@link Sensor#isWakeUpSensor()} to check 712 * whether a sensor is a wake-up sensor. See 713 * {@link #registerListener(SensorEventListener, Sensor, int, int)} for information on how to 714 * reduce the power impact of registering to wake-up sensors. 715 * </p> 716 * <p class="note"> 717 * Note: Don't use this method with one-shot trigger sensors such as 718 * {@link Sensor#TYPE_SIGNIFICANT_MOTION}. Use 719 * {@link #requestTriggerSensor(TriggerEventListener, Sensor)} instead. Use 720 * {@link Sensor#getReportingMode()} to obtain the reporting mode of a given sensor. 721 * </p> 722 * 723 * @param listener A {@link android.hardware.SensorEventListener SensorEventListener} object. 724 * @param sensor The {@link android.hardware.Sensor Sensor} to register to. 725 * @param samplingPeriodUs The rate {@link android.hardware.SensorEvent sensor events} are 726 * delivered at. This is only a hint to the system. Events may be received faster or 727 * slower than the specified rate. Usually events are received faster. The value must 728 * be one of {@link #SENSOR_DELAY_NORMAL}, {@link #SENSOR_DELAY_UI}, 729 * {@link #SENSOR_DELAY_GAME}, or {@link #SENSOR_DELAY_FASTEST} or, the desired delay 730 * between events in microseconds. Specifying the delay in microseconds only works 731 * from Android 2.3 (API level 9) onwards. For earlier releases, you must use one of 732 * the {@code SENSOR_DELAY_*} constants. 733 * @return <code>true</code> if the sensor is supported and successfully enabled. 734 * @see #registerListener(SensorEventListener, Sensor, int, Handler) 735 * @see #unregisterListener(SensorEventListener) 736 * @see #unregisterListener(SensorEventListener, Sensor) 737 */ registerListener(SensorEventListener listener, Sensor sensor, int samplingPeriodUs)738 public boolean registerListener(SensorEventListener listener, Sensor sensor, 739 int samplingPeriodUs) { 740 return registerListener(listener, sensor, samplingPeriodUs, null); 741 } 742 743 /** 744 * Registers a {@link android.hardware.SensorEventListener SensorEventListener} for the given 745 * sensor at the given sampling frequency and the given maximum reporting latency. 746 * <p> 747 * This function is similar to {@link #registerListener(SensorEventListener, Sensor, int)} but 748 * it allows events to stay temporarily in the hardware FIFO (queue) before being delivered. The 749 * events can be stored in the hardware FIFO up to {@code maxReportLatencyUs} microseconds. Once 750 * one of the events in the FIFO needs to be reported, all of the events in the FIFO are 751 * reported sequentially. This means that some events will be reported before the maximum 752 * reporting latency has elapsed. 753 * </p><p> 754 * When {@code maxReportLatencyUs} is 0, the call is equivalent to a call to 755 * {@link #registerListener(SensorEventListener, Sensor, int)}, as it requires the events to be 756 * delivered as soon as possible. 757 * </p><p> 758 * When {@code sensor.maxFifoEventCount()} is 0, the sensor does not use a FIFO, so the call 759 * will also be equivalent to {@link #registerListener(SensorEventListener, Sensor, int)}. 760 * </p><p> 761 * Setting {@code maxReportLatencyUs} to a positive value allows to reduce the number of 762 * interrupts the AP (Application Processor) receives, hence reducing power consumption, as the 763 * AP can switch to a lower power state while the sensor is capturing the data. This is 764 * especially important when registering to wake-up sensors, for which each interrupt causes the 765 * AP to wake up if it was in suspend mode. See {@link Sensor#isWakeUpSensor()} for more 766 * information on wake-up sensors. 767 * </p> 768 * <p class="note"> 769 * </p> 770 * Note: Don't use this method with one-shot trigger sensors such as 771 * {@link Sensor#TYPE_SIGNIFICANT_MOTION}. Use 772 * {@link #requestTriggerSensor(TriggerEventListener, Sensor)} instead. </p> 773 * 774 * @param listener A {@link android.hardware.SensorEventListener SensorEventListener} object 775 * that will receive the sensor events. If the application is interested in receiving 776 * flush complete notifications, it should register with 777 * {@link android.hardware.SensorEventListener SensorEventListener2} instead. 778 * @param sensor The {@link android.hardware.Sensor Sensor} to register to. 779 * @param samplingPeriodUs The desired delay between two consecutive events in microseconds. 780 * This is only a hint to the system. Events may be received faster or slower than 781 * the specified rate. Usually events are received faster. Can be one of 782 * {@link #SENSOR_DELAY_NORMAL}, {@link #SENSOR_DELAY_UI}, 783 * {@link #SENSOR_DELAY_GAME}, {@link #SENSOR_DELAY_FASTEST} or the delay in 784 * microseconds. 785 * @param maxReportLatencyUs Maximum time in microseconds that events can be delayed before 786 * being reported to the application. A large value allows reducing the power 787 * consumption associated with the sensor. If maxReportLatencyUs is set to zero, 788 * events are delivered as soon as they are available, which is equivalent to calling 789 * {@link #registerListener(SensorEventListener, Sensor, int)}. 790 * @return <code>true</code> if the sensor is supported and successfully enabled. 791 * @see #registerListener(SensorEventListener, Sensor, int) 792 * @see #unregisterListener(SensorEventListener) 793 * @see #flush(SensorEventListener) 794 */ registerListener(SensorEventListener listener, Sensor sensor, int samplingPeriodUs, int maxReportLatencyUs)795 public boolean registerListener(SensorEventListener listener, Sensor sensor, 796 int samplingPeriodUs, int maxReportLatencyUs) { 797 int delay = getDelay(samplingPeriodUs); 798 return registerListenerImpl(listener, sensor, delay, null, maxReportLatencyUs, 0); 799 } 800 801 /** 802 * Registers a {@link android.hardware.SensorEventListener SensorEventListener} for the given 803 * sensor. Events are delivered in continuous mode as soon as they are available. To reduce the 804 * power consumption, applications can use 805 * {@link #registerListener(SensorEventListener, Sensor, int, int)} instead and specify a 806 * positive non-zero maximum reporting latency. 807 * <p class="note"> 808 * </p> 809 * Note: Don't use this method with a one shot trigger sensor such as 810 * {@link Sensor#TYPE_SIGNIFICANT_MOTION}. Use 811 * {@link #requestTriggerSensor(TriggerEventListener, Sensor)} instead. </p> 812 * 813 * @param listener A {@link android.hardware.SensorEventListener SensorEventListener} object. 814 * @param sensor The {@link android.hardware.Sensor Sensor} to register to. 815 * @param samplingPeriodUs The rate {@link android.hardware.SensorEvent sensor events} are 816 * delivered at. This is only a hint to the system. Events may be received faster or 817 * slower than the specified rate. Usually events are received faster. The value must 818 * be one of {@link #SENSOR_DELAY_NORMAL}, {@link #SENSOR_DELAY_UI}, 819 * {@link #SENSOR_DELAY_GAME}, or {@link #SENSOR_DELAY_FASTEST} or, the desired 820 * delay between events in microseconds. Specifying the delay in microseconds only 821 * works from Android 2.3 (API level 9) onwards. For earlier releases, you must use 822 * one of the {@code SENSOR_DELAY_*} constants. 823 * @param handler The {@link android.os.Handler Handler} the {@link android.hardware.SensorEvent 824 * sensor events} will be delivered to. 825 * @return <code>true</code> if the sensor is supported and successfully enabled. 826 * @see #registerListener(SensorEventListener, Sensor, int) 827 * @see #unregisterListener(SensorEventListener) 828 * @see #unregisterListener(SensorEventListener, Sensor) 829 */ registerListener(SensorEventListener listener, Sensor sensor, int samplingPeriodUs, Handler handler)830 public boolean registerListener(SensorEventListener listener, Sensor sensor, 831 int samplingPeriodUs, Handler handler) { 832 int delay = getDelay(samplingPeriodUs); 833 return registerListenerImpl(listener, sensor, delay, handler, 0, 0); 834 } 835 836 /** 837 * Registers a {@link android.hardware.SensorEventListener SensorEventListener} for the given 838 * sensor at the given sampling frequency and the given maximum reporting latency. 839 * 840 * @param listener A {@link android.hardware.SensorEventListener SensorEventListener} object 841 * that will receive the sensor events. If the application is interested in receiving 842 * flush complete notifications, it should register with 843 * {@link android.hardware.SensorEventListener SensorEventListener2} instead. 844 * @param sensor The {@link android.hardware.Sensor Sensor} to register to. 845 * @param samplingPeriodUs The desired delay between two consecutive events in microseconds. 846 * This is only a hint to the system. Events may be received faster or slower than 847 * the specified rate. Usually events are received faster. Can be one of 848 * {@link #SENSOR_DELAY_NORMAL}, {@link #SENSOR_DELAY_UI}, 849 * {@link #SENSOR_DELAY_GAME}, {@link #SENSOR_DELAY_FASTEST} or the delay in 850 * microseconds. 851 * @param maxReportLatencyUs Maximum time in microseconds that events can be delayed before 852 * being reported to the application. A large value allows reducing the power 853 * consumption associated with the sensor. If maxReportLatencyUs is set to zero, 854 * events are delivered as soon as they are available, which is equivalent to calling 855 * {@link #registerListener(SensorEventListener, Sensor, int)}. 856 * @param handler The {@link android.os.Handler Handler} the {@link android.hardware.SensorEvent 857 * sensor events} will be delivered to. 858 * @return <code>true</code> if the sensor is supported and successfully enabled. 859 * @see #registerListener(SensorEventListener, Sensor, int, int) 860 */ registerListener(SensorEventListener listener, Sensor sensor, int samplingPeriodUs, int maxReportLatencyUs, Handler handler)861 public boolean registerListener(SensorEventListener listener, Sensor sensor, 862 int samplingPeriodUs, int maxReportLatencyUs, Handler handler) { 863 int delayUs = getDelay(samplingPeriodUs); 864 return registerListenerImpl(listener, sensor, delayUs, handler, maxReportLatencyUs, 0); 865 } 866 867 /** @hide */ registerListenerImpl(SensorEventListener listener, Sensor sensor, int delayUs, Handler handler, int maxReportLatencyUs, int reservedFlags)868 protected abstract boolean registerListenerImpl(SensorEventListener listener, Sensor sensor, 869 int delayUs, Handler handler, int maxReportLatencyUs, int reservedFlags); 870 871 872 /** 873 * Flushes the FIFO of all the sensors registered for this listener. If there are events 874 * in the FIFO of the sensor, they are returned as if the maxReportLantecy of the FIFO has 875 * expired. Events are returned in the usual way through the SensorEventListener. 876 * This call doesn't affect the maxReportLantecy for this sensor. This call is asynchronous and 877 * returns immediately. 878 * {@link android.hardware.SensorEventListener2#onFlushCompleted onFlushCompleted} is called 879 * after all the events in the batch at the time of calling this method have been delivered 880 * successfully. If the hardware doesn't support flush, it still returns true and a trivial 881 * flush complete event is sent after the current event for all the clients registered for this 882 * sensor. 883 * 884 * @param listener A {@link android.hardware.SensorEventListener SensorEventListener} object 885 * which was previously used in a registerListener call. 886 * @return <code>true</code> if the flush is initiated successfully on all the sensors 887 * registered for this listener, false if no sensor is previously registered for this 888 * listener or flush on one of the sensors fails. 889 * @see #registerListener(SensorEventListener, Sensor, int, int) 890 * @throws IllegalArgumentException when listener is null. 891 */ flush(SensorEventListener listener)892 public boolean flush(SensorEventListener listener) { 893 return flushImpl(listener); 894 } 895 896 /** @hide */ flushImpl(SensorEventListener listener)897 protected abstract boolean flushImpl(SensorEventListener listener); 898 899 900 /** 901 * Create a sensor direct channel backed by shared memory wrapped in MemoryFile object. 902 * 903 * The resulting channel can be used for delivering sensor events to native code, other 904 * processes, GPU/DSP or other co-processors without CPU intervention. This is the recommanded 905 * for high performance sensor applications that use high sensor rates (e.g. greater than 200Hz) 906 * and cares about sensor event latency. 907 * 908 * Use the returned {@link android.hardware.SensorDirectChannel} object to configure direct 909 * report of sensor events. After use, call {@link android.hardware.SensorDirectChannel#close()} 910 * to free up resource in sensor system associated with the direct channel. 911 * 912 * @param mem A {@link android.os.MemoryFile} shared memory object. 913 * @return A {@link android.hardware.SensorDirectChannel} object. 914 * @throws NullPointerException when mem is null. 915 * @throws UncheckedIOException if not able to create channel. 916 * @see SensorDirectChannel#close() 917 */ createDirectChannel(MemoryFile mem)918 public SensorDirectChannel createDirectChannel(MemoryFile mem) { 919 return createDirectChannelImpl(mem, null); 920 } 921 922 /** 923 * Create a sensor direct channel backed by shared memory wrapped in HardwareBuffer object. 924 * 925 * The resulting channel can be used for delivering sensor events to native code, other 926 * processes, GPU/DSP or other co-processors without CPU intervention. This is the recommanded 927 * for high performance sensor applications that use high sensor rates (e.g. greater than 200Hz) 928 * and cares about sensor event latency. 929 * 930 * Use the returned {@link android.hardware.SensorDirectChannel} object to configure direct 931 * report of sensor events. After use, call {@link android.hardware.SensorDirectChannel#close()} 932 * to free up resource in sensor system associated with the direct channel. 933 * 934 * @param mem A {@link android.hardware.HardwareBuffer} shared memory object. 935 * @return A {@link android.hardware.SensorDirectChannel} object. 936 * @throws NullPointerException when mem is null. 937 * @throws UncheckedIOException if not able to create channel. 938 * @see SensorDirectChannel#close() 939 */ createDirectChannel(HardwareBuffer mem)940 public SensorDirectChannel createDirectChannel(HardwareBuffer mem) { 941 return createDirectChannelImpl(null, mem); 942 } 943 944 /** @hide */ createDirectChannelImpl( MemoryFile memoryFile, HardwareBuffer hardwareBuffer)945 protected abstract SensorDirectChannel createDirectChannelImpl( 946 MemoryFile memoryFile, HardwareBuffer hardwareBuffer); 947 948 /** @hide */ destroyDirectChannel(SensorDirectChannel channel)949 void destroyDirectChannel(SensorDirectChannel channel) { 950 destroyDirectChannelImpl(channel); 951 } 952 953 /** @hide */ destroyDirectChannelImpl(SensorDirectChannel channel)954 protected abstract void destroyDirectChannelImpl(SensorDirectChannel channel); 955 956 /** @hide */ configureDirectChannelImpl( SensorDirectChannel channel, Sensor s, int rate)957 protected abstract int configureDirectChannelImpl( 958 SensorDirectChannel channel, Sensor s, int rate); 959 960 /** 961 * Used for receiving notifications from the SensorManager when dynamic sensors are connected or 962 * disconnected. 963 */ 964 public abstract static class DynamicSensorCallback { 965 /** 966 * Called when there is a dynamic sensor being connected to the system. 967 * 968 * @param sensor the newly connected sensor. See {@link android.hardware.Sensor Sensor}. 969 */ onDynamicSensorConnected(Sensor sensor)970 public void onDynamicSensorConnected(Sensor sensor) {} 971 972 /** 973 * Called when there is a dynamic sensor being disconnected from the system. 974 * 975 * @param sensor the disconnected sensor. See {@link android.hardware.Sensor Sensor}. 976 */ onDynamicSensorDisconnected(Sensor sensor)977 public void onDynamicSensorDisconnected(Sensor sensor) {} 978 } 979 980 981 /** 982 * Add a {@link android.hardware.SensorManager.DynamicSensorCallback 983 * DynamicSensorCallback} to receive dynamic sensor connection callbacks. Repeat 984 * registration with the already registered callback object will have no additional effect. 985 * 986 * @param callback An object that implements the 987 * {@link android.hardware.SensorManager.DynamicSensorCallback 988 * DynamicSensorCallback} 989 * interface for receiving callbacks. 990 * @see #registerDynamicSensorCallback(DynamicSensorCallback, Handler) 991 * 992 * @throws IllegalArgumentException when callback is null. 993 */ registerDynamicSensorCallback(DynamicSensorCallback callback)994 public void registerDynamicSensorCallback(DynamicSensorCallback callback) { 995 registerDynamicSensorCallback(callback, null); 996 } 997 998 /** 999 * Add a {@link android.hardware.SensorManager.DynamicSensorCallback 1000 * DynamicSensorCallback} to receive dynamic sensor connection callbacks. Repeat 1001 * registration with the already registered callback object will have no additional effect. 1002 * 1003 * @param callback An object that implements the 1004 * {@link android.hardware.SensorManager.DynamicSensorCallback 1005 * DynamicSensorCallback} interface for receiving callbacks. 1006 * @param handler The {@link android.os.Handler Handler} the {@link 1007 * android.hardware.SensorManager.DynamicSensorCallback 1008 * sensor connection events} will be delivered to. 1009 * 1010 * @throws IllegalArgumentException when callback is null. 1011 */ registerDynamicSensorCallback( DynamicSensorCallback callback, Handler handler)1012 public void registerDynamicSensorCallback( 1013 DynamicSensorCallback callback, Handler handler) { 1014 registerDynamicSensorCallbackImpl(callback, handler); 1015 } 1016 1017 /** 1018 * Remove a {@link android.hardware.SensorManager.DynamicSensorCallback 1019 * DynamicSensorCallback} to stop sending dynamic sensor connection events to that 1020 * callback. 1021 * 1022 * @param callback An object that implements the 1023 * {@link android.hardware.SensorManager.DynamicSensorCallback 1024 * DynamicSensorCallback} 1025 * interface for receiving callbacks. 1026 */ unregisterDynamicSensorCallback(DynamicSensorCallback callback)1027 public void unregisterDynamicSensorCallback(DynamicSensorCallback callback) { 1028 unregisterDynamicSensorCallbackImpl(callback); 1029 } 1030 1031 /** 1032 * Tell if dynamic sensor discovery feature is supported by system. 1033 * 1034 * @return <code>true</code> if dynamic sensor discovery is supported, <code>false</code> 1035 * otherwise. 1036 */ isDynamicSensorDiscoverySupported()1037 public boolean isDynamicSensorDiscoverySupported() { 1038 List<Sensor> sensors = getSensorList(Sensor.TYPE_DYNAMIC_SENSOR_META); 1039 return sensors.size() > 0; 1040 } 1041 1042 /** @hide */ registerDynamicSensorCallbackImpl( DynamicSensorCallback callback, Handler handler)1043 protected abstract void registerDynamicSensorCallbackImpl( 1044 DynamicSensorCallback callback, Handler handler); 1045 1046 /** @hide */ unregisterDynamicSensorCallbackImpl( DynamicSensorCallback callback)1047 protected abstract void unregisterDynamicSensorCallbackImpl( 1048 DynamicSensorCallback callback); 1049 1050 /** 1051 * <p> 1052 * Computes the inclination matrix <b>I</b> as well as the rotation matrix 1053 * <b>R</b> transforming a vector from the device coordinate system to the 1054 * world's coordinate system which is defined as a direct orthonormal basis, 1055 * where: 1056 * </p> 1057 * 1058 * <ul> 1059 * <li>X is defined as the vector product <b>Y.Z</b> (It is tangential to 1060 * the ground at the device's current location and roughly points East).</li> 1061 * <li>Y is tangential to the ground at the device's current location and 1062 * points towards the magnetic North Pole.</li> 1063 * <li>Z points towards the sky and is perpendicular to the ground.</li> 1064 * </ul> 1065 * 1066 * <p> 1067 * <center><img src="../../../images/axis_globe.png" 1068 * alt="World coordinate-system diagram." border="0" /></center> 1069 * </p> 1070 * 1071 * <p> 1072 * <hr> 1073 * <p> 1074 * By definition: 1075 * <p> 1076 * [0 0 g] = <b>R</b> * <b>gravity</b> (g = magnitude of gravity) 1077 * <p> 1078 * [0 m 0] = <b>I</b> * <b>R</b> * <b>geomagnetic</b> (m = magnitude of 1079 * geomagnetic field) 1080 * <p> 1081 * <b>R</b> is the identity matrix when the device is aligned with the 1082 * world's coordinate system, that is, when the device's X axis points 1083 * toward East, the Y axis points to the North Pole and the device is facing 1084 * the sky. 1085 * 1086 * <p> 1087 * <b>I</b> is a rotation matrix transforming the geomagnetic vector into 1088 * the same coordinate space as gravity (the world's coordinate space). 1089 * <b>I</b> is a simple rotation around the X axis. The inclination angle in 1090 * radians can be computed with {@link #getInclination}. 1091 * <hr> 1092 * 1093 * <p> 1094 * Each matrix is returned either as a 3x3 or 4x4 row-major matrix depending 1095 * on the length of the passed array: 1096 * <p> 1097 * <u>If the array length is 16:</u> 1098 * 1099 * <pre> 1100 * / M[ 0] M[ 1] M[ 2] M[ 3] \ 1101 * | M[ 4] M[ 5] M[ 6] M[ 7] | 1102 * | M[ 8] M[ 9] M[10] M[11] | 1103 * \ M[12] M[13] M[14] M[15] / 1104 *</pre> 1105 * 1106 * This matrix is ready to be used by OpenGL ES's 1107 * {@link javax.microedition.khronos.opengles.GL10#glLoadMatrixf(float[], int) 1108 * glLoadMatrixf(float[], int)}. 1109 * <p> 1110 * Note that because OpenGL matrices are column-major matrices you must 1111 * transpose the matrix before using it. However, since the matrix is a 1112 * rotation matrix, its transpose is also its inverse, conveniently, it is 1113 * often the inverse of the rotation that is needed for rendering; it can 1114 * therefore be used with OpenGL ES directly. 1115 * <p> 1116 * Also note that the returned matrices always have this form: 1117 * 1118 * <pre> 1119 * / M[ 0] M[ 1] M[ 2] 0 \ 1120 * | M[ 4] M[ 5] M[ 6] 0 | 1121 * | M[ 8] M[ 9] M[10] 0 | 1122 * \ 0 0 0 1 / 1123 *</pre> 1124 * 1125 * <p> 1126 * <u>If the array length is 9:</u> 1127 * 1128 * <pre> 1129 * / M[ 0] M[ 1] M[ 2] \ 1130 * | M[ 3] M[ 4] M[ 5] | 1131 * \ M[ 6] M[ 7] M[ 8] / 1132 *</pre> 1133 * 1134 * <hr> 1135 * <p> 1136 * The inverse of each matrix can be computed easily by taking its 1137 * transpose. 1138 * 1139 * <p> 1140 * The matrices returned by this function are meaningful only when the 1141 * device is not free-falling and it is not close to the magnetic north. If 1142 * the device is accelerating, or placed into a strong magnetic field, the 1143 * returned matrices may be inaccurate. 1144 * 1145 * @param R 1146 * is an array of 9 floats holding the rotation matrix <b>R</b> when 1147 * this function returns. R can be null. 1148 * <p> 1149 * 1150 * @param I 1151 * is an array of 9 floats holding the rotation matrix <b>I</b> when 1152 * this function returns. I can be null. 1153 * <p> 1154 * 1155 * @param gravity 1156 * is an array of 3 floats containing the gravity vector expressed in 1157 * the device's coordinate. You can simply use the 1158 * {@link android.hardware.SensorEvent#values values} returned by a 1159 * {@link android.hardware.SensorEvent SensorEvent} of a 1160 * {@link android.hardware.Sensor Sensor} of type 1161 * {@link android.hardware.Sensor#TYPE_ACCELEROMETER 1162 * TYPE_ACCELEROMETER}. 1163 * <p> 1164 * 1165 * @param geomagnetic 1166 * is an array of 3 floats containing the geomagnetic vector 1167 * expressed in the device's coordinate. You can simply use the 1168 * {@link android.hardware.SensorEvent#values values} returned by a 1169 * {@link android.hardware.SensorEvent SensorEvent} of a 1170 * {@link android.hardware.Sensor Sensor} of type 1171 * {@link android.hardware.Sensor#TYPE_MAGNETIC_FIELD 1172 * TYPE_MAGNETIC_FIELD}. 1173 * 1174 * @return <code>true</code> on success, <code>false</code> on failure (for 1175 * instance, if the device is in free fall). Free fall is defined as 1176 * condition when the magnitude of the gravity is less than 1/10 of 1177 * the nominal value. On failure the output matrices are not modified. 1178 * 1179 * @see #getInclination(float[]) 1180 * @see #getOrientation(float[], float[]) 1181 * @see #remapCoordinateSystem(float[], int, int, float[]) 1182 */ 1183 getRotationMatrix(float[] R, float[] I, float[] gravity, float[] geomagnetic)1184 public static boolean getRotationMatrix(float[] R, float[] I, 1185 float[] gravity, float[] geomagnetic) { 1186 // TODO: move this to native code for efficiency 1187 float Ax = gravity[0]; 1188 float Ay = gravity[1]; 1189 float Az = gravity[2]; 1190 1191 final float normsqA = (Ax * Ax + Ay * Ay + Az * Az); 1192 final float g = 9.81f; 1193 final float freeFallGravitySquared = 0.01f * g * g; 1194 if (normsqA < freeFallGravitySquared) { 1195 // gravity less than 10% of normal value 1196 return false; 1197 } 1198 1199 final float Ex = geomagnetic[0]; 1200 final float Ey = geomagnetic[1]; 1201 final float Ez = geomagnetic[2]; 1202 float Hx = Ey * Az - Ez * Ay; 1203 float Hy = Ez * Ax - Ex * Az; 1204 float Hz = Ex * Ay - Ey * Ax; 1205 final float normH = (float) Math.sqrt(Hx * Hx + Hy * Hy + Hz * Hz); 1206 1207 if (normH < 0.1f) { 1208 // device is close to free fall (or in space?), or close to 1209 // magnetic north pole. Typical values are > 100. 1210 return false; 1211 } 1212 final float invH = 1.0f / normH; 1213 Hx *= invH; 1214 Hy *= invH; 1215 Hz *= invH; 1216 final float invA = 1.0f / (float) Math.sqrt(Ax * Ax + Ay * Ay + Az * Az); 1217 Ax *= invA; 1218 Ay *= invA; 1219 Az *= invA; 1220 final float Mx = Ay * Hz - Az * Hy; 1221 final float My = Az * Hx - Ax * Hz; 1222 final float Mz = Ax * Hy - Ay * Hx; 1223 if (R != null) { 1224 if (R.length == 9) { 1225 R[0] = Hx; R[1] = Hy; R[2] = Hz; 1226 R[3] = Mx; R[4] = My; R[5] = Mz; 1227 R[6] = Ax; R[7] = Ay; R[8] = Az; 1228 } else if (R.length == 16) { 1229 R[0] = Hx; R[1] = Hy; R[2] = Hz; R[3] = 0; 1230 R[4] = Mx; R[5] = My; R[6] = Mz; R[7] = 0; 1231 R[8] = Ax; R[9] = Ay; R[10] = Az; R[11] = 0; 1232 R[12] = 0; R[13] = 0; R[14] = 0; R[15] = 1; 1233 } 1234 } 1235 if (I != null) { 1236 // compute the inclination matrix by projecting the geomagnetic 1237 // vector onto the Z (gravity) and X (horizontal component 1238 // of geomagnetic vector) axes. 1239 final float invE = 1.0f / (float) Math.sqrt(Ex * Ex + Ey * Ey + Ez * Ez); 1240 final float c = (Ex * Mx + Ey * My + Ez * Mz) * invE; 1241 final float s = (Ex * Ax + Ey * Ay + Ez * Az) * invE; 1242 if (I.length == 9) { 1243 I[0] = 1; I[1] = 0; I[2] = 0; 1244 I[3] = 0; I[4] = c; I[5] = s; 1245 I[6] = 0; I[7] = -s; I[8] = c; 1246 } else if (I.length == 16) { 1247 I[0] = 1; I[1] = 0; I[2] = 0; 1248 I[4] = 0; I[5] = c; I[6] = s; 1249 I[8] = 0; I[9] = -s; I[10] = c; 1250 I[3] = I[7] = I[11] = I[12] = I[13] = I[14] = 0; 1251 I[15] = 1; 1252 } 1253 } 1254 return true; 1255 } 1256 1257 /** 1258 * Computes the geomagnetic inclination angle in radians from the 1259 * inclination matrix <b>I</b> returned by {@link #getRotationMatrix}. 1260 * 1261 * @param I 1262 * inclination matrix see {@link #getRotationMatrix}. 1263 * 1264 * @return The geomagnetic inclination angle in radians. 1265 * 1266 * @see #getRotationMatrix(float[], float[], float[], float[]) 1267 * @see #getOrientation(float[], float[]) 1268 * @see GeomagneticField 1269 * 1270 */ getInclination(float[] I)1271 public static float getInclination(float[] I) { 1272 if (I.length == 9) { 1273 return (float) Math.atan2(I[5], I[4]); 1274 } else { 1275 return (float) Math.atan2(I[6], I[5]); 1276 } 1277 } 1278 1279 /** 1280 * <p> 1281 * Rotates the supplied rotation matrix so it is expressed in a different 1282 * coordinate system. This is typically used when an application needs to 1283 * compute the three orientation angles of the device (see 1284 * {@link #getOrientation}) in a different coordinate system. 1285 * </p> 1286 * 1287 * <p> 1288 * When the rotation matrix is used for drawing (for instance with OpenGL 1289 * ES), it usually <b>doesn't need</b> to be transformed by this function, 1290 * unless the screen is physically rotated, in which case you can use 1291 * {@link android.view.Display#getRotation() Display.getRotation()} to 1292 * retrieve the current rotation of the screen. Note that because the user 1293 * is generally free to rotate their screen, you often should consider the 1294 * rotation in deciding the parameters to use here. 1295 * </p> 1296 * 1297 * <p> 1298 * <u>Examples:</u> 1299 * <p> 1300 * 1301 * <ul> 1302 * <li>Using the camera (Y axis along the camera's axis) for an augmented 1303 * reality application where the rotation angles are needed:</li> 1304 * 1305 * <p> 1306 * <ul> 1307 * <code>remapCoordinateSystem(inR, AXIS_X, AXIS_Z, outR);</code> 1308 * </ul> 1309 * </p> 1310 * 1311 * <li>Using the device as a mechanical compass when rotation is 1312 * {@link android.view.Surface#ROTATION_90 Surface.ROTATION_90}:</li> 1313 * 1314 * <p> 1315 * <ul> 1316 * <code>remapCoordinateSystem(inR, AXIS_Y, AXIS_MINUS_X, outR);</code> 1317 * </ul> 1318 * </p> 1319 * 1320 * Beware of the above example. This call is needed only to account for a 1321 * rotation from its natural orientation when calculating the rotation 1322 * angles (see {@link #getOrientation}). If the rotation matrix is also used 1323 * for rendering, it may not need to be transformed, for instance if your 1324 * {@link android.app.Activity Activity} is running in landscape mode. 1325 * </ul> 1326 * 1327 * <p> 1328 * Since the resulting coordinate system is orthonormal, only two axes need 1329 * to be specified. 1330 * 1331 * @param inR 1332 * the rotation matrix to be transformed. Usually it is the matrix 1333 * returned by {@link #getRotationMatrix}. 1334 * 1335 * @param X 1336 * defines the axis of the new cooridinate system that coincide with the X axis of the 1337 * original coordinate system. 1338 * 1339 * @param Y 1340 * defines the axis of the new cooridinate system that coincide with the Y axis of the 1341 * original coordinate system. 1342 * 1343 * @param outR 1344 * the transformed rotation matrix. inR and outR should not be the same 1345 * array. 1346 * 1347 * @return <code>true</code> on success. <code>false</code> if the input 1348 * parameters are incorrect, for instance if X and Y define the same 1349 * axis. Or if inR and outR don't have the same length. 1350 * 1351 * @see #getRotationMatrix(float[], float[], float[], float[]) 1352 */ 1353 remapCoordinateSystem(float[] inR, int X, int Y, float[] outR)1354 public static boolean remapCoordinateSystem(float[] inR, int X, int Y, float[] outR) { 1355 if (inR == outR) { 1356 final float[] temp = sTempMatrix; 1357 synchronized (temp) { 1358 // we don't expect to have a lot of contention 1359 if (remapCoordinateSystemImpl(inR, X, Y, temp)) { 1360 final int size = outR.length; 1361 for (int i = 0; i < size; i++) { 1362 outR[i] = temp[i]; 1363 } 1364 return true; 1365 } 1366 } 1367 } 1368 return remapCoordinateSystemImpl(inR, X, Y, outR); 1369 } 1370 remapCoordinateSystemImpl(float[] inR, int X, int Y, float[] outR)1371 private static boolean remapCoordinateSystemImpl(float[] inR, int X, int Y, float[] outR) { 1372 /* 1373 * X and Y define a rotation matrix 'r': 1374 * 1375 * (X==1)?((X&0x80)?-1:1):0 (X==2)?((X&0x80)?-1:1):0 (X==3)?((X&0x80)?-1:1):0 1376 * (Y==1)?((Y&0x80)?-1:1):0 (Y==2)?((Y&0x80)?-1:1):0 (Y==3)?((X&0x80)?-1:1):0 1377 * r[0] ^ r[1] 1378 * 1379 * where the 3rd line is the vector product of the first 2 lines 1380 * 1381 */ 1382 1383 final int length = outR.length; 1384 if (inR.length != length) { 1385 return false; // invalid parameter 1386 } 1387 if ((X & 0x7C) != 0 || (Y & 0x7C) != 0) { 1388 return false; // invalid parameter 1389 } 1390 if (((X & 0x3) == 0) || ((Y & 0x3) == 0)) { 1391 return false; // no axis specified 1392 } 1393 if ((X & 0x3) == (Y & 0x3)) { 1394 return false; // same axis specified 1395 } 1396 1397 // Z is "the other" axis, its sign is either +/- sign(X)*sign(Y) 1398 // this can be calculated by exclusive-or'ing X and Y; except for 1399 // the sign inversion (+/-) which is calculated below. 1400 int Z = X ^ Y; 1401 1402 // extract the axis (remove the sign), offset in the range 0 to 2. 1403 final int x = (X & 0x3) - 1; 1404 final int y = (Y & 0x3) - 1; 1405 final int z = (Z & 0x3) - 1; 1406 1407 // compute the sign of Z (whether it needs to be inverted) 1408 final int axis_y = (z + 1) % 3; 1409 final int axis_z = (z + 2) % 3; 1410 if (((x ^ axis_y) | (y ^ axis_z)) != 0) { 1411 Z ^= 0x80; 1412 } 1413 1414 final boolean sx = (X >= 0x80); 1415 final boolean sy = (Y >= 0x80); 1416 final boolean sz = (Z >= 0x80); 1417 1418 // Perform R * r, in avoiding actual muls and adds. 1419 final int rowLength = ((length == 16) ? 4 : 3); 1420 for (int j = 0; j < 3; j++) { 1421 final int offset = j * rowLength; 1422 for (int i = 0; i < 3; i++) { 1423 if (x == i) outR[offset + i] = sx ? -inR[offset + 0] : inR[offset + 0]; 1424 if (y == i) outR[offset + i] = sy ? -inR[offset + 1] : inR[offset + 1]; 1425 if (z == i) outR[offset + i] = sz ? -inR[offset + 2] : inR[offset + 2]; 1426 } 1427 } 1428 if (length == 16) { 1429 outR[3] = outR[7] = outR[11] = outR[12] = outR[13] = outR[14] = 0; 1430 outR[15] = 1; 1431 } 1432 return true; 1433 } 1434 1435 /** 1436 * Computes the device's orientation based on the rotation matrix. 1437 * <p> 1438 * When it returns, the array values are as follows: 1439 * <ul> 1440 * <li>values[0]: <i>Azimuth</i>, angle of rotation about the -z axis. 1441 * This value represents the angle between the device's y 1442 * axis and the magnetic north pole. When facing north, this 1443 * angle is 0, when facing south, this angle is π. 1444 * Likewise, when facing east, this angle is π/2, and 1445 * when facing west, this angle is -π/2. The range of 1446 * values is -π to π.</li> 1447 * <li>values[1]: <i>Pitch</i>, angle of rotation about the x axis. 1448 * This value represents the angle between a plane parallel 1449 * to the device's screen and a plane parallel to the ground. 1450 * Assuming that the bottom edge of the device faces the 1451 * user and that the screen is face-up, tilting the top edge 1452 * of the device toward the ground creates a positive pitch 1453 * angle. The range of values is -π/2 to π/2.</li> 1454 * <li>values[2]: <i>Roll</i>, angle of rotation about the y axis. This 1455 * value represents the angle between a plane perpendicular 1456 * to the device's screen and a plane perpendicular to the 1457 * ground. Assuming that the bottom edge of the device faces 1458 * the user and that the screen is face-up, tilting the left 1459 * edge of the device toward the ground creates a positive 1460 * roll angle. The range of values is -π to π.</li> 1461 * </ul> 1462 * <p> 1463 * Applying these three rotations in the azimuth, pitch, roll order 1464 * transforms an identity matrix to the rotation matrix passed into this 1465 * method. Also, note that all three orientation angles are expressed in 1466 * <b>radians</b>. 1467 * 1468 * @param R 1469 * rotation matrix see {@link #getRotationMatrix}. 1470 * 1471 * @param values 1472 * an array of 3 floats to hold the result. 1473 * 1474 * @return The array values passed as argument. 1475 * 1476 * @see #getRotationMatrix(float[], float[], float[], float[]) 1477 * @see GeomagneticField 1478 */ getOrientation(float[] R, float[] values)1479 public static float[] getOrientation(float[] R, float[] values) { 1480 /* 1481 * 4x4 (length=16) case: 1482 * / R[ 0] R[ 1] R[ 2] 0 \ 1483 * | R[ 4] R[ 5] R[ 6] 0 | 1484 * | R[ 8] R[ 9] R[10] 0 | 1485 * \ 0 0 0 1 / 1486 * 1487 * 3x3 (length=9) case: 1488 * / R[ 0] R[ 1] R[ 2] \ 1489 * | R[ 3] R[ 4] R[ 5] | 1490 * \ R[ 6] R[ 7] R[ 8] / 1491 * 1492 */ 1493 if (R.length == 9) { 1494 values[0] = (float) Math.atan2(R[1], R[4]); 1495 values[1] = (float) Math.asin(-R[7]); 1496 values[2] = (float) Math.atan2(-R[6], R[8]); 1497 } else { 1498 values[0] = (float) Math.atan2(R[1], R[5]); 1499 values[1] = (float) Math.asin(-R[9]); 1500 values[2] = (float) Math.atan2(-R[8], R[10]); 1501 } 1502 1503 return values; 1504 } 1505 1506 /** 1507 * Computes the Altitude in meters from the atmospheric pressure and the 1508 * pressure at sea level. 1509 * <p> 1510 * Typically the atmospheric pressure is read from a 1511 * {@link Sensor#TYPE_PRESSURE} sensor. The pressure at sea level must be 1512 * known, usually it can be retrieved from airport databases in the 1513 * vicinity. If unknown, you can use {@link #PRESSURE_STANDARD_ATMOSPHERE} 1514 * as an approximation, but absolute altitudes won't be accurate. 1515 * </p> 1516 * <p> 1517 * To calculate altitude differences, you must calculate the difference 1518 * between the altitudes at both points. If you don't know the altitude 1519 * as sea level, you can use {@link #PRESSURE_STANDARD_ATMOSPHERE} instead, 1520 * which will give good results considering the range of pressure typically 1521 * involved. 1522 * </p> 1523 * <p> 1524 * <code><ul> 1525 * float altitude_difference = 1526 * getAltitude(SensorManager.PRESSURE_STANDARD_ATMOSPHERE, pressure_at_point2) 1527 * - getAltitude(SensorManager.PRESSURE_STANDARD_ATMOSPHERE, pressure_at_point1); 1528 * </ul></code> 1529 * </p> 1530 * 1531 * @param p0 pressure at sea level 1532 * @param p atmospheric pressure 1533 * @return Altitude in meters 1534 */ getAltitude(float p0, float p)1535 public static float getAltitude(float p0, float p) { 1536 final float coef = 1.0f / 5.255f; 1537 return 44330.0f * (1.0f - (float) Math.pow(p / p0, coef)); 1538 } 1539 1540 /** Helper function to compute the angle change between two rotation matrices. 1541 * Given a current rotation matrix (R) and a previous rotation matrix 1542 * (prevR) computes the intrinsic rotation around the z, x, and y axes which 1543 * transforms prevR to R. 1544 * outputs a 3 element vector containing the z, x, and y angle 1545 * change at indexes 0, 1, and 2 respectively. 1546 * <p> Each input matrix is either as a 3x3 or 4x4 row-major matrix 1547 * depending on the length of the passed array: 1548 * <p>If the array length is 9, then the array elements represent this matrix 1549 * <pre> 1550 * / R[ 0] R[ 1] R[ 2] \ 1551 * | R[ 3] R[ 4] R[ 5] | 1552 * \ R[ 6] R[ 7] R[ 8] / 1553 *</pre> 1554 * <p>If the array length is 16, then the array elements represent this matrix 1555 * <pre> 1556 * / R[ 0] R[ 1] R[ 2] R[ 3] \ 1557 * | R[ 4] R[ 5] R[ 6] R[ 7] | 1558 * | R[ 8] R[ 9] R[10] R[11] | 1559 * \ R[12] R[13] R[14] R[15] / 1560 *</pre> 1561 * 1562 * See {@link #getOrientation} for more detailed definition of the output. 1563 * 1564 * @param R current rotation matrix 1565 * @param prevR previous rotation matrix 1566 * @param angleChange an an array of floats (z, x, and y) in which the angle change 1567 * (in radians) is stored 1568 */ 1569 getAngleChange(float[] angleChange, float[] R, float[] prevR)1570 public static void getAngleChange(float[] angleChange, float[] R, float[] prevR) { 1571 float rd1 = 0, rd4 = 0, rd6 = 0, rd7 = 0, rd8 = 0; 1572 float ri0 = 0, ri1 = 0, ri2 = 0, ri3 = 0, ri4 = 0, ri5 = 0, ri6 = 0, ri7 = 0, ri8 = 0; 1573 float pri0 = 0, pri1 = 0, pri2 = 0, pri3 = 0, pri4 = 0; 1574 float pri5 = 0, pri6 = 0, pri7 = 0, pri8 = 0; 1575 1576 if (R.length == 9) { 1577 ri0 = R[0]; 1578 ri1 = R[1]; 1579 ri2 = R[2]; 1580 ri3 = R[3]; 1581 ri4 = R[4]; 1582 ri5 = R[5]; 1583 ri6 = R[6]; 1584 ri7 = R[7]; 1585 ri8 = R[8]; 1586 } else if (R.length == 16) { 1587 ri0 = R[0]; 1588 ri1 = R[1]; 1589 ri2 = R[2]; 1590 ri3 = R[4]; 1591 ri4 = R[5]; 1592 ri5 = R[6]; 1593 ri6 = R[8]; 1594 ri7 = R[9]; 1595 ri8 = R[10]; 1596 } 1597 1598 if (prevR.length == 9) { 1599 pri0 = prevR[0]; 1600 pri1 = prevR[1]; 1601 pri2 = prevR[2]; 1602 pri3 = prevR[3]; 1603 pri4 = prevR[4]; 1604 pri5 = prevR[5]; 1605 pri6 = prevR[6]; 1606 pri7 = prevR[7]; 1607 pri8 = prevR[8]; 1608 } else if (prevR.length == 16) { 1609 pri0 = prevR[0]; 1610 pri1 = prevR[1]; 1611 pri2 = prevR[2]; 1612 pri3 = prevR[4]; 1613 pri4 = prevR[5]; 1614 pri5 = prevR[6]; 1615 pri6 = prevR[8]; 1616 pri7 = prevR[9]; 1617 pri8 = prevR[10]; 1618 } 1619 1620 // calculate the parts of the rotation difference matrix we need 1621 // rd[i][j] = pri[0][i] * ri[0][j] + pri[1][i] * ri[1][j] + pri[2][i] * ri[2][j]; 1622 1623 rd1 = pri0 * ri1 + pri3 * ri4 + pri6 * ri7; //rd[0][1] 1624 rd4 = pri1 * ri1 + pri4 * ri4 + pri7 * ri7; //rd[1][1] 1625 rd6 = pri2 * ri0 + pri5 * ri3 + pri8 * ri6; //rd[2][0] 1626 rd7 = pri2 * ri1 + pri5 * ri4 + pri8 * ri7; //rd[2][1] 1627 rd8 = pri2 * ri2 + pri5 * ri5 + pri8 * ri8; //rd[2][2] 1628 1629 angleChange[0] = (float) Math.atan2(rd1, rd4); 1630 angleChange[1] = (float) Math.asin(-rd7); 1631 angleChange[2] = (float) Math.atan2(-rd6, rd8); 1632 1633 } 1634 1635 /** Helper function to convert a rotation vector to a rotation matrix. 1636 * Given a rotation vector (presumably from a ROTATION_VECTOR sensor), returns a 1637 * 9 or 16 element rotation matrix in the array R. R must have length 9 or 16. 1638 * If R.length == 9, the following matrix is returned: 1639 * <pre> 1640 * / R[ 0] R[ 1] R[ 2] \ 1641 * | R[ 3] R[ 4] R[ 5] | 1642 * \ R[ 6] R[ 7] R[ 8] / 1643 *</pre> 1644 * If R.length == 16, the following matrix is returned: 1645 * <pre> 1646 * / R[ 0] R[ 1] R[ 2] 0 \ 1647 * | R[ 4] R[ 5] R[ 6] 0 | 1648 * | R[ 8] R[ 9] R[10] 0 | 1649 * \ 0 0 0 1 / 1650 *</pre> 1651 * @param rotationVector the rotation vector to convert 1652 * @param R an array of floats in which to store the rotation matrix 1653 */ getRotationMatrixFromVector(float[] R, float[] rotationVector)1654 public static void getRotationMatrixFromVector(float[] R, float[] rotationVector) { 1655 1656 float q0; 1657 float q1 = rotationVector[0]; 1658 float q2 = rotationVector[1]; 1659 float q3 = rotationVector[2]; 1660 1661 if (rotationVector.length >= 4) { 1662 q0 = rotationVector[3]; 1663 } else { 1664 q0 = 1 - q1 * q1 - q2 * q2 - q3 * q3; 1665 q0 = (q0 > 0) ? (float) Math.sqrt(q0) : 0; 1666 } 1667 1668 float sq_q1 = 2 * q1 * q1; 1669 float sq_q2 = 2 * q2 * q2; 1670 float sq_q3 = 2 * q3 * q3; 1671 float q1_q2 = 2 * q1 * q2; 1672 float q3_q0 = 2 * q3 * q0; 1673 float q1_q3 = 2 * q1 * q3; 1674 float q2_q0 = 2 * q2 * q0; 1675 float q2_q3 = 2 * q2 * q3; 1676 float q1_q0 = 2 * q1 * q0; 1677 1678 if (R.length == 9) { 1679 R[0] = 1 - sq_q2 - sq_q3; 1680 R[1] = q1_q2 - q3_q0; 1681 R[2] = q1_q3 + q2_q0; 1682 1683 R[3] = q1_q2 + q3_q0; 1684 R[4] = 1 - sq_q1 - sq_q3; 1685 R[5] = q2_q3 - q1_q0; 1686 1687 R[6] = q1_q3 - q2_q0; 1688 R[7] = q2_q3 + q1_q0; 1689 R[8] = 1 - sq_q1 - sq_q2; 1690 } else if (R.length == 16) { 1691 R[0] = 1 - sq_q2 - sq_q3; 1692 R[1] = q1_q2 - q3_q0; 1693 R[2] = q1_q3 + q2_q0; 1694 R[3] = 0.0f; 1695 1696 R[4] = q1_q2 + q3_q0; 1697 R[5] = 1 - sq_q1 - sq_q3; 1698 R[6] = q2_q3 - q1_q0; 1699 R[7] = 0.0f; 1700 1701 R[8] = q1_q3 - q2_q0; 1702 R[9] = q2_q3 + q1_q0; 1703 R[10] = 1 - sq_q1 - sq_q2; 1704 R[11] = 0.0f; 1705 1706 R[12] = R[13] = R[14] = 0.0f; 1707 R[15] = 1.0f; 1708 } 1709 } 1710 1711 /** Helper function to convert a rotation vector to a normalized quaternion. 1712 * Given a rotation vector (presumably from a ROTATION_VECTOR sensor), returns a normalized 1713 * quaternion in the array Q. The quaternion is stored as [w, x, y, z] 1714 * @param rv the rotation vector to convert 1715 * @param Q an array of floats in which to store the computed quaternion 1716 */ getQuaternionFromVector(float[] Q, float[] rv)1717 public static void getQuaternionFromVector(float[] Q, float[] rv) { 1718 if (rv.length >= 4) { 1719 Q[0] = rv[3]; 1720 } else { 1721 Q[0] = 1 - rv[0] * rv[0] - rv[1] * rv[1] - rv[2] * rv[2]; 1722 Q[0] = (Q[0] > 0) ? (float) Math.sqrt(Q[0]) : 0; 1723 } 1724 Q[1] = rv[0]; 1725 Q[2] = rv[1]; 1726 Q[3] = rv[2]; 1727 } 1728 1729 /** 1730 * Requests receiving trigger events for a trigger sensor. 1731 * 1732 * <p> 1733 * When the sensor detects a trigger event condition, such as significant motion in 1734 * the case of the {@link Sensor#TYPE_SIGNIFICANT_MOTION}, the provided trigger listener 1735 * will be invoked once and then its request to receive trigger events will be canceled. 1736 * To continue receiving trigger events, the application must request to receive trigger 1737 * events again. 1738 * </p> 1739 * 1740 * @param listener The listener on which the 1741 * {@link TriggerEventListener#onTrigger(TriggerEvent)} will be delivered. 1742 * @param sensor The sensor to be enabled. 1743 * 1744 * @return true if the sensor was successfully enabled. 1745 * 1746 * @throws IllegalArgumentException when sensor is null or not a trigger sensor. 1747 */ requestTriggerSensor(TriggerEventListener listener, Sensor sensor)1748 public boolean requestTriggerSensor(TriggerEventListener listener, Sensor sensor) { 1749 return requestTriggerSensorImpl(listener, sensor); 1750 } 1751 1752 /** 1753 * @hide 1754 */ requestTriggerSensorImpl(TriggerEventListener listener, Sensor sensor)1755 protected abstract boolean requestTriggerSensorImpl(TriggerEventListener listener, 1756 Sensor sensor); 1757 1758 /** 1759 * Cancels receiving trigger events for a trigger sensor. 1760 * 1761 * <p> 1762 * Note that a Trigger sensor will be auto disabled if 1763 * {@link TriggerEventListener#onTrigger(TriggerEvent)} has triggered. 1764 * This method is provided in case the user wants to explicitly cancel the request 1765 * to receive trigger events. 1766 * </p> 1767 * 1768 * @param listener The listener on which the 1769 * {@link TriggerEventListener#onTrigger(TriggerEvent)} 1770 * is delivered.It should be the same as the one used 1771 * in {@link #requestTriggerSensor(TriggerEventListener, Sensor)} 1772 * @param sensor The sensor for which the trigger request should be canceled. 1773 * If null, it cancels receiving trigger for all sensors associated 1774 * with the listener. 1775 * 1776 * @return true if successfully canceled. 1777 * 1778 * @throws IllegalArgumentException when sensor is a trigger sensor. 1779 */ cancelTriggerSensor(TriggerEventListener listener, Sensor sensor)1780 public boolean cancelTriggerSensor(TriggerEventListener listener, Sensor sensor) { 1781 return cancelTriggerSensorImpl(listener, sensor, true); 1782 } 1783 1784 /** 1785 * @hide 1786 */ cancelTriggerSensorImpl(TriggerEventListener listener, Sensor sensor, boolean disable)1787 protected abstract boolean cancelTriggerSensorImpl(TriggerEventListener listener, 1788 Sensor sensor, boolean disable); 1789 1790 1791 /** 1792 * For testing purposes only. Not for third party applications. 1793 * 1794 * Initialize data injection mode and create a client for data injection. SensorService should 1795 * already be operating in DATA_INJECTION mode for this call succeed. To set SensorService into 1796 * DATA_INJECTION mode "adb shell dumpsys sensorservice data_injection" needs to be called 1797 * through adb. Typically this is done using a host side test. This mode is expected to be used 1798 * only for testing purposes. If the HAL is set to data injection mode, it will ignore the input 1799 * from physical sensors and read sensor data that is injected from the test application. This 1800 * mode is used for testing vendor implementations for various algorithms like Rotation Vector, 1801 * Significant Motion, Step Counter etc. Not all HALs support DATA_INJECTION. This method will 1802 * fail in those cases. Once this method succeeds, the test can call 1803 * {@link injectSensorData(Sensor, float[], int, long)} to inject sensor data into the HAL. 1804 * 1805 * @param enable True to initialize a client in DATA_INJECTION mode. 1806 * False to clean up the native resources. 1807 * 1808 * @return true if the HAL supports data injection and false 1809 * otherwise. 1810 * @hide 1811 */ 1812 @SystemApi initDataInjection(boolean enable)1813 public boolean initDataInjection(boolean enable) { 1814 return initDataInjectionImpl(enable); 1815 } 1816 1817 /** 1818 * @hide 1819 */ initDataInjectionImpl(boolean enable)1820 protected abstract boolean initDataInjectionImpl(boolean enable); 1821 1822 /** 1823 * For testing purposes only. Not for third party applications. 1824 * 1825 * This method is used to inject raw sensor data into the HAL. Call {@link 1826 * initDataInjection(boolean)} before this method to set the HAL in data injection mode. This 1827 * method should be called only if a previous call to initDataInjection has been successful and 1828 * the HAL and SensorService are already opreating in data injection mode. 1829 * 1830 * @param sensor The sensor to inject. 1831 * @param values Sensor values to inject. The length of this 1832 * array must be exactly equal to the number of 1833 * values reported by the sensor type. 1834 * @param accuracy Accuracy of the sensor. 1835 * @param timestamp Sensor timestamp associated with the event. 1836 * 1837 * @return boolean True if the data injection succeeds, false 1838 * otherwise. 1839 * @throws IllegalArgumentException when the sensor is null, 1840 * data injection is not supported by the sensor, values 1841 * are null, incorrect number of values for the sensor, 1842 * sensor accuracy is incorrect or timestamps are 1843 * invalid. 1844 * @hide 1845 */ 1846 @SystemApi injectSensorData(Sensor sensor, float[] values, int accuracy, long timestamp)1847 public boolean injectSensorData(Sensor sensor, float[] values, int accuracy, 1848 long timestamp) { 1849 if (sensor == null) { 1850 throw new IllegalArgumentException("sensor cannot be null"); 1851 } 1852 if (!sensor.isDataInjectionSupported()) { 1853 throw new IllegalArgumentException("sensor does not support data injection"); 1854 } 1855 if (values == null) { 1856 throw new IllegalArgumentException("sensor data cannot be null"); 1857 } 1858 int expectedNumValues = Sensor.getMaxLengthValuesArray(sensor, Build.VERSION_CODES.M); 1859 if (values.length != expectedNumValues) { 1860 throw new IllegalArgumentException("Wrong number of values for sensor " 1861 + sensor.getName() + " actual=" + values.length + " expected=" 1862 + expectedNumValues); 1863 } 1864 if (accuracy < SENSOR_STATUS_NO_CONTACT || accuracy > SENSOR_STATUS_ACCURACY_HIGH) { 1865 throw new IllegalArgumentException("Invalid sensor accuracy"); 1866 } 1867 if (timestamp <= 0) { 1868 throw new IllegalArgumentException("Negative or zero sensor timestamp"); 1869 } 1870 return injectSensorDataImpl(sensor, values, accuracy, timestamp); 1871 } 1872 1873 /** 1874 * @hide 1875 */ injectSensorDataImpl(Sensor sensor, float[] values, int accuracy, long timestamp)1876 protected abstract boolean injectSensorDataImpl(Sensor sensor, float[] values, int accuracy, 1877 long timestamp); 1878 getLegacySensorManager()1879 private LegacySensorManager getLegacySensorManager() { 1880 synchronized (mSensorListByType) { 1881 if (mLegacySensorManager == null) { 1882 Log.i(TAG, "This application is using deprecated SensorManager API which will " 1883 + "be removed someday. Please consider switching to the new API."); 1884 mLegacySensorManager = new LegacySensorManager(this); 1885 } 1886 return mLegacySensorManager; 1887 } 1888 } 1889 getDelay(int rate)1890 private static int getDelay(int rate) { 1891 int delay = -1; 1892 switch (rate) { 1893 case SENSOR_DELAY_FASTEST: 1894 delay = 0; 1895 break; 1896 case SENSOR_DELAY_GAME: 1897 delay = 20000; 1898 break; 1899 case SENSOR_DELAY_UI: 1900 delay = 66667; 1901 break; 1902 case SENSOR_DELAY_NORMAL: 1903 delay = 200000; 1904 break; 1905 default: 1906 delay = rate; 1907 break; 1908 } 1909 return delay; 1910 } 1911 1912 /** @hide */ setOperationParameter(SensorAdditionalInfo parameter)1913 public boolean setOperationParameter(SensorAdditionalInfo parameter) { 1914 return setOperationParameterImpl(parameter); 1915 } 1916 1917 /** @hide */ setOperationParameterImpl(SensorAdditionalInfo parameter)1918 protected abstract boolean setOperationParameterImpl(SensorAdditionalInfo parameter); 1919 } 1920