mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
Merge pull request #6292 from bergzand/jc42
drivers/jc42: Add JEDEC JC42.4 temperature sensors
This commit is contained in:
commit
937e3a28c2
@ -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
|
||||
|
@ -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
|
||||
|
125
drivers/include/jc42.h
Normal file
125
drivers/include/jc42.h
Normal file
@ -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 <koen@bergzand.net>
|
||||
*/
|
||||
|
||||
#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_ */
|
2
drivers/jc42/Makefile
Normal file
2
drivers/jc42/Makefile
Normal file
@ -0,0 +1,2 @@
|
||||
MODULE = jc42
|
||||
include $(RIOTBASE)/Makefile.base
|
50
drivers/jc42/include/jc42_internal.h
Normal file
50
drivers/jc42/include/jc42_internal.h
Normal file
@ -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 <koen@bergzand.net>
|
||||
*/
|
||||
|
||||
#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_ */
|
65
drivers/jc42/include/jc42_params.h
Normal file
65
drivers/jc42/include/jc42_params.h
Normal file
@ -0,0 +1,65 @@
|
||||
/*
|
||||
* Copyright (C) 2017 Koen Zandberg <koen@bergzand.net>
|
||||
*
|
||||
* 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 <koen@bergzand.net>
|
||||
*/
|
||||
|
||||
#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 */
|
||||
/** @} */
|
102
drivers/jc42/jc42.c
Normal file
102
drivers/jc42/jc42.c
Normal file
@ -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 <koen@bergzand.net>
|
||||
*
|
||||
* @}
|
||||
*/
|
||||
|
||||
|
||||
#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;
|
||||
}
|
44
drivers/jc42/jc42_saul.c
Normal file
44
drivers/jc42/jc42_saul.c
Normal file
@ -0,0 +1,44 @@
|
||||
/*
|
||||
* Copyright (C) 2017 Koen Zandberg <koen@bergzand.net>
|
||||
*
|
||||
* 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 <koen@bergzand.net>
|
||||
*
|
||||
* @}
|
||||
*/
|
||||
|
||||
#include <string.h>
|
||||
|
||||
#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
|
||||
};
|
@ -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 */
|
||||
|
||||
|
82
sys/auto_init/saul/auto_init_jc42.c
Normal file
82
sys/auto_init/saul/auto_init_jc42.c
Normal file
@ -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 <koen@bergzand.net>
|
||||
*
|
||||
* @}
|
||||
*/
|
||||
|
||||
#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 */
|
17
tests/driver_jc42/Makefile
Normal file
17
tests/driver_jc42/Makefile
Normal file
@ -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
|
14
tests/driver_jc42/README.md
Normal file
14
tests/driver_jc42/README.md
Normal file
@ -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.
|
82
tests/driver_jc42/main.c
Normal file
82
tests/driver_jc42/main.c
Normal file
@ -0,0 +1,82 @@
|
||||
/*
|
||||
* Copyright (C) 2017 Koen Zandberg <koen@bergzand.net>
|
||||
*
|
||||
* 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 <koen@bergzand.net>
|
||||
*
|
||||
* @}
|
||||
*/
|
||||
|
||||
#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 <stdio.h>
|
||||
|
||||
#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;
|
||||
}
|
Loading…
Reference in New Issue
Block a user