1 /*
2  * Copyright (c) 2022 Chromium OS Authors
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 #define DT_DRV_COMPAT ti_tca6424a
7 
8 #include <zephyr/drivers/gpio/gpio_utils.h>
9 #include <zephyr/drivers/gpio.h>
10 #include <zephyr/drivers/i2c.h>
11 #include <zephyr/logging/log.h>
12 #include <zephyr/sys/byteorder.h>
13 #include <zephyr/sys/util.h>
14 LOG_MODULE_REGISTER(gpio_tca6424a, CONFIG_GPIO_LOG_LEVEL);
15 
16 /* TCA6424A auto increment register addresses */
17 #define TCA6424A_REG_INPUT			0x80
18 #define TCA6424A_REG_OUTPUT			0x84
19 #define TCA6424A_REG_POLARITY_INVERSION		0x88
20 #define TCA6424A_REG_CONFIGURATION		0x8C
21 
22 /** Cache of the output configuration and data of the pins. */
23 struct tca6424a_pins_state {
24 	uint32_t input;
25 	uint32_t output;
26 	uint32_t config;
27 };
28 
29 struct tca6424a_irq_state {
30 	uint32_t rising;
31 	uint32_t falling;
32 };
33 
34 /** Configuration data */
35 struct tca6424a_drv_cfg {
36 	/* gpio_driver_config needs to be first */
37 	struct gpio_driver_config common;
38 
39 	struct i2c_dt_spec i2c_spec;
40 	struct gpio_dt_spec int_gpio;
41 	struct gpio_dt_spec reset_gpio;
42 };
43 
44 /** Runtime driver data */
45 struct tca6424a_drv_data {
46 	/* gpio_driver_data needs to be first */
47 	struct gpio_driver_data common;
48 
49 	sys_slist_t callbacks;
50 	struct k_sem lock;
51 	struct k_work work;
52 	const struct device *dev;
53 	struct gpio_callback int_gpio_cb;
54 	struct tca6424a_pins_state pins_state;
55 	struct tca6424a_irq_state irq_state;
56 };
57 
58 /**
59  * @brief Gets the state of a specified block of 3 registers from the TCA6424A
60  * and stores it in specified cache and buffer variables.
61  *
62  * @param dev Pointer to the device structure for the driver instance.
63  * @param reg Address of the first of 3 registers to be read.
64  * @param cache Pointer to the location to cache the pins states.
65  * @param buf Pointer to the buffer to output the register.
66  *
67  * @retval 0 if successful.
68  * @retval Negative value for error code.
69  */
read_port_regs(const struct device * dev,uint8_t reg,uint32_t * buf)70 static int read_port_regs(const struct device *dev, uint8_t reg, uint32_t *buf)
71 {
72 	const struct tca6424a_drv_cfg *const config = dev->config;
73 	uint32_t port_data, value;
74 	int ret;
75 
76 	ret = i2c_burst_read_dt(&config->i2c_spec, reg, (uint8_t *)&port_data, 3);
77 	if (ret != 0) {
78 		LOG_ERR("%s: error reading register 0x%X (%d)", dev->name,
79 			reg, ret);
80 		return ret;
81 	}
82 
83 	value = sys_le24_to_cpu(port_data);
84 	*buf = value;
85 	LOG_DBG("%s: Read: REG[0x%X] = 0x%X, REG[0x%X] = 0x%X, "
86 		"REG[0x%X] = 0x%X",
87 		dev->name, reg, (*buf & 0xFF), (reg + 1), ((*buf >> 8) & 0xFF),
88 		(reg + 2), ((*buf >> 16) & 0xFF));
89 
90 	return 0;
91 }
92 
93 /**
94  * @brief writes to a specified block of 3 registers from the TCA6424A
95  * and stores it in a specified cache variable.
96  *
97  * @param dev Pointer to the device structure for the driver instance.
98  * @param reg Address of the first of 3 registers to be written.
99  * @param cache Pointer to the location to cache the pins states.
100  * @param value The pin value to be written into the registers.
101  *
102  * @retval 0 if successful.
103  * @retval Negative value for error code.
104  */
write_port_regs(const struct device * dev,uint8_t reg,uint32_t value)105 static int write_port_regs(const struct device *dev, uint8_t reg, uint32_t value)
106 {
107 	const struct tca6424a_drv_cfg *const config = dev->config;
108 	uint8_t buf[4];
109 	int ret;
110 
111 	LOG_DBG("%s: Write: REG[0x%X] = 0x%X, REG[0x%X] = 0x%X, "
112 		"REG[0x%X] = 0x%X",
113 		dev->name, reg, (value & 0xFF), (reg + 1), ((value >> 8) & 0xFF),
114 		(reg + 2), ((value >> 16) & 0xFF));
115 
116 	buf[0] = reg;
117 	sys_put_le24(value, &buf[1]);
118 	ret = i2c_write_dt(&config->i2c_spec, buf, sizeof(buf));
119 	if (ret != 0) {
120 		LOG_ERR("%s: error writing to register 0x%X "
121 			"(%d)",
122 			dev->name, reg, ret);
123 	}
124 	return ret;
125 }
126 
update_input_regs(const struct device * dev,uint32_t * buf)127 static inline int update_input_regs(const struct device *dev, uint32_t *buf)
128 {
129 	int ret;
130 	struct tca6424a_drv_data *const drv_data = dev->data;
131 
132 	ret = read_port_regs(dev, TCA6424A_REG_INPUT, buf);
133 	if (ret == 0) {
134 		drv_data->pins_state.input = *buf;
135 	}
136 
137 	return ret;
138 }
139 
update_output_regs(const struct device * dev,uint32_t value)140 static inline int update_output_regs(const struct device *dev, uint32_t value)
141 {
142 	int ret;
143 	struct tca6424a_drv_data *const drv_data = dev->data;
144 
145 	ret = write_port_regs(dev, TCA6424A_REG_OUTPUT, value);
146 	if (ret == 0) {
147 		drv_data->pins_state.output = value;
148 	}
149 
150 	return ret;
151 }
152 
update_invers_regs(const struct device * dev,uint32_t value)153 static inline int update_invers_regs(const struct device *dev, uint32_t value)
154 {
155 	return write_port_regs(dev, TCA6424A_REG_POLARITY_INVERSION, value);
156 }
157 
update_config_regs(const struct device * dev,uint32_t value)158 static inline int update_config_regs(const struct device *dev, uint32_t value)
159 {
160 	int ret;
161 	struct tca6424a_drv_data *const drv_data = dev->data;
162 
163 	ret = write_port_regs(dev, TCA6424A_REG_CONFIGURATION, value);
164 	if (ret == 0) {
165 		drv_data->pins_state.config = value;
166 	}
167 
168 	return ret;
169 }
170 
171 /**
172  * @brief Handles interrupt triggered by the interrupt pin of TCA6424A.
173  *
174  * If int_gpios is configured in device tree then this will be triggered each
175  * time a gpio configured as an input changes state. The gpio input states are
176  * read in this function which clears the interrupt.
177  *
178  * @param dev Pointer to the device structure for the driver instance.
179  */
tca6424a_handle_interrupt(const struct device * dev)180 static void tca6424a_handle_interrupt(const struct device *dev)
181 {
182 	struct tca6424a_drv_data *drv_data = dev->data;
183 	struct tca6424a_irq_state *irq_state = &drv_data->irq_state;
184 	int ret;
185 	uint32_t previous_state;
186 	uint32_t current_state;
187 	uint32_t transitioned_pins;
188 	uint32_t interrupt_status;
189 
190 	k_sem_take(&drv_data->lock, K_FOREVER);
191 
192 	/* Any interrupts enabled? */
193 	if (!irq_state->rising && !irq_state->falling) {
194 		k_sem_give(&drv_data->lock);
195 		return;
196 	}
197 
198 	/* Store previous input state then read new value */
199 	previous_state = drv_data->pins_state.input;
200 	ret = update_input_regs(dev, &current_state);
201 	if (ret != 0) {
202 		k_sem_give(&drv_data->lock);
203 		return;
204 	}
205 
206 	/* Find out which input pins have changed state */
207 	transitioned_pins = previous_state ^ current_state;
208 
209 	/* Mask gpio transactions with rising/falling edge interrupt config */
210 	interrupt_status = (irq_state->rising & transitioned_pins & current_state);
211 	interrupt_status |= (irq_state->falling & transitioned_pins & previous_state);
212 	k_sem_give(&drv_data->lock);
213 
214 	if (interrupt_status) {
215 		gpio_fire_callbacks(&drv_data->callbacks, dev, interrupt_status);
216 	}
217 }
218 
219 /**
220  * @brief Work handler for TCA6424A interrupt
221  *
222  * @param work Work struct that contains pointer to interrupt handler function
223  */
tca6424a_work_handler(struct k_work * work)224 static void tca6424a_work_handler(struct k_work *work)
225 {
226 	struct tca6424a_drv_data *drv_data = CONTAINER_OF(work, struct tca6424a_drv_data, work);
227 
228 	tca6424a_handle_interrupt(drv_data->dev);
229 }
230 
231 /**
232  * @brief ISR for interrupt pin of TCA6424A
233  *
234  * @param dev Pointer to the device structure for the driver instance.
235  * @param gpio_cb Pointer to callback function struct
236  * @param pins Bitmask of pins that triggered interrupt
237  */
tca6424a_int_gpio_handler(const struct device * dev,struct gpio_callback * gpio_cb,uint32_t pins)238 static void tca6424a_int_gpio_handler(const struct device *dev, struct gpio_callback *gpio_cb,
239 				      uint32_t pins)
240 {
241 	ARG_UNUSED(dev);
242 	ARG_UNUSED(pins);
243 
244 	struct tca6424a_drv_data *drv_data =
245 		CONTAINER_OF(gpio_cb, struct tca6424a_drv_data, int_gpio_cb);
246 
247 	k_work_submit(&drv_data->work);
248 }
249 
tca6424a_setup_pin(const struct device * dev,gpio_pin_t pin,gpio_flags_t flags)250 static int tca6424a_setup_pin(const struct device *dev, gpio_pin_t pin, gpio_flags_t flags)
251 {
252 	struct tca6424a_drv_data *const drv_data = dev->data;
253 	uint32_t reg_cfg = drv_data->pins_state.config;
254 	uint32_t reg_out = drv_data->pins_state.output;
255 	int ret;
256 
257 	/* For each pin, 0 == output, 1 == input */
258 	if ((flags & GPIO_OUTPUT) != 0) {
259 		if ((flags & GPIO_OUTPUT_INIT_HIGH) != 0) {
260 			reg_out |= BIT(pin);
261 		} else if ((flags & GPIO_OUTPUT_INIT_LOW) != 0) {
262 			reg_out &= ~BIT(pin);
263 		}
264 		ret = update_output_regs(dev, reg_out);
265 		if (ret != 0) {
266 			return ret;
267 		}
268 		reg_cfg &= ~BIT(pin);
269 	} else {
270 		reg_cfg |= BIT(pin);
271 	}
272 
273 	ret = update_config_regs(dev, reg_cfg);
274 
275 	return ret;
276 }
277 
tca6424a_pin_config(const struct device * dev,gpio_pin_t pin,gpio_flags_t flags)278 static int tca6424a_pin_config(const struct device *dev, gpio_pin_t pin, gpio_flags_t flags)
279 {
280 	int ret;
281 	struct tca6424a_drv_data *const drv_data = dev->data;
282 
283 	/* Does not support disconnected pin */
284 	if ((flags & (GPIO_INPUT | GPIO_OUTPUT)) == GPIO_DISCONNECTED) {
285 		return -ENOTSUP;
286 	}
287 
288 	/* Single Ended lines (Open drain and open source) not supported */
289 	if ((flags & GPIO_SINGLE_ENDED) != 0) {
290 		return -ENOTSUP;
291 	}
292 
293 	/* The TCA6424A has no internal pull up support */
294 	if (((flags & GPIO_PULL_UP) != 0) || ((flags & GPIO_PULL_DOWN) != 0)) {
295 		return -ENOTSUP;
296 	}
297 
298 	/* Simultaneous input & output mode not supported */
299 	if (((flags & GPIO_INPUT) != 0) && ((flags & GPIO_OUTPUT) != 0)) {
300 		return -ENOTSUP;
301 	}
302 
303 	/* Can't do I2C bus operations from an ISR */
304 	if (k_is_in_isr()) {
305 		return -EWOULDBLOCK;
306 	}
307 
308 	k_sem_take(&drv_data->lock, K_FOREVER);
309 
310 	ret = tca6424a_setup_pin(dev, pin, flags);
311 	if (ret != 0) {
312 		LOG_ERR("%s: error setting pin direction (%d)", dev->name, ret);
313 	}
314 
315 	k_sem_give(&drv_data->lock);
316 	return ret;
317 }
318 
tca6424a_port_get_raw(const struct device * dev,gpio_port_value_t * value)319 static int tca6424a_port_get_raw(const struct device *dev, gpio_port_value_t *value)
320 {
321 	struct tca6424a_drv_data *const drv_data = dev->data;
322 	uint32_t buf;
323 	int ret;
324 
325 	/* Can't do I2C bus operations from an ISR */
326 	if (k_is_in_isr()) {
327 		return -EWOULDBLOCK;
328 	}
329 
330 	k_sem_take(&drv_data->lock, K_FOREVER);
331 
332 	ret = update_input_regs(dev, &buf);
333 	if (ret == 0) {
334 		*value = buf;
335 	}
336 
337 	k_sem_give(&drv_data->lock);
338 	return ret;
339 }
340 
tca6424a_port_set_masked_raw(const struct device * dev,gpio_port_pins_t mask,gpio_port_value_t value)341 static int tca6424a_port_set_masked_raw(const struct device *dev, gpio_port_pins_t mask,
342 					gpio_port_value_t value)
343 {
344 	struct tca6424a_drv_data *const drv_data = dev->data;
345 	uint32_t reg_out;
346 	int ret;
347 
348 	/* Can't do I2C bus operations from an ISR */
349 	if (k_is_in_isr()) {
350 		return -EWOULDBLOCK;
351 	}
352 
353 	k_sem_take(&drv_data->lock, K_FOREVER);
354 
355 	reg_out = drv_data->pins_state.output;
356 	reg_out = (reg_out & ~mask) | (mask & value);
357 
358 	ret = update_output_regs(dev, reg_out);
359 
360 	k_sem_give(&drv_data->lock);
361 
362 	return ret;
363 }
364 
tca6424a_port_set_bits_raw(const struct device * dev,gpio_port_pins_t mask)365 static int tca6424a_port_set_bits_raw(const struct device *dev, gpio_port_pins_t mask)
366 {
367 	return tca6424a_port_set_masked_raw(dev, mask, mask);
368 }
369 
tca6424a_port_clear_bits_raw(const struct device * dev,gpio_port_pins_t mask)370 static int tca6424a_port_clear_bits_raw(const struct device *dev, gpio_port_pins_t mask)
371 {
372 	return tca6424a_port_set_masked_raw(dev, mask, 0);
373 }
374 
tca6424a_port_toggle_bits(const struct device * dev,gpio_port_pins_t mask)375 static int tca6424a_port_toggle_bits(const struct device *dev, gpio_port_pins_t mask)
376 {
377 	struct tca6424a_drv_data *const drv_data = dev->data;
378 	uint32_t reg_out;
379 	int ret;
380 
381 	/* Can't do I2C bus operations from an ISR */
382 	if (k_is_in_isr()) {
383 		return -EWOULDBLOCK;
384 	}
385 
386 	k_sem_take(&drv_data->lock, K_FOREVER);
387 
388 	reg_out = drv_data->pins_state.output;
389 	reg_out ^= mask;
390 
391 	ret = update_output_regs(dev, reg_out);
392 
393 	k_sem_give(&drv_data->lock);
394 
395 	return ret;
396 }
397 
tca6424a_pin_interrupt_configure(const struct device * dev,gpio_pin_t pin,enum gpio_int_mode mode,enum gpio_int_trig trig)398 static int tca6424a_pin_interrupt_configure(const struct device *dev, gpio_pin_t pin,
399 					    enum gpio_int_mode mode, enum gpio_int_trig trig)
400 {
401 	struct tca6424a_drv_data *drv_data = dev->data;
402 	struct tca6424a_irq_state *irq = &drv_data->irq_state;
403 
404 	/* Device does not support level-triggered interrupts. */
405 	if (mode == GPIO_INT_MODE_LEVEL) {
406 		return -ENOTSUP;
407 	}
408 
409 	k_sem_take(&drv_data->lock, K_FOREVER);
410 
411 	if (mode == GPIO_INT_MODE_DISABLED) {
412 		irq->falling &= ~BIT(pin);
413 		irq->rising &= ~BIT(pin);
414 	} else { /* GPIO_INT_MODE_EDGE */
415 		if (trig == GPIO_INT_TRIG_BOTH) {
416 			irq->falling |= BIT(pin);
417 			irq->rising |= BIT(pin);
418 		} else if (trig == GPIO_INT_TRIG_LOW) {
419 			irq->falling |= BIT(pin);
420 			irq->rising &= ~BIT(pin);
421 		} else if (trig == GPIO_INT_TRIG_HIGH) {
422 			irq->falling &= ~BIT(pin);
423 			irq->rising |= BIT(pin);
424 		}
425 	}
426 
427 	k_sem_give(&drv_data->lock);
428 
429 	return 0;
430 }
431 
tca6424a_manage_callback(const struct device * dev,struct gpio_callback * callback,bool set)432 static int tca6424a_manage_callback(const struct device *dev, struct gpio_callback *callback,
433 				    bool set)
434 {
435 	struct tca6424a_drv_data *drv_data = dev->data;
436 
437 	return gpio_manage_callback(&drv_data->callbacks, callback, set);
438 }
439 
440 static DEVICE_API(gpio, tca6424a_drv_api) = {
441 	.pin_configure = tca6424a_pin_config,
442 	.port_get_raw = tca6424a_port_get_raw,
443 	.port_set_masked_raw = tca6424a_port_set_masked_raw,
444 	.port_set_bits_raw = tca6424a_port_set_bits_raw,
445 	.port_clear_bits_raw = tca6424a_port_clear_bits_raw,
446 	.port_toggle_bits = tca6424a_port_toggle_bits,
447 	.pin_interrupt_configure = tca6424a_pin_interrupt_configure,
448 	.manage_callback = tca6424a_manage_callback,
449 };
450 
451 /**
452  * @brief Initialization function of TCA6424A
453  *
454  * This sets initial input/output configuration and output states.
455  * The interrupt is configured if it is enabled.
456  *
457  * @param dev Device struct
458  *
459  * @retval 0 if successful.
460  * @retval Negative value for error code.
461  */
tca6424a_init(const struct device * dev)462 static int tca6424a_init(const struct device *dev)
463 {
464 	const struct tca6424a_drv_cfg *drv_cfg = dev->config;
465 	struct tca6424a_drv_data *drv_data = dev->data;
466 	int ret;
467 
468 	if (!device_is_ready(drv_cfg->i2c_spec.bus)) {
469 		LOG_ERR("I2C device not found");
470 		return -ENODEV;
471 	}
472 	/* If the RESET line is available, use it to reset the expander.
473 	 * Otherwise, write reset values to registers that are not used by
474 	 * this driver.
475 	 */
476 	if (drv_cfg->reset_gpio.port) {
477 		if (!gpio_is_ready_dt(&drv_cfg->reset_gpio)) {
478 			LOG_ERR("%s is not ready", drv_cfg->reset_gpio.port->name);
479 			return -ENODEV;
480 		}
481 		ret = gpio_pin_configure_dt(&drv_cfg->reset_gpio, GPIO_OUTPUT_ACTIVE);
482 		if (ret != 0) {
483 			LOG_ERR("%s: failed to configure RESET line: %d", dev->name, ret);
484 			return ret;
485 		}
486 		/* RESET signal needs to be active for a minimum of 30 ns. */
487 		k_busy_wait(1);
488 
489 		ret = gpio_pin_set_dt(&drv_cfg->reset_gpio, 0);
490 		if (ret != 0) {
491 			LOG_ERR("%s: failed to deactivate RESET line: %d", dev->name, ret);
492 			return ret;
493 		}
494 		/* Give the expander at least 200 ns to recover after reset. */
495 		k_busy_wait(1);
496 	} else {
497 		ret = update_invers_regs(dev, 0x0);
498 		if (ret != 0) {
499 			LOG_ERR("%s: failed to reset inversion register: %d", dev->name, ret);
500 			return ret;
501 		}
502 	}
503 	/* Set initial configuration of the pins. */
504 	ret = update_config_regs(dev, 0xFFFFFF);
505 	if (ret != 0) {
506 		return ret;
507 	}
508 
509 	ret = update_output_regs(dev, 0x0);
510 	if (ret != 0) {
511 		return ret;
512 	}
513 
514 	/* Read initial state of the input port register. */
515 	ret = update_input_regs(dev, &drv_data->pins_state.input);
516 	if (ret != 0) {
517 		LOG_ERR("%s: failed to initially read input port: %d", dev->name, ret);
518 		return ret;
519 	}
520 
521 	/* If the INT line is available, configure the callback for it. */
522 	if (drv_cfg->int_gpio.port) {
523 		if (!gpio_is_ready_dt(&drv_cfg->int_gpio)) {
524 			LOG_ERR("Cannot get pointer to gpio interrupt device "
525 				"%s init failed", dev->name);
526 			return -EINVAL;
527 		}
528 
529 		drv_data->dev = dev;
530 
531 		k_work_init(&drv_data->work, tca6424a_work_handler);
532 
533 		ret = gpio_pin_configure_dt(&drv_cfg->int_gpio, GPIO_INPUT);
534 		if (ret != 0) {
535 			LOG_ERR("%s init failed: %d", dev->name, ret);
536 			return ret;
537 		}
538 
539 		ret = gpio_pin_interrupt_configure_dt(&drv_cfg->int_gpio, GPIO_INT_EDGE_TO_ACTIVE);
540 		if (ret != 0) {
541 			LOG_ERR("%s init failed: %d", dev->name, ret);
542 			return ret;
543 		}
544 
545 		gpio_init_callback(&drv_data->int_gpio_cb, tca6424a_int_gpio_handler,
546 				   BIT(drv_cfg->int_gpio.pin));
547 
548 		ret = gpio_add_callback(drv_cfg->int_gpio.port, &drv_data->int_gpio_cb);
549 		if (ret != 0) {
550 			LOG_ERR("%s init failed: %d", dev->name, ret);
551 			return ret;
552 		}
553 	}
554 
555 	LOG_DBG("%s init ok", dev->name);
556 	return ret;
557 }
558 
559 #define TCA6424A_INST(idx)                                                                         \
560 	static const struct tca6424a_drv_cfg tca6424a_cfg##idx = {                                 \
561 		.common = {                                                                        \
562 			.port_pin_mask = GPIO_PORT_PIN_MASK_FROM_DT_INST(idx),                     \
563 		},                                                                                 \
564 		.i2c_spec = I2C_DT_SPEC_INST_GET(idx),                                             \
565 		.int_gpio = GPIO_DT_SPEC_INST_GET_OR(idx, int_gpios, {0}),                         \
566 		.reset_gpio = GPIO_DT_SPEC_INST_GET_OR(idx, reset_gpios, {0}),                     \
567 	};                                                                                         \
568 	static struct tca6424a_drv_data tca6424a_data##idx = {                                     \
569 		.lock = Z_SEM_INITIALIZER(tca6424a_data##idx.lock, 1, 1),                          \
570 		.work = Z_WORK_INITIALIZER(tca6424a_work_handler),                                 \
571 		.dev = DEVICE_DT_INST_GET(idx),                                                    \
572 	};                                                                                         \
573 	DEVICE_DT_INST_DEFINE(idx, tca6424a_init, NULL, &tca6424a_data##idx, &tca6424a_cfg##idx,   \
574 			      POST_KERNEL, CONFIG_GPIO_TCA6424A_INIT_PRIORITY, &tca6424a_drv_api);
575 
576 DT_INST_FOREACH_STATUS_OKAY(TCA6424A_INST)
577