2017-10-26 10:21:12 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 RWTH-Aachen
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2022-02-25 17:17:11 +01:00
|
|
|
* @ingroup drivers_shtcx
|
2017-10-26 10:21:12 +02:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
2022-02-25 17:17:11 +01:00
|
|
|
* @brief Device driver implementation for the Sensirion SHTCX temperature
|
2017-10-26 10:21:12 +02:00
|
|
|
* and humidity sensor.
|
|
|
|
*
|
|
|
|
* @author Steffen Robertz <steffen.robertz@rwth-aachen.de>
|
|
|
|
* @author Josua Arndt <jarndt@ias.rwth-aachen.de>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
#include <string.h>
|
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
#include "log.h"
|
|
|
|
#include "assert.h"
|
|
|
|
#include "checksum/crc8.h"
|
2022-02-25 17:17:11 +01:00
|
|
|
#include "shtcx.h"
|
|
|
|
#include "shtcx_regs.h"
|
2017-10-26 10:21:12 +02:00
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2017-10-26 10:21:12 +02:00
|
|
|
#include "debug.h"
|
|
|
|
|
2022-02-25 17:17:11 +01:00
|
|
|
int8_t shtcx_init(shtcx_t *const dev, const shtcx_params_t *params)
|
2017-10-26 10:21:12 +02:00
|
|
|
{
|
|
|
|
uint16_t id;
|
|
|
|
|
|
|
|
/* check for a valid device descriptor and parameters */
|
|
|
|
assert(dev && params);
|
|
|
|
/* copy settings into the device descriptor */
|
|
|
|
dev->params = *params;
|
2022-02-25 17:17:11 +01:00
|
|
|
/* Verify the connection by reading the SHTCX's ID and checking its value */
|
|
|
|
if (shtcx_id(dev, &id) != SHTCX_OK) {
|
|
|
|
return SHTCX_ERROR_BUS;
|
2017-10-26 10:21:12 +02:00
|
|
|
}
|
2022-02-25 17:17:11 +01:00
|
|
|
else if ((id & 0x3F) != SHTCX_ID) {
|
|
|
|
return SHTCX_ERROR_CRC;
|
2017-10-26 10:21:12 +02:00
|
|
|
}
|
|
|
|
else {
|
2022-02-25 17:17:11 +01:00
|
|
|
return SHTCX_OK;
|
2017-10-26 10:21:12 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-25 17:17:11 +01:00
|
|
|
int8_t shtcx_read(const shtcx_t *dev, uint16_t *rel_humidity,
|
2017-10-26 10:21:12 +02:00
|
|
|
int16_t *temperature)
|
|
|
|
{
|
|
|
|
uint8_t received[6];
|
|
|
|
/* Build and issue the measurement command */
|
|
|
|
uint8_t cmd[] =
|
2022-02-25 17:17:11 +01:00
|
|
|
{ SHTCX_MEASURE_CLOCK_STRETCHING_TEMP_HIGH,
|
|
|
|
SHTCX_MEASURE_CLOCK_STRETCHING_TEMP_LOW };
|
2017-10-26 10:21:12 +02:00
|
|
|
|
|
|
|
i2c_acquire(dev->params.i2c_dev);
|
|
|
|
|
|
|
|
if (i2c_write_bytes(dev->params.i2c_dev, dev->params.i2c_addr, cmd, 2, 0)) {
|
2022-02-25 17:17:11 +01:00
|
|
|
return SHTCX_ERROR_BUS;
|
2017-10-26 10:21:12 +02:00
|
|
|
}
|
|
|
|
/* Receive the measurement */
|
|
|
|
/* 16 bit Temperature
|
|
|
|
* 8 bit CRC temp
|
|
|
|
* 16 Bit Absolute Humidity
|
|
|
|
* 8 bit CRC Hum
|
|
|
|
*/
|
|
|
|
if (i2c_read_bytes(dev->params.i2c_dev, dev->params.i2c_addr, received, 6,
|
|
|
|
0)) {
|
2022-02-25 17:17:11 +01:00
|
|
|
return SHTCX_ERROR_BUS;
|
2017-10-26 10:21:12 +02:00
|
|
|
}
|
|
|
|
i2c_release(dev->params.i2c_dev);
|
|
|
|
|
|
|
|
if (rel_humidity != NULL) {
|
|
|
|
uint32_t abs_humidity = ((received[3] << 8) | received[4]);
|
|
|
|
/* 10000 * ( abs_humidity /65536) */
|
|
|
|
*rel_humidity = (10000 * abs_humidity) >> 16;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (temperature != NULL) {
|
|
|
|
uint16_t temp_f = ((received[0] << 8) | received[1]);
|
|
|
|
/* calculate the relative humidity and convert the temperature to centi °C */
|
|
|
|
/* (175.0 * 100 * temp_f / 65536) - 45 ; */
|
|
|
|
*temperature = ((17500 * (uint32_t)temp_f) >> 16) - 4500;
|
|
|
|
}
|
|
|
|
|
2022-02-25 17:17:11 +01:00
|
|
|
if (!((crc8(&received[0], 2, SHTCX_CRC, 0xFF) == received[2]) &&
|
|
|
|
(crc8(&received[3], 2, SHTCX_CRC, 0xFF) == received[5]))) {
|
2017-10-26 10:21:12 +02:00
|
|
|
/* crc check failed */
|
|
|
|
DEBUG("CRC Error\n");
|
2022-02-25 17:17:11 +01:00
|
|
|
return SHTCX_ERROR_CRC;
|
2017-10-26 10:21:12 +02:00
|
|
|
}
|
|
|
|
DEBUG("CRC Passed! \n");
|
2022-02-25 17:17:11 +01:00
|
|
|
return SHTCX_OK;
|
2017-10-26 10:21:12 +02:00
|
|
|
}
|
|
|
|
|
2022-02-25 17:17:11 +01:00
|
|
|
int8_t shtcx_id(const shtcx_t *dev, uint16_t *id)
|
2017-10-26 10:21:12 +02:00
|
|
|
{
|
|
|
|
/* Build and send measurement command */
|
2022-02-25 17:17:11 +01:00
|
|
|
uint8_t data[] = { SHTCX_COMMAND_ID_HIGH, SHTCX_COMMAND_ID_LOW };
|
2017-10-26 10:21:12 +02:00
|
|
|
|
|
|
|
i2c_acquire(dev->params.i2c_dev);
|
|
|
|
if (i2c_write_bytes(dev->params.i2c_dev, dev->params.i2c_addr, data, 2,
|
|
|
|
0)) {
|
2022-02-25 17:17:11 +01:00
|
|
|
return SHTCX_ERROR_BUS;
|
2017-10-26 10:21:12 +02:00
|
|
|
}
|
|
|
|
/* receive ID and check if the send and receive commands were successful */
|
|
|
|
if (i2c_read_bytes(dev->params.i2c_dev, dev->params.i2c_addr, data, 2, 0)) {
|
2022-02-25 17:17:11 +01:00
|
|
|
return SHTCX_ERROR_BUS;
|
2017-10-26 10:21:12 +02:00
|
|
|
}
|
|
|
|
i2c_release(dev->params.i2c_dev);
|
|
|
|
|
|
|
|
/* Save ID in device descriptor */
|
|
|
|
*id = (data[0] << 8) | data[1];
|
2022-02-25 17:17:11 +01:00
|
|
|
return SHTCX_OK;
|
2017-10-26 10:21:12 +02:00
|
|
|
}
|
|
|
|
|
2022-02-25 17:17:11 +01:00
|
|
|
int8_t shtcx_reset(const shtcx_t *const dev)
|
2017-10-26 10:21:12 +02:00
|
|
|
{
|
|
|
|
/* Build and issue the reset command */
|
2022-02-25 17:17:11 +01:00
|
|
|
uint8_t data[] = { SHTCX_COMMAND_RESET_HIGH, SHTCX_COMMAND_RESET_LOW };
|
2017-10-26 10:21:12 +02:00
|
|
|
|
|
|
|
i2c_acquire(dev->params.i2c_dev);
|
|
|
|
if (i2c_write_bytes(dev->params.i2c_dev, dev->params.i2c_addr, data, 2,
|
|
|
|
0)) {
|
2022-02-25 17:17:11 +01:00
|
|
|
return SHTCX_ERROR_BUS;
|
2017-10-26 10:21:12 +02:00
|
|
|
}
|
|
|
|
i2c_release(dev->params.i2c_dev);
|
2022-02-25 17:17:11 +01:00
|
|
|
return SHTCX_OK;
|
2017-10-26 10:21:12 +02:00
|
|
|
}
|