logo

qmk_firmware

custom branch of QMK firmware git clone https://anongit.hacktivis.me/git/qmk_firmware.git

is31fl3218-mono.c (5171B)


  1. /* Copyright 2018 Jason Williams (Wilba)
  2. *
  3. * This program is free software: you can redistribute it and/or modify
  4. * it under the terms of the GNU General Public License as published by
  5. * the Free Software Foundation, either version 2 of the License, or
  6. * (at your option) any later version.
  7. *
  8. * This program is distributed in the hope that it will be useful,
  9. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. * GNU General Public License for more details.
  12. *
  13. * You should have received a copy of the GNU General Public License
  14. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  15. */
  16. #include "is31fl3218-mono.h"
  17. #include "i2c_master.h"
  18. #include "gpio.h"
  19. #define IS31FL3218_PWM_REGISTER_COUNT 18
  20. #define IS31FL3218_LED_CONTROL_REGISTER_COUNT 3
  21. #ifndef IS31FL3218_I2C_TIMEOUT
  22. # define IS31FL3218_I2C_TIMEOUT 100
  23. #endif
  24. #ifndef IS31FL3218_I2C_PERSISTENCE
  25. # define IS31FL3218_I2C_PERSISTENCE 0
  26. #endif
  27. typedef struct is31fl3218_driver_t {
  28. uint8_t pwm_buffer[IS31FL3218_PWM_REGISTER_COUNT];
  29. bool pwm_buffer_dirty;
  30. uint8_t led_control_buffer[IS31FL3218_LED_CONTROL_REGISTER_COUNT];
  31. bool led_control_buffer_dirty;
  32. } PACKED is31fl3218_driver_t;
  33. // IS31FL3218 has 18 PWM outputs and a fixed I2C address, so no chaining.
  34. is31fl3218_driver_t driver_buffers = {
  35. .pwm_buffer = {0},
  36. .pwm_buffer_dirty = false,
  37. .led_control_buffer = {0},
  38. .led_control_buffer_dirty = false,
  39. };
  40. void is31fl3218_write_register(uint8_t reg, uint8_t data) {
  41. #if IS31FL3218_I2C_PERSISTENCE > 0
  42. for (uint8_t i = 0; i < IS31FL3218_I2C_PERSISTENCE; i++) {
  43. if (i2c_write_register(IS31FL3218_I2C_ADDRESS << 1, reg, &data, 1, IS31FL3218_I2C_TIMEOUT) == I2C_STATUS_SUCCESS) break;
  44. }
  45. #else
  46. i2c_write_register(IS31FL3218_I2C_ADDRESS << 1, reg, &data, 1, IS31FL3218_I2C_TIMEOUT);
  47. #endif
  48. }
  49. void is31fl3218_write_pwm_buffer(void) {
  50. #if IS31FL3218_I2C_PERSISTENCE > 0
  51. for (uint8_t i = 0; i < IS31FL3218_I2C_PERSISTENCE; i++) {
  52. if (i2c_write_register(IS31FL3218_I2C_ADDRESS << 1, IS31FL3218_REG_PWM, driver_buffers.pwm_buffer, 18, IS31FL3218_I2C_TIMEOUT) == I2C_STATUS_SUCCESS) break;
  53. }
  54. #else
  55. i2c_write_register(IS31FL3218_I2C_ADDRESS << 1, IS31FL3218_REG_PWM, driver_buffers.pwm_buffer, 18, IS31FL3218_I2C_TIMEOUT);
  56. #endif
  57. }
  58. void is31fl3218_init(void) {
  59. i2c_init();
  60. #if defined(IS31FL3218_SDB_PIN)
  61. gpio_set_pin_output(IS31FL3218_SDB_PIN);
  62. gpio_write_pin_high(IS31FL3218_SDB_PIN);
  63. #endif
  64. // In case we ever want to reinitialize (?)
  65. is31fl3218_write_register(IS31FL3218_REG_RESET, 0x00);
  66. // Turn off software shutdown
  67. is31fl3218_write_register(IS31FL3218_REG_SHUTDOWN, 0x01);
  68. // Set all PWM values to zero
  69. for (uint8_t i = 0; i < IS31FL3218_PWM_REGISTER_COUNT; i++) {
  70. is31fl3218_write_register(IS31FL3218_REG_PWM + i, 0x00);
  71. }
  72. // turn off all LEDs in the LED control register
  73. for (uint8_t i = 0; i < IS31FL3218_LED_CONTROL_REGISTER_COUNT; i++) {
  74. is31fl3218_write_register(IS31FL3218_REG_LED_CONTROL_1 + i, 0x00);
  75. }
  76. // Load PWM registers and LED Control register data
  77. is31fl3218_write_register(IS31FL3218_REG_UPDATE, 0x01);
  78. for (int i = 0; i < IS31FL3218_LED_COUNT; i++) {
  79. is31fl3218_set_led_control_register(i, true);
  80. }
  81. is31fl3218_update_led_control_registers();
  82. }
  83. void is31fl3218_set_value(int index, uint8_t value) {
  84. is31fl3218_led_t led;
  85. if (index >= 0 && index < IS31FL3218_LED_COUNT) {
  86. memcpy_P(&led, (&g_is31fl3218_leds[index]), sizeof(led));
  87. if (driver_buffers.pwm_buffer[led.v] == value) {
  88. return;
  89. }
  90. driver_buffers.pwm_buffer[led.v] = value;
  91. driver_buffers.pwm_buffer_dirty = true;
  92. }
  93. }
  94. void is31fl3218_set_value_all(uint8_t value) {
  95. for (int i = 0; i < IS31FL3218_LED_COUNT; i++) {
  96. is31fl3218_set_value(i, value);
  97. }
  98. }
  99. void is31fl3218_set_led_control_register(uint8_t index, bool value) {
  100. is31fl3218_led_t led;
  101. memcpy_P(&led, (&g_is31fl3218_leds[index]), sizeof(led));
  102. uint8_t control_register = led.v / 6;
  103. uint8_t bit_value = led.v % 6;
  104. if (value) {
  105. driver_buffers.led_control_buffer[control_register] |= (1 << bit_value);
  106. } else {
  107. driver_buffers.led_control_buffer[control_register] &= ~(1 << bit_value);
  108. }
  109. driver_buffers.led_control_buffer_dirty = true;
  110. }
  111. void is31fl3218_update_pwm_buffers(void) {
  112. if (driver_buffers.pwm_buffer_dirty) {
  113. is31fl3218_write_pwm_buffer();
  114. // Load PWM registers and LED Control register data
  115. is31fl3218_write_register(IS31FL3218_REG_UPDATE, 0x01);
  116. driver_buffers.pwm_buffer_dirty = false;
  117. }
  118. }
  119. void is31fl3218_update_led_control_registers(void) {
  120. if (driver_buffers.led_control_buffer_dirty) {
  121. for (uint8_t i = 0; i < IS31FL3218_LED_CONTROL_REGISTER_COUNT; i++) {
  122. is31fl3218_write_register(IS31FL3218_REG_LED_CONTROL_1 + i, driver_buffers.led_control_buffer[i]);
  123. }
  124. driver_buffers.led_control_buffer_dirty = false;
  125. }
  126. }