Updated and Validated
[betaflight.git] / src / main / drivers / barometer / barometer_ms5611.c
blobb747868865d84e882f44eb836b7e229a01b1ff89
1 /*
2 * This file is part of Cleanflight and Betaflight.
4 * Cleanflight and Betaflight are free software. You can redistribute
5 * this software and/or modify this software under the terms of the
6 * GNU General Public License as published by the Free Software
7 * Foundation, either version 3 of the License, or (at your option)
8 * any later version.
10 * Cleanflight and Betaflight are distributed in the hope that they
11 * will be useful, but WITHOUT ANY WARRANTY; without even the implied
12 * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
13 * See the GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this software.
18 * If not, see <http://www.gnu.org/licenses/>.
21 #include <stdbool.h>
22 #include <stdint.h>
24 #include "platform.h"
26 #if defined(USE_BARO) && (defined(USE_BARO_MS5611) || defined(USE_BARO_SPI_MS5611))
28 #include "build/build_config.h"
30 #include "barometer.h"
31 #include "barometer_ms5611.h"
33 #include "drivers/bus_i2c.h"
34 #include "drivers/bus_i2c_busdev.h"
35 #include "drivers/bus_spi.h"
36 #include "drivers/io.h"
37 #include "drivers/time.h"
39 // 10 MHz max SPI frequency
40 #define MS5611_MAX_SPI_CLK_HZ 10000000
42 // MS5611, Standard address 0x77
43 #define MS5611_I2C_ADDR 0x77
45 #define CMD_RESET 0x1E // ADC reset command
46 #define CMD_ADC_READ 0x00 // ADC read command
47 #define CMD_ADC_CONV 0x40 // ADC conversion command
48 #define CMD_ADC_D1 0x00 // ADC D1 conversion
49 #define CMD_ADC_D2 0x10 // ADC D2 conversion
50 #define CMD_ADC_256 0x00 // ADC OSR=256
51 #define CMD_ADC_512 0x02 // ADC OSR=512
52 #define CMD_ADC_1024 0x04 // ADC OSR=1024
53 #define CMD_ADC_2048 0x06 // ADC OSR=2048
54 #define CMD_ADC_4096 0x08 // ADC OSR=4096
55 #define CMD_PROM_RD 0xA0 // Prom read command
56 #define PROM_NB 8
58 STATIC_UNIT_TESTED uint32_t ms5611_ut; // static result of temperature measurement
59 STATIC_UNIT_TESTED uint32_t ms5611_up; // static result of pressure measurement
60 STATIC_UNIT_TESTED uint16_t ms5611_c[PROM_NB]; // on-chip ROM
61 static uint8_t ms5611_osr = CMD_ADC_4096;
62 #define MS5611_DATA_FRAME_SIZE 3
63 static DMA_DATA_ZERO_INIT uint8_t sensor_data[MS5611_DATA_FRAME_SIZE];
65 void ms5611BusInit(const extDevice_t *dev)
67 #ifdef USE_BARO_SPI_MS5611
68 if (dev->bus->busType == BUS_TYPE_SPI) {
69 IOHi(dev->busType_u.spi.csnPin); // Disable
70 IOInit(dev->busType_u.spi.csnPin, OWNER_BARO_CS, 0);
71 IOConfigGPIO(dev->busType_u.spi.csnPin, IOCFG_OUT_PP);
72 spiSetClkDivisor(dev, spiCalculateDivider(MS5611_MAX_SPI_CLK_HZ));
74 #else
75 UNUSED(dev);
76 #endif
79 void ms5611BusDeinit(const extDevice_t *dev)
81 #ifdef USE_BARO_SPI_MS5611
82 if (dev->bus->busType == BUS_TYPE_SPI) {
83 spiPreinitByIO(dev->busType_u.spi.csnPin);
85 #else
86 UNUSED(dev);
87 #endif
90 static void ms5611Reset(const extDevice_t *dev)
92 busRawWriteRegister(dev, CMD_RESET, 1);
94 delayMicroseconds(2800);
97 static uint16_t ms5611Prom(const extDevice_t *dev, int8_t coef_num)
99 uint8_t rxbuf[2] = { 0, 0 };
101 busRawReadRegisterBuffer(dev, CMD_PROM_RD + coef_num * 2, rxbuf, 2); // send PROM READ command
103 return rxbuf[0] << 8 | rxbuf[1];
106 STATIC_UNIT_TESTED int8_t ms5611CRC(uint16_t *prom)
108 int32_t i, j;
109 uint32_t res = 0;
110 uint8_t crc = prom[7] & 0xF;
111 prom[7] &= 0xFF00;
113 bool blankEeprom = true;
115 for (i = 0; i < 16; i++) {
116 if (prom[i >> 1]) {
117 blankEeprom = false;
119 if (i & 1)
120 res ^= ((prom[i >> 1]) & 0x00FF);
121 else
122 res ^= (prom[i >> 1] >> 8);
123 for (j = 8; j > 0; j--) {
124 if (res & 0x8000)
125 res ^= 0x1800;
126 res <<= 1;
129 prom[7] |= crc;
130 if (!blankEeprom && crc == ((res >> 12) & 0xF))
131 return 0;
133 return -1;
136 static void ms5611ReadAdc(const extDevice_t *dev)
138 busRawReadRegisterBufferStart(dev, CMD_ADC_READ, sensor_data, MS5611_DATA_FRAME_SIZE); // read ADC
141 static void ms5611StartUT(baroDev_t *baro)
143 busRawWriteRegisterStart(&baro->dev, CMD_ADC_CONV + CMD_ADC_D2 + ms5611_osr, 1); // D2 (temperature) conversion start!
146 static bool ms5611ReadUT(baroDev_t *baro)
148 if (busBusy(&baro->dev, NULL)) {
149 return false;
152 ms5611ReadAdc(&baro->dev);
154 return true;
157 static bool ms5611GetUT(baroDev_t *baro)
159 if (busBusy(&baro->dev, NULL)) {
160 return false;
163 ms5611_ut = sensor_data[0] << 16 | sensor_data[1] << 8 | sensor_data[2];
165 return true;
168 static void ms5611StartUP(baroDev_t *baro)
170 busRawWriteRegisterStart(&baro->dev, CMD_ADC_CONV + CMD_ADC_D1 + ms5611_osr, 1); // D1 (pressure) conversion start!
173 static bool ms5611ReadUP(baroDev_t *baro)
175 if (busBusy(&baro->dev, NULL)) {
176 return false;
179 ms5611ReadAdc(&baro->dev);
181 return true;
184 static bool ms5611GetUP(baroDev_t *baro)
186 if (busBusy(&baro->dev, NULL)) {
187 return false;
190 ms5611_up = sensor_data[0] << 16 | sensor_data[1] << 8 | sensor_data[2];
192 return true;
195 STATIC_UNIT_TESTED void ms5611Calculate(int32_t *pressure, int32_t *temperature)
197 uint32_t press;
198 int64_t temp;
199 int64_t delt;
200 int64_t dT = (int64_t)ms5611_ut - ((uint64_t)ms5611_c[5] * 256);
201 int64_t off = ((int64_t)ms5611_c[2] << 16) + (((int64_t)ms5611_c[4] * dT) >> 7);
202 int64_t sens = ((int64_t)ms5611_c[1] << 15) + (((int64_t)ms5611_c[3] * dT) >> 8);
203 temp = 2000 + ((dT * (int64_t)ms5611_c[6]) >> 23);
205 if (temp < 2000) { // temperature lower than 20degC
206 delt = temp - 2000;
207 delt = 5 * delt * delt;
208 off -= delt >> 1;
209 sens -= delt >> 2;
210 if (temp < -1500) { // temperature lower than -15degC
211 delt = temp + 1500;
212 delt = delt * delt;
213 off -= 7 * delt;
214 sens -= (11 * delt) >> 1;
216 temp -= ((dT * dT) >> 31);
218 press = ((((int64_t)ms5611_up * sens) >> 21) - off) >> 15;
221 if (pressure)
222 *pressure = press;
223 if (temperature)
224 *temperature = temp;
227 bool ms5611Detect(baroDev_t *baro)
229 uint8_t sig;
230 int i;
231 bool defaultAddressApplied = false;
233 delay(10); // No idea how long the chip takes to power-up, but let's make it 10ms
235 extDevice_t *dev = &baro->dev;
237 ms5611BusInit(dev);
239 if ((dev->bus->busType == BUS_TYPE_I2C) && (dev->busType_u.i2c.address == 0)) {
240 // Default address for MS5611
241 dev->busType_u.i2c.address = MS5611_I2C_ADDR;
242 defaultAddressApplied = true;
245 if (!busRawReadRegisterBuffer(dev, CMD_PROM_RD, &sig, 1) || sig == 0xFF) {
246 goto fail;
249 ms5611Reset(dev);
251 // read all coefficients
252 for (i = 0; i < PROM_NB; i++)
253 ms5611_c[i] = ms5611Prom(dev, i);
255 // check crc, bail out if wrong - we are probably talking to BMP085 w/o XCLR line!
256 if (ms5611CRC(ms5611_c) != 0) {
257 goto fail;
260 busDeviceRegister(dev);
262 // TODO prom + CRC
263 baro->ut_delay = 10000;
264 baro->up_delay = 10000;
265 baro->start_ut = ms5611StartUT;
266 baro->read_ut = ms5611ReadUT;
267 baro->get_ut = ms5611GetUT;
268 baro->start_up = ms5611StartUP;
269 baro->read_up = ms5611ReadUP;
270 baro->get_up = ms5611GetUP;
271 baro->calculate = ms5611Calculate;
273 return true;
275 fail:;
276 ms5611BusDeinit(dev);
278 if (defaultAddressApplied) {
279 dev->busType_u.i2c.address = 0;
282 return false;
284 #endif