Compare commits
No commits in common. "7e8652c3b19c092cf34c40e06c52452adeb6c977" and "7ec153d397754d383f6b71b26376395c3d9b57cf" have entirely different histories.
7e8652c3b1
...
7ec153d397
114
README.md
114
README.md
@ -7,7 +7,7 @@ A C89 driver for the 3-axis accelerometer LIS3DH. Supports both i2c and SPI.
|
||||
> - HP filter (4 c/o freq)
|
||||
> - 2G, 4G, 8G and 16G
|
||||
> - All power modes
|
||||
> - Interrupt generation (partial)
|
||||
> - Interrupt generation (soon)
|
||||
> - Free-fall detection (soon)
|
||||
> - Single and double click detection (soon)
|
||||
> - 4D/6D orientation detection (soon)
|
||||
@ -152,7 +152,7 @@ x: 0.516000, y: -0.852000, z: -0.112000
|
||||
Instead of polling for every single [x y z] set, a FIFO with programmable capacity ("watermark") can be used like such:
|
||||
|
||||
All FIFO readings use 10-bit resolution regardless of the mode set in `lis.cfg.mode`.
|
||||
The watermark level can also be adjusted to a value [0-31] inclusive by modifying the `lis.cfg.fifo.fth` property before calling configure().
|
||||
The watermark level can also be adjusted to a value [0-32] inclusive by modifying the `lis.cfg.fifo.fth` property before calling configure().
|
||||
```c
|
||||
#define _GNU_SOURCE
|
||||
#include <unistd.h>
|
||||
@ -333,116 +333,6 @@ x: -0.016000, y: -0.032000, z: -0.040000
|
||||
x: -0.016000, y: -0.032000, z: -0.040000
|
||||
x: -0.008000, y: -0.024000, z: -0.008000
|
||||
```
|
||||
## Using interrupts
|
||||
|
||||
The LIS3DH supports two different interrupt "output pins," `INT1` and `INT2`. The appropriate flag must be set in either `cfg.int1` or `cfg.int2` (only one of such flags can be set at a time!) and the interrupt source must be configured to trigger into `INT1` or `INT2`. Below is example code that listens and receives an interrupt when the FIFO watermark is reached i.e. it is full.
|
||||
|
||||
```c
|
||||
#define _GNU_SOURCE
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
#include <math.h>
|
||||
#include "lis3dh.h"
|
||||
#include "interrupt.h"
|
||||
#include "i2c.h"
|
||||
|
||||
/* GPIO 12 or Pin 32 */
|
||||
#define GPIO_INTERRUPT_PIN 12
|
||||
|
||||
int main() {
|
||||
|
||||
lis3dh_t lis;
|
||||
struct lis3dh_fifo_data fifo;
|
||||
int k;
|
||||
|
||||
lis.dev.init = i2c_init;
|
||||
lis.dev.read = i2c_read;
|
||||
lis.dev.write = i2c_write;
|
||||
lis.dev.sleep = usleep;
|
||||
lis.dev.deinit = i2c_deinit;
|
||||
|
||||
if (lis3dh_init(&lis)) {
|
||||
/* error handling */
|
||||
}
|
||||
|
||||
/* register interrupt */
|
||||
if (int_register(GPIO_INTERRUPT_PIN)) {
|
||||
/* error handling */
|
||||
}
|
||||
|
||||
lis.cfg.mode = LIS3DH_MODE_NORMAL;
|
||||
lis.cfg.range = LIS3DH_FS_2G;
|
||||
lis.cfg.rate = LIS3DH_ODR_100_HZ;
|
||||
lis.cfg.fifo.mode = LIS3DH_FIFO_MODE_STREAM;
|
||||
lis.cfg.fifo.trig = LIS3DH_FIFO_TRIG_INT1; /* trigger into INT1 */
|
||||
lis.cfg.int1.wtm = 1; /* trigger upon watermark level reached */
|
||||
|
||||
if (lis3dh_configure(&lis)) {
|
||||
/* error handling */
|
||||
}
|
||||
|
||||
/* wait for interrupt from LIS3DH */
|
||||
if (int_poll(GPIO_INTERRUPT_PIN)) {
|
||||
/* error handling */
|
||||
}
|
||||
|
||||
if (lis3dh_read_fifo(&lis, &fifo)) {
|
||||
/* error handling */
|
||||
}
|
||||
|
||||
for(k=0; k<fifo.size; k++) {
|
||||
printf("%04.04f %04.04f %04.04f %04.04f\n", fifo.x[k], fifo.y[k], fifo.z[k]);
|
||||
}
|
||||
|
||||
/* unregister interrupt */
|
||||
if (int_unregister(GPIO_INTERRUPT_PIN)) {
|
||||
/* error handling */
|
||||
}
|
||||
|
||||
if (lis3dh_deinit(&lis)) {
|
||||
/* error handling */
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
```
|
||||
|
||||
Output:
|
||||
```
|
||||
$ ./lis3dh
|
||||
0.2040 -1.0120 -0.1720 1.0466
|
||||
0.2200 -1.0200 -0.1600 1.0557
|
||||
0.2160 -1.0200 -0.1600 1.0548
|
||||
0.2120 -1.0240 -0.1600 1.0579
|
||||
0.2200 -1.0160 -0.1760 1.0543
|
||||
0.2080 -0.9960 -0.1720 1.0319
|
||||
0.2080 -0.9960 -0.1760 1.0326
|
||||
0.2200 -1.0200 -0.1600 1.0557
|
||||
0.2200 -1.0160 -0.1560 1.0512
|
||||
0.2160 -1.0200 -0.1600 1.0548
|
||||
0.2120 -1.0240 -0.1520 1.0567
|
||||
0.2200 -1.0240 -0.1520 1.0583
|
||||
0.2160 -1.0200 -0.1520 1.0536
|
||||
0.2160 -1.0200 -0.1560 1.0542
|
||||
0.2080 -0.9960 -0.1760 1.0326
|
||||
0.2200 -1.0240 -0.1600 1.0595
|
||||
0.2120 -1.0000 -0.1720 1.0366
|
||||
0.2120 -0.9960 -0.1760 1.0334
|
||||
0.2200 -1.0240 -0.1600 1.0595
|
||||
0.2200 -1.0200 -0.1600 1.0557
|
||||
0.2080 -0.9960 -0.1640 1.0306
|
||||
0.1920 -1.0080 -0.1600 1.0385
|
||||
0.2080 -1.0080 -0.1600 1.0416
|
||||
0.2200 -1.0240 -0.1520 1.0583
|
||||
0.2080 -1.0000 -0.1720 1.0358
|
||||
0.2080 -0.9960 -0.1480 1.0282
|
||||
0.2040 -1.0240 -0.1560 1.0557
|
||||
0.2200 -1.0240 -0.1560 1.0589
|
||||
0.2120 -1.0040 -0.1520 1.0373
|
||||
0.2120 -1.0200 -0.1560 1.0534
|
||||
0.2200 -1.0240 -0.1560 1.0589
|
||||
```
|
||||
|
||||
### Using i2c on STM32
|
||||
Simple example code
|
||||
|
56
lis3dh.c
56
lis3dh.c
@ -58,7 +58,7 @@ int lis3dh_init(lis3dh_t *lis3dh) {
|
||||
|
||||
lis3dh->cfg.fifo.mode = 0xFF; /* in use if neq 0xFF */
|
||||
lis3dh->cfg.fifo.trig = 0;
|
||||
lis3dh->cfg.fifo.fth = 31; /* default watermark level. */
|
||||
lis3dh->cfg.fifo.fth = 32; /* default watermark level. */
|
||||
|
||||
lis3dh->cfg.filter.mode = 0xFF; /* in use if neq 0xFF */
|
||||
lis3dh->cfg.filter.cutoff = 0;
|
||||
@ -67,20 +67,20 @@ int lis3dh_init(lis3dh_t *lis3dh) {
|
||||
lis3dh->cfg.filter.ia1 = 0;
|
||||
lis3dh->cfg.filter.ia2 = 0;
|
||||
|
||||
lis3dh->cfg.int1.click = 0;
|
||||
lis3dh->cfg.int1.ia1 = 0;
|
||||
lis3dh->cfg.int1.ia2 = 0;
|
||||
lis3dh->cfg.int1.drdy_zyxda = 0;
|
||||
lis3dh->cfg.int1.drdy_321 = 0;
|
||||
lis3dh->cfg.int1.wtm = 0;
|
||||
lis3dh->cfg.int1.overrun = 0;
|
||||
lis3dh->cfg.pin1.click = 0;
|
||||
lis3dh->cfg.pin1.ia1 = 0;
|
||||
lis3dh->cfg.pin1.ia2 = 0;
|
||||
lis3dh->cfg.pin1.drdy_zyxda = 0;
|
||||
lis3dh->cfg.pin1.drdy_321 = 0;
|
||||
lis3dh->cfg.pin1.wtm = 0;
|
||||
lis3dh->cfg.pin1.overrun = 0;
|
||||
|
||||
lis3dh->cfg.int2.click = 0;
|
||||
lis3dh->cfg.int2.ia1 = 0;
|
||||
lis3dh->cfg.int2.ia2 = 0;
|
||||
lis3dh->cfg.int2.boot = 0;
|
||||
lis3dh->cfg.int2.act = 0;
|
||||
lis3dh->cfg.int2.polarity = 0;
|
||||
lis3dh->cfg.pin2.click = 0;
|
||||
lis3dh->cfg.pin2.ia1 = 0;
|
||||
lis3dh->cfg.pin2.ia2 = 0;
|
||||
lis3dh->cfg.pin2.boot = 0;
|
||||
lis3dh->cfg.pin2.act = 0;
|
||||
lis3dh->cfg.pin2.polarity = 0;
|
||||
|
||||
err |= lis3dh_reset(lis3dh);
|
||||
|
||||
@ -103,21 +103,21 @@ int lis3dh_configure(lis3dh_t *lis3dh) {
|
||||
ctrl_reg6 = 0;
|
||||
fifo_ctrl_reg = 0;
|
||||
|
||||
/* set interrupt registers */
|
||||
ctrl_reg3 |= (lis3dh->cfg.int1.click & 1) << 7;
|
||||
ctrl_reg3 |= (lis3dh->cfg.int1.ia1 & 1) << 6;
|
||||
ctrl_reg3 |= (lis3dh->cfg.int1.ia2 & 1) << 5;
|
||||
ctrl_reg3 |= (lis3dh->cfg.int1.drdy_zyxda & 1) << 4;
|
||||
ctrl_reg3 |= (lis3dh->cfg.int1.drdy_321 & 1) << 3;
|
||||
ctrl_reg3 |= (lis3dh->cfg.int1.wtm & 1) << 2;
|
||||
ctrl_reg3 |= (lis3dh->cfg.int1.overrun & 1) << 1;
|
||||
/* set pin interrupt settings */
|
||||
ctrl_reg3 |= (lis3dh->cfg.pin1.click & 1) << 7;
|
||||
ctrl_reg3 |= (lis3dh->cfg.pin1.ia1 & 1) << 6;
|
||||
ctrl_reg3 |= (lis3dh->cfg.pin1.ia2 & 1) << 5;
|
||||
ctrl_reg3 |= (lis3dh->cfg.pin1.drdy_zyxda & 1) << 4;
|
||||
ctrl_reg3 |= (lis3dh->cfg.pin1.drdy_321 & 1) << 3;
|
||||
ctrl_reg3 |= (lis3dh->cfg.pin1.wtm & 1) << 2;
|
||||
ctrl_reg3 |= (lis3dh->cfg.pin1.overrun & 1) << 1;
|
||||
|
||||
ctrl_reg6 |= (lis3dh->cfg.int2.click & 1) << 7;
|
||||
ctrl_reg6 |= (lis3dh->cfg.int2.ia1 & 1) << 6;
|
||||
ctrl_reg6 |= (lis3dh->cfg.int2.ia2 & 1) << 5;
|
||||
ctrl_reg6 |= (lis3dh->cfg.int2.boot & 1) << 4;
|
||||
ctrl_reg6 |= (lis3dh->cfg.int2.act & 1) << 3;
|
||||
ctrl_reg6 |= (lis3dh->cfg.int2.polarity & 1) << 1;
|
||||
ctrl_reg6 |= (lis3dh->cfg.pin2.click & 1) << 7;
|
||||
ctrl_reg6 |= (lis3dh->cfg.pin2.ia1 & 1) << 6;
|
||||
ctrl_reg6 |= (lis3dh->cfg.pin2.ia2 & 1) << 5;
|
||||
ctrl_reg6 |= (lis3dh->cfg.pin2.boot & 1) << 4;
|
||||
ctrl_reg6 |= (lis3dh->cfg.pin2.act & 1) << 3;
|
||||
ctrl_reg6 |= (lis3dh->cfg.pin2.polarity & 1) << 1;
|
||||
|
||||
/* set enable FIFO */
|
||||
if (lis3dh->cfg.fifo.mode != 0xFF) {
|
||||
|
8
lis3dh.h
8
lis3dh.h
@ -63,7 +63,7 @@ struct lis3dh_device {
|
||||
int (*deinit)(void);
|
||||
};
|
||||
|
||||
struct lis3dh_int2_config {
|
||||
struct lis3dh_pin2_config {
|
||||
uint8_t click; /* CLICK interrupt */
|
||||
uint8_t ia1; /* IA1 interrupt */
|
||||
uint8_t ia2; /* IA2 interrupt */
|
||||
@ -72,7 +72,7 @@ struct lis3dh_int2_config {
|
||||
uint8_t polarity; /* INT1 & INT2 polarity. 0 active high, 1 active low */
|
||||
};
|
||||
|
||||
struct lis3dh_int1_config {
|
||||
struct lis3dh_pin1_config {
|
||||
uint8_t click; /* CLICK interrupt */
|
||||
uint8_t ia1; /* IA1 interrupt */
|
||||
uint8_t ia2; /* IA2 interrupt */
|
||||
@ -103,8 +103,8 @@ struct lis3dh_config {
|
||||
uint8_t mode; /* LPen and HR */
|
||||
struct lis3dh_fifo_config fifo;
|
||||
struct lis3dh_filter_config filter;
|
||||
struct lis3dh_int1_config int1;
|
||||
struct lis3dh_int2_config int2;
|
||||
struct lis3dh_pin1_config pin1;
|
||||
struct lis3dh_pin2_config pin2;
|
||||
};
|
||||
|
||||
struct lis3dh_acceleration {
|
||||
|
Loading…
Reference in New Issue
Block a user