• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* Copyright (C) 2009 The Android Open Source Project
2 **
3 ** This software is licensed under the terms of the GNU General Public
4 ** License version 2, as published by the Free Software Foundation, and
5 ** may be copied, distributed, and modified under those terms.
6 **
7 ** This program is distributed in the hope that it will be useful,
8 ** but WITHOUT ANY WARRANTY; without even the implied warranty of
9 ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
10 ** GNU General Public License for more details.
11 */
12 
13 #include <math.h>
14 #include "android/hw-sensors.h"
15 #include "android/utils/debug.h"
16 #include "android/utils/misc.h"
17 #include "android/utils/system.h"
18 #include "android/hw-qemud.h"
19 #include "android/globals.h"
20 #include "hw/hw.h"
21 #include "qemu-char.h"
22 #include "qemu-timer.h"
23 #include "android/sensors-port.h"
24 
25 #define  E(...)    derror(__VA_ARGS__)
26 #define  W(...)    dwarning(__VA_ARGS__)
27 #define  D(...)  VERBOSE_PRINT(sensors,__VA_ARGS__)
28 #define  V(...)  VERBOSE_PRINT(init,__VA_ARGS__)
29 
30 /* define T_ACTIVE to 1 to debug transport communications */
31 #define  T_ACTIVE  0
32 
33 #if T_ACTIVE
34 #define  T(...)  VERBOSE_PRINT(sensors,__VA_ARGS__)
35 #else
36 #define  T(...)   ((void)0)
37 #endif
38 
39 /* this code supports emulated sensor hardware
40  *
41  * Note that currently, only the accelerometer is really emulated, and only
42  * for the purpose of allowing auto-rotating the screen in keyboard-less
43  * configurations.
44  *
45  *
46  */
47 
48 
49 static const struct {
50     const char*  name;
51     int          id;
52 } _sSensors[MAX_SENSORS] = {
53 #define SENSOR_(x,y)  { y, ANDROID_SENSOR_##x },
54   SENSORS_LIST
55 #undef SENSOR_
56 };
57 
58 
59 static int
_sensorIdFromName(const char * name)60 _sensorIdFromName( const char*  name )
61 {
62     int  nn;
63     for (nn = 0; nn < MAX_SENSORS; nn++)
64         if (!strcmp(_sSensors[nn].name,name))
65             return _sSensors[nn].id;
66     return -1;
67 }
68 
69 static const char*
_sensorNameFromId(int id)70 _sensorNameFromId( int  id )
71 {
72     int  nn;
73     for (nn = 0; nn < MAX_SENSORS; nn++)
74         if (id == _sSensors[nn].id)
75             return _sSensors[nn].name;
76     return NULL;
77 }
78 
79 /* For common Sensor Value struct */
80 typedef struct {
81     float a, b, c;
82 } SensorValues;
83 
84 typedef struct {
85     float   x, y, z;
86 } Acceleration;
87 
88 
89 typedef struct {
90     float  x, y, z;
91 } MagneticField;
92 
93 
94 typedef struct {
95     float  azimuth;
96     float  pitch;
97     float  roll;
98 } Orientation;
99 
100 
101 typedef struct {
102     float  celsius;
103 } Temperature;
104 
105 
106 typedef struct {
107     float  value;
108 } Proximity;
109 
110 typedef struct {
111     char       enabled;
112     union {
113         SensorValues   value;
114         Acceleration   acceleration;
115         MagneticField  magnetic;
116         Orientation    orientation;
117         Temperature    temperature;
118         Proximity      proximity;
119     } u;
120 } Sensor;
121 
122 /*
123  * - when the qemu-specific sensors HAL module starts, it sends
124  *   "list-sensors"
125  *
126  * - this code replies with a string containing an integer corresponding
127  *   to a bitmap of available hardware sensors in the current AVD
128  *   configuration (e.g. "1" a.k.a (1 << ANDROID_SENSOR_ACCELERATION))
129  *
130  * - the HAL module sends "set:<sensor>:<flag>" to enable or disable
131  *   the report of a given sensor state. <sensor> must be the name of
132  *   a given sensor (e.g. "accelerometer"), and <flag> must be either
133  *   "1" (to enable) or "0" (to disable).
134  *
135  * - Once at least one sensor is "enabled", this code should periodically
136  *   send information about the corresponding enabled sensors. The default
137  *   period is 200ms.
138  *
139  * - the HAL module sends "set-delay:<delay>", where <delay> is an integer
140  *   corresponding to a time delay in milli-seconds. This corresponds to
141  *   a new interval between sensor events sent by this code to the HAL
142  *   module.
143  *
144  * - the HAL module can also send a "wake" command. This code should simply
145  *   send the "wake" back to the module. This is used internally to wake a
146  *   blocking read that happens in a different thread. This ping-pong makes
147  *   the code in the HAL module very simple.
148  *
149  * - each timer tick, this code sends sensor reports in the following
150  *   format (each line corresponds to a different line sent to the module):
151  *
152  *      acceleration:<x>:<y>:<z>
153  *      magnetic-field:<x>:<y>:<z>
154  *      orientation:<azimuth>:<pitch>:<roll>
155  *      temperature:<celsius>
156  *      sync:<time_us>
157  *
158  *   Where each line before the sync:<time_us> is optional and will only
159  *   appear if the corresponding sensor has been enabled by the HAL module.
160  *
161  *   Note that <time_us> is the VM time in micro-seconds when the report
162  *   was "taken" by this code. This is adjusted by the HAL module to
163  *   emulated system time (using the first sync: to compute an adjustment
164  *   offset).
165  */
166 #define  HEADER_SIZE  4
167 #define  BUFFER_SIZE  512
168 
169 typedef struct HwSensorClient   HwSensorClient;
170 
171 typedef struct {
172     QemudService*       service;
173     Sensor              sensors[MAX_SENSORS];
174     HwSensorClient*     clients;
175     AndroidSensorsPort* sensors_port;
176 } HwSensors;
177 
178 struct HwSensorClient {
179     HwSensorClient*  next;
180     HwSensors*       sensors;
181     QemudClient*     client;
182     QEMUTimer*       timer;
183     uint32_t         enabledMask;
184     int32_t          delay_ms;
185 };
186 
187 static void
_hwSensorClient_free(HwSensorClient * cl)188 _hwSensorClient_free( HwSensorClient*  cl )
189 {
190     /* remove from sensors's list */
191     if (cl->sensors) {
192         HwSensorClient**  pnode = &cl->sensors->clients;
193         for (;;) {
194             HwSensorClient*  node = *pnode;
195             if (node == NULL)
196                 break;
197             if (node == cl) {
198                 *pnode = cl->next;
199                 break;
200             }
201             pnode = &node->next;
202         }
203         cl->next    = NULL;
204         cl->sensors = NULL;
205     }
206 
207     /* close QEMUD client, if any */
208     if (cl->client) {
209         qemud_client_close(cl->client);
210         cl->client = NULL;
211     }
212     /* remove timer, if any */
213     if (cl->timer) {
214         qemu_del_timer(cl->timer);
215         qemu_free_timer(cl->timer);
216         cl->timer = NULL;
217     }
218     AFREE(cl);
219 }
220 
221 /* forward */
222 static void  _hwSensorClient_tick(void*  opaque);
223 
224 
225 static HwSensorClient*
_hwSensorClient_new(HwSensors * sensors)226 _hwSensorClient_new( HwSensors*  sensors )
227 {
228     HwSensorClient*  cl;
229 
230     ANEW0(cl);
231 
232     cl->sensors     = sensors;
233     cl->enabledMask = 0;
234     cl->delay_ms    = 800;
235     cl->timer       = qemu_new_timer_ns(vm_clock, _hwSensorClient_tick, cl);
236 
237     cl->next         = sensors->clients;
238     sensors->clients = cl;
239 
240     return cl;
241 }
242 
243 /* forward */
244 
245 static void  _hwSensorClient_receive( HwSensorClient*  cl,
246                                       uint8_t*         query,
247                                       int              querylen );
248 
249 /* Qemud service management */
250 
251 static void
_hwSensorClient_recv(void * opaque,uint8_t * msg,int msglen,QemudClient * client)252 _hwSensorClient_recv( void*  opaque, uint8_t*  msg, int  msglen,
253                       QemudClient*  client )
254 {
255     HwSensorClient*  cl = opaque;
256 
257     _hwSensorClient_receive(cl, msg, msglen);
258 }
259 
260 static void
_hwSensorClient_close(void * opaque)261 _hwSensorClient_close( void*  opaque )
262 {
263     HwSensorClient*  cl = opaque;
264 
265     /* the client is already closed here */
266     cl->client = NULL;
267     _hwSensorClient_free(cl);
268 }
269 
270 /* send a one-line message to the HAL module through a qemud channel */
271 static void
_hwSensorClient_send(HwSensorClient * cl,const uint8_t * msg,int msglen)272 _hwSensorClient_send( HwSensorClient*  cl, const uint8_t*  msg, int  msglen )
273 {
274     D("%s: '%s'", __FUNCTION__, quote_bytes((const void*)msg, msglen));
275     qemud_client_send(cl->client, msg, msglen);
276 }
277 
278 static int
_hwSensorClient_enabled(HwSensorClient * cl,int sensorId)279 _hwSensorClient_enabled( HwSensorClient*  cl, int  sensorId )
280 {
281     return (cl->enabledMask & (1 << sensorId)) != 0;
282 }
283 
284 /* this function is called periodically to send sensor reports
285  * to the HAL module, and re-arm the timer if necessary
286  */
287 static void
_hwSensorClient_tick(void * opaque)288 _hwSensorClient_tick( void*  opaque )
289 {
290     HwSensorClient*  cl = opaque;
291     HwSensors*       hw  = cl->sensors;
292     int64_t          delay = cl->delay_ms;
293     int64_t          now_ns;
294     uint32_t         mask  = cl->enabledMask;
295     Sensor*          sensor;
296     char             buffer[128];
297 
298     if (_hwSensorClient_enabled(cl, ANDROID_SENSOR_ACCELERATION)) {
299         sensor = &hw->sensors[ANDROID_SENSOR_ACCELERATION];
300         snprintf(buffer, sizeof buffer, "acceleration:%g:%g:%g",
301                  sensor->u.acceleration.x,
302                  sensor->u.acceleration.y,
303                  sensor->u.acceleration.z);
304         _hwSensorClient_send(cl, (uint8_t*)buffer, strlen(buffer));
305     }
306 
307     if (_hwSensorClient_enabled(cl, ANDROID_SENSOR_MAGNETIC_FIELD)) {
308         sensor = &hw->sensors[ANDROID_SENSOR_MAGNETIC_FIELD];
309         /* NOTE: sensors HAL expects "magnetic", not "magnetic-field" name here. */
310         snprintf(buffer, sizeof buffer, "magnetic:%g:%g:%g",
311                  sensor->u.magnetic.x,
312                  sensor->u.magnetic.y,
313                  sensor->u.magnetic.z);
314         _hwSensorClient_send(cl, (uint8_t*)buffer, strlen(buffer));
315     }
316 
317     if (_hwSensorClient_enabled(cl, ANDROID_SENSOR_ORIENTATION)) {
318         sensor = &hw->sensors[ANDROID_SENSOR_ORIENTATION];
319         snprintf(buffer, sizeof buffer, "orientation:%g:%g:%g",
320                  sensor->u.orientation.azimuth,
321                  sensor->u.orientation.pitch,
322                  sensor->u.orientation.roll);
323         _hwSensorClient_send(cl, (uint8_t*)buffer, strlen(buffer));
324     }
325 
326     if (_hwSensorClient_enabled(cl, ANDROID_SENSOR_TEMPERATURE)) {
327         sensor = &hw->sensors[ANDROID_SENSOR_TEMPERATURE];
328         snprintf(buffer, sizeof buffer, "temperature:%g",
329                  sensor->u.temperature.celsius);
330         _hwSensorClient_send(cl, (uint8_t*)buffer, strlen(buffer));
331     }
332 
333     if (_hwSensorClient_enabled(cl, ANDROID_SENSOR_PROXIMITY)) {
334         sensor = &hw->sensors[ANDROID_SENSOR_PROXIMITY];
335         snprintf(buffer, sizeof buffer, "proximity:%g",
336                  sensor->u.proximity.value);
337         _hwSensorClient_send(cl, (uint8_t*) buffer, strlen(buffer));
338     }
339 
340     now_ns = qemu_get_clock_ns(vm_clock);
341 
342     snprintf(buffer, sizeof buffer, "sync:%" PRId64, now_ns/1000);
343     _hwSensorClient_send(cl, (uint8_t*)buffer, strlen(buffer));
344 
345     /* rearm timer, use a minimum delay of 20 ms, just to
346      * be safe.
347      */
348     if (mask == 0)
349         return;
350 
351     if (delay < 20)
352         delay = 20;
353 
354     delay *= 1000000LL;  /* convert to nanoseconds */
355     qemu_mod_timer(cl->timer, now_ns + delay);
356 }
357 
358 /* handle incoming messages from the HAL module */
359 static void
_hwSensorClient_receive(HwSensorClient * cl,uint8_t * msg,int msglen)360 _hwSensorClient_receive( HwSensorClient*  cl, uint8_t*  msg, int  msglen )
361 {
362     HwSensors*  hw = cl->sensors;
363 
364     D("%s: '%.*s'", __FUNCTION__, msglen, msg);
365 
366     /* "list-sensors" is used to get an integer bit map of
367      * available emulated sensors. We compute the mask from the
368      * current hardware configuration.
369      */
370     if (msglen == 12 && !memcmp(msg, "list-sensors", 12)) {
371         char  buff[12];
372         int   mask = 0;
373         int   nn;
374 
375         for (nn = 0; nn < MAX_SENSORS; nn++) {
376             if (hw->sensors[nn].enabled)
377                 mask |= (1 << nn);
378         }
379 
380         snprintf(buff, sizeof buff, "%d", mask);
381         _hwSensorClient_send(cl, (const uint8_t*)buff, strlen(buff));
382         return;
383     }
384 
385     /* "wake" is a special message that must be sent back through
386      * the channel. It is used to exit a blocking read.
387      */
388     if (msglen == 4 && !memcmp(msg, "wake", 4)) {
389         _hwSensorClient_send(cl, (const uint8_t*)"wake", 4);
390         return;
391     }
392 
393     /* "set-delay:<delay>" is used to set the delay in milliseconds
394      * between sensor events
395      */
396     if (msglen > 10 && !memcmp(msg, "set-delay:", 10)) {
397         cl->delay_ms = atoi((const char*)msg+10);
398         if (cl->enabledMask != 0)
399             _hwSensorClient_tick(cl);
400 
401         return;
402     }
403 
404     /* "set:<name>:<state>" is used to enable/disable a given
405      * sensor. <state> must be 0 or 1
406      */
407     if (msglen > 4 && !memcmp(msg, "set:", 4)) {
408         char*  q;
409         int    id, enabled, oldEnabledMask = cl->enabledMask;
410         msg += 4;
411         q    = strchr((char*)msg, ':');
412         if (q == NULL) {  /* should not happen */
413             D("%s: ignore bad 'set' command", __FUNCTION__);
414             return;
415         }
416         *q++ = 0;
417 
418         id = _sensorIdFromName((const char*)msg);
419         if (id < 0 || id >= MAX_SENSORS) {
420             D("%s: ignore unknown sensor name '%s'", __FUNCTION__, msg);
421             return;
422         }
423 
424         if (!hw->sensors[id].enabled) {
425             D("%s: trying to set disabled %s sensor", __FUNCTION__, msg);
426             return;
427         }
428         enabled = (q[0] == '1');
429 
430         if (enabled)
431             cl->enabledMask |= (1 << id);
432         else
433             cl->enabledMask &= ~(1 << id);
434 
435         if (cl->enabledMask != oldEnabledMask) {
436             D("%s: %s %s sensor", __FUNCTION__,
437                 (cl->enabledMask & (1 << id))  ? "enabling" : "disabling",  msg);
438         }
439 
440         /* If emulating device is connected update sensor state there too. */
441         if (hw->sensors_port != NULL && sensors_port_is_connected(hw->sensors_port)) {
442             if (enabled) {
443                 sensors_port_enable_sensor(hw->sensors_port, (const char*)msg);
444             } else {
445                 sensors_port_disable_sensor(hw->sensors_port, (const char*)msg);
446             }
447         }
448 
449         _hwSensorClient_tick(cl);
450         return;
451     }
452 
453     D("%s: ignoring unknown query", __FUNCTION__);
454 }
455 
456 /* Saves sensor-specific client data to snapshot */
457 static void
_hwSensorClient_save(QEMUFile * f,QemudClient * client,void * opaque)458 _hwSensorClient_save( QEMUFile*  f, QemudClient*  client, void*  opaque  )
459 {
460     HwSensorClient* sc = opaque;
461 
462     qemu_put_be32(f, sc->delay_ms);
463     qemu_put_be32(f, sc->enabledMask);
464     qemu_put_timer(f, sc->timer);
465 }
466 
467 /* Loads sensor-specific client data from snapshot */
468 static int
_hwSensorClient_load(QEMUFile * f,QemudClient * client,void * opaque)469 _hwSensorClient_load( QEMUFile*  f, QemudClient*  client, void*  opaque  )
470 {
471     HwSensorClient* sc = opaque;
472 
473     sc->delay_ms = qemu_get_be32(f);
474     sc->enabledMask = qemu_get_be32(f);
475     qemu_get_timer(f, sc->timer);
476 
477     return 0;
478 }
479 
480 static QemudClient*
_hwSensors_connect(void * opaque,QemudService * service,int channel,const char * client_param)481 _hwSensors_connect( void*  opaque,
482                     QemudService*  service,
483                     int  channel,
484                     const char* client_param )
485 {
486     HwSensors*       sensors = opaque;
487     HwSensorClient*  cl      = _hwSensorClient_new(sensors);
488     QemudClient*     client  = qemud_client_new(service, channel, client_param, cl,
489                                                 _hwSensorClient_recv,
490                                                 _hwSensorClient_close,
491                                                 _hwSensorClient_save,
492                                                 _hwSensorClient_load );
493     qemud_client_set_framing(client, 1);
494     cl->client = client;
495 
496     return client;
497 }
498 
499 /* change the value of the emulated sensor vector */
500 static void
_hwSensors_setSensorValue(HwSensors * h,int sensor_id,float a,float b,float c)501 _hwSensors_setSensorValue( HwSensors*  h, int sensor_id, float a, float b, float c )
502 {
503     Sensor* s = &h->sensors[sensor_id];
504 
505     s->u.value.a = a;
506     s->u.value.b = b;
507     s->u.value.c = c;
508 }
509 
510 /* Saves available sensors to allow checking availability when loaded.
511  */
512 static void
_hwSensors_save(QEMUFile * f,QemudService * sv,void * opaque)513 _hwSensors_save( QEMUFile*  f, QemudService*  sv, void*  opaque)
514 {
515     HwSensors* h = opaque;
516 
517     // number of sensors
518     qemu_put_be32(f, MAX_SENSORS);
519     AndroidSensor i;
520     for (i = 0 ; i < MAX_SENSORS; i++) {
521         Sensor* s = &h->sensors[i];
522         qemu_put_be32(f, s->enabled);
523 
524         /* this switch ensures that a warning is raised when a new sensor is
525          * added and is not added here as well.
526          */
527         switch (i) {
528         case ANDROID_SENSOR_ACCELERATION:
529             qemu_put_float(f, s->u.acceleration.x);
530             qemu_put_float(f, s->u.acceleration.y);
531             qemu_put_float(f, s->u.acceleration.z);
532             break;
533         case ANDROID_SENSOR_MAGNETIC_FIELD:
534             qemu_put_float(f, s->u.magnetic.x);
535             qemu_put_float(f, s->u.magnetic.y);
536             qemu_put_float(f, s->u.magnetic.z);
537             break;
538         case ANDROID_SENSOR_ORIENTATION:
539             qemu_put_float(f, s->u.orientation.azimuth);
540             qemu_put_float(f, s->u.orientation.pitch);
541             qemu_put_float(f, s->u.orientation.roll);
542             break;
543         case ANDROID_SENSOR_TEMPERATURE:
544             qemu_put_float(f, s->u.temperature.celsius);
545             break;
546         case ANDROID_SENSOR_PROXIMITY:
547             qemu_put_float(f, s->u.proximity.value);
548             break;
549         case MAX_SENSORS:
550             break;
551         }
552     }
553 }
554 
555 
556 static int
_hwSensors_load(QEMUFile * f,QemudService * s,void * opaque)557 _hwSensors_load( QEMUFile*  f, QemudService*  s, void*  opaque)
558 {
559     HwSensors* h = opaque;
560 
561     /* check number of sensors */
562     int32_t num_sensors = qemu_get_be32(f);
563     if (num_sensors > MAX_SENSORS) {
564         D("%s: cannot load: snapshot requires %d sensors, %d available\n",
565           __FUNCTION__, num_sensors, MAX_SENSORS);
566         return -EIO;
567     }
568 
569     /* load sensor state */
570     AndroidSensor i;
571     for (i = 0 ; i < num_sensors; i++) {
572         Sensor* s = &h->sensors[i];
573         s->enabled = qemu_get_be32(f);
574 
575         /* this switch ensures that a warning is raised when a new sensor is
576          * added and is not added here as well.
577          */
578         switch (i) {
579         case ANDROID_SENSOR_ACCELERATION:
580             s->u.acceleration.x = qemu_get_float(f);
581             s->u.acceleration.y = qemu_get_float(f);
582             s->u.acceleration.z = qemu_get_float(f);
583             break;
584         case ANDROID_SENSOR_MAGNETIC_FIELD:
585             s->u.magnetic.x = qemu_get_float(f);
586             s->u.magnetic.y = qemu_get_float(f);
587             s->u.magnetic.z = qemu_get_float(f);
588             break;
589         case ANDROID_SENSOR_ORIENTATION:
590             s->u.orientation.azimuth = qemu_get_float(f);
591             s->u.orientation.pitch   = qemu_get_float(f);
592             s->u.orientation.roll    = qemu_get_float(f);
593             break;
594         case ANDROID_SENSOR_TEMPERATURE:
595             s->u.temperature.celsius = qemu_get_float(f);
596             break;
597         case ANDROID_SENSOR_PROXIMITY:
598             s->u.proximity.value = qemu_get_float(f);
599             break;
600         case MAX_SENSORS:
601             break;
602         }
603     }
604 
605     /* The following is necessary when we resume a snaphost
606      * created by an older version of the emulator that provided
607      * less hardware sensors.
608      */
609     for ( ; i < MAX_SENSORS; i++ ) {
610         h->sensors[i].enabled = 0;
611     }
612 
613     return 0;
614 }
615 
616 
617 /* change the emulated proximity */
618 static void
_hwSensors_setProximity(HwSensors * h,float value)619 _hwSensors_setProximity( HwSensors*  h, float value )
620 {
621     Sensor*  s = &h->sensors[ANDROID_SENSOR_PROXIMITY];
622     s->u.proximity.value = value;
623 }
624 
625 /* change the coarse orientation (landscape/portrait) of the emulated device */
626 static void
_hwSensors_setCoarseOrientation(HwSensors * h,AndroidCoarseOrientation orient)627 _hwSensors_setCoarseOrientation( HwSensors*  h, AndroidCoarseOrientation  orient )
628 {
629     /* The Android framework computes the orientation by looking at
630      * the accelerometer sensor (*not* the orientation sensor !)
631      *
632      * That's because the gravity is a constant 9.81 vector that
633      * can be determined quite easily.
634      *
635      * Also, for some reason, the framework code considers that the phone should
636      * be inclined by 30 degrees along the phone's X axis to be considered
637      * in its ideal "vertical" position
638      *
639      * If the phone is completely vertical, rotating it will not do anything !
640      */
641     const double  g      = 9.81;
642     const double  angle  = 20.0;
643     const double  cos_angle = cos(angle/M_PI);
644     const double  sin_angle = sin(angle/M_PI);
645 
646     switch (orient) {
647     case ANDROID_COARSE_PORTRAIT:
648         _hwSensors_setSensorValue( h, ANDROID_SENSOR_ACCELERATION, 0., g*cos_angle, g*sin_angle );
649         break;
650 
651     case ANDROID_COARSE_LANDSCAPE:
652         _hwSensors_setSensorValue( h, ANDROID_SENSOR_ACCELERATION, g*cos_angle, 0., g*sin_angle );
653         break;
654     default:
655         ;
656     }
657 }
658 
659 
660 /* initialize the sensors state */
661 static void
_hwSensors_init(HwSensors * h)662 _hwSensors_init( HwSensors*  h )
663 {
664     /* Try to see if there is a device attached that can be used for
665      * sensor emulation. */
666     h->sensors_port = sensors_port_create(h);
667     if (h->sensors_port == NULL) {
668         V("Realistic sensor emulation is not available, since the remote controller is not accessible:\n %s",
669           strerror(errno));
670     }
671 
672     h->service = qemud_service_register("sensors", 0, h, _hwSensors_connect,
673                                         _hwSensors_save, _hwSensors_load);
674 
675     if (android_hw->hw_accelerometer) {
676         h->sensors[ANDROID_SENSOR_ACCELERATION].enabled = 1;
677     }
678 
679     if (android_hw->hw_sensors_proximity) {
680         h->sensors[ANDROID_SENSOR_PROXIMITY].enabled = 1;
681     }
682 
683     if (android_hw->hw_sensors_magnetic_field) {
684         h->sensors[ANDROID_SENSOR_MAGNETIC_FIELD].enabled = 1;
685     }
686 
687     if (android_hw->hw_sensors_orientation) {
688         h->sensors[ANDROID_SENSOR_ORIENTATION].enabled = 1;
689     }
690 
691     if (android_hw->hw_sensors_temperature) {
692         h->sensors[ANDROID_SENSOR_TEMPERATURE].enabled = 1;
693     }
694 
695     if (h->sensors_port != NULL) {
696         /* Init sensors on the attached device. */
697         sensors_port_init_sensors(h->sensors_port);
698     }
699 
700     /* XXX: TODO: Add other tests when we add the corresponding
701         * properties to hardware-properties.ini et al. */
702 
703     _hwSensors_setCoarseOrientation(h, ANDROID_COARSE_PORTRAIT);
704     _hwSensors_setProximity(h, 1);
705 }
706 
707 static HwSensors    _sensorsState[1];
708 
709 void
android_hw_sensors_init(void)710 android_hw_sensors_init( void )
711 {
712     HwSensors*  hw = _sensorsState;
713 
714     if (hw->service == NULL) {
715         _hwSensors_init(hw);
716         D("%s: sensors qemud service initialized", __FUNCTION__);
717     }
718 }
719 
720 /* change the coarse orientation value */
721 extern void
android_sensors_set_coarse_orientation(AndroidCoarseOrientation orient)722 android_sensors_set_coarse_orientation( AndroidCoarseOrientation  orient )
723 {
724     android_hw_sensors_init();
725     _hwSensors_setCoarseOrientation(_sensorsState, orient);
726 }
727 
728 /* Get sensor name from sensor id */
729 extern const char*
android_sensors_get_name_from_id(int sensor_id)730 android_sensors_get_name_from_id( int sensor_id )
731 {
732     if (sensor_id < 0 || sensor_id >= MAX_SENSORS)
733         return NULL;
734 
735     return _sensorNameFromId(sensor_id);
736 }
737 
738 /* Get sensor id from sensor name */
739 extern int
android_sensors_get_id_from_name(char * sensorname)740 android_sensors_get_id_from_name( char* sensorname )
741 {
742     HwSensors* hw = _sensorsState;
743 
744     if (sensorname == NULL)
745         return SENSOR_STATUS_UNKNOWN;
746 
747     int id = _sensorIdFromName(sensorname);
748 
749     if (id < 0 || id >= MAX_SENSORS)
750         return SENSOR_STATUS_UNKNOWN;
751 
752     if (hw->service != NULL) {
753         if (! hw->sensors[id].enabled)
754             return SENSOR_STATUS_DISABLED;
755     } else
756         return SENSOR_STATUS_NO_SERVICE;
757 
758     return id;
759 }
760 
761 /* Interface of reading the data for all sensors */
762 extern int
android_sensors_get(int sensor_id,float * a,float * b,float * c)763 android_sensors_get( int sensor_id, float* a, float* b, float* c )
764 {
765     HwSensors* hw = _sensorsState;
766 
767     *a = 0;
768     *b = 0;
769     *c = 0;
770 
771     if (sensor_id < 0 || sensor_id >= MAX_SENSORS)
772         return SENSOR_STATUS_UNKNOWN;
773 
774     Sensor* sensor = &hw->sensors[sensor_id];
775     if (hw->service != NULL) {
776         if (! sensor->enabled)
777             return SENSOR_STATUS_DISABLED;
778     } else
779         return SENSOR_STATUS_NO_SERVICE;
780 
781     *a = sensor->u.value.a;
782     *b = sensor->u.value.b;
783     *c = sensor->u.value.c;
784 
785     return SENSOR_STATUS_OK;
786 }
787 
788 /* Interface of setting the data for all sensors */
789 extern int
android_sensors_set(int sensor_id,float a,float b,float c)790 android_sensors_set( int sensor_id, float a, float b, float c )
791 {
792     HwSensors* hw = _sensorsState;
793 
794     if (sensor_id < 0 || sensor_id >= MAX_SENSORS)
795         return SENSOR_STATUS_UNKNOWN;
796 
797     if (hw->service != NULL) {
798         if (! hw->sensors[sensor_id].enabled)
799             return SENSOR_STATUS_DISABLED;
800     } else
801         return SENSOR_STATUS_NO_SERVICE;
802 
803     _hwSensors_setSensorValue(hw, sensor_id, a, b, c);
804 
805     return SENSOR_STATUS_OK;
806 }
807 
808 /* Get Sensor from sensor id */
809 extern uint8_t
android_sensors_get_sensor_status(int sensor_id)810 android_sensors_get_sensor_status( int sensor_id )
811 {
812     HwSensors* hw = _sensorsState;
813 
814     if (sensor_id < 0 || sensor_id >= MAX_SENSORS)
815         return SENSOR_STATUS_UNKNOWN;
816 
817     return hw->sensors[sensor_id].enabled;
818 }
819