/* * Copyright (C) 2014 PHYTEC Messtechnik GmbH * * 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_tmp006 * @{ * * @file * @brief Driver for the TI TMP006 Infrared Thermopile Sensor. * * @author Johann Fischer * @author Peter Kietzmann * * @} */ #include #include #include #include "periph/i2c.h" #include "tmp006.h" #define ENABLE_DEBUG (0) #include "debug.h" #define TMP006_V_OBJECT 0x00 /**< Sensor Voltage Register */ #define TMP006_T_AMBIENT 0x01 /**< Ambient Temperature Register */ #define TMP006_CONFIG 0x02 /**< Configuration Register */ #define TMP006_MANUFACTURER_ID 0xFE /**< Manufacturer ID Register */ #define TMP006_DEVICE_ID 0xFF /**< Device ID Register */ #define TMP006_CONFIG_RST (1 << 15) #define TMP006_CONFIG_MOD_SHIFT 12 #define TMP006_CONFIG_MOD_MASK 0x7000 #define TMP006_CONFIG_MOD(x) (((uint16_t)(((uint16_t)(x))<i2c); status = i2c_read_regs(dev->i2c, dev->addr, TMP006_DEVICE_ID, reg, 2); if (status != 2) { /* Release the bus for other threads. */ i2c_release(dev->i2c); return -1; } i2c_release(dev->i2c); tmp = ((uint16_t)reg[0] << 8) | reg[1]; if (tmp != TMP006_DID_VALUE) { return -1; } return 0; } int tmp006_init(tmp006_t *dev, i2c_t i2c, uint8_t address, uint8_t conv_rate) { int status; char reg[2]; /* write device descriptor */ dev->i2c = i2c; dev->addr = address; dev->initialized = false; if (conv_rate > TMP006_CONFIG_CR_AS16) { return -1; } i2c_acquire(dev->i2c); /* initialize the I2C bus */ status = i2c_init_master(i2c, I2C_SPEED); if (status < 0) { i2c_release(dev->i2c); return -2; } i2c_release(dev->i2c); if (tmp006_test(dev)) { return -3; } uint16_t tmp = TMP006_CONFIG_CR(conv_rate); reg[0] = (uint8_t)(tmp >> 8); reg[1] = (uint8_t)tmp; /* Acquire exclusive access to the bus. */ i2c_acquire(dev->i2c); status = i2c_write_regs(dev->i2c, dev->addr, TMP006_CONFIG, reg, 2); if (status != 2) { i2c_release(dev->i2c); return -4; } i2c_release(dev->i2c); dev->initialized = true; return 0; } int tmp006_reset(tmp006_t *dev) { int status; char reg[2]; uint16_t tmp = TMP006_CONFIG_RST; reg[0] = (uint8_t)(tmp >> 8); reg[1] = (uint8_t)tmp; dev->initialized = false; /* Acquire exclusive access to the bus. */ i2c_acquire(dev->i2c); status = i2c_write_regs(dev->i2c, dev->addr, TMP006_CONFIG, reg, 2); if (status != 2) { i2c_release(dev->i2c); return -1; } i2c_release(dev->i2c); return 0; } int tmp006_set_active(tmp006_t *dev) { int status; char reg[2]; if (dev->initialized == false) { return -1; } i2c_acquire(dev->i2c); status = i2c_read_regs(dev->i2c, dev->addr, TMP006_CONFIG, reg, 2); if (status != 2) { i2c_release(dev->i2c); return -1; } reg[0] |= (uint8_t)(TMP006_CONFIG_MOD(TMP006_CONFIG_MOD_CC) >> 8); status = i2c_write_regs(dev->i2c, dev->addr, TMP006_CONFIG, reg, 2); if (status != 2) { i2c_release(dev->i2c); return -1; } i2c_release(dev->i2c); return 0; } int tmp006_set_standby(tmp006_t *dev) { int status; char reg[2]; i2c_acquire(dev->i2c); status = i2c_read_regs(dev->i2c, dev->addr, TMP006_CONFIG, reg, 2); if (status != 2) { i2c_release(dev->i2c); return -1; } i2c_release(dev->i2c); reg[0] &= ~(uint8_t)(TMP006_CONFIG_MOD(TMP006_CONFIG_MOD_CC) >> 8); i2c_acquire(dev->i2c); status = i2c_write_regs(dev->i2c, dev->addr, TMP006_CONFIG, reg, 2); if (status != 2) { i2c_release(dev->i2c); return -1; } i2c_release(dev->i2c); return 0; } int tmp006_read(tmp006_t *dev, int16_t *rawv, int16_t *rawt, uint8_t *drdy) { int status; char buf[2]; if (dev->initialized == false) { return -1; } i2c_acquire(dev->i2c); /* Register bytes are sent MSB first. */ status = i2c_read_regs(dev->i2c, dev->addr, TMP006_CONFIG, buf, 2); if (status != 2) { i2c_release(dev->i2c); return -1; } i2c_release(dev->i2c); *drdy = buf[1] & (uint8_t)(TMP006_CONFIG_DRDY); if (!(*drdy)) { /* conversion in progress */ return -1; } i2c_acquire(dev->i2c); status = i2c_read_regs(dev->i2c, dev->addr, TMP006_V_OBJECT, buf, 2); if (status != 2) { i2c_release(dev->i2c); return -1; } i2c_release(dev->i2c); *rawv = ((uint16_t)buf[0] << 8) | buf[1]; i2c_acquire(dev->i2c); status = i2c_read_regs(dev->i2c, dev->addr, TMP006_T_AMBIENT, buf, 2); if (status != 2) { i2c_release(dev->i2c); return -1; } i2c_release(dev->i2c); *rawt = ((uint16_t)buf[0] << 8) | buf[1]; return 0; } void tmp006_convert(int16_t rawv, int16_t rawt, float *tamb, float *tobj) { /* calculate die temperature */ *tamb = (double)rawt / 128.0; /* die temperature in Kelvin */ double tdie_k = *tamb + 273.15; /* calculate sensor voltage */ double sens_v = (double)rawv * TMP006_CCONST_LSB_SIZE; double tdiff = tdie_k - TMP006_CCONST_TREF; double tdiff_pow2 = pow(tdiff, 2); double s = TMP006_CCONST_S0 * (1 + TMP006_CCONST_A1 * tdiff + TMP006_CCONST_A2 * tdiff_pow2); double v_os = TMP006_CCONST_B0 + TMP006_CCONST_B1 * tdiff + TMP006_CCONST_B2 * tdiff_pow2; double f_obj = (sens_v - v_os) + TMP006_CCONST_C2 * pow((sens_v - v_os), 2); double t = pow(pow(tdie_k, 4) + (f_obj / s), 0.25); /* calculate object temperature in Celsius */ *tobj = (t - 273.15); }