1 /*
2  * Copyright (c) 2018 blik GmbH
3  * Copyright (c) 2018,2024 NXP
4  *
5  * SPDX-License-Identifier: Apache-2.0
6  */
7 
8 #define DT_DRV_COMPAT nxp_rtc
9 
10 #include <zephyr/drivers/counter.h>
11 #include <zephyr/irq.h>
12 #include <zephyr/kernel.h>
13 #include <zephyr/sys_clock.h>
14 #include <fsl_rtc.h>
15 #include <zephyr/logging/log.h>
16 
17 /*
18  * FSL_FEATURE_* is defined with paranethesis
19  * which is not acceptable when using the IS_ENABLED() macro
20  */
21 #if (defined(FSL_FEATURE_RTC_HAS_LPO_ADJUST) && FSL_FEATURE_RTC_HAS_LPO_ADJUST)
22 #define NXP_RTC_HAS_LPO_ADJUST 1
23 #else
24 #define NXP_RTC_HAS_LPO_ADJUST 0
25 #endif
26 
27 LOG_MODULE_REGISTER(mcux_rtc, CONFIG_COUNTER_LOG_LEVEL);
28 
29 struct mcux_rtc_data {
30 	counter_alarm_callback_t alarm_callback;
31 	counter_top_callback_t top_callback;
32 	void *alarm_user_data;
33 	void *top_user_data;
34 };
35 
36 struct mcux_rtc_config {
37 	struct counter_config_info info;
38 	RTC_Type *base;
39 	void (*irq_config_func)(const struct device *dev);
40 };
41 
mcux_rtc_start(const struct device * dev)42 static int mcux_rtc_start(const struct device *dev)
43 {
44 	const struct counter_config_info *info = dev->config;
45 	const struct mcux_rtc_config *config =
46 		CONTAINER_OF(info, struct mcux_rtc_config, info);
47 
48 	RTC_StartTimer(config->base);
49 	RTC_EnableInterrupts(config->base,
50 			     kRTC_AlarmInterruptEnable |
51 			     kRTC_TimeOverflowInterruptEnable |
52 			     kRTC_TimeInvalidInterruptEnable);
53 
54 	return 0;
55 }
56 
mcux_rtc_stop(const struct device * dev)57 static int mcux_rtc_stop(const struct device *dev)
58 {
59 	const struct counter_config_info *info = dev->config;
60 	const struct mcux_rtc_config *config =
61 		CONTAINER_OF(info, struct mcux_rtc_config, info);
62 
63 	RTC_DisableInterrupts(config->base,
64 			      kRTC_AlarmInterruptEnable |
65 			      kRTC_TimeOverflowInterruptEnable |
66 			      kRTC_TimeInvalidInterruptEnable);
67 	RTC_StopTimer(config->base);
68 
69 	/* clear out any set alarms */
70 	config->base->TAR = 0;
71 
72 	return 0;
73 }
74 
mcux_rtc_read(const struct device * dev)75 static uint32_t mcux_rtc_read(const struct device *dev)
76 {
77 	const struct counter_config_info *info = dev->config;
78 	const struct mcux_rtc_config *config =
79 		CONTAINER_OF(info, struct mcux_rtc_config, info);
80 
81 	uint32_t ticks = config->base->TSR;
82 
83 	/*
84 	 * Read TSR seconds twice in case it glitches during an update.
85 	 * This can happen when a read occurs at the time the register is
86 	 * incrementing.
87 	 */
88 	if (config->base->TSR == ticks) {
89 		return ticks;
90 	}
91 
92 	ticks = config->base->TSR;
93 
94 	return ticks;
95 }
96 
mcux_rtc_get_value(const struct device * dev,uint32_t * ticks)97 static int mcux_rtc_get_value(const struct device *dev, uint32_t *ticks)
98 {
99 	*ticks = mcux_rtc_read(dev);
100 	return 0;
101 }
102 
mcux_rtc_set_alarm(const struct device * dev,uint8_t chan_id,const struct counter_alarm_cfg * alarm_cfg)103 static int mcux_rtc_set_alarm(const struct device *dev, uint8_t chan_id,
104 			      const struct counter_alarm_cfg *alarm_cfg)
105 {
106 	const struct counter_config_info *info = dev->config;
107 	const struct mcux_rtc_config *config =
108 		CONTAINER_OF(info, struct mcux_rtc_config, info);
109 	struct mcux_rtc_data *data = dev->data;
110 
111 	uint32_t ticks = alarm_cfg->ticks;
112 	uint32_t current = mcux_rtc_read(dev);
113 
114 	LOG_DBG("Current time is %d ticks", current);
115 
116 	if (chan_id != 0U) {
117 		LOG_ERR("Invalid channel id");
118 		return -EINVAL;
119 	}
120 
121 	if (data->alarm_callback != NULL) {
122 		return -EBUSY;
123 	}
124 
125 	if ((alarm_cfg->flags & COUNTER_ALARM_CFG_ABSOLUTE) == 0) {
126 		ticks += current;
127 	}
128 
129 	if (ticks < current) {
130 		LOG_ERR("Alarm cannot be earlier than current time");
131 		return -EINVAL;
132 	}
133 
134 	data->alarm_callback = alarm_cfg->callback;
135 	data->alarm_user_data = alarm_cfg->user_data;
136 
137 	config->base->TAR = ticks;
138 	LOG_DBG("Alarm set to %d ticks", ticks);
139 
140 	return 0;
141 }
142 
mcux_rtc_cancel_alarm(const struct device * dev,uint8_t chan_id)143 static int mcux_rtc_cancel_alarm(const struct device *dev, uint8_t chan_id)
144 {
145 	struct mcux_rtc_data *data = dev->data;
146 
147 	if (chan_id != 0U) {
148 		LOG_ERR("Invalid channel id");
149 		return -EINVAL;
150 	}
151 
152 	data->alarm_callback = NULL;
153 
154 	return 0;
155 }
156 
mcux_rtc_set_top_value(const struct device * dev,const struct counter_top_cfg * cfg)157 static int mcux_rtc_set_top_value(const struct device *dev,
158 				  const struct counter_top_cfg *cfg)
159 {
160 	const struct counter_config_info *info = dev->config;
161 	const struct mcux_rtc_config *config =
162 			CONTAINER_OF(info, struct mcux_rtc_config, info);
163 	struct mcux_rtc_data *data = dev->data;
164 
165 	if (cfg->ticks != info->max_top_value) {
166 		LOG_ERR("Wrap can only be set to 0x%x.", info->max_top_value);
167 		return -ENOTSUP;
168 	}
169 
170 	if (!(cfg->flags & COUNTER_TOP_CFG_DONT_RESET)) {
171 		RTC_StopTimer(config->base);
172 		config->base->TSR = 0;
173 		RTC_StartTimer(config->base);
174 	}
175 
176 	data->top_callback = cfg->callback;
177 	data->top_user_data = cfg->user_data;
178 
179 	return 0;
180 }
181 
mcux_rtc_get_pending_int(const struct device * dev)182 static uint32_t mcux_rtc_get_pending_int(const struct device *dev)
183 {
184 	const struct counter_config_info *info = dev->config;
185 	const struct mcux_rtc_config *config =
186 		CONTAINER_OF(info, struct mcux_rtc_config, info);
187 
188 	return RTC_GetStatusFlags(config->base) & RTC_SR_TAF_MASK;
189 }
190 
mcux_rtc_get_top_value(const struct device * dev)191 static uint32_t mcux_rtc_get_top_value(const struct device *dev)
192 {
193 	const struct counter_config_info *info = dev->config;
194 
195 	return info->max_top_value;
196 }
197 
mcux_rtc_isr(const struct device * dev)198 static void mcux_rtc_isr(const struct device *dev)
199 {
200 	const struct counter_config_info *info = dev->config;
201 	const struct mcux_rtc_config *config =
202 		CONTAINER_OF(info, struct mcux_rtc_config, info);
203 	struct mcux_rtc_data *data = dev->data;
204 	counter_alarm_callback_t cb;
205 	uint32_t current = mcux_rtc_read(dev);
206 
207 
208 	LOG_DBG("Current time is %d ticks", current);
209 
210 	if ((RTC_GetStatusFlags(config->base) & RTC_SR_TAF_MASK) &&
211 	    (data->alarm_callback)) {
212 		cb = data->alarm_callback;
213 		data->alarm_callback = NULL;
214 		cb(dev, 0, current, data->alarm_user_data);
215 	}
216 
217 	if ((RTC_GetStatusFlags(config->base) & RTC_SR_TOF_MASK) &&
218 	    (data->top_callback)) {
219 		data->top_callback(dev, data->top_user_data);
220 	}
221 
222 	/*
223 	 * Clear any conditions to ack the IRQ
224 	 *
225 	 * callback may have already reset the alarm flag if a new
226 	 * alarm value was programmed to the TAR
227 	 */
228 	RTC_StopTimer(config->base);
229 	if (RTC_GetStatusFlags(config->base) & RTC_SR_TAF_MASK) {
230 		RTC_ClearStatusFlags(config->base, kRTC_AlarmFlag);
231 	} else if (RTC_GetStatusFlags(config->base) & RTC_SR_TIF_MASK) {
232 		RTC_ClearStatusFlags(config->base, kRTC_TimeInvalidFlag);
233 	} else if (RTC_GetStatusFlags(config->base) & RTC_SR_TOF_MASK) {
234 		RTC_ClearStatusFlags(config->base, kRTC_TimeOverflowFlag);
235 	}
236 	RTC_StartTimer(config->base);
237 }
238 
mcux_rtc_init(const struct device * dev)239 static int mcux_rtc_init(const struct device *dev)
240 {
241 	const struct counter_config_info *info = dev->config;
242 	const struct mcux_rtc_config *config =
243 		CONTAINER_OF(info, struct mcux_rtc_config, info);
244 	rtc_config_t rtc_config;
245 
246 	RTC_GetDefaultConfig(&rtc_config);
247 	RTC_Init(config->base, &rtc_config);
248 
249 	/* DT_ENUM_IDX(DT_NODELABEL(rtc), clock_source):
250 	 * "RTC": 0
251 	 * "LPO": 1
252 	 */
253 	BUILD_ASSERT((((DT_INST_ENUM_IDX(0, clock_source) == 1) &&
254 		NXP_RTC_HAS_LPO_ADJUST) ||
255 		DT_INST_ENUM_IDX(0, clock_source) == 0),
256 		"Cannot choose the LPO clock for that instance of the RTC");
257 
258 #if (defined(FSL_FEATURE_RTC_HAS_LPO_ADJUST) && FSL_FEATURE_RTC_HAS_LPO_ADJUST)
259 	/* The RTC prescaler increments using the LPO 1 kHz clock
260 	 * instead of the RTC clock
261 	 */
262 	RTC_EnableLPOClock(config->base, DT_INST_ENUM_IDX(0, clock_source));
263 #endif
264 
265 #if !(defined(FSL_FEATURE_RTC_HAS_NO_CR_OSCE) && FSL_FEATURE_RTC_HAS_NO_CR_OSCE)
266 	/* Enable 32kHz oscillator and wait for 1ms to settle */
267 	RTC_SetClockSource(config->base);
268 	k_busy_wait(USEC_PER_MSEC);
269 #endif /* !FSL_FEATURE_RTC_HAS_NO_CR_OSCE */
270 
271 	config->irq_config_func(dev);
272 
273 	return 0;
274 }
275 
276 static DEVICE_API(counter, mcux_rtc_driver_api) = {
277 	.start = mcux_rtc_start,
278 	.stop = mcux_rtc_stop,
279 	.get_value = mcux_rtc_get_value,
280 	.set_alarm = mcux_rtc_set_alarm,
281 	.cancel_alarm = mcux_rtc_cancel_alarm,
282 	.set_top_value = mcux_rtc_set_top_value,
283 	.get_pending_int = mcux_rtc_get_pending_int,
284 	.get_top_value = mcux_rtc_get_top_value,
285 };
286 
287 static struct mcux_rtc_data mcux_rtc_data_0;
288 
289 static void mcux_rtc_irq_config_0(const struct device *dev);
290 
291 static struct mcux_rtc_config mcux_rtc_config_0 = {
292 	.base = (RTC_Type *)DT_INST_REG_ADDR(0),
293 	.irq_config_func = mcux_rtc_irq_config_0,
294 	.info = {
295 		.max_top_value = UINT32_MAX,
296 		.freq = DT_INST_PROP(0, clock_frequency) /
297 				DT_INST_PROP(0, prescaler),
298 		.flags = COUNTER_CONFIG_INFO_COUNT_UP,
299 		.channels = 1,
300 	},
301 };
302 
303 DEVICE_DT_INST_DEFINE(0, &mcux_rtc_init, NULL,
304 		    &mcux_rtc_data_0, &mcux_rtc_config_0.info,
305 		    POST_KERNEL, CONFIG_COUNTER_INIT_PRIORITY,
306 		    &mcux_rtc_driver_api);
307 
mcux_rtc_irq_config_0(const struct device * dev)308 static void mcux_rtc_irq_config_0(const struct device *dev)
309 {
310 	IRQ_CONNECT(DT_INST_IRQN(0),
311 		    DT_INST_IRQ(0, priority),
312 		    mcux_rtc_isr, DEVICE_DT_INST_GET(0), 0);
313 	irq_enable(DT_INST_IRQN(0));
314 }
315