1 /*
2 * Copyright (c) 2018 Aapo Vienamo
3 * Copyright (c) 2018 Peter Bigot Consulting, LLC
4 * Copyright (c) 2019-2020 Nordic Semiconductor ASA
5 * Copyright (c) 2020 ZedBlox Ltd.
6 *
7 * SPDX-License-Identifier: Apache-2.0
8 */
9
10 #define DT_DRV_COMPAT semtech_sx1509b
11
12 #include <errno.h>
13
14 #include <zephyr/kernel.h>
15 #include <zephyr/device.h>
16 #include <zephyr/init.h>
17 #include <zephyr/drivers/gpio.h>
18 #include <zephyr/drivers/gpio/gpio_sx1509b.h>
19 #include <zephyr/dt-bindings/gpio/semtech-sx1509b.h>
20 #include <zephyr/drivers/i2c.h>
21 #include <zephyr/sys/byteorder.h>
22 #include <zephyr/sys/util.h>
23
24 #include <zephyr/logging/log.h>
25 LOG_MODULE_REGISTER(sx1509b, CONFIG_GPIO_LOG_LEVEL);
26
27 #include <zephyr/drivers/gpio/gpio_utils.h>
28
29 /* Number of pins supported by the device */
30 #define NUM_PINS 16
31
32 /* Max to select all pins supported on the device. */
33 #define ALL_PINS ((uint16_t)BIT_MASK(NUM_PINS))
34
35 /* Reset delay is 2.5 ms, round up for Zephyr resolution */
36 #define RESET_DELAY_MS 3
37
38 /** Cache of the output configuration and data of the pins. */
39 struct sx1509b_pin_state {
40 uint16_t input_disable; /* 0x00 */
41 uint16_t long_slew; /* 0x02 */
42 uint16_t low_drive; /* 0x04 */
43 uint16_t pull_up; /* 0x06 */
44 uint16_t pull_down; /* 0x08 */
45 uint16_t open_drain; /* 0x0A */
46 uint16_t polarity; /* 0x0C */
47 uint16_t dir; /* 0x0E */
48 uint16_t data; /* 0x10 */
49 } __packed;
50
51 struct sx1509b_irq_state {
52 uint16_t interrupt_mask; /* 0x12 */
53 uint32_t interrupt_sense; /* 0x14, 0x16 */
54 } __packed;
55
56 struct sx1509b_debounce_state {
57 uint8_t debounce_config; /* 0x22 */
58 uint16_t debounce_enable; /* 0x23 */
59 } __packed;
60
61 /** Runtime driver data */
62 struct sx1509b_drv_data {
63 /* gpio_driver_data needs to be first */
64 struct gpio_driver_data common;
65 struct sx1509b_pin_state pin_state;
66 uint16_t led_drv_enable;
67 struct sx1509b_debounce_state debounce_state;
68 struct k_sem lock;
69
70 #ifdef CONFIG_GPIO_SX1509B_INTERRUPT
71 struct gpio_callback gpio_cb;
72 struct k_work work;
73 struct sx1509b_irq_state irq_state;
74 const struct device *dev;
75 /* user ISR cb */
76 sys_slist_t cb;
77 #endif /* CONFIG_GPIO_SX1509B_INTERRUPT */
78
79 };
80
81 /** Configuration data */
82 struct sx1509b_config {
83 /* gpio_driver_config needs to be first */
84 struct gpio_driver_config common;
85 struct i2c_dt_spec bus;
86 #ifdef CONFIG_GPIO_SX1509B_INTERRUPT
87 struct gpio_dt_spec nint_gpio;
88 #endif /* CONFIG_GPIO_SX1509B_INTERRUPT */
89 };
90
91 /* General configuration register addresses */
92 enum {
93 /* TODO: Add rest of the regs */
94 SX1509B_REG_CLOCK = 0x1e,
95 SX1509B_REG_RESET = 0x7d,
96 };
97
98 /* Magic values for softreset */
99 enum {
100 SX1509B_REG_RESET_MAGIC0 = 0x12,
101 SX1509B_REG_RESET_MAGIC1 = 0x34,
102 };
103
104 /* Register bits for SX1509B_REG_CLOCK */
105 enum {
106 SX1509B_REG_CLOCK_FOSC_OFF = 0 << 5,
107 SX1509B_REG_CLOCK_FOSC_EXT = 1 << 5,
108 SX1509B_REG_CLOCK_FOSC_INT_2MHZ = 2 << 5,
109 };
110
111 /* Register bits for SX1509B_REG_MISC */
112 enum {
113 SX1509B_REG_MISC_LOG_A = 1 << 3,
114 SX1509B_REG_MISC_LOG_B = 1 << 7,
115 /* ClkX = fOSC */
116 SX1509B_REG_MISC_FREQ = 1 << 4,
117 };
118
119 /* Pin configuration register addresses */
120 enum {
121 SX1509B_REG_INPUT_DISABLE = 0x00,
122 SX1509B_REG_PULL_UP = 0x06,
123 SX1509B_REG_PULL_DOWN = 0x08,
124 SX1509B_REG_OPEN_DRAIN = 0x0a,
125 SX1509B_REG_DIR = 0x0e,
126 SX1509B_REG_DATA = 0x10,
127 SX1509B_REG_INTERRUPT_MASK = 0x12,
128 SX1509B_REG_INTERRUPT_SENSE = 0x14,
129 SX1509B_REG_INTERRUPT_SENSE_B = 0x14,
130 SX1509B_REG_INTERRUPT_SENSE_A = 0x16,
131 SX1509B_REG_INTERRUPT_SOURCE = 0x18,
132 SX1509B_REG_MISC = 0x1f,
133 SX1509B_REG_LED_DRV_ENABLE = 0x20,
134 SX1509B_REG_DEBOUNCE_CONFIG = 0x22,
135 SX1509B_REG_DEBOUNCE_ENABLE = 0x23,
136 };
137
138 /* Edge sensitivity types */
139 enum {
140 SX1509B_EDGE_NONE = 0x00,
141 SX1509B_EDGE_RISING = 0x01,
142 SX1509B_EDGE_FALLING = 0x02,
143 SX1509B_EDGE_BOTH = 0x03,
144 };
145
146 /* Intensity register addresses for all 16 pins */
147 static const uint8_t intensity_registers[16] = { 0x2a, 0x2d, 0x30, 0x33,
148 0x36, 0x3b, 0x40, 0x45,
149 0x4a, 0x4d, 0x50, 0x53,
150 0x56, 0x5b, 0x60, 0x65 };
151
152 /**
153 * @brief Write a big-endian word to an internal address of an I2C slave.
154 *
155 * @param dev Pointer to the I2C bus spec.
156 * @param reg_addr Address of the internal register being written.
157 * @param value Value to be written to internal register.
158 *
159 * @retval 0 If successful.
160 * @retval -EIO General input / output error.
161 */
i2c_reg_write_word_be(const struct i2c_dt_spec * bus,uint8_t reg_addr,uint16_t value)162 static inline int i2c_reg_write_word_be(const struct i2c_dt_spec *bus,
163 uint8_t reg_addr, uint16_t value)
164 {
165 uint8_t tx_buf[3] = { reg_addr, value >> 8, value & 0xff };
166
167 return i2c_write_dt(bus, tx_buf, 3);
168 }
169
170 /**
171 * @brief Write a big-endian byte to an internal address of an I2C slave.
172 *
173 * @param bus Pointer to the I2C bus spec.
174 * @param reg_addr Address of the internal register being written.
175 * @param value Value to be written to internal register.
176 *
177 * @retval 0 If successful.
178 * @retval -EIO General input / output error.
179 */
i2c_reg_write_byte_be(const struct i2c_dt_spec * bus,uint8_t reg_addr,uint8_t value)180 static inline int i2c_reg_write_byte_be(const struct i2c_dt_spec *bus,
181 uint8_t reg_addr, uint8_t value)
182 {
183 uint8_t tx_buf[3] = { reg_addr, value };
184
185 return i2c_write_dt(bus, tx_buf, 2);
186 }
187
188 #ifdef CONFIG_GPIO_SX1509B_INTERRUPT
sx1509b_handle_interrupt(const struct device * dev)189 static int sx1509b_handle_interrupt(const struct device *dev)
190 {
191 const struct sx1509b_config *cfg = dev->config;
192 struct sx1509b_drv_data *drv_data = dev->data;
193 int ret = 0;
194 uint16_t int_source;
195 uint8_t cmd = SX1509B_REG_INTERRUPT_SOURCE;
196
197 k_sem_take(&drv_data->lock, K_FOREVER);
198
199 ret = i2c_write_read_dt(&cfg->bus, &cmd, sizeof(cmd),
200 (uint8_t *)&int_source, sizeof(int_source));
201 if (ret != 0) {
202 goto out;
203 }
204
205 int_source = sys_be16_to_cpu(int_source);
206
207 /* reset interrupts before invoking callbacks */
208 ret = i2c_reg_write_word_be(&cfg->bus, SX1509B_REG_INTERRUPT_SOURCE,
209 int_source);
210
211 out:
212 k_sem_give(&drv_data->lock);
213
214 if (ret == 0) {
215 gpio_fire_callbacks(&drv_data->cb, dev, int_source);
216 }
217
218 return ret;
219 }
220
sx1509b_work_handler(struct k_work * work)221 static void sx1509b_work_handler(struct k_work *work)
222 {
223 struct sx1509b_drv_data *drv_data =
224 CONTAINER_OF(work, struct sx1509b_drv_data, work);
225
226 sx1509b_handle_interrupt(drv_data->dev);
227 }
228
sx1509_int_cb(const struct device * dev,struct gpio_callback * gpio_cb,uint32_t pins)229 static void sx1509_int_cb(const struct device *dev,
230 struct gpio_callback *gpio_cb,
231 uint32_t pins)
232 {
233 struct sx1509b_drv_data *drv_data = CONTAINER_OF(gpio_cb,
234 struct sx1509b_drv_data, gpio_cb);
235
236 ARG_UNUSED(pins);
237
238 k_work_submit(&drv_data->work);
239 }
240 #endif
241
write_pin_state(const struct sx1509b_config * cfg,struct sx1509b_drv_data * drv_data,struct sx1509b_pin_state * pins,bool data_first)242 static int write_pin_state(const struct sx1509b_config *cfg,
243 struct sx1509b_drv_data *drv_data,
244 struct sx1509b_pin_state *pins, bool data_first)
245 {
246 struct {
247 uint8_t reg;
248 struct sx1509b_pin_state pins;
249 } __packed pin_buf;
250 int rc;
251
252 pin_buf.reg = SX1509B_REG_INPUT_DISABLE;
253 pin_buf.pins.input_disable = sys_cpu_to_be16(pins->input_disable);
254 pin_buf.pins.long_slew = sys_cpu_to_be16(pins->long_slew);
255 pin_buf.pins.low_drive = sys_cpu_to_be16(pins->low_drive);
256 pin_buf.pins.pull_up = sys_cpu_to_be16(pins->pull_up);
257 pin_buf.pins.pull_down = sys_cpu_to_be16(pins->pull_down);
258 pin_buf.pins.open_drain = sys_cpu_to_be16(pins->open_drain);
259 pin_buf.pins.polarity = sys_cpu_to_be16(pins->polarity);
260 pin_buf.pins.dir = sys_cpu_to_be16(pins->dir);
261 pin_buf.pins.data = sys_cpu_to_be16(pins->data);
262
263 if (data_first) {
264 rc = i2c_reg_write_word_be(&cfg->bus, SX1509B_REG_DATA,
265 pins->data);
266 if (rc == 0) {
267 rc = i2c_write_dt(&cfg->bus, &pin_buf.reg,
268 sizeof(pin_buf) - sizeof(pins->data));
269 }
270 } else {
271 rc = i2c_write_dt(&cfg->bus, &pin_buf.reg, sizeof(pin_buf));
272 }
273
274 return rc;
275 }
276
sx1509b_config(const struct device * dev,gpio_pin_t pin,gpio_flags_t flags)277 static int sx1509b_config(const struct device *dev,
278 gpio_pin_t pin,
279 gpio_flags_t flags)
280 {
281 const struct sx1509b_config *cfg = dev->config;
282 struct sx1509b_drv_data *drv_data = dev->data;
283 struct sx1509b_pin_state *pins = &drv_data->pin_state;
284 struct sx1509b_debounce_state *debounce = &drv_data->debounce_state;
285 int rc = 0;
286 bool data_first = false;
287
288 /* Can't do I2C bus operations from an ISR */
289 if (k_is_in_isr()) {
290 return -EWOULDBLOCK;
291 }
292
293 k_sem_take(&drv_data->lock, K_FOREVER);
294
295 if (drv_data->led_drv_enable & BIT(pin)) {
296 /* Disable LED driver */
297 drv_data->led_drv_enable &= ~BIT(pin);
298 rc = i2c_reg_write_word_be(&cfg->bus,
299 SX1509B_REG_LED_DRV_ENABLE,
300 drv_data->led_drv_enable);
301
302 if (rc) {
303 goto out;
304 }
305 }
306
307 pins->open_drain &= ~BIT(pin);
308 if ((flags & GPIO_SINGLE_ENDED) != 0) {
309 if ((flags & GPIO_LINE_OPEN_DRAIN) != 0) {
310 pins->open_drain |= BIT(pin);
311 } else {
312 /* Open source not supported */
313 rc = -ENOTSUP;
314 goto out;
315 }
316 }
317
318 if ((flags & GPIO_PULL_UP) != 0) {
319 pins->pull_up |= BIT(pin);
320 } else {
321 pins->pull_up &= ~BIT(pin);
322 }
323 if ((flags & GPIO_PULL_DOWN) != 0) {
324 pins->pull_down |= BIT(pin);
325 } else {
326 pins->pull_down &= ~BIT(pin);
327 }
328
329 if ((flags & GPIO_INPUT) != 0) {
330 pins->input_disable &= ~BIT(pin);
331 } else {
332 pins->input_disable |= BIT(pin);
333 }
334
335 if ((flags & GPIO_OUTPUT) != 0) {
336 pins->dir &= ~BIT(pin);
337 if ((flags & GPIO_OUTPUT_INIT_LOW) != 0) {
338 pins->data &= ~BIT(pin);
339 data_first = true;
340 } else if ((flags & GPIO_OUTPUT_INIT_HIGH) != 0) {
341 pins->data |= BIT(pin);
342 data_first = true;
343 }
344 } else {
345 pins->dir |= BIT(pin);
346 }
347
348 if ((flags & SX1509B_GPIO_DEBOUNCE) != 0) {
349 debounce->debounce_enable |= BIT(pin);
350 } else {
351 debounce->debounce_enable &= ~BIT(pin);
352 }
353
354 LOG_DBG("CFG %u %x : ID %04x ; PU %04x ; PD %04x ; DIR %04x ; DAT %04x",
355 pin, flags,
356 pins->input_disable, pins->pull_up, pins->pull_down,
357 pins->dir, pins->data);
358
359 rc = write_pin_state(cfg, drv_data, pins, data_first);
360
361 if (rc == 0) {
362 struct {
363 uint8_t reg;
364 struct sx1509b_debounce_state debounce;
365 } __packed debounce_buf;
366
367 debounce_buf.reg = SX1509B_REG_DEBOUNCE_CONFIG;
368 debounce_buf.debounce.debounce_config
369 = debounce->debounce_config;
370 debounce_buf.debounce.debounce_enable
371 = sys_cpu_to_be16(debounce->debounce_enable);
372
373 rc = i2c_write_dt(&cfg->bus, &debounce_buf.reg,
374 sizeof(debounce_buf));
375 }
376
377 out:
378 k_sem_give(&drv_data->lock);
379 return rc;
380 }
381
port_get(const struct device * dev,gpio_port_value_t * value)382 static int port_get(const struct device *dev,
383 gpio_port_value_t *value)
384 {
385 const struct sx1509b_config *cfg = dev->config;
386 struct sx1509b_drv_data *drv_data = dev->data;
387 uint16_t pin_data;
388 int rc = 0;
389
390 /* Can't do I2C bus operations from an ISR */
391 if (k_is_in_isr()) {
392 return -EWOULDBLOCK;
393 }
394
395 k_sem_take(&drv_data->lock, K_FOREVER);
396
397 uint8_t cmd = SX1509B_REG_DATA;
398
399 rc = i2c_write_read_dt(&cfg->bus, &cmd, sizeof(cmd), &pin_data,
400 sizeof(pin_data));
401 LOG_DBG("read %04x got %d", sys_be16_to_cpu(pin_data), rc);
402 if (rc != 0) {
403 goto out;
404 }
405
406 *value = sys_be16_to_cpu(pin_data);
407
408 out:
409 k_sem_give(&drv_data->lock);
410 return rc;
411 }
412
port_write(const struct device * dev,gpio_port_pins_t mask,gpio_port_value_t value,gpio_port_value_t toggle)413 static int port_write(const struct device *dev,
414 gpio_port_pins_t mask,
415 gpio_port_value_t value,
416 gpio_port_value_t toggle)
417 {
418 /* Can't do I2C bus operations from an ISR */
419 if (k_is_in_isr()) {
420 return -EWOULDBLOCK;
421 }
422
423 const struct sx1509b_config *cfg = dev->config;
424 struct sx1509b_drv_data *drv_data = dev->data;
425 void *data = &drv_data->pin_state.data;
426 uint16_t *outp = data;
427
428 __ASSERT_NO_MSG(IS_PTR_ALIGNED(data, uint16_t));
429
430 k_sem_take(&drv_data->lock, K_FOREVER);
431
432 uint16_t orig_out = *outp;
433 uint16_t out = ((orig_out & ~mask) | (value & mask)) ^ toggle;
434 int rc = i2c_reg_write_word_be(&cfg->bus, SX1509B_REG_DATA, out);
435 if (rc == 0) {
436 *outp = out;
437 }
438
439 k_sem_give(&drv_data->lock);
440
441 LOG_DBG("write %04x msk %04x val %04x => %04x: %d", orig_out, mask, value, out, rc);
442
443 return rc;
444 }
445
port_set_masked(const struct device * dev,gpio_port_pins_t mask,gpio_port_value_t value)446 static int port_set_masked(const struct device *dev,
447 gpio_port_pins_t mask,
448 gpio_port_value_t value)
449 {
450 return port_write(dev, mask, value, 0);
451 }
452
port_set_bits(const struct device * dev,gpio_port_pins_t pins)453 static int port_set_bits(const struct device *dev,
454 gpio_port_pins_t pins)
455 {
456 return port_write(dev, pins, pins, 0);
457 }
458
port_clear_bits(const struct device * dev,gpio_port_pins_t pins)459 static int port_clear_bits(const struct device *dev,
460 gpio_port_pins_t pins)
461 {
462 return port_write(dev, pins, 0, 0);
463 }
464
port_toggle_bits(const struct device * dev,gpio_port_pins_t pins)465 static int port_toggle_bits(const struct device *dev,
466 gpio_port_pins_t pins)
467 {
468 return port_write(dev, 0, 0, pins);
469 }
470
471 #ifdef CONFIG_GPIO_SX1509B_INTERRUPT
pin_interrupt_configure(const struct device * dev,gpio_pin_t pin,enum gpio_int_mode mode,enum gpio_int_trig trig)472 static int pin_interrupt_configure(const struct device *dev,
473 gpio_pin_t pin,
474 enum gpio_int_mode mode,
475 enum gpio_int_trig trig)
476 {
477 int rc = 0;
478
479 /* Device does not support level-triggered interrupts. */
480 if (mode == GPIO_INT_MODE_LEVEL) {
481 return -ENOTSUP;
482 }
483
484 const struct sx1509b_config *cfg = dev->config;
485 struct sx1509b_drv_data *drv_data = dev->data;
486 struct sx1509b_irq_state *irq = &drv_data->irq_state;
487 struct {
488 uint8_t reg;
489 struct sx1509b_irq_state irq;
490 } __packed irq_buf;
491
492 /* Only level triggered interrupts are supported, and those
493 * only if interrupt support is enabled.
494 */
495 if (IS_ENABLED(CONFIG_GPIO_SX1509B_INTERRUPT)) {
496 if (mode == GPIO_INT_MODE_LEVEL) {
497 return -ENOTSUP;
498 }
499 } else if (mode != GPIO_INT_MODE_DISABLED) {
500 return -ENOTSUP;
501 }
502
503 k_sem_take(&drv_data->lock, K_FOREVER);
504
505 irq->interrupt_sense &= ~(SX1509B_EDGE_BOTH << (pin * 2));
506 if (mode == GPIO_INT_MODE_DISABLED) {
507 irq->interrupt_mask |= BIT(pin);
508 } else { /* GPIO_INT_MODE_EDGE */
509 irq->interrupt_mask &= ~BIT(pin);
510 if (trig == GPIO_INT_TRIG_BOTH) {
511 irq->interrupt_sense |= (SX1509B_EDGE_BOTH <<
512 (pin * 2));
513 } else if (trig == GPIO_INT_TRIG_LOW) {
514 irq->interrupt_sense |= (SX1509B_EDGE_FALLING <<
515 (pin * 2));
516 } else if (trig == GPIO_INT_TRIG_HIGH) {
517 irq->interrupt_sense |= (SX1509B_EDGE_RISING <<
518 (pin * 2));
519 }
520 }
521
522 irq_buf.reg = SX1509B_REG_INTERRUPT_MASK;
523 irq_buf.irq.interrupt_mask = sys_cpu_to_be16(irq->interrupt_mask);
524 irq_buf.irq.interrupt_sense = sys_cpu_to_be32(irq->interrupt_sense);
525
526 rc = i2c_write_dt(&cfg->bus, &irq_buf.reg, sizeof(irq_buf));
527
528 k_sem_give(&drv_data->lock);
529
530 return rc;
531 }
532 #endif /* CONFIG_GPIO_SX1509B_INTERRUPT */
533
534 /**
535 * @brief Initialization function of SX1509B
536 *
537 * @param dev Device struct
538 * @return 0 if successful, failed otherwise.
539 */
sx1509b_init(const struct device * dev)540 static int sx1509b_init(const struct device *dev)
541 {
542 const struct sx1509b_config *cfg = dev->config;
543 struct sx1509b_drv_data *drv_data = dev->data;
544 int rc;
545
546 if (!device_is_ready(cfg->bus.bus)) {
547 LOG_ERR("I2C bus not ready");
548 rc = -ENODEV;
549 goto out;
550 }
551
552 #ifdef CONFIG_GPIO_SX1509B_INTERRUPT
553 drv_data->dev = dev;
554
555 if (!gpio_is_ready_dt(&cfg->nint_gpio)) {
556 rc = -ENODEV;
557 goto out;
558 }
559 k_work_init(&drv_data->work, sx1509b_work_handler);
560
561 gpio_pin_configure_dt(&cfg->nint_gpio, GPIO_INPUT);
562 gpio_pin_interrupt_configure_dt(&cfg->nint_gpio,
563 GPIO_INT_EDGE_TO_ACTIVE);
564
565 gpio_init_callback(&drv_data->gpio_cb, sx1509_int_cb,
566 BIT(cfg->nint_gpio.pin));
567 gpio_add_callback(cfg->nint_gpio.port, &drv_data->gpio_cb);
568
569 drv_data->irq_state = (struct sx1509b_irq_state) {
570 .interrupt_mask = ALL_PINS,
571 };
572 #endif
573
574 rc = i2c_reg_write_byte_dt(&cfg->bus, SX1509B_REG_RESET,
575 SX1509B_REG_RESET_MAGIC0);
576 if (rc != 0) {
577 LOG_ERR("%s: reset m0 failed: %d\n", dev->name, rc);
578 goto out;
579 }
580 rc = i2c_reg_write_byte_dt(&cfg->bus, SX1509B_REG_RESET,
581 SX1509B_REG_RESET_MAGIC1);
582 if (rc != 0) {
583 goto out;
584 }
585
586 k_sleep(K_MSEC(RESET_DELAY_MS));
587
588 /* Reset state mediated by initial configuration */
589 drv_data->pin_state = (struct sx1509b_pin_state) {
590 .dir = (ALL_PINS
591 & ~(DT_INST_PROP(0, init_out_low)
592 | DT_INST_PROP(0, init_out_high))),
593 .data = (ALL_PINS
594 & ~DT_INST_PROP(0, init_out_low)),
595 };
596 drv_data->debounce_state = (struct sx1509b_debounce_state) {
597 .debounce_config = CONFIG_GPIO_SX1509B_DEBOUNCE_TIME,
598 };
599
600 rc = i2c_reg_write_byte_dt(&cfg->bus, SX1509B_REG_CLOCK,
601 SX1509B_REG_CLOCK_FOSC_INT_2MHZ);
602 if (rc == 0) {
603 rc = i2c_reg_write_word_be(&cfg->bus, SX1509B_REG_DATA,
604 drv_data->pin_state.data);
605 }
606 if (rc == 0) {
607 rc = i2c_reg_write_word_be(&cfg->bus, SX1509B_REG_DIR,
608 drv_data->pin_state.dir);
609 }
610 if (rc == 0) {
611 rc = i2c_reg_write_byte_be(&cfg->bus, SX1509B_REG_MISC,
612 SX1509B_REG_MISC_LOG_A |
613 SX1509B_REG_MISC_LOG_B |
614 SX1509B_REG_MISC_FREQ);
615 }
616
617 out:
618 if (rc != 0) {
619 LOG_ERR("%s init failed: %d", dev->name, rc);
620 } else {
621 LOG_INF("%s init ok", dev->name);
622 }
623 k_sem_give(&drv_data->lock);
624 return rc;
625 }
626
627 #ifdef CONFIG_GPIO_SX1509B_INTERRUPT
gpio_sx1509b_manage_callback(const struct device * dev,struct gpio_callback * callback,bool set)628 static int gpio_sx1509b_manage_callback(const struct device *dev,
629 struct gpio_callback *callback,
630 bool set)
631 {
632 struct sx1509b_drv_data *data = dev->data;
633
634 return gpio_manage_callback(&data->cb, callback, set);
635 }
636 #endif
637
638 static DEVICE_API(gpio, api_table) = {
639 .pin_configure = sx1509b_config,
640 .port_get_raw = port_get,
641 .port_set_masked_raw = port_set_masked,
642 .port_set_bits_raw = port_set_bits,
643 .port_clear_bits_raw = port_clear_bits,
644 .port_toggle_bits = port_toggle_bits,
645 #ifdef CONFIG_GPIO_SX1509B_INTERRUPT
646 .pin_interrupt_configure = pin_interrupt_configure,
647 .manage_callback = gpio_sx1509b_manage_callback,
648 #endif
649 };
650
sx1509b_led_intensity_pin_configure(const struct device * dev,gpio_pin_t pin)651 int sx1509b_led_intensity_pin_configure(const struct device *dev,
652 gpio_pin_t pin)
653 {
654 const struct sx1509b_config *cfg = dev->config;
655 struct sx1509b_drv_data *drv_data = dev->data;
656 struct sx1509b_pin_state *pins = &drv_data->pin_state;
657 int rc;
658
659 /* Can't do I2C bus operations from an ISR */
660 if (k_is_in_isr()) {
661 return -EWOULDBLOCK;
662 }
663
664 if (pin >= ARRAY_SIZE(intensity_registers)) {
665 return -ERANGE;
666 }
667
668 k_sem_take(&drv_data->lock, K_FOREVER);
669
670 /* Enable LED driver */
671 drv_data->led_drv_enable |= BIT(pin);
672 rc = i2c_reg_write_word_be(&cfg->bus, SX1509B_REG_LED_DRV_ENABLE,
673 drv_data->led_drv_enable);
674
675 /* Set intensity to 0 */
676 if (rc == 0) {
677 rc = i2c_reg_write_byte_be(&cfg->bus, intensity_registers[pin], 0);
678 } else {
679 goto out;
680 }
681
682 pins->input_disable |= BIT(pin);
683 pins->pull_up &= ~BIT(pin);
684 pins->dir &= ~BIT(pin);
685 pins->data &= ~BIT(pin);
686
687 if (rc == 0) {
688 rc = write_pin_state(cfg, drv_data, pins, false);
689 }
690
691 out:
692 k_sem_give(&drv_data->lock);
693 return rc;
694 }
695
sx1509b_led_intensity_pin_set(const struct device * dev,gpio_pin_t pin,uint8_t intensity_val)696 int sx1509b_led_intensity_pin_set(const struct device *dev, gpio_pin_t pin,
697 uint8_t intensity_val)
698 {
699 const struct sx1509b_config *cfg = dev->config;
700 struct sx1509b_drv_data *drv_data = dev->data;
701 int rc;
702
703 /* Can't do I2C bus operations from an ISR */
704 if (k_is_in_isr()) {
705 return -EWOULDBLOCK;
706 }
707
708 if (pin >= ARRAY_SIZE(intensity_registers)) {
709 return -ERANGE;
710 }
711
712 k_sem_take(&drv_data->lock, K_FOREVER);
713
714 rc = i2c_reg_write_byte_be(&cfg->bus, intensity_registers[pin],
715 intensity_val);
716
717 k_sem_give(&drv_data->lock);
718
719 return rc;
720 }
721
722 #define GPIO_SX1509B_DEFINE(inst) \
723 static const struct sx1509b_config sx1509b_cfg##inst = { \
724 .common = { \
725 .port_pin_mask = GPIO_PORT_PIN_MASK_FROM_DT_INST(inst),\
726 }, \
727 .bus = I2C_DT_SPEC_INST_GET(inst), \
728 IF_ENABLED(CONFIG_GPIO_SX1509B_INTERRUPT, \
729 (GPIO_DT_SPEC_INST_GET(inst, nint_gpios))) \
730 }; \
731 \
732 static struct sx1509b_drv_data sx1509b_drvdata##inst = { \
733 .lock = Z_SEM_INITIALIZER(sx1509b_drvdata##inst.lock, 1, 1), \
734 }; \
735 \
736 DEVICE_DT_INST_DEFINE(inst, sx1509b_init, NULL, &sx1509b_drvdata##inst,\
737 &sx1509b_cfg##inst, POST_KERNEL, \
738 CONFIG_GPIO_SX1509B_INIT_PRIORITY, &api_table);
739
740 DT_INST_FOREACH_STATUS_OKAY(GPIO_SX1509B_DEFINE)
741