# LIS3DH A C89 driver for the 3-axis accelerometer LIS3DH. Supports both i2c and SPI. > ### Features > - FIFO of varying watermark level, up to 32 > - HP filter (4 c/o freq) > - 2G, 4G, 8G and 16G > - All power modes > - Interrupt generation (soon) > - Free-fall detection (soon) > - Single and double click detection (soon) > - 4D/6D orientation detection (soon) ## Implementation This driver requires the user to provide pointers to the following abstractely named functions: This project has example interface code for I2C used on Raspberry Pi. ```c /* initialise the "interface" */ int init(void); /* read from device register `reg`, `size` amount of bytes and write them to `dst` */ int read(uint8_t reg, uint8_t *dst, uint32_t size); /* write `value` to device register `reg` */ int write(uint8_t reg, uint8_t value); /* sleep for `dur_us` microseconds */ int sleep(uint32_t dur_us); /* deinitalise the "interface" */ int deinit(void); ``` All above functions return `0` on success. ## Usage Simple example: ```c #define _GNU_SOURCE #include /* usleep() */ #include #include "lis3dh.h" #include "i2c.h" int main() { lis3dh_t lis; 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 */ } lis.cfg.mode = LIS3DH_MODE_HR; lis.cfg.range = LIS3DH_FS_4G; lis.cfg.rate = LIS3DH_ODR_100_HZ; if (lis3dh_configure(&lis)) { /* error handling */ } if (lis3dh_poll(&lis)) { /* error handling */ } if (lis3dh_read(&lis)) { /* error handling */ } printf("x: %f, y: %f, z: %f\n", lis.acc.x, lis.acc.y, lis.acc.z); if (lis3dh_deinit(&lis)) { /* error handling */ } return 0; } ``` The output should be something similar to this: ``` $ ./lis3dh x: 0.540000, y: -0.882000, z: -0.100000 ``` poll() and read() can be indefinitely looped for a constant data stream, like this: ```c #define _GNU_SOURCE #include #include #include "lis3dh.h" #include "i2c.h" int main() { lis3dh_t lis; int i; 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 */ } lis.cfg.mode = LIS3DH_MODE_HR; lis.cfg.range = LIS3DH_FS_4G; lis.cfg.rate = LIS3DH_ODR_100_HZ; if (lis3dh_configure(&lis)) { /* error handling */ } for(i=0; i<10; i++) { if (lis3dh_poll(&lis)) { /* error handling */ } if (lis3dh_read(&lis)) { /* error handling */ } printf("x: %f, y: %f, z: %f\n", lis.acc.x, lis.acc.y, lis.acc.z); } if (lis3dh_deinit(&lis)) { /* error handling */ } return 0; } ``` Output: ``` $ ./lis3dh x: 0.534000, y: -0.882000, z: -0.102000 x: 0.538000, y: -0.866000, z: -0.136000 x: 0.518000, y: -0.846000, z: -0.100000 x: 0.518000, y: -0.840000, z: -0.098000 x: 0.542000, y: -0.876000, z: -0.098000 x: 0.518000, y: -0.834000, z: -0.146000 x: 0.512000, y: -0.854000, z: -0.106000 x: 0.574000, y: -0.870000, z: -0.122000 x: 0.518000, y: -0.846000, z: -0.098000 x: 0.516000, y: -0.852000, z: -0.112000 ``` ## Using FIFO 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-32] inclusive by modifying the `lis.cfg.fifo.fth` property before calling configure(). ```c #define _GNU_SOURCE #include #include #include "lis3dh.h" #include "i2c.h" int main() { lis3dh_t lis; struct lis3dh_fifo_data data; int i; 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 */ } lis.cfg.mode = LIS3DH_MODE_HR; lis.cfg.range = LIS3DH_FS_4G; lis.cfg.rate = LIS3DH_ODR_100_HZ; lis.cfg.fifo.mode = LIS3DH_FIFO_MODE_NORMAL; if (lis3dh_configure(&lis)) { /* error handling */ } if (lis3dh_poll_fifo(&lis)) { /* error handling */ } if (lis3dh_read_fifo(&lis, &data)) { /* error handling */ } /* read out fifo buffer data */ for(i=0; i #include #include "lis3dh.h" #include "i2c.h" int main() { lis3dh_t lis; struct lis3dh_fifo_data data; int i; 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 */ } lis.cfg.mode = LIS3DH_MODE_HR; lis.cfg.range = LIS3DH_FS_4G; lis.cfg.rate = LIS3DH_ODR_100_HZ; lis.cfg.fifo.mode = LIS3DH_FIFO_MODE_NORMAL; lis.cfg.filter.mode = LIS3DH_FILTER_MODE_AUTORESET; lis.cfg.filter.cutoff = LIS3DH_FILTER_CUTOFF_8; if (lis3dh_configure(&lis)) { /* error handling */ } if (lis3dh_poll_fifo(&lis)) { /* error handling */ } if (lis3dh_read_fifo(&lis, &data)) { /* error handling */ } /* read out fifo buffer data */ for(i=0; i