1 /*
2  * Copyright (c) 2018 Justin Watson
3  * Copyright (c) 2020-2023 Gerson Fernando Budke <nandojve@gmail.com>
4  *
5  * SPDX-License-Identifier: Apache-2.0
6  */
7 
8 #define DT_DRV_COMPAT atmel_sam4l_gpio
9 
10 #include <errno.h>
11 #include <zephyr/kernel.h>
12 #include <zephyr/device.h>
13 #include <zephyr/init.h>
14 #include <soc.h>
15 #include <zephyr/drivers/gpio.h>
16 #include <zephyr/drivers/clock_control/atmel_sam_pmc.h>
17 #include <zephyr/irq.h>
18 
19 #include <zephyr/drivers/gpio/gpio_utils.h>
20 
21 typedef void (*config_func_t)(const struct device *dev);
22 
23 struct gpio_sam_config {
24 	/* gpio_driver_config needs to be first */
25 	struct gpio_driver_config common;
26 	Gpio *regs;
27 	config_func_t config_func;
28 
29 	const struct atmel_sam_pmc_config clock_cfg;
30 };
31 
32 struct gpio_sam_runtime {
33 	/* gpio_driver_data needs to be first */
34 	struct gpio_driver_data common;
35 	sys_slist_t cb;
36 };
37 
38 #define GPIO_SAM_ALL_PINS    0xFFFFFFFF
39 
gpio_sam_port_configure(const struct device * dev,uint32_t mask,gpio_flags_t flags)40 static int gpio_sam_port_configure(const struct device *dev,
41 				   uint32_t mask,
42 				   gpio_flags_t flags)
43 {
44 	const struct gpio_sam_config * const cfg = dev->config;
45 	Gpio * const gpio = cfg->regs;
46 
47 	/* No hardware support */
48 	if (flags & GPIO_SINGLE_ENDED) {
49 		return -ENOTSUP;
50 	}
51 
52 	if (!(flags & (GPIO_OUTPUT | GPIO_INPUT))) {
53 		gpio->IERC = mask;
54 		gpio->PUERC = mask;
55 		gpio->PDERC = mask;
56 		gpio->GPERS = mask;
57 		gpio->ODERC = mask;
58 		gpio->STERC = mask;
59 
60 		return 0;
61 	}
62 
63 	/*
64 	 * Always enable schmitt-trigger because SAM4L GPIO Ctrl
65 	 * is Input only or Input/Output.
66 	 */
67 	gpio->STERS = mask;
68 
69 	if (flags & GPIO_OUTPUT) {
70 		if (flags & GPIO_OUTPUT_INIT_HIGH) {
71 			gpio->OVRS = mask;
72 		}
73 		if (flags & GPIO_OUTPUT_INIT_LOW) {
74 			gpio->OVRC = mask;
75 		}
76 		gpio->ODERS = mask;
77 	} else {
78 		gpio->ODERC = mask;
79 	}
80 
81 	gpio->PUERC = mask;
82 	gpio->PDERC = mask;
83 	if (flags & GPIO_PULL_UP) {
84 		gpio->PUERS = mask;
85 	} else if (flags & GPIO_PULL_DOWN) {
86 		gpio->PDERS = mask;
87 	}
88 
89 	/* Enable the GPIO to control the pin (instead of a peripheral). */
90 	gpio->GPERS = mask;
91 
92 	return 0;
93 }
94 
gpio_sam_config(const struct device * dev,gpio_pin_t pin,gpio_flags_t flags)95 static int gpio_sam_config(const struct device *dev,
96 			   gpio_pin_t pin,
97 			   gpio_flags_t flags)
98 {
99 	return gpio_sam_port_configure(dev, BIT(pin), flags);
100 }
101 
gpio_sam_port_get_raw(const struct device * dev,uint32_t * value)102 static int gpio_sam_port_get_raw(const struct device *dev,
103 				 uint32_t *value)
104 {
105 	const struct gpio_sam_config * const cfg = dev->config;
106 	Gpio * const gpio = cfg->regs;
107 
108 	*value = gpio->PVR;
109 
110 	return 0;
111 }
112 
gpio_sam_port_set_masked_raw(const struct device * dev,uint32_t mask,uint32_t value)113 static int gpio_sam_port_set_masked_raw(const struct device *dev,
114 					uint32_t mask,
115 					uint32_t value)
116 {
117 	const struct gpio_sam_config * const cfg = dev->config;
118 	Gpio * const gpio = cfg->regs;
119 
120 	gpio->OVR = (gpio->PVR & ~mask) | (mask & value);
121 
122 	return 0;
123 }
124 
gpio_sam_port_set_bits_raw(const struct device * dev,uint32_t mask)125 static int gpio_sam_port_set_bits_raw(const struct device *dev,
126 				      uint32_t mask)
127 {
128 	const struct gpio_sam_config * const cfg = dev->config;
129 	Gpio * const gpio = cfg->regs;
130 
131 	gpio->OVRS = mask;
132 
133 	return 0;
134 }
135 
gpio_sam_port_clear_bits_raw(const struct device * dev,uint32_t mask)136 static int gpio_sam_port_clear_bits_raw(const struct device *dev,
137 					uint32_t mask)
138 {
139 	const struct gpio_sam_config * const cfg = dev->config;
140 	Gpio * const gpio = cfg->regs;
141 
142 	gpio->OVRC = mask;
143 
144 	return 0;
145 }
146 
gpio_sam_port_toggle_bits(const struct device * dev,uint32_t mask)147 static int gpio_sam_port_toggle_bits(const struct device *dev,
148 				     uint32_t mask)
149 {
150 	const struct gpio_sam_config * const cfg = dev->config;
151 	Gpio * const gpio = cfg->regs;
152 
153 	gpio->OVRT = mask;
154 
155 	return 0;
156 }
157 
gpio_sam_port_interrupt_configure(const struct device * dev,uint32_t mask,enum gpio_int_mode mode,enum gpio_int_trig trig)158 static int gpio_sam_port_interrupt_configure(const struct device *dev,
159 					     uint32_t mask,
160 					     enum gpio_int_mode mode,
161 					     enum gpio_int_trig trig)
162 {
163 	const struct gpio_sam_config * const cfg = dev->config;
164 	Gpio * const gpio = cfg->regs;
165 
166 	if (mode == GPIO_INT_MODE_LEVEL) {
167 		return -ENOTSUP;
168 	}
169 
170 	gpio->IERC = mask;
171 	gpio->IMR0C = mask;
172 	gpio->IMR1C = mask;
173 
174 	if (trig != GPIO_INT_TRIG_BOTH) {
175 		if (trig == GPIO_INT_TRIG_HIGH) {
176 			gpio->IMR0S = mask;
177 		} else {
178 			gpio->IMR1S = mask;
179 		}
180 	}
181 
182 	if (mode != GPIO_INT_MODE_DISABLED) {
183 		gpio->IFRC = mask;
184 		gpio->IERS = mask;
185 	}
186 
187 	return 0;
188 }
189 
gpio_sam_pin_interrupt_configure(const struct device * dev,gpio_pin_t pin,enum gpio_int_mode mode,enum gpio_int_trig trig)190 static int gpio_sam_pin_interrupt_configure(const struct device *dev,
191 					    gpio_pin_t pin,
192 					    enum gpio_int_mode mode,
193 					    enum gpio_int_trig trig)
194 {
195 	return gpio_sam_port_interrupt_configure(dev, BIT(pin), mode, trig);
196 }
197 
gpio_sam_isr(const struct device * dev)198 static void gpio_sam_isr(const struct device *dev)
199 {
200 	const struct gpio_sam_config * const cfg = dev->config;
201 	Gpio * const gpio = cfg->regs;
202 	struct gpio_sam_runtime *context = dev->data;
203 	uint32_t int_stat;
204 
205 	int_stat = gpio->IFR;
206 	gpio->IFRC = int_stat;
207 
208 	gpio_fire_callbacks(&context->cb, dev, int_stat);
209 }
210 
gpio_sam_manage_callback(const struct device * port,struct gpio_callback * callback,bool set)211 static int gpio_sam_manage_callback(const struct device *port,
212 				    struct gpio_callback *callback,
213 				    bool set)
214 {
215 	struct gpio_sam_runtime *context = port->data;
216 
217 	return gpio_manage_callback(&context->cb, callback, set);
218 }
219 
220 static DEVICE_API(gpio, gpio_sam_api) = {
221 	.pin_configure = gpio_sam_config,
222 	.port_get_raw = gpio_sam_port_get_raw,
223 	.port_set_masked_raw = gpio_sam_port_set_masked_raw,
224 	.port_set_bits_raw = gpio_sam_port_set_bits_raw,
225 	.port_clear_bits_raw = gpio_sam_port_clear_bits_raw,
226 	.port_toggle_bits = gpio_sam_port_toggle_bits,
227 	.pin_interrupt_configure = gpio_sam_pin_interrupt_configure,
228 	.manage_callback = gpio_sam_manage_callback,
229 };
230 
gpio_sam_init(const struct device * dev)231 int gpio_sam_init(const struct device *dev)
232 {
233 	const struct gpio_sam_config * const cfg = dev->config;
234 
235 	/* Enable GPIO clock in PM */
236 	(void)clock_control_on(SAM_DT_PMC_CONTROLLER,
237 			       (clock_control_subsys_t)&cfg->clock_cfg);
238 
239 	cfg->config_func(dev);
240 
241 	return 0;
242 }
243 
244 #define GPIO_SAM_IRQ_CONNECT(n, m)					\
245 	do {								\
246 		IRQ_CONNECT(DT_INST_IRQ_BY_IDX(n, m, irq),		\
247 			    DT_INST_IRQ_BY_IDX(n, m, priority),		\
248 			    gpio_sam_isr,				\
249 			    DEVICE_DT_INST_GET(n), 0);			\
250 		irq_enable(DT_INST_IRQ_BY_IDX(n, m, irq));		\
251 	} while (false)
252 
253 #define GPIO_SAM_INIT(n)						\
254 	static void port_##n##_sam_config_func(const struct device *dev);\
255 									\
256 	static const struct gpio_sam_config port_##n##_sam_config = {	\
257 		.common = {						\
258 			.port_pin_mask = GPIO_PORT_PIN_MASK_FROM_DT_INST(n),\
259 		},							\
260 		.regs = (Gpio *)DT_INST_REG_ADDR(n),			\
261 		.clock_cfg = SAM_DT_INST_CLOCK_PMC_CFG(n),		\
262 		.config_func = port_##n##_sam_config_func,		\
263 	};								\
264 									\
265 	static struct gpio_sam_runtime port_##n##_sam_runtime;		\
266 									\
267 	DEVICE_DT_INST_DEFINE(n, gpio_sam_init, NULL,			\
268 			    &port_##n##_sam_runtime,			\
269 			    &port_##n##_sam_config, PRE_KERNEL_1,	\
270 			    CONFIG_GPIO_INIT_PRIORITY,			\
271 			    &gpio_sam_api);				\
272 									\
273 	static void port_##n##_sam_config_func(const struct device *dev)\
274 	{								\
275 		GPIO_SAM_IRQ_CONNECT(n, 0);				\
276 		GPIO_SAM_IRQ_CONNECT(n, 1);				\
277 		GPIO_SAM_IRQ_CONNECT(n, 2);				\
278 		GPIO_SAM_IRQ_CONNECT(n, 3);				\
279 	}
280 
281 DT_INST_FOREACH_STATUS_OKAY(GPIO_SAM_INIT)
282