diff --git a/drivers/Makefile.dep b/drivers/Makefile.dep index e5a0c9991d53d51fc76dd25ee04e5c88d17c0393..6a3de6c93dc87d5bbaa1defff51448a8208deb05 100644 --- a/drivers/Makefile.dep +++ b/drivers/Makefile.dep @@ -452,6 +452,11 @@ ifneq (,$(filter uart_half_duplex,$(USEMODULE))) USEMODULE += xtimer endif +ifneq (,$(filter vcnl40%0,$(USEMODULE))) + USEMODULE += vcnl40x0 + FEATURES_REQUIRED += periph_i2c +endif + ifneq (,$(filter veml6070,$(USEMODULE))) FEATURES_REQUIRED += periph_i2c endif diff --git a/drivers/Makefile.include b/drivers/Makefile.include index 83c994ef376412e9fc178daffb99d69bb84b6040..19bb780f39d6d8294f5aded1df906dd3f33bf8f3 100644 --- a/drivers/Makefile.include +++ b/drivers/Makefile.include @@ -246,6 +246,10 @@ ifneq (,$(filter uart_half_duplex,$(USEMODULE))) USEMODULE_INCLUDES += $(RIOTBASE)/drivers/uart_half_duplex/include endif +ifneq (,$(filter vcnl40x0,$(USEMODULE))) + USEMODULE_INCLUDES += $(RIOTBASE)/drivers/vcnl40x0/include +endif + ifneq (,$(filter veml6070,$(USEMODULE))) USEMODULE_INCLUDES += $(RIOTBASE)/drivers/veml6070/include endif diff --git a/drivers/include/saul.h b/drivers/include/saul.h index c678d6b19f9d78fd413e2921cd1399f09267a9a7..75b12ca5737af44981150060b549787ba36106b6 100644 --- a/drivers/include/saul.h +++ b/drivers/include/saul.h @@ -97,6 +97,7 @@ enum { SAUL_SENSE_CO2 = 0x8f, /**< sensor: CO2 Gas */ SAUL_SENSE_TVOC = 0x90, /**< sensor: TVOC Gas */ SAUL_SENSE_OCCUP = 0x91, /**< sensor: occupancy */ + SAUL_SENSE_PROXIMITY= 0x92, /**< sensor: proximity */ SAUL_CLASS_ANY = 0xff /**< any device - wildcard */ /* extend this list as needed... */ }; diff --git a/drivers/include/vcnl40x0.h b/drivers/include/vcnl40x0.h new file mode 100644 index 0000000000000000000000000000000000000000..43006a8d346a860cd0b0df1ab1e112c153d7a01d --- /dev/null +++ b/drivers/include/vcnl40x0.h @@ -0,0 +1,149 @@ +/* + * Copyright (C) 2017 Inria + * + * 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_vcnl40x0 VCNL4010/VCNL4020/VCNL4040 Proximity and Ambient Light Sensors + * @ingroup drivers_sensors + * @brief Device driver interface for the VCNL40X0 Proximity and Ambient Light Sensors. + * @{ + * + * @file + * @brief Device driver interface for the VCNL40X0 sensors family. + * + * @note VCNL4010, VCNL4020 and VCNL4040 are supported. + * + * @author Alexandre Abadie <alexandre.abadie@inria.fr> + */ + +#ifndef VCNL40X0_H +#define VCNL40X0_H + +#include "saul.h" +#include "periph/i2c.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @brief Proximity measure rate + */ +enum { + VCNL40X0_PROXIMITY_RATE_2 = 0, /**< 1.95 measurements/s (default) */ + VCNL40X0_PROXIMITY_RATE_4, /**< 3.90625 measurements/s */ + VCNL40X0_PROXIMITY_RATE_8, /**< 7.8125 measurements/s */ + VCNL40X0_PROXIMITY_RATE_16, /**< 16.625 measurements/s */ + VCNL40X0_PROXIMITY_RATE_31, /**< 31.25 measurements/s */ + VCNL40X0_PROXIMITY_RATE_62, /**< 62.5 measurements/s */ + VCNL40X0_PROXIMITY_RATE_125, /**< 125 measurements/s */ + VCNL40X0_PROXIMITY_RATE_250, /**< 250 measurements/s */ +}; + +/** + * @brief Ambient light measurement rate + */ +enum { + VCNL40X0_AMBIENT_RATE_1 = 0, /**< 1 sample/s */ + VCNL40X0_AMBIENT_RATE_2, /**< 2 sample/s (default) */ + VCNL40X0_AMBIENT_RATE_3, /**< 3 sample/s */ + VCNL40X0_AMBIENT_RATE_4, /**< 4 sample/s */ + VCNL40X0_AMBIENT_RATE_5, /**< 5 sample/s */ + VCNL40X0_AMBIENT_RATE_6, /**< 6 sample/s */ + VCNL40X0_AMBIENT_RATE_8, /**< 8 sample/s */ + VCNL40X0_AMBIENT_RATE_10, /**< 10 sample/s */ +}; + +/** + * @brief Ambient light number of conversions during one measurement cycle + * + * Number of conversions = 2^decimal value + */ +enum { + VCNL40X0_AMBIENT_AVERAGE_1 = 0, /**< Decimal value 1 */ + VCNL40X0_AMBIENT_AVERAGE_2, /**< Decimal value 2 */ + VCNL40X0_AMBIENT_AVERAGE_4, /**< Decimal value 4 */ + VCNL40X0_AMBIENT_AVERAGE_8, /**< Decimal value 8 */ + VCNL40X0_AMBIENT_AVERAGE_16, /**< Decimal value 16 */ + VCNL40X0_AMBIENT_AVERAGE_32, /**< Decimal value 32 (default) */ + VCNL40X0_AMBIENT_AVERAGE_64, /**< Decimal value 64 */ + VCNL40X0_AMBIENT_AVERAGE_128, /**< Decimal value 128 */ +}; + +/** + * @brief Status and error return codes + */ +enum { + VCNL40X0_OK = 0, /**< Everything was fine */ + VCNL40X0_ERR_I2C, /**< Error initializing the I2C bus */ + VCNL40X0_ERR_NODEV /**< Error wrong device */ +}; + +/** + * @brief Device initialization parameters + */ +typedef struct { + i2c_t i2c_dev; /**< I2C device which is used */ + uint8_t i2c_addr; /**< Address on I2C bus */ + uint8_t led_current; /**< LED current */ + uint8_t proximity_rate; /**< Proximity rate */ + uint8_t ambient_rate; /**< Ambient light rate */ + uint8_t ambient_avg; /**< Ambient light conversion average */ +} vcnl40x0_params_t; + +/** + * @brief Device descriptor for the VCNL40X0 sensor + */ +typedef struct { + vcnl40x0_params_t params; /**< Device parameters */ +} vcnl40x0_t; + +/** + * @brief Initialize the given VCNL40X0 device + * + * @param[out] dev Initialized device descriptor of VCNL40X0 device + * @param[in] params The parameters for the VCNL40X0 device + * + * @return VCNL40X0_OK on success + * @return -VCNL40X0_ERR_I2C if given I2C is not enabled in board config + * @return -VCNL40X0_ERR_NODEV if not a vcnl40X0 device + */ +int vcnl40x0_init(vcnl40x0_t *dev, const vcnl40x0_params_t *params); + +/** + * @brief Read proximity value from the vcnl40X0 device + * + * @param[in] dev Device descriptor of VCNL40X0 device to read from + * + * @return Proximity in counts + */ +uint16_t vcnl40x0_read_proximity(const vcnl40x0_t *dev); + +/** + * @brief Read ambient light value from the vcnl40X0 device + * + * @param[in] dev Device descriptor of VCNL40X0 device to read from + * + * @return Ambient light in counts + */ +uint16_t vcnl40x0_read_ambient_light(const vcnl40x0_t *dev); + +/** + * @brief Read illuminance value from the vcnl40X0 device + * + * @param[in] dev Device descriptor of VCNL40X0 device to read from + * + * @return Illuminance in lux + */ +uint16_t vcnl40x0_read_illuminance(const vcnl40x0_t *dev); + +#ifdef __cplusplus +} +#endif + +#endif /* VCNL40X0_H */ +/** @} */ diff --git a/drivers/saul/saul_str.c b/drivers/saul/saul_str.c index 255037d8792ea369f71b1882246527c91058866e..32233c73c00d16222e12147d9bb02bcdbe211974 100644 --- a/drivers/saul/saul_str.c +++ b/drivers/saul/saul_str.c @@ -53,6 +53,7 @@ const char *saul_class_to_str(const uint8_t class_id) case SAUL_SENSE_DISTANCE: return "SENSE_DISTANCE"; case SAUL_SENSE_CO2: return "SENSE_CO2"; case SAUL_SENSE_TVOC: return "SENSE_TVOC"; + case SAUL_SENSE_PROXIMITY: return "SENSE_PROXIMITY"; case SAUL_CLASS_ANY: return "CLASS_ANY"; case SAUL_SENSE_OCCUP: return "SENSE_OCCUP"; default: return "CLASS_UNKNOWN"; diff --git a/drivers/vcnl40x0/Makefile b/drivers/vcnl40x0/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..48422e909a47d7cd428d10fa73825060ccc8d8c2 --- /dev/null +++ b/drivers/vcnl40x0/Makefile @@ -0,0 +1 @@ +include $(RIOTBASE)/Makefile.base diff --git a/drivers/vcnl40x0/include/vcnl40x0_internals.h b/drivers/vcnl40x0/include/vcnl40x0_internals.h new file mode 100644 index 0000000000000000000000000000000000000000..50851eefcbb5c67863b518211717b5f2faf0a8b9 --- /dev/null +++ b/drivers/vcnl40x0/include/vcnl40x0_internals.h @@ -0,0 +1,139 @@ +/* + * Copyright (C) 2017 Inria + * + * 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_vcnl40x0 + * @{ + * + * @file + * @brief Internal addresses, registers, constants for the VCNL40X0 devices. + * + * @author Alexandre Abadie <alexandre.abadie@inria.fr> + * @} + */ + +#ifndef VCNL40X0_INTERNALS_H +#define VCNL40X0_INTERNALS_H + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @brief VCNL40X0 I2C address + */ +#define VCNL40X0_ADDR (0x13) + +/** + * @brief VCNL40X0 product ID + */ +#define VCNL40X0_PRODUCT_ID (0x20) + +/** + * @name VCNL40X0 registers + * @{ + */ +#define VCNL40X0_REG_COMMAND (0x80) +#define VCNL40X0_REG_PRODUCT_ID (0x81) +#define VCNL40X0_REG_PROXIMITY_RATE (0x82) +#define VCNL40X0_REG_PROXIMITY_CURRENT (0x83) +#define VCNL40X0_REG_AMBIENT_PARAMETER (0x84) +#define VCNL40X0_REG_AMBIENT_VALUE (0x85) +#define VCNL40X0_REG_PROXIMITY_VALUE (0x87) +#define VCNL40X0_REG_INTERRUPT_CONTROL (0x89) +#define VCNL40X0_REG_INTERRUPT_LOW_THRES (0x8a) +#define VCNL40X0_REG_HIGH_THRES (0x8c) +#define VCNL40X0_REG_INTERRUPT_STATUS (0x8e) +#define VCNL40X0_REG_PROX_TIMING (0xf9) +#define VCNL40X0_REG_AMBIENT_IR_LIGHT_LEVEL (0x90) /* should not be used */ +/** @} */ + +/** + * @name VCNL40X0 command register constants + * @{ + */ +#define VCNL40X0_COMMAND_ALL_DISABLE (0x00) +#define VCNL40X0_COMMAND_SELFTIMED_MODE_ENABLE (0x01) +#define VCNL40X0_COMMAND_PROX_ENABLE (0x02) +#define VCNL40X0_COMMAND_AMBI_ENABLE (0x04) +#define VCNL40X0_COMMAND_PROX_ON_DEMAND (0x08) +#define VCNL40X0_COMMAND_AMBI_ON_DEMAND (0x10) +#define VCNL40X0_COMMAND_MASK_PROX_DATA_READY (0x20) +#define VCNL40X0_COMMAND_MASK_AMBI_DATA_READY (0x40) +#define VCNL40X0_COMMAND_MASK_LOCK (0x80) +/** @} */ + +/** + * @name VCNL40X0 product ID register constants + * @{ + */ +#define VCNL40X0_PRODUCT_MASK_REVISION_ID (0x0f) +#define VCNL40X0_PRODUCT_MASK_PRODUCT_ID (0xf0) +/** @} */ + +/** + * @name VCNL40X0 proximity rate register constants + * @{ + */ +#define VCNL40X0_PROXIMITY_MASK_MEASUREMENT_RATE (0x07) +#define VCNL40X0_PROXIMITY_MASK_LED_CURRENT (0x3f) +#define VCNL40X0_PROXIMITY_MASK_FUSE_PROG_ID (0xc0) +/** @} */ + +/** + * @name VCNL40X0 ambient light parameter register constants + * @{ + */ +#define VCNL40X0_AMBIENT_MASK_PARA_AVERAGE (0x07) +#define VCNL40X0_AMBIENT_PARA_AUTO_OFFSET_ENABLE (0x08) +#define VCNL40X0_AMBIENT_MASK_PARA_AUTO_OFFSET (0x08) +#define VCNL40X0_AMBIENT_MASK_PARA_MEAS_RATE (0x70) +#define VCNL40X0_AMBIENT_PARA_CONT_CONV_ENABLE (0x80) +#define VCNL40X0_AMBIENT_MASK_PARA_CONT_CONV (0x80) +/** @} */ + +/** + * @name VCNL40X0 interrupt control register constants + * @{ + */ +#define VCNL40X0_INTERRUPT_THRES_SEL_PROX (0x00) +#define VCNL40X0_INTERRUPT_THRES_SEL_ALS (0x01) +#define VCNL40X0_INTERRUPT_THRES_ENABLE (0x02) +#define VCNL40X0_INTERRUPT_ALS_READY_ENABLE (0x04) +#define VCNL40X0_INTERRUPT_PROX_READY_ENABLE (0x08) +#define VCNL40X0_INTERRUPT_COUNT_EXCEED_1 (0x00) +#define VCNL40X0_INTERRUPT_COUNT_EXCEED_2 (0x20) +#define VCNL40X0_INTERRUPT_COUNT_EXCEED_4 (0x40) +#define VCNL40X0_INTERRUPT_COUNT_EXCEED_8 (0x60) +#define VCNL40X0_INTERRUPT_COUNT_EXCEED_16 (0x80) +#define VCNL40X0_INTERRUPT_COUNT_EXCEED_32 (0xa0) +#define VCNL40X0_INTERRUPT_COUNT_EXCEED_64 (0xc0) +#define VCNL40X0_INTERRUPT_COUNT_EXCEED_128 (0xe0) +#define VCNL40X0_INTERRUPT_MASK_COUNT_EXCEED (0xe0) +/** @} */ + +/** + * @name VCNL40X0 interrupt status register constants + * @{ + */ +#define VCNL40X0_INTERRUPT_STATUS_THRES_HI (0x01) +#define VCNL40X0_INTERRUPT_STATUS_THRES_LO (0x02) +#define VCNL40X0_INTERRUPT_STATUS_ALS_READY (0x04) +#define VCNL40X0_INTERRUPT_STATUS_PROX_READY (0x08) +#define VCNL40X0_INTERRUPT_MASK_STATUS_THRES_HI (0x01) +#define VCNL40X0_INTERRUPT_MASK_THRES_LO (0x02) +#define VCNL40X0_INTERRUPT_MASK_ALS_READY (0x04) +#define VCNL40X0_INTERRUPT_MASK_PROX_READY (0x08) +/** @} */ + +#ifdef __cplusplus +} +#endif + +#endif /* VCNL40X0_INTERNALS_H */ +/** @} */ diff --git a/drivers/vcnl40x0/include/vcnl40x0_params.h b/drivers/vcnl40x0/include/vcnl40x0_params.h new file mode 100644 index 0000000000000000000000000000000000000000..c04c900385c93540b05464452a03a4afe4bf5e0f --- /dev/null +++ b/drivers/vcnl40x0/include/vcnl40x0_params.h @@ -0,0 +1,83 @@ +/* + * Copyright (C) 2017 Inria + * + * 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_vcnl40x0 + * + * @{ + * @file + * @brief Default configuration for VCNL40X0 devices + * + * @author Alexandre Abadie <alexandre.abadie@inria.fr> + */ + +#ifndef VCNL40X0_PARAMS_H +#define VCNL40X0_PARAMS_H + +#include "board.h" +#include "vcnl40x0.h" +#include "vcnl40x0_internals.h" +#include "saul_reg.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @name Default configuration parameters for the VCNL40X0 + * @{ + */ +#ifndef VCNL40X0_PARAM_I2C_DEV +#define VCNL40X0_PARAM_I2C_DEV I2C_DEV(0) +#endif +#ifndef VCNL40X0_PARAM_I2C_ADDR +#define VCNL40X0_PARAM_I2C_ADDR (VCNL40X0_ADDR) +#endif +#ifndef VCNL40X0_PARAM_LED_CURRENT +#define VCNL40X0_PARAM_LED_CURRENT (2U) +#endif +#ifndef VCNL40X0_PARAM_PROXIMITY_RATE +#define VCNL40X0_PARAM_PROXIMITY_RATE (VCNL40X0_PROXIMITY_RATE_2) +#endif +#ifndef VCNL40X0_PARAM_AMBIENT_AVG +#define VCNL40X0_PARAM_AMBIENT_AVG (VCNL40X0_AMBIENT_AVERAGE_32) +#endif +#ifndef VCNL40X0_PARAM_AMBIENT_RATE +#define VCNL40X0_PARAM_AMBIENT_RATE (VCNL40X0_AMBIENT_RATE_2) +#endif + +#define VCNL40X0_PARAMS { .i2c_dev = VCNL40X0_PARAM_I2C_DEV, \ + .i2c_addr = VCNL40X0_PARAM_I2C_ADDR, \ + .led_current = VCNL40X0_PARAM_LED_CURRENT, \ + .proximity_rate = VCNL40X0_PARAM_PROXIMITY_RATE, \ + .ambient_avg = VCNL40X0_PARAM_AMBIENT_AVG, \ + .ambient_rate = VCNL40X0_PARAM_AMBIENT_RATE } +/**@}*/ + +/** + * @brief Configure VCNL40X0 + */ +static const vcnl40x0_params_t vcnl40x0_params[] = +{ + VCNL40X0_PARAMS, +}; + +/** + * @brief Configure SAUL registry entries + */ +static const saul_reg_info_t vcnl40x0_saul_reg_info[] = +{ + { .name = "vcnl40x0" } +}; + +#ifdef __cplusplus +} +#endif + +#endif /* VCNL40X0_PARAMS_H */ +/** @} */ diff --git a/drivers/vcnl40x0/vcnl40x0.c b/drivers/vcnl40x0/vcnl40x0.c new file mode 100644 index 0000000000000000000000000000000000000000..06abfdc8d898598685a88c0d937f4afd0e7fdf2f --- /dev/null +++ b/drivers/vcnl40x0/vcnl40x0.c @@ -0,0 +1,172 @@ +/* + * Copyright (C) 2017 Inria + * + * 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_vcnl40x0 + * @{ + * + * @file + * @brief Device driver implementation for VCNL40X0 Proximity and Ambient Light devices. + * + * @author Alexandre Abadie <alexandre.abadie@inria.fr> + * + * @} + */ + +#include "vcnl40x0.h" +#include "vcnl40x0_internals.h" +#include "vcnl40x0_params.h" +#include "periph/i2c.h" + +#define ENABLE_DEBUG (0) +#include "debug.h" + +#define DEV_I2C (dev->params.i2c_dev) +#define DEV_ADDR (dev->params.i2c_addr) + +/* Internal functions */ +static int _set_command_reg(const vcnl40x0_t *dev, uint8_t reg) +{ + if (i2c_write_reg(DEV_I2C, DEV_ADDR, VCNL40X0_REG_COMMAND, reg, 0) != 0) { + return -1; + } + + return 0; +} + +static int _get_command_reg(const vcnl40x0_t *dev, uint8_t *reg) +{ + if (i2c_read_reg(DEV_I2C, DEV_ADDR, VCNL40X0_REG_COMMAND, reg, 0) != 0) { + return -1; + } + + return 0; +} + +/*---------------------------------------------------------------------------* + * VCNL40X0 Core API * + *---------------------------------------------------------------------------*/ + +int vcnl40x0_init(vcnl40x0_t *dev, const vcnl40x0_params_t *params) +{ + dev->params = *params; + + /* Acquire exclusive access */ + i2c_acquire(DEV_I2C); + + /* Check sensor ID */ + uint8_t checkid; + i2c_read_reg(DEV_I2C, DEV_ADDR, VCNL40X0_REG_PRODUCT_ID, + &checkid, 0); + if ((checkid & VCNL40X0_PRODUCT_MASK_PRODUCT_ID) != VCNL40X0_PRODUCT_ID) { + DEBUG("[vcnl40x0] Error: wrong device ID '0x%02x', expected '0x%02x'\n", + checkid, VCNL40X0_PRODUCT_ID); + i2c_release(DEV_I2C); + return -VCNL40X0_ERR_NODEV; + } + + /* LED current cannot be above 20 */ + if (dev->params.led_current > 20) { + dev->params.led_current = 20; + } + + if (i2c_write_reg(DEV_I2C, DEV_ADDR, VCNL40X0_REG_PROXIMITY_CURRENT, + dev->params.led_current, 0) != 0) { + DEBUG("[vcnl40x0] Error: failed to set proximity current\n"); + i2c_release(DEV_I2C); + return -VCNL40X0_ERR_I2C; + } + + if (_set_command_reg(dev, VCNL40X0_COMMAND_ALL_DISABLE) < 0) { + DEBUG("[vcnl40x0] Error: failed to set disable all commands\n"); + i2c_release(DEV_I2C); + return -VCNL40X0_ERR_I2C; + } + + if (i2c_write_reg(DEV_I2C, DEV_ADDR, VCNL40X0_REG_PROXIMITY_RATE, + dev->params.proximity_rate, 0) != 0) { + DEBUG("[vcnl40x0] Error: failed to set proximity rate\n"); + i2c_release(DEV_I2C); + return -VCNL40X0_ERR_I2C; + } + + if (i2c_write_reg(DEV_I2C, DEV_ADDR, VCNL40X0_REG_AMBIENT_PARAMETER, + dev->params.ambient_rate | + VCNL40X0_AMBIENT_PARA_AUTO_OFFSET_ENABLE | + dev->params.ambient_avg, 0) != 0) { + DEBUG("[vcnl40x0] Error: failed to set ambient light rate\n"); + i2c_release(DEV_I2C); + return -VCNL40X0_ERR_I2C; + } + + /* Release I2C device */ + i2c_release(DEV_I2C); + + DEBUG("[vcnl40x0] info: vcnl40x0 sensor initialized with success\n"); + + return VCNL40X0_OK; +} + +uint16_t vcnl40x0_read_proximity(const vcnl40x0_t *dev) +{ + i2c_acquire(DEV_I2C); + + _set_command_reg(dev, + VCNL40X0_COMMAND_PROX_ENABLE | + VCNL40X0_COMMAND_PROX_ON_DEMAND); + uint16_t cnt = 0xffff; + while (cnt--) { + uint8_t reg; + _get_command_reg(dev, ®); + if (reg & VCNL40X0_COMMAND_MASK_PROX_DATA_READY) { + uint8_t prox_buf[2]; + i2c_read_regs(DEV_I2C, DEV_ADDR, VCNL40X0_REG_PROXIMITY_VALUE, + &prox_buf, 2, 0); + uint16_t res = (uint16_t)((prox_buf[0] << 8) | prox_buf[1]); + DEBUG("[vcnl40x0] Proximity read: %i\n", res); + i2c_release(DEV_I2C); + return res; + } + } + + i2c_release(DEV_I2C); + return 0; +} + +uint16_t vcnl40x0_read_ambient_light(const vcnl40x0_t *dev) +{ + i2c_acquire(DEV_I2C); + + _set_command_reg(dev, + VCNL40X0_COMMAND_AMBI_ENABLE | + VCNL40X0_COMMAND_AMBI_ON_DEMAND); + + uint16_t cnt = 0xffff; + while (cnt--) { + uint8_t reg; + _get_command_reg(dev, ®); + if (reg & VCNL40X0_COMMAND_MASK_AMBI_DATA_READY) { + uint8_t ambient_buf[2]; + i2c_read_regs(DEV_I2C, DEV_ADDR, VCNL40X0_REG_AMBIENT_VALUE, + &ambient_buf, 2, 0); + uint16_t res = (uint16_t)((ambient_buf[0] << 8) | ambient_buf[1]); + DEBUG("[vcnl40x0] Ambient light read: %i\n", res); + i2c_release(DEV_I2C); + return res; + } + } + + i2c_release(DEV_I2C); + + return 0; +} + +uint16_t vcnl40x0_read_illuminance(const vcnl40x0_t *dev) +{ + return vcnl40x0_read_ambient_light(dev) >> 2; +} diff --git a/drivers/vcnl40x0/vcnl40x0_saul.c b/drivers/vcnl40x0/vcnl40x0_saul.c new file mode 100644 index 0000000000000000000000000000000000000000..b2bc4df217b787761a2b05fe6b88030da796756c --- /dev/null +++ b/drivers/vcnl40x0/vcnl40x0_saul.c @@ -0,0 +1,59 @@ +/* + * Copyright (C) 2017 Inria + * + * 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_vcnl40x0 + * @{ + * + * @file + * @brief SAUL adaption for VCNL40X0 devices + * + * @author Hauke Petersen <hauke.petersen@fu-berlin.de> + * @author Alexandre Abadie <alexandre.abadie@inria.fr> + * + * @} + */ + +#include <string.h> + +#include "saul.h" +#include "vcnl40x0.h" +#include "vcnl40x0_params.h" +#include "xtimer.h" + +static int read_proximity(const void *dev, phydat_t *res) +{ + const vcnl40x0_t *d = (vcnl40x0_t *)dev; + + res->val[0] = (int16_t)vcnl40x0_read_proximity(d); + res->unit = UNIT_CTS; + res->scale = 0; + return 1; +} + +static int read_illuminance(const void *dev, phydat_t *res) +{ + const vcnl40x0_t *d = (vcnl40x0_t *)dev; + + res->val[0] = (int16_t)vcnl40x0_read_illuminance(d); + res->unit = UNIT_LUX; + res->scale = 0; + return 1; +} + +const saul_driver_t vcnl40x0_proximity_saul_driver = { + .read = read_proximity, + .write = saul_notsup, + .type = SAUL_SENSE_PROXIMITY +}; + +const saul_driver_t vcnl40x0_illuminance_saul_driver = { + .read = read_illuminance, + .write = saul_notsup, + .type = SAUL_SENSE_LIGHT +}; diff --git a/makefiles/pseudomodules.inc.mk b/makefiles/pseudomodules.inc.mk index c5e37daaefed1663c5a830e0bec86284fa812465..9a3b4491595463a269aa1c86f1dc227900698ef0 100644 --- a/makefiles/pseudomodules.inc.mk +++ b/makefiles/pseudomodules.inc.mk @@ -116,6 +116,11 @@ PSEUDOMODULES += si7021 PSEUDOMODULES += rn2483 PSEUDOMODULES += rn2903 +# include variants of VCNL40x0 drivers as pseudo modules +PSEUDOMODULES += vcnl4010 +PSEUDOMODULES += vcnl4020 +PSEUDOMODULES += vcnl4040 + # add all pseudo random number generator variants as pseudomodules PSEUDOMODULES += prng_% diff --git a/sys/auto_init/auto_init.c b/sys/auto_init/auto_init.c index 755f5f0ec1e4b0c37f8ae7711d80cfeebd181981..31a01675c0c8cb4e79ce7f41558e0c5a90e09a3b 100644 --- a/sys/auto_init/auto_init.c +++ b/sys/auto_init/auto_init.c @@ -433,6 +433,10 @@ void auto_init(void) extern void auto_init_tsl2561(void); auto_init_tsl2561(); #endif +#ifdef MODULE_VCNL40X0 + extern void auto_init_vcnl40x0(void); + auto_init_vcnl40x0(); +#endif #ifdef MODULE_VEML6070 extern void auto_init_veml6070(void); auto_init_veml6070(); diff --git a/sys/auto_init/saul/auto_init_vcnl40x0.c b/sys/auto_init/saul/auto_init_vcnl40x0.c new file mode 100644 index 0000000000000000000000000000000000000000..b8d1e3fedaac722e97fab93d4e5ffe28b4e2cf79 --- /dev/null +++ b/sys/auto_init/saul/auto_init_vcnl40x0.c @@ -0,0 +1,81 @@ +/* + * Copyright (C) 2017 Inria + * + * 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 VCNL40X0 driver. + * + * @author Alexandre Abadie <alexandre.abadie@inria.fr> + * + * @} + */ + +#ifdef MODULE_VCNL40X0 + +#include "log.h" +#include "saul_reg.h" + +#include "vcnl40x0_params.h" + +#define ENABLE_DEBUG (0) + +/** + * @brief Define the number of configured sensors + */ +#define VCNL40X0_NUMOF (sizeof(vcnl40x0_params) / sizeof(vcnl40x0_params[0])) + +/** + * @brief Allocation of memory for device descriptors + */ +static vcnl40x0_t vcnl40x0_devs[VCNL40X0_NUMOF]; + +/** + * @brief Memory for the SAUL registry entries + */ +static saul_reg_t saul_entries[VCNL40X0_NUMOF * 2]; + +/** + * @brief Reference the driver structs. + * @{ + */ +extern const saul_driver_t vcnl40x0_proximity_saul_driver; +extern const saul_driver_t vcnl40x0_illuminance_saul_driver; +/** @} */ + +void auto_init_vcnl40x0(void) +{ + for (unsigned i = 0; i < VCNL40X0_NUMOF; i++) { + LOG_DEBUG("[auto_init_saul] initializing vcnl40x0 #%u\n", i); + + if (vcnl40x0_init(&vcnl40x0_devs[i], + &vcnl40x0_params[i]) != VCNL40X0_OK) { + LOG_ERROR("[auto_init_saul] error initializing vcnl40x0 #%u\n", i); + return; + } + + /* proximity */ + saul_entries[(i * 2)].dev = &(vcnl40x0_devs[i]); + saul_entries[(i * 2)].name = vcnl40x0_saul_reg_info[i].name; + saul_entries[(i * 2)].driver = &vcnl40x0_proximity_saul_driver; + + /* illuminance */ + saul_entries[(i * 2) + 1].dev = &(vcnl40x0_devs[i]); + saul_entries[(i * 2) + 1].name = vcnl40x0_saul_reg_info[i].name; + saul_entries[(i * 2) + 1].driver = &vcnl40x0_illuminance_saul_driver; + + /* register to saul */ + saul_reg_add(&(saul_entries[(i * 2)])); + saul_reg_add(&(saul_entries[(i * 2) + 1])); + } +} +#else +typedef int dont_be_pedantic; +#endif /* MODULE_VCNL40X0 */ diff --git a/sys/include/phydat.h b/sys/include/phydat.h index b94aecb8a4bc2b361b059d3a36fe12d9ec40b3fb..946a9337039fdb1113a8d7fea8c201cb67dbae1a 100644 --- a/sys/include/phydat.h +++ b/sys/include/phydat.h @@ -101,6 +101,7 @@ enum { UNIT_CD, /**< Candela */ /* logical */ UNIT_BOOL, /**< boolean value [0|1] */ + UNIT_CTS, /**< counts */ UNIT_PERCENT, /**< out of 100 */ UNIT_PERMILL, /**< out of 1000 */ UNIT_PPM, /**< part per million */ diff --git a/sys/phydat/phydat_str.c b/sys/phydat/phydat_str.c index 7d5755c35dbe3a9d0647029a7dcafc8022c2004f..a10fd7aee64a80e2a07d133e2a259c29c9bd4506 100644 --- a/sys/phydat/phydat_str.c +++ b/sys/phydat/phydat_str.c @@ -98,6 +98,7 @@ const char *phydat_unit_to_str(uint8_t unit) case UNIT_PPB: return "ppb"; case UNIT_CD: return "cd"; case UNIT_PERCENT: return "%"; + case UNIT_CTS: return "cts"; default: return ""; } } diff --git a/tests/driver_vcnl40x0/Makefile b/tests/driver_vcnl40x0/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..fa901ba619be4b571eddada864c6231f6d2fff4d --- /dev/null +++ b/tests/driver_vcnl40x0/Makefile @@ -0,0 +1,7 @@ +APPLICATION = driver_vcnl40x0 +include ../Makefile.tests_common + +USEMODULE += vcnl4010 +USEMODULE += xtimer + +include $(RIOTBASE)/Makefile.include diff --git a/tests/driver_vcnl40x0/README.md b/tests/driver_vcnl40x0/README.md new file mode 100644 index 0000000000000000000000000000000000000000..f89dcf0e9c3f4f5fda6e997c455ba8e7f434c081 --- /dev/null +++ b/tests/driver_vcnl40x0/README.md @@ -0,0 +1,10 @@ +## About +This is a test application for the VCNL40X0 proximity and ambient light sensor. + +## Usage + +After initialization, every 2 seconds, the application: +* reads the proximity (cts); +* reads the ambient light (cts); +* reads the illuminance (computed from ambient light by dividing it by 4); +* those values are printed to STDOUT. diff --git a/tests/driver_vcnl40x0/main.c b/tests/driver_vcnl40x0/main.c new file mode 100644 index 0000000000000000000000000000000000000000..0ed5c7d1ca24050dd41204b5bece687b16145385 --- /dev/null +++ b/tests/driver_vcnl40x0/main.c @@ -0,0 +1,65 @@ +/* + * Copyright (C) 2017 Inria + * + * 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 VNCL40X0 proximity and ambient light sensor. + * @author Alexandre Abadie <alexandre.abadie@inria.fr> + * + * @} + */ + +#include <stdio.h> +#include <inttypes.h> + +#include "vcnl40x0.h" +#include "vcnl40x0_params.h" +#include "xtimer.h" +#include "board.h" + +#define SLEEP_2S (2U) /* 2 seconds delay between printf */ + +int main(void) +{ + vcnl40x0_t dev; + int result; + + puts("VCNL40X0 test application\n"); + + printf("+------------Initializing------------+\n"); + result = vcnl40x0_init(&dev, &vcnl40x0_params[0]); + if (result == -VCNL40X0_ERR_I2C) { + puts("[Error] The given i2c is not enabled"); + return 1; + } + else if (result == -VCNL40X0_ERR_NODEV) { + puts("[Error] The sensor did not answer correctly on the given address"); + return 1; + } + else { + printf("Initialization successful\n\n"); + } + + printf("\n+--------Starting Measurements--------+\n"); + while (1) { + printf("Proximity [cts]: %d\n" + "Ambient light [cts]: %d\n" + "Illuminance [lx]: %d\n" + "\n+-------------------------------------+\n", + vcnl40x0_read_proximity(&dev), + vcnl40x0_read_ambient_light(&dev), + vcnl40x0_read_illuminance(&dev)); + + xtimer_sleep(SLEEP_2S); + } + + return 0; +}