zephyr/drivers/adc/adc_ads1x1x.c
Benedikt Schmidt 49ce054200 drivers: adc: fix thread function signatures
Fix the data acquisition thread function signatures to avoid a stack
corruption on thread exit.
Fixes #62637

Signed-off-by: Benedikt Schmidt <benedikt.schmidt@embedded-solutions.at>
2023-10-30 12:24:34 +01:00

701 lines
20 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/* TI ADS1X1X ADC
*
* Copyright (c) 2021 Facebook, Inc
*
* SPDX-License-Identifier: Apache-2.0
*/
#include <stdbool.h>
#include <zephyr/device.h>
#include <zephyr/devicetree.h>
#include <zephyr/drivers/adc.h>
#include <zephyr/logging/log.h>
#include <zephyr/drivers/i2c.h>
#include <zephyr/kernel.h>
#include <zephyr/sys/byteorder.h>
#include <zephyr/sys/util.h>
#define ADC_CONTEXT_USES_KERNEL_TIMER
#include "adc_context.h"
LOG_MODULE_REGISTER(ADS1X1X, CONFIG_ADC_LOG_LEVEL);
#define ADS1X1X_CONFIG_OS BIT(15)
#define ADS1X1X_CONFIG_MUX(x) ((x) << 12)
#define ADS1X1X_CONFIG_PGA(x) ((x) << 9)
#define ADS1X1X_CONFIG_MODE BIT(8)
#define ADS1X1X_CONFIG_DR(x) ((x) << 5)
#define ADS1X1X_CONFIG_COMP_MODE BIT(4)
#define ADS1X1X_CONFIG_COMP_POL BIT(3)
#define ADS1X1X_CONFIG_COMP_LAT BIT(2)
#define ADS1X1X_CONFIG_COMP_QUE(x) (x)
enum ads1x1x_reg {
ADS1X1X_REG_CONV = 0x00,
ADS1X1X_REG_CONFIG = 0x01,
ADS1X1X_REG_LO_THRESH = 0x02,
ADS1X1X_REG_HI_THRESH = 0x03,
};
enum {
ADS1X15_CONFIG_MUX_DIFF_0_1 = 0,
ADS1X15_CONFIG_MUX_DIFF_0_3 = 1,
ADS1X15_CONFIG_MUX_DIFF_1_3 = 2,
ADS1X15_CONFIG_MUX_DIFF_2_3 = 3,
ADS1X15_CONFIG_MUX_SINGLE_0 = 4,
ADS1X15_CONFIG_MUX_SINGLE_1 = 5,
ADS1X15_CONFIG_MUX_SINGLE_2 = 6,
ADS1X15_CONFIG_MUX_SINGLE_3 = 7,
};
enum {
/* ADS111X, ADS101X samples per second */
/* 8, 128 samples per second */
ADS1X1X_CONFIG_DR_8_128 = 0,
/* 16, 250 samples per second */
ADS1X1X_CONFIG_DR_16_250 = 1,
/* 32, 490 samples per second */
ADS1X1X_CONFIG_DR_32_490 = 2,
/* 64, 920 samples per second */
ADS1X1X_CONFIG_DR_64_920 = 3,
/* 128, 1600 samples per second (default) */
ADS1X1X_CONFIG_DR_128_1600 = 4,
/* 250, 2400 samples per second */
ADS1X1X_CONFIG_DR_250_2400 = 5,
/* 475, 3300 samples per second */
ADS1X1X_CONFIG_DR_475_3300 = 6,
/* 860, 3300 samples per second */
ADS1X1X_CONFIG_DR_860_3300 = 7,
/* Default data rate */
ADS1X1X_CONFIG_DR_DEFAULT = ADS1X1X_CONFIG_DR_128_1600
};
enum {
/* +/-6.144V range = Gain 1/3 */
ADS1X1X_CONFIG_PGA_6144 = 0,
/* +/-4.096V range = Gain 1/2 */
ADS1X1X_CONFIG_PGA_4096 = 1,
/* +/-2.048V range = Gain 1 (default) */
ADS1X1X_CONFIG_PGA_2048 = 2,
/* +/-1.024V range = Gain 2 */
ADS1X1X_CONFIG_PGA_1024 = 3,
/* +/-0.512V range = Gain 4 */
ADS1X1X_CONFIG_PGA_512 = 4,
/* +/-0.256V range = Gain 8 */
ADS1X1X_CONFIG_PGA_256 = 5
};
enum {
ADS1X1X_CONFIG_MODE_CONTINUOUS = 0,
ADS1X1X_CONFIG_MODE_SINGLE_SHOT = 1,
};
enum {
/* Traditional comparator with hysteresis (default) */
ADS1X1X_CONFIG_COMP_MODE_TRADITIONAL = 0,
/* Window comparator */
ADS1X1X_CONFIG_COMP_MODE_WINDOW = 1
};
enum {
/* ALERT/RDY pin is low when active (default) */
ADS1X1X_CONFIG_COMP_POLARITY_ACTIVE_LO = 0,
/* ALERT/RDY pin is high when active */
ADS1X1X_CONFIG_COMP_POLARITY_ACTIVE_HI = 1
};
enum {
/* Non-latching comparator (default) */
ADS1X1X_CONFIG_COMP_NON_LATCHING = 0,
/* Latching comparator */
ADS1X1X_CONFIG_COMP_LATCHING = 1
};
enum {
/* Assert ALERT/RDY after one conversions */
ADS1X1X_CONFIG_COMP_QUEUE_1 = 0,
/* Assert ALERT/RDY after two conversions */
ADS1X1X_CONFIG_COMP_QUEUE_2 = 1,
/* Assert ALERT/RDY after four conversions */
ADS1X1X_CONFIG_COMP_QUEUE_4 = 2,
/* Disable the comparator and put ALERT/RDY in high state (default) */
ADS1X1X_CONFIG_COMP_QUEUE_NONE = 3
};
struct ads1x1x_config {
struct i2c_dt_spec bus;
const uint32_t odr_delay[8];
uint8_t resolution;
bool multiplexer;
bool pga;
};
struct ads1x1x_data {
const struct device *dev;
struct adc_context ctx;
k_timeout_t ready_time;
struct k_sem acq_sem;
int16_t *buffer;
int16_t *repeat_buffer;
struct k_thread thread;
bool differential;
K_KERNEL_STACK_MEMBER(stack, CONFIG_ADC_ADS1X1X_ACQUISITION_THREAD_STACK_SIZE);
};
static int ads1x1x_read_reg(const struct device *dev, enum ads1x1x_reg reg_addr, uint16_t *buf)
{
const struct ads1x1x_config *config = dev->config;
uint16_t reg_val;
int ret;
ret = i2c_burst_read_dt(&config->bus, reg_addr, (uint8_t *)&reg_val, sizeof(reg_val));
if (ret != 0) {
LOG_ERR("ADS1X1X[0x%X]: error reading register 0x%X (%d)", config->bus.addr,
reg_addr, ret);
return ret;
}
*buf = sys_be16_to_cpu(reg_val);
return 0;
}
static int ads1x1x_write_reg(const struct device *dev, enum ads1x1x_reg reg_addr, uint16_t reg_val)
{
const struct ads1x1x_config *config = dev->config;
uint8_t buf[3];
int ret;
buf[0] = reg_addr;
sys_put_be16(reg_val, &buf[1]);
ret = i2c_write_dt(&config->bus, buf, sizeof(buf));
if (ret != 0) {
LOG_ERR("ADS1X1X[0x%X]: error writing register 0x%X (%d)", config->bus.addr,
reg_addr, ret);
return ret;
}
return 0;
}
static int ads1x1x_start_conversion(const struct device *dev)
{
/* send start sampling command */
uint16_t config;
int ret;
ret = ads1x1x_read_reg(dev, ADS1X1X_REG_CONFIG, &config);
if (ret != 0) {
return ret;
}
config |= ADS1X1X_CONFIG_OS;
ret = ads1x1x_write_reg(dev, ADS1X1X_REG_CONFIG, config);
return ret;
}
static inline int ads1x1x_acq_time_to_dr(const struct device *dev, uint16_t acq_time)
{
struct ads1x1x_data *data = dev->data;
const struct ads1x1x_config *ads_config = dev->config;
const uint32_t *odr_delay = ads_config->odr_delay;
uint32_t odr_delay_us = 0;
int odr = -EINVAL;
uint16_t acq_value = ADC_ACQ_TIME_VALUE(acq_time);
/* The ADS1x1x uses samples per seconds units with the lowest being 8SPS
* and with acquisition_time only having 14b for time, this will not fit
* within here for microsecond units. Use Tick units and allow the user to
* specify the ODR directly.
*/
if (acq_time != ADC_ACQ_TIME_DEFAULT && ADC_ACQ_TIME_UNIT(acq_time) != ADC_ACQ_TIME_TICKS) {
return -EINVAL;
}
if (acq_time == ADC_ACQ_TIME_DEFAULT) {
odr = ADS1X1X_CONFIG_DR_DEFAULT;
odr_delay_us = odr_delay[ADS1X1X_CONFIG_DR_DEFAULT];
} else {
switch (acq_value) {
case ADS1X1X_CONFIG_DR_8_128:
odr = ADS1X1X_CONFIG_DR_8_128;
odr_delay_us = odr_delay[ADS1X1X_CONFIG_DR_8_128];
break;
case ADS1X1X_CONFIG_DR_16_250:
odr = ADS1X1X_CONFIG_DR_16_250;
odr_delay_us = odr_delay[ADS1X1X_CONFIG_DR_16_250];
break;
case ADS1X1X_CONFIG_DR_32_490:
odr = ADS1X1X_CONFIG_DR_32_490;
odr_delay_us = odr_delay[ADS1X1X_CONFIG_DR_32_490];
break;
case ADS1X1X_CONFIG_DR_64_920:
odr = ADS1X1X_CONFIG_DR_64_920;
odr_delay_us = odr_delay[ADS1X1X_CONFIG_DR_64_920];
break;
case ADS1X1X_CONFIG_DR_128_1600:
odr = ADS1X1X_CONFIG_DR_128_1600;
odr_delay_us = odr_delay[ADS1X1X_CONFIG_DR_128_1600];
break;
case ADS1X1X_CONFIG_DR_250_2400:
odr = ADS1X1X_CONFIG_DR_250_2400;
odr_delay_us = odr_delay[ADS1X1X_CONFIG_DR_250_2400];
break;
case ADS1X1X_CONFIG_DR_475_3300:
odr = ADS1X1X_CONFIG_DR_475_3300;
odr_delay_us = odr_delay[ADS1X1X_CONFIG_DR_475_3300];
break;
case ADS1X1X_CONFIG_DR_860_3300:
odr = ADS1X1X_CONFIG_DR_860_3300;
odr_delay_us = odr_delay[ADS1X1X_CONFIG_DR_860_3300];
break;
default:
break;
}
}
/* As per the datasheet, 25us is needed to wake-up from power down mode
*/
odr_delay_us += 25;
data->ready_time = K_USEC(odr_delay_us);
return odr;
}
static int ads1x1x_wait_data_ready(const struct device *dev)
{
int rc;
struct ads1x1x_data *data = dev->data;
k_sleep(data->ready_time);
uint16_t status = 0;
rc = ads1x1x_read_reg(dev, ADS1X1X_REG_CONFIG, &status);
if (rc != 0) {
return rc;
}
while (!(status & ADS1X1X_CONFIG_OS)) {
k_sleep(K_USEC(100));
rc = ads1x1x_read_reg(dev, ADS1X1X_REG_CONFIG, &status);
if (rc != 0) {
return rc;
}
}
return rc;
}
static int ads1x1x_channel_setup(const struct device *dev,
const struct adc_channel_cfg *channel_cfg)
{
const struct ads1x1x_config *ads_config = dev->config;
struct ads1x1x_data *data = dev->data;
uint16_t config = 0;
int dr = 0;
if (channel_cfg->channel_id != 0) {
LOG_ERR("unsupported channel id '%d'", channel_cfg->channel_id);
return -ENOTSUP;
}
if (channel_cfg->reference != ADC_REF_INTERNAL) {
LOG_ERR("unsupported channel reference type '%d'", channel_cfg->reference);
return -ENOTSUP;
}
if (ads_config->multiplexer) {
/* the device has an input multiplexer */
if (channel_cfg->differential) {
if (channel_cfg->input_positive == 0 && channel_cfg->input_negative == 1) {
config |= ADS1X1X_CONFIG_MUX(ADS1X15_CONFIG_MUX_DIFF_0_1);
} else if (channel_cfg->input_positive == 0 &&
channel_cfg->input_negative == 3) {
config |= ADS1X1X_CONFIG_MUX(ADS1X15_CONFIG_MUX_DIFF_0_3);
} else if (channel_cfg->input_positive == 1 &&
channel_cfg->input_negative == 3) {
config |= ADS1X1X_CONFIG_MUX(ADS1X15_CONFIG_MUX_DIFF_1_3);
} else if (channel_cfg->input_positive == 2 &&
channel_cfg->input_negative == 3) {
config |= ADS1X1X_CONFIG_MUX(ADS1X15_CONFIG_MUX_DIFF_2_3);
} else {
LOG_ERR("unsupported input positive '%d' and input negative '%d'",
channel_cfg->input_positive, channel_cfg->input_negative);
return -ENOTSUP;
}
} else {
if (channel_cfg->input_positive == 0) {
config |= ADS1X1X_CONFIG_MUX(ADS1X15_CONFIG_MUX_SINGLE_0);
} else if (channel_cfg->input_positive == 1) {
config |= ADS1X1X_CONFIG_MUX(ADS1X15_CONFIG_MUX_SINGLE_1);
} else if (channel_cfg->input_positive == 2) {
config |= ADS1X1X_CONFIG_MUX(ADS1X15_CONFIG_MUX_SINGLE_2);
} else if (channel_cfg->input_positive == 3) {
config |= ADS1X1X_CONFIG_MUX(ADS1X15_CONFIG_MUX_SINGLE_3);
} else {
LOG_ERR("unsupported input positive '%d'",
channel_cfg->input_positive);
return -ENOTSUP;
}
}
} else {
/* only differential supported without multiplexer */
if (!((channel_cfg->differential) &&
(channel_cfg->input_positive == 0 && channel_cfg->input_negative == 1))) {
LOG_ERR("unsupported input positive '%d' and input negative '%d'",
channel_cfg->input_positive, channel_cfg->input_negative);
return -ENOTSUP;
}
}
/* store differential mode to determine supported resolution */
data->differential = channel_cfg->differential;
dr = ads1x1x_acq_time_to_dr(dev, channel_cfg->acquisition_time);
if (dr < 0) {
LOG_ERR("unsupported channel acquisition time 0x%02x",
channel_cfg->acquisition_time);
return -ENOTSUP;
}
config |= ADS1X1X_CONFIG_DR(dr);
if (ads_config->pga) {
/* programmable gain amplifier support */
switch (channel_cfg->gain) {
case ADC_GAIN_1_3:
config |= ADS1X1X_CONFIG_PGA(ADS1X1X_CONFIG_PGA_6144);
break;
case ADC_GAIN_1_2:
config |= ADS1X1X_CONFIG_PGA(ADS1X1X_CONFIG_PGA_4096);
break;
case ADC_GAIN_1:
config |= ADS1X1X_CONFIG_PGA(ADS1X1X_CONFIG_PGA_2048);
break;
case ADC_GAIN_2:
config |= ADS1X1X_CONFIG_PGA(ADS1X1X_CONFIG_PGA_1024);
break;
case ADC_GAIN_4:
config |= ADS1X1X_CONFIG_PGA(ADS1X1X_CONFIG_PGA_512);
break;
case ADC_GAIN_8:
config |= ADS1X1X_CONFIG_PGA(ADS1X1X_CONFIG_PGA_256);
break;
default:
LOG_ERR("unsupported channel gain '%d'", channel_cfg->gain);
return -ENOTSUP;
}
} else {
/* no programmable gain amplifier, so only allow ADC_GAIN_1 */
if (channel_cfg->gain != ADC_GAIN_1) {
LOG_ERR("unsupported channel gain '%d'", channel_cfg->gain);
return -ENOTSUP;
}
}
/* Only single shot supported */
config |= ADS1X1X_CONFIG_MODE;
/* disable comparator */
config |= ADS1X1X_CONFIG_COMP_MODE;
return ads1x1x_write_reg(dev, ADS1X1X_REG_CONFIG, config);
}
static int ads1x1x_validate_buffer_size(const struct adc_sequence *sequence)
{
size_t needed = sizeof(int16_t);
if (sequence->options) {
needed *= (1 + sequence->options->extra_samplings);
}
if (sequence->buffer_size < needed) {
return -ENOMEM;
}
return 0;
}
static int ads1x1x_validate_sequence(const struct device *dev, const struct adc_sequence *sequence)
{
const struct ads1x1x_config *config = dev->config;
struct ads1x1x_data *data = dev->data;
uint8_t resolution = data->differential ? config->resolution : config->resolution - 1;
int err;
if (sequence->resolution != resolution) {
LOG_ERR("unsupported resolution %d", sequence->resolution);
return -ENOTSUP;
}
if (sequence->channels != BIT(0)) {
LOG_ERR("only channel 0 supported");
return -ENOTSUP;
}
if (sequence->oversampling) {
LOG_ERR("oversampling not supported");
return -ENOTSUP;
}
err = ads1x1x_validate_buffer_size(sequence);
if (err) {
LOG_ERR("buffer size too small");
return -ENOTSUP;
}
return 0;
}
static void adc_context_update_buffer_pointer(struct adc_context *ctx, bool repeat_sampling)
{
struct ads1x1x_data *data = CONTAINER_OF(ctx, struct ads1x1x_data, ctx);
if (repeat_sampling) {
data->buffer = data->repeat_buffer;
}
}
static void adc_context_start_sampling(struct adc_context *ctx)
{
struct ads1x1x_data *data = CONTAINER_OF(ctx, struct ads1x1x_data, ctx);
int ret;
data->repeat_buffer = data->buffer;
ret = ads1x1x_start_conversion(data->dev);
if (ret != 0) {
/* if we fail to complete the I2C operations to start
* sampling, return an immediate error (likely -EIO) rather
* than handing it off to the acquisition thread.
*/
adc_context_complete(ctx, ret);
return;
}
k_sem_give(&data->acq_sem);
}
static int ads1x1x_adc_start_read(const struct device *dev, const struct adc_sequence *sequence)
{
int rc;
struct ads1x1x_data *data = dev->data;
rc = ads1x1x_validate_sequence(dev, sequence);
if (rc != 0) {
return rc;
}
data->buffer = sequence->buffer;
adc_context_start_read(&data->ctx, sequence);
return adc_context_wait_for_completion(&data->ctx);
}
static int ads1x1x_adc_read_async(const struct device *dev, const struct adc_sequence *sequence,
struct k_poll_signal *async)
{
int rc;
struct ads1x1x_data *data = dev->data;
adc_context_lock(&data->ctx, async ? true : false, async);
rc = ads1x1x_adc_start_read(dev, sequence);
adc_context_release(&data->ctx, rc);
return rc;
}
static int ads1x1x_adc_perform_read(const struct device *dev)
{
int rc;
struct ads1x1x_data *data = dev->data;
const struct ads1x1x_config *config = dev->config;
int16_t buf;
rc = ads1x1x_read_reg(dev, ADS1X1X_REG_CONV, &buf);
if (rc != 0) {
adc_context_complete(&data->ctx, rc);
return rc;
}
/* The ads101x stores it's 12b data in the upper part
* while the ads111x uses all 16b in the register, so
* shift down. Data is also signed, so perform
* division rather than shifting
*/
*data->buffer++ = buf / (1 << (16 - config->resolution));
adc_context_on_sampling_done(&data->ctx, dev);
return rc;
}
static int ads1x1x_read(const struct device *dev, const struct adc_sequence *sequence)
{
return ads1x1x_adc_read_async(dev, sequence, NULL);
}
static void ads1x1x_acquisition_thread(void *p1, void *p2, void *p3)
{
ARG_UNUSED(p2);
ARG_UNUSED(p3);
const struct device *dev = p1;
struct ads1x1x_data *data = dev->data;
int rc;
while (true) {
k_sem_take(&data->acq_sem, K_FOREVER);
rc = ads1x1x_wait_data_ready(dev);
if (rc != 0) {
LOG_ERR("failed to get ready status (err %d)", rc);
adc_context_complete(&data->ctx, rc);
continue;
}
ads1x1x_adc_perform_read(dev);
}
}
static int ads1x1x_init(const struct device *dev)
{
const struct ads1x1x_config *config = dev->config;
struct ads1x1x_data *data = dev->data;
data->dev = dev;
k_sem_init(&data->acq_sem, 0, 1);
if (!device_is_ready(config->bus.bus)) {
LOG_ERR("I2C bus %s not ready", config->bus.bus->name);
return -ENODEV;
}
k_tid_t tid =
k_thread_create(&data->thread, data->stack, K_THREAD_STACK_SIZEOF(data->stack),
ads1x1x_acquisition_thread, (void *)dev, NULL,
NULL, CONFIG_ADC_ADS1X1X_ACQUISITION_THREAD_PRIO, 0, K_NO_WAIT);
k_thread_name_set(tid, "adc_ads1x1x");
adc_context_unlock_unconditionally(&data->ctx);
return 0;
}
static const struct adc_driver_api ads1x1x_api = {
.channel_setup = ads1x1x_channel_setup,
.read = ads1x1x_read,
.ref_internal = 2048,
#ifdef CONFIG_ADC_ASYNC
.read_async = ads1x1x_adc_read_async,
#endif
};
#define DT_INST_ADS1X1X(inst, t) DT_INST(inst, ti_ads##t)
#define ADS1X1X_INIT(t, n, odr_delay_us, res, mux, pgab) \
static const struct ads1x1x_config ads##t##_config_##n = { \
.bus = I2C_DT_SPEC_GET(DT_INST_ADS1X1X(n, t)), \
.odr_delay = odr_delay_us, \
.resolution = res, \
.multiplexer = mux, \
.pga = pgab, \
}; \
static struct ads1x1x_data ads##t##_data_##n = { \
ADC_CONTEXT_INIT_LOCK(ads##t##_data_##n, ctx), \
ADC_CONTEXT_INIT_TIMER(ads##t##_data_##n, ctx), \
ADC_CONTEXT_INIT_SYNC(ads##t##_data_##n, ctx), \
}; \
DEVICE_DT_DEFINE(DT_INST_ADS1X1X(n, t), ads1x1x_init, NULL, &ads##t##_data_##n, \
&ads##t##_config_##n, POST_KERNEL, CONFIG_ADC_ADS1X1X_INIT_PRIORITY, \
&ads1x1x_api);
/* The ADS111X provides 16 bits of data in binary two's complement format
* A positive full-scale (+FS) input produces an output code of 7FFFh and a
* negative full-scale (FS) input produces an output code of 8000h. Single
* ended signal measurements only only use the positive code range from
* 0000h to 7FFFh
*/
#define ADS111X_RESOLUTION 16
/*
* Approximated ADS111x acquisition times in microseconds. These are
* used for the initial delay when polling for data ready.
* {8 SPS, 16 SPS, 32 SPS, 64 SPS, 128 SPS (default), 250 SPS, 475 SPS, 860 SPS}
*/
#define ADS111X_ODR_DELAY_US \
{ \
125000, 62500, 31250, 15625, 7813, 4000, 2105, 1163 \
}
/*
* ADS1115: 16 bit, multiplexer, programmable gain amplifier
*/
#define ADS1115_INIT(n) ADS1X1X_INIT(1115, n, ADS111X_ODR_DELAY_US, ADS111X_RESOLUTION, true, true)
#undef DT_DRV_COMPAT
#define DT_DRV_COMPAT ti_ads1115
DT_INST_FOREACH_STATUS_OKAY(ADS1115_INIT)
/*
* ADS1114: 16 bit, no multiplexer, programmable gain amplifier
*/
#define ADS1114_INIT(n) ADS1X1X_INIT(1114, n, ADS111X_ODR_DELAY_US, ADS111X_RESOLUTION, false, true)
#undef DT_DRV_COMPAT
#define DT_DRV_COMPAT ti_ads1114
DT_INST_FOREACH_STATUS_OKAY(ADS1114_INIT)
/*
* ADS1113: 16 bit, no multiplexer, no programmable gain amplifier
*/
#define ADS1113_INIT(n) \
ADS1X1X_INIT(1113, n, ADS111X_ODR_DELAY_US, ADS111X_RESOLUTION, false, false)
#undef DT_DRV_COMPAT
#define DT_DRV_COMPAT ti_ads1113
DT_INST_FOREACH_STATUS_OKAY(ADS1113_INIT)
/* The ADS101X provides 12 bits of data in binary two's complement format
* A positive full-scale (+FS) input produces an output code of 7FFh and a
* negative full-scale (FS) input produces an output code of 800h. Single
* ended signal measurements only only use the positive code range from
* 000h to 7FFh
*/
#define ADS101X_RESOLUTION 12
/*
* Approximated ADS101x acquisition times in microseconds. These are
* used for the initial delay when polling for data ready.
* {128 SPS, 250 SPS, 490 SPS, 920 SPS, 1600 SPS (default), 2400 SPS, 3300 SPS, 3300 SPS}
*/
#define ADS101X_ODR_DELAY_US \
{ \
7813, 4000, 2041, 1087, 625, 417, 303, 303 \
}
/*
* ADS1015: 12 bit, multiplexer, programmable gain amplifier
*/
#define ADS1015_INIT(n) ADS1X1X_INIT(1015, n, ADS101X_ODR_DELAY_US, ADS101X_RESOLUTION, true, true)
#undef DT_DRV_COMPAT
#define DT_DRV_COMPAT ti_ads1015
DT_INST_FOREACH_STATUS_OKAY(ADS1015_INIT)
/*
* ADS1014: 12 bit, no multiplexer, programmable gain amplifier
*/
#define ADS1014_INIT(n) ADS1X1X_INIT(1014, n, ADS101X_ODR_DELAY_US, ADS101X_RESOLUTION, false, true)
#undef DT_DRV_COMPAT
#define DT_DRV_COMPAT ti_ads1014
DT_INST_FOREACH_STATUS_OKAY(ADS1014_INIT)
/*
* ADS1013: 12 bit, no multiplexer, no programmable gain amplifier
*/
#define ADS1013_INIT(n) \
ADS1X1X_INIT(1013, n, ADS101X_ODR_DELAY_US, ADS101X_RESOLUTION, false, false)
#undef DT_DRV_COMPAT
#define DT_DRV_COMPAT ti_ads1013
DT_INST_FOREACH_STATUS_OKAY(ADS1013_INIT)