1 /*
2 * Copyright (c) 2023 Antmicro <www.antmicro.com>
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT silabs_gecko_adc
8
9 #include <zephyr/drivers/adc.h>
10
11 #include <em_adc.h>
12 #include <em_cmu.h>
13
14 #define ADC_CONTEXT_USES_KERNEL_TIMER
15 #include "adc_context.h"
16
17 #include <zephyr/logging/log.h>
18 LOG_MODULE_REGISTER(adc_gecko, CONFIG_ADC_LOG_LEVEL);
19
20 /* Number of channels available. */
21 #define GECKO_CHANNEL_COUNT 16
22
23 struct adc_gecko_channel_config {
24 bool initialized;
25 ADC_Ref_TypeDef reference;
26 ADC_PosSel_TypeDef input_select;
27 };
28
29 struct adc_gecko_data {
30 const struct device *dev;
31 struct adc_context ctx;
32 uint16_t *buffer;
33 uint16_t *repeat_buffer;
34 uint32_t channels;
35 uint8_t channel_id;
36 ADC_Res_TypeDef resolution;
37 struct adc_gecko_channel_config channel_config[GECKO_CHANNEL_COUNT];
38 };
39
40 struct adc_gecko_config {
41 ADC_TypeDef *base;
42 void (*irq_cfg_func)(void);
43 uint32_t frequency;
44 };
45
adc_gecko_set_config(const struct device * dev)46 static void adc_gecko_set_config(const struct device *dev)
47 {
48 struct adc_gecko_data *data = dev->data;
49 struct adc_gecko_channel_config *channel_config = NULL;
50 const struct adc_gecko_config *config = dev->config;
51 ADC_TypeDef *adc_base = (ADC_TypeDef *)config->base;
52
53 ADC_Init_TypeDef init = ADC_INIT_DEFAULT;
54 ADC_InitSingle_TypeDef initSingle = ADC_INITSINGLE_DEFAULT;
55
56 channel_config = &data->channel_config[data->channel_id];
57
58 init.prescale = ADC_PrescaleCalc(config->frequency, 0);
59 init.timebase = ADC_TimebaseCalc(0);
60
61 initSingle.diff = false;
62 initSingle.reference = channel_config->reference;
63 initSingle.resolution = data->resolution;
64 initSingle.acqTime = adcAcqTime4;
65
66 initSingle.posSel = channel_config->input_select;
67
68 ADC_Init(adc_base, &init);
69 ADC_InitSingle(adc_base, &initSingle);
70 }
71
adc_gecko_check_buffer_size(const struct adc_sequence * sequence,uint8_t active_channels)72 static int adc_gecko_check_buffer_size(const struct adc_sequence *sequence,
73 uint8_t active_channels)
74 {
75 size_t needed_buffer_size;
76
77 needed_buffer_size = active_channels * sizeof(uint16_t);
78
79 if (sequence->options) {
80 needed_buffer_size *= (1 + sequence->options->extra_samplings);
81 }
82
83 if (sequence->buffer_size < needed_buffer_size) {
84 LOG_DBG("Provided buffer is too small (%u/%u)",
85 sequence->buffer_size, needed_buffer_size);
86 return -ENOMEM;
87 }
88
89 return 0;
90 }
91
start_read(const struct device * dev,const struct adc_sequence * sequence)92 static int start_read(const struct device *dev, const struct adc_sequence *sequence)
93 {
94
95 struct adc_gecko_data *data = dev->data;
96 uint32_t channels;
97 uint8_t channel_count;
98 uint8_t index;
99 int res;
100
101 /* Check if at least 1 channel is requested */
102 if (sequence->channels == 0) {
103 LOG_DBG("No channel requested");
104 return -EINVAL;
105 }
106
107 if (sequence->oversampling) {
108 LOG_ERR("Oversampling is not supported");
109 return -ENOTSUP;
110 }
111
112 /* Verify all requested channels are initialized and store resolution */
113 channels = sequence->channels;
114 channel_count = 0;
115 while (channels) {
116 /* Iterate through all channels and check if they are initialized */
117 index = find_lsb_set(channels) - 1;
118 if (index >= GECKO_CHANNEL_COUNT) {
119 LOG_DBG("Requested channel index not available: %d", index);
120 return -EINVAL;
121 }
122
123 if (!data->channel_config[index].initialized) {
124 LOG_DBG("Channel not initialized");
125 return -EINVAL;
126 }
127 channel_count++;
128 channels &= ~BIT(index);
129 }
130
131 res = adc_gecko_check_buffer_size(sequence, channel_count);
132 if (res < 0) {
133 return res;
134 }
135
136 data->buffer = sequence->buffer;
137
138 adc_context_start_read(&data->ctx, sequence);
139
140 res = adc_context_wait_for_completion(&data->ctx);
141
142 return res;
143 }
144
adc_gecko_start_channel(const struct device * dev)145 static void adc_gecko_start_channel(const struct device *dev)
146 {
147 const struct adc_gecko_config *config = dev->config;
148 struct adc_gecko_data *data = dev->data;
149 ADC_TypeDef *adc_base = (ADC_TypeDef *)config->base;
150
151 data->channel_id = find_lsb_set(data->channels) - 1;
152 adc_gecko_set_config(data->dev);
153
154 ADC_IntEnable(adc_base, ADC_IEN_SINGLE);
155 ADC_Start(adc_base, adcStartSingle);
156 }
157
adc_context_start_sampling(struct adc_context * ctx)158 static void adc_context_start_sampling(struct adc_context *ctx)
159 {
160 struct adc_gecko_data *data = CONTAINER_OF(ctx, struct adc_gecko_data, ctx);
161
162 data->channels = ctx->sequence.channels;
163 adc_gecko_start_channel(data->dev);
164 }
165
adc_context_update_buffer_pointer(struct adc_context * ctx,bool repeat_sampling)166 static void adc_context_update_buffer_pointer(struct adc_context *ctx, bool repeat_sampling)
167 {
168 struct adc_gecko_data *data = CONTAINER_OF(ctx, struct adc_gecko_data, ctx);
169
170 if (repeat_sampling) {
171 data->buffer = data->repeat_buffer;
172 }
173 }
174
adc_gecko_isr(void * arg)175 static void adc_gecko_isr(void *arg)
176 {
177 const struct device *dev = (const struct device *)arg;
178 const struct adc_gecko_config *config = dev->config;
179 struct adc_gecko_data *data = dev->data;
180 ADC_TypeDef *adc_base = config->base;
181
182 uint32_t sample = 0;
183 uint32_t flags, err;
184
185 flags = ADC_IntGet(adc_base);
186
187 __ASSERT(flags & ADC_IF_SINGLE, "unexpected ADC IRQ (flags=0x%08x)!", flags);
188
189 err = flags & (ADC_IF_EM23ERR | ADC_IF_PROGERR | ADC_IF_VREFOV | ADC_IF_SINGLEOF);
190
191 if (!err) {
192 sample = ADC_DataSingleGet(adc_base);
193 *data->buffer++ = (uint16_t)sample;
194 data->channels &= ~BIT(data->channel_id);
195
196 if (data->channels) {
197 adc_gecko_start_channel(dev);
198 } else {
199 adc_context_on_sampling_done(&data->ctx, dev);
200 }
201 } else {
202 LOG_ERR("ADC conversion error, flags=%08x", err);
203 adc_context_complete(&data->ctx, -EIO);
204 }
205 ADC_IntClear(adc_base, ADC_IF_SINGLE | err);
206 }
207
adc_gecko_read(const struct device * dev,const struct adc_sequence * sequence)208 static int adc_gecko_read(const struct device *dev,
209 const struct adc_sequence *sequence)
210 {
211 struct adc_gecko_data *data = dev->data;
212 int error;
213
214 adc_context_lock(&data->ctx, false, NULL);
215 error = start_read(dev, sequence);
216 adc_context_release(&data->ctx, error);
217
218 return error;
219 }
220
adc_gecko_channel_setup(const struct device * dev,const struct adc_channel_cfg * channel_cfg)221 static int adc_gecko_channel_setup(const struct device *dev,
222 const struct adc_channel_cfg *channel_cfg)
223 {
224 struct adc_gecko_data *data = dev->data;
225 struct adc_gecko_channel_config *channel_config = NULL;
226
227 if (channel_cfg->channel_id < GECKO_CHANNEL_COUNT) {
228 channel_config = &data->channel_config[channel_cfg->channel_id];
229 } else {
230 LOG_DBG("Requested channel index not available: %d", channel_cfg->channel_id);
231 return -EINVAL;
232 }
233
234 channel_config->initialized = false;
235
236 channel_config->input_select = channel_cfg->input_positive;
237
238 switch (channel_cfg->gain) {
239 case ADC_GAIN_1:
240 break;
241 default:
242 LOG_ERR("unsupported channel gain '%d'", channel_cfg->gain);
243 return -ENOTSUP;
244 }
245
246 switch (channel_cfg->reference) {
247 case ADC_REF_VDD_1:
248 channel_config->reference = adcRef5V;
249 break;
250 case ADC_REF_VDD_1_2:
251 channel_config->reference = adcRef2V5;
252 break;
253 case ADC_REF_VDD_1_4:
254 channel_config->reference = adcRef1V25;
255 break;
256 default:
257 LOG_ERR("unsupported channel reference type '%d'", channel_cfg->reference);
258 return -ENOTSUP;
259 }
260
261 channel_config->initialized = true;
262 return 0;
263 }
264
adc_gecko_init(const struct device * dev)265 static int adc_gecko_init(const struct device *dev)
266 {
267 const struct adc_gecko_config *config = dev->config;
268 struct adc_gecko_data *data = dev->data;
269
270 CMU_ClockEnable(cmuClock_HFPER, true);
271 CMU_ClockEnable(cmuClock_ADC0, true);
272
273 data->dev = dev;
274 data->resolution = adcRes12Bit;
275
276 config->irq_cfg_func();
277
278 adc_context_unlock_unconditionally(&data->ctx);
279
280 return 0;
281 }
282
283 static DEVICE_API(adc, api_gecko_adc_driver_api) = {
284 .channel_setup = adc_gecko_channel_setup,
285 .read = adc_gecko_read,
286 };
287
288 #define GECKO_ADC_INIT(n) \
289 \
290 static void adc_gecko_config_func_##n(void); \
291 \
292 const static struct adc_gecko_config adc_gecko_config_##n = { \
293 .base = (ADC_TypeDef *)DT_INST_REG_ADDR(n), \
294 .irq_cfg_func = adc_gecko_config_func_##n, \
295 .frequency = DT_INST_PROP(n, frequency), \
296 }; \
297 static struct adc_gecko_data adc_gecko_data_##n = { \
298 ADC_CONTEXT_INIT_TIMER(adc_gecko_data_##n, ctx), \
299 ADC_CONTEXT_INIT_LOCK(adc_gecko_data_##n, ctx), \
300 ADC_CONTEXT_INIT_SYNC(adc_gecko_data_##n, ctx), \
301 }; \
302 static void adc_gecko_config_func_##n(void) \
303 { \
304 IRQ_CONNECT(DT_INST_IRQN(n), \
305 DT_INST_IRQ(n, priority), \
306 adc_gecko_isr, DEVICE_DT_INST_GET(n), 0); \
307 irq_enable(DT_INST_IRQN(n)); \
308 }; \
309 DEVICE_DT_INST_DEFINE(n, \
310 &adc_gecko_init, NULL, \
311 &adc_gecko_data_##n, &adc_gecko_config_##n,\
312 POST_KERNEL, CONFIG_ADC_INIT_PRIORITY, \
313 &api_gecko_adc_driver_api);
314
315 DT_INST_FOREACH_STATUS_OKAY(GECKO_ADC_INIT)
316