Files
mgos_i2c_mock/src/mgos_barometer_mpl3115.c
2018-04-22 09:42:47 +02:00

108 lines
2.4 KiB
C

#include "mgos_barometer_mpl3115.h"
#include "mgos_i2c.h"
// Datasheet:
// https://cdn-shop.adafruit.com/datasheets/1893_datasheet.pdf
bool mgos_barometer_mpl3115_detect(struct mgos_barometer *dev) {
int val;
if (!dev) {
return false;
}
if ((val = mgos_i2c_read_reg_b(dev->i2c, dev->i2caddr, MPL3115_REG_WHOAMI)) < 0) {
return false;
}
if (val != 0xC4) {
return false;
}
return true;
}
bool mgos_barometer_mpl3115_create(struct mgos_barometer *dev) {
if (!dev) {
return false;
}
// Reset
if (!mgos_i2c_write_reg_b(dev->i2c, dev->i2caddr, MPL3115_REG_CTRL1, 0x04)) {
return false;
}
mgos_usleep(20000);
// Set sample period to 1sec ST[3:0], period 2^ST seconds
if (!mgos_i2c_write_reg_b(dev->i2c, dev->i2caddr, MPL3115_REG_CTRL2, 0x00)) {
return false;
}
// Set Barometer Mode, OS[2:0], oversampling 2^OS times, continuous sampling
if (!mgos_i2c_write_reg_b(dev->i2c, dev->i2caddr, MPL3115_REG_CTRL1, 0x39)) {
return false;
}
// Set event flags for temp+pressure
if (!mgos_i2c_write_reg_b(dev->i2c, dev->i2caddr, MPL3115_REG_PT_DATA, 0x07)) {
return false;
}
dev->capabilities|=MGOS_BAROMETER_CAP_BAROMETER;
dev->capabilities|=MGOS_BAROMETER_CAP_THERMOMETER;
return true;
}
bool mgos_barometer_mpl3115_read(struct mgos_barometer *dev) {
if (!dev) {
return false;
}
uint8_t val = 0;
uint8_t retries=100;
if ((val = mgos_i2c_read_reg_b(dev->i2c, dev->i2caddr, MPL3115_REG_STATUS)) < 0)
return false;
while (!(val & 0x08) && retries>0) { // Data Ready
if ((val = mgos_i2c_read_reg_b(dev->i2c, dev->i2caddr, MPL3115_REG_STATUS)) < 0) {
return false;
}
mgos_usleep(10000);
// LOG(LL_DEBUG, ("Snoozing, retries=%d", retries));
retries--;
}
if (retries==0) {
LOG(LL_ERROR, ("Timed out waiting for data ready"));
return false;
}
uint32_t pressure;
int16_t temperature;
uint8_t data[5];
if (!mgos_i2c_read_reg_n(dev->i2c, dev->i2caddr, MPL3115_REG_PRESSURE_MSB, 5, data)) {
return false;
}
pressure = data[0];
pressure <<= 8;
pressure |= data[1];
pressure <<= 8;
pressure |= data[2];
pressure >>= 4;
temperature = data[3];
temperature <<= 8;
temperature |= data[4];
temperature >>= 4;
if (temperature & 0x800) {
temperature |= 0xF000;
}
dev->pressure = pressure / 4.0;
dev->temperature = temperature / 16.0;
return true;
}