]> git.donarmstrong.com Git - qmk_firmware.git/blob - keyboards/zeal60/zeal60.c
092235ca61bbd8b8c160a70ed64166bb84c119f2
[qmk_firmware.git] / keyboards / zeal60 / zeal60.c
1 /* Copyright 2017 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 "zeal60.h"
17 #include "zeal60_api.h"
18
19 // Check that no backlight functions are called
20 #if RGB_BACKLIGHT_ENABLED
21 #include "rgb_backlight.h"
22 #endif // BACKLIGHT_ENABLED
23
24 #include "raw_hid.h"
25 #include "dynamic_keymap.h"
26 #include "timer.h"
27 #include "tmk_core/common/eeprom.h"
28
29 bool eeprom_is_valid(void)
30 {
31         return (eeprom_read_word(((void*)EEPROM_MAGIC_ADDR)) == EEPROM_MAGIC &&
32                         eeprom_read_byte(((void*)EEPROM_VERSION_ADDR)) == EEPROM_VERSION);
33 }
34
35 void eeprom_set_valid(bool valid)
36 {
37         eeprom_update_word(((void*)EEPROM_MAGIC_ADDR), valid ? EEPROM_MAGIC : 0xFFFF);
38         eeprom_update_byte(((void*)EEPROM_VERSION_ADDR), valid ? EEPROM_VERSION : 0xFF);
39 }
40
41 #ifdef RAW_ENABLE
42
43 void raw_hid_receive( uint8_t *data, uint8_t length )
44 {
45         uint8_t *command_id = &(data[0]);
46         uint8_t *command_data = &(data[1]);
47         switch ( *command_id )
48         {
49                 case id_get_protocol_version:
50                 {
51                         command_data[0] = PROTOCOL_VERSION >> 8;
52                         command_data[1] = PROTOCOL_VERSION & 0xFF;
53                         break;
54                 }
55                 case id_get_keyboard_value:
56                 {
57                         if ( command_data[0] == 0x01 )
58                         {
59                                 uint32_t value = timer_read32();
60                                 command_data[1] = (value >> 24 ) & 0xFF;
61                                 command_data[2] = (value >> 16 ) & 0xFF;
62                                 command_data[3] = (value >> 8 ) & 0xFF;
63                                 command_data[4] = value & 0xFF;
64                         }
65                         else
66                         {
67                                 *command_id = id_unhandled;
68                         }
69                         break;
70                 }
71 #ifdef DYNAMIC_KEYMAP_ENABLE
72                 case id_dynamic_keymap_get_keycode:
73                 {
74                         uint16_t keycode = dynamic_keymap_get_keycode( command_data[0], command_data[1], command_data[2] );
75                         command_data[3] = keycode >> 8;
76                         command_data[4] = keycode & 0xFF;
77                         break;
78                 }
79                 case id_dynamic_keymap_set_keycode:
80                 {
81                         dynamic_keymap_set_keycode( command_data[0], command_data[1], command_data[2], ( command_data[3] << 8 ) | command_data[4] );
82                         break;
83                 }
84                 case id_dynamic_keymap_reset:
85                 {
86                         dynamic_keymap_reset();
87                         break;
88                 }
89 #endif // DYNAMIC_KEYMAP_ENABLE
90 #if RGB_BACKLIGHT_ENABLED
91                 case id_backlight_config_set_value:
92                 {
93                         backlight_config_set_value(command_data);
94                         break;
95                 }
96                 case id_backlight_config_get_value:
97                 {
98                         backlight_config_get_value(command_data);
99                         break;
100                 }
101                 case id_backlight_config_save:
102                 {
103                         backlight_config_save();
104                         break;
105                 }
106 #endif // RGB_BACKLIGHT_ENABLED
107                 default:
108                 {
109                         // Unhandled message.
110                         *command_id = id_unhandled;
111                         break;
112                 }
113         }
114
115         // Return same buffer with values changed
116         raw_hid_send( data, length );
117
118 }
119
120 #endif
121
122 void bootmagic_lite(void)
123 {
124         // The lite version of TMK's bootmagic.
125         // 100% less potential for accidentally making the
126         // keyboard do stupid things.
127
128         // We need multiple scans because debouncing can't be turned off.
129         matrix_scan();
130         wait_ms(DEBOUNCING_DELAY);
131         wait_ms(DEBOUNCING_DELAY);
132         matrix_scan();
133
134         // If the Esc and space bar are held down on power up,
135         // reset the EEPROM valid state and jump to bootloader.
136         // Assumes Esc is at [0,0] and spacebar is at [4,7].
137         // This isn't very generalized, but we need something that doesn't
138         // rely on user's keymaps in firmware or EEPROM.
139         if ( ( matrix_get_row(0) & (1<<0) ) &&
140                 ( matrix_get_row(4) & (1<<7) ) )
141         {
142                 // Set the Zeal60 specific EEPROM state as invalid.
143                 eeprom_set_valid(false);
144                 // Set the TMK/QMK EEPROM state as invalid.
145                 eeconfig_disable();
146                 // Jump to bootloader.
147                 bootloader_jump();
148         }
149 }
150
151 void matrix_init_kb(void)
152 {
153         bootmagic_lite();
154
155         // If the EEPROM has the magic, the data is good.
156         // OK to load from EEPROM.
157         if (eeprom_is_valid())
158         {
159 #if RGB_BACKLIGHT_ENABLED
160                 backlight_config_load();
161 #endif // RGB_BACKLIGHT_ENABLED
162                 // TODO: do something to "turn on" keymaps in EEPROM?
163         }
164         else
165         {
166 #if RGB_BACKLIGHT_ENABLED
167                 // If the EEPROM has not been saved before, or is out of date,
168                 // save the default values to the EEPROM. Default values
169                 // come from construction of the zeal_backlight_config instance.
170                 backlight_config_save();
171 #endif // RGB_BACKLIGHT_ENABLED
172
173 #ifdef DYNAMIC_KEYMAP_ENABLE
174                 // This resets the keymaps in EEPROM to what is in flash.
175                 dynamic_keymap_reset();
176 #endif
177
178                 // Save the magic number last, in case saving was interrupted
179                 eeprom_set_valid(true);
180         }
181
182 #if RGB_BACKLIGHT_ENABLED
183         // Initialize LED drivers for backlight.
184         backlight_init_drivers();
185         
186         backlight_timer_init();
187         backlight_timer_enable();
188 #endif // RGB_BACKLIGHT_ENABLED
189
190         matrix_init_user();
191 }
192
193 void matrix_scan_kb(void)
194 {
195 #if RGB_BACKLIGHT_ENABLED
196         // This only updates the LED driver buffers if something has changed.
197         backlight_update_pwm_buffers();
198 #endif // BACKLIGHT_ENABLED
199         matrix_scan_user();
200 }
201
202 bool process_record_kb(uint16_t keycode, keyrecord_t *record)
203 {
204 #if RGB_BACKLIGHT_ENABLED
205         process_record_backlight(keycode, record);
206 #endif // BACKLIGHT_ENABLED
207
208         switch(keycode)
209         {
210                 case FN_MO13:
211                         if (record->event.pressed)
212                         {
213                                 layer_on(1);
214                                 update_tri_layer(1, 2, 3);
215                         }
216                         else
217                         {
218                                 layer_off(1);
219                                 update_tri_layer(1, 2, 3);
220                         }
221                         return false;
222                         break;
223                 case FN_MO23:
224                         if (record->event.pressed)
225                         {
226                                 layer_on(2);
227                                 update_tri_layer(1, 2, 3);
228                         }
229                         else
230                         {
231                                 layer_off(2);
232                                 update_tri_layer(1, 2, 3);
233                         }
234                         return false;
235                         break;
236         }
237         
238         return process_record_user(keycode, record);
239 }
240
241 // This overrides the one in quantum/keymap_common.c
242 uint16_t keymap_function_id_to_action( uint16_t function_id )
243 {
244         // Zeal60 specific "action functions" are 0xF00 to 0xFFF
245         // i.e. F(0xF00) to F(0xFFF) are mapped to
246         // enum zeal60_action_functions by masking last 8 bits.
247         if ( function_id >= 0x0F00 && function_id <= 0x0FFF )
248         {
249                 uint8_t id = function_id & 0xFF;
250                 switch ( id )
251                 {
252                         case TRIPLE_TAP_1_3:
253                         case TRIPLE_TAP_2_3:
254                         {
255                                 return ACTION_FUNCTION_TAP(id);
256                                 break;
257                         }
258                         default:
259                                 break;
260                 }
261         }
262
263 #if USE_KEYMAPS_IN_EEPROM
264
265 #if 0
266         // This is how to implement actions stored in EEPROM.
267         // Not yet implemented. Not sure if it's worth the trouble
268         // before we have a nice GUI for keymap editing.
269         if ( eeprom_is_valid() &&
270                  function_id < 32 ) // TODO: replace magic number
271         {
272                 uint16_t action = keymap_action_load(function_id);
273
274                 // If action is not "empty", return it, otherwise
275                 // drop down to return the one in flash
276                 if ( action != 0x0000 ) // TODO: replace magic number
277                 {
278                         return action;
279                 }
280         }
281 #endif
282
283 #endif // USE_KEYMAPS_IN_EEPROM
284
285         return pgm_read_word(&fn_actions[function_id]);
286 }
287
288
289 // Zeal60 specific "action functions"
290 void action_function(keyrecord_t *record, uint8_t id, uint8_t opt)
291 {
292         switch (id)
293         {
294         case TRIPLE_TAP_1_3:
295         case TRIPLE_TAP_2_3:
296                 if (record->event.pressed)
297                 {
298                         layer_on( id == TRIPLE_TAP_1_3 ? 1 : 2 );
299
300                         if (record->tap.count && !record->tap.interrupted)
301                         {
302                                 if (record->tap.count >= 3)
303                                 {
304                                         layer_invert(3);
305                                 }
306                         }
307                         else
308                         {
309                                 record->tap.count = 0;
310                         }
311                 }
312                 else
313                 {
314                         layer_off( id == TRIPLE_TAP_1_3 ? 1 : 2 );
315                 }
316                 break;
317         }
318 }
319
320 void led_set_kb(uint8_t usb_led)
321 {
322 #if RGB_BACKLIGHT_ENABLED
323         backlight_set_indicator_state(usb_led);
324 #endif // RGB_BACKLIGHT_ENABLED
325 }
326
327 void suspend_power_down_kb(void)
328 {
329 #if RGB_BACKLIGHT_ENABLED
330         backlight_set_suspend_state(true);
331 #endif // RGB_BACKLIGHT_ENABLED
332 }
333
334 void suspend_wakeup_init_kb(void)
335 {
336 #if RGB_BACKLIGHT_ENABLED
337         backlight_set_suspend_state(false);
338 #endif // RGB_BACKLIGHT_ENABLED
339 }
340