2021-11-07 23:40:37 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 Gunar Schorcht
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup tests
|
|
|
|
* @brief Test application for the L3Gxxxx 3-axis gyroscope sensor family driver
|
|
|
|
* @author Gunar Schorcht <gunar@schorcht.net>
|
|
|
|
* @file
|
|
|
|
*
|
|
|
|
* # About
|
|
|
|
*
|
|
|
|
* This is a manual test application for the driver of the L3GXXXX 3-axis
|
|
|
|
* gyroscope sensor family. It demonstrates the use of a L3Gxxxx sensor.
|
|
|
|
* The different functionalities of the driver are enabled and tested by
|
|
|
|
* using different modules.
|
|
|
|
*
|
|
|
|
* # Usage
|
|
|
|
*
|
|
|
|
* The used sensor must be specified by setting the variable DRIVER to
|
|
|
|
* `l3gd20h`, `l3gd20`, `l3g4200d_ng`, `i3g4250d` or `a3g4250d` which enables
|
|
|
|
* the corresponding pseudomodule:
|
|
|
|
* ```
|
2023-03-30 16:12:49 +02:00
|
|
|
* DRIVER=l3gd20h BOARD=... make -C tests/drivers/l3gxxxx
|
2021-11-07 23:40:37 +01:00
|
|
|
* ```
|
|
|
|
* If the DRIVER variable is not set, `l3gd20h` is used.
|
|
|
|
*
|
|
|
|
* By default, the application will initialize the gyroscope with the default
|
|
|
|
* configuration parameter set as defined in the `l3gxxxx_params.h` file:
|
|
|
|
*
|
|
|
|
* Configuration Parameter | Default value
|
|
|
|
* :---------------------------|:----------------------
|
|
|
|
* Output data rate | 100 Hz
|
|
|
|
* LPF2 cutoff frequency | 25 Hz
|
|
|
|
* Full scale | 245 dps
|
|
|
|
* Filter selection | HPF and LPF2 are used
|
|
|
|
* HPF mode | Normal with autoreset
|
|
|
|
* HPF cutoff frequency | 8 Hz
|
|
|
|
*
|
|
|
|
* The default configuration uses the I2C interface (module `l3gxxxx_i2c`)
|
|
|
|
* of the sensor and polls the data periodically every 200 ms. If the SPI
|
|
|
|
* interface of the sensor should be used, the module `l3gxxxx_spi` has to
|
|
|
|
* be used:
|
|
|
|
* ```
|
2023-03-30 16:12:49 +02:00
|
|
|
* DRIVER=l3gd20h USEMODULE=l3gxxxx_spi BOARD=... make -C tests/drivers/l3gxxxx
|
2021-11-07 23:40:37 +01:00
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* The application is configurable by using different pseudomodules to test
|
|
|
|
* most of the functionalities of the L3Gxxxx driver.
|
|
|
|
*
|
|
|
|
* - Module `l3gxxxx_irq_drdy`:
|
|
|
|
* Instead of polling the sensor regularly for data, the data ready interrupt
|
|
|
|
* on signal `INT2/DRDY` is used to read the data. If module `l3gxxxx_fifo`
|
|
|
|
* is used at the same time, the FIFO status interrupts are enabled instead
|
|
|
|
* of the data ready interrupt. The GPIO pin used for the interrupt signal
|
|
|
|
* can be overridden, for example:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* DRIVER=l3gd20h USEMODULE=l3gxxxx_irq_drdy \
|
2023-03-30 16:12:49 +02:00
|
|
|
* CFLAGS='-DL3GXXXX_INT2_PIN=GPIO_PIN\(0,5\) BOARD=... make -C tests/drivers/l3gxxxx
|
2021-11-07 23:40:37 +01:00
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* - Module `l3gxxxx_irq_event`:
|
|
|
|
* Event interrupt generation is configured according to default
|
|
|
|
* configuration parameters in the `l3gxxxx_params.h` file to generate
|
|
|
|
* interrupts on signal `INT1` when the absolute value of the angular rate
|
|
|
|
* of any axis exceeds the default threshold. The GPIO pin used for the
|
|
|
|
* interrupt signal can be overridden, for example:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* DRIVER=l3gd20h USEMODULE=l3gxxxx_irq_event \
|
2023-03-30 16:12:49 +02:00
|
|
|
* CFLAGS='-DL3GXXXX_INT1_PIN=GPIO_PIN\(0,6\) BOARD=... make -C tests/drivers/l3gxxxx
|
2021-11-07 23:40:37 +01:00
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* - Module `l3gxxxx_fifo`:
|
|
|
|
* Instead of fetching single data samples from the sensor, the FIFO of
|
|
|
|
* the sensor is enabled. Depending on the use of module `l3gxxxx_irq_drdy`,
|
|
|
|
* the FIFO state is either polled regularly or FIFO status interrupts on
|
|
|
|
* signal `INT2/DRDY` are used. All available data samples are read from
|
|
|
|
* the FIFO with a single burst access.
|
|
|
|
*
|
|
|
|
* - Module `l3gxxxx_config`:
|
|
|
|
* The configuration of the event interrupt generation is changed at
|
|
|
|
* runtime.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#include "thread.h"
|
|
|
|
#include "ztimer.h"
|
|
|
|
|
|
|
|
#include "l3gxxxx.h"
|
|
|
|
#include "l3gxxxx_params.h"
|
|
|
|
|
|
|
|
#define TESTS_DRIVER_L3GXXXX_SLEEP (200) /* 200 ms */
|
|
|
|
|
|
|
|
#if IS_USED(MODULE_L3GXXXX_IRQ)
|
|
|
|
int handle_event(const l3gxxxx_t *dev, l3gxxxx_int_event_src_t source)
|
|
|
|
{
|
|
|
|
(void)dev;
|
|
|
|
|
|
|
|
/* display the event */
|
|
|
|
if (!source.active) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (source.x_low) {
|
|
|
|
printf("|x| below ");
|
|
|
|
}
|
|
|
|
if (source.y_low) {
|
|
|
|
printf("|y| below ");
|
|
|
|
}
|
|
|
|
if (source.z_low) {
|
|
|
|
printf("|z| below ");
|
|
|
|
}
|
|
|
|
if (source.x_high) {
|
|
|
|
printf("|x| above ");
|
|
|
|
}
|
|
|
|
if (source.y_high) {
|
|
|
|
printf("|y| above ");
|
|
|
|
}
|
|
|
|
if (source.z_high) {
|
|
|
|
printf("|z| above ");
|
|
|
|
}
|
|
|
|
printf("defined thresholds: ");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif /* IS_USED(MODULE_L3GXXXX_IRQ) */
|
|
|
|
|
|
|
|
int read_data(const l3gxxxx_t *dev)
|
|
|
|
{
|
|
|
|
#if IS_USED(MODULE_L3GXXXX_FIFO)
|
|
|
|
|
|
|
|
l3gxxxx_data_fifo_t data;
|
|
|
|
|
|
|
|
int num = l3gxxxx_read_fifo(dev, data);
|
|
|
|
|
|
|
|
printf("FIFO samples num=%d\n", num);
|
|
|
|
for (int i = 0; i < num; i++) {
|
|
|
|
printf("gyro [dps] x: %+5" PRIi32 ", y: %+5" PRIi32 ", z: %+5" PRIi32 "\n",
|
|
|
|
data[i].x / 1000, data[i].y / 1000, data[i].z / 1000);
|
|
|
|
}
|
|
|
|
|
|
|
|
return num;
|
|
|
|
|
|
|
|
#else /* IS_USED(MODULE_L3GXXXX_FIFO) */
|
|
|
|
|
|
|
|
l3gxxxx_data_t data;
|
|
|
|
l3gxxxx_read(dev, &data);
|
|
|
|
|
|
|
|
printf("gyro [dps] x: %+5" PRIi32 ", y: %+5" PRIi32 ", z: %+5" PRIi32 "\n",
|
|
|
|
data.x / 1000, data.y / 1000, data.z / 1000);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
#endif /* IS_USED(MODULE_L3GXXXX_FIFO) */
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
l3gxxxx_t dev;
|
|
|
|
|
|
|
|
puts("L3GXXXX gyroscope driver test application\n");
|
|
|
|
puts("Initializing L3GXXXX sensor");
|
|
|
|
|
|
|
|
/* initialize the sensor with default configuration parameters */
|
|
|
|
if (l3gxxxx_init(&dev, &l3gxxxx_params[0]) == L3GXXXX_OK) {
|
|
|
|
puts("[OK]\n");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
puts("[Failed]");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if IS_USED(MODULE_L3GXXXX_CONFIG)
|
|
|
|
/* select LPF2 and HPF, configure HPF and reset REFERENCE by dummy read */
|
|
|
|
int8_t ref;
|
|
|
|
l3gxxxx_select_output_filter(&dev, L3GXXXX_HPF_ONLY);
|
|
|
|
l3gxxxx_config_hpf(&dev, L3GXXXX_HPF_NORMAL, 0);
|
|
|
|
l3gxxxx_get_hpf_ref(&dev, &ref);
|
|
|
|
#endif /* IS_USED(MODULE_L3GXXXX_CONFIG) */
|
|
|
|
|
|
|
|
#if IS_USED(MODULE_L3GXXXX_FIFO) && IS_USED(MODULE_L3GXXXX_CONFIG)
|
|
|
|
/* clear FIFO and activate FIFO Stream mode */
|
|
|
|
l3gxxxx_set_fifo_mode(&dev, L3GXXXX_BYPASS, 0);
|
|
|
|
l3gxxxx_set_fifo_mode(&dev, L3GXXXX_STREAM, 10);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if IS_USED(MODULE_L3GXXXX_IRQ_EVENT) && IS_USED(MODULE_L3GXXXX_CONFIG)
|
|
|
|
/* enable event interrupts (axis movement and wake-up) */
|
|
|
|
l3gxxxx_int_event_cfg_t int_cfg;
|
|
|
|
|
|
|
|
l3gxxxx_get_int_event_cfg(&dev, &int_cfg);
|
|
|
|
|
|
|
|
int_cfg.x_high_enabled = true;
|
|
|
|
int_cfg.y_high_enabled = true;
|
|
|
|
int_cfg.z_high_enabled = true;
|
|
|
|
int_cfg.x_low_enabled = false;
|
|
|
|
int_cfg.y_low_enabled = false;
|
|
|
|
int_cfg.z_low_enabled = false;
|
|
|
|
int_cfg.x_threshold = 1000;
|
|
|
|
int_cfg.y_threshold = 1000;
|
|
|
|
int_cfg.z_threshold = 1000;
|
|
|
|
|
|
|
|
int_cfg.filter = L3GXXXX_HPF_ONLY;
|
|
|
|
int_cfg.and_or = false;
|
|
|
|
int_cfg.duration = 0;
|
|
|
|
int_cfg.latch = true;
|
|
|
|
|
|
|
|
l3gxxxx_set_int_event_cfg(&dev, &int_cfg);
|
|
|
|
#endif /* IS_USED(MODULE_L3GXXXX_CONFIG) && IS_USED(MODULE_L3GXXXX_IRQ_EVENT) */
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
|
|
|
|
#if IS_USED(MODULE_L3GXXXX_IRQ)
|
|
|
|
/* wait for next interrupt */
|
|
|
|
l3gxxxx_int_src_t int_src = l3gxxxx_wait_int(&dev);
|
|
|
|
(void)int_src;
|
|
|
|
|
|
|
|
/* in case of event we determine the source */
|
|
|
|
if (int_src.event.active) {
|
|
|
|
handle_event(&dev, int_src.event);
|
|
|
|
}
|
|
|
|
/* read and print data in any case */
|
|
|
|
read_data(&dev);
|
|
|
|
#else
|
|
|
|
/* read and print data */
|
|
|
|
read_data(&dev);
|
|
|
|
ztimer_sleep(ZTIMER_MSEC, TESTS_DRIVER_L3GXXXX_SLEEP);
|
|
|
|
#endif /* IS_USED(MODULE_L3GXXXX_IRQ) */
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|