1 /*
2  * Copyright 2021-23, NXP
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #include <zephyr/drivers/counter.h>
8 #include <zephyr/irq.h>
9 #include <fsl_rtc.h>
10 #include "fsl_power.h"
11 #include <zephyr/logging/log.h>
12 
13 LOG_MODULE_REGISTER(mcux_rtc, CONFIG_COUNTER_LOG_LEVEL);
14 
15 struct mcux_lpc_rtc_data {
16 	counter_alarm_callback_t alarm_callback;
17 	counter_top_callback_t top_callback;
18 	void *alarm_user_data;
19 	void *top_user_data;
20 	uint32_t value;
21 };
22 
23 struct mcux_lpc_rtc_config {
24 	struct counter_config_info info;
25 	RTC_Type *base;
26 	const struct device *rtc_dev;
27 	void (*irq_config_func)(const struct device *dev);
28 	/* Device defined as wake-up source */
29 	bool wakeup_source;
30 };
31 
32 #if CONFIG_COUNTER_MCUX_LPC_RTC_HIGHRES
33 static int mcux_lpc_rtc_highres_start(const struct device *dev);
34 #endif
35 
mcux_lpc_rtc_isr(const struct device * dev)36 static void mcux_lpc_rtc_isr(const struct device *dev)
37 {
38 	const struct counter_config_info *info = dev->config;
39 	const struct mcux_lpc_rtc_config *config =
40 		CONTAINER_OF(info, struct mcux_lpc_rtc_config, info);
41 	struct mcux_lpc_rtc_data *data = dev->data;
42 	counter_alarm_callback_t cb;
43 	uint32_t current = RTC_GetSecondsTimerCount(config->base);
44 
45 	LOG_DBG("Current time is %d ticks", current);
46 
47 	if ((RTC_GetStatusFlags(config->base) & RTC_CTRL_ALARM1HZ_MASK) &&
48 	    (data->alarm_callback)) {
49 		cb = data->alarm_callback;
50 		data->alarm_callback = NULL;
51 		cb(dev, 0, current, data->alarm_user_data);
52 	}
53 
54 	if (data->top_callback) {
55 		data->top_callback(dev, data->top_user_data);
56 	}
57 
58 	/*
59 	 * Clear any conditions to ack the IRQ
60 	 *
61 	 * callback may have already reset the alarm flag if a new
62 	 * alarm value was programmed to the TAR
63 	 */
64 	if (RTC_GetStatusFlags(config->base) & RTC_CTRL_ALARM1HZ_MASK) {
65 		RTC_ClearStatusFlags(config->base, kRTC_AlarmFlag);
66 	}
67 
68 	/* Check if the Wake counter interrupt was set */
69 	if (RTC_GetStatusFlags(config->base) & RTC_CTRL_WAKE1KHZ_MASK) {
70 		RTC_ClearStatusFlags(config->base, kRTC_WakeupFlag);
71 #if CONFIG_COUNTER_MCUX_LPC_RTC_HIGHRES
72 		if (config->base->CTRL & RTC_CTRL_RTC1KHZ_EN_MASK) {
73 			mcux_lpc_rtc_highres_start(dev);
74 		}
75 #endif
76 	}
77 }
78 
79 #if CONFIG_COUNTER_MCUX_LPC_RTC_1HZ
80 
81 #define DT_DRV_COMPAT nxp_lpc_rtc
82 
mcux_lpc_rtc_start(const struct device * dev)83 static int mcux_lpc_rtc_start(const struct device *dev)
84 {
85 	const struct counter_config_info *info = dev->config;
86 	const struct mcux_lpc_rtc_config *config =
87 		CONTAINER_OF(info, struct mcux_lpc_rtc_config, info);
88 
89 	RTC_EnableTimer(config->base, true);
90 
91 	return 0;
92 }
93 
mcux_lpc_rtc_stop(const struct device * dev)94 static int mcux_lpc_rtc_stop(const struct device *dev)
95 {
96 	const struct counter_config_info *info = dev->config;
97 	const struct mcux_lpc_rtc_config *config =
98 		CONTAINER_OF(info, struct mcux_lpc_rtc_config, info);
99 
100 	RTC_EnableTimer(config->base, false);
101 
102 	/* clear out any set alarms */
103 	RTC_SetSecondsTimerMatch(config->base, 0);
104 
105 	return 0;
106 }
107 
mcux_lpc_rtc_read(const struct device * dev)108 static uint32_t mcux_lpc_rtc_read(const struct device *dev)
109 {
110 	const struct counter_config_info *info = dev->config;
111 	const struct mcux_lpc_rtc_config *config =
112 		CONTAINER_OF(info, struct mcux_lpc_rtc_config, info);
113 
114 	uint32_t ticks = RTC_GetSecondsTimerCount(config->base);
115 
116 	return ticks;
117 }
118 
mcux_lpc_rtc_get_value(const struct device * dev,uint32_t * ticks)119 static int mcux_lpc_rtc_get_value(const struct device *dev, uint32_t *ticks)
120 {
121 	*ticks = mcux_lpc_rtc_read(dev);
122 	return 0;
123 }
124 
mcux_lpc_rtc_set_alarm(const struct device * dev,uint8_t chan_id,const struct counter_alarm_cfg * alarm_cfg)125 static int mcux_lpc_rtc_set_alarm(const struct device *dev, uint8_t chan_id,
126 			      const struct counter_alarm_cfg *alarm_cfg)
127 {
128 	const struct counter_config_info *info = dev->config;
129 	const struct mcux_lpc_rtc_config *config =
130 		CONTAINER_OF(info, struct mcux_lpc_rtc_config, info);
131 	struct mcux_lpc_rtc_data *data = dev->data;
132 
133 	uint32_t ticks = alarm_cfg->ticks;
134 	uint32_t current = mcux_lpc_rtc_read(dev);
135 
136 	LOG_DBG("Current time is %d ticks", current);
137 
138 	if (chan_id != 0U) {
139 		LOG_ERR("Invalid channel id");
140 		return -EINVAL;
141 	}
142 
143 	if (data->alarm_callback != NULL) {
144 		return -EBUSY;
145 	}
146 
147 	if ((alarm_cfg->flags & COUNTER_ALARM_CFG_ABSOLUTE) == 0) {
148 		ticks += current;
149 	}
150 
151 	if (ticks < current) {
152 		LOG_ERR("Alarm cannot be earlier than current time");
153 		return -EINVAL;
154 	}
155 
156 	data->alarm_callback = alarm_cfg->callback;
157 	data->alarm_user_data = alarm_cfg->user_data;
158 
159 	RTC_SetSecondsTimerMatch(config->base, ticks);
160 	LOG_DBG("Alarm set to %d ticks", ticks);
161 
162 	return 0;
163 }
164 
mcux_lpc_rtc_cancel_alarm(const struct device * dev,uint8_t chan_id)165 static int mcux_lpc_rtc_cancel_alarm(const struct device *dev, uint8_t chan_id)
166 {
167 	struct mcux_lpc_rtc_data *data = dev->data;
168 
169 	if (chan_id != 0U) {
170 		LOG_ERR("Invalid channel id");
171 		return -EINVAL;
172 	}
173 
174 	data->alarm_callback = NULL;
175 
176 	return 0;
177 }
178 
mcux_lpc_rtc_set_top_value(const struct device * dev,const struct counter_top_cfg * cfg)179 static int mcux_lpc_rtc_set_top_value(const struct device *dev,
180 				  const struct counter_top_cfg *cfg)
181 {
182 	return -ENOTSUP;
183 }
184 
mcux_lpc_rtc_get_pending_int(const struct device * dev)185 static uint32_t mcux_lpc_rtc_get_pending_int(const struct device *dev)
186 {
187 	const struct counter_config_info *info = dev->config;
188 	const struct mcux_lpc_rtc_config *config =
189 		CONTAINER_OF(info, struct mcux_lpc_rtc_config, info);
190 
191 	return RTC_GetStatusFlags(config->base) & RTC_CTRL_ALARM1HZ_MASK;
192 }
193 
mcux_lpc_rtc_get_top_value(const struct device * dev)194 static uint32_t mcux_lpc_rtc_get_top_value(const struct device *dev)
195 {
196 	const struct counter_config_info *info = dev->config;
197 
198 	return info->max_top_value;
199 }
200 
mcux_lpc_rtc_init(const struct device * dev)201 static int mcux_lpc_rtc_init(const struct device *dev)
202 {
203 	const struct counter_config_info *info = dev->config;
204 	const struct mcux_lpc_rtc_config *config =
205 		CONTAINER_OF(info, struct mcux_lpc_rtc_config, info);
206 
207 	RTC_Init(config->base);
208 
209 	/* Issue a software reset to set the registers to init state */
210 	RTC_Reset(config->base);
211 
212 	config->irq_config_func(dev);
213 
214 	if (config->wakeup_source) {
215 		/* Enable the bit to wakeup from Deep Power Down mode */
216 		RTC_EnableAlarmTimerInterruptFromDPD(config->base, true);
217 	}
218 
219 	return 0;
220 }
221 
222 static DEVICE_API(counter, mcux_rtc_driver_api) = {
223 	.start = mcux_lpc_rtc_start,
224 	.stop = mcux_lpc_rtc_stop,
225 	.get_value = mcux_lpc_rtc_get_value,
226 	.set_alarm = mcux_lpc_rtc_set_alarm,
227 	.cancel_alarm = mcux_lpc_rtc_cancel_alarm,
228 	.set_top_value = mcux_lpc_rtc_set_top_value,
229 	.get_pending_int = mcux_lpc_rtc_get_pending_int,
230 	.get_top_value = mcux_lpc_rtc_get_top_value,
231 };
232 
233 #define COUNTER_LPC_RTC_DEVICE(id)						\
234 	static void mcux_lpc_rtc_irq_config_##id(const struct device *dev);	\
235 	static const struct mcux_lpc_rtc_config mcux_lpc_rtc_config_##id = {	\
236 		.base = (RTC_Type *)DT_INST_REG_ADDR(id),			\
237 		.irq_config_func = mcux_lpc_rtc_irq_config_##id,		\
238 		.rtc_dev = DEVICE_DT_GET_OR_NULL(DT_INST_CHILD(id, rtc_highres)),	\
239 		.info = {						\
240 			.max_top_value = UINT32_MAX,			\
241 			.freq = 1,					\
242 			.flags = COUNTER_CONFIG_INFO_COUNT_UP,		\
243 			.channels = 1,					\
244 		},							\
245 		.wakeup_source = DT_INST_PROP(id, wakeup_source)	\
246 	};								\
247 	static struct mcux_lpc_rtc_data mcux_lpc_rtc_data_##id;		\
248 	DEVICE_DT_INST_DEFINE(id, &mcux_lpc_rtc_init, NULL,		\
249 				&mcux_lpc_rtc_data_##id, &mcux_lpc_rtc_config_##id.info,	\
250 				POST_KERNEL, CONFIG_COUNTER_INIT_PRIORITY,		\
251 				&mcux_rtc_driver_api);					\
252 	static void mcux_lpc_rtc_irq_config_##id(const struct device *dev)	\
253 	{									\
254 		IRQ_CONNECT(DT_INST_IRQN(id),					\
255 			DT_INST_IRQ(id, priority),				\
256 			mcux_lpc_rtc_isr, DEVICE_DT_INST_GET(id), 0);		\
257 		irq_enable(DT_INST_IRQN(id));					\
258 		if (DT_INST_PROP(id, wakeup_source)) {				\
259 			EnableDeepSleepIRQ(DT_INST_IRQN(id));			\
260 		}								\
261 	}
262 
DT_INST_FOREACH_STATUS_OKAY(COUNTER_LPC_RTC_DEVICE)263 DT_INST_FOREACH_STATUS_OKAY(COUNTER_LPC_RTC_DEVICE)
264 #endif
265 
266 #if CONFIG_COUNTER_MCUX_LPC_RTC_HIGHRES
267 
268 #undef DT_DRV_COMPAT
269 #define DT_DRV_COMPAT nxp_lpc_rtc_highres
270 
271 static int mcux_lpc_rtc_highres_start(const struct device *dev)
272 {
273 	const struct counter_config_info *info = dev->config;
274 	const struct mcux_lpc_rtc_config *config =
275 		CONTAINER_OF(info, struct mcux_lpc_rtc_config, info);
276 	struct mcux_lpc_rtc_data *data = dev->data;
277 
278 	if (config->rtc_dev) {
279 		/* We have another RTC driver enabled, check if RTC is enabled */
280 		if ((config->base->CTRL & RTC_CTRL_RTC_EN_MASK) == 0) {
281 			/* RTC is not enabled and we do not turn it on as it will effect
282 			 * the RTC counter value thereby affecting the RTC counter drivers
283 			 */
284 			LOG_ERR("RTC Wake counter cannot be started as RTC is not enabled.");
285 			return -EINVAL;
286 		}
287 	} else {
288 		if ((config->base->CTRL & RTC_CTRL_RTC_EN_MASK) == 0) {
289 			RTC_EnableTimer(config->base, true);
290 		}
291 	}
292 
293 	if (data->value == 0) {
294 		/* Start from the max value */
295 		RTC_SetWakeupCount(config->base, counter_get_top_value(dev));
296 	} else {
297 		RTC_SetWakeupCount(config->base, data->value);
298 	}
299 
300 	return 0;
301 }
302 
mcux_lpc_rtc_highres_stop(const struct device * dev)303 static int mcux_lpc_rtc_highres_stop(const struct device *dev)
304 {
305 	const struct counter_config_info *info = dev->config;
306 	const struct mcux_lpc_rtc_config *config =
307 		CONTAINER_OF(info, struct mcux_lpc_rtc_config, info);
308 
309 	config->base->CTRL &= ~RTC_CTRL_RTC1KHZ_EN_MASK;
310 
311 	if (config->rtc_dev == NULL) {
312 		/* Disable RTC as no other driver is using it */
313 		RTC_EnableTimer(config->base, false);
314 	}
315 
316 	return 0;
317 }
318 
mcux_lpc_rtc_highres_read(const struct device * dev)319 static uint32_t mcux_lpc_rtc_highres_read(const struct device *dev)
320 {
321 	const struct counter_config_info *info = dev->config;
322 	const struct mcux_lpc_rtc_config *config =
323 		CONTAINER_OF(info, struct mcux_lpc_rtc_config, info);
324 
325 	uint32_t ticks = RTC_GetWakeupCount(config->base);
326 
327 	return ticks;
328 }
329 
mcux_lpc_rtc_highres_set_alarm(const struct device * dev,uint8_t chan_id,const struct counter_alarm_cfg * alarm_cfg)330 static int mcux_lpc_rtc_highres_set_alarm(const struct device *dev, uint8_t chan_id,
331 					  const struct counter_alarm_cfg *alarm_cfg)
332 {
333 	return -ENOTSUP;
334 }
335 
mcux_lpc_rtc_highres_cancel_alarm(const struct device * dev,uint8_t chan_id)336 static int mcux_lpc_rtc_highres_cancel_alarm(const struct device *dev, uint8_t chan_id)
337 {
338 	return -ENOTSUP;
339 }
340 
341 
mcux_lpc_rtc_highres_get_value(const struct device * dev,uint32_t * ticks)342 static int mcux_lpc_rtc_highres_get_value(const struct device *dev, uint32_t *ticks)
343 {
344 	*ticks = mcux_lpc_rtc_highres_read(dev);
345 	return 0;
346 }
347 
mcux_lpc_rtc_highres_set_top_value(const struct device * dev,const struct counter_top_cfg * cfg)348 static int mcux_lpc_rtc_highres_set_top_value(const struct device *dev,
349 				  const struct counter_top_cfg *cfg)
350 {
351 	const struct counter_config_info *info = dev->config;
352 	const struct mcux_lpc_rtc_config *config =
353 		CONTAINER_OF(info, struct mcux_lpc_rtc_config, info);
354 	struct mcux_lpc_rtc_data *data = dev->data;
355 
356 	if (cfg->flags & COUNTER_TOP_CFG_DONT_RESET) {
357 		return -ENOTSUP;
358 	}
359 
360 	data->value = cfg->ticks;
361 	data->top_callback = cfg->callback;
362 	data->top_user_data = cfg->user_data;
363 
364 	if (config->base->CTRL & RTC_CTRL_RTC1KHZ_EN_MASK) {
365 		return mcux_lpc_rtc_highres_start(dev);
366 	}
367 
368 	return 0;
369 }
370 
mcux_lpc_rtc_highres_get_pending_int(const struct device * dev)371 static uint32_t mcux_lpc_rtc_highres_get_pending_int(const struct device *dev)
372 {
373 	const struct counter_config_info *info = dev->config;
374 	const struct mcux_lpc_rtc_config *config =
375 		CONTAINER_OF(info, struct mcux_lpc_rtc_config, info);
376 
377 	return RTC_GetStatusFlags(config->base) & RTC_CTRL_WAKE1KHZ_MASK;
378 }
379 
mcux_lpc_rtc_highres_get_top_value(const struct device * dev)380 static uint32_t mcux_lpc_rtc_highres_get_top_value(const struct device *dev)
381 {
382 	struct mcux_lpc_rtc_data *data = dev->data;
383 	const struct counter_config_info *info = dev->config;
384 
385 	if (data->value == 0) {
386 		return info->max_top_value;
387 	} else {
388 		return data->value;
389 	}
390 }
391 
mcux_lpc_rtc_highres_init(const struct device * dev)392 static int mcux_lpc_rtc_highres_init(const struct device *dev)
393 {
394 	const struct counter_config_info *info = dev->config;
395 	const struct mcux_lpc_rtc_config *config =
396 		CONTAINER_OF(info, struct mcux_lpc_rtc_config, info);
397 
398 	/* Initialize the RTC if this is only driver using it */
399 	if (config->rtc_dev == NULL) {
400 		RTC_Init(config->base);
401 
402 		/* Issue a software reset to set the registers to init state */
403 		RTC_Reset(config->base);
404 
405 		config->irq_config_func(dev);
406 	}
407 
408 	if (config->wakeup_source) {
409 		/* Enable the bit to wakeup from Deep Power Down mode */
410 		RTC_EnableWakeUpTimerInterruptFromDPD(config->base, true);
411 	}
412 
413 	return 0;
414 }
415 
416 static DEVICE_API(counter, mcux_rtc_highres_driver_api) = {
417 	.start = mcux_lpc_rtc_highres_start,
418 	.stop = mcux_lpc_rtc_highres_stop,
419 	.get_value = mcux_lpc_rtc_highres_get_value,
420 	.set_alarm = mcux_lpc_rtc_highres_set_alarm,
421 	.cancel_alarm = mcux_lpc_rtc_highres_cancel_alarm,
422 	.set_top_value = mcux_lpc_rtc_highres_set_top_value,
423 	.get_pending_int = mcux_lpc_rtc_highres_get_pending_int,
424 	.get_top_value = mcux_lpc_rtc_highres_get_top_value,
425 };
426 
427 #define COUNTER_LPC_RTC_HIGHRES_IRQ_INIT(n)							\
428 	do {											\
429 		IRQ_CONNECT(DT_IRQN(DT_INST_PARENT(n)),						\
430 			DT_IRQ(DT_INST_PARENT(n), priority),					\
431 			mcux_lpc_rtc_isr,							\
432 			DEVICE_DT_INST_GET(n), 0);						\
433 		irq_enable(DT_IRQN(DT_INST_PARENT(n)));						\
434 		if (DT_INST_PROP(n, wakeup_source)) {						\
435 			EnableDeepSleepIRQ(DT_IRQN(DT_INST_PARENT(n)));				\
436 		}										\
437 	} while (false)
438 
439 #define COUNTER_LPC_RTC_HIGHRES_DEVICE(id)							\
440 	static void mcux_lpc_rtc_highres_irq_config_##id(const struct device *dev);		\
441 	static const struct mcux_lpc_rtc_config mcux_lpc_rtc_highres_config_##id = {		\
442 		.base = (RTC_Type *)DT_REG_ADDR(DT_INST_PARENT(id)),				\
443 		.rtc_dev = DEVICE_DT_GET_OR_NULL(DT_INST_PARENT(id)),				\
444 		.irq_config_func = mcux_lpc_rtc_highres_irq_config_##id,			\
445 		.info = {									\
446 			.max_top_value = UINT16_MAX,						\
447 			.freq = 1000,								\
448 			.channels = 0,								\
449 		},										\
450 		.wakeup_source = DT_INST_PROP(id, wakeup_source)				\
451 	};											\
452 	static struct mcux_lpc_rtc_data mcux_lpc_rtc_highres_data_##id;				\
453 	DEVICE_DT_INST_DEFINE(id, &mcux_lpc_rtc_highres_init, NULL,				\
454 				&mcux_lpc_rtc_highres_data_##id,				\
455 				&mcux_lpc_rtc_highres_config_##id.info,				\
456 				POST_KERNEL, CONFIG_COUNTER_INIT_PRIORITY,			\
457 				&mcux_rtc_highres_driver_api);					\
458 	static void mcux_lpc_rtc_highres_irq_config_##id(const struct device *dev)		\
459 	{											\
460 		COND_CODE_1(IS_ENABLED(DT_HAS_COMPAT_STATUS_OKAY(nxp_lpc_rtc)),			\
461 			 (), (COUNTER_LPC_RTC_HIGHRES_IRQ_INIT(id));)				\
462 	}
463 
464 DT_INST_FOREACH_STATUS_OKAY(COUNTER_LPC_RTC_HIGHRES_DEVICE)
465 
466 #endif
467