You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

108 lines
3.0 KiB

  1. // Copyright 2022 zvecr<git@zvecr.com>
  2. // SPDX-License-Identifier: GPL-2.0-or-later
  3. #include "mcp23018.h"
  4. #include "i2c_master.h"
  5. #include "wait.h"
  6. #include "debug.h"
  7. #define SLAVE_TO_ADDR(n) (n << 1)
  8. #define TIMEOUT 100
  9. enum {
  10. CMD_IODIRA = 0x00, // i/o direction register
  11. CMD_IODIRB = 0x01,
  12. CMD_GPPUA = 0x0C, // GPIO pull-up resistor register
  13. CMD_GPPUB = 0x0D,
  14. CMD_GPIOA = 0x12, // general purpose i/o port register (write modifies OLAT)
  15. CMD_GPIOB = 0x13,
  16. };
  17. void mcp23018_init(uint8_t addr) {
  18. static uint8_t s_init = 0;
  19. if (!s_init) {
  20. i2c_init();
  21. wait_ms(1000);
  22. s_init = 1;
  23. }
  24. }
  25. bool mcp23018_set_config(uint8_t slave_addr, mcp23018_port_t port, uint8_t conf) {
  26. uint8_t addr = SLAVE_TO_ADDR(slave_addr);
  27. uint8_t cmdDirection = port ? CMD_IODIRB : CMD_IODIRA;
  28. uint8_t cmdPullup = port ? CMD_GPPUB : CMD_GPPUA;
  29. i2c_status_t ret = i2c_writeReg(addr, cmdDirection, &conf, sizeof(conf), TIMEOUT);
  30. if (ret != I2C_STATUS_SUCCESS) {
  31. dprintf("mcp23018_set_config::directionFAILED::%u\n", ret);
  32. return false;
  33. }
  34. ret = i2c_writeReg(addr, cmdPullup, &conf, sizeof(conf), TIMEOUT);
  35. if (ret != I2C_STATUS_SUCCESS) {
  36. dprintf("mcp23018_set_config::pullupFAILED::%u\n", ret);
  37. return false;
  38. }
  39. return true;
  40. }
  41. bool mcp23018_set_output(uint8_t slave_addr, mcp23018_port_t port, uint8_t conf) {
  42. uint8_t addr = SLAVE_TO_ADDR(slave_addr);
  43. uint8_t cmd = port ? CMD_GPIOB : CMD_GPIOA;
  44. i2c_status_t ret = i2c_writeReg(addr, cmd, &conf, sizeof(conf), TIMEOUT);
  45. if (ret != I2C_STATUS_SUCCESS) {
  46. dprintf("mcp23018_set_output::FAILED::%u\n", ret);
  47. return false;
  48. }
  49. return true;
  50. }
  51. bool mcp23018_set_output_all(uint8_t slave_addr, uint8_t confA, uint8_t confB) {
  52. uint8_t addr = SLAVE_TO_ADDR(slave_addr);
  53. uint8_t conf[2] = {confA, confB};
  54. i2c_status_t ret = i2c_writeReg(addr, CMD_GPIOA, &conf[0], sizeof(conf), TIMEOUT);
  55. if (ret != I2C_STATUS_SUCCESS) {
  56. dprintf("mcp23018_set_output::FAILED::%u\n", ret);
  57. return false;
  58. }
  59. return true;
  60. }
  61. bool mcp23018_readPins(uint8_t slave_addr, mcp23018_port_t port, uint8_t* out) {
  62. uint8_t addr = SLAVE_TO_ADDR(slave_addr);
  63. uint8_t cmd = port ? CMD_GPIOB : CMD_GPIOA;
  64. i2c_status_t ret = i2c_readReg(addr, cmd, out, sizeof(uint8_t), TIMEOUT);
  65. if (ret != I2C_STATUS_SUCCESS) {
  66. dprintf("mcp23018_readPins::FAILED::%u\n", ret);
  67. return false;
  68. }
  69. return true;
  70. }
  71. bool mcp23018_readPins_all(uint8_t slave_addr, uint16_t* out) {
  72. uint8_t addr = SLAVE_TO_ADDR(slave_addr);
  73. typedef union {
  74. uint8_t u8[2];
  75. uint16_t u16;
  76. } data16;
  77. data16 data = {.u16 = 0};
  78. i2c_status_t ret = i2c_readReg(addr, CMD_GPIOA, &data.u8[0], sizeof(data), TIMEOUT);
  79. if (ret != I2C_STATUS_SUCCESS) {
  80. dprintf("mcp23018_readPins::FAILED::%u\n", ret);
  81. return false;
  82. }
  83. *out = data.u16;
  84. return true;
  85. }