1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/drivers/ds18/ds18.c
Guillaume Gonnet 1783b3da55 drivers/ds18: fix ds18_read with negative temperatures
When DS18 returns 0xFE6F (corresponding to -25.0625 °C), `ds18_read` returns
13877 whereas it should return -2506. This commit fixes this issue.
2019-05-16 20:12:22 +02:00

222 lines
5.0 KiB
C

/*
* Copyright (C) 2017 Frits Kuipers
* 2018 HAW Hamburg
*
* 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_ds18
* @{
*
* @file
* @brief Device driver implementation for the Maxim Integrated DS1822 and
* DS18B20 temperature sensors.
*
* @author Frits Kuipers <frits.kuipers@gmail.com>
* @author Leandro Lanzieri <leandro.lanzieri@haw-hamburg.de>
* @}
*/
#include "ds18.h"
#include "ds18_internal.h"
#include "log.h"
#include "periph/gpio.h"
#include "xtimer.h"
#define ENABLE_DEBUG (0)
#include "debug.h"
static void ds18_low(const ds18_t *dev)
{
/* Set gpio as output and clear pin */
gpio_init(dev->params.pin, GPIO_OUT);
gpio_clear(dev->params.pin);
}
static void ds18_release(const ds18_t *dev)
{
/* Init pin as input */
gpio_init(dev->params.pin, dev->params.in_mode);
}
static void ds18_write_bit(const ds18_t *dev, uint8_t bit)
{
/* Initiate write slot */
ds18_low(dev);
/* Release pin when bit==1 */
if (bit) {
ds18_release(dev);
}
/* Wait for slot to end */
xtimer_usleep(DS18_DELAY_SLOT);
ds18_release(dev);
xtimer_usleep(1);
}
static int ds18_read_bit(const ds18_t *dev, uint8_t *bit)
{
/* Initiate read slot */
ds18_low(dev);
ds18_release(dev);
#if defined(MODULE_DS18_OPTIMIZED)
xtimer_usleep(DS18_SAMPLE_TIME);
*bit = gpio_read(dev->params.pin);
xtimer_usleep(DS18_DELAY_R_RECOVER);
return DS18_OK;
#else
uint32_t start, measurement = 0;
/* Measure time low of device pin, timeout after slot time*/
start = xtimer_now_usec();
while (!gpio_read(dev->params.pin) && measurement < DS18_DELAY_SLOT) {
measurement = xtimer_now_usec() - start;
}
/* If there was a timeout return error */
if (measurement >= DS18_DELAY_SLOT) {
return DS18_ERROR;
}
/* When gpio was low for less than the sample time, bit is high*/
*bit = measurement < DS18_SAMPLE_TIME;
/* Wait for slot to end */
xtimer_usleep(DS18_DELAY_SLOT - measurement);
return DS18_OK;
#endif
}
static int ds18_read_byte(const ds18_t *dev, uint8_t *byte)
{
uint8_t bit = 0;
*byte = 0;
for (int i = 0; i < 8; i++) {
if (ds18_read_bit(dev, &bit) == DS18_OK) {
*byte |= (bit << i);
}
else {
return DS18_ERROR;
}
}
return DS18_OK;
}
static void ds18_write_byte(const ds18_t *dev, uint8_t byte)
{
for (int i = 0; i < 8; i++) {
ds18_write_bit(dev, byte & (0x01 << i));
}
}
static int ds18_reset(const ds18_t *dev)
{
int res;
/* Line low and sleep the reset delay */
ds18_low(dev);
xtimer_usleep(DS18_DELAY_RESET);
/* Release and wait for the presence response */
ds18_release(dev);
xtimer_usleep(DS18_DELAY_PRESENCE);
/* Check device presence */
res = gpio_read(dev->params.pin);
/* Sleep for reset delay */
xtimer_usleep(DS18_DELAY_RESET);
return res;
}
int ds18_trigger(const ds18_t *dev)
{
int res;
res = ds18_reset(dev);
if (res) {
return DS18_ERROR;
}
/* Please note that this command triggers a conversion on all devices
* connected to the bus. */
ds18_write_byte(dev, DS18_CMD_SKIPROM);
ds18_write_byte(dev, DS18_CMD_CONVERT);
return DS18_OK;
}
int ds18_read(const ds18_t *dev, int16_t *temperature)
{
int res;
uint8_t b1 = 0, b2 = 0;
DEBUG("[DS18] Reset and read scratchpad\n");
res = ds18_reset(dev);
if (res) {
return DS18_ERROR;
}
ds18_write_byte(dev, DS18_CMD_SKIPROM);
ds18_write_byte(dev, DS18_CMD_RSCRATCHPAD);
if (ds18_read_byte(dev, &b1) != DS18_OK) {
DEBUG("[DS18] Error reading temperature byte 1\n");
return DS18_ERROR;
}
DEBUG("[DS18] Received byte: 0x%02x\n", b1);
if (ds18_read_byte(dev, &b2) != DS18_OK) {
DEBUG("[DS18] Error reading temperature byte 2\n");
return DS18_ERROR;
}
DEBUG("[DS18] Received byte: 0x%02x\n", b2);
int32_t measurement = ((int16_t)(b2 << 8 | b1) * 625);
*temperature = (int16_t)(measurement / 100);
return DS18_OK;
}
int ds18_get_temperature(const ds18_t *dev, int16_t *temperature)
{
DEBUG("[DS18] Convert T\n");
if (ds18_trigger(dev)) {
return DS18_ERROR;
}
DEBUG("[DS18] Wait for convert T\n");
xtimer_usleep(DS18_DELAY_CONVERT);
return ds18_read(dev, temperature);
}
int ds18_init(ds18_t *dev, const ds18_params_t *params)
{
int res;
dev->params = *params;
/* Deduct the input mode from the output mode. If pull-up resistors are
* used for output then will be used for input as well. */
dev->params.in_mode = (dev->params.out_mode == GPIO_OD_PU) ? GPIO_IN_PU : GPIO_IN;
/* Initialize the device and the pin */
res = gpio_init(dev->params.pin, dev->params.in_mode) == 0 ? DS18_OK : DS18_ERROR;
return res;
}