1 /*
2 * Copyright (c) 2020 Friedt Professional Engineering Services, Inc
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #define DT_DRV_COMPAT zephyr_gpio_emul
8
9 #include <zephyr/device.h>
10 #include <zephyr/drivers/gpio.h>
11 #include <zephyr/drivers/gpio/gpio_emul.h>
12 #include <errno.h>
13 #include <zephyr/kernel.h>
14 #include <zephyr/pm/device.h>
15
16 #include <zephyr/drivers/gpio/gpio_utils.h>
17
18 #define LOG_LEVEL CONFIG_GPIO_LOG_LEVEL
19 #include <zephyr/logging/log.h>
20 LOG_MODULE_REGISTER(gpio_emul);
21
22 #define GPIO_EMUL_INT_BITMASK \
23 (GPIO_INT_DISABLE | GPIO_INT_ENABLE | GPIO_INT_LEVELS_LOGICAL | \
24 GPIO_INT_EDGE | GPIO_INT_LOW_0 | GPIO_INT_HIGH_1)
25
26 /**
27 * @brief GPIO Emulator interrupt capabilities
28 *
29 * These enumerations are used as a bitmask and allow the GPIO Emulator to
30 * model GPIO interrupt controllers with varying interrupt trigger support.
31 *
32 * For example, some controllers to not support level interrupts,
33 * some controllers do not support rising and falling edge simultaneously,
34 * etc.
35 *
36 * This primarily affects the behaviour of @ref gpio_pin_interrupt_configure.
37 */
38 enum gpio_emul_interrupt_cap {
39 GPIO_EMUL_INT_CAP_EDGE_RISING = 1,
40 GPIO_EMUL_INT_CAP_EDGE_FALLING = 2,
41 GPIO_EMUL_INT_CAP_LEVEL_HIGH = 16,
42 GPIO_EMUL_INT_CAP_LEVEL_LOW = 32,
43 };
44
45 /**
46 * @brief Emulated GPIO controller configuration data
47 *
48 * This structure contains all of the state for a given emulated GPIO
49 * controller as well as all of the pins associated with it.
50 *
51 * The @a flags member is a pointer to an array which is @a num_pins in size.
52 *
53 * @a num_pins must be in the range [1, @ref GPIO_MAX_PINS_PER_PORT].
54 *
55 * Pin direction as well as other pin properties are set using
56 * specific bits in @a flags. For more details, see @ref gpio_interface.
57 *
58 * Changes are synchronized using @ref gpio_emul_data.mu.
59 */
60 struct gpio_emul_config {
61 /** Common @ref gpio_driver_config */
62 const struct gpio_driver_config common;
63 /** Number of pins available in the given GPIO controller instance */
64 const gpio_pin_t num_pins;
65 /** Supported interrupts */
66 const enum gpio_emul_interrupt_cap interrupt_caps;
67 };
68
69 /**
70 * @brief Emulated GPIO controller data
71 *
72 * This structure contains data structures used by a emulated GPIO
73 * controller.
74 *
75 * If the application wishes to specify a "wiring" for the emulated
76 * GPIO, then a @ref gpio_callback_handler_t should be registered using
77 * @ref gpio_add_callback.
78 *
79 * Changes are to @ref gpio_emul_data and @ref gpio_emul_config are
80 * synchronized using @a k_spinlock.
81 */
82 struct gpio_emul_data {
83 /** Common @ref gpio_driver_data */
84 struct gpio_driver_data common;
85 /** Pointer to an array of flags is @a num_pins in size */
86 gpio_flags_t *flags;
87 /** Input values for each pin */
88 gpio_port_value_t input_vals;
89 /** Output values for each pin */
90 gpio_port_value_t output_vals;
91 /** Interrupt status for each pin */
92 gpio_port_pins_t interrupts;
93 /** Spinlock to synchronize accesses to driver data and config */
94 struct k_spinlock lock;
95 /** Is interrupt enabled for each pin */
96 gpio_port_pins_t enabled_interrupts;
97 /** Singly-linked list of callbacks associated with the controller */
98 sys_slist_t callbacks;
99 };
100
101 /**
102 * @brief Obtain a mask of pins that match all of the provided @p flags
103 *
104 * Use this function to see which pins match the current GPIO configuration.
105 *
106 * The caller must ensure that @ref gpio_emul_data.lock is locked.
107 *
108 * @param port The emulated GPIO device pointer
109 * @param mask A mask of flags to match
110 * @param flags The flags to match
111 *
112 * @return a mask of the pins with matching @p flags
113 */
114 static gpio_port_pins_t
get_pins_with_flags(const struct device * port,gpio_port_pins_t mask,gpio_flags_t flags)115 get_pins_with_flags(const struct device *port, gpio_port_pins_t mask,
116 gpio_flags_t flags)
117 {
118 size_t i;
119 gpio_port_pins_t matched = 0;
120 struct gpio_emul_data *drv_data =
121 (struct gpio_emul_data *)port->data;
122 const struct gpio_emul_config *config =
123 (const struct gpio_emul_config *)port->config;
124
125 for (i = 0; i < config->num_pins; ++i) {
126 if ((drv_data->flags[i] & mask) == flags) {
127 matched |= BIT(i);
128 }
129 }
130
131 return matched;
132 }
133
134 /**
135 * @brief Obtain a mask of pins that are configured as @ref GPIO_INPUT
136 *
137 * The caller must ensure that @ref gpio_emul_data.lock is locked.
138 *
139 * @param port The emulated GPIO device pointer
140 *
141 * @return a mask of pins that are configured as @ref GPIO_INPUT
142 */
get_input_pins(const struct device * port)143 static inline gpio_port_pins_t get_input_pins(const struct device *port)
144 {
145 return get_pins_with_flags(port, GPIO_INPUT, GPIO_INPUT);
146 }
147
148 /**
149 * @brief Obtain a mask of pins that are configured as @ref GPIO_OUTPUT
150 *
151 * The caller must ensure that @ref gpio_emul_data.lock is locked.
152 *
153 * @param port The emulated GPIO device pointer
154 *
155 * @return a mask of pins that are configured as @ref GPIO_OUTPUT
156 */
get_output_pins(const struct device * port)157 static inline gpio_port_pins_t get_output_pins(const struct device *port)
158 {
159 return get_pins_with_flags(port, GPIO_OUTPUT, GPIO_OUTPUT);
160 }
161
162 /**
163 * Check if @p port has capabilities specified in @p caps
164 *
165 * @param port The emulated GPIO device pointer
166 * @param caps A bitmask of @ref gpio_emul_interrupt_cap
167 *
168 * @return true if all @p caps are present, otherwise false
169 */
gpio_emul_config_has_caps(const struct device * port,int caps)170 static inline bool gpio_emul_config_has_caps(const struct device *port,
171 int caps) {
172
173 const struct gpio_emul_config *config =
174 (const struct gpio_emul_config *)port->config;
175
176 return (caps & config->interrupt_caps) == caps;
177 }
178
179 /*
180 * GPIO backend API (for setting input pin values)
181 */
gpio_emul_gen_interrupt_bits(const struct device * port,gpio_port_pins_t mask,gpio_port_value_t prev_values,gpio_port_value_t values,gpio_port_pins_t * interrupts,bool detect_edge)182 static void gpio_emul_gen_interrupt_bits(const struct device *port,
183 gpio_port_pins_t mask,
184 gpio_port_value_t prev_values,
185 gpio_port_value_t values,
186 gpio_port_pins_t *interrupts,
187 bool detect_edge)
188 {
189 size_t i;
190 bool bit;
191 bool prev_bit;
192 struct gpio_emul_data *drv_data =
193 (struct gpio_emul_data *)port->data;
194 const struct gpio_emul_config *config =
195 (const struct gpio_emul_config *)port->config;
196
197 for (i = 0, *interrupts = 0; mask && i < config->num_pins;
198 ++i, mask >>= 1, prev_values >>= 1, values >>= 1) {
199 if ((mask & 1) == 0) {
200 continue;
201 }
202
203 prev_bit = ((prev_values & 1) != 0);
204 bit = ((values & 1) != 0);
205
206 switch (drv_data->flags[i] & GPIO_EMUL_INT_BITMASK) {
207 case GPIO_INT_EDGE_RISING:
208 if (gpio_emul_config_has_caps(port, GPIO_EMUL_INT_CAP_EDGE_RISING)) {
209 if (detect_edge && !prev_bit && bit) {
210 drv_data->interrupts |= BIT(i);
211 *interrupts |= (BIT(i) & drv_data->enabled_interrupts);
212 }
213 }
214 break;
215 case GPIO_INT_EDGE_FALLING:
216 if (gpio_emul_config_has_caps(port, GPIO_EMUL_INT_CAP_EDGE_FALLING)) {
217 if (detect_edge && prev_bit && !bit) {
218 drv_data->interrupts |= BIT(i);
219 *interrupts |= (BIT(i) & drv_data->enabled_interrupts);
220 }
221 }
222 break;
223 case GPIO_INT_EDGE_BOTH:
224 if (gpio_emul_config_has_caps(port,
225 GPIO_EMUL_INT_CAP_EDGE_RISING | GPIO_EMUL_INT_CAP_EDGE_FALLING)) {
226 if (detect_edge && prev_bit != bit) {
227 drv_data->interrupts |= BIT(i);
228 *interrupts |= (BIT(i) & drv_data->enabled_interrupts);
229 }
230 }
231 break;
232 case GPIO_INT_LEVEL_LOW:
233 if (gpio_emul_config_has_caps(port, GPIO_EMUL_INT_CAP_LEVEL_LOW)) {
234 if (!bit) {
235 drv_data->interrupts |= BIT(i);
236 *interrupts |= (BIT(i) & drv_data->enabled_interrupts);
237 }
238 }
239 break;
240 case GPIO_INT_LEVEL_HIGH:
241 if (gpio_emul_config_has_caps(port, GPIO_EMUL_INT_CAP_LEVEL_HIGH)) {
242 if (bit) {
243 drv_data->interrupts |= BIT(i);
244 *interrupts |= (BIT(i) & drv_data->enabled_interrupts);
245 }
246 }
247 break;
248 case 0:
249 case GPIO_INT_DISABLE:
250 break;
251 default:
252 LOG_DBG("unhandled case %u",
253 drv_data->flags[i] & GPIO_EMUL_INT_BITMASK);
254 break;
255 }
256 }
257 }
258
259 /**
260 * @brief Trigger possible interrupt events after an input pin has changed
261 *
262 * For more information, see @ref gpio_interface.
263 *
264 * The caller must ensure that @ref gpio_emul_data.lock is locked.
265 *
266 * @param port The emulated GPIO port
267 * @param mask The mask of pins that have changed
268 * @param prev_values Previous pin values
269 * @param values Current pin values
270 */
gpio_emul_pend_interrupt(const struct device * port,gpio_port_pins_t mask,gpio_port_value_t prev_values,gpio_port_value_t values)271 static void gpio_emul_pend_interrupt(const struct device *port, gpio_port_pins_t mask,
272 gpio_port_value_t prev_values,
273 gpio_port_value_t values)
274 {
275 gpio_port_pins_t interrupts;
276 struct gpio_emul_data *drv_data =
277 (struct gpio_emul_data *)port->data;
278 k_spinlock_key_t key;
279
280 key = k_spin_lock(&drv_data->lock);
281 gpio_emul_gen_interrupt_bits(port, mask, prev_values, values,
282 &interrupts, true);
283 while (interrupts != 0) {
284 k_spin_unlock(&drv_data->lock, key);
285 gpio_fire_callbacks(&drv_data->callbacks, port, interrupts);
286 key = k_spin_lock(&drv_data->lock);
287 /* Clear handled interrupts */
288 drv_data->interrupts &= ~interrupts;
289 gpio_emul_gen_interrupt_bits(port, mask, prev_values, values,
290 &interrupts, false);
291 }
292
293 k_spin_unlock(&drv_data->lock, key);
294 }
295
gpio_emul_input_set_masked_int(const struct device * port,gpio_port_pins_t mask,gpio_port_value_t values)296 int gpio_emul_input_set_masked_int(const struct device *port,
297 gpio_port_pins_t mask,
298 gpio_port_value_t values)
299 {
300 gpio_port_pins_t input_mask;
301 gpio_port_pins_t prev_values;
302 struct gpio_emul_data *drv_data =
303 (struct gpio_emul_data *)port->data;
304 const struct gpio_emul_config *config =
305 (const struct gpio_emul_config *)port->config;
306
307 if (mask == 0) {
308 return 0;
309 }
310
311 if (~config->common.port_pin_mask & mask) {
312 LOG_ERR("Pin not supported port_pin_mask=%x mask=%x",
313 config->common.port_pin_mask, mask);
314 return -EINVAL;
315 }
316
317 input_mask = get_input_pins(port);
318 if (~input_mask & mask) {
319 LOG_ERR("Not input pin input_mask=%x mask=%x", input_mask,
320 mask);
321 return -EINVAL;
322 }
323
324 prev_values = drv_data->input_vals;
325 drv_data->input_vals &= ~mask;
326 drv_data->input_vals |= values & mask;
327
328 return 0;
329 }
330
331 /* documented in drivers/gpio/gpio_emul.h */
gpio_emul_input_set_masked(const struct device * port,gpio_port_pins_t mask,gpio_port_value_t values)332 int gpio_emul_input_set_masked(const struct device *port, gpio_port_pins_t mask,
333 gpio_port_value_t values)
334 {
335 struct gpio_emul_data *drv_data = (struct gpio_emul_data *)port->data;
336 gpio_port_pins_t prev_input_values;
337 gpio_port_pins_t input_values;
338 k_spinlock_key_t key;
339 int rv;
340
341 key = k_spin_lock(&drv_data->lock);
342 prev_input_values = drv_data->input_vals;
343 rv = gpio_emul_input_set_masked_int(port, mask, values);
344 input_values = drv_data->input_vals;
345 k_spin_unlock(&drv_data->lock, key);
346 if (rv) {
347 return rv;
348 }
349
350 gpio_emul_pend_interrupt(port, mask, prev_input_values, input_values);
351 return 0;
352 }
353
354 /* documented in drivers/gpio/gpio_emul.h */
gpio_emul_output_get_masked(const struct device * port,gpio_port_pins_t mask,gpio_port_value_t * values)355 int gpio_emul_output_get_masked(const struct device *port, gpio_port_pins_t mask,
356 gpio_port_value_t *values)
357 {
358 struct gpio_emul_data *drv_data =
359 (struct gpio_emul_data *)port->data;
360 const struct gpio_emul_config *config =
361 (const struct gpio_emul_config *)port->config;
362 k_spinlock_key_t key;
363
364 if (mask == 0) {
365 return 0;
366 }
367
368 if (~config->common.port_pin_mask & mask) {
369 return -EINVAL;
370 }
371
372 key = k_spin_lock(&drv_data->lock);
373 *values = drv_data->output_vals & get_output_pins(port);
374 k_spin_unlock(&drv_data->lock, key);
375
376 return 0;
377 }
378
379 /* documented in drivers/gpio/gpio_emul.h */
gpio_emul_flags_get(const struct device * port,gpio_pin_t pin,gpio_flags_t * flags)380 int gpio_emul_flags_get(const struct device *port, gpio_pin_t pin, gpio_flags_t *flags)
381 {
382 struct gpio_emul_data *drv_data =
383 (struct gpio_emul_data *)port->data;
384 const struct gpio_emul_config *config =
385 (const struct gpio_emul_config *)port->config;
386 k_spinlock_key_t key;
387
388 if (flags == NULL) {
389 return -EINVAL;
390 }
391
392 if ((config->common.port_pin_mask & BIT(pin)) == 0) {
393 return -EINVAL;
394 }
395
396 key = k_spin_lock(&drv_data->lock);
397 *flags = drv_data->flags[pin];
398 k_spin_unlock(&drv_data->lock, key);
399
400 return 0;
401 }
402
403 /*
404 * GPIO Driver API
405 *
406 * API is documented at drivers/gpio.h
407 */
408
gpio_emul_pin_configure(const struct device * port,gpio_pin_t pin,gpio_flags_t flags)409 static int gpio_emul_pin_configure(const struct device *port, gpio_pin_t pin,
410 gpio_flags_t flags)
411 {
412 struct gpio_emul_data *drv_data =
413 (struct gpio_emul_data *)port->data;
414 const struct gpio_emul_config *config =
415 (const struct gpio_emul_config *)port->config;
416 k_spinlock_key_t key;
417 int rv;
418
419 if (flags & GPIO_OPEN_DRAIN) {
420 return -ENOTSUP;
421 }
422
423 if (flags & GPIO_OPEN_SOURCE) {
424 return -ENOTSUP;
425 }
426
427 if ((config->common.port_pin_mask & BIT(pin)) == 0) {
428 return -EINVAL;
429 }
430
431 key = k_spin_lock(&drv_data->lock);
432 drv_data->flags[pin] = flags;
433 if (flags & GPIO_OUTPUT) {
434 if (flags & GPIO_OUTPUT_INIT_LOW) {
435 drv_data->output_vals &= ~BIT(pin);
436 if (flags & GPIO_INPUT) {
437 /* for push-pull mode to generate interrupts */
438 rv = gpio_emul_input_set_masked_int(
439 port, BIT(pin), drv_data->output_vals);
440 __ASSERT_NO_MSG(rv == 0);
441 }
442 } else if (flags & GPIO_OUTPUT_INIT_HIGH) {
443 drv_data->output_vals |= BIT(pin);
444 if (flags & GPIO_INPUT) {
445 /* for push-pull mode to generate interrupts */
446 rv = gpio_emul_input_set_masked_int(
447 port, BIT(pin), drv_data->output_vals);
448 __ASSERT_NO_MSG(rv == 0);
449 }
450 }
451 } else if (flags & GPIO_INPUT) {
452 if (flags & GPIO_PULL_UP) {
453 rv = gpio_emul_input_set_masked_int(port, BIT(pin), BIT(pin));
454 __ASSERT_NO_MSG(rv == 0);
455 } else if (flags & GPIO_PULL_DOWN) {
456 rv = gpio_emul_input_set_masked_int(
457 port, BIT(pin), 0);
458 __ASSERT_NO_MSG(rv == 0);
459 }
460 }
461
462 k_spin_unlock(&drv_data->lock, key);
463 gpio_fire_callbacks(&drv_data->callbacks, port, BIT(pin));
464 /* GPIO pin configuration changed so clear the pending interrupt. */
465 drv_data->interrupts &= ~((gpio_port_pins_t)BIT(pin));
466
467 return 0;
468 }
469
470 #ifdef CONFIG_GPIO_GET_CONFIG
gpio_emul_pin_get_config(const struct device * port,gpio_pin_t pin,gpio_flags_t * out_flags)471 static int gpio_emul_pin_get_config(const struct device *port, gpio_pin_t pin,
472 gpio_flags_t *out_flags)
473 {
474 struct gpio_emul_data *drv_data =
475 (struct gpio_emul_data *)port->data;
476 k_spinlock_key_t key;
477
478 key = k_spin_lock(&drv_data->lock);
479
480 *out_flags = drv_data->flags[pin] &
481 ~(GPIO_OUTPUT_INIT_LOW | GPIO_OUTPUT_INIT_HIGH
482 | GPIO_OUTPUT_INIT_LOGICAL);
483 if (drv_data->flags[pin] & GPIO_OUTPUT) {
484 if (drv_data->output_vals & BIT(pin)) {
485 *out_flags |= GPIO_OUTPUT_HIGH;
486 } else {
487 *out_flags |= GPIO_OUTPUT_LOW;
488 }
489 }
490
491 k_spin_unlock(&drv_data->lock, key);
492
493 return 0;
494 }
495 #endif
496
gpio_emul_port_get_raw(const struct device * port,gpio_port_value_t * values)497 static int gpio_emul_port_get_raw(const struct device *port, gpio_port_value_t *values)
498 {
499 struct gpio_emul_data *drv_data =
500 (struct gpio_emul_data *)port->data;
501 k_spinlock_key_t key;
502
503 if (values == NULL) {
504 return -EINVAL;
505 }
506
507 key = k_spin_lock(&drv_data->lock);
508 *values = drv_data->input_vals & get_input_pins(port);
509 k_spin_unlock(&drv_data->lock, key);
510
511 return 0;
512 }
513
gpio_emul_port_set_masked_raw(const struct device * port,gpio_port_pins_t mask,gpio_port_value_t values)514 static int gpio_emul_port_set_masked_raw(const struct device *port,
515 gpio_port_pins_t mask,
516 gpio_port_value_t values)
517 {
518 gpio_port_pins_t output_mask;
519 gpio_port_pins_t prev_values;
520 gpio_port_pins_t prev_input_values;
521 gpio_port_pins_t input_values;
522 gpio_port_pins_t input_mask;
523 struct gpio_emul_data *drv_data =
524 (struct gpio_emul_data *)port->data;
525 k_spinlock_key_t key;
526 int rv;
527
528 key = k_spin_lock(&drv_data->lock);
529 output_mask = get_output_pins(port);
530 mask &= output_mask;
531 prev_values = drv_data->output_vals;
532 prev_values &= output_mask;
533 values &= mask;
534 drv_data->output_vals &= ~mask;
535 drv_data->output_vals |= values;
536 /* in push-pull, set input values & fire interrupts */
537 prev_input_values = drv_data->input_vals;
538 input_mask = mask & get_input_pins(port);
539 rv = gpio_emul_input_set_masked_int(port, input_mask,
540 drv_data->output_vals);
541 input_values = drv_data->input_vals;
542 k_spin_unlock(&drv_data->lock, key);
543 __ASSERT_NO_MSG(rv == 0);
544 gpio_emul_pend_interrupt(port, input_mask, prev_input_values, input_values);
545
546 /* for output-wiring, so the user can take action based on output */
547 if (prev_values ^ values) {
548 gpio_fire_callbacks(&drv_data->callbacks, port, mask & ~get_input_pins(port));
549 }
550
551 return 0;
552 }
553
gpio_emul_port_set_bits_raw(const struct device * port,gpio_port_pins_t pins)554 static int gpio_emul_port_set_bits_raw(const struct device *port,
555 gpio_port_pins_t pins)
556 {
557 struct gpio_emul_data *drv_data =
558 (struct gpio_emul_data *)port->data;
559 k_spinlock_key_t key;
560 gpio_port_pins_t prev_input_values;
561 gpio_port_pins_t input_values;
562 gpio_port_pins_t input_mask;
563 int rv;
564
565 key = k_spin_lock(&drv_data->lock);
566 pins &= get_output_pins(port);
567 drv_data->output_vals |= pins;
568 prev_input_values = drv_data->input_vals;
569 input_mask = pins & get_input_pins(port);
570 rv = gpio_emul_input_set_masked_int(port, input_mask,
571 drv_data->output_vals);
572 input_values = drv_data->input_vals;
573 k_spin_unlock(&drv_data->lock, key);
574 __ASSERT_NO_MSG(rv == 0);
575 gpio_emul_pend_interrupt(port, input_mask, prev_input_values, input_values);
576 /* for output-wiring, so the user can take action based on output */
577 gpio_fire_callbacks(&drv_data->callbacks, port, pins & ~get_input_pins(port));
578
579 return 0;
580 }
581
gpio_emul_port_clear_bits_raw(const struct device * port,gpio_port_pins_t pins)582 static int gpio_emul_port_clear_bits_raw(const struct device *port,
583 gpio_port_pins_t pins)
584 {
585 struct gpio_emul_data *drv_data =
586 (struct gpio_emul_data *)port->data;
587 k_spinlock_key_t key;
588 gpio_port_pins_t prev_input_values;
589 gpio_port_pins_t input_values;
590 gpio_port_pins_t input_mask;
591 int rv;
592
593 key = k_spin_lock(&drv_data->lock);
594 pins &= get_output_pins(port);
595 drv_data->output_vals &= ~pins;
596 prev_input_values = drv_data->input_vals;
597 input_mask = pins & get_input_pins(port);
598 rv = gpio_emul_input_set_masked_int(port, input_mask, drv_data->output_vals);
599 input_values = drv_data->input_vals;
600 k_spin_unlock(&drv_data->lock, key);
601 __ASSERT_NO_MSG(rv == 0);
602 gpio_emul_pend_interrupt(port, input_mask, prev_input_values, input_values);
603 /* for output-wiring, so the user can take action based on output */
604 gpio_fire_callbacks(&drv_data->callbacks, port, pins & ~get_input_pins(port));
605
606 return 0;
607 }
608
gpio_emul_port_toggle_bits(const struct device * port,gpio_port_pins_t pins)609 static int gpio_emul_port_toggle_bits(const struct device *port, gpio_port_pins_t pins)
610 {
611 struct gpio_emul_data *drv_data =
612 (struct gpio_emul_data *)port->data;
613 k_spinlock_key_t key;
614 int rv;
615
616 key = k_spin_lock(&drv_data->lock);
617 drv_data->output_vals ^= (pins & get_output_pins(port));
618 /* in push-pull, set input values but do not fire interrupts (yet) */
619 rv = gpio_emul_input_set_masked_int(port, pins & get_input_pins(port),
620 drv_data->output_vals);
621 k_spin_unlock(&drv_data->lock, key);
622 __ASSERT_NO_MSG(rv == 0);
623 /* for output-wiring, so the user can take action based on output */
624 gpio_fire_callbacks(&drv_data->callbacks, port, pins);
625
626 return 0;
627 }
628
gpio_emul_level_trigger_supported(const struct device * port,enum gpio_int_trig trig)629 static bool gpio_emul_level_trigger_supported(const struct device *port,
630 enum gpio_int_trig trig)
631 {
632 switch (trig) {
633 case GPIO_INT_TRIG_LOW:
634 return gpio_emul_config_has_caps(port, GPIO_EMUL_INT_CAP_LEVEL_LOW);
635 case GPIO_INT_TRIG_HIGH:
636 return gpio_emul_config_has_caps(port, GPIO_EMUL_INT_CAP_LEVEL_HIGH);
637 case GPIO_INT_TRIG_BOTH:
638 return gpio_emul_config_has_caps(port, GPIO_EMUL_INT_CAP_LEVEL_LOW
639 | GPIO_EMUL_INT_CAP_LEVEL_HIGH);
640 default:
641 return false;
642 }
643 }
644
gpio_emul_edge_trigger_supported(const struct device * port,enum gpio_int_trig trig)645 static bool gpio_emul_edge_trigger_supported(const struct device *port,
646 enum gpio_int_trig trig)
647 {
648 switch (trig) {
649 case GPIO_INT_TRIG_LOW:
650 return gpio_emul_config_has_caps(port, GPIO_EMUL_INT_CAP_EDGE_FALLING);
651 case GPIO_INT_TRIG_HIGH:
652 return gpio_emul_config_has_caps(port, GPIO_EMUL_INT_CAP_EDGE_RISING);
653 case GPIO_INT_TRIG_BOTH:
654 return gpio_emul_config_has_caps(port, GPIO_EMUL_INT_CAP_EDGE_FALLING
655 | GPIO_EMUL_INT_CAP_EDGE_RISING);
656 default:
657 return false;
658 }
659 }
660
gpio_emul_pin_interrupt_configure(const struct device * port,gpio_pin_t pin,enum gpio_int_mode mode,enum gpio_int_trig trig)661 static int gpio_emul_pin_interrupt_configure(const struct device *port, gpio_pin_t pin,
662 enum gpio_int_mode mode,
663 enum gpio_int_trig trig)
664 {
665 int ret;
666 struct gpio_emul_data *drv_data =
667 (struct gpio_emul_data *)port->data;
668 const struct gpio_emul_config *config =
669 (const struct gpio_emul_config *)port->config;
670 k_spinlock_key_t key;
671
672 if ((BIT(pin) & config->common.port_pin_mask) == 0) {
673 return -EINVAL;
674 }
675
676 #ifdef CONFIG_GPIO_ENABLE_DISABLE_INTERRUPT
677 if (mode != GPIO_INT_MODE_DISABLED && !(mode & GPIO_INT_ENABLE_DISABLE_ONLY)) {
678 #else
679 if (mode != GPIO_INT_MODE_DISABLED) {
680 #endif /* CONFIG_GPIO_ENABLE_DISABLE_INTERRUPT */
681 switch (trig) {
682 case GPIO_INT_TRIG_LOW:
683 case GPIO_INT_TRIG_HIGH:
684 case GPIO_INT_TRIG_BOTH:
685 break;
686 default:
687 return -EINVAL;
688 }
689 }
690
691 if (mode == GPIO_INT_MODE_LEVEL) {
692 if (!gpio_emul_level_trigger_supported(port, trig)) {
693 return -ENOTSUP;
694 }
695 }
696
697 if (mode == GPIO_INT_MODE_EDGE) {
698 if (!gpio_emul_edge_trigger_supported(port, trig)) {
699 return -ENOTSUP;
700 }
701 }
702
703 key = k_spin_lock(&drv_data->lock);
704
705 #ifdef CONFIG_GPIO_ENABLE_DISABLE_INTERRUPT
706 /* According to the GPIO interrupt configuration flag documentation,
707 * changes to the interrupt trigger properties should clear pending
708 * interrupts.
709 */
710 if (!(mode & GPIO_INT_ENABLE_DISABLE_ONLY)) {
711 drv_data->interrupts &= ~((gpio_port_pins_t)BIT(pin));
712 }
713 #else
714 drv_data->interrupts &= ~((gpio_port_pins_t)BIT(pin));
715 #endif /* CONFIG_GPIO_ENABLE_DISABLE_INTERRUPT */
716
717 switch (mode) {
718 case GPIO_INT_MODE_DISABLED:
719 drv_data->flags[pin] &= ~GPIO_EMUL_INT_BITMASK;
720 drv_data->flags[pin] |= GPIO_INT_DISABLE;
721 #ifdef CONFIG_GPIO_ENABLE_DISABLE_INTERRUPT
722 __fallthrough;
723 case GPIO_INT_MODE_DISABLE_ONLY:
724 #endif /* CONFIG_GPIO_ENABLE_DISABLE_INTERRUPT */
725 drv_data->enabled_interrupts &= ~((gpio_port_pins_t)BIT(pin));
726 break;
727 case GPIO_INT_MODE_LEVEL:
728 case GPIO_INT_MODE_EDGE:
729 drv_data->flags[pin] &= ~GPIO_EMUL_INT_BITMASK;
730 drv_data->flags[pin] |= (mode | trig);
731 #ifdef CONFIG_GPIO_ENABLE_DISABLE_INTERRUPT
732 __fallthrough;
733 case GPIO_INT_MODE_ENABLE_ONLY:
734 #endif /* CONFIG_GPIO_ENABLE_DISABLE_INTERRUPT */
735 drv_data->enabled_interrupts |= BIT(pin);
736 break;
737 default:
738 ret = -EINVAL;
739 goto unlock;
740 }
741
742 ret = 0;
743
744 unlock:
745 k_spin_unlock(&drv_data->lock, key);
746
747 /* Trigger callback if this pin has pending interrupt */
748 if (BIT(pin) & (drv_data->interrupts & drv_data->enabled_interrupts)) {
749 gpio_fire_callbacks(&drv_data->callbacks, port, BIT(pin));
750 drv_data->interrupts &= ~((gpio_port_pins_t)BIT(pin));
751 }
752
753 return ret;
754 }
755
756 static int gpio_emul_manage_callback(const struct device *port,
757 struct gpio_callback *cb, bool set)
758 {
759 struct gpio_emul_data *drv_data =
760 (struct gpio_emul_data *)port->data;
761
762 return gpio_manage_callback(&drv_data->callbacks, cb, set);
763 }
764
765 static uint32_t gpio_emul_get_pending_int(const struct device *dev)
766 {
767 struct gpio_emul_data *drv_data =
768 (struct gpio_emul_data *)dev->data;
769
770 return drv_data->interrupts;
771 }
772
773 #ifdef CONFIG_GPIO_GET_DIRECTION
774 static int gpio_emul_port_get_direction(const struct device *port, gpio_port_pins_t map,
775 gpio_port_pins_t *inputs, gpio_port_pins_t *outputs)
776 {
777 int i;
778 gpio_port_pins_t ip = 0;
779 gpio_port_pins_t op = 0;
780 struct gpio_emul_data *const drv_data = (struct gpio_emul_data *)port->data;
781 const struct gpio_emul_config *config = (const struct gpio_emul_config *)port->config;
782
783 map &= config->common.port_pin_mask;
784
785 if (inputs != NULL) {
786 for (i = find_lsb_set(map) - 1; map;
787 map &= ~BIT(i), i = find_lsb_set(map) - 1) {
788 ip |= !!(drv_data->flags[i] & GPIO_INPUT) * BIT(i);
789 }
790
791 *inputs = ip;
792 }
793
794 if (outputs != NULL) {
795 for (i = find_lsb_set(map) - 1; map;
796 map &= ~BIT(i), i = find_lsb_set(map) - 1) {
797 op |= !!(drv_data->flags[i] & GPIO_OUTPUT) * BIT(i);
798 }
799
800 *outputs = op;
801 }
802
803 return 0;
804 }
805 #endif /* CONFIG_GPIO_GET_DIRECTION */
806
807 static DEVICE_API(gpio, gpio_emul_driver) = {
808 .pin_configure = gpio_emul_pin_configure,
809 #ifdef CONFIG_GPIO_GET_CONFIG
810 .pin_get_config = gpio_emul_pin_get_config,
811 #endif
812 .port_get_raw = gpio_emul_port_get_raw,
813 .port_set_masked_raw = gpio_emul_port_set_masked_raw,
814 .port_set_bits_raw = gpio_emul_port_set_bits_raw,
815 .port_clear_bits_raw = gpio_emul_port_clear_bits_raw,
816 .port_toggle_bits = gpio_emul_port_toggle_bits,
817 .pin_interrupt_configure = gpio_emul_pin_interrupt_configure,
818 .manage_callback = gpio_emul_manage_callback,
819 .get_pending_int = gpio_emul_get_pending_int,
820 #ifdef CONFIG_GPIO_GET_DIRECTION
821 .port_get_direction = gpio_emul_port_get_direction,
822 #endif /* CONFIG_GPIO_GET_DIRECTION */
823 };
824
825 static int gpio_emul_init(const struct device *dev)
826 {
827 struct gpio_emul_data *drv_data =
828 (struct gpio_emul_data *)dev->data;
829
830 sys_slist_init(&drv_data->callbacks);
831 return 0;
832 }
833
834 #ifdef CONFIG_PM_DEVICE
835 static int gpio_emul_pm_device_pm_action(const struct device *dev,
836 enum pm_device_action action)
837 {
838 ARG_UNUSED(dev);
839 ARG_UNUSED(action);
840
841 return 0;
842 }
843 #endif
844
845 /*
846 * Device Initialization
847 */
848
849 #define GPIO_EMUL_INT_CAPS(_num) (0 \
850 + DT_INST_PROP(_num, rising_edge) \
851 * GPIO_EMUL_INT_CAP_EDGE_RISING \
852 + DT_INST_PROP(_num, falling_edge) \
853 * GPIO_EMUL_INT_CAP_EDGE_FALLING \
854 + DT_INST_PROP(_num, high_level) \
855 * GPIO_EMUL_INT_CAP_LEVEL_HIGH \
856 + DT_INST_PROP(_num, low_level) \
857 * GPIO_EMUL_INT_CAP_LEVEL_LOW \
858 )
859
860 #define DEFINE_GPIO_EMUL(_num) \
861 \
862 static gpio_flags_t \
863 gpio_emul_flags_##_num[DT_INST_PROP(_num, ngpios)]; \
864 \
865 static const struct gpio_emul_config gpio_emul_config_##_num = {\
866 .common = { \
867 .port_pin_mask = \
868 GPIO_PORT_PIN_MASK_FROM_DT_INST(_num), \
869 }, \
870 .num_pins = DT_INST_PROP(_num, ngpios), \
871 .interrupt_caps = GPIO_EMUL_INT_CAPS(_num) \
872 }; \
873 BUILD_ASSERT( \
874 DT_INST_PROP(_num, ngpios) <= GPIO_MAX_PINS_PER_PORT, \
875 "Too many ngpios"); \
876 \
877 static struct gpio_emul_data gpio_emul_data_##_num = { \
878 .flags = gpio_emul_flags_##_num, \
879 }; \
880 \
881 PM_DEVICE_DT_INST_DEFINE(_num, gpio_emul_pm_device_pm_action); \
882 \
883 DEVICE_DT_INST_DEFINE(_num, gpio_emul_init, \
884 PM_DEVICE_DT_INST_GET(_num), \
885 &gpio_emul_data_##_num, \
886 &gpio_emul_config_##_num, POST_KERNEL, \
887 CONFIG_GPIO_INIT_PRIORITY, \
888 &gpio_emul_driver);
889
890 DT_INST_FOREACH_STATUS_OKAY(DEFINE_GPIO_EMUL)
891