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