1 /* Copyright (c) 2023 Intel Corporation.
2  *
3  * SPDX-License-Identifier: Apache-2.0
4  */
5 
6 #define DT_DRV_COMPAT intel_sedi_gpio
7 
8 #include "sedi_driver_gpio.h"
9 #include <zephyr/drivers/gpio.h>
10 #include <zephyr/kernel.h>
11 #include <zephyr/drivers/gpio/gpio_utils.h>
12 #include <zephyr/pm/device.h>
13 
14 struct gpio_sedi_config {
15 	/* gpio_driver_data needs to be first */
16 	struct gpio_driver_config common;
17 	sedi_gpio_t device;
18 	uint32_t pin_nums;
19 	void (*irq_config)(void);
20 
21 	DEVICE_MMIO_ROM;
22 };
23 
24 struct gpio_sedi_data {
25 	/* gpio_driver_data needs to be first */
26 	struct gpio_driver_config common;
27 	sys_slist_t callbacks;
28 
29 	DEVICE_MMIO_RAM;
30 };
31 
32 static int gpio_sedi_init(const struct device *dev);
33 
34 #ifdef CONFIG_PM_DEVICE
gpio_sedi_suspend_device(const struct device * dev)35 static int gpio_sedi_suspend_device(const struct device *dev)
36 {
37 	const struct gpio_sedi_config *config = dev->config;
38 	sedi_gpio_t gpio_dev = config->device;
39 	int ret;
40 
41 	if (pm_device_is_busy(dev)) {
42 		return -EBUSY;
43 	}
44 
45 	ret = sedi_gpio_set_power(gpio_dev, SEDI_POWER_SUSPEND);
46 
47 	if (ret != SEDI_DRIVER_OK) {
48 		return -EIO;
49 	}
50 
51 	return 0;
52 }
53 
gpio_sedi_resume_device_from_suspend(const struct device * dev)54 static int gpio_sedi_resume_device_from_suspend(const struct device *dev)
55 {
56 	const struct gpio_sedi_config *config = dev->config;
57 	sedi_gpio_t gpio_dev = config->device;
58 	int ret;
59 
60 	ret = sedi_gpio_set_power(gpio_dev, SEDI_POWER_FULL);
61 	if (ret != SEDI_DRIVER_OK) {
62 		return -EIO;
63 	}
64 
65 	return 0;
66 }
67 
gpio_sedi_pm_action(const struct device * dev,enum pm_device_action action)68 static int gpio_sedi_pm_action(const struct device *dev,
69 			      enum pm_device_action action)
70 {
71 	int ret = 0;
72 
73 	switch (action) {
74 	case PM_DEVICE_ACTION_SUSPEND:
75 		ret = gpio_sedi_suspend_device(dev);
76 		break;
77 	case PM_DEVICE_ACTION_RESUME:
78 		ret = gpio_sedi_resume_device_from_suspend(dev);
79 		break;
80 
81 	default:
82 		ret = -ENOTSUP;
83 	}
84 
85 	return ret;
86 }
87 #endif /* CONFIG_PM_DEVICE */
88 
gpio_sedi_callback(const uint32_t pin_mask,const sedi_gpio_port_t port,void * param)89 static void gpio_sedi_callback(const uint32_t pin_mask,
90 			       const sedi_gpio_port_t port,
91 			       void *param)
92 {
93 	ARG_UNUSED(port);
94 	struct device *dev = (struct device *)param;
95 	struct gpio_sedi_data *data =
96 		(struct gpio_sedi_data *)(dev->data);
97 
98 	/* call the callbacks */
99 	gpio_fire_callbacks(&data->callbacks, dev, pin_mask);
100 }
101 
gpio_sedi_write_raw(const struct device * dev,uint32_t pins,bool is_clear)102 static void gpio_sedi_write_raw(const struct device *dev,
103 				uint32_t pins,
104 				bool is_clear)
105 {
106 	uint8_t i;
107 	const struct gpio_sedi_config *config = dev->config;
108 	sedi_gpio_t gpio_dev = config->device;
109 	sedi_gpio_pin_state_t val;
110 
111 	if (is_clear) {
112 		val = SEDI_GPIO_STATE_LOW;
113 	} else {
114 		val = SEDI_GPIO_STATE_HIGH;
115 	}
116 
117 	for (i = 0; i < config->pin_nums; i++) {
118 		if (pins & 0x1) {
119 			sedi_gpio_write_pin(gpio_dev, i, val);
120 		}
121 		pins >>= 1;
122 		if (pins == 0) {
123 			break;
124 		}
125 	}
126 }
127 
gpio_sedi_configure(const struct device * dev,gpio_pin_t pin,gpio_flags_t flags)128 static int gpio_sedi_configure(const struct device *dev, gpio_pin_t pin,
129 			       gpio_flags_t flags)
130 {
131 	const struct gpio_sedi_config *config = dev->config;
132 	sedi_gpio_t gpio_dev = config->device;
133 	sedi_gpio_pin_config_t pin_config = { 0 };
134 
135 	if ((flags & GPIO_OUTPUT) && (flags & GPIO_INPUT)) {
136 		/* Pin cannot be configured as input and output */
137 		return -ENOTSUP;
138 	} else if (!(flags & (GPIO_INPUT | GPIO_OUTPUT))) {
139 		/* Pin has to be configured as input or output */
140 		return -ENOTSUP;
141 	}
142 
143 	pin_config.enable_interrupt = false;
144 	/* Map direction */
145 	if (flags & GPIO_OUTPUT) {
146 		pin_config.direction = SEDI_GPIO_DIR_MODE_OUTPUT;
147 		sedi_gpio_config_pin(gpio_dev, pin, pin_config);
148 		/* Set start state */
149 		if ((flags & GPIO_OUTPUT_INIT_HIGH) != 0) {
150 			sedi_gpio_write_pin(gpio_dev, pin, 1);
151 		} else if ((flags & GPIO_OUTPUT_INIT_LOW) != 0) {
152 			sedi_gpio_write_pin(gpio_dev, pin, 0);
153 		}
154 	} else {
155 		pin_config.direction = SEDI_GPIO_DIR_MODE_INPUT;
156 		sedi_gpio_config_pin(gpio_dev, pin, pin_config);
157 	}
158 
159 	return 0;
160 }
161 
gpio_sedi_get_raw(const struct device * dev,uint32_t * value)162 static int gpio_sedi_get_raw(const struct device *dev, uint32_t *value)
163 {
164 	const struct gpio_sedi_config *config = dev->config;
165 	sedi_gpio_t gpio_dev = config->device;
166 
167 	*value = sedi_gpio_read_pin_32bits(gpio_dev, 0);
168 
169 	return 0;
170 }
171 
gpio_sedi_set_masked_raw(const struct device * dev,uint32_t mask,uint32_t value)172 static int gpio_sedi_set_masked_raw(const struct device *dev,
173 				    uint32_t mask,
174 				    uint32_t value)
175 {
176 	gpio_sedi_write_raw(dev, (mask & value), false);
177 
178 	return 0;
179 }
180 
gpio_sedi_set_bits_raw(const struct device * dev,uint32_t pins)181 static int gpio_sedi_set_bits_raw(const struct device *dev, uint32_t pins)
182 {
183 	gpio_sedi_write_raw(dev, pins, false);
184 
185 	return 0;
186 }
187 
gpio_sedi_clear_bits_raw(const struct device * dev,uint32_t pins)188 static int gpio_sedi_clear_bits_raw(const struct device *dev, uint32_t pins)
189 {
190 	gpio_sedi_write_raw(dev, pins, true);
191 
192 	return 0;
193 }
194 
gpio_sedi_toggle_bits(const struct device * dev,uint32_t pins)195 static int gpio_sedi_toggle_bits(const struct device *dev, uint32_t pins)
196 {
197 	const struct gpio_sedi_config *config = dev->config;
198 	sedi_gpio_t gpio_dev = config->device;
199 	uint8_t i;
200 
201 	for (i = 0; i < config->pin_nums; i++) {
202 		if (pins & 0x1) {
203 			sedi_gpio_toggle_pin(gpio_dev, i);
204 		}
205 		pins >>= 1;
206 		if (pins == 0) {
207 			break;
208 		}
209 	}
210 
211 	return 0;
212 }
213 
gpio_sedi_interrupt_configure(const struct device * dev,gpio_pin_t pin,enum gpio_int_mode mode,enum gpio_int_trig trig)214 static int gpio_sedi_interrupt_configure(const struct device *dev,
215 					 gpio_pin_t pin,
216 					 enum gpio_int_mode mode,
217 					 enum gpio_int_trig trig)
218 {
219 	const struct gpio_sedi_config *config = dev->config;
220 	sedi_gpio_t gpio_dev = config->device;
221 	sedi_gpio_pin_config_t pin_config = { 0 };
222 
223 	/* Not support level trigger */
224 	if (mode == GPIO_INT_MODE_LEVEL) {
225 		return -EINVAL;
226 	}
227 	/* Only input needs interrupt enabled */
228 	pin_config.direction = SEDI_GPIO_DIR_MODE_INPUT;
229 	pin_config.enable_wakeup = true;
230 	if (mode == GPIO_INT_MODE_DISABLED) {
231 		pin_config.enable_interrupt = false;
232 	} else {
233 		pin_config.enable_interrupt = true;
234 		switch (trig) {
235 		case GPIO_INT_TRIG_LOW:
236 			pin_config.interrupt_mode =
237 				SEDI_GPIO_INT_MODE_FALLING_EDGE;
238 			break;
239 		case GPIO_INT_TRIG_HIGH:
240 			pin_config.interrupt_mode =
241 				SEDI_GPIO_INT_MODE_RISING_EDGE;
242 			break;
243 		case GPIO_INT_TRIG_BOTH:
244 			pin_config.interrupt_mode =
245 				SEDI_GPIO_INT_MODE_BOTH_EDGE;
246 			break;
247 		default:
248 			return -EINVAL;
249 		}
250 	}
251 	/* Configure interrupt mode */
252 	sedi_gpio_config_pin(gpio_dev, pin, pin_config);
253 
254 	return 0;
255 }
256 
gpio_sedi_manage_callback(const struct device * dev,struct gpio_callback * callback,bool set)257 static int gpio_sedi_manage_callback(const struct device *dev,
258 				     struct gpio_callback *callback,
259 				     bool set)
260 {
261 	struct gpio_sedi_data *data = dev->data;
262 
263 	gpio_manage_callback(&(data->callbacks), callback, set);
264 
265 	return 0;
266 }
267 
gpio_sedi_get_pending(const struct device * dev)268 static uint32_t gpio_sedi_get_pending(const struct device *dev)
269 {
270 	const struct gpio_sedi_config *config = dev->config;
271 	sedi_gpio_t gpio_dev = config->device;
272 
273 	return sedi_gpio_get_gisr(gpio_dev, 0);
274 }
275 
276 static DEVICE_API(gpio, gpio_sedi_driver_api) = {
277 	.pin_configure = gpio_sedi_configure,
278 	.port_get_raw = gpio_sedi_get_raw,
279 	.port_set_masked_raw = gpio_sedi_set_masked_raw,
280 	.port_set_bits_raw = gpio_sedi_set_bits_raw,
281 	.port_clear_bits_raw = gpio_sedi_clear_bits_raw,
282 	.port_toggle_bits = gpio_sedi_toggle_bits,
283 	.pin_interrupt_configure = gpio_sedi_interrupt_configure,
284 	.manage_callback = gpio_sedi_manage_callback,
285 	.get_pending_int = gpio_sedi_get_pending
286 };
287 
288 extern void gpio_isr(IN sedi_gpio_t gpio_device);
289 
gpio_sedi_init(const struct device * dev)290 static int gpio_sedi_init(const struct device *dev)
291 {
292 	int ret;
293 	const struct gpio_sedi_config *config = dev->config;
294 	sedi_gpio_t gpio_dev = config->device;
295 
296 	DEVICE_MMIO_MAP(dev, K_MEM_CACHE_NONE);
297 
298 	/* Call sedi gpio init */
299 	ret = sedi_gpio_init(gpio_dev, gpio_sedi_callback, (void *)dev);
300 
301 	if (ret != 0) {
302 		return ret;
303 	}
304 	sedi_gpio_set_power(gpio_dev, SEDI_POWER_FULL);
305 
306 	config->irq_config();
307 
308 	return 0;
309 }
310 
311 #define GPIO_SEDI_IRQ_FLAGS_SENSE0(n) 0
312 #define GPIO_SEDI_IRQ_FLAGS_SENSE1(n) DT_INST_IRQ(n, sense)
313 #define GPIO_SEDI_IRQ_FLAGS(n) \
314 	_CONCAT(GPIO_SEDI_IRQ_FLAGS_SENSE, DT_INST_IRQ_HAS_CELL(n, sense))(n)
315 
316 #define GPIO_DEVICE_INIT_SEDI(n)				       \
317 	static struct gpio_sedi_data gpio##n##_data;	               \
318 	static void gpio_sedi_irq_config_##n(void)		       \
319 	{							       \
320 		IRQ_CONNECT(DT_INST_IRQN(n), DT_INST_IRQ(n, priority), \
321 			    gpio_isr, n,			       \
322 			    GPIO_SEDI_IRQ_FLAGS(n));		       \
323 		irq_enable(DT_INST_IRQN(n));			       \
324 	};							       \
325 	static const struct gpio_sedi_config gpio##n##_config = {      \
326 		DEVICE_MMIO_ROM_INIT(DT_DRV_INST(n)),                  \
327 		.common = { 0xFFFFFFFF },			       \
328 		.device = DT_INST_PROP(n, peripheral_id),              \
329 		.pin_nums = DT_INST_PROP(n, ngpios),                   \
330 		.irq_config = gpio_sedi_irq_config_##n,	               \
331 	};							       \
332 	PM_DEVICE_DEFINE(gpio_##n, gpio_sedi_pm_action);               \
333 	DEVICE_DT_INST_DEFINE(n,				       \
334 		      gpio_sedi_init,				       \
335 		      PM_DEVICE_GET(gpio_##n),		               \
336 		      &gpio##n##_data,			               \
337 		      &gpio##n##_config,			       \
338 		      POST_KERNEL,				       \
339 		      CONFIG_GPIO_INIT_PRIORITY,	               \
340 		      &gpio_sedi_driver_api);
341 
342 DT_INST_FOREACH_STATUS_OKAY(GPIO_DEVICE_INIT_SEDI)
343