logo

qmk_firmware

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

ergodox_ez.c (13933B)


  1. /*
  2. Copyright 2012 Jun Wako <wakojun@gmail.com>
  3. Copyright 2013 Oleg Kostyuk <cub.uanic@gmail.com>
  4. Copyright 2015 ZSA Technology Labs Inc (@zsa)
  5. Copyright 2020 Christopher Courtney <drashna@live.com> (@drashna)
  6. This program is free software: you can redistribute it and/or modify
  7. it under the terms of the GNU General Public License as published by
  8. the Free Software Foundation, either version 2 of the License, or
  9. (at your option) any later version.
  10. This program is distributed in the hope that it will be useful,
  11. but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. GNU General Public License for more details.
  14. You should have received a copy of the GNU General Public License
  15. along with this program. If not, see <http://www.gnu.org/licenses/>.
  16. */
  17. #include "ergodox_ez.h"
  18. extern inline void ergodox_board_led_on(void);
  19. extern inline void ergodox_right_led_1_on(void);
  20. extern inline void ergodox_right_led_2_on(void);
  21. extern inline void ergodox_right_led_3_on(void);
  22. extern inline void ergodox_board_led_off(void);
  23. extern inline void ergodox_right_led_1_off(void);
  24. extern inline void ergodox_right_led_2_off(void);
  25. extern inline void ergodox_right_led_3_off(void);
  26. extern inline void ergodox_led_all_on(void);
  27. extern inline void ergodox_led_all_off(void);
  28. extern inline void ergodox_right_led_1_set(uint8_t n);
  29. extern inline void ergodox_right_led_2_set(uint8_t n);
  30. extern inline void ergodox_right_led_3_set(uint8_t n);
  31. extern inline void ergodox_right_led_set(uint8_t led, uint8_t n);
  32. extern inline void ergodox_led_all_set(uint8_t n);
  33. keyboard_config_t keyboard_config;
  34. bool i2c_initialized = 0;
  35. i2c_status_t mcp23018_status = 0x20;
  36. void matrix_init_kb(void) {
  37. // keyboard LEDs (see "PWM on ports OC1(A|B|C)" in "teensy-2-0.md")
  38. TCCR1A = 0b10101001; // set and configure fast PWM
  39. TCCR1B = 0b00001001; // set and configure fast PWM
  40. // (tied to Vcc for hardware convenience)
  41. gpio_set_pin_input(B4); // set B(4) as input, internal pull-up disabled
  42. // unused pins - C7, D4, D5, E6
  43. // set as input with internal pull-up enabled
  44. gpio_set_pin_input_high(C7);
  45. gpio_set_pin_input_high(D4);
  46. gpio_set_pin_input_high(D5);
  47. gpio_set_pin_input_high(E6);
  48. keyboard_config.raw = eeconfig_read_kb();
  49. ergodox_led_all_set((uint8_t)keyboard_config.led_level * 255 / 4);
  50. #ifdef RGB_MATRIX_ENABLE
  51. if (keyboard_config.rgb_matrix_enable) {
  52. rgb_matrix_set_flags(LED_FLAG_ALL);
  53. } else {
  54. rgb_matrix_set_flags(LED_FLAG_NONE);
  55. }
  56. #endif
  57. ergodox_blink_all_leds();
  58. matrix_init_user();
  59. }
  60. void ergodox_blink_all_leds(void) {
  61. ergodox_led_all_off();
  62. ergodox_led_all_set(LED_BRIGHTNESS_DEFAULT);
  63. ergodox_right_led_1_on();
  64. _delay_ms(50);
  65. ergodox_right_led_2_on();
  66. _delay_ms(50);
  67. ergodox_right_led_3_on();
  68. _delay_ms(50);
  69. #ifdef LEFT_LEDS
  70. ergodox_left_led_1_on();
  71. _delay_ms(50);
  72. if (!mcp23018_status) {
  73. mcp23018_status = ergodox_left_leds_update();
  74. }
  75. ergodox_left_led_2_on();
  76. _delay_ms(50);
  77. if (!mcp23018_status) {
  78. mcp23018_status = ergodox_left_leds_update();
  79. }
  80. ergodox_left_led_3_on();
  81. _delay_ms(50);
  82. if (!mcp23018_status) {
  83. mcp23018_status = ergodox_left_leds_update();
  84. }
  85. #endif
  86. ergodox_right_led_1_off();
  87. _delay_ms(50);
  88. ergodox_right_led_2_off();
  89. _delay_ms(50);
  90. ergodox_right_led_3_off();
  91. #ifdef LEFT_LEDS
  92. _delay_ms(50);
  93. ergodox_left_led_1_off();
  94. if (!mcp23018_status) {
  95. mcp23018_status = ergodox_left_leds_update();
  96. }
  97. _delay_ms(50);
  98. ergodox_left_led_2_off();
  99. if (!mcp23018_status) {
  100. mcp23018_status = ergodox_left_leds_update();
  101. }
  102. _delay_ms(50);
  103. ergodox_left_led_3_off();
  104. if (!mcp23018_status) {
  105. mcp23018_status = ergodox_left_leds_update();
  106. }
  107. #endif
  108. // ergodox_led_all_on();
  109. //_delay_ms(333);
  110. ergodox_led_all_off();
  111. }
  112. uint8_t init_mcp23018(void) {
  113. mcp23018_status = 0x20;
  114. // I2C subsystem
  115. // uint8_t sreg_prev;
  116. // sreg_prev=SREG;
  117. // cli();
  118. if (i2c_initialized == 0) {
  119. i2c_init(); // on pins D(1,0)
  120. i2c_initialized = true;
  121. _delay_ms(1000);
  122. }
  123. // i2c_init(); // on pins D(1,0)
  124. // _delay_ms(1000);
  125. // set pin direction
  126. // - unused : input : 1
  127. // - input : input : 1
  128. // - driving : output : 0
  129. uint8_t data[] = {0b00000000, 0b00111111};
  130. mcp23018_status = i2c_write_register(I2C_ADDR, IODIRA, data, 2, ERGODOX_EZ_I2C_TIMEOUT);
  131. if (!mcp23018_status) {
  132. // set pull-up
  133. // - unused : on : 1
  134. // - input : on : 1
  135. // - driving : off : 0
  136. mcp23018_status = i2c_write_register(I2C_ADDR, GPPUA, data, 2, ERGODOX_EZ_I2C_TIMEOUT);
  137. }
  138. #ifdef LEFT_LEDS
  139. if (!mcp23018_status) mcp23018_status = ergodox_left_leds_update();
  140. #endif // LEFT_LEDS
  141. // SREG=sreg_prev;
  142. return mcp23018_status;
  143. }
  144. #ifdef LEFT_LEDS
  145. uint8_t ergodox_left_leds_update(void) {
  146. if (mcp23018_status) { // if there was an error
  147. return mcp23018_status;
  148. }
  149. # define LEFT_LED_1_SHIFT 7 // in MCP23018 port B
  150. # define LEFT_LED_2_SHIFT 6 // in MCP23018 port B
  151. # define LEFT_LED_3_SHIFT 7 // in MCP23018 port A
  152. // set logical value (doesn't matter on inputs)
  153. // - unused : hi-Z : 1
  154. // - input : hi-Z : 1
  155. // - driving : hi-Z : 1
  156. uint8_t data[2];
  157. data[0] = 0b11111111 & ~(ergodox_left_led_3 << LEFT_LED_3_SHIFT);
  158. data[1] = 0b11111111 & ~(ergodox_left_led_2 << LEFT_LED_2_SHIFT) & ~(ergodox_left_led_1 << LEFT_LED_1_SHIFT);
  159. mcp23018_status = i2c_write_register(I2C_ADDR, OLATA, data, 2, ERGODOX_EZ_I2C_TIMEOUT);
  160. return mcp23018_status;
  161. }
  162. #endif
  163. #ifdef SWAP_HANDS_ENABLE
  164. __attribute__((weak))
  165. // swap-hands action needs a matrix to define the swap
  166. const keypos_t PROGMEM hand_swap_config[MATRIX_ROWS][MATRIX_COLS] = {
  167. /* Left hand, matrix positions */
  168. {{0, 13}, {1, 13}, {2, 13}, {3, 13}, {4, 13}, {5, 13}},
  169. {{0, 12}, {1, 12}, {2, 12}, {3, 12}, {4, 12}, {5, 12}},
  170. {{0, 11}, {1, 11}, {2, 11}, {3, 11}, {4, 11}, {5, 11}},
  171. {{0, 10}, {1, 10}, {2, 10}, {3, 10}, {4, 10}, {5, 10}},
  172. {{0, 9}, {1, 9}, {2, 9}, {3, 9}, {4, 9}, {5, 9}},
  173. {{0, 8}, {1, 8}, {2, 8}, {3, 8}, {4, 8}, {5, 8}},
  174. {{0, 7}, {1, 7}, {2, 7}, {3, 7}, {4, 7}, {5, 7}},
  175. /* Right hand, matrix positions */
  176. {{0, 6}, {1, 6}, {2, 6}, {3, 6}, {4, 6}, {5, 6}},
  177. {{0, 5}, {1, 5}, {2, 5}, {3, 5}, {4, 5}, {5, 5}},
  178. {{0, 4}, {1, 4}, {2, 4}, {3, 4}, {4, 4}, {5, 4}},
  179. {{0, 3}, {1, 3}, {2, 3}, {3, 3}, {4, 3}, {5, 3}},
  180. {{0, 2}, {1, 2}, {2, 2}, {3, 2}, {4, 2}, {5, 2}},
  181. {{0, 1}, {1, 1}, {2, 1}, {3, 1}, {4, 1}, {5, 1}},
  182. {{0, 0}, {1, 0}, {2, 0}, {3, 0}, {4, 0}, {5, 0}},
  183. };
  184. #endif
  185. #ifdef RGB_MATRIX_ENABLE
  186. // clang-format off
  187. const is31fl3731_led_t PROGMEM g_is31fl3731_leds[IS31FL3731_LED_COUNT] = {
  188. /* driver
  189. * | R location
  190. * | | G location
  191. * | | | B location
  192. * | | | | */
  193. {0, C3_1, C2_1, C4_1}, // LED1 on right
  194. {0, C6_1, C5_1, C7_1}, // LED2
  195. {0, C4_2, C3_2, C5_2}, // LED3
  196. {0, C7_2, C6_2, C8_2}, // LED4
  197. {0, C2_3, C1_3, C3_3}, // LED5
  198. {0, C5_3, C4_3, C6_3}, // LED6
  199. {0, C8_3, C7_3, C9_3}, // LED7
  200. {0, C2_4, C1_4, C3_4}, // LED8
  201. {0, C6_4, C5_4, C7_4}, // LED9
  202. {0, C2_5, C1_5, C3_5}, // LED10
  203. {0, C7_5, C6_5, C8_5}, // LED11
  204. {0, C2_6, C1_6, C3_6}, // LED12
  205. {0, C5_6, C4_6, C6_6}, // LED13
  206. {0, C8_6, C7_6, C9_6}, // LED14
  207. {0, C2_7, C1_7, C3_7}, // LED15
  208. {0, C5_7, C4_7, C6_7}, // LED16
  209. {0, C2_8, C1_8, C3_8}, // LED17
  210. {0, C5_8, C4_8, C6_8}, // LED18
  211. {0, C3_9, C2_9, C4_9}, // LED19
  212. {0, C6_9, C5_9, C7_9}, // LED20
  213. {0, C4_10, C3_10, C5_10}, // LED21
  214. {0, C7_10, C6_10, C8_10}, // LED22
  215. {0, C2_11, C1_11, C3_11}, // LED23
  216. {0, C5_11, C4_11, C6_11}, // LED24
  217. {1, C3_1, C2_1, C4_1}, // LED1 on left
  218. {1, C6_1, C5_1, C7_1}, // LED2
  219. {1, C4_2, C3_2, C5_2}, // LED3
  220. {1, C7_2, C6_2, C8_2}, // LED4
  221. {1, C2_3, C1_3, C3_3}, // LED5
  222. {1, C5_3, C4_3, C6_3}, // LED6
  223. {1, C8_3, C7_3, C9_3}, // LED7
  224. {1, C2_4, C1_4, C3_4}, // LED8
  225. {1, C6_4, C5_4, C7_4}, // LED9
  226. {1, C2_5, C1_5, C3_5}, // LED10
  227. {1, C7_5, C6_5, C8_5}, // LED11
  228. {1, C2_6, C1_6, C3_6}, // LED12
  229. {1, C5_6, C4_6, C6_6}, // LED13
  230. {1, C8_6, C7_6, C9_6}, // LED14
  231. {1, C2_7, C1_7, C3_7}, // LED15
  232. {1, C5_7, C4_7, C6_7}, // LED16
  233. {1, C2_8, C1_8, C3_8}, // LED17
  234. {1, C5_8, C4_8, C6_8}, // LED18
  235. {1, C3_9, C2_9, C4_9}, // LED19
  236. {1, C6_9, C5_9, C7_9}, // LED20
  237. {1, C4_10, C3_10, C5_10}, // LED21
  238. {1, C7_10, C6_10, C8_10}, // LED22
  239. {1, C2_11, C1_11, C3_11}, // LED23
  240. {1, C5_11, C4_11, C6_11} // LED24
  241. };
  242. led_config_t g_led_config = { {
  243. { NO_LED, NO_LED, NO_LED, NO_LED, NO_LED, NO_LED },
  244. { 28, 33, 38, 43, 47, NO_LED },
  245. { 27, 32, 37, 42, 46, NO_LED },
  246. { 26, 31, 36, 41, 45, NO_LED },
  247. { 25, 30, 35, 40, 44, NO_LED },
  248. { 24, 29, 34, 39, NO_LED, NO_LED },
  249. { NO_LED, NO_LED, NO_LED, NO_LED, NO_LED, NO_LED },
  250. { NO_LED, NO_LED, NO_LED, NO_LED, NO_LED, NO_LED },
  251. { 0, 5, 10, 15, NO_LED, NO_LED },
  252. { 1, 6, 11, 16, 20, NO_LED },
  253. { 2, 7, 12, 17, 21, NO_LED },
  254. { 3, 8, 13, 18, 22, NO_LED },
  255. { 4, 9, 14, 19, 23, NO_LED },
  256. { NO_LED, NO_LED, NO_LED, NO_LED, NO_LED, NO_LED }
  257. }, {
  258. { 137, 0 }, { 154, 0 }, { 172, 0 }, { 189, 0 }, { 206, 0 }, { 137, 12 },
  259. { 154, 12 }, { 172, 12 }, { 189, 12 }, { 206, 12 }, { 137, 25 }, { 154, 25 },
  260. { 172, 25 }, { 189, 25 }, { 206, 25 }, { 137, 38 }, { 154, 38 }, { 172, 38 },
  261. { 189, 38 }, { 206, 38 }, { 154, 51 }, { 172, 51 }, { 189, 51 }, { 206, 51 },
  262. { 86, 0 }, { 68, 0 }, { 51, 0 }, { 34, 0 }, { 17, 0 }, { 86, 12 },
  263. { 68, 12 }, { 51, 12 }, { 34, 12 }, { 17, 12 }, { 86, 25 }, { 68, 25 },
  264. { 51, 25 }, { 34, 25 }, { 17, 25 }, { 86, 38 }, { 68, 38 }, { 51, 38 },
  265. { 34, 38 }, { 17, 38 }, { 68, 51 }, { 51, 51 }, { 34, 51 }, { 17, 51 }
  266. }, {
  267. 4, 4, 4, 4, 4, 4,
  268. 4, 4, 4, 4, 4, 4,
  269. 4, 4, 4, 4, 4, 4,
  270. 4, 4, 1, 1, 1, 1,
  271. 4, 4, 4, 4, 4, 4,
  272. 4, 4, 4, 4, 4, 4,
  273. 4, 4, 4, 4, 4, 4,
  274. 4, 4, 1, 1, 1, 1
  275. } };
  276. // clang-format on
  277. # ifdef ORYX_CONFIGURATOR
  278. void keyboard_post_init_kb(void) {
  279. rgb_matrix_enable_noeeprom();
  280. keyboard_post_init_user();
  281. }
  282. # endif
  283. #endif
  284. #ifdef ORYX_CONFIGURATOR
  285. bool process_record_kb(uint16_t keycode, keyrecord_t *record) {
  286. switch (keycode) {
  287. case LED_LEVEL:
  288. if (record->event.pressed) {
  289. keyboard_config.led_level++;
  290. if (keyboard_config.led_level > 4) {
  291. keyboard_config.led_level = 0;
  292. }
  293. ergodox_led_all_set((uint8_t)keyboard_config.led_level * 255 / 4);
  294. eeconfig_update_kb(keyboard_config.raw);
  295. layer_state_set_kb(layer_state);
  296. }
  297. break;
  298. # ifdef RGB_MATRIX_ENABLE
  299. case TOGGLE_LAYER_COLOR:
  300. if (record->event.pressed) {
  301. keyboard_config.disable_layer_led ^= 1;
  302. if (keyboard_config.disable_layer_led) rgb_matrix_set_color_all(0, 0, 0);
  303. eeconfig_update_kb(keyboard_config.raw);
  304. }
  305. break;
  306. case QK_RGB_MATRIX_TOGGLE:
  307. if (record->event.pressed) {
  308. switch (rgb_matrix_get_flags()) {
  309. case LED_FLAG_ALL: {
  310. rgb_matrix_set_flags(LED_FLAG_NONE);
  311. keyboard_config.rgb_matrix_enable = false;
  312. rgb_matrix_set_color_all(0, 0, 0);
  313. } break;
  314. default: {
  315. rgb_matrix_set_flags(LED_FLAG_ALL);
  316. keyboard_config.rgb_matrix_enable = true;
  317. } break;
  318. }
  319. eeconfig_update_kb(keyboard_config.raw);
  320. }
  321. return false;
  322. # endif
  323. }
  324. return process_record_user(keycode, record);
  325. }
  326. #endif
  327. void eeconfig_init_kb(void) { // EEPROM is getting reset!
  328. keyboard_config.raw = 0;
  329. keyboard_config.led_level = 4;
  330. keyboard_config.rgb_matrix_enable = true;
  331. eeconfig_update_kb(keyboard_config.raw);
  332. eeconfig_init_user();
  333. }
  334. #ifdef ORYX_ENABLE
  335. static uint16_t loops = 0;
  336. static bool is_on = false;
  337. #endif
  338. #ifdef DYNAMIC_MACRO_ENABLE
  339. static bool is_dynamic_recording = false;
  340. static uint16_t dynamic_loop_timer;
  341. bool dynamic_macro_record_start_kb(int8_t direction) {
  342. if (!dynamic_macro_record_start_user(direction)) {
  343. return false;
  344. }
  345. is_dynamic_recording = true;
  346. dynamic_loop_timer = timer_read();
  347. ergodox_right_led_1_on();
  348. return true;
  349. }
  350. bool dynamic_macro_record_end_kb(int8_t direction) {
  351. if (!dynamic_macro_record_end_user(direction)) {
  352. return false;
  353. }
  354. is_dynamic_recording = false;
  355. layer_state_set_user(layer_state);
  356. return true;
  357. }
  358. #endif
  359. void matrix_scan_kb(void) {
  360. #ifdef DYNAMIC_MACRO_ENABLE
  361. if (is_dynamic_recording) {
  362. ergodox_right_led_1_off();
  363. // if (timer_elapsed(dynamic_loop_timer) > 5)
  364. {
  365. static uint8_t counter;
  366. counter++;
  367. if (counter > 100) ergodox_right_led_1_on();
  368. dynamic_loop_timer = timer_read();
  369. }
  370. }
  371. #endif
  372. #ifdef CAPS_LOCK_STATUS
  373. led_t led_state = host_keyboard_led_state();
  374. if(led_state.caps_lock) {
  375. ergodox_right_led_3_on();
  376. }
  377. else {
  378. uint8_t layer = get_highest_layer(layer_state);
  379. if(layer != 1) {
  380. ergodox_right_led_3_off();
  381. }
  382. }
  383. #endif
  384. matrix_scan_user();
  385. }