1 /*
2  * Copyright (c) 2023 Cypress Semiconductor Corporation (an Infineon company) or
3  * an affiliate of Cypress Semiconductor Corporation
4  *
5  * SPDX-License-Identifier: Apache-2.0
6  */
7 
8 /**
9  * @brief ADC driver for Infineon CAT1 MCU family.
10  */
11 
12 #define DT_DRV_COMPAT infineon_cat1_adc
13 
14 #include <zephyr/drivers/adc.h>
15 #include <cyhal_adc.h>
16 #include <cyhal_utils_impl.h>
17 
18 #define ADC_CONTEXT_USES_KERNEL_TIMER
19 #include "adc_context.h"
20 
21 #include <zephyr/logging/log.h>
22 LOG_MODULE_REGISTER(ifx_cat1_adc, CONFIG_ADC_LOG_LEVEL);
23 
24 #if defined(PASS_SARMUX_PADS0_PORT)
25 	#define _ADCSAR_PORT PASS_SARMUX_PADS0_PORT
26 #else
27 	#error The selected device does not supported ADC
28 #endif
29 
30 #define ADC_CAT1_EVENTS_MASK (CYHAL_ADC_EOS | CYHAL_ADC_ASYNC_READ_COMPLETE)
31 
32 #define ADC_CAT1_DEFAULT_ACQUISITION_NS (1000u)
33 #define ADC_CAT1_RESOLUTION             (12u)
34 #define ADC_CAT1_REF_INTERNAL_MV        (1200u)
35 
36 struct ifx_cat1_adc_data {
37 	struct adc_context ctx;
38 	const struct device *dev;
39 	cyhal_adc_t adc_obj;
40 	cyhal_adc_channel_t adc_chan_obj[CY_SAR_SEQ_NUM_CHANNELS];
41 	uint16_t *buffer;
42 	uint16_t *repeat_buffer;
43 	uint32_t channels;
44 	uint32_t channels_mask;
45 };
46 
47 struct ifx_cat1_adc_config {
48 	uint8_t irq_priority;
49 };
50 
_cyhal_adc_event_callback(void * callback_arg,cyhal_adc_event_t event)51 static void _cyhal_adc_event_callback(void *callback_arg, cyhal_adc_event_t event)
52 {
53 	const struct device *dev = (const struct device *) callback_arg;
54 	struct ifx_cat1_adc_data *data = dev->data;
55 	uint32_t channels = data->channels;
56 	int32_t result;
57 	uint32_t channel_id;
58 
59 	while (channels != 0) {
60 		channel_id = find_lsb_set(channels) - 1;
61 		channels &= ~BIT(channel_id);
62 
63 		result = Cy_SAR_GetResult32(data->adc_chan_obj[channel_id].adc->base,
64 					    data->adc_chan_obj[channel_id].channel_idx);
65 		/* Legacy API for BWC. Convert from signed to unsigned by adding 0x800 to
66 		 * convert the lowest signed 12-bit number to 0x0.
67 		 */
68 		*data->buffer = (uint16_t)(result + 0x800);
69 		data->buffer++;
70 	}
71 
72 	adc_context_on_sampling_done(&data->ctx, dev);
73 
74 	LOG_DBG("%s ISR triggered.", dev->name);
75 }
76 
adc_context_start_sampling(struct adc_context * ctx)77 static void adc_context_start_sampling(struct adc_context *ctx)
78 {
79 	struct ifx_cat1_adc_data *data = CONTAINER_OF(ctx, struct ifx_cat1_adc_data, ctx);
80 
81 	data->repeat_buffer = data->buffer;
82 
83 	Cy_SAR_StartConvert(data->adc_obj.base, CY_SAR_START_CONVERT_SINGLE_SHOT);
84 }
85 
adc_context_update_buffer_pointer(struct adc_context * ctx,bool repeat_sampling)86 static void adc_context_update_buffer_pointer(struct adc_context *ctx,
87 					      bool repeat_sampling)
88 {
89 	struct ifx_cat1_adc_data *data = CONTAINER_OF(ctx, struct ifx_cat1_adc_data, ctx);
90 
91 	if (repeat_sampling) {
92 		data->buffer = data->repeat_buffer;
93 	}
94 }
95 
ifx_cat1_adc_channel_setup(const struct device * dev,const struct adc_channel_cfg * channel_cfg)96 static int ifx_cat1_adc_channel_setup(const struct device *dev,
97 				      const struct adc_channel_cfg *channel_cfg)
98 {
99 	struct ifx_cat1_adc_data *data = dev->data;
100 	cy_rslt_t result;
101 
102 	cyhal_gpio_t vplus = CYHAL_GET_GPIO(_ADCSAR_PORT, channel_cfg->input_positive);
103 	cyhal_gpio_t vminus = channel_cfg->differential ?
104 			      CYHAL_GET_GPIO(_ADCSAR_PORT, channel_cfg->input_negative) :
105 			      CYHAL_ADC_VNEG;
106 	uint32_t acquisition_ns = ADC_CAT1_DEFAULT_ACQUISITION_NS;
107 
108 	if (channel_cfg->reference != ADC_REF_INTERNAL) {
109 		LOG_ERR("Selected ADC reference is not valid");
110 		return -EINVAL;
111 	}
112 
113 	if (channel_cfg->gain != ADC_GAIN_1) {
114 		LOG_ERR("Selected ADC gain is not valid");
115 		return -EINVAL;
116 	}
117 
118 	if (channel_cfg->acquisition_time != ADC_ACQ_TIME_DEFAULT) {
119 		switch (ADC_ACQ_TIME_UNIT(channel_cfg->acquisition_time)) {
120 		case ADC_ACQ_TIME_MICROSECONDS:
121 			acquisition_ns = ADC_ACQ_TIME_VALUE(channel_cfg->acquisition_time) * 1000;
122 			break;
123 		case ADC_ACQ_TIME_NANOSECONDS:
124 			acquisition_ns = ADC_ACQ_TIME_VALUE(channel_cfg->acquisition_time);
125 			break;
126 		default:
127 			LOG_ERR("Selected ADC acquisition time units is not valid");
128 			return -EINVAL;
129 		}
130 	}
131 
132 	/* ADC channel configuration */
133 	const cyhal_adc_channel_config_t channel_config = {
134 		/* Disable averaging for channel */
135 		.enable_averaging = false,
136 		/* Minimum acquisition time set to 1us */
137 		.min_acquisition_ns = acquisition_ns,
138 		/* Sample channel when ADC performs a scan */
139 		.enabled = true
140 	};
141 
142 	/* Initialize a channel and configure it to scan the input pin(s). */
143 	cyhal_adc_channel_free(&data->adc_chan_obj[channel_cfg->channel_id]);
144 	result = cyhal_adc_channel_init_diff(&data->adc_chan_obj[channel_cfg->channel_id],
145 					     &data->adc_obj, vplus, vminus, &channel_config);
146 	if (result != CY_RSLT_SUCCESS) {
147 		LOG_ERR("ADC channel initialization failed. Error: 0x%08X\n", (unsigned int)result);
148 		return -EIO;
149 	}
150 
151 	data->channels_mask |= BIT(channel_cfg->channel_id);
152 
153 	return 0;
154 }
155 
validate_buffer_size(const struct adc_sequence * sequence)156 static int validate_buffer_size(const struct adc_sequence *sequence)
157 {
158 	int active_channels = 0;
159 	int total_buffer_size;
160 
161 	for (int i = 0; i < CY_SAR_SEQ_NUM_CHANNELS; i++) {
162 		if (sequence->channels & BIT(i)) {
163 			active_channels++;
164 		}
165 	}
166 
167 	total_buffer_size = active_channels * sizeof(uint16_t);
168 
169 	if (sequence->options) {
170 		total_buffer_size *= (1 + sequence->options->extra_samplings);
171 	}
172 
173 	if (sequence->buffer_size < total_buffer_size) {
174 		return -ENOMEM;
175 	}
176 
177 	return 0;
178 }
179 
start_read(const struct device * dev,const struct adc_sequence * sequence)180 static int start_read(const struct device *dev,
181 		      const struct adc_sequence *sequence)
182 {
183 	struct ifx_cat1_adc_data *data = dev->data;
184 	uint32_t channels = sequence->channels;
185 	uint32_t unconfigured_channels = channels & ~data->channels_mask;
186 
187 	if (sequence->resolution != ADC_CAT1_RESOLUTION) {
188 		LOG_ERR("Invalid ADC resolution (%d)", sequence->resolution);
189 		return -EINVAL;
190 	}
191 
192 	if (unconfigured_channels != 0) {
193 		LOG_ERR("ADC channel(s) not configured: 0x%08X\n", unconfigured_channels);
194 		return -EINVAL;
195 	}
196 
197 	if (sequence->oversampling) {
198 		LOG_ERR("Oversampling not supported");
199 		return -ENOTSUP;
200 	}
201 
202 	int return_val = validate_buffer_size(sequence);
203 
204 	if (return_val < 0) {
205 		LOG_ERR("Invalid sequence buffer size");
206 		return return_val;
207 	}
208 
209 	data->channels = channels;
210 	data->buffer = sequence->buffer;
211 	adc_context_start_read(&data->ctx, sequence);
212 
213 	return adc_context_wait_for_completion(&data->ctx);
214 }
215 
ifx_cat1_adc_read(const struct device * dev,const struct adc_sequence * sequence)216 static int ifx_cat1_adc_read(const struct device *dev,
217 			     const struct adc_sequence *sequence)
218 {
219 	int ret;
220 	struct ifx_cat1_adc_data *data = dev->data;
221 
222 	adc_context_lock(&data->ctx, false, NULL);
223 	ret = start_read(dev, sequence);
224 	adc_context_release(&data->ctx, ret);
225 	return ret;
226 }
227 
228 #ifdef CONFIG_ADC_ASYNC
ifx_cat1_adc_read_async(const struct device * dev,const struct adc_sequence * sequence,struct k_poll_signal * async)229 static int ifx_cat1_adc_read_async(const struct device *dev,
230 				   const struct adc_sequence *sequence,
231 				   struct k_poll_signal *async)
232 {
233 	int ret;
234 	struct ifx_cat1_adc_data *data = dev->data;
235 
236 	adc_context_lock(&data->ctx, true, async);
237 	ret = start_read(dev, sequence);
238 	adc_context_release(&data->ctx, ret);
239 
240 	return ret;
241 }
242 #endif
243 
ifx_cat1_adc_init(const struct device * dev)244 static int ifx_cat1_adc_init(const struct device *dev)
245 {
246 	struct ifx_cat1_adc_data *data = dev->data;
247 	const struct ifx_cat1_adc_config *config = dev->config;
248 	cy_rslt_t result;
249 
250 	data->dev = dev;
251 
252 	/* Initialize ADC. The ADC block which can connect to the input pin is selected */
253 	result = cyhal_adc_init(&data->adc_obj, CYHAL_GET_GPIO(_ADCSAR_PORT, 0), NULL);
254 	if (result != CY_RSLT_SUCCESS) {
255 		LOG_ERR("ADC initialization failed. Error: 0x%08X\n", (unsigned int)result);
256 		return -EIO;
257 	}
258 
259 	/* Enable ADC Interrupt */
260 	cyhal_adc_enable_event(&data->adc_obj, (cyhal_adc_event_t)ADC_CAT1_EVENTS_MASK,
261 			       config->irq_priority, true);
262 	cyhal_adc_register_callback(&data->adc_obj, _cyhal_adc_event_callback, (void *) dev);
263 
264 	adc_context_unlock_unconditionally(&data->ctx);
265 
266 	return 0;
267 }
268 
269 static DEVICE_API(adc, adc_cat1_driver_api) = {
270 	.channel_setup = ifx_cat1_adc_channel_setup,
271 	.read = ifx_cat1_adc_read,
272 	#ifdef CONFIG_ADC_ASYNC
273 	.read_async = ifx_cat1_adc_read_async,
274 	#endif
275 	.ref_internal = ADC_CAT1_REF_INTERNAL_MV
276 };
277 
278 /* Macros for ADC instance declaration */
279 #define INFINEON_CAT1_ADC_INIT(n)					       \
280 	static struct ifx_cat1_adc_data ifx_cat1_adc_data##n = {	       \
281 		ADC_CONTEXT_INIT_TIMER(ifx_cat1_adc_data##n, ctx),	       \
282 		ADC_CONTEXT_INIT_LOCK(ifx_cat1_adc_data##n, ctx),	       \
283 		ADC_CONTEXT_INIT_SYNC(ifx_cat1_adc_data##n, ctx),	       \
284 	};								       \
285 									       \
286 	static const struct ifx_cat1_adc_config adc_cat1_cfg_##n = {	       \
287 		.irq_priority = DT_INST_IRQ(n, priority),		       \
288 	};								       \
289 									       \
290 	DEVICE_DT_INST_DEFINE(n, ifx_cat1_adc_init,			       \
291 			      NULL, &ifx_cat1_adc_data##n,		       \
292 			      &adc_cat1_cfg_##n,			       \
293 			      POST_KERNEL, CONFIG_ADC_INIT_PRIORITY,	       \
294 			      &adc_cat1_driver_api);
295 
296 DT_INST_FOREACH_STATUS_OKAY(INFINEON_CAT1_ADC_INIT)
297