1 /*
2 * Copyright (c) 2018-2019 NXP
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6 #define DT_DRV_COMPAT nxp_imx_epit
7
8 #include <zephyr/drivers/counter.h>
9 #include <zephyr/device.h>
10 #include <zephyr/irq.h>
11 #include "clock_freq.h"
12 #include "epit.h"
13
14 #define COUNTER_MAX_RELOAD 0xFFFFFFFF
15
16 struct imx_epit_config {
17 struct counter_config_info info;
18 EPIT_Type *base;
19 uint16_t prescaler;
20 };
21
22 struct imx_epit_data {
23 volatile counter_top_callback_t callback;
24 volatile void *user_data;
25 };
26
get_epit_config(const struct device * dev)27 static inline const struct imx_epit_config *get_epit_config(const struct device *dev)
28 {
29 return CONTAINER_OF(dev->config, struct imx_epit_config,
30 info);
31 }
32
imx_epit_isr(const struct device * dev)33 static void imx_epit_isr(const struct device *dev)
34 {
35 EPIT_Type *base = get_epit_config(dev)->base;
36 struct imx_epit_data *driver_data = dev->data;
37
38 EPIT_ClearStatusFlag(base);
39
40 if (driver_data->callback != NULL) {
41 driver_data->callback(dev, (void *)driver_data->user_data);
42 }
43 }
44
imx_epit_init(const struct device * dev)45 static void imx_epit_init(const struct device *dev)
46 {
47 struct imx_epit_config *config = (struct imx_epit_config *)
48 get_epit_config(dev);
49 EPIT_Type *base = config->base;
50 epit_init_config_t epit_config = {
51 .freeRun = true,
52 .waitEnable = true,
53 .stopEnable = true,
54 .dbgEnable = true,
55 .enableMode = true
56 };
57
58 /* Adjust frequency in the counter configuration info */
59 config->info.freq = get_epit_clock_freq(base)/(config->prescaler + 1U);
60
61 EPIT_Init(base, &epit_config);
62 }
63
imx_epit_start(const struct device * dev)64 static int imx_epit_start(const struct device *dev)
65 {
66 EPIT_Type *base = get_epit_config(dev)->base;
67
68 /* Set EPIT clock source */
69 EPIT_SetClockSource(base, epitClockSourcePeriph);
70
71 /* Set prescaler */
72 EPIT_SetPrescaler(base, get_epit_config(dev)->prescaler);
73
74 /* Start the counter */
75 EPIT_Enable(base);
76
77 return 0;
78 }
79
imx_epit_stop(const struct device * dev)80 static int imx_epit_stop(const struct device *dev)
81 {
82 EPIT_Type *base = get_epit_config(dev)->base;
83
84 /* Disable EPIT */
85 EPIT_Disable(base);
86
87 return 0;
88 }
89
imx_epit_get_value(const struct device * dev,uint32_t * ticks)90 static int imx_epit_get_value(const struct device *dev, uint32_t *ticks)
91 {
92 EPIT_Type *base = get_epit_config(dev)->base;
93
94 *ticks = EPIT_GetCounterLoadValue(base) - EPIT_ReadCounter(base);
95
96 return 0;
97 }
98
imx_epit_set_top_value(const struct device * dev,const struct counter_top_cfg * cfg)99 static int imx_epit_set_top_value(const struct device *dev,
100 const struct counter_top_cfg *cfg)
101 {
102 EPIT_Type *base = get_epit_config(dev)->base;
103 struct imx_epit_data *driver_data = dev->data;
104
105 /* Disable EPIT Output Compare interrupt for consistency */
106 EPIT_SetIntCmd(base, false);
107
108 driver_data->callback = cfg->callback;
109 driver_data->user_data = cfg->user_data;
110
111 /* Set reload value and optionally counter to "ticks" */
112 EPIT_SetOverwriteCounter(base,
113 !(cfg->flags & COUNTER_TOP_CFG_DONT_RESET));
114 EPIT_SetCounterLoadValue(base, cfg->ticks);
115
116 if (cfg->callback != NULL) {
117 /* (Re)enable EPIT Output Compare interrupt */
118 EPIT_SetIntCmd(base, true);
119 }
120
121 return 0;
122 }
123
imx_epit_get_pending_int(const struct device * dev)124 static uint32_t imx_epit_get_pending_int(const struct device *dev)
125 {
126 EPIT_Type *base = get_epit_config(dev)->base;
127
128 return EPIT_GetStatusFlag(base) ? 1U : 0U;
129 }
130
imx_epit_get_top_value(const struct device * dev)131 static uint32_t imx_epit_get_top_value(const struct device *dev)
132 {
133 EPIT_Type *base = get_epit_config(dev)->base;
134
135 return EPIT_GetCounterLoadValue(base);
136 }
137
138 static DEVICE_API(counter, imx_epit_driver_api) = {
139 .start = imx_epit_start,
140 .stop = imx_epit_stop,
141 .get_value = imx_epit_get_value,
142 .set_top_value = imx_epit_set_top_value,
143 .get_pending_int = imx_epit_get_pending_int,
144 .get_top_value = imx_epit_get_top_value,
145 };
146
147 #define COUNTER_IMX_EPIT_DEVICE(idx) \
148 static int imx_epit_config_func_##idx(const struct device *dev); \
149 static const struct imx_epit_config imx_epit_##idx##z_config = { \
150 .info = { \
151 .max_top_value = COUNTER_MAX_RELOAD, \
152 .freq = 1U, \
153 .flags = 0, \
154 .channels = 0U, \
155 }, \
156 .base = (EPIT_Type *)DT_INST_REG_ADDR(idx), \
157 .prescaler = DT_INST_PROP(idx, prescaler), \
158 }; \
159 static struct imx_epit_data imx_epit_##idx##_data; \
160 DEVICE_DT_INST_DEFINE(idx, \
161 &imx_epit_config_func_##idx, \
162 NULL, \
163 &imx_epit_##idx##_data, &imx_epit_##idx##z_config.info, \
164 PRE_KERNEL_1, CONFIG_COUNTER_INIT_PRIORITY, \
165 &imx_epit_driver_api); \
166 static int imx_epit_config_func_##idx(const struct device *dev) \
167 { \
168 imx_epit_init(dev); \
169 IRQ_CONNECT(DT_INST_IRQN(idx), \
170 DT_INST_IRQ(idx, priority), \
171 imx_epit_isr, DEVICE_DT_INST_GET(idx), 0); \
172 irq_enable(DT_INST_IRQN(idx)); \
173 return 0; \
174 }
175
176 DT_INST_FOREACH_STATUS_OKAY(COUNTER_IMX_EPIT_DEVICE)
177