]> git.donarmstrong.com Git - tmk_firmware.git/blob - common/action_util.c
Fix mod stuck of MODS_KEY when leaving layer #62
[tmk_firmware.git] / common / action_util.c
1 /*
2 Copyright 2013 Jun Wako <wakojun@gmail.com>
3
4 This program is free software: you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation, either version 2 of the License, or
7 (at your option) any later version.
8
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12 GNU General Public License for more details.
13
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 "host.h"
18 #include "report.h"
19 #include "debug.h"
20 #include "action_util.h"
21
22 static inline void add_key_byte(uint8_t code);
23 static inline void del_key_byte(uint8_t code);
24 #ifdef NKRO_ENABLE
25 static inline void add_key_bit(uint8_t code);
26 static inline void del_key_bit(uint8_t code);
27 #endif
28
29 static uint8_t real_mods = 0;
30 static uint8_t weak_mods = 0;
31
32
33 // TODO: pointer variable is not needed
34 //report_keyboard_t keyboard_report = {};
35 report_keyboard_t *keyboard_report = &(report_keyboard_t){};
36
37 #ifndef NO_ACTION_ONESHOT
38 static bool oneshot_enabled = true;
39 static int8_t oneshot_mods = 0;
40 #endif
41
42 void send_keyboard_report(void) {
43     keyboard_report->mods  = real_mods;
44     keyboard_report->mods |= weak_mods;
45 #ifndef NO_ACTION_ONESHOT
46     keyboard_report->mods |= oneshot_mods;
47     if (has_anykey()) {
48         clear_oneshot_mods();
49     }
50 #endif
51     host_keyboard_send(keyboard_report);
52 }
53
54 /* key */
55 void add_key(uint8_t key)
56 {
57 #ifdef NKRO_ENABLE
58     if (keyboard_nkro) {
59         add_key_bit(key);
60         return;
61     }
62 #endif
63     add_key_byte(key);
64 }
65
66 void del_key(uint8_t key)
67 {
68 #ifdef NKRO_ENABLE
69     if (keyboard_nkro) {
70         del_key_bit(key);
71         return;
72     }
73 #endif
74     del_key_byte(key);
75 }
76
77 void clear_keys(void)
78 {
79     // not clear mods
80     for (int8_t i = 1; i < REPORT_SIZE; i++) {
81         keyboard_report->raw[i] = 0;
82     }
83 }
84
85
86 /* modifier */
87 uint8_t get_mods(void) { return real_mods; }
88 void add_mods(uint8_t mods) { real_mods |= mods; }
89 void del_mods(uint8_t mods) { real_mods &= ~mods; }
90 void set_mods(uint8_t mods) { real_mods = mods; }
91 void clear_mods(void) { real_mods = 0; }
92
93 /* weak modifier */
94 uint8_t get_weak_mods(void) { return weak_mods; }
95 void add_weak_mods(uint8_t mods) { weak_mods |= mods; }
96 void del_weak_mods(uint8_t mods) { weak_mods &= ~mods; }
97 void set_weak_mods(uint8_t mods) { weak_mods = mods; }
98 void clear_weak_mods(void) { weak_mods = 0; }
99
100 /* Oneshot modifier */
101 #ifndef NO_ACTION_ONESHOT
102 void set_oneshot_mods(uint8_t mods) { oneshot_mods = mods; }
103 void clear_oneshot_mods(void) { oneshot_mods = 0; }
104 void oneshot_toggle(void) { oneshot_enabled = !oneshot_enabled; }
105 void oneshot_enable(void) { oneshot_enabled = true; }
106 void oneshot_disable(void) { oneshot_enabled = false; }
107 #endif
108
109
110
111
112 /*
113  * inspect keyboard state
114  */
115 uint8_t has_anykey(void)
116 {
117     uint8_t cnt = 0;
118     for (uint8_t i = 1; i < REPORT_SIZE; i++) {
119         if (keyboard_report->raw[i])
120             cnt++;
121     }
122     return cnt;
123 }
124
125 uint8_t has_anymod(void)
126 {
127     return bitpop(real_mods);
128 }
129
130 uint8_t get_first_key(void)
131 {
132 #ifdef NKRO_ENABLE
133     if (keyboard_nkro) {
134         uint8_t i = 0;
135         for (; i < REPORT_BITS && !keyboard_report->nkro.bits[i]; i++)
136             ;
137         return i<<3 | biton(keyboard_report->nkro.bits[i]);
138     }
139 #endif
140     return keyboard_report->keys[0];
141 }
142
143
144
145 /* local functions */
146 static inline void add_key_byte(uint8_t code)
147 {
148     int8_t i = 0;
149     int8_t empty = -1;
150     for (; i < REPORT_KEYS; i++) {
151         if (keyboard_report->keys[i] == code) {
152             break;
153         }
154         if (empty == -1 && keyboard_report->keys[i] == 0) {
155             empty = i;
156         }
157     }
158     if (i == REPORT_KEYS) {
159         if (empty != -1) {
160             keyboard_report->keys[empty] = code;
161         }
162     }
163 }
164
165 static inline void del_key_byte(uint8_t code)
166 {
167     for (uint8_t i = 0; i < REPORT_KEYS; i++) {
168         if (keyboard_report->keys[i] == code) {
169             keyboard_report->keys[i] = 0;
170         }
171     }
172 }
173
174 #ifdef NKRO_ENABLE
175 static inline void add_key_bit(uint8_t code)
176 {
177     if ((code>>3) < REPORT_BITS) {
178         keyboard_report->nkro.bits[code>>3] |= 1<<(code&7);
179     } else {
180         dprintf("add_key_bit: can't add: %02X\n", code);
181     }
182 }
183
184 static inline void del_key_bit(uint8_t code)
185 {
186     if ((code>>3) < REPORT_BITS) {
187         keyboard_report->nkro.bits[code>>3] &= ~(1<<(code&7));
188     } else {
189         dprintf("del_key_bit: can't del: %02X\n", code);
190     }
191 }
192 #endif