1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00

drivers/tsl4531x: Add power modes; redesign

Changes include:
- Implementation of two power modes for the driver: low and high
- Redesign and API change to the description in tsl4531x.h
- Full documentation
- Changing file structure and implementation to fit best practices as
  described in
  https://github.com/RIOT-OS/RIOT/wiki/Guide%3A-Writing-a-device-driver-in-RIOT
- Including I2C addresses for the rest of the range
This commit is contained in:
danpetry 2018-10-11 11:56:01 +02:00
parent 5604abcedc
commit a1a834a726
8 changed files with 538 additions and 150 deletions

View File

@ -446,6 +446,11 @@ ifneq (,$(filter tsl2561,$(USEMODULE)))
USEMODULE += xtimer
endif
ifneq (,$(filter tsl4531x,$(USEMODULE)))
FEATURES_REQUIRED += periph_i2c
USEMODULE += xtimer
endif
ifneq (,$(filter uart_half_duplex,$(USEMODULE)))
FEATURES_REQUIRED += periph_gpio
FEATURES_REQUIRED += periph_uart
@ -475,7 +480,3 @@ ifneq (,$(filter xbee,$(USEMODULE)))
USEMODULE += xtimer
USEMODULE += netif
endif
ifneq (,$(filter tsl4531x,$(USEMODULE)))
FEATURES_REQUIRED += periph_i2c
endif

View File

@ -242,6 +242,10 @@ ifneq (,$(filter tsl2561,$(USEMODULE)))
USEMODULE_INCLUDES += $(RIOTBASE)/drivers/tsl2561/include
endif
ifneq (,$(filter tsl4531x,$(USEMODULE)))
USEMODULE_INCLUDES += $(RIOTBASE)/drivers/tsl4531x/include
endif
ifneq (,$(filter uart_half_duplex,$(USEMODULE)))
USEMODULE_INCLUDES += $(RIOTBASE)/drivers/uart_half_duplex/include
endif

View File

@ -12,49 +12,184 @@
* @brief Device driver for the AMS TSL4531 sensor
* @{
*
* # Power modes
*
* This driver has two power modes: high and low. Its startup mode is configured
* in the initialization parameters, and it can also be changed during runtime.
*
* In high power mode, the user application can read from the device using the
* tsl4531x_simple_read function, and it will return immediately.
*
* In low power mode, the user application can interact with the driver in a
* synchronous or asynchronous manner. For synchronous operation, the application
* can call tsl4531x_simple_read, and the driver will block for the integration
* time defined in the initialization parameters (plus a small margin, to encompass
* the max times indicated in the datasheet). For asyncronous operation, the
* application needs to use the functions tsl4531x_start_sample,
* tsl4531x_time_until_sample_ready and tsl4531x_get_sample, as described in those
* functions' descriptions.
*
* Both modes will work through SAUL, with the low-power mode being synchronous.
*
* @file
* @brief Device driver for the AMS TSL4531 sensor
*
* @author Juan I Carrano <j.carrano@fu-berlin.de>
* @author Daniel Petry <daniel.petry@fu-berlin.de>
*/
#ifndef TSL4531x_H
#define TSL4531x_H
#ifndef TSL4531X_H
#define TSL4531X_H
#include <stdint.h>
#include "periph/i2c.h"
static const uint16_t TSL4531x_INTEGRATE_400ms = 0; /* 0b00 */
static const uint16_t TSL4531x_INTEGRATE_200ms = 1; /* 0b01 */
static const uint16_t TSL4531x_INTEGRATE_100ms = 2; /* 0b10 */
#ifdef __cplusplus
extern "C" {
#endif
/**
* @brief Represents a single TSL4531x device.
* @brief Integration times
*/
typedef struct tsl4531x {
i2c_t i2c_dev;
uint8_t integration_time;
typedef enum {
TSL4531X_INTEGRATE_400MS = 0, /* 0b00 */
TSL4531X_INTEGRATE_200MS = 1, /* 0b01 */
TSL4531X_INTEGRATE_100MS = 2, /* 0b10 */
}tsl4531x_intgn_time_t;
/**
* @name Fixed values for different members of the TSL4531x series
* @{
*
* @brief Part numbers
*/
#define TSL45311_PARTNO (0x8)
#define TSL45313_PARTNO (0x9)
#define TSL45315_PARTNO (0xA)
#define TSL45317_PARTNO (0xB)
/**
* @brief TSL4531x I2C addresses
*/
#define TSL45311_ADDR (0x39)
#define TSL45313_ADDR (0x39)
#define TSL45315_ADDR (0x29)
#define TSL45317_ADDR (0x29)
/** @} */
/**
* @brief Device initialization parameters
*/
typedef struct {
i2c_t i2c_dev; /**< I2C device which is used */
i2c_t i2c_addr; /**< I2C address of sensor */
tsl4531x_intgn_time_t integration_time; /**< integration time */
uint8_t low_power_mode : 1; /**< low power mode */
uint8_t part_number; /**< part number, according to variant */
} tsl4531x_params_t;
/**
* @brief Device descriptor
*/
typedef struct {
i2c_t i2c_dev; /**< I2C device which is used */
i2c_t i2c_addr; /**< I2C address of sensor */
tsl4531x_intgn_time_t integration_time; /**< integration time */
uint8_t low_power_mode : 1; /**< low power mode */
uint8_t high_power_mode_started_up : 1; /**< high power mode started up flag */
uint32_t sample_start_time; /**< sample start time */
} tsl4531x_t;
/**
* Initialize TSL4513 device.
* Initialize the TSL4531x device.
*
* @param dev Device object.
* @param i2c_dev I2C interface to use.
* @param integration_time one of TSL4531x_INTEGRATE_400ms,
* TSL4531x_INTEGRATE_200ms, TSL4531x_INTEGRATE_100ms.
* @param[out] dev Initialized device descriptor
* @param[in] params Device initialization parameters
*
* @return Zero on success, negative on error.
* @return Zero on success
* @return -ENODEV if I2C bus can't be acquired
* @return -ENXIO if device can't be read or configured
* @return -ENOTSUP if ID, once read, is wrong
*/
int tsl4531x_init(tsl4531x_t *dev, i2c_t i2c_dev, uint8_t integration_time);
int tsl4531x_init(tsl4531x_t *dev, const tsl4531x_params_t *params);
/**
* Get the last measurements of illuminance, in lux.
* Set the low power mode of the driver on or off.
*
* @return 0 on sucess, negative on error. If there is an error, the value of
* `*result_lux` is undefined.
* @param dev Initialized device descriptor
* @param low_power_on Bool indicating whether low power mode is on or off
*
* @return Zero
*/
int tsl4531x_read(const tsl4531x_t *dev, uint16_t *result_lux);
int tsl4531x_set_low_power_mode(tsl4531x_t *dev, uint8_t low_power_on);
#endif /* TSL4531x_H */
/**
* Start collecting sample in low power mode. This provides asynchronous operation
* along with tsl4531x_time_until_sample_ready and tsl4531x_get_sample. It does
* nothing in high power mode.
*
* @param dev Initialized device descriptor
*
* @return Zero
*/
int tsl4531x_start_sample(tsl4531x_t *dev);
/**
* Deliver time in microseconds until sample is ready, or zero if it is ready.
* In low power mode, this counts down from the point at which tsl4531x_start_sample
* is called, and along with that function and tsl4531x_get_sample, provides
* asynchronous operation. In high power mode, this counts down from the point
* at which the driver is switched into high power mode or started up, and
* indicates whether enough time has passed for a full sample to be collected.
*
* Note that for low power mode this rolls over and repeats its behaviour every
* 1.2 hours. The sample should have been collected well before this, however.
*
* The countdown time equals the integration time, which can be set in the
* device initialisation parameters, plus 5% margin to encompass the max times
* indicated in the datasheet.
*
* @param dev Initialized device descriptor
*
* @return Time in microseconds until sample is ready
*/
uint32_t tsl4531x_time_until_sample_ready(tsl4531x_t *dev);
/**
* Reads the sample from the device immediately. In high power mode, this does
* the same as tsl4531x_simple_read once the device has performed one
* integration cycle. In low power mode, this provides asynchronous operation
* along with tsl4531x_start_sample and tsl4531x_time_until_sample_ready which
* determine whether the device has performed an integration cycle.
*
* Note that this function will always return the value stored in the device's
* internal register, and this value will be sensible physically, representing
* the last time an integration cycle has been performed. However, in order for
* it to be accurate, the start_sample and time_until_sample_ready functions
* need to also be used, or alternatively the simple_read function can be used.
*
* @param dev Initialized device descriptor
*
* @return The sample value in Lux
*/
int tsl4531x_get_sample(const tsl4531x_t *dev);
/**
* Reads the sample from the device. In low power mode, or in high power mode
* just after startup, this starts collecting the sample, blocks until the
* sample is ready (400/200/100ms depending on the integration time set in the
* initialization parameters), and then reads and returns the sample.
*
* @param dev Initialized device descriptor
*
* @return The sample value in Lux
*/
int tsl4531x_simple_read(tsl4531x_t *dev);
#ifdef __cplusplus
}
#endif
#endif /* TSL4531X_H */
/** @} */

View File

@ -1,31 +0,0 @@
/*
* Copyright (C) 2018 Freie Universität Berlin
*
* 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_tsl4531x
* @{
*
* @file
* @brief SAUL interface for TSL4531 Luminosity sensor.
*
* @author Juan I Carrano <j.carrano@fu-berlin.de>
*
* @}
*/
#ifndef TSL4531x_SAUL_H
#define TSL4531x_SAUL_H
#include "saul.h"
/** SAUL-compatible structure for illuminance sensor
*/
const saul_driver_t tsl4531x_saul_driver;
#endif /* TSL4531x_SAUL_H */

View File

@ -0,0 +1,105 @@
/*
* Copyright (C) 2017 Inria
* Copyright (C) 2018 Freie Universität Berlin
*
* 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_tsl4531x
* @brief Internal addresses, registers, constants for the TSL4531x sensor.
* @{
*
* @file
* @brief Internal addresses, registers, constants for the TSL4531x sensor.
*
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
* @author Daniel Petry <danielpetry@cantab.net>
* @author Juan I Carrano <j.carrano@fu-berlin.de>
*
* Derived from the internals.h file for the tsl2561 driver.
*/
#ifndef TSL4531X_INTERNALS_H
#define TSL4531X_INTERNALS_H
#ifdef __cplusplus
extern "C" {
#endif
/**
* @name TSL4531x internal registers
* @{
*/
#define TSL4531X_CONTROL_REG (0x0) /**< sets power mode */
#define TSL4531X_CONFIGURATION_REG (0x01) /**< sets power mode */
#define TSL4531X_ALSDATA1_REG (0x04) /**< contains DATALOW */
#define TSL4531X_ALSDATA2_REG (0x05) /**< contains DATAHIGH */
#define TSL4531X_ID_REG (0x0A) /**< contains part no above*/
/** @} */
/**
* @name TSL4531x power modes
* @{
*/
#define TSL4531X_MODE_POWER_DOWN (0x00)
#define TSL4531X_MODE_RESERVED (0x01)
#define TSL4531X_MODE_SINGLE_ADC_CYCLE (0x02) /**< Runs a single ADC cycle then
returns to power down. */
#define TSL4531X_MODE_NORMAL (0x03)
/* PowerSave saves some power in full power mode. PowerSave skip turns this off.
* Currently PowerSave skip is hard-coded to be on for simplicity, as it's just
* an intermediate between normal mode and low-power mode. */
#define TSL4531X_PSAVESKIP_OFF (0)
#define TSL4531X_PSAVESKIP_ON (1)
/** @} */
/**
* @name TSL4531x macros
* @{
*/
/* Assemble the Command register */
#define TSL4531X_COMMAND(addr) ((1 << 7) | (addr))
/* Assemble the Configuration register */
#define TSL4531X_CONFIG(psaveskip, tcntrl) (((!!(psaveskip)) << 3) | (tcntrl))
/* Assemble the Control register */
#define TSL4531X_CONTROL(mode) (mode)
#define TSL4531X_GET_PARTNO(id_reg) ((id_reg) >> 4)
/* Data multiplier according to integration time.
*
* From the manual:
* MULTIPLIER = 1 for TCNTRL = 00 (Tint = 400 ms)
* MULTIPLIER = 2 for TCNTRL = 01 (Tint = 200 ms)
* MULTIPLIER = 4 for TCNTRL = 10 (Tint = 100 ms)
*/
#define MULTIPLY_DATA(data_raw, integration_time) ((data_raw) << (integration_time))
/* This allows either full power mode or power down mode. */
#define TSL4531X_POWER_MODE(mode) ((!mode) * (0x03))
/* The full integration time is a sum of:
* 1. A worst-case time for each of the integration times, which according to
* the datasheet is 5% more than the nominal time
* 2. 60, 30 or 15 ms, according to the integration time, if PowerSave is on;
* which is the power down period in between integration cycles in this mode.
* Note that in the current implementation, the "PowerSave skip" option is
* hard-coded to be on, as PowerSave only gives slightly less power
* consumption than normal mode.
*/
#define TSL4531X_GET_INTEGRATION_TIME_USEC(int_time_setting, psaveskip) \
(uint32_t)1000 * (((0x04 >> int_time_setting) * 105) + ((!psaveskip) * (60 >> int_time_setting)))
/** @} */
#ifdef __cplusplus
}
#endif
#endif /* TSL4531X_INTERNALS_H */
/** @} */

View File

@ -0,0 +1,87 @@
/*
* Copyright (C) 2017 Inria
* Copyright (C) 2018 Freie Universität Berlin
*
* 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_tsl4531x
*
* @{
* @file
* @brief Default configuration for tsl4531x light sensor.
*
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
* @author Daniel Petry <danielpetry@cantab.net>
*
* Derived from the default configuration for the tsl2561 driver.
*/
#ifndef TSL4531X_PARAMS_H
#define TSL4531X_PARAMS_H
#include "board.h"
#include "saul_reg.h"
#include "tsl4531x.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @name Set default configuration parameters for the TSL4531x
* @{
*/
#ifndef TSL4531X_PARAM_I2C_DEV
#define TSL4531X_PARAM_I2C_DEV I2C_DEV(0)
#endif
#ifndef TSL4531X_PARAM_I2C_ADDR
#define TSL4531X_PARAM_I2C_ADDR TSL45315_ADDR
#endif
#ifndef TSL4531X_PARAM_INTEGRATION
#define TSL4531X_PARAM_INTEGRATION TSL4531X_INTEGRATE_400MS
#endif
#ifndef TSL4531X_LOW_POWER_MODE
#define TSL4531X_LOW_POWER_MODE (false)
#endif
#ifndef TSL4531X_PARAM_PARTNO
#define TSL4531X_PARAM_PARTNO TSL45315_PARTNO
#endif
#ifndef TSL4531X_PARAMS
#define TSL4531X_PARAMS { .i2c_dev = TSL4531X_PARAM_I2C_DEV, \
.i2c_addr = TSL4531X_PARAM_I2C_ADDR, \
.integration_time = TSL4531X_PARAM_INTEGRATION, \
.low_power_mode = TSL4531X_LOW_POWER_MODE, \
.part_number = TSL4531X_PARAM_PARTNO }
#endif
#ifndef TSL4531X_SAUL_INFO
#define TSL4531X_SAUL_INFO { .name = "tsl4531x" }
#endif
/**@}*/
/**
* @brief Configure TSL4531x
*/
static const tsl4531x_params_t tsl4531x_params[] =
{
TSL4531X_PARAMS
};
/**
* @brief Allocate and configure entries to the SAUL registry
*/
saul_reg_info_t tsl4531x_saul_info[] =
{
TSL4531X_SAUL_INFO
};
#ifdef __cplusplus
}
#endif
#endif /* TSL4531X_PARAMS_H */
/** @} */

View File

@ -1,4 +1,5 @@
/*
* Copyright (C) 2016 Inria
* Copyright (C) 2018 Freie Universität Berlin
*
* This file is subject to the terms and conditions of the GNU Lesser
@ -11,15 +12,13 @@
* @{
*
* @file
* @brief Device driver for the TSL4531 Luminosity sensor.
* @brief Device driver for the TSL4531x Luminosity sensor.
*
* @author Daniel Petry <daniel.petry@fu-berlin.de>
* @author Juan I Carrano <j.carrano@fu-berlin.de>
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
*
* This driver was derived from the one for TSL2561.
*
* @todo The i2c address is hardcoded
* @todo The only operating mode is "Normal Operation"
* @todo PowerSave is not supported
* This driver was derived from the TSL2561 driver.
*
* @}
*/
@ -28,113 +27,202 @@
#include "log.h"
#include "tsl4531x.h"
#include "tsl4531x_internals.h"
#include "xtimer.h"
#define ENABLE_DEBUG (0)
#include "debug.h"
static const uint16_t TSL4531_Addr = 0x29;
static const uint16_t TSL4531_PARTNO = 0xA; /* 0b1010 */
#define _DATALOW 0
#define _DATAHIGH 1
static const uint16_t Control_Reg = 0x0;
static const uint16_t Configuration_Reg = 0x01;
static const uint16_t ALSData1_Reg = 0x04; /* contains DATALOW */
static const uint16_t ALSData2_Reg = 0x05; /* contains DATAHIGH */
static const uint16_t ID_Reg = 0x0A;
static const uint16_t TSL4531x_MODE_Normal = 0x03; /* 0b11 */
/* Assemble the Command register */
#define TSL4531_COMMAND(addr) ((1<<7)|(addr))
/* Assemble the Configuration register */
#define TSL4531_CONFIG(psaveskip, tcntrl) (((!!(psaveskip)) << 3) | (tcntrl))
/* Assemble the Control register */
#define TSL4531_CONTROL(mode) (mode)
#define TSL4531_GET_PARTNO(id_reg) ((id_reg) >> 4)
/* Apply the multiplier corresponding to the active integration time */
/* From the manual:
* MULTIPLIER = 1 for TCNTRL = 00 (Tint = 400 ms),
* MULTIPLIER = 2 for TCNTRL = 01 (Tint = 200 ms), and
* MULTIPLIER = 4 for TCNTRL = 10 (Tint = 100 ms), and
*/
#define MULTIPLY_DATA(data_raw, integration_time) ((data_raw) << (integration_time))
int tsl4531x_init(tsl4531x_t *dev, i2c_t i2c_dev, uint8_t integration_time)
int tsl4531x_init(tsl4531x_t *dev, const tsl4531x_params_t *params)
{
int r;
uint8_t id;
dev->i2c_dev = i2c_dev;
dev->integration_time = integration_time;
if ((r = i2c_acquire(dev->i2c_dev)) < 0) {
DEBUG("[Error] Cannot acquire device: %d\n", r);
goto acquire_error;
/* Initialise I2C bus */
if ((r = i2c_acquire(params->i2c_dev)) < 0) {
DEBUG("I2C_dev is: %d.", params->i2c_dev);
DEBUG("[Error] Cannot acquire device. I2C error: %d\n", r);
return -ENODEV;
}
/* Verify sensor ID */
if ((r = i2c_read_reg(dev->i2c_dev, TSL4531_Addr, TSL4531_COMMAND(ID_Reg),
/* Test for connectivity - verify ID and compare against stored value */
if ((r = i2c_read_reg(params->i2c_dev,
params->i2c_addr,
TSL4531X_COMMAND(TSL4531X_ID_REG),
&id, 0)) < 0) {
DEBUG("[Error] Cannot read ID register: %d\n", r);
goto init_error;
}
DEBUG("[Info] ID ? %d\n", id);
if (TSL4531_GET_PARTNO(id) != TSL4531_PARTNO) {
DEBUG("[Error] not a TSL4531 sensor\n");
r = -ENOTSUP;
goto init_error;
DEBUG("[Error] Cannot read ID register. I2C error: %d\n", r);
i2c_release(params->i2c_dev);
return -ENXIO;
}
/* configure device */
if (((r = i2c_write_reg(dev->i2c_dev , TSL4531_Addr, TSL4531_COMMAND(Control_Reg),
TSL4531_CONTROL(TSL4531x_MODE_Normal), 0)) < 0)
||
((r = i2c_write_reg(dev->i2c_dev , TSL4531_Addr, TSL4531_COMMAND(Configuration_Reg),
TSL4531_CONFIG(1, integration_time), 0)) < 0)) {
DEBUG("[Error] Cannot configure device %d\n", r);
goto init_error;
DEBUG("[Info] tsl4531x sensor ID ? %d\n", id);
if (TSL4531X_GET_PARTNO(id) != params->part_number) {
DEBUG("[Error] not a TSL4531 sensor.\n");
i2c_release(params->i2c_dev);
return -ENOTSUP;
}
init_error:
/* ignore errors on device release */
i2c_release(dev->i2c_dev);
/* Configure device. In low power mode, we initially power the sensor down. */
if (((r = i2c_write_reg(params->i2c_dev,
params->i2c_addr,
TSL4531X_COMMAND(TSL4531X_CONTROL_REG),
TSL4531X_CONTROL(TSL4531X_POWER_MODE(params->low_power_mode)),
0)) < 0)
||
((r = i2c_write_reg(params->i2c_dev,
params->i2c_addr,
TSL4531X_COMMAND(TSL4531X_CONFIGURATION_REG),
TSL4531X_CONFIG(TSL4531X_PSAVESKIP_ON, params->integration_time),
0)) < 0)) {
acquire_error:
DEBUG("[Error] Cannot configure device. I2C error: %d\n", r);
i2c_release(params->i2c_dev);
return -ENXIO;
}
return r;
/* If device was configured correctly, initialise the device descriptor */
dev->i2c_dev = params->i2c_dev;
dev->i2c_addr = params->i2c_addr;
dev->integration_time = params->integration_time;
dev->low_power_mode = params->low_power_mode;
dev->high_power_mode_started_up = false;
dev->sample_start_time = 0; /* Device assumed to start up at same time as
microcontroller - i.e. when it hits this line */
i2c_release(params->i2c_dev);
return 0;
}
#define _DATALOW 0
#define _DATAHIGH 1
int tsl4531x_read(const tsl4531x_t *dev, uint16_t *result_lux)
int tsl4531x_set_low_power_mode(tsl4531x_t *dev, uint8_t low_power_mode)
{
assert(dev);
int r;
dev->low_power_mode = low_power_mode;
if ((r = i2c_acquire(dev->i2c_dev)) < 0) {
DEBUG("[Error] Cannot acquire device. I2C error: %d\n", r);
return -ENODEV;
}
if ((r = i2c_write_reg(dev->i2c_dev,
dev->i2c_addr,
TSL4531X_COMMAND(TSL4531X_CONTROL_REG),
TSL4531X_CONTROL(TSL4531X_POWER_MODE(low_power_mode)),
0)) < 0) {
DEBUG("[Error] Cannot write power mode. I2C error: %d\n", r);
i2c_release(dev->i2c_dev);
return -ENXIO;
}
i2c_release(dev->i2c_dev);
/* In high power mode only, we restart the sample ready timer, because only
in this mode it's used to indicate readiness after startup. */
if (!dev->low_power_mode) {
dev->sample_start_time = xtimer_now_usec();
}
return 0;
}
int tsl4531x_start_sample(tsl4531x_t *dev)
{
assert(dev);
/* Don't change the mode to one-shot if the device is in high power mode. */
if (dev->low_power_mode) {
int r;
if ((r = i2c_acquire(dev->i2c_dev)) < 0) {
DEBUG("[Error] Cannot acquire device. I2C error: %d\n", r);
return -ENODEV;
}
if ((r = i2c_write_reg(dev->i2c_dev,
dev->i2c_addr,
TSL4531X_COMMAND(TSL4531X_CONTROL_REG),
TSL4531X_CONTROL(TSL4531X_MODE_SINGLE_ADC_CYCLE),
0)) < 0) {
DEBUG("[Error] Cannot write power mode. I2C error: %d\n", r);
i2c_release(dev->i2c_dev);
return -ENXIO;
}
i2c_release(dev->i2c_dev);
dev->sample_start_time = xtimer_now_usec();
}
return 0;
}
uint32_t tsl4531x_time_until_sample_ready(tsl4531x_t *dev)
{
assert(dev);
uint32_t t = TSL4531X_GET_INTEGRATION_TIME_USEC(dev->integration_time, TSL4531X_PSAVESKIP_ON) -
(xtimer_now_usec() - dev->sample_start_time);
/* Clamp t at zero */
t = (t <= TSL4531X_GET_INTEGRATION_TIME_USEC(dev->integration_time, TSL4531X_PSAVESKIP_ON) ?
t : 0);
if (!dev->low_power_mode) {
if (t == 0) {
dev->high_power_mode_started_up = true;
}
if (dev->high_power_mode_started_up) {
return 0;
}
}
return t;
}
int tsl4531x_get_sample(const tsl4531x_t *dev)
{
assert(dev);
int r;
uint8_t als_data[2]; /* = {[DATALOW], [DATAHIGH]} */
if ((r = i2c_acquire(dev->i2c_dev)) < 0) {
DEBUG("[Error] Cannot acquire device: %d\n", r);
goto read_acquire_error;
DEBUG("[Error] Cannot acquire device. I2C error: %d\n", r);
return -ENODEV;
}
if ((r = i2c_read_regs(dev->i2c_dev, TSL4531_Addr, TSL4531_COMMAND(ALSData1_Reg),
als_data, 2, 0)) < 0) {
(void)ALSData2_Reg; /* suppress warning */
DEBUG("[Error] Cannot read device: %d\n", r);
goto read_error;
if ((r = i2c_read_regs(dev->i2c_dev,
dev->i2c_addr,
TSL4531X_COMMAND(TSL4531X_ALSDATA1_REG),
als_data, 2, 0)) < 0) {
DEBUG("[Error] Cannot read data register. I2C error: %d\n", r);
i2c_release(dev->i2c_dev);
return -ENXIO;
}
*result_lux = MULTIPLY_DATA((((uint16_t)als_data[_DATAHIGH]) << 8)
+ als_data[_DATALOW], dev->integration_time);
read_error:
/* ignore errors on device release */
i2c_release(dev->i2c_dev);
read_acquire_error:
return r;
return MULTIPLY_DATA((((uint16_t)als_data[_DATAHIGH]) << 8)
+ als_data[_DATALOW], dev->integration_time);
}
int tsl4531x_simple_read(tsl4531x_t *dev)
{
assert(dev);
if (dev->low_power_mode) {
tsl4531x_start_sample(dev);
}
xtimer_usleep(tsl4531x_time_until_sample_ready(dev));
return tsl4531x_get_sample(dev);
}

View File

@ -1,4 +1,5 @@
/*
* Copyright (C) 2016 Inria
* Copyright (C) 2018 Freie Universität Berlin
*
* This file is subject to the terms and conditions of the GNU Lesser
@ -11,27 +12,25 @@
* @{
*
* @file
* @brief SAUL interface for TSL4531 Luminosity sensor.
* @brief SAUL interface for TSL4531x Luminosity sensor.
*
* @author Juan I Carrano <j.carrano@fu-berlin.de>
* @author Daniel Petry <daniel.petry@fu-berlin.de>
*
* @}
*/
#include <string.h>
#include "saul.h"
#include "tsl4531x.h"
static int _read(const void *dev, phydat_t *res)
static int _read(const void *dev, phydat_t *data)
{
int nvals;
uint16_t lux;
nvals = (tsl4531x_read(dev, &lux) >= 0)? 1 : 0;
res->val[0] = lux;
res->unit = UNIT_LUX;
res->scale = 0;
return nvals;
memset(data, 0, sizeof(phydat_t));
data->val[0] = tsl4531x_simple_read((const tsl4531x_t *)dev);
data->unit = UNIT_LUX;
data->scale = 0;
return 1;
}
const saul_driver_t tsl4531x_saul_driver = {