1 /*
2  * Copyright (c) 2022 Microchip Technology Inc.
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #define DT_DRV_COMPAT microchip_mpfs_gpio
8 
9 #include <errno.h>
10 #include <zephyr/kernel.h>
11 #include <zephyr/device.h>
12 #include <zephyr/drivers/gpio.h>
13 #include <zephyr/sys/util.h>
14 #include <zephyr/irq.h>
15 
16 #include <zephyr/drivers/gpio/gpio_utils.h>
17 
18 
19 #define MSS_GPIO_INPUT_MODE               0x02
20 #define MSS_GPIO_OUTPUT_MODE              0x05
21 #define MSS_GPIO_INOUT_MODE               0x03
22 #define MSS_GPIO_IRQ_LEVEL_HIGH           0x00
23 #define MSS_GPIO_IRQ_LEVEL_LOW            0x20
24 #define MSS_GPIO_IRQ_EDGE_POSITIVE        0x40
25 #define MSS_GPIO_IRQ_EDGE_NEGATIVE        0x60
26 #define MSS_GPIO_IRQ_EDGE_BOTH            0x80
27 #define MSS_GPIO_INT_ENABLE_MASK          0x08
28 #define MSS_OUTPUT_BUFFER_ENABLE_MASK     0x04
29 
30 struct mss_gpio_t {
31 	uint32_t gpio_cfg[32];
32 	uint32_t gpio_irq;
33 	const uint32_t gpio_in;
34 	uint32_t gpio_out;
35 	uint32_t gpio_cfg_all;
36 	uint32_t gpio_cfg_byte[4];
37 	uint32_t gpio_clr_bits;
38 	uint32_t gpio_set_bits;
39 
40 };
41 
42 typedef void (*mss_gpio_cfg_func_t)(void);
43 
44 
45 struct mss_gpio_config {
46 	/* gpio_driver_config needs to be first */
47 	struct gpio_driver_config common;
48 	uintptr_t            gpio_base_addr;
49 	uint32_t                gpio_irq_base;
50 	mss_gpio_cfg_func_t    gpio_cfg_func;
51 };
52 
53 struct mss_gpio_data {
54 	/* gpio_driver_data needs to be first */
55 	struct gpio_driver_data common;
56 	/* list of callbacks */
57 	sys_slist_t cb;
58 };
59 
60 
61 /* Helper Macros for GPIO */
62 #define DEV_GPIO_CFG(dev)	\
63 	((const struct mss_gpio_config * const)(dev)->config)
64 #define DEV_GPIO(dev)	\
65 	((volatile struct mss_gpio_t *)(DEV_GPIO_CFG(dev))->gpio_base_addr)
66 #define DEV_GPIO_DATA(dev)	\
67 	((struct mss_gpio_data *)(dev)->data)
68 
69 
mss_gpio_config(const struct device * dev,gpio_pin_t pin,gpio_flags_t flags)70 static int mss_gpio_config(const struct device *dev,
71 					gpio_pin_t pin,
72 					gpio_flags_t flags)
73 {
74 	volatile struct mss_gpio_t *gpio = DEV_GPIO(dev);
75 	uint32_t io_flags = flags & (GPIO_INPUT | GPIO_OUTPUT);
76 
77 	if (io_flags == GPIO_DISCONNECTED) {
78 
79 		return -ENOTSUP;
80 	}
81 
82 	if (flags & GPIO_OUTPUT) {
83 
84 		gpio->gpio_cfg[pin] |= MSS_GPIO_OUTPUT_MODE;
85 
86 		if (flags & GPIO_OUTPUT_INIT_HIGH) {
87 			gpio->gpio_out |= BIT(pin);
88 
89 		} else if (flags & GPIO_OUTPUT_INIT_LOW) {
90 			gpio->gpio_out &= ~BIT(pin);
91 
92 		}
93 
94 	} else if (flags & GPIO_INPUT) {
95 		gpio->gpio_cfg[pin] |= MSS_GPIO_INPUT_MODE;
96 
97 	} else {
98 		return -ENOTSUP;
99 	}
100 
101 	return 0;
102 }
103 
mss_gpio_port_toggle_bits(const struct device * dev,gpio_port_pins_t mask)104 static int mss_gpio_port_toggle_bits(const struct device *dev,
105 					gpio_port_pins_t mask)
106 {
107 	volatile struct mss_gpio_t *gpio = DEV_GPIO(dev);
108 
109 	gpio->gpio_out ^= mask;
110 
111 	return 0;
112 }
113 
mss_gpio_port_get_raw(const struct device * dev,gpio_port_value_t * value)114 static int mss_gpio_port_get_raw(const struct device *dev,
115 					gpio_port_value_t *value)
116 {
117 	volatile struct mss_gpio_t *gpio = DEV_GPIO(dev);
118 
119 	*value = gpio->gpio_in;
120 
121 	return 0;
122 }
123 
mss_gpio_port_set_masked_raw(const struct device * dev,gpio_port_pins_t mask,gpio_port_value_t value)124 static int mss_gpio_port_set_masked_raw(const struct device *dev,
125 					   gpio_port_pins_t mask,
126 					   gpio_port_value_t value)
127 {
128 	volatile struct mss_gpio_t *gpio = DEV_GPIO(dev);
129 
130 	gpio->gpio_out = (gpio->gpio_out & ~mask) | (value & mask);
131 
132 	return 0;
133 }
134 
mss_gpio_port_set_bits_raw(const struct device * dev,gpio_port_pins_t mask)135 static int mss_gpio_port_set_bits_raw(const struct device *dev,
136 					 gpio_port_pins_t mask)
137 {
138 	volatile struct mss_gpio_t *gpio = DEV_GPIO(dev);
139 
140 	gpio->gpio_out |= mask;
141 
142 	return 0;
143 }
144 
mss_gpio_port_clear_bits_raw(const struct device * dev,gpio_port_pins_t mask)145 static int mss_gpio_port_clear_bits_raw(const struct device *dev,
146 					   gpio_port_pins_t mask)
147 {
148 	volatile struct mss_gpio_t *gpio = DEV_GPIO(dev);
149 
150 	gpio->gpio_out &= ~mask;
151 
152 	return 0;
153 }
154 
mss_gpio_pin_interrupt_configure(const struct device * dev,gpio_pin_t pin,enum gpio_int_mode mode,enum gpio_int_trig trig)155 static int mss_gpio_pin_interrupt_configure(const struct device *dev,
156 						 gpio_pin_t pin,
157 						 enum gpio_int_mode mode,
158 						 enum gpio_int_trig trig)
159 {
160 	ARG_UNUSED(trig);
161 	volatile struct mss_gpio_t *gpio = DEV_GPIO(dev);
162 
163 	gpio->gpio_cfg[pin] |= (MSS_GPIO_INT_ENABLE_MASK);
164 
165 	switch (mode | trig) {
166 	case GPIO_INT_EDGE_BOTH:
167 		gpio->gpio_cfg[pin] |= (MSS_GPIO_IRQ_EDGE_BOTH);
168 		break;
169 	case GPIO_INT_EDGE_RISING:
170 		gpio->gpio_cfg[pin] |= (MSS_GPIO_IRQ_EDGE_POSITIVE);
171 		break;
172 	case GPIO_INT_EDGE_FALLING:
173 		gpio->gpio_cfg[pin] |= (MSS_GPIO_IRQ_EDGE_NEGATIVE);
174 		break;
175 	case GPIO_INT_LEVEL_LOW:
176 		gpio->gpio_cfg[pin] |= (MSS_GPIO_IRQ_LEVEL_LOW);
177 		break;
178 	case GPIO_INT_LEVEL_HIGH:
179 		gpio->gpio_cfg[pin] |= (MSS_GPIO_IRQ_LEVEL_HIGH);
180 		break;
181 	default:
182 		gpio->gpio_cfg[pin] &= ~MSS_GPIO_INT_ENABLE_MASK;
183 		break;
184 	}
185 	return 0;
186 }
187 
188 
mss_gpio_manage_callback(const struct device * dev,struct gpio_callback * callback,bool set)189 static int mss_gpio_manage_callback(const struct device *dev,
190 					   struct gpio_callback *callback,
191 					   bool set)
192 {
193 	struct mss_gpio_data *data = dev->data;
194 
195 	return gpio_manage_callback(&data->cb, callback, set);
196 }
197 static DEVICE_API(gpio, mss_gpio_driver) = {
198 	.pin_configure           = mss_gpio_config,
199 	.port_toggle_bits        = mss_gpio_port_toggle_bits,
200 	.port_get_raw            = mss_gpio_port_get_raw,
201 	.port_set_masked_raw     = mss_gpio_port_set_masked_raw,
202 	.port_set_bits_raw       = mss_gpio_port_set_bits_raw,
203 	.port_clear_bits_raw     = mss_gpio_port_clear_bits_raw,
204 	.pin_interrupt_configure = mss_gpio_pin_interrupt_configure,
205 	.manage_callback         = mss_gpio_manage_callback
206 };
207 
208 
mss_gpio_init(const struct device * dev)209 static int mss_gpio_init(const struct device *dev)
210 {
211 	volatile struct mss_gpio_t *gpio = DEV_GPIO(dev);
212 
213 	gpio->gpio_irq = 0xFFFFFFFFU;
214 
215 	const struct mss_gpio_config *cfg = DEV_GPIO_CFG(dev);
216 	/* Configure GPIO device */
217 	cfg->gpio_cfg_func();
218 	return 0;
219 }
220 
mss_gpio_irq_handler(const struct device * dev)221 static void mss_gpio_irq_handler(const struct device *dev)
222 {
223 	volatile struct mss_gpio_t *gpio = DEV_GPIO(dev);
224 	uint32_t interrupt_status = gpio->gpio_irq;
225 
226 	gpio->gpio_irq = gpio->gpio_irq;
227 	struct mss_gpio_data *data = dev->data;
228 
229 	gpio_fire_callbacks(&data->cb, dev, interrupt_status);
230 }
231 
232 #define MSS_GPIO_INIT(n)	\
233 	static struct mss_gpio_data mss_gpio_data_##n;	\
234 	static void gpio_mss_gpio_cfg_func_##n(void);	\
235 	\
236 	static const struct mss_gpio_config mss_gpio_config_##n = {	\
237 		.common = {	\
238 			.port_pin_mask = GPIO_PORT_PIN_MASK_FROM_DT_INST(n),	\
239 		},	\
240 		.gpio_base_addr = DT_INST_REG_ADDR(n),	\
241 		.gpio_irq_base  = DT_INST_IRQN(n),	\
242 		.gpio_cfg_func = gpio_mss_gpio_cfg_func_##n	\
243 	};	\
244 	\
245 	DEVICE_DT_INST_DEFINE(n,	\
246 				mss_gpio_init,	\
247 				NULL,	\
248 				&mss_gpio_data_##n, &mss_gpio_config_##n,	\
249 				PRE_KERNEL_1, CONFIG_GPIO_INIT_PRIORITY,	\
250 				&mss_gpio_driver);	\
251 	\
252 	static void gpio_mss_gpio_cfg_func_##n(void)	\
253 	{	\
254 		IRQ_CONNECT(DT_INST_IRQN(n),	\
255 				DT_INST_IRQ(n, priority),	\
256 				mss_gpio_irq_handler,	\
257 				DEVICE_DT_INST_GET(n),	\
258 				0);	\
259 		irq_enable(DT_INST_IRQN(n));	\
260 	}	\
261 
262 DT_INST_FOREACH_STATUS_OKAY(MSS_GPIO_INIT)
263