diff --git a/drivers/Makefile.dep b/drivers/Makefile.dep index 7e8a3cb634..0b6ace63c6 100644 --- a/drivers/Makefile.dep +++ b/drivers/Makefile.dep @@ -82,6 +82,10 @@ ifneq (,$(filter io1_xplained,$(USEMODULE))) USEMODULE += at30tse75x endif +ifneq (,$(filter jc42,$(USEMODULE))) + FEATURES_REQUIRED += periph_i2c +endif + ifneq (,$(filter kw2xrf,$(USEMODULE))) USEMODULE += ieee802154 USEMODULE += netif diff --git a/drivers/Makefile.include b/drivers/Makefile.include index a7bee4e6c4..7c1e2836cf 100644 --- a/drivers/Makefile.include +++ b/drivers/Makefile.include @@ -64,6 +64,9 @@ endif ifneq (,$(filter bmp180,$(USEMODULE))) USEMODULE_INCLUDES += $(RIOTBASE)/drivers/bmp180/include endif +ifneq (,$(filter jc42,$(USEMODULE))) + USEMODULE_INCLUDES += $(RIOTBASE)/drivers/jc42/include +endif ifneq (,$(filter cc2420,$(USEMODULE))) USEMODULE_INCLUDES += $(RIOTBASE)/drivers/cc2420/include endif diff --git a/drivers/include/jc42.h b/drivers/include/jc42.h new file mode 100644 index 0000000000..a11384b3a9 --- /dev/null +++ b/drivers/include/jc42.h @@ -0,0 +1,125 @@ +/* + * Copyright (C) 2017 Koen Zandberg + * + * 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. + */ + +/** + * @defgroup drivers_jc42 JC42 compliant temperature sensor driver + * @ingroup drivers_sensors + * + * @brief JC42 compliant temperature sensor driver + * + * ## Description + * + * The connection between the MCU and jc42 compliant temperature sensors is + * based on a I2C-interface. There are multiple jc42 compliant temperature + * sensors available such as the mcp9804 and the MAX6604. This driver reads the + * temperature from these sensors. The sensors support a alarm wire, but this + * is not supported by this driver. + * + * @{ + * + * @file + * @brief Driver for jc42 compliant temperature sensors + * + * @author Koen Zandberg + */ + +#ifndef JC42_TEMP_H_ +#define JC42_TEMP_H_ + +#include "periph/i2c.h" +#include "saul.h" + + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @name jc42 status return codes + * @{ + */ +#define JC42_OK (0) +#define JC42_NOI2C (-1) +#define JC42_NODEV (-2) +/** @} */ + + +/** + * @brief Device descriptor for a jc42 device + */ +typedef struct { + i2c_t i2c; /**< I2C device that sensor is connected to */ + uint8_t addr; /**< I2C address of this particular sensor */ +} jc42_t; + +/** + * @brief Device initialization parameters + */ +typedef struct { + i2c_t i2c; /**< I2C device that sensor is connected to */ + i2c_speed_t speed; /**< I2C device speed */ + uint8_t addr; /**< Configured address of the sensor */ +} jc42_params_t; + +/** + * @brief export SAUL endpoint + */ +extern const saul_driver_t jc42_temperature_saul_driver; + +/** + * @brief Initialize a jc42 device + * + * @param[out] dev device descriptor + * @param[in] params jc42 initialization struct + * + * + * @return 0 on success + * @return -1 on error + */ +int jc42_init(jc42_t* dev, jc42_params_t* params); + +/** + * @brief Get content of configuration register + * + * @param[in] dev device descriptor + * @param[out] data buffer where config register will be written to + * + * @return 0 on success + * @return -1 on error + */ +int jc42_get_config(jc42_t* dev, uint16_t* data); + +/** + * @brief Set content of configuration register + * + * @param[in] dev device descriptor + * @param[in] data new value for configuration register + * + * @return 0 on success + * @return -1 on error + */ +int jc42_set_config(jc42_t* dev, uint16_t data); + + +/** + * @brief Get measured temperature + * + * @param[in] dev device descriptor + * @param[out] temperature uint16_t buffer where temperature will be written to in centi-degree + * + * @return 0 on success + * @return -1 on error + */ +int jc42_get_temperature(jc42_t* dev, int16_t* temperature); + +#ifdef __cplusplus +} +#endif + +/** @} */ +#endif /* JC42_TEMP_H_ */ diff --git a/drivers/jc42/Makefile b/drivers/jc42/Makefile new file mode 100644 index 0000000000..d77abd36c6 --- /dev/null +++ b/drivers/jc42/Makefile @@ -0,0 +1,2 @@ +MODULE = jc42 +include $(RIOTBASE)/Makefile.base diff --git a/drivers/jc42/include/jc42_internal.h b/drivers/jc42/include/jc42_internal.h new file mode 100644 index 0000000000..dbafae7028 --- /dev/null +++ b/drivers/jc42/include/jc42_internal.h @@ -0,0 +1,50 @@ +/* + * Copyright (C) 2017 Koen Zandberg + * + * 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_sensors + * @{ + * + * @file + * @brief Internal addresses, registers, constants for jc42 sensors. + * + * @author Koen Zandberg + */ + +#ifndef JC42_INTERNAL_H_ +#define JC42_INTERNAL_H_ + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @name JC42 register addresses + * @{ + */ +#define JC42_REG_CAP (0x00) +#define JC42_REG_CONFIG (0x01) +#define JC42_REG_TEMP_UPPER (0x02) +#define JC42_REG_TEMP_LOWER (0x03) +#define JC42_REG_TEMP_CRITICAL (0x04) +#define JC42_REG_TEMP (0x05) +#define JC42_REG_MANID (0x06) +#define JC42_REG_DEVICEID (0x07) + +#define jc42_BUS_FREE_TIME_US (1) +/** @} */ + + +#ifdef __cplusplus +} +#endif + +/** @} */ + + +#endif /* JC42_INTERNAL_H_ */ diff --git a/drivers/jc42/include/jc42_params.h b/drivers/jc42/include/jc42_params.h new file mode 100644 index 0000000000..687ea72b2b --- /dev/null +++ b/drivers/jc42/include/jc42_params.h @@ -0,0 +1,65 @@ +/* + * Copyright (C) 2017 Koen Zandberg + * + * 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_jc42 + * + * @{ + * @file + * @brief Default configuration for jc42 + * + * @author Koen Zandberg + */ + +#ifndef JC42_PARAMS_H +#define JC42_PARAMS_H + +#include "jc42.h" +#include "periph/i2c.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @brief Set default configuration parameters for the JC42 + * @{ + */ +#ifndef JC42_PARAM_I2C_DEV +#define JC42_PARAM_I2C_DEV (0) +#endif +#ifndef JC42_PARAM_ADDR +#define JC42_PARAM_ADDR (0x18) +#endif +#ifndef JC42_PARAM_SPEED +#define JC42_PARAM_SPEED I2C_SPEED_FAST +#endif + +#define JC42_PARAMS_DEFAULT {.i2c = JC42_PARAM_I2C_DEV, \ + .speed = JC42_PARAM_SPEED, \ + .addr = JC42_PARAM_ADDR } +/**@}*/ + +/** + * @brief Configure JC42 + */ +static const jc42_params_t jc42_params[] = +{ +#ifdef JC42_PARAMS_BOARD + JC42_PARAMS_BOARD, +#else + JC42_PARAMS_DEFAULT, +#endif +}; + +#ifdef __cplusplus +} +#endif + +#endif /* JC42_PARAMS_H */ +/** @} */ diff --git a/drivers/jc42/jc42.c b/drivers/jc42/jc42.c new file mode 100644 index 0000000000..92b61acd5b --- /dev/null +++ b/drivers/jc42/jc42.c @@ -0,0 +1,102 @@ +/* + * Copyright (C) 2017 Koen Zandberg + * + * 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_jc42 + * @{ + * + * @file + * @brief Device driver implementation for the JEDEC jc42.4 compliant temperature sensors + * + * @author Koen Zandberg + * + * @} + */ + + +#include "periph/i2c.h" +#include "byteorder.h" + +#include "jc42.h" +#include "jc42_internal.h" + +#define ENABLE_DEBUG (0) +#include "debug.h" + +static int jc42_get_register(jc42_t* dev, uint8_t reg, uint16_t* data) +{ + i2c_acquire(dev->i2c); + if (i2c_read_regs(dev->i2c, dev->addr, reg, data, 2) <= 0) { + DEBUG("[jc42] Problem reading register 0x%x\n", reg); + i2c_release(dev->i2c); + return JC42_NODEV; + } + i2c_release(dev->i2c); + return JC42_OK; +} + +static int jc42_set_register(jc42_t* dev, uint8_t reg, uint16_t* data) +{ + i2c_acquire(dev->i2c); + if (i2c_write_regs(dev->i2c, dev->addr, reg, data, 2) <= 0) { + DEBUG("[jc42] Problem writing to register 0x%x\n", reg); + i2c_release(dev->i2c); + return JC42_NODEV; + } + i2c_release(dev->i2c); + + return JC42_OK; +} + +int jc42_get_config(jc42_t* dev, uint16_t* data) +{ + return jc42_get_register(dev, JC42_REG_CONFIG, data); +} + +int jc42_set_config(jc42_t* dev, uint16_t data) +{ + return jc42_set_register(dev, JC42_REG_CONFIG, &data); +} + +int jc42_get_temperature(jc42_t* dev, int16_t* temperature) +{ + struct { signed int x:12;} s; + uint16_t tmp; + + /* Read temperature */ + if (jc42_get_register(dev, JC42_REG_TEMP, &tmp) != 0) { + return JC42_NODEV; + } + tmp = NTOHS(tmp); + /* Convert fixed point to uint16_t */ + *temperature = ((s.x = tmp)*100)>>4; + return JC42_OK; +} + +int jc42_init(jc42_t* dev, jc42_params_t* params) +{ + uint16_t config; + dev->i2c = params->i2c; + dev->addr = params->addr; + i2c_acquire(dev->i2c); + if (i2c_init_master(dev->i2c, params->speed) != 0) { + DEBUG("[jc42] Problem initializing I2C master\n"); + i2c_release(dev->i2c); + return JC42_NOI2C; + } + i2c_release(dev->i2c); + + /* Poll the device, fail if unavailable */ + if (jc42_get_config(dev, &config) != 0) { + return JC42_NODEV; + } + + DEBUG("[jc42] Config: 0x%x\n", config); + + return JC42_OK; +} diff --git a/drivers/jc42/jc42_saul.c b/drivers/jc42/jc42_saul.c new file mode 100644 index 0000000000..a0cebdb7e9 --- /dev/null +++ b/drivers/jc42/jc42_saul.c @@ -0,0 +1,44 @@ +/* + * Copyright (C) 2017 Koen Zandberg + * + * 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_jc42 + * @{ + * + * @file + * @brief SAUL adaption for jc42 compatible device + * + * @author Koen Zandberg + * + * @} + */ + +#include + +#include "saul.h" +#include "jc42.h" + + + +static int read_temperature(void *dev, phydat_t *res) +{ + int16_t temperature; + jc42_t *d = (jc42_t *)dev; + + jc42_get_temperature(d, &temperature); + res->val[0] = temperature; + res->unit = UNIT_TEMP_C; + res->scale = -2; + return 1; +} + +const saul_driver_t jc42_temperature_saul_driver = { + .read = read_temperature, + .write = saul_notsup, + .type = SAUL_SENSE_TEMP +}; diff --git a/sys/auto_init/auto_init.c b/sys/auto_init/auto_init.c index 56d5d72945..e4d9a82344 100644 --- a/sys/auto_init/auto_init.c +++ b/sys/auto_init/auto_init.c @@ -294,6 +294,10 @@ void auto_init(void) extern void auto_init_bmp180(void); auto_init_bmp180(); #endif +#ifdef MODULE_JC42 + extern void auto_init_jc42(void); + auto_init_jc42(); +#endif #endif /* MODULE_AUTO_INIT_SAUL */ diff --git a/sys/auto_init/saul/auto_init_jc42.c b/sys/auto_init/saul/auto_init_jc42.c new file mode 100644 index 0000000000..d2cab3f4cc --- /dev/null +++ b/sys/auto_init/saul/auto_init_jc42.c @@ -0,0 +1,82 @@ +/* + * Copyright (C) 2017 Koen Zandberg + * + * 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 auto_init_saul + * @{ + * + * @file + * @brief Auto initialization of jc42 compatible driver. + * + * @author Koen Zandberg + * + * @} + */ + +#ifdef MODULE_JC42 + +#include "log.h" +#include "saul_reg.h" + +#include "jc42.h" +#include "jc42_params.h" + +/** + * @brief Define the number of configured sensors + */ +#define JC42_NUMOF (sizeof(jc42_params) / sizeof(jc42_params[0])) + +/** + * @brief Allocation of memory for device descriptors + */ +static jc42_t jc42_devs[JC42_NUMOF]; + +/** + * @brief Memory for the SAUL registry entries + */ +static saul_reg_t saul_entries[JC42_NUMOF]; + +/** + * @brief Reference the driver structs. + * @{ + */ +extern const saul_driver_t jc42_temperature_saul_driver; +/** @} */ + +/** + * @brief Allocate and configure entries to the SAUL registry + */ +saul_reg_t jc42_saul_reg_info[]= +{ + { + .name= "jc42", + .driver = &jc42_temperature_saul_driver + } +}; + +void auto_init_jc42(void) +{ + for (unsigned i = 0; i < JC42_NUMOF; i++) { + const jc42_params_t *p = &jc42_params[i]; + if (jc42_init(&jc42_devs[i], (jc42_params_t*) p) < 0) { + LOG_ERROR("Unable to initialize jc42 sensor #%i\n", i); + return; + } + + /* temperature */ + saul_entries[i].dev = &(jc42_devs[i]); + saul_entries[i].name = jc42_saul_reg_info[i].name; + saul_entries[i].driver = &jc42_temperature_saul_driver; + + /* register to saul */ + saul_reg_add(&(saul_entries[i])); + } +} +#else +typedef int dont_be_pedantic; +#endif /* MODULE_JC42 */ diff --git a/tests/driver_jc42/Makefile b/tests/driver_jc42/Makefile new file mode 100644 index 0000000000..4560f2025d --- /dev/null +++ b/tests/driver_jc42/Makefile @@ -0,0 +1,17 @@ +APPLICATION = driver_jc42 +include ../Makefile.tests_common + +USEMODULE += jc42 +USEMODULE += xtimer + +# set default device parameters in case they are undefined +TEST_I2C ?= 0 +TEST_I2C_ADDR ?= 0x18 +TEST_I2C_SPEED ?= I2C_SPEED_NORMAL + +# export parameters +CFLAGS += -DTEST_I2C=$(TEST_I2C) +CFLAGS += -DTEST_I2C_ADDR=$(TEST_I2C_ADDR) +CFLAGS += -DTEST_I2C_SPEED=$(TEST_I2C_SPEED) + +include $(RIOTBASE)/Makefile.include diff --git a/tests/driver_jc42/README.md b/tests/driver_jc42/README.md new file mode 100644 index 0000000000..4b27efcb92 --- /dev/null +++ b/tests/driver_jc42/README.md @@ -0,0 +1,14 @@ +# JC42 Driver Test + +## Introduction +This test will test if the jc42 compatible temperature sensor is working. + +## Configuration +There are three parameters to configure: + +* `TEST_I2C` — I2C device to use. +* `TEST_I2C_ADDR` — The sensor address (usually 0x18). +* `TEST_I2C_SPEED` — The sensor address (usually I2C_SPEED_NORMAL). + +## Expected result +The sensor should continuously (every 1 sec) output the temperature. The precision should be two digits. diff --git a/tests/driver_jc42/main.c b/tests/driver_jc42/main.c new file mode 100644 index 0000000000..7e5f477d1b --- /dev/null +++ b/tests/driver_jc42/main.c @@ -0,0 +1,82 @@ +/* + * Copyright (C) 2017 Koen Zandberg + * + * 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 tests + * @{ + * + * @file + * @brief Test application for the jc42 sensor driver + * + * @author Koen Zandberg + * + * @} + */ + +#ifndef TEST_I2C +#error "TEST_I2C not defined" +#endif + +#ifndef TEST_I2C_ADDR +#error "TEST_I2C_ADDR not defined" +#endif + +#ifndef TEST_I2C_SPEED +#error "TEST_I2C_SPEED not defined" +#endif + +#include + +#include "xtimer.h" + +#include "periph/i2c.h" + +#include "jc42.h" + +int main(void) +{ + jc42_t dev; + jc42_params_t params = { + .i2c = TEST_I2C, + .addr = TEST_I2C_ADDR, + .speed = TEST_I2C_SPEED, + }; + + puts("JC42 temperature sensor test application\n"); + + /* initialize the sensor */ + printf("Initializing sensor..."); + + if (jc42_init(&dev, ¶ms) == 0) { + puts("[OK]"); + } + else { + puts("[Failed]"); + return 1; + } + + /* read temperature every 1 seconds */ + int16_t temperature; + while (1) { + printf("Testing sensor communication..."); + if (jc42_get_temperature(&dev, &temperature) == 0) { + puts("[OK]"); + } + else { + puts("[Failed]"); + return 1; + } + /* display results */ + printf("temperature: %d.%02d C\n", temperature / 100, temperature % 100); + + /* sleep between measurements */ + xtimer_usleep(1000 * MS_IN_USEC); + } + + return 0; +}