efl/src/lib/eeze/eeze_sensor.c

413 lines
12 KiB
C

#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <Eina.h>
#include <Ecore.h>
#include <Eeze_Sensor.h>
#include "eeze_sensor_private.h"
#include "../../static_libs/buildsystem/buildsystem.h"
EAPI int EEZE_SENSOR_EVENT_ACCELEROMETER;
EAPI int EEZE_SENSOR_EVENT_GRAVITY;
EAPI int EEZE_SENSOR_EVENT_LINEAR_ACCELERATION;
EAPI int EEZE_SENSOR_EVENT_DEVICE_ORIENTATION;
EAPI int EEZE_SENSOR_EVENT_MAGNETIC;
EAPI int EEZE_SENSOR_EVENT_ORIENTATION;
EAPI int EEZE_SENSOR_EVENT_GYROSCOPE;
EAPI int EEZE_SENSOR_EVENT_LIGHT;
EAPI int EEZE_SENSOR_EVENT_PROXIMITY;
EAPI int EEZE_SENSOR_EVENT_SNAP;
EAPI int EEZE_SENSOR_EVENT_SHAKE;
EAPI int EEZE_SENSOR_EVENT_DOUBLETAP;
EAPI int EEZE_SENSOR_EVENT_PANNING;
EAPI int EEZE_SENSOR_EVENT_PANNING_BROWSE;
EAPI int EEZE_SENSOR_EVENT_TILT;
EAPI int EEZE_SENSOR_EVENT_FACEDOWN;
EAPI int EEZE_SENSOR_EVENT_DIRECT_CALL;
EAPI int EEZE_SENSOR_EVENT_SMART_ALERT;
EAPI int EEZE_SENSOR_EVENT_NO_MOVE;
EAPI int EEZE_SENSOR_EVENT_BAROMETER;
EAPI int EEZE_SENSOR_EVENT_TEMPERATURE;
static Eeze_Sensor *g_handle;
static Eina_Prefix *pfx;
/* Priority order for modules. The one with the highest order of the available
* ones will be used. This in good enough for now as we only have three modules
* and one is a test harness anyway. If the number of modules grows we might
* re-think the priority handling, but we should do this when the need arise.
*/
static const char *_module_priority[] = {
"fake",
"udev",
NULL
};
/* Search through the list of loaded module and return the one with the highest
* priority.
*/
Eeze_Sensor_Module *
_highest_priority_module_get(void)
{
Eeze_Sensor_Module *module = NULL;
int i = 0;
while (_module_priority[i] != NULL)
{
module = eina_hash_find(g_handle->modules, _module_priority[i]);
if (module) return module;
i++;
}
return NULL;
}
/* Utility function to take the given sensor type and get the matching sensor
* object from the highest priority module.
*/
EAPI Eeze_Sensor_Obj *
eeze_sensor_obj_get(Eeze_Sensor_Type sensor_type)
{
Eina_List *l;
Eeze_Sensor_Obj *obj, *sens;
Eeze_Sensor_Module *module;
module = _highest_priority_module_get();
if (!module) return NULL;
EINA_LIST_FOREACH(module->sensor_list, l, obj)
{
if (obj->type == sensor_type)
{
sens = calloc(1, sizeof(Eeze_Sensor_Obj));
if (!sens) return NULL;
memcpy(sens, obj, sizeof(Eeze_Sensor_Obj));
return sens;
}
}
return NULL;
}
static void
eeze_sensor_modules_load(void)
{
char buf[PATH_MAX];
/* Check for available runtime modules and load them. In some cases the
* un-installed modules to be used from the local build dir. Coverage check
* is one of these items. We do load the modules from the builddir if the
* environment is set. Normal case is to use installed modules from system
*/
#ifdef NEED_RUN_IN_TREE
if (
#if defined(HAVE_GETUID) && defined(HAVE_GETEUID)
(getuid() == geteuid()) &&
#endif
(getenv("EFL_RUN_IN_TREE")))
{
const char **itr;
for (itr = _module_priority; *itr != NULL; itr++)
{
bs_mod_dir_get(buf, sizeof(buf), "eeze/sensor", *itr);
g_handle->modules_array = eina_module_list_get(
g_handle->modules_array, buf, EINA_FALSE, NULL, NULL);
}
}
else
#endif
{
snprintf(buf, sizeof(buf), "%s/eeze/modules/sensor",
eina_prefix_lib_get(pfx));
g_handle->modules_array = eina_module_arch_list_get(NULL, buf,
MODULE_ARCH);
}
if (!g_handle->modules_array)
{
ERR("No modules found!");
return;
}
// XXX: MODFIX: do not list ALL modules and load them ALL! this is
// this will, for example, load both udev AND fake modules - run
// their init funcs etc. etc. why? no!
eina_module_list_load(g_handle->modules_array);
}
static void
eeze_sensor_modules_unload(void)
{
if (!g_handle->modules_array) return;
eina_module_list_unload(g_handle->modules_array);
eina_module_list_free(g_handle->modules_array);
eina_array_free(g_handle->modules_array);
g_handle->modules_array = NULL;
}
/* This function is offered to the modules to register itself after they have
* been loaded in initialized. They stay in the hash function until they
* unregister themself.
*/
EAPI Eina_Bool
eeze_sensor_module_register(const char *name, Eeze_Sensor_Module *mod)
{
Eeze_Sensor_Module *module = NULL;
if (!mod) return EINA_FALSE;
module = mod;
if (!module->init) return EINA_FALSE;
if (!(module->init())) return EINA_FALSE;
INF("Registered module %s", name);
return eina_hash_add(g_handle->modules, name, module);
}
/* This function is offered to the modules to unregsiter itself. When requested
* we remove them safely from the hash.
*/
EAPI Eina_Bool
eeze_sensor_module_unregister(const char *name)
{
DBG("Unregister module %s", name);
Eeze_Sensor_Module *module = NULL;
module = eina_hash_find(g_handle->modules, name);
if (!module) return EINA_FALSE;
if (module->shutdown)
module->shutdown();
return eina_hash_del(g_handle->modules, name, NULL);
}
/* Create a new sensor object for a given sensor type. This functions allocates
* the needed memory and links it with the matching sensor from the loaded
* modules. It also does an initial synchronous read to fill the sensor object
* with values.
* Make sure to use the eeze_sensor_free function to remove this sensor object
* when it is no longer needed.
*/
EAPI Eeze_Sensor_Obj *
eeze_sensor_new(Eeze_Sensor_Type type)
{
Eeze_Sensor_Obj *sens;
Eeze_Sensor_Module *module = NULL;
sens = eeze_sensor_obj_get(type);
if (!sens) return NULL;
module = _highest_priority_module_get();
if (!module)
{
free(sens);
return NULL;
}
if (!module->read)
{
free(sens);
return NULL;
}
/* The read is asynchronous here as we want to make sure that the sensor
* object has valid data when created. As we give back cached values we
* have a race condition when we do a asynchronous read here and the
* application asks for cached data before the reply came in. This logic has
* the downside that the sensor creation takes longer. But that is only a
*initial cost.
*/
if (module->read(sens))
return sens;
free(sens);
return NULL;
}
/* Free sensor object created with eeze_sensor_new */
EAPI void
eeze_sensor_free(Eeze_Sensor_Obj *sens)
{
if (!sens) return;
free(sens);
}
/* All of the below getter function do access the cached data from the last
* sensor read. It is way faster this way but also means that the timestamp
* should be checked to ensure recent data if needed.
*/
EAPI Eina_Bool
eeze_sensor_accuracy_get(Eeze_Sensor_Obj *sens, int *accuracy)
{
if (!sens) return EINA_FALSE;
*accuracy = sens->accuracy;
return EINA_TRUE;
}
EAPI Eina_Bool
eeze_sensor_xyz_get(Eeze_Sensor_Obj *sens, float *x, float *y, float *z)
{
if (!sens) return EINA_FALSE;
*x = sens->data[0];
*y = sens->data[1];
*z = sens->data[2];
return EINA_TRUE;
}
EAPI Eina_Bool
eeze_sensor_xy_get(Eeze_Sensor_Obj *sens, float *x, float *y)
{
if (!sens) return EINA_FALSE;
*x = sens->data[0];
*y = sens->data[1];
return EINA_TRUE;
}
EAPI Eina_Bool
eeze_sensor_x_get(Eeze_Sensor_Obj *sens, float *x)
{
if (!sens) return EINA_FALSE;
*x = sens->data[0];
return EINA_TRUE;
}
EAPI Eina_Bool
eeze_sensor_timestamp_get(Eeze_Sensor_Obj *sens, double *timestamp)
{
if (!sens) return EINA_FALSE;
*timestamp = sens->timestamp;
return EINA_TRUE;
}
/* Synchronous read. Blocked until the data was readout from the hardware
* sensor
*/
EAPI Eina_Bool
eeze_sensor_read(Eeze_Sensor_Obj *sens)
{
Eeze_Sensor_Module *module = NULL;
if (!sens) return EINA_FALSE;
module = _highest_priority_module_get();
if (!module) return EINA_FALSE;
if (module->read)
return module->read(sens);
return EINA_FALSE;
}
/* Asynchronous read. Schedule a new read out that will update the cached values
* as soon as it arrives.
*/
EAPI Eina_Bool
eeze_sensor_async_read(Eeze_Sensor_Obj *sens, void *user_data)
{
Eeze_Sensor_Module *module = NULL;
if (!sens) return EINA_FALSE;
module = _highest_priority_module_get();
if (!module) return EINA_FALSE;
if (module->async_read)
return module->async_read(sens, user_data);
return EINA_FALSE;
}
void
eeze_sensor_shutdown(void)
{
ecore_event_type_flush(EEZE_SENSOR_EVENT_ACCELEROMETER,
EEZE_SENSOR_EVENT_GRAVITY,
EEZE_SENSOR_EVENT_LINEAR_ACCELERATION,
EEZE_SENSOR_EVENT_DEVICE_ORIENTATION,
EEZE_SENSOR_EVENT_MAGNETIC,
EEZE_SENSOR_EVENT_ORIENTATION,
EEZE_SENSOR_EVENT_GYROSCOPE,
EEZE_SENSOR_EVENT_LIGHT,
EEZE_SENSOR_EVENT_PROXIMITY,
EEZE_SENSOR_EVENT_SNAP,
EEZE_SENSOR_EVENT_SHAKE,
EEZE_SENSOR_EVENT_DOUBLETAP,
EEZE_SENSOR_EVENT_PANNING,
EEZE_SENSOR_EVENT_PANNING_BROWSE,
EEZE_SENSOR_EVENT_TILT,
EEZE_SENSOR_EVENT_FACEDOWN,
EEZE_SENSOR_EVENT_DIRECT_CALL,
EEZE_SENSOR_EVENT_SMART_ALERT,
EEZE_SENSOR_EVENT_NO_MOVE,
EEZE_SENSOR_EVENT_BAROMETER,
EEZE_SENSOR_EVENT_TEMPERATURE);
eeze_sensor_modules_unload();
eina_hash_free(g_handle->modules);
g_handle->modules = NULL;
free(g_handle);
g_handle = NULL;
eina_prefix_free(pfx);
pfx = NULL;
eina_shutdown();
}
Eina_Bool
eeze_sensor_init(void)
{
if (!eina_init()) return EINA_FALSE;
pfx = eina_prefix_new(NULL, eeze_sensor_init, "EEZE", "eeze", "checkme",
PACKAGE_BIN_DIR, PACKAGE_LIB_DIR,
PACKAGE_DATA_DIR, PACKAGE_DATA_DIR);
g_handle = calloc(1, sizeof(Eeze_Sensor));
if (!g_handle) return EINA_FALSE;
g_handle->modules_array = NULL;
g_handle->modules = eina_hash_string_small_new(NULL);
if (!g_handle->modules) return EINA_FALSE;
/* Make sure we create new ecore event types before using them */
EEZE_SENSOR_EVENT_ACCELEROMETER = ecore_event_type_new();
EEZE_SENSOR_EVENT_GRAVITY = ecore_event_type_new();
EEZE_SENSOR_EVENT_LINEAR_ACCELERATION = ecore_event_type_new();
EEZE_SENSOR_EVENT_DEVICE_ORIENTATION = ecore_event_type_new();
EEZE_SENSOR_EVENT_MAGNETIC = ecore_event_type_new();
EEZE_SENSOR_EVENT_ORIENTATION = ecore_event_type_new();
EEZE_SENSOR_EVENT_GYROSCOPE = ecore_event_type_new();
EEZE_SENSOR_EVENT_LIGHT = ecore_event_type_new();
EEZE_SENSOR_EVENT_PROXIMITY = ecore_event_type_new();
EEZE_SENSOR_EVENT_SNAP = ecore_event_type_new();
EEZE_SENSOR_EVENT_SHAKE = ecore_event_type_new();
EEZE_SENSOR_EVENT_DOUBLETAP = ecore_event_type_new();
EEZE_SENSOR_EVENT_PANNING = ecore_event_type_new();
EEZE_SENSOR_EVENT_PANNING_BROWSE = ecore_event_type_new();
EEZE_SENSOR_EVENT_TILT = ecore_event_type_new();
EEZE_SENSOR_EVENT_FACEDOWN = ecore_event_type_new();
EEZE_SENSOR_EVENT_DIRECT_CALL = ecore_event_type_new();
EEZE_SENSOR_EVENT_SMART_ALERT = ecore_event_type_new();
EEZE_SENSOR_EVENT_NO_MOVE = ecore_event_type_new();
EEZE_SENSOR_EVENT_BAROMETER = ecore_event_type_new();
EEZE_SENSOR_EVENT_TEMPERATURE = ecore_event_type_new();
/* Core is ready so we can load the modules from disk now */
eeze_sensor_modules_load();
return EINA_TRUE;
}