1 /*
2  * Copyright (c) 2019, Linaro Limited.
3  * Copyright 2024 NXP
4  *
5  * SPDX-License-Identifier: Apache-2.0
6  */
7 
8 #define DT_DRV_COMPAT nxp_imx_gpt
9 
10 #include <zephyr/drivers/counter.h>
11 #include <zephyr/drivers/clock_control.h>
12 #include <zephyr/irq.h>
13 #include <fsl_gpt.h>
14 #include <zephyr/logging/log.h>
15 #include <zephyr/sys/barrier.h>
16 
17 LOG_MODULE_REGISTER(mcux_gpt, CONFIG_COUNTER_LOG_LEVEL);
18 
19 #define DEV_CFG(_dev) ((const struct mcux_gpt_config *)(_dev)->config)
20 #define DEV_DATA(_dev) ((struct mcux_gpt_data *)(_dev)->data)
21 
22 struct mcux_gpt_config {
23 	/* info must be first element */
24 	struct counter_config_info info;
25 
26 	DEVICE_MMIO_NAMED_ROM(gpt_mmio);
27 
28 	const struct device *clock_dev;
29 	clock_control_subsys_t clock_subsys;
30 	clock_name_t clock_source;
31 };
32 
33 struct mcux_gpt_data {
34 	DEVICE_MMIO_NAMED_RAM(gpt_mmio);
35 	counter_alarm_callback_t alarm_callback;
36 	counter_top_callback_t top_callback;
37 	void *alarm_user_data;
38 	void *top_user_data;
39 };
40 
get_base(const struct device * dev)41 static GPT_Type *get_base(const struct device *dev)
42 {
43 	return (GPT_Type *)DEVICE_MMIO_NAMED_GET(dev, gpt_mmio);
44 }
45 
mcux_gpt_start(const struct device * dev)46 static int mcux_gpt_start(const struct device *dev)
47 {
48 	GPT_Type *base = get_base(dev);
49 
50 	GPT_StartTimer(base);
51 
52 	return 0;
53 }
54 
mcux_gpt_stop(const struct device * dev)55 static int mcux_gpt_stop(const struct device *dev)
56 {
57 	GPT_Type *base = get_base(dev);
58 
59 	GPT_StopTimer(base);
60 
61 	return 0;
62 }
63 
mcux_gpt_get_value(const struct device * dev,uint32_t * ticks)64 static int mcux_gpt_get_value(const struct device *dev, uint32_t *ticks)
65 {
66 	GPT_Type *base = get_base(dev);
67 
68 	*ticks = GPT_GetCurrentTimerCount(base);
69 	return 0;
70 }
71 
mcux_gpt_set_alarm(const struct device * dev,uint8_t chan_id,const struct counter_alarm_cfg * alarm_cfg)72 static int mcux_gpt_set_alarm(const struct device *dev, uint8_t chan_id,
73 			      const struct counter_alarm_cfg *alarm_cfg)
74 {
75 	GPT_Type *base = get_base(dev);
76 	struct mcux_gpt_data *data = dev->data;
77 
78 	uint32_t current = GPT_GetCurrentTimerCount(base);
79 	uint32_t ticks = alarm_cfg->ticks;
80 
81 	if (chan_id != 0) {
82 		LOG_ERR("Invalid channel id");
83 		return -EINVAL;
84 	}
85 
86 	if ((alarm_cfg->flags & COUNTER_ALARM_CFG_ABSOLUTE) == 0) {
87 		ticks += current;
88 	}
89 
90 	if (data->alarm_callback) {
91 		return -EBUSY;
92 	}
93 
94 	data->alarm_callback = alarm_cfg->callback;
95 	data->alarm_user_data = alarm_cfg->user_data;
96 
97 	GPT_SetOutputCompareValue(base, kGPT_OutputCompare_Channel1,
98 				  ticks);
99 	GPT_EnableInterrupts(base, kGPT_OutputCompare1InterruptEnable);
100 
101 	return 0;
102 }
103 
mcux_gpt_cancel_alarm(const struct device * dev,uint8_t chan_id)104 static int mcux_gpt_cancel_alarm(const struct device *dev, uint8_t chan_id)
105 {
106 	GPT_Type *base = get_base(dev);
107 	struct mcux_gpt_data *data = dev->data;
108 
109 	if (chan_id != 0) {
110 		LOG_ERR("Invalid channel id");
111 		return -EINVAL;
112 	}
113 
114 	GPT_DisableInterrupts(base, kGPT_OutputCompare1InterruptEnable);
115 	data->alarm_callback = NULL;
116 
117 	return 0;
118 }
119 
mcux_gpt_isr(const struct device * dev)120 void mcux_gpt_isr(const struct device *dev)
121 {
122 	GPT_Type *base = get_base(dev);
123 	struct mcux_gpt_data *data = dev->data;
124 	uint32_t current = GPT_GetCurrentTimerCount(base);
125 	uint32_t status;
126 
127 	status =  GPT_GetStatusFlags(base, kGPT_OutputCompare1Flag |
128 				     kGPT_RollOverFlag);
129 	GPT_ClearStatusFlags(base, status);
130 	barrier_dsync_fence_full();
131 
132 	if ((status & kGPT_OutputCompare1Flag) && data->alarm_callback) {
133 		GPT_DisableInterrupts(base,
134 				      kGPT_OutputCompare1InterruptEnable);
135 		counter_alarm_callback_t alarm_cb = data->alarm_callback;
136 		data->alarm_callback = NULL;
137 		alarm_cb(dev, 0, current, data->alarm_user_data);
138 	}
139 
140 	if ((status & kGPT_RollOverFlag) && data->top_callback) {
141 		data->top_callback(dev, data->top_user_data);
142 	}
143 }
144 
mcux_gpt_get_pending_int(const struct device * dev)145 static uint32_t mcux_gpt_get_pending_int(const struct device *dev)
146 {
147 	GPT_Type *base = get_base(dev);
148 
149 	return GPT_GetStatusFlags(base, kGPT_OutputCompare1Flag);
150 }
151 
mcux_gpt_set_top_value(const struct device * dev,const struct counter_top_cfg * cfg)152 static int mcux_gpt_set_top_value(const struct device *dev,
153 				  const struct counter_top_cfg *cfg)
154 {
155 	const struct mcux_gpt_config *config = dev->config;
156 	GPT_Type *base = get_base(dev);
157 	struct mcux_gpt_data *data = dev->data;
158 
159 	if (cfg->ticks != config->info.max_top_value) {
160 		LOG_ERR("Wrap can only be set to 0x%x",
161 			config->info.max_top_value);
162 		return -ENOTSUP;
163 	}
164 
165 	data->top_callback = cfg->callback;
166 	data->top_user_data = cfg->user_data;
167 
168 	GPT_EnableInterrupts(base, kGPT_RollOverFlagInterruptEnable);
169 
170 	return 0;
171 }
172 
mcux_gpt_get_top_value(const struct device * dev)173 static uint32_t mcux_gpt_get_top_value(const struct device *dev)
174 {
175 	const struct mcux_gpt_config *config = dev->config;
176 
177 	return config->info.max_top_value;
178 }
179 
mcux_gpt_init(const struct device * dev)180 static int mcux_gpt_init(const struct device *dev)
181 {
182 	const struct mcux_gpt_config *config = dev->config;
183 	gpt_config_t gptConfig;
184 	uint32_t clock_freq;
185 	GPT_Type *base;
186 
187 	DEVICE_MMIO_NAMED_MAP(dev, gpt_mmio, K_MEM_CACHE_NONE | K_MEM_DIRECT_MAP);
188 
189 	if (!device_is_ready(config->clock_dev)) {
190 		LOG_ERR("clock control device not ready");
191 		return -ENODEV;
192 	}
193 
194 	if (clock_control_get_rate(config->clock_dev, config->clock_subsys,
195 				   &clock_freq)) {
196 		return -EINVAL;
197 	}
198 
199 	/* Adjust divider to match expected freq */
200 	if (clock_freq % config->info.freq) {
201 		LOG_ERR("Cannot Adjust GPT freq to %u\n", config->info.freq);
202 		LOG_ERR("clock src is %u\n", clock_freq);
203 		return -EINVAL;
204 	}
205 
206 	GPT_GetDefaultConfig(&gptConfig);
207 	gptConfig.enableFreeRun = true; /* Do not reset on compare */
208 	gptConfig.clockSource = kGPT_ClockSource_Periph;
209 	gptConfig.divider = clock_freq / config->info.freq;
210 	base = get_base(dev);
211 	GPT_Init(base, &gptConfig);
212 
213 	return 0;
214 }
215 
216 static DEVICE_API(counter, mcux_gpt_driver_api) = {
217 	.start = mcux_gpt_start,
218 	.stop = mcux_gpt_stop,
219 	.get_value = mcux_gpt_get_value,
220 	.set_alarm = mcux_gpt_set_alarm,
221 	.cancel_alarm = mcux_gpt_cancel_alarm,
222 	.set_top_value = mcux_gpt_set_top_value,
223 	.get_pending_int = mcux_gpt_get_pending_int,
224 	.get_top_value = mcux_gpt_get_top_value,
225 };
226 
227 #define GPT_DEVICE_INIT_MCUX(n)						\
228 	static struct mcux_gpt_data mcux_gpt_data_ ## n;		\
229 									\
230 	static const struct mcux_gpt_config mcux_gpt_config_ ## n = {	\
231 		DEVICE_MMIO_NAMED_ROM_INIT(gpt_mmio, DT_DRV_INST(n)),	\
232 		.clock_dev = DEVICE_DT_GET(DT_INST_CLOCKS_CTLR(n)),	\
233 		.clock_subsys =						\
234 			(clock_control_subsys_t)DT_INST_CLOCKS_CELL(n, name),\
235 		.info = {						\
236 			.max_top_value = UINT32_MAX,			\
237 			.freq = DT_INST_PROP(n, gptfreq),           \
238 			.channels = 1,					\
239 			.flags = COUNTER_CONFIG_INFO_COUNT_UP,		\
240 		},							\
241 	};								\
242 									\
243 	static int mcux_gpt_## n ##_init(const struct device *dev);	\
244 	DEVICE_DT_INST_DEFINE(n,					\
245 			    mcux_gpt_## n ##_init,			\
246 			    NULL,					\
247 			    &mcux_gpt_data_ ## n,			\
248 			    &mcux_gpt_config_ ## n,			\
249 			    POST_KERNEL,				\
250 			    CONFIG_COUNTER_INIT_PRIORITY,		\
251 			    &mcux_gpt_driver_api);			\
252 									\
253 	static int mcux_gpt_## n ##_init(const struct device *dev)	\
254 	{								\
255 		IRQ_CONNECT(DT_INST_IRQN(n),				\
256 			    DT_INST_IRQ(n, priority),			\
257 			    mcux_gpt_isr, DEVICE_DT_INST_GET(n), 0);	\
258 		irq_enable(DT_INST_IRQN(n));				\
259 		return mcux_gpt_init(dev);				\
260 	}								\
261 
262 DT_INST_FOREACH_STATUS_OKAY(GPT_DEVICE_INIT_MCUX)
263