/* * Copyright (c) 2023 Renesas Electronics Corporation * * SPDX-License-Identifier: Apache-2.0 */ #define DT_DRV_COMPAT renesas_smartbond_sdadc #define ADC_CONTEXT_USES_KERNEL_TIMER #include #include #include "adc_context.h" #include #define LOG_LEVEL CONFIG_ADC_LOG_LEVEL #include #include #include #include #include #include #include LOG_MODULE_REGISTER(adc_smartbond_sdadc); struct sdadc_smartbond_cfg { const struct pinctrl_dev_config *pcfg; /** Value for SDADC_CLK_FREQ */ uint8_t sdadc_clk_freq; }; struct sdadc_smartbond_data { struct adc_context ctx; /* Buffer to store channel data */ uint16_t *buffer; /* Copy of channel mask from sequence */ uint32_t channel_read_mask; /* Number of bits in sequence channels */ uint8_t sequence_channel_count; /* Index in buffer to store current value to */ uint8_t result_index; }; #define SMARTBOND_SDADC_CHANNEL_COUNT 8 struct sdadc_smartbond_channel_cfg { uint32_t sd_adc_ctrl_reg; }; static struct sdadc_smartbond_channel_cfg m_sdchannels[SMARTBOND_SDADC_CHANNEL_COUNT]; /* Implementation of the ADC driver API function: adc_channel_setup. */ static int sdadc_smartbond_channel_setup(const struct device *dev, const struct adc_channel_cfg *channel_cfg) { uint8_t channel_id = channel_cfg->channel_id; struct sdadc_smartbond_channel_cfg *config = &m_sdchannels[channel_id]; if (channel_id >= SMARTBOND_SDADC_CHANNEL_COUNT) { return -EINVAL; } if (channel_cfg->acquisition_time != ADC_ACQ_TIME_DEFAULT) { LOG_ERR("Selected ADC acquisition time is not valid"); return -EINVAL; } if (channel_cfg->input_positive > SMARTBOND_SDADC_VBAT) { LOG_ERR("Channels out of range"); return -EINVAL; } if (channel_cfg->differential) { if (channel_cfg->input_negative >= SMARTBOND_SDADC_VBAT) { LOG_ERR("Differential negative channels out of range"); return -EINVAL; } } config->sd_adc_ctrl_reg = 0; if ((channel_cfg->input_positive == SMARTBOND_SDADC_VBAT && channel_cfg->gain != ADC_GAIN_1_4) || (channel_cfg->input_positive != SMARTBOND_SDADC_VBAT && channel_cfg->gain != ADC_GAIN_1)) { LOG_ERR("ADC gain should be 1/4 for VBAT and 1 for all other channels"); return -EINVAL; } switch (channel_cfg->reference) { case ADC_REF_INTERNAL: break; default: LOG_ERR("Selected ADC reference is not valid"); return -EINVAL; } config->sd_adc_ctrl_reg = channel_cfg->input_positive << SDADC_SDADC_CTRL_REG_SDADC_INP_SEL_Pos; if (channel_cfg->differential) { config->sd_adc_ctrl_reg |= channel_cfg->input_negative << SDADC_SDADC_CTRL_REG_SDADC_INN_SEL_Pos; } else { config->sd_adc_ctrl_reg |= SDADC_SDADC_CTRL_REG_SDADC_SE_Msk; } return 0; } #define PER_CHANNEL_ADC_CONFIG_MASK (SDADC_SDADC_CTRL_REG_SDADC_INP_SEL_Msk | \ SDADC_SDADC_CTRL_REG_SDADC_INN_SEL_Msk | \ SDADC_SDADC_CTRL_REG_SDADC_SE_Msk \ ) static inline void sdadc_smartbond_pm_policy_state_lock_get(const struct device *dev, struct sdadc_smartbond_data *data) { #if defined(CONFIG_PM_DEVICE) pm_device_runtime_get(dev); /* * Prevent the SoC from entering the normal sleep state. */ pm_policy_state_lock_get(PM_STATE_STANDBY, PM_ALL_SUBSTATES); #endif } static inline void sdadc_smartbond_pm_policy_state_lock_put(const struct device *dev, struct sdadc_smartbond_data *data) { #if defined(CONFIG_PM_DEVICE) /* * Allow the SoC to enter the normal sleep state once sdadc is done. */ pm_policy_state_lock_put(PM_STATE_STANDBY, PM_ALL_SUBSTATES); pm_device_runtime_put(dev); #endif } static int pop_count(uint32_t n) { return __builtin_popcount(n); } static void adc_context_start_sampling(struct adc_context *ctx) { uint32_t val; struct sdadc_smartbond_data *data = CONTAINER_OF(ctx, struct sdadc_smartbond_data, ctx); /* Extract lower channel from sequence mask */ int current_channel = u32_count_trailing_zeros(data->channel_read_mask); /* Wait until the SDADC LDO stabilizes */ while (!(SDADC->SDADC_CTRL_REG & SDADC_SDADC_CTRL_REG_SDADC_LDO_OK_Msk)) { __NOP(); } if (ctx->sequence.calibrate) { /* TODO: Add calibration code */ } else { val = SDADC->SDADC_CTRL_REG & ~PER_CHANNEL_ADC_CONFIG_MASK; val |= m_sdchannels[current_channel].sd_adc_ctrl_reg; val |= SDADC_SDADC_CTRL_REG_SDADC_START_Msk | SDADC_SDADC_CTRL_REG_SDADC_MINT_Msk; val |= (ctx->sequence.oversampling - 7) << SDADC_SDADC_CTRL_REG_SDADC_OSR_Pos; SDADC->SDADC_CTRL_REG = val; } } static void adc_context_update_buffer_pointer(struct adc_context *ctx, bool repeat) { struct sdadc_smartbond_data *data = CONTAINER_OF(ctx, struct sdadc_smartbond_data, ctx); if (!repeat) { data->buffer += data->sequence_channel_count; } } static int check_buffer_size(const struct adc_sequence *sequence, uint8_t active_channels) { size_t needed_buffer_size; needed_buffer_size = active_channels * sizeof(uint16_t); if (sequence->options) { needed_buffer_size *= (1 + sequence->options->extra_samplings); } if (sequence->buffer_size < needed_buffer_size) { LOG_ERR("Provided buffer is too small (%u/%u)", sequence->buffer_size, needed_buffer_size); return -ENOMEM; } return 0; } static int start_read(const struct device *dev, const struct adc_sequence *sequence) { int error; struct sdadc_smartbond_data *data = dev->data; if (sequence->oversampling < 7U || sequence->oversampling > 10) { LOG_ERR("Invalid oversampling"); return -EINVAL; } if ((sequence->channels == 0) || ((sequence->channels & ~BIT_MASK(SMARTBOND_SDADC_CHANNEL_COUNT)) != 0)) { LOG_ERR("Channel scanning is not supported"); return -EINVAL; } if (sequence->resolution < 8 || sequence->resolution > 15) { LOG_ERR("ADC resolution value %d is not valid", sequence->resolution); return -EINVAL; } error = check_buffer_size(sequence, 1); if (error) { return error; } data->buffer = sequence->buffer; data->channel_read_mask = sequence->channels; data->sequence_channel_count = pop_count(sequence->channels); data->result_index = 0; adc_context_start_read(&data->ctx, sequence); error = adc_context_wait_for_completion(&data->ctx); return error; } static void sdadc_smartbond_isr(const struct device *dev) { struct sdadc_smartbond_data *data = dev->data; int current_channel = u32_count_trailing_zeros(data->channel_read_mask); SDADC->SDADC_CLEAR_INT_REG = 0; /* Store current channel value, result is left justified, move bits right */ data->buffer[data->result_index++] = ((uint16_t)SDADC->SDADC_RESULT_REG) >> (16 - data->ctx.sequence.resolution); /* Exclude channel from mask for further reading */ data->channel_read_mask ^= 1 << current_channel; if (data->channel_read_mask == 0) { sdadc_smartbond_pm_policy_state_lock_put(dev, data); adc_context_on_sampling_done(&data->ctx, dev); } else { adc_context_start_sampling(&data->ctx); } LOG_DBG("%s ISR triggered.", dev->name); } /* Implementation of the ADC driver API function: adc_read. */ static int sdadc_smartbond_read(const struct device *dev, const struct adc_sequence *sequence) { int error; struct sdadc_smartbond_data *data = dev->data; adc_context_lock(&data->ctx, false, NULL); sdadc_smartbond_pm_policy_state_lock_get(dev, data); error = start_read(dev, sequence); adc_context_release(&data->ctx, error); return error; } #if defined(CONFIG_ADC_ASYNC) /* Implementation of the ADC driver API function: adc_read_sync. */ static int sdadc_smartbond_read_async(const struct device *dev, const struct adc_sequence *sequence, struct k_poll_signal *async) { struct sdadc_smartbond_data *data = dev->data; int error; adc_context_lock(&data->ctx, true, async); sdadc_smartbond_pm_policy_state_lock_get(dev, data); error = start_read(dev, sequence); adc_context_release(&data->ctx, error); return error; } #endif /* CONFIG_ADC_ASYNC */ static int sdadc_smartbond_resume(const struct device *dev) { int ret; const struct sdadc_smartbond_cfg *config = dev->config; da1469x_pd_acquire(MCU_PD_DOMAIN_COM); SDADC->SDADC_TEST_REG = (SDADC->SDADC_TEST_REG & ~SDADC_SDADC_TEST_REG_SDADC_CLK_FREQ_Msk) | (config->sdadc_clk_freq) << SDADC_SDADC_TEST_REG_SDADC_CLK_FREQ_Pos; SDADC->SDADC_CTRL_REG = SDADC_SDADC_CTRL_REG_SDADC_EN_Msk; /* * Configure dt provided device signals when available. * pinctrl is optional so ENOENT is not setup failure. */ ret = pinctrl_apply_state(config->pcfg, PINCTRL_STATE_DEFAULT); if (ret < 0 && ret != -ENOENT) { SDADC->SDADC_CTRL_REG = 0; /* Release the comms domain */ da1469x_pd_release(MCU_PD_DOMAIN_COM); LOG_ERR("ADC pinctrl setup failed (%d)", ret); return ret; } return 0; } #ifdef CONFIG_PM_DEVICE static int sdadc_smartbond_suspend(const struct device *dev) { int ret; const struct sdadc_smartbond_cfg *config = dev->config; /* Disable the sdadc LDO */ SDADC->SDADC_CTRL_REG = 0; /* Release the comms domain */ da1469x_pd_release(MCU_PD_DOMAIN_COM); /* * Configure dt provided device signals for sleep. * pinctrl is optional so ENOENT is not setup failure. */ ret = pinctrl_apply_state(config->pcfg, PINCTRL_STATE_SLEEP); if (ret < 0 && ret != -ENOENT) { LOG_WRN("Failed to configure the sdadc pins to inactive state"); return ret; } return 0; } static int sdadc_smartbond_pm_action(const struct device *dev, enum pm_device_action action) { int ret; switch (action) { case PM_DEVICE_ACTION_RESUME: ret = sdadc_smartbond_resume(dev); break; case PM_DEVICE_ACTION_SUSPEND: ret = sdadc_smartbond_suspend(dev); break; default: return -ENOTSUP; } return ret; } #endif /* CONFIG_PM_DEVICE */ static int sdadc_smartbond_init(const struct device *dev) { int ret; struct sdadc_smartbond_data *data = dev->data; #ifdef CONFIG_PM_DEVICE_RUNTIME /* Make sure device state is marked as suspended */ pm_device_init_suspended(dev); ret = pm_device_runtime_enable(dev); #else ret = sdadc_smartbond_resume(dev); #endif IRQ_CONNECT(DT_INST_IRQN(0), DT_INST_IRQ(0, priority), sdadc_smartbond_isr, DEVICE_DT_INST_GET(0), 0); NVIC_ClearPendingIRQ(DT_INST_IRQN(0)); NVIC_EnableIRQ(DT_INST_IRQN(0)); adc_context_unlock_unconditionally(&data->ctx); return ret; } static DEVICE_API(adc, sdadc_smartbond_driver_api) = { .channel_setup = sdadc_smartbond_channel_setup, .read = sdadc_smartbond_read, #ifdef CONFIG_ADC_ASYNC .read_async = sdadc_smartbond_read_async, #endif .ref_internal = 1200, }; /* * There is only one instance on supported SoCs, so inst is guaranteed * to be 0 if any instance is okay. (We use adc_0 above, so the driver * is relying on the numeric instance value in a way that happens to * be safe.) * * Just in case that assumption becomes invalid in the future, we use * a BUILD_ASSERT(). */ #define SDADC_INIT(inst) \ BUILD_ASSERT((inst) == 0, \ "multiple instances not supported"); \ PINCTRL_DT_INST_DEFINE(inst); \ static const struct sdadc_smartbond_cfg sdadc_smartbond_cfg_##inst = { \ .pcfg = PINCTRL_DT_INST_DEV_CONFIG_GET(inst), \ .sdadc_clk_freq = DT_INST_PROP(inst, clock_freq), \ }; \ static struct sdadc_smartbond_data sdadc_smartbond_data_##inst = { \ ADC_CONTEXT_INIT_TIMER(sdadc_smartbond_data_##inst, ctx), \ ADC_CONTEXT_INIT_LOCK(sdadc_smartbond_data_##inst, ctx), \ ADC_CONTEXT_INIT_SYNC(sdadc_smartbond_data_##inst, ctx), \ }; \ PM_DEVICE_DT_INST_DEFINE(inst, sdadc_smartbond_pm_action); \ DEVICE_DT_INST_DEFINE(inst, \ sdadc_smartbond_init, \ PM_DEVICE_DT_INST_GET(inst), \ &sdadc_smartbond_data_##inst, \ &sdadc_smartbond_cfg_##inst, \ POST_KERNEL, \ CONFIG_ADC_INIT_PRIORITY, \ &sdadc_smartbond_driver_api); DT_INST_FOREACH_STATUS_OKAY(SDADC_INIT)