| /* Driver for Infineon DPS310 temperature and pressure sensor */ |
| |
| /* |
| * Copyright (c) 2019 Infineon Technologies AG |
| * |
| * SPDX-License-Identifier: Apache-2.0 |
| */ |
| |
| #define DT_DRV_COMPAT infineon_dps310 |
| |
| #include <zephyr/kernel.h> |
| #include <zephyr/drivers/sensor.h> |
| #include <zephyr/init.h> |
| #include <zephyr/sys/byteorder.h> |
| #include <zephyr/sys/util.h> |
| |
| #include <zephyr/drivers/i2c.h> |
| #include <zephyr/logging/log.h> |
| |
| LOG_MODULE_REGISTER(DPS310, CONFIG_SENSOR_LOG_LEVEL); |
| |
| /* Register addresses as in the datasheet */ |
| #define IFX_DPS310_REG_ADDR_PSR_B2 (0x00) |
| #define IFX_DPS310_REG_ADDR_TMP_B2 (0x03) |
| #define IFX_DPS310_REG_ADDR_PRS_CFG (0x06) |
| #define IFX_DPS310_REG_ADDR_TMP_CFG (0x07) |
| #define IFX_DPS310_REG_ADDR_MEAS_CFG (0x08) |
| #define IFX_DPS310_REG_ADDR_CFG_REG (0x09) |
| #define IFX_DPS310_REG_ADDR_INT_STS (0x0A) |
| #define IFX_DPS310_REG_ADDR_FIFO_STS (0x0B) |
| #define IFX_DPS310_REG_ADDR_RESET (0x0C) |
| #define IFX_DPS310_REG_ADDR_PRODUCT_ID (0x0D) |
| #define IFX_DPS310_REG_ADDR_COEF_0 (0x10) |
| #define IFX_DPS310_REG_ADDR_COEF_SRCE (0x28) |
| |
| enum { |
| IFX_DPS310_MODE_IDLE = 0x00, |
| IFX_DPS310_MODE_COMMAND_PRESSURE = 0x01, |
| IFX_DPS310_MODE_COMMAND_TEMPERATURE = 0x02, |
| IFX_DPS310_MODE_BACKGROUND_PRESSURE = 0x05, |
| IFX_DPS310_MODE_BACKGROUND_TEMPERATURE = 0x06, |
| IFX_DPS310_MODE_BACKGROUND_ALL = 0x07 |
| }; |
| |
| /* Bits in registers as in the datasheet */ |
| #define IFX_DPS310_REG_ADDR_MEAS_CFG_PRS_RDY (0x10) |
| #define IFX_DPS310_REG_ADDR_MEAS_CFG_TMP_RDY (0x20) |
| /* |
| * If sensor is ready after self initialization |
| * bits 6 and 7 in register MEAS_CFG (0x08) should be "1" |
| */ |
| #define IFX_DPS310_REG_ADDR_MEAS_CFG_SELF_INIT_OK (0xC0) |
| #define IFX_DPS310_COEF_SRCE_MASK (0x80) |
| #define IFX_DPS310_PRODUCT_ID (0x10) |
| |
| /* Polling time in ms*/ |
| #define POLL_TIME_MS (K_MSEC(10)) |
| /* Number of times to poll before timeout */ |
| #define POLL_TRIES 3 |
| |
| /* |
| * Measurement times in ms for different oversampling rates |
| * From Table 16 in the datasheet, rounded up for safety margin |
| */ |
| enum { |
| IFX_DPS310_MEAS_TIME_OSR_1 = 4, |
| IFX_DPS310_MEAS_TIME_OSR_2 = 6, |
| IFX_DPS310_MEAS_TIME_OSR_4 = 9, |
| IFX_DPS310_MEAS_TIME_OSR_8 = 15, |
| IFX_DPS310_MEAS_TIME_OSR_16 = 28, |
| IFX_DPS310_MEAS_TIME_OSR_32 = 54, |
| IFX_DPS310_MEAS_TIME_OSR_64 = 105, |
| IFX_DPS310_MEAS_TIME_OSR_128 = 207 |
| }; |
| |
| /* Compensation scale factors from Table 9 in the datasheet */ |
| enum { |
| IFX_DPS310_SF_OSR_1 = 524288, |
| IFX_DPS310_SF_OSR_2 = 1572864, |
| IFX_DPS310_SF_OSR_4 = 3670016, |
| IFX_DPS310_SF_OSR_8 = 7864320, |
| IFX_DPS310_SF_OSR_16 = 253952, |
| IFX_DPS310_SF_OSR_32 = 516096, |
| IFX_DPS310_SF_OSR_64 = 1040384, |
| IFX_DPS310_SF_OSR_128 = 2088960 |
| }; |
| |
| /* |
| * Oversampling and measurement rates configuration for pressure and temperature |
| * sensor According to Table 16 of the datasheet |
| */ |
| enum { |
| IFX_DPS310_RATE_1 = 0x00, |
| IFX_DPS310_RATE_2 = 0x01, |
| IFX_DPS310_RATE_4 = 0x02, |
| IFX_DPS310_RATE_8 = 0x03, |
| IFX_DPS310_RATE_16 = 0x04, |
| IFX_DPS310_RATE_32 = 0x05, |
| IFX_DPS310_RATE_64 = 0x06, |
| IFX_DPS310_RATE_128 = 0x07 |
| }; |
| |
| /* Helper macro to set temperature and pressure config register */ |
| #define CFG_REG(MEAS_RATE, OSR_RATE) \ |
| ((((MEAS_RATE)&0x07) << 4) | ((OSR_RATE)&0x07)) |
| |
| /* Setup constants depending on temperature oversampling factor */ |
| #if defined CONFIG_DPS310_TEMP_OSR_1X |
| #define IFX_DPS310_SF_TMP IFX_DPS310_SF_OSR_1 |
| #define IFX_DPS310_TMP_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_1 |
| #define IFX_DPS310_TMP_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_1) |
| #define IFX_DPS310_T_SHIFT 0 |
| #elif defined CONFIG_DPS310_TEMP_OSR_2X |
| #define IFX_DPS310_SF_TMP IFX_DPS310_SF_OSR_2 |
| #define IFX_DPS310_TMP_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_2 |
| #define IFX_DPS310_TMP_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_2) |
| #define IFX_DPS310_T_SHIFT 0 |
| #elif defined CONFIG_DPS310_TEMP_OSR_4X |
| #define IFX_DPS310_SF_TMP IFX_DPS310_SF_OSR_4 |
| #define IFX_DPS310_TMP_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_4 |
| #define IFX_DPS310_TMP_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_4) |
| #define IFX_DPS310_T_SHIFT 0 |
| #elif defined CONFIG_DPS310_TEMP_OSR_8X |
| #define IFX_DPS310_SF_TMP IFX_DPS310_SF_OSR_8 |
| #define IFX_DPS310_TMP_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_8 |
| #define IFX_DPS310_TMP_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_8) |
| #define IFX_DPS310_T_SHIFT 0 |
| #elif defined CONFIG_DPS310_TEMP_OSR_16X |
| #define IFX_DPS310_SF_TMP IFX_DPS310_SF_OSR_16 |
| #define IFX_DPS310_TMP_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_16 |
| #define IFX_DPS310_TMP_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_16) |
| #define IFX_DPS310_T_SHIFT 1 |
| #elif defined CONFIG_DPS310_TEMP_OSR_32X |
| #define IFX_DPS310_SF_TMP IFX_DPS310_SF_OSR_32 |
| #define IFX_DPS310_TMP_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_32 |
| #define IFX_DPS310_TMP_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_32) |
| #define IFX_DPS310_T_SHIFT 1 |
| #elif defined CONFIG_DPS310_TEMP_OSR_64X |
| #define IFX_DPS310_SF_TMP IFX_DPS310_SF_OSR_64 |
| #define IFX_DPS310_TMP_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_64 |
| #define IFX_DPS310_TMP_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_64) |
| #define IFX_DPS310_T_SHIFT 1 |
| #elif defined CONFIG_DPS310_TEMP_OSR_128X |
| #define IFX_DPS310_SF_TMP IFX_DPS310_SF_OSR_128 |
| #define IFX_DPS310_TMP_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_128 |
| #define IFX_DPS310_TMP_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_128) |
| #define IFX_DPS310_T_SHIFT 1 |
| #endif |
| |
| /* Setup constants depending on pressure oversampling factor */ |
| #if defined CONFIG_DPS310_PRESS_OSR_1X |
| #define IFX_DPS310_SF_PSR IFX_DPS310_SF_OSR_1 |
| #define IFX_DPS310_PSR_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_1 |
| #define IFX_DPS310_PSR_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_1) |
| #define IFX_DPS310_P_SHIFT 0 |
| #elif defined CONFIG_DPS310_PRESS_OSR_2X |
| #define IFX_DPS310_SF_PSR IFX_DPS310_SF_OSR_2 |
| #define IFX_DPS310_PSR_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_2 |
| #define IFX_DPS310_PSR_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_2) |
| #define IFX_DPS310_P_SHIFT 0 |
| #elif defined CONFIG_DPS310_PRESS_OSR_4X |
| #define IFX_DPS310_SF_PSR IFX_DPS310_SF_OSR_4 |
| #define IFX_DPS310_PSR_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_4 |
| #define IFX_DPS310_PSR_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_4) |
| #define IFX_DPS310_P_SHIFT 0 |
| #elif defined CONFIG_DPS310_PRESS_OSR_8X |
| #define IFX_DPS310_SF_PSR IFX_DPS310_SF_OSR_8 |
| #define IFX_DPS310_PSR_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_8 |
| #define IFX_DPS310_PSR_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_8) |
| #define IFX_DPS310_P_SHIFT 0 |
| #elif defined CONFIG_DPS310_PRESS_OSR_16X |
| #define IFX_DPS310_SF_PSR IFX_DPS310_SF_OSR_16 |
| #define IFX_DPS310_PSR_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_16 |
| #define IFX_DPS310_PSR_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_16) |
| #define IFX_DPS310_P_SHIFT 1 |
| #elif defined CONFIG_DPS310_PRESS_OSR_32X |
| #define IFX_DPS310_SF_PSR IFX_DPS310_SF_OSR_32 |
| #define IFX_DPS310_PSR_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_32 |
| #define IFX_DPS310_PSR_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_32) |
| #define IFX_DPS310_P_SHIFT 1 |
| #elif defined CONFIG_DPS310_PRESS_OSR_64X |
| #define IFX_DPS310_SF_PSR IFX_DPS310_SF_OSR_64 |
| #define IFX_DPS310_PSR_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_64 |
| #define IFX_DPS310_PSR_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_64) |
| #define IFX_DPS310_P_SHIFT 1 |
| #elif defined CONFIG_DPS310_PRESS_OSR_128X |
| #define IFX_DPS310_SF_PSR IFX_DPS310_SF_OSR_128 |
| #define IFX_DPS310_PSR_MEAS_TIME IFX_DPS310_MEAS_TIME_OSR_128 |
| #define IFX_DPS310_PSR_CFG CFG_REG(IFX_DPS310_RATE_1, IFX_DPS310_RATE_128) |
| #define IFX_DPS310_P_SHIFT 1 |
| #endif |
| |
| #define DPS310_CFG_REG \ |
| (((IFX_DPS310_T_SHIFT & 0x01) << 3) | \ |
| ((IFX_DPS310_P_SHIFT & 0x01) << 2)) |
| |
| #define HW_BUG_FIX_SEQUENCE_LEN 5 |
| |
| #define POW_2_23_MINUS_1 BIT_MASK(23) |
| #define POW_2_24 BIT(24) |
| #define POW_2_15_MINUS_1 BIT_MASK(15) |
| #define POW_2_16 BIT(16) |
| #define POW_2_11_MINUS_1 BIT_MASK(11) |
| #define POW_2_12 BIT(12) |
| #define POW_2_20 BIT(20) |
| #define POW_2_19_MINUS_1 BIT_MASK(19) |
| |
| /* Needed because the values are referenced by pointer. */ |
| static const uint8_t REG_ADDR_MEAS_CFG = IFX_DPS310_REG_ADDR_MEAS_CFG; |
| static const uint8_t REG_ADDR_CALIB_COEFF_0 = IFX_DPS310_REG_ADDR_COEF_0; |
| static const uint8_t REG_ADDR_TMP_B2 = IFX_DPS310_REG_ADDR_TMP_B2; |
| static const uint8_t REG_ADDR_PSR_B2 = IFX_DPS310_REG_ADDR_PSR_B2; |
| static const uint8_t REG_ADDR_COEF_SRCE = IFX_DPS310_REG_ADDR_COEF_SRCE; |
| |
| /* calibration coefficients */ |
| struct dps310_cal_coeff { |
| /* Pressure Sensor Calibration Coefficients */ |
| int32_t c00; /* 20bit */ |
| int32_t c10; /* 20bit */ |
| int16_t c01; /* 16bit */ |
| int16_t c11; /* 16bit */ |
| int16_t c20; /* 16bit */ |
| int16_t c21; /* 16bit */ |
| int16_t c30; /* 16bit */ |
| /* Temperature Sensor Calibration Coefficients */ |
| int16_t c0; /* 12bit */ |
| int16_t c1; /* 12bit */ |
| }; |
| |
| struct dps310_data { |
| struct dps310_cal_coeff comp; |
| /* Temperature Values */ |
| int32_t tmp_val1; |
| int32_t tmp_val2; |
| /* Last raw temperature value for temperature compensation */ |
| int32_t raw_tmp; |
| /* Pressure Values */ |
| int32_t psr_val1; |
| int32_t psr_val2; |
| }; |
| |
| struct dps310_cfg { |
| struct i2c_dt_spec i2c; |
| }; |
| |
| /* |
| * Convert the bytes from calibration memory to calibration coefficients |
| * structure |
| */ |
| static void dps310_calib_coeff_creation(const uint8_t raw_coef[18], |
| struct dps310_cal_coeff *comp) |
| { |
| /* Temperature sensor compensation values */ |
| comp->c0 = (((uint16_t)raw_coef[0]) << 4) + (raw_coef[1] >> 4); |
| /* coefficient is 2nd compliment */ |
| if (comp->c0 > POW_2_11_MINUS_1) { |
| comp->c0 = comp->c0 - POW_2_12; |
| } |
| |
| comp->c1 = (((uint16_t)(raw_coef[1] & 0x0F)) << 8) + raw_coef[2]; |
| /* coefficient is 2nd compliment */ |
| if (comp->c1 > POW_2_11_MINUS_1) { |
| comp->c1 = comp->c1 - POW_2_12; |
| } |
| |
| /* Pressure sensor compensation values */ |
| comp->c00 = (((uint32_t)raw_coef[3]) << 12) + (((uint16_t)raw_coef[4]) << 4) + |
| (raw_coef[5] >> 4); |
| /* coefficient is 2nd compliment */ |
| if (comp->c00 > POW_2_19_MINUS_1) { |
| comp->c00 = comp->c00 - POW_2_20; |
| } |
| |
| comp->c10 = (((uint32_t)(raw_coef[5] & 0x0F)) << 16) + |
| (((uint16_t)raw_coef[6]) << 8) + raw_coef[7]; |
| /* coefficient is 2nd compliment */ |
| if (comp->c10 > POW_2_19_MINUS_1) { |
| comp->c10 = comp->c10 - POW_2_20; |
| } |
| |
| comp->c01 = (int16_t) sys_get_be16(&raw_coef[8]); |
| comp->c11 = (int16_t) sys_get_be16(&raw_coef[10]); |
| comp->c20 = (int16_t) sys_get_be16(&raw_coef[12]); |
| comp->c21 = (int16_t) sys_get_be16(&raw_coef[14]); |
| comp->c30 = (int16_t) sys_get_be16(&raw_coef[16]); |
| } |
| |
| /* Poll one or multiple bits given by ready_mask in reg_addr */ |
| static bool poll_rdy(const struct device *dev, uint8_t reg_addr, uint8_t ready_mask) |
| { |
| const struct dps310_cfg *config = dev->config; |
| /* Try only a finite number of times */ |
| for (int i = 0; i < POLL_TRIES; i++) { |
| uint8_t reg = 0; |
| int res = i2c_reg_read_byte_dt(&config->i2c, reg_addr, ®); |
| if (res < 0) { |
| LOG_WRN("I2C error: %d", res); |
| return false; |
| } |
| |
| if ((reg & ready_mask) == ready_mask) { |
| /* measurement is ready */ |
| return true; |
| } |
| |
| /* give the sensor more time */ |
| k_sleep(POLL_TIME_MS); |
| } |
| |
| return false; |
| } |
| |
| /* Trigger a temperature measurement and wait until the result is stored */ |
| static bool dps310_trigger_temperature(const struct device *dev) |
| { |
| const struct dps310_cfg *config = dev->config; |
| |
| /* command to start temperature measurement */ |
| static const uint8_t tmp_meas_cmd[] = { |
| IFX_DPS310_REG_ADDR_MEAS_CFG, |
| IFX_DPS310_MODE_COMMAND_TEMPERATURE |
| }; |
| |
| /* trigger temperature measurement */ |
| int res = i2c_write_dt(&config->i2c, tmp_meas_cmd, sizeof(tmp_meas_cmd)); |
| if (res < 0) { |
| LOG_WRN("I2C error: %d", res); |
| return false; |
| } |
| |
| /* give the sensor time to store measured values internally */ |
| k_msleep(IFX_DPS310_TMP_MEAS_TIME); |
| |
| if (!poll_rdy(dev, IFX_DPS310_REG_ADDR_MEAS_CFG, IFX_DPS310_REG_ADDR_MEAS_CFG_TMP_RDY)) { |
| LOG_DBG("Poll timeout for temperature"); |
| return false; |
| } |
| |
| return true; |
| } |
| |
| /* Trigger a pressure measurement and wait until the result is stored */ |
| static bool dps310_trigger_pressure(const struct device *dev) |
| { |
| const struct dps310_cfg *config = dev->config; |
| |
| /* command to start pressure measurement */ |
| static const uint8_t psr_meas_cmd[] = { |
| IFX_DPS310_REG_ADDR_MEAS_CFG, |
| IFX_DPS310_MODE_COMMAND_PRESSURE |
| }; |
| |
| /* trigger pressure measurement */ |
| int res = i2c_write_dt(&config->i2c, psr_meas_cmd, sizeof(psr_meas_cmd)); |
| if (res < 0) { |
| LOG_WRN("I2C error: %d", res); |
| return false; |
| } |
| |
| /* give the sensor time to store measured values internally */ |
| k_msleep(IFX_DPS310_PSR_MEAS_TIME); |
| |
| if (!poll_rdy(dev, IFX_DPS310_REG_ADDR_MEAS_CFG, IFX_DPS310_REG_ADDR_MEAS_CFG_PRS_RDY)) { |
| LOG_DBG("Poll timeout for pressure"); |
| return false; |
| } |
| |
| return true; |
| } |
| |
| /* |
| * function to fix a hardware problem on some devices |
| * you have this bug if you measure around 60°C when temperature is around 20°C |
| * call dps310_hw_bug_fix() directly in the init() function to fix this issue |
| */ |
| static void dps310_hw_bug_fix(const struct device *dev) |
| { |
| const struct dps310_cfg *config = dev->config; |
| |
| /* setup the necessary 5 sequences to fix the hw bug */ |
| static const uint8_t hw_bug_fix_sequence[HW_BUG_FIX_SEQUENCE_LEN][2] = { |
| /* |
| * First write valid signature on 0x0e and 0x0f |
| * to unlock address 0x62 |
| */ |
| { 0x0E, 0xA5 }, |
| { 0x0F, 0x96 }, |
| /* Then update high gain value for Temperature */ |
| { 0x62, 0x02 }, |
| /* Finally lock back the location 0x62 */ |
| { 0x0E, 0x00 }, |
| { 0x0F, 0x00 } |
| }; |
| |
| /* execute sequence for hw bug fix */ |
| for (int i = 0; i < HW_BUG_FIX_SEQUENCE_LEN; i++) { |
| int res = i2c_write_dt(&config->i2c, hw_bug_fix_sequence[i], 2); |
| if (res < 0) { |
| LOG_WRN("I2C error: %d", res); |
| return; |
| } |
| } |
| } |
| |
| /* |
| * Scale and compensate the raw temperature measurement value to micro °C |
| * The formula is based on the Chapter 4.9.2 in the datasheet and was |
| * modified to need only integer arithmetic. |
| */ |
| static void dps310_scale_temperature(const struct device *dev, int32_t tmp_raw) |
| { |
| struct dps310_data *data = dev->data; |
| const struct dps310_cal_coeff *comp = &data->comp; |
| |
| /* first term, rescaled to micro °C */ |
| int32_t tmp_p0 = (1000000 / 2) * comp->c0; |
| |
| /* second term, rescaled to micro °C */ |
| int32_t tmp_p1 = |
| (((int64_t)1000000) * comp->c1 * tmp_raw) / IFX_DPS310_SF_TMP; |
| |
| /* calculate the temperature corresponding to the datasheet */ |
| int32_t tmp_final = tmp_p0 + tmp_p1; /* value is in micro °C */ |
| |
| /* store calculated value */ |
| data->tmp_val1 = tmp_final / 1000000; |
| data->tmp_val2 = tmp_final % 1000000; |
| } |
| |
| /* |
| * Scale and temperature compensate the raw pressure measurement value to |
| * Kilopascal. The formula is based on the Chapter 4.9.1 in the datasheet. |
| */ |
| static void dps310_scale_pressure(const struct device *dev, int32_t tmp_raw, int32_t psr_raw) |
| { |
| struct dps310_data *data = dev->data; |
| const struct dps310_cal_coeff *comp = &data->comp; |
| |
| float psr = ((float)psr_raw) / IFX_DPS310_SF_PSR; |
| float tmp = ((float)tmp_raw) / IFX_DPS310_SF_TMP; |
| |
| /* scale according to formula from datasheet */ |
| float psr_final = comp->c00; |
| |
| psr_final += psr * (comp->c10 + psr * (comp->c20 + psr * comp->c30)); |
| psr_final += tmp * comp->c01; |
| psr_final += tmp * psr * (comp->c11 + psr * comp->c21); |
| |
| /* rescale from Pascal to Kilopascal */ |
| psr_final /= 1000; |
| |
| /* store calculated value */ |
| data->psr_val1 = psr_final; |
| data->psr_val2 = (psr_final - data->psr_val1) * 1000000; |
| } |
| |
| /* Convert the raw sensor data to int32_t */ |
| static int32_t raw_to_int24(const uint8_t raw[3]) |
| { |
| /* convert from twos complement */ |
| int32_t res = (int32_t) sys_get_be24(raw); |
| |
| if (res > POW_2_23_MINUS_1) { |
| res -= POW_2_24; |
| } |
| |
| return res; |
| } |
| |
| /* perform a single measurement of temperature and pressure */ |
| static bool dps310_measure_tmp_psr(const struct device *dev) |
| { |
| struct dps310_data *data = dev->data; |
| const struct dps310_cfg *config = dev->config; |
| |
| if (!dps310_trigger_temperature(dev)) { |
| return false; |
| } |
| |
| if (!dps310_trigger_pressure(dev)) { |
| return false; |
| } |
| |
| /* memory for pressure and temperature raw values */ |
| uint8_t value_raw[6]; |
| |
| /* read pressure and temperature raw values in one continuous read */ |
| int res = i2c_write_read_dt(&config->i2c, ®_ADDR_PSR_B2, 1, |
| &value_raw, sizeof(value_raw)); |
| if (res < 0) { |
| LOG_WRN("I2C error: %d", res); |
| return false; |
| } |
| |
| /* convert raw data to int */ |
| int32_t psr_raw = raw_to_int24(&value_raw[0]); |
| |
| data->raw_tmp = raw_to_int24(&value_raw[3]); |
| |
| dps310_scale_temperature(dev, data->raw_tmp); |
| dps310_scale_pressure(dev, data->raw_tmp, psr_raw); |
| |
| return true; |
| } |
| |
| /* |
| * perform a single pressure measurement |
| * uses the stored temperature value for sensor temperature compensation |
| * temperature must be measured regularly for good temperature compensation |
| */ |
| static bool dps310_measure_psr(const struct device *dev) |
| { |
| struct dps310_data *data = dev->data; |
| const struct dps310_cfg *config = dev->config; |
| |
| /* measure pressure */ |
| if (!dps310_trigger_pressure(dev)) { |
| return false; |
| } |
| |
| /* memory for pressure raw value */ |
| uint8_t value_raw[3]; |
| |
| /* read pressure raw values in one continuous read */ |
| int res = i2c_write_read_dt(&config->i2c, ®_ADDR_PSR_B2, 1, |
| &value_raw, sizeof(value_raw)); |
| if (res < 0) { |
| LOG_WRN("I2C error: %d", res); |
| return false; |
| } |
| |
| /* convert raw data to int */ |
| int32_t psr_raw = raw_to_int24(&value_raw[0]); |
| |
| dps310_scale_pressure(dev, data->raw_tmp, psr_raw); |
| |
| return true; |
| } |
| |
| /* perform a single temperature measurement */ |
| static bool dps310_measure_tmp(const struct device *dev) |
| { |
| struct dps310_data *data = dev->data; |
| const struct dps310_cfg *config = dev->config; |
| |
| /* measure temperature */ |
| if (!dps310_trigger_temperature(dev)) { |
| return false; |
| } |
| |
| /* memory for temperature raw value */ |
| uint8_t value_raw[3]; |
| |
| /* read temperature raw values in one continuous read */ |
| int res = i2c_write_read_dt(&config->i2c, ®_ADDR_TMP_B2, 1, |
| &value_raw, sizeof(value_raw)); |
| if (res < 0) { |
| LOG_WRN("I2C error: %d", res); |
| return false; |
| } |
| |
| /* convert raw data to int */ |
| data->raw_tmp = raw_to_int24(&value_raw[0]); |
| |
| dps310_scale_temperature(dev, data->raw_tmp); |
| |
| return true; |
| } |
| |
| /* Initialize the sensor and apply the configuration */ |
| static int dps310_init(const struct device *dev) |
| { |
| struct dps310_data *data = dev->data; |
| const struct dps310_cfg *config = dev->config; |
| |
| if (!device_is_ready(config->i2c.bus)) { |
| LOG_ERR("I2C bus device not ready"); |
| return -ENODEV; |
| } |
| |
| uint8_t product_id = 0; |
| int res = i2c_reg_read_byte_dt(&config->i2c, IFX_DPS310_REG_ADDR_PRODUCT_ID, &product_id); |
| |
| if (res < 0) { |
| LOG_ERR("No device found"); |
| return -EINVAL; |
| } |
| |
| if (product_id != IFX_DPS310_PRODUCT_ID) { |
| LOG_ERR("Device is not a DPS310"); |
| return -EINVAL; |
| } |
| |
| LOG_DBG("Init DPS310"); |
| /* give the sensor time to load the calibration data */ |
| k_sleep(K_MSEC(40)); |
| |
| /* wait for the sensor to load the calibration data */ |
| if (!poll_rdy(dev, REG_ADDR_MEAS_CFG, IFX_DPS310_REG_ADDR_MEAS_CFG_SELF_INIT_OK)) { |
| LOG_DBG("Sensor not ready"); |
| return -EIO; |
| } |
| |
| /* read calibration coefficients */ |
| uint8_t raw_coef[18] = { 0 }; |
| |
| res = i2c_write_read_dt(&config->i2c, ®_ADDR_CALIB_COEFF_0, 1, &raw_coef, 18); |
| if (res < 0) { |
| LOG_WRN("I2C error: %d", res); |
| return -EIO; |
| } |
| |
| /* convert calibration coefficients */ |
| dps310_calib_coeff_creation(raw_coef, &data->comp); |
| |
| /* |
| * check which temperature sensor was used for calibration and use it |
| * for measurements. |
| */ |
| uint8_t tmp_coef_srce = 0; |
| |
| res = i2c_write_read_dt(&config->i2c, ®_ADDR_COEF_SRCE, 1, &tmp_coef_srce, |
| sizeof(tmp_coef_srce)); |
| if (res < 0) { |
| LOG_WRN("I2C error: %d", res); |
| return -EIO; |
| } |
| |
| /* clear all other bits */ |
| tmp_coef_srce &= IFX_DPS310_COEF_SRCE_MASK; |
| |
| /* merge with temperature measurement configuration */ |
| tmp_coef_srce |= IFX_DPS310_TMP_CFG; |
| |
| /* set complete configuration in one write */ |
| const uint8_t config_seq[] = { |
| IFX_DPS310_REG_ADDR_PRS_CFG, /* start register address */ |
| IFX_DPS310_PSR_CFG, /* PSR_CFG */ |
| tmp_coef_srce, /* TMP_CFG */ |
| 0x00, /* MEAS_CFG */ |
| DPS310_CFG_REG /* CFG_REG */ |
| }; |
| |
| res = i2c_write_dt(&config->i2c, config_seq, sizeof(config_seq)); |
| if (res < 0) { |
| LOG_WRN("I2C error: %d", res); |
| return -EIO; |
| } |
| |
| dps310_hw_bug_fix(dev); |
| dps310_measure_tmp_psr(dev); |
| |
| LOG_DBG("Init OK"); |
| return 0; |
| } |
| |
| /* Do a measurement and fetch the data from the sensor */ |
| static int dps310_sample_fetch(const struct device *dev, |
| enum sensor_channel chan) |
| { |
| LOG_DBG("Fetching sample from DPS310"); |
| |
| switch (chan) { |
| case SENSOR_CHAN_AMBIENT_TEMP: |
| if (!dps310_measure_tmp(dev)) { |
| LOG_ERR("Failed to measure temperature"); |
| return -EIO; |
| } |
| break; |
| case SENSOR_CHAN_PRESS: |
| if (!dps310_measure_psr(dev)) { |
| LOG_ERR("Failed to measure pressure"); |
| return -EIO; |
| } |
| break; |
| case SENSOR_CHAN_ALL: |
| if (!dps310_measure_tmp_psr(dev)) { |
| LOG_ERR("Failed to measure temperature and pressure"); |
| return -EIO; |
| } |
| break; |
| default: |
| return -EINVAL; |
| } |
| |
| return 0; |
| } |
| |
| /* Get the measurement data */ |
| static int dps310_channel_get(const struct device *dev, |
| enum sensor_channel chan, |
| struct sensor_value *val) |
| { |
| struct dps310_data *data = dev->data; |
| |
| switch (chan) { |
| case SENSOR_CHAN_AMBIENT_TEMP: |
| val->val1 = data->tmp_val1; |
| val->val2 = data->tmp_val2; |
| break; |
| case SENSOR_CHAN_PRESS: |
| val->val1 = data->psr_val1; |
| val->val2 = data->psr_val2; |
| break; |
| default: |
| return -EINVAL; |
| } |
| |
| return 0; |
| } |
| |
| static const struct sensor_driver_api dps310_api_funcs = { |
| .sample_fetch = dps310_sample_fetch, |
| .channel_get = dps310_channel_get, |
| }; |
| |
| #define DPS310_DEFINE(inst) \ |
| static struct dps310_data dps310_data_##inst; \ |
| \ |
| static const struct dps310_cfg dps310_cfg_##inst = { \ |
| .i2c = I2C_DT_SPEC_INST_GET(inst), \ |
| }; \ |
| \ |
| DEVICE_DT_INST_DEFINE(inst, dps310_init, NULL, \ |
| &dps310_data_##inst, &dps310_cfg_##inst, POST_KERNEL, \ |
| CONFIG_SENSOR_INIT_PRIORITY, &dps310_api_funcs); \ |
| |
| DT_INST_FOREACH_STATUS_OKAY(DPS310_DEFINE) |