• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 &pi;.
1444      *                Likewise, when facing east, this angle is &pi;/2, and
1445      *                when facing west, this angle is -&pi;/2. The range of
1446      *                values is -&pi; to &pi;.</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 -&pi;/2 to &pi;/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 -&pi; to &pi;.</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