/* * Copyright (C) 2020 iosabi * * 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_si1133 * @{ * * @file * @brief Device driver implementation for the SI1133 UV/IR/Ambient light * sensor with I2C interface. * * @author iosabi <iosabi@protonmail.com> * * @} */ #include <math.h> #include <string.h> #include <stdint.h> #include <stdbool.h> #include "xtimer.h" #include "periph/i2c.h" #include "si1133.h" #include "si1133_internals.h" #define ENABLE_DEBUG 0 #include "debug.h" /** * @brief The command execution timeout in blocking mode. * * When forcing a measurement (SI1133_CMD_FORCE) the command is not done until * all the measurements from all the channels are taken. This may take some time * depending on the configuration. After the expected sampling time is over we * wait up to this command timeout value for the command to be ready. */ #define SI1133_COMMAND_TIMEOUT_USEC 10000 /** * @brief Return the expected sampling time in us for a FORCE command. */ static uint32_t _si1133_force_time_us(si1133_t *dev) { /* Sample time measured in 512 clocks of the 21 MHz, or about 24.4 us. */ uint32_t sample_time = 0; for (uint32_t i = 0; i < dev->num_channels; i++) { /* A forced sample consists of (1 << sw_gain) measurements where each * one has a total 155 us (about 7 * 24.4 us) of processing time, plus * two ADC samples. Each ADC sample requires a t_adstart time (48.8 us * or 2 * 24.4 us) plus the time configured by the sample_time_log. */ sample_time += (2 * (2 + (1u << dev->channel[i].sample_time_log)) + 7) << dev->channel[i].sw_gain; } /* The resulting sample_time value in microseconds is sample_time * 512 / 21 * the sample_time is already a 24-bit number so we split the logic to * fit in 32-bit arithmetic using the fact that 512 == 24 * 21 + 8. */ return sample_time * 24 + sample_time * 8 / 21; } /** * @brief Run a command with no arguments. * * For commands CMD_PARAM_SET or CMD_PARAM_QUERY use @ref _si1133_set_param * and @ref _si1133_get_param respectively instead. */ static int _si1133_run_command(si1133_t *dev, uint8_t command) { DEBUG("[si1133] run_command: 0x%.2x, cmd_counter=%d\n", (unsigned)command, dev->cmd_counter); int ret; ret = i2c_write_reg(dev->i2c_dev, dev->address, SI1133_REG_COMMAND, command, 0 /* flags */); if (ret) { DEBUG("[si1133] write command I2C error: %s\n", strerror(-ret)); return SI1133_ERR_I2C; } if (command == SI1133_CMD_FORCE) { /* Wait for the expected force acquisition time. */ xtimer_usleep(_si1133_force_time_us(dev)); } if (command == SI1133_CMD_RESET_SW) { /* Reset command sets the command counter of 0x0f. */ dev->cmd_counter = 0x0f; /* Reset command puts the device in "Initialization Mode" which requires * us to wait until the device is ready. */ xtimer_msleep(SI1133_STARTUP_TIME_MS); } else if (command == SI1133_CMD_RESET_CMD_CTR) { /* The reset cmd_counter command, well, resets it to 0. */ dev->cmd_counter = 0; } else { /* Increment the expected 4-bit command counter value. */ dev->cmd_counter = (dev->cmd_counter + 1) & SI1133_RESP0_COUNTER_MASK; } uint8_t new_cmd_ctr; xtimer_ticks32_t start_time; bool retry = false; while (1) { ret = i2c_read_reg(dev->i2c_dev, dev->address, SI1133_REG_RESPONSE0, &new_cmd_ctr, 0 /* flags */); if (ret) { DEBUG("[si1133] read RESPONSE0 I2C error: %s\n", strerror(-ret)); return SI1133_ERR_I2C; } if (new_cmd_ctr & SI1133_RESP0_CMD_ERR_MASK) { DEBUG("[si1133] Command 0x%.2x returned error %d\n", (unsigned)command, new_cmd_ctr & SI1133_RESP0_COUNTER_MASK); /* Error code 2 is "ADC or accumulation overflow", while error code * 3 is output buffer overflow which can only occur in BURST mode. * However, in FORCE mode after the first overflow we can get an * error code 3 if we change the settings in between the overflow * and a new FORCE command, probably due to a silicon bug when * handling sw_gain overflows since it is not possible to write more * than 18 bytes of output in FORCE mode while the output buffer is * 26 bytes long. */ if ((new_cmd_ctr & SI1133_RESP0_COUNTER_MASK) >= 2) { return SI1133_ERR_OVERFLOW; } return SI1133_ERR_LOGIC; } /* The reset command is done when the RUNNING flag is clear, the other * commands are done when the command value is set to the expected * one. */ if ((command == SI1133_CMD_RESET_SW) ? !(new_cmd_ctr & SI1133_RESP0_RUNNING_MASK) : (dev->cmd_counter == (new_cmd_ctr & SI1133_RESP0_COUNTER_MASK))) { break; } /* The command didn't yet finish in this case so it should be in running * state and we need to retry the loop with a timeout. This avoids * calling xtimer for commands that are immediate. */ if (retry) { if (xtimer_usec_from_ticks(xtimer_diff(xtimer_now(), start_time)) > SI1133_COMMAND_TIMEOUT_USEC) { DEBUG("[si1133] Command 0x%.2x timeout.\n", (unsigned)command); return SI1133_ERR_LOGIC; } } else { retry = true; start_time = xtimer_now(); } } if (retry) { DEBUG("[si1133] Command overtime: %" PRIu32 " us.\n", xtimer_usec_from_ticks(xtimer_diff(xtimer_now(), start_time))); } return SI1133_OK; } static int _si1133_set_param(si1133_t *dev, uint8_t param, uint8_t value) { int ret; ret = i2c_write_reg(dev->i2c_dev, dev->address, SI1133_REG_HOSTIN0, value, 0 /* flags */); if (ret) { DEBUG("[si1133] write HOSTIN0 I2C error: %s\n", strerror(-ret)); return SI1133_ERR_I2C; } ret = _si1133_run_command(dev, param | SI1133_CMD_PARAM_SET); if (ret) { return ret; } uint8_t resp1; ret = i2c_read_reg(dev->i2c_dev, dev->address, SI1133_REG_RESPONSE1, &resp1, 0); if (ret) { DEBUG("[si1133] read RESPONSE1 I2C error: %s\n", strerror(-ret)); return SI1133_ERR_I2C; } if (resp1 != value) { DEBUG("[si1133] Expected to read back value 0x%.2" PRIu8 " when setting param 0x%.2" PRIu8 " but got 0x%.2" PRIu8 "\n", value, param, resp1); return SI1133_ERR_LOGIC; } return SI1133_OK; } /** * @brief Reset the device. */ static int _si1133_reset(si1133_t *dev) { DEBUG("[si1133] reset()\n"); dev->num_channels = 0; for (uint32_t i = 0; i < SI1133_NUM_CHANNELS; i++) { /* Initialize the config with invalid values to force the first config * call to update it. */ dev->channel[i].sw_gain = 0xff; } int ret = _si1133_run_command(dev, SI1133_CMD_RESET_SW); if (ret) { return ret; } return _si1133_run_command(dev, SI1133_CMD_RESET_CMD_CTR); } /** * @brief Configure a single channel with the passed parameters. */ static si1133_ret_code_t _si1133_configure_channel( si1133_t *dev, uint32_t channel_id, const si1133_channel_t *channel) { if (channel->sample_time_log > 14 || channel->sw_gain > 7) { return SI1133_ERR_PARAMS; } if (!memcmp(channel, &dev->channel[channel_id], sizeof(si1133_channel_t))) { /* Avoid the I2C roundtrip if the channel configuration didn't * change. */ return SI1133_OK; } /* Generate the channel configuration. */ si1133_channel_params_t config; #define SI1133_SENS_CASE(x) \ case SI1133_SENS_ ## x: \ config.adcconfig = \ SI1133_ADCMUX_ ## x << SI1133_ADCCONFIG_ADCMUX_SHIFT; \ break; switch (channel->sensor) { SI1133_SENS_CASE(SMALL_IR) SI1133_SENS_CASE(MEDIUM_IR) SI1133_SENS_CASE(LARGE_IR) SI1133_SENS_CASE(WHITE) SI1133_SENS_CASE(LARGE_WHITE) SI1133_SENS_CASE(UV) SI1133_SENS_CASE(DEEP_UV) default: return SI1133_ERR_PARAMS; } /* Use normal decimation (1) except in the extremes where we must use * the other decimation values. */ const uint8_t hw_gain = channel->sample_time_log == 0 ? 0 : (channel->sample_time_log <= 11 ? channel->sample_time_log - 1 : 11); /* Select decimation. A value of "0" in this field is the "normal" * decimation, which corresponds to 1 in our "decimation" equation. * The values in this hardware field are offset by 3. */ const uint8_t decimation = (channel->sample_time_log - hw_gain + 3) & 3; config.adcconfig |= decimation << SI1133_ADCCONFIG_DECIM_RATE_SHIFT; /* HSIG = 0, SW_GAIN and HW_GAIN as configured. */ config.adcsens = (hw_gain << SI1133_ADCSENS_HW_GAIN_SHIFT) | (channel->sw_gain << SI1133_ADCSENS_SW_GAIN_SHIFT); /* 24-bit output, no output shift, no threshold. */ config.adcpost = SI1133_ADCPOST_24BIT_OUT_MASK; /* No counter, this will only be used in Force mode. */ config.measconfig = 0; DEBUG("[si1133] config: %.2x %.2x %.2x %.2x\n", ((uint8_t *)&config)[0], ((uint8_t *)&config)[1], ((uint8_t *)&config)[2], ((uint8_t *)&config)[3]); for (uint8_t i = 0; i < sizeof(config); i++) { uint8_t param = SI1133_PARAM_ADCCONFIG0 + sizeof(config) * channel_id + i; si1133_ret_code_t ret = _si1133_set_param(dev, param, ((uint8_t *)&config)[i]); if (ret) { return ret; } } memcpy(&(dev->channel[channel_id]), channel, sizeof(si1133_channel_t)); return SI1133_OK; } static int _si1133_read_values(si1133_t *dev, int32_t *values, uint32_t num_channels) { /* We can read all registers in a single I2C burst. */ uint8_t data[3 * num_channels]; /* We only request 24-bit values from the device so they are all 3 byte * long. */ int ret = i2c_read_regs(dev->i2c_dev, dev->address, SI1133_REG_HOSTOUTx, data, 3 * num_channels, 0 /* flags */); if (ret) { return SI1133_ERR_I2C; } uint8_t *offset = data; for (uint8_t i = 0; i < num_channels; i++) { /* Sign-extend the first 8-bit value before shifting. */ int32_t value = ((int32_t)(int8_t)*(offset++)) << 16; value |= *(offset++) << 8u; value |= *(offset++); values[i] = value; } return SI1133_OK; } si1133_ret_code_t si1133_init(si1133_t *dev, const si1133_params_t *params) { dev->i2c_dev = params->i2c_dev; dev->address = params->address; /* After leaving "Off Mode" the SI1133 enters an "Initialization Mode" for * a period of time in which it can't be reached over I2C. After this time * the device will be in Standby Mode. */ xtimer_msleep(SI1133_STARTUP_TIME_MS); i2c_acquire(params->i2c_dev); /* check sensor ID */ uint8_t partid = 0; int ret = i2c_read_reg(params->i2c_dev, params->address, SI1133_REG_PART_ID, &partid, 0); if (ret != 0) { DEBUG("[si1133] i2c communication error: %s.\n", strerror(-ret)); i2c_release(params->i2c_dev); return SI1133_ERR_I2C; } if (partid != SI1133_ID) { DEBUG("[si1133] Invalid part id: 0x%.2u\n", (unsigned)partid); i2c_release(params->i2c_dev); return SI1133_ERR_NODEV; } #if ENABLE_DEBUG uint8_t rev_id, hw_id; i2c_read_reg(params->i2c_dev, params->address, SI1133_REG_REV_ID, &rev_id, 0 /* flags */); i2c_read_reg(params->i2c_dev, params->address, SI1133_REG_HW_ID, &hw_id, 0 /* flags */); DEBUG("[si1133] impl code: %u, silicon HW rev: %u, rev: %u.%u\n", hw_id & 0x1f, hw_id >> 5, rev_id >> 4, rev_id & 0x0f); #endif /* ENABLE_DEBUG */ /* We don't know the state in which the device is at this point so we need * to perform a reset, unfortunately this requires another start-up wait. */ ret = _si1133_reset(dev); if (ret) { i2c_release(dev->i2c_dev); return ret; } i2c_release(dev->i2c_dev); return SI1133_OK; } si1133_ret_code_t si1133_configure_channels(si1133_t *dev, const si1133_channel_t *channels, uint32_t num_channels) { DEBUG("[si1133] configure_channels(num=%" PRIu32 ")\n", num_channels); if (num_channels > SI1133_NUM_CHANNELS) { return SI1133_ERR_PARAMS; } i2c_acquire(dev->i2c_dev); si1133_ret_code_t ret; for (uint32_t i = 0; i < num_channels; i++) { ret = _si1133_configure_channel(dev, i, channels + i); if (ret) { dev->num_channels = 0; i2c_release(dev->i2c_dev); return ret; } } /* CHAN_LIST is a bit mask of channels used. */ ret = _si1133_set_param(dev, SI1133_PARAM_CHAN_LIST, (1u << num_channels) - 1); if (ret) { num_channels = 0; } dev->num_channels = num_channels; i2c_release(dev->i2c_dev); DEBUG("[si1133] Sample Time %" PRIu32 " us\n", _si1133_force_time_us(dev)); return ret; } si1133_ret_code_t si1133_easy_configure(si1133_t *dev, si1133_sensor_t sensor_mask, uint8_t sample_time_log, uint8_t sw_gain) { DEBUG("[si1133] easy_configure(0x%.2x)\n", (unsigned)sensor_mask); i2c_acquire(dev->i2c_dev); si1133_ret_code_t ret; uint8_t num_channels = 0; for (uint8_t mask = sensor_mask; mask; num_channels++, mask = mask & (mask - 1)) { if (num_channels >= SI1133_NUM_CHANNELS) { dev->num_channels = 0; i2c_release(dev->i2c_dev); return SI1133_ERR_PARAMS; } si1133_channel_t channel; channel.sensor = mask ^ (mask & (mask - 1)); channel.sample_time_log = sample_time_log; channel.sw_gain = sw_gain; ret = _si1133_configure_channel(dev, num_channels, &channel); if (ret) { dev->num_channels = 0; i2c_release(dev->i2c_dev); return ret; } } /* CHAN_LIST is a bit mask of channels used. */ ret = _si1133_set_param(dev, SI1133_PARAM_CHAN_LIST, (1u << num_channels) - 1); if (ret) { num_channels = 0; } dev->num_channels = num_channels; i2c_release(dev->i2c_dev); DEBUG("[si1133] Sample Time %" PRIu32 " us\n", _si1133_force_time_us(dev)); return ret; } si1133_ret_code_t si1133_capture_sensors(si1133_t *dev, int32_t *values, uint32_t num_channels) { if (!dev->num_channels) { /* Must be configured before calling capture_sensors. */ return SI1133_ERR_PARAMS; } i2c_acquire(dev->i2c_dev); si1133_ret_code_t force_ret; force_ret = _si1133_run_command(dev, SI1133_CMD_FORCE); if (force_ret != SI1133_OK && force_ret != SI1133_ERR_OVERFLOW) { i2c_release(dev->i2c_dev); DEBUG("[si1133] force read command error: %d\n", force_ret); return force_ret; } si1133_ret_code_t ret = SI1133_OK; if (force_ret == SI1133_ERR_OVERFLOW) { /* We need to reset the overflow condition with a RESET_CMD_CTR */ ret = _si1133_run_command(dev, SI1133_CMD_RESET_CMD_CTR); } if (ret == SI1133_OK) { ret = _si1133_read_values(dev, values, num_channels); } i2c_release(dev->i2c_dev); /* If there was an error reading the I2C values then return that error, * otherwise we want to return the CMD_FORCE return value because there * could be an overflow non-fatal error to report. */ return ret ? ret : force_ret; }