1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-15 23:53:08 +01:00
RIOT/drivers/qmc5883l/qmc5883l.c
2019-11-13 12:56:26 +01:00

204 lines
4.9 KiB
C

/*
* Copyright (C) 2019 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.
*/
/**
* @ingroup drivers_qmc5883l
* @{
*
* @file
* @brief Implementation of the QMC5883L device driver
*
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
*
* @}
*/
#include <limits.h>
#include "assert.h"
#include "periph/gpio.h"
#include "qmc5883l.h"
#include "qmc5883l_internal.h"
/* shortcut to the i2c device address */
#define ADDR QMC5883L_ADDR
static int _reg_read(const qmc5883l_t *dev, uint8_t reg,
uint8_t *val, int acquire, int release)
{
if (acquire) {
if (i2c_acquire(dev->i2c) != 0) {
return QMC5883L_BUSERR;
}
}
int res = i2c_read_reg(dev->i2c, ADDR, reg, val, 0);
if ((release) || (res != 0)) {
i2c_release(dev->i2c);
}
return (res == 0) ? QMC5883L_OK : QMC5883L_BUSERR;
}
static int _reg_write(const qmc5883l_t *dev, uint8_t reg,
uint8_t val, int acquire, int release)
{
if (acquire) {
if (i2c_acquire(dev->i2c) != 0) {
return QMC5883L_BUSERR;
}
}
int res = i2c_write_reg(dev->i2c, ADDR, reg, val, 0);
if ((release) || (res != 0)) {
i2c_release(dev->i2c);
}
return (res == 0) ? QMC5883L_OK : QMC5883L_BUSERR;
}
int qmc5883l_init(qmc5883l_t *dev, const qmc5883l_params_t *params)
{
assert(dev);
assert(params);
int res;
dev->i2c = params->i2c;
dev->pin_drdy = params->pin_drdy;
dev->cfg = (params->odr | params->rng | params->osr | QMC5883L_CONT);
/* lets start with a soft reset */
res = _reg_write(dev, QMC5883L_CTRL2, QMC5883L_SOFT_RST, 1, 0);
if (res != QMC5883L_OK) {
return res;
}
/* verify the reset by reading the status register, should be all zeros */
uint8_t tmp;
if (i2c_read_reg(dev->i2c, ADDR, QMC5883L_STATUS, &tmp, 0) != 0) {
i2c_release(dev->i2c);
return QMC5883L_BUSERR;
}
if (tmp != 0) {
i2c_release(dev->i2c);
return QMC5883L_NOCFG;
}
/* write the actual device configuration */
res = _reg_write(dev, QMC5883L_SETRESET, 0x01, 0, 0);
if (res != QMC5883L_OK) {
return res;
}
res = _reg_write(dev, QMC5883L_CTRL2, QMC5883L_INT_ENB, 0, 0);
if (res != QMC5883L_OK) {
return res;
}
return _reg_write(dev, QMC5883L_CTRL1, (dev->cfg | QMC5883L_CONT), 0, 1);
}
int qmc5883l_data_ready(const qmc5883l_t *dev)
{
assert(dev);
uint8_t status;
int res = _reg_read(dev, QMC5883L_STATUS, &status, 1, 1);
if (res != QMC5883L_OK) {
return res;
}
return (status & QMC5883L_DRDY) ? QMC5883L_OK : QMC5883L_NODATA;
}
int qmc5883l_read(const qmc5883l_t *dev, int16_t *data_out)
{
assert(data_out);
int16_t tmp[3];
int res = qmc5883l_read_raw(dev, tmp);
if ((res == QMC5883L_OK) || (res == QMC5883L_OVERFLOW)) {
uint16_t scale = (dev->cfg & QMC5883L_RNG_8G) ? 3 : 12;
for (unsigned i = 0; i < 3; i++) {
data_out[i] = tmp[i] / scale;
}
}
return res;
}
int qmc5883l_read_raw(const qmc5883l_t *dev, int16_t *data_out)
{
assert(dev);
assert(data_out);
int res;
uint8_t status;
uint8_t tmp[6];
res = _reg_read(dev, QMC5883L_STATUS, &status, 1, 0);
if (res != QMC5883L_OK) {
return res;
}
if (!(status & QMC5883L_DRDY)) {
res = QMC5883L_NODATA;
goto done;
}
if (status & QMC5883L_OVL) {
res = QMC5883L_OVERFLOW;
}
if (i2c_read_regs(dev->i2c, ADDR, QMC5883L_DOXL, tmp, 6, 0) != 0) {
res = QMC5883L_BUSERR;
goto done;
}
/* convert data to host byte order */
data_out[0] = (int16_t)tmp[1] << 8 | tmp[0];
data_out[1] = (int16_t)tmp[3] << 8 | tmp[2];
data_out[2] = (int16_t)tmp[5] << 8 | tmp[4];
done:
i2c_release(dev->i2c);
return res;
}
int qmc5883l_poweron(const qmc5883l_t *dev)
{
assert(dev);
return _reg_write(dev, QMC5883L_CTRL1, dev->cfg, 1, 1);
}
int qmc5883l_poweroff(const qmc5883l_t *dev)
{
assert(dev);
return _reg_write(dev, QMC5883L_CTRL1, 0, 1, 1);
}
#ifdef MODULE_QMC5883L_INT
int qmc5883l_init_int(const qmc5883l_t *dev, gpio_cb_t cb, void *arg)
{
assert(dev);
assert(cb);
if (dev->pin_drdy == GPIO_UNDEF) {
return QMC5883L_NOCFG;
}
if (gpio_init_int(dev->pin_drdy, GPIO_IN, GPIO_RISING, cb, arg) != 0) {
return QMC5883L_NOCFG;
}
return _reg_write(dev, QMC5883L_CTRL2, 0, 1, 1);
}
int qmc5883l_irq_enable(const qmc5883l_t *dev)
{
assert(dev);
gpio_irq_enable(dev->pin_drdy);
return _reg_write(dev, QMC5883L_CTRL2, 0, 1, 1);
}
int qmc5883l_irq_disable(const qmc5883l_t *dev)
{
assert(dev);
gpio_irq_disable(dev->pin_drdy);
return _reg_write(dev, QMC5883L_CTRL2, QMC5883L_INT_ENB, 1, 1);
}
#endif /* MODULE_QMC5883L_INT */