2014-11-25 17:29:19 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 PHYTEC Messtechnik GmbH
|
2017-06-09 22:17:02 +02:00
|
|
|
* 2017 HAW Hamburg
|
2014-11-25 17:29:19 +01:00
|
|
|
*
|
|
|
|
* 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 <j.fischer@phytec.de>
|
2015-01-20 10:05:17 +01:00
|
|
|
* @author Peter Kietzmann <peter.kietzmann@haw-hamburg.de>
|
2017-06-09 22:17:02 +02:00
|
|
|
* @author Sebastian Meiling <s@mlng.net>
|
2014-11-25 17:29:19 +01:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdbool.h>
|
2017-06-09 22:17:02 +02:00
|
|
|
#include <string.h>
|
2014-11-25 17:29:19 +01:00
|
|
|
#include <math.h>
|
2017-06-09 22:17:02 +02:00
|
|
|
|
|
|
|
#include "log.h"
|
2014-11-25 17:29:19 +01:00
|
|
|
#include "periph/i2c.h"
|
|
|
|
#include "tmp006.h"
|
2017-06-09 22:17:02 +02:00
|
|
|
#include "tmp006_regs.h"
|
2017-11-06 09:04:50 +01:00
|
|
|
#if TMP006_USE_LOW_POWER
|
|
|
|
#include "xtimer.h"
|
|
|
|
#endif
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
#define ENABLE_DEBUG (0)
|
2014-11-25 17:29:19 +01:00
|
|
|
#include "debug.h"
|
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
#define TMP006_CONFIG_RST (1 << 15)
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
#define TMP006_CONFIG_MOD_SHIFT (12U)
|
|
|
|
#define TMP006_CONFIG_MOD_MASK (0x7000)
|
|
|
|
#define TMP006_CONFIG_MOD(x) (((uint16_t)(((uint16_t)(x)) << TMP006_CONFIG_MOD_SHIFT))\
|
|
|
|
& TMP006_CONFIG_MOD_MASK)
|
|
|
|
#define TMP006_CONFIG_MOD_CC (0x07)
|
|
|
|
#define TMP006_CONFIG_MOD_OFF (0x00)
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
#define TMP006_CONFIG_CR_SHIFT (9U)
|
|
|
|
#define TMP006_CONFIG_CR_MASK (0x0E00)
|
|
|
|
#define TMP006_CONFIG_CR(x) (((uint16_t)(((uint16_t)(x)) << TMP006_CONFIG_CR_SHIFT))\
|
|
|
|
& TMP006_CONFIG_CR_MASK)
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
#define TMP006_CONFIG_DRDY_PIN_EN (1 << 8)
|
|
|
|
#define TMP006_CONFIG_DRDY (1 << 7)
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
#define TMP006_MID_VALUE (0x5449) /**< Manufacturer ID */
|
|
|
|
#define TMP006_DID_VALUE (0x0067) /**< Device ID */
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
#define BUS (dev->p.i2c)
|
|
|
|
#define ADDR (dev->p.addr)
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
int tmp006_init(tmp006_t *dev, const tmp006_params_t *params)
|
2014-11-25 17:29:19 +01:00
|
|
|
{
|
2017-06-09 22:17:02 +02:00
|
|
|
/* check parameters */
|
|
|
|
assert(dev && params);
|
|
|
|
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg[2];
|
2014-11-25 17:29:19 +01:00
|
|
|
uint16_t tmp;
|
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
/* initialize the device descriptor */
|
|
|
|
memcpy(&dev->p, params, sizeof(tmp006_params_t));
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
if (dev->p.rate > TMP006_CONFIG_CR_AS16) {
|
|
|
|
LOG_ERROR("tmp006_init: invalid conversion rate!\n");
|
|
|
|
return -TMP006_ERROR_CONF;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
/* test device id */
|
2018-06-28 22:32:02 +02:00
|
|
|
i2c_acquire(BUS);
|
|
|
|
if (i2c_read_regs(BUS, ADDR, TMP006_REGS_DEVICE_ID, reg, 2, 0) < 0) {
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
LOG_ERROR("tmp006_init: error reading device ID!\n");
|
|
|
|
return -TMP006_ERROR_BUS;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
2017-06-09 22:17:02 +02:00
|
|
|
tmp = ((uint16_t)reg[0] << 8) | reg[1];
|
|
|
|
if (tmp != TMP006_DID_VALUE) {
|
|
|
|
return -TMP006_ERROR_DEV;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
2017-06-09 22:17:02 +02:00
|
|
|
/* set conversion rate */
|
|
|
|
tmp = TMP006_CONFIG_CR(dev->p.rate);
|
2016-09-30 23:01:46 +02:00
|
|
|
reg[0] = (tmp >> 8);
|
|
|
|
reg[1] = tmp;
|
2018-06-28 22:32:02 +02:00
|
|
|
if (i2c_write_regs(BUS, ADDR, TMP006_REGS_CONFIG, reg, 2, 0) < 0) {
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
LOG_ERROR("tmp006_init: error setting conversion rate!\n");
|
|
|
|
return -TMP006_ERROR_BUS;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
|
|
|
|
return TMP006_OK;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
int tmp006_reset(const tmp006_t *dev)
|
2014-11-25 17:29:19 +01:00
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg[2];
|
2014-11-25 17:29:19 +01:00
|
|
|
uint16_t tmp = TMP006_CONFIG_RST;
|
2016-09-30 23:01:46 +02:00
|
|
|
reg[0] = (tmp >> 8);
|
|
|
|
reg[1] = tmp;
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2015-01-20 10:05:17 +01:00
|
|
|
/* Acquire exclusive access to the bus. */
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_acquire(BUS);
|
2018-06-28 22:32:02 +02:00
|
|
|
if (i2c_write_regs(BUS, ADDR, TMP006_REGS_CONFIG, reg, 2, 0) < 0) {
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
return -TMP006_ERROR_BUS;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
return TMP006_OK;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
int tmp006_set_active(const tmp006_t *dev)
|
2014-11-25 17:29:19 +01:00
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg[2];
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_acquire(BUS);
|
2018-06-28 22:32:02 +02:00
|
|
|
if (i2c_read_regs(BUS, ADDR, TMP006_REGS_CONFIG, reg, 2, 0) < 0) {
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
return -TMP006_ERROR_BUS;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
|
|
|
|
2016-09-30 23:01:46 +02:00
|
|
|
reg[0] |= (TMP006_CONFIG_MOD(TMP006_CONFIG_MOD_CC) >> 8);
|
2018-06-28 22:32:02 +02:00
|
|
|
if (i2c_write_regs(BUS, ADDR, TMP006_REGS_CONFIG, reg, 2, 0) < 0) {
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
return -TMP006_ERROR_BUS;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
return TMP006_OK;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
int tmp006_set_standby(const tmp006_t *dev)
|
2014-11-25 17:29:19 +01:00
|
|
|
{
|
2016-09-30 23:01:46 +02:00
|
|
|
uint8_t reg[2];
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_acquire(BUS);
|
2018-06-28 22:32:02 +02:00
|
|
|
if (i2c_read_regs(BUS, ADDR, TMP006_REGS_CONFIG, reg, 2, 0) < 0) {
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
return -TMP006_ERROR_BUS;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
|
|
|
|
2016-09-30 23:01:46 +02:00
|
|
|
reg[0] &= ~(TMP006_CONFIG_MOD(TMP006_CONFIG_MOD_CC) >> 8);
|
2018-06-28 22:32:02 +02:00
|
|
|
if (i2c_write_regs(BUS, ADDR, TMP006_REGS_CONFIG, reg, 2, 0) < 0) {
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
return -TMP006_ERROR_BUS;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
return TMP006_OK;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
int tmp006_read(const tmp006_t *dev, int16_t *rawv, int16_t *rawt, uint8_t *drdy)
|
2014-11-25 17:29:19 +01:00
|
|
|
{
|
2017-06-09 22:17:02 +02:00
|
|
|
uint8_t reg[2];
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_acquire(BUS);
|
2014-11-25 17:29:19 +01:00
|
|
|
/* Register bytes are sent MSB first. */
|
2018-06-28 22:32:02 +02:00
|
|
|
if (i2c_read_regs(BUS, ADDR, TMP006_REGS_CONFIG, reg, 2, 0) < 0) {
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
return -TMP006_ERROR_BUS;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
*drdy = reg[1] & (TMP006_CONFIG_DRDY);
|
2014-11-25 17:29:19 +01:00
|
|
|
if (!(*drdy)) {
|
2017-06-09 22:17:02 +02:00
|
|
|
LOG_DEBUG("tmp006_read: conversion in progress!\n");
|
|
|
|
return -TMP006_ERROR;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_acquire(BUS);
|
2018-06-28 22:32:02 +02:00
|
|
|
if (i2c_read_regs(BUS, ADDR, TMP006_REGS_V_OBJECT, reg, 2, 0) < 0) {
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
return -TMP006_ERROR_BUS;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
*rawv = ((uint16_t)reg[0] << 8) | reg[1];
|
2014-11-25 17:29:19 +01:00
|
|
|
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_acquire(BUS);
|
2018-06-28 22:32:02 +02:00
|
|
|
if (i2c_read_regs(BUS, ADDR, TMP006_REGS_T_AMBIENT, reg, 2, 0) < 0) {
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
return -TMP006_ERROR_BUS;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
2017-06-09 22:17:02 +02:00
|
|
|
i2c_release(BUS);
|
|
|
|
*rawt = ((uint16_t)reg[0] << 8) | reg[1];
|
|
|
|
return TMP006_OK;
|
2014-11-25 17:29:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
2017-06-09 23:03:54 +02:00
|
|
|
|
2017-06-20 17:32:45 +02:00
|
|
|
int tmp006_read_temperature(const tmp006_t *dev, int16_t *ta, int16_t *to)
|
2017-06-09 23:03:54 +02:00
|
|
|
{
|
2017-11-06 09:04:50 +01:00
|
|
|
|
|
|
|
uint8_t drdy;
|
|
|
|
#if (!TMP006_USE_RAW_VALUES)
|
2017-06-09 23:03:54 +02:00
|
|
|
int16_t rawtemp, rawvolt;
|
|
|
|
float tamb, tobj;
|
2017-11-06 09:04:50 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if TMP006_USE_LOW_POWER
|
|
|
|
if (tmp006_set_active(dev)) {
|
|
|
|
return TMP006_ERROR;
|
|
|
|
}
|
|
|
|
xtimer_usleep(TMP006_CONVERSION_TIME);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if TMP006_USE_RAW_VALUES
|
|
|
|
tmp006_read(dev, to, ta, &drdy);
|
|
|
|
|
|
|
|
if (!drdy) {
|
|
|
|
return TMP006_ERROR;
|
|
|
|
}
|
|
|
|
#else
|
2017-06-09 23:03:54 +02:00
|
|
|
tmp006_read(dev, &rawvolt, &rawtemp, &drdy);
|
|
|
|
|
|
|
|
if (!drdy) {
|
|
|
|
return TMP006_ERROR;
|
|
|
|
}
|
|
|
|
tmp006_convert(rawvolt, rawtemp, &tamb, &tobj);
|
|
|
|
*ta = (int16_t)(tamb*100);
|
|
|
|
*to = (int16_t)(tobj*100);
|
2017-11-06 09:04:50 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if TMP006_USE_LOW_POWER
|
|
|
|
if (tmp006_set_standby(dev)) {
|
|
|
|
return TMP006_ERROR;
|
|
|
|
}
|
|
|
|
#endif
|
2017-06-09 23:03:54 +02:00
|
|
|
|
|
|
|
return TMP006_OK;
|
|
|
|
}
|