2015-11-18 16:06:16 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Freie Universität Berlin
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @defgroup sys_phydat Phydat
|
|
|
|
* @ingroup sys
|
|
|
|
* @brief Generic data container for physical data and utility functions
|
|
|
|
*
|
|
|
|
* The purpose of this module is to introduce a common view on physical data
|
|
|
|
* throughout RIOT. This data is typically the output from sensor readings, data
|
|
|
|
* aggregation, and also the input for actuators.
|
|
|
|
*
|
|
|
|
* The idea is to enable different sensor/actuator drivers and other RIOT
|
|
|
|
* modules to exchange and have the same view on this kind of data. Labeling
|
|
|
|
* data with a unit type it's scaling makes it possible to pipe data between
|
|
|
|
* modules in an automated fashion without the need of specialized software
|
|
|
|
* wrappers and/or data normalization modules.
|
|
|
|
*
|
|
|
|
* @todo It might make sense to introduce additional data types for
|
|
|
|
* increased precision, i.e. something like phydat_float_t...
|
|
|
|
*
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Generic data container for physical data interface
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*/
|
|
|
|
|
2017-05-23 18:19:52 +02:00
|
|
|
#ifndef PHYDAT_H
|
|
|
|
#define PHYDAT_H
|
2015-11-18 16:06:16 +01:00
|
|
|
|
2017-06-26 14:44:38 +02:00
|
|
|
#include <stddef.h>
|
2015-11-18 16:06:16 +01:00
|
|
|
#include <stdint.h>
|
2019-07-18 15:16:43 +02:00
|
|
|
#include "kernel_defines.h"
|
2015-11-18 16:06:16 +01:00
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief The fixed number of dimensions we work with
|
|
|
|
*
|
|
|
|
* We use a fixed number of 3 dimensions, as many physical values we encounter
|
|
|
|
* can be expressed this way. In practice we have e.g. readings from
|
|
|
|
* accelerometers, gyros, color sensors, or set data for RGB LEDs.
|
|
|
|
*
|
|
|
|
* When expressing 1-dimensional data we just ignore the 2 higher dimension.
|
|
|
|
* This leads to a slight overhead of some byte of memory - but we benefit from
|
|
|
|
* a unified data structure for passing around physical data.
|
|
|
|
*/
|
|
|
|
#define PHYDAT_DIM (3U)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief The maximum length of a scaling string
|
|
|
|
*/
|
|
|
|
#define PHYDAT_SCALE_STR_MAXLEN (sizeof("*E-128\0"))
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Definition of physical units and comparable data types
|
|
|
|
*
|
|
|
|
* This list should contain all needed physical units (e.g. SI units), but also
|
|
|
|
* non-physical units that can be used to define the type of data passed around.
|
|
|
|
* This can be for example BOOL or aggregate values. As rule of thumb, the unit
|
|
|
|
* list can contain anything that helps two modules automatically negotiate, if
|
|
|
|
* they can understand each other.
|
|
|
|
*
|
|
|
|
* @note Extent this list as needed.
|
|
|
|
*/
|
|
|
|
enum {
|
|
|
|
/* generic values */
|
|
|
|
UNIT_UNDEF, /**< unit undefined */
|
|
|
|
UNIT_NONE, /**< data has no physical unit */
|
|
|
|
/* temperature */
|
|
|
|
UNIT_TEMP_C, /**< degree Celsius */
|
|
|
|
UNIT_TEMP_F, /**< degree Fahrenheit */
|
|
|
|
UNIT_TEMP_K, /**< Kelvin */
|
2016-06-05 16:38:32 +02:00
|
|
|
/* illuminance */
|
|
|
|
UNIT_LUX, /**< Lux (lx) */
|
2015-11-18 16:06:16 +01:00
|
|
|
/* dimension */
|
|
|
|
UNIT_M, /**< meters */
|
|
|
|
UNIT_M2, /**< square meters */
|
|
|
|
UNIT_M3, /**< cubic meters */
|
|
|
|
/* kinetic */
|
|
|
|
UNIT_G, /**< gravitational force */
|
|
|
|
UNIT_DPS, /**< degree per second */
|
|
|
|
/* weight */
|
|
|
|
UNIT_GR, /**< grams - not using the SI unit (kg) here to make scale
|
|
|
|
* handling simpler */
|
|
|
|
/* electricity */
|
|
|
|
UNIT_A, /**< Ampere */
|
|
|
|
UNIT_V, /**< Volts */
|
2019-11-05 12:37:41 +01:00
|
|
|
UNIT_W, /**< Watt */
|
2015-11-18 16:06:16 +01:00
|
|
|
UNIT_GS, /**< gauss */
|
2022-01-25 13:46:08 +01:00
|
|
|
UNIT_T, /**< Tesla */
|
2018-11-02 13:46:04 +01:00
|
|
|
UNIT_DBM, /**< decibel-milliwatts */
|
2018-07-31 13:42:17 +02:00
|
|
|
UNIT_COULOMB, /**< coulomb */
|
2019-02-15 13:33:46 +01:00
|
|
|
UNIT_F, /**< Farad */
|
2020-02-27 17:07:15 +01:00
|
|
|
UNIT_OHM, /**< Ohm */
|
2019-02-10 23:02:54 +01:00
|
|
|
/* electrochemical */
|
|
|
|
UNIT_PH, /**< pH */
|
2015-11-18 16:06:16 +01:00
|
|
|
/* pressure */
|
|
|
|
UNIT_BAR, /**< Beer? */
|
|
|
|
UNIT_PA, /**< Pascal */
|
|
|
|
/* light */
|
|
|
|
UNIT_CD, /**< Candela */
|
|
|
|
/* logical */
|
|
|
|
UNIT_BOOL, /**< boolean value [0|1] */
|
2017-06-20 11:03:56 +02:00
|
|
|
UNIT_CTS, /**< counts */
|
2015-11-18 16:06:16 +01:00
|
|
|
UNIT_PERCENT, /**< out of 100 */
|
|
|
|
UNIT_PERMILL, /**< out of 1000 */
|
|
|
|
UNIT_PPM, /**< part per million */
|
2018-06-11 15:46:46 +02:00
|
|
|
UNIT_PPB, /**< part per billion */
|
2015-11-18 16:06:16 +01:00
|
|
|
/* aggregate values */
|
|
|
|
UNIT_TIME, /**< the three dimensions contain sec, min, and hours */
|
2018-11-22 19:14:20 +01:00
|
|
|
UNIT_DATE, /**< the 3 dimensions contain days, months and years */
|
|
|
|
/* mass concentration */
|
2020-02-03 17:16:53 +01:00
|
|
|
UNIT_GPM3, /**< grams per cubic meter */
|
|
|
|
/* number concentration */
|
|
|
|
UNIT_CPM3 /**< count per cubic meter */
|
2015-11-18 16:06:16 +01:00
|
|
|
/* extend this list as needed */
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Generic data structure for expressing physical values
|
|
|
|
*
|
2019-10-23 21:16:22 +02:00
|
|
|
* Physical data is expressed in a 3-dimensional tuple of values. In addition
|
2015-12-01 14:32:37 +01:00
|
|
|
* to the data fields, this struct contains further the (physical) unit and the
|
2015-11-18 16:06:16 +01:00
|
|
|
* scale factor of the data. The unit is expressed as constant. The scale factor
|
|
|
|
* is expressed as power of 10 (10^factor).
|
|
|
|
*
|
|
|
|
* The combination of signed 16-bit numbers with and the scale factor gives us a
|
2015-12-01 14:32:37 +01:00
|
|
|
* very high dynamic range (from -32*10^-131 to 32*10^130). In a wider sense we
|
2015-11-18 16:06:16 +01:00
|
|
|
* are saving the values as fixed floating points...
|
|
|
|
*
|
|
|
|
* The scale factor is identical for all 3 values.
|
|
|
|
*
|
|
|
|
* In a traditional (scientific) computational system the obvious choice for the
|
|
|
|
* used data type would be to use floats. We are however on heavily resource
|
|
|
|
* constrained (even 8-bit) embedded systems, so we use int16_t here. As most
|
|
|
|
* sensor are in some way ADC based, they normally do not use a higher accuracy
|
|
|
|
* than 12-14bit, so using 16-bit integers to represent this data is good enough
|
|
|
|
* in most cases.
|
|
|
|
*/
|
|
|
|
typedef struct {
|
|
|
|
int16_t val[PHYDAT_DIM]; /**< the 3 generic dimensions of data */
|
|
|
|
uint8_t unit; /**< the (physical) unit of the data */
|
|
|
|
int8_t scale; /**< the scale factor, 10^*scale* */
|
|
|
|
} phydat_t;
|
|
|
|
|
2018-02-18 07:48:49 +01:00
|
|
|
/**
|
|
|
|
* @brief Minimum value for phydat_t::val
|
|
|
|
*/
|
|
|
|
#define PHYDAT_MIN (INT16_MIN)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Maximum value for phydat_t::val
|
|
|
|
*/
|
|
|
|
#define PHYDAT_MAX (INT16_MAX)
|
|
|
|
|
2015-11-18 16:06:16 +01:00
|
|
|
/**
|
|
|
|
* @brief Dump the given data container to STDIO
|
|
|
|
*
|
|
|
|
* @param[in] data data container to dump
|
|
|
|
* @param[in] dim number of dimension of @p data to dump
|
|
|
|
*/
|
|
|
|
void phydat_dump(phydat_t *data, uint8_t dim);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Convert the given unit to a string
|
|
|
|
*
|
|
|
|
* @param[in] unit unit to convert
|
|
|
|
*
|
|
|
|
* @return string representation of given unit (e.g. V or m)
|
|
|
|
* @return NULL if unit was not recognized
|
|
|
|
*/
|
|
|
|
const char *phydat_unit_to_str(uint8_t unit);
|
|
|
|
|
2017-06-26 14:44:38 +02:00
|
|
|
/**
|
|
|
|
* @brief Return a string representation for every unit, including
|
|
|
|
* non-physical units like 'none' or 'time'
|
|
|
|
*
|
|
|
|
* This function is useful when converting phydat_t structures to non-binary
|
|
|
|
* representations like JSON or XML.
|
|
|
|
*
|
|
|
|
* In practice, this function extends phydat_unit_to_str() with additional
|
|
|
|
* identifiers for non physical units.
|
|
|
|
*
|
|
|
|
* @param[in] unit unit to convert
|
|
|
|
*
|
|
|
|
* @return string representation of given unit
|
|
|
|
* @return empty string ("") if unit was not recognized
|
|
|
|
*/
|
|
|
|
const char *phydat_unit_to_str_verbose(uint8_t unit);
|
|
|
|
|
2015-11-18 16:06:16 +01:00
|
|
|
/**
|
2018-02-18 07:38:45 +01:00
|
|
|
* @brief Convert the given scale factor to an SI prefix
|
2015-11-18 16:06:16 +01:00
|
|
|
*
|
2018-02-18 07:38:45 +01:00
|
|
|
* The given scaling factor is returned as a SI unit prefix (e.g. M for Mega, u
|
|
|
|
* for micro, etc), or `\0` otherwise.
|
2015-11-18 16:06:16 +01:00
|
|
|
*
|
|
|
|
* @param[in] scale scale factor to convert
|
2018-02-18 07:38:45 +01:00
|
|
|
*
|
|
|
|
* @return SI prefix if applicable
|
|
|
|
* @return `\0` if no SI prefix was found
|
2015-11-18 16:06:16 +01:00
|
|
|
*/
|
2018-02-18 07:38:45 +01:00
|
|
|
char phydat_prefix_from_scale(int8_t scale);
|
2015-11-18 16:06:16 +01:00
|
|
|
|
2018-02-19 08:43:39 +01:00
|
|
|
/**
|
sys/phydat: New phydat_fit API
The current phydat_fit implementation the following limitations:
- The API is way more complicated to use than needed
- It doesn't perform any rounding
- It uses `long` in a place where actual width (or better range) of the type
is pretty important.
This commit addresses these limitations and uses lookup-tables to reduce the
number of divisions required.
Before this commit code using it looked like this:
``` C
long values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .scale = 42, .unit = UNIT_V };
phydat_fit(&dat, values[0], 0, phydat_fit(&dat, values[1], 1, phydat_fit(&dat, values[2], 2, 0)));
```
Now it can be used like this:
``` C
int32_t values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .unit = UNIT_V, .scale = 42 };
phydat_fit(&dat, values, 3);
```
2018-10-24 13:04:09 +02:00
|
|
|
* @brief Scale integer value(s) to fit into a @ref phydat_t
|
2018-02-19 08:43:39 +01:00
|
|
|
*
|
sys/phydat: New phydat_fit API
The current phydat_fit implementation the following limitations:
- The API is way more complicated to use than needed
- It doesn't perform any rounding
- It uses `long` in a place where actual width (or better range) of the type
is pretty important.
This commit addresses these limitations and uses lookup-tables to reduce the
number of divisions required.
Before this commit code using it looked like this:
``` C
long values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .scale = 42, .unit = UNIT_V };
phydat_fit(&dat, values[0], 0, phydat_fit(&dat, values[1], 1, phydat_fit(&dat, values[2], 2, 0)));
```
Now it can be used like this:
``` C
int32_t values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .unit = UNIT_V, .scale = 42 };
phydat_fit(&dat, values, 3);
```
2018-10-24 13:04:09 +02:00
|
|
|
* Inserts the @p values in the given @p dat so that all @p dim values in
|
|
|
|
* @p values fit inside the limits of the data type,
|
|
|
|
* [@ref PHYDAT_MIN, @ref PHYDAT_MAX], and updates the stored scale factor.
|
2021-03-04 21:25:34 +01:00
|
|
|
* The @ref phydat_t::scale member in @p dat is used as the the original scale
|
|
|
|
* of the @p values.
|
sys/phydat: New phydat_fit API
The current phydat_fit implementation the following limitations:
- The API is way more complicated to use than needed
- It doesn't perform any rounding
- It uses `long` in a place where actual width (or better range) of the type
is pretty important.
This commit addresses these limitations and uses lookup-tables to reduce the
number of divisions required.
Before this commit code using it looked like this:
``` C
long values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .scale = 42, .unit = UNIT_V };
phydat_fit(&dat, values[0], 0, phydat_fit(&dat, values[1], 1, phydat_fit(&dat, values[2], 2, 0)));
```
Now it can be used like this:
``` C
int32_t values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .unit = UNIT_V, .scale = 42 };
phydat_fit(&dat, values, 3);
```
2018-10-24 13:04:09 +02:00
|
|
|
* The value is rounded to the nearest integer if possible, otherwise away from
|
|
|
|
* zero. E.g. `0.5` and `0.6` are rounded to `1`, `0.4` and `-0.4` are rounded
|
|
|
|
* to `0`, `-0.5` and `-0.6` are rounded to `-1`.
|
2018-02-19 08:43:39 +01:00
|
|
|
*
|
|
|
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.c}
|
sys/phydat: New phydat_fit API
The current phydat_fit implementation the following limitations:
- The API is way more complicated to use than needed
- It doesn't perform any rounding
- It uses `long` in a place where actual width (or better range) of the type
is pretty important.
This commit addresses these limitations and uses lookup-tables to reduce the
number of divisions required.
Before this commit code using it looked like this:
``` C
long values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .scale = 42, .unit = UNIT_V };
phydat_fit(&dat, values[0], 0, phydat_fit(&dat, values[1], 1, phydat_fit(&dat, values[2], 2, 0)));
```
Now it can be used like this:
``` C
int32_t values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .unit = UNIT_V, .scale = 42 };
phydat_fit(&dat, values, 3);
```
2018-10-24 13:04:09 +02:00
|
|
|
* int32_t values[] = { 100000, 2000000, 30000000 };
|
|
|
|
* phydat_t dat = { .scale = 0 };
|
|
|
|
* phydat_fit(&dat, values, 3);
|
2018-02-19 08:43:39 +01:00
|
|
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
*
|
sys/phydat: New phydat_fit API
The current phydat_fit implementation the following limitations:
- The API is way more complicated to use than needed
- It doesn't perform any rounding
- It uses `long` in a place where actual width (or better range) of the type
is pretty important.
This commit addresses these limitations and uses lookup-tables to reduce the
number of divisions required.
Before this commit code using it looked like this:
``` C
long values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .scale = 42, .unit = UNIT_V };
phydat_fit(&dat, values[0], 0, phydat_fit(&dat, values[1], 1, phydat_fit(&dat, values[2], 2, 0)));
```
Now it can be used like this:
``` C
int32_t values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .unit = UNIT_V, .scale = 42 };
phydat_fit(&dat, values, 3);
```
2018-10-24 13:04:09 +02:00
|
|
|
* @note Unless compiled with `-DPHYDAT_FIT_TRADE_PRECISION_FOR_ROM=0`, this
|
|
|
|
* function will scale the value `-32768`, even though it would fit into a
|
|
|
|
* @ref phydat_t. Statistically, this precision loss happens in 0.00153%
|
|
|
|
* of the calls. This optimization saves a bit more than 20 bytes.
|
2018-02-19 08:43:39 +01:00
|
|
|
*
|
2021-03-04 21:25:34 +01:00
|
|
|
* @pre The @ref phydat_t::scale member in @p dat is initialized to the scale
|
|
|
|
* of the @p values by the caller prior to calling this function.
|
2018-02-19 08:43:39 +01:00
|
|
|
*
|
sys/phydat: New phydat_fit API
The current phydat_fit implementation the following limitations:
- The API is way more complicated to use than needed
- It doesn't perform any rounding
- It uses `long` in a place where actual width (or better range) of the type
is pretty important.
This commit addresses these limitations and uses lookup-tables to reduce the
number of divisions required.
Before this commit code using it looked like this:
``` C
long values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .scale = 42, .unit = UNIT_V };
phydat_fit(&dat, values[0], 0, phydat_fit(&dat, values[1], 1, phydat_fit(&dat, values[2], 2, 0)));
```
Now it can be used like this:
``` C
int32_t values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .unit = UNIT_V, .scale = 42 };
phydat_fit(&dat, values, 3);
```
2018-10-24 13:04:09 +02:00
|
|
|
* @param[in, out] dat the value will be written into this data array
|
|
|
|
* @param[in] values value(s) to rescale
|
|
|
|
* @param[in] dim Number of elements in @p values
|
2018-02-19 08:43:39 +01:00
|
|
|
*/
|
sys/phydat: New phydat_fit API
The current phydat_fit implementation the following limitations:
- The API is way more complicated to use than needed
- It doesn't perform any rounding
- It uses `long` in a place where actual width (or better range) of the type
is pretty important.
This commit addresses these limitations and uses lookup-tables to reduce the
number of divisions required.
Before this commit code using it looked like this:
``` C
long values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .scale = 42, .unit = UNIT_V };
phydat_fit(&dat, values[0], 0, phydat_fit(&dat, values[1], 1, phydat_fit(&dat, values[2], 2, 0)));
```
Now it can be used like this:
``` C
int32_t values[] = { 100000, 2000000, 30000000 };
phydat_t dat = { .unit = UNIT_V, .scale = 42 };
phydat_fit(&dat, values, 3);
```
2018-10-24 13:04:09 +02:00
|
|
|
void phydat_fit(phydat_t *dat, const int32_t *values, unsigned int dim);
|
2018-02-19 08:43:39 +01:00
|
|
|
|
2017-06-29 12:53:58 +02:00
|
|
|
/**
|
|
|
|
* @brief Convert the given phydat_t structure into a JSON string
|
|
|
|
*
|
|
|
|
* The output string written to @p buf will be `\0` terminated. You must make
|
|
|
|
* sure, that the given @p buf is large enough to hold the resulting string. You
|
|
|
|
* can call the function with `@p buf := NULL` to simply calculate the size of
|
|
|
|
* the JSON string without writing anything.
|
|
|
|
*
|
|
|
|
* The formatted JSON string will have the following format:
|
|
|
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.json}
|
|
|
|
* // case (dim == 1):
|
|
|
|
* {
|
|
|
|
* "d": 21.45,
|
|
|
|
* "u": "°C"
|
|
|
|
* }
|
|
|
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
*
|
|
|
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.json}
|
|
|
|
* // case (dim > 1), dim := 3 in this case:
|
|
|
|
* {
|
|
|
|
* "d": [1.02, 0.23, -0.81],
|
|
|
|
* "u": "g"
|
|
|
|
* }
|
|
|
|
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
*
|
|
|
|
* The data will be encoded as fixed point number based on the given scale
|
|
|
|
* factor.
|
|
|
|
*
|
|
|
|
* For encoding the unit, this function uses the extended
|
|
|
|
* phydat_unit_to_str_verbose() function to also print units for non-SI types,
|
|
|
|
* e.g. it will produce `..."u":"date"}` for @ref UNIT_DATE or `..."u":"none"}`
|
|
|
|
* for @ref UNIT_NONE.
|
|
|
|
*
|
|
|
|
* @param[in] data data to encode
|
2022-05-18 17:36:54 +02:00
|
|
|
* @param[in] dim dimensions used in @p data, MUST be > 0 and <= PHYDAT_DIM
|
2017-06-29 12:53:58 +02:00
|
|
|
* @param[out] buf target buffer for the JSON string, or NULL
|
|
|
|
*
|
|
|
|
* @pre @p dim > 0
|
2022-05-18 17:36:54 +02:00
|
|
|
* @pre @p dim <= PHYDAT_DIM
|
2017-06-29 12:53:58 +02:00
|
|
|
*
|
|
|
|
* @return number of bytes (potentially) written to @p buf, including `\0`
|
|
|
|
* terminator
|
|
|
|
*/
|
|
|
|
size_t phydat_to_json(const phydat_t *data, size_t dim, char *buf);
|
|
|
|
|
2015-11-18 16:06:16 +01:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-05-23 18:19:52 +02:00
|
|
|
#endif /* PHYDAT_H */
|
2015-11-18 16:06:16 +01:00
|
|
|
/** @} */
|