]> git.donarmstrong.com Git - tmk_firmware.git/blob - keyboard/hhkb_rn42/rn42/rn42.c
556c61af9589e7e971a11e621d46d37e4a3a6a2a
[tmk_firmware.git] / keyboard / hhkb_rn42 / rn42 / rn42.c
1 #include <avr/io.h>
2 #include "host.h"
3 #include "host_driver.h"
4 #include "serial.h"
5 #include "rn42.h"
6 #include "print.h"
7 #include "wait.h"
8
9
10 /* Host driver */
11 static uint8_t keyboard_leds(void);
12 static void send_keyboard(report_keyboard_t *report);
13 static void send_mouse(report_mouse_t *report);
14 static void send_system(uint16_t data);
15 static void send_consumer(uint16_t data);
16
17 host_driver_t rn42_driver = {
18     keyboard_leds,
19     send_keyboard,
20     send_mouse,
21     send_system,
22     send_consumer
23 };
24
25
26 void rn42_init(void)
27 {
28     // JTAG disable for PORT F. write JTD bit twice within four cycles.
29     MCUCR |= (1<<JTD);
30     MCUCR |= (1<<JTD);
31
32     // PF7: BT connection control(high: connect, low: disconnect)
33     rn42_autoconnect();
34
35     // PF6: linked(input without pull-up)
36     DDRF  &= ~(1<<6);
37     PORTF &= ~(1<<6);
38
39     // PF1: RTS(low: allowed to send, high: not allowed)
40     DDRF &= ~(1<<1);
41     PORTF &= ~(1<<1);
42
43     // PD5: CTS(low: allow to send, high:not allow)
44     DDRD |= (1<<5);
45     PORTD &= ~(1<<5);
46
47     serial_init();
48 }
49
50 void rn42_putc(uint8_t c)
51 {
52     serial_send(c);
53 }
54
55 bool rn42_autoconnecting(void)
56 {
57     // GPIO6 for control connection(high: auto connect, low: disconnect)
58     // Note that this needs config: SM,4(Auto-Connect DTR Mode)
59     return (PORTF & (1<<7) ? true : false);
60 }
61
62 void rn42_autoconnect(void)
63 {
64     // hi to auto connect
65     DDRF |= (1<<7);
66     PORTF |= (1<<7);
67 }
68
69 void rn42_disconnect(void)
70 {
71     // low to disconnect
72     DDRF |= (1<<7);
73     PORTF &= ~(1<<7);
74 }
75
76 bool rn42_rts(void)
77 {
78     // low when RN-42 is powered and ready to receive
79     return PINF&(1<<1);
80 }
81
82 void rn42_cts_hi(void)
83 {
84     // not allow to send
85     PORTD |= (1<<5);
86 }
87
88 void rn42_cts_lo(void)
89 {
90     // allow to send
91     PORTD &= ~(1<<5);
92 }
93
94 bool rn42_linked(void)
95 {
96     return PINF&(1<<6);
97 }
98
99
100 static uint8_t keyboard_leds(void) { return 0; }
101
102 static void send_keyboard(report_keyboard_t *report)
103 {
104     // wake from deep sleep
105 /*
106     PORTD |= (1<<5);    // high
107     wait_ms(5);
108     PORTD &= ~(1<<5);   // low
109 */
110
111     serial_send(0xFD);  // Raw report mode
112     serial_send(9);     // length
113     serial_send(1);     // descriptor type
114     serial_send(report->mods);
115     serial_send(0x00);
116     serial_send(report->keys[0]);
117     serial_send(report->keys[1]);
118     serial_send(report->keys[2]);
119     serial_send(report->keys[3]);
120     serial_send(report->keys[4]);
121     serial_send(report->keys[5]);
122 }
123
124 static void send_mouse(report_mouse_t *report)
125 {
126     // wake from deep sleep
127 /*
128     PORTD |= (1<<5);    // high
129     wait_ms(5);
130     PORTD &= ~(1<<5);   // low
131 */
132
133     serial_send(0xFD);  // Raw report mode
134     serial_send(5);     // length
135     serial_send(2);     // descriptor type
136     serial_send(report->buttons);
137     serial_send(report->x);
138     serial_send(report->y);
139     serial_send(report->v);
140 }
141
142 static void send_system(uint16_t data)
143 {
144     // Table 5-6 of RN-BT-DATA-UB
145     // 81,82,83 scan codes can be used?
146 }
147
148
149 static uint16_t usage2bits(uint16_t usage)
150 {
151     switch (usage) {
152         case AC_HOME:               return 0x01;
153         case AL_EMAIL:              return 0x02;
154         case AC_SEARCH:             return 0x04;
155         //case AL_KBD_LAYOUT:         return 0x08;  // Apple virtual keybaord toggle
156         case AUDIO_VOL_UP:          return 0x10;
157         case AUDIO_VOL_DOWN:        return 0x20;
158         case AUDIO_MUTE:            return 0x40;
159         case TRANSPORT_PLAY_PAUSE:  return 0x80;
160         case TRANSPORT_NEXT_TRACK:  return 0x100;
161         case TRANSPORT_PREV_TRACK:  return 0x200;
162         case TRANSPORT_STOP:        return 0x400;
163         case TRANSPORT_STOP_EJECT:  return 0x800;
164         //case return 0x1000;   // Fast forward
165         //case return 0x2000;   // Rewind
166         //case return 0x4000;   // Stop/eject
167         //case return 0x8000;   // Internet browser
168     };
169     return 0;
170 }
171
172 static void send_consumer(uint16_t data)
173 {
174     uint16_t bits = usage2bits(data);
175     serial_send(0xFD);  // Raw report mode
176     serial_send(3);     // length
177     serial_send(3);     // descriptor type
178     serial_send(bits&0xFF);
179     serial_send((bits>>8)&0xFF);
180 }
181
182
183 /* Null driver for config_mode */
184 static uint8_t config_keyboard_leds(void);
185 static void config_send_keyboard(report_keyboard_t *report);
186 static void config_send_mouse(report_mouse_t *report);
187 static void config_send_system(uint16_t data);
188 static void config_send_consumer(uint16_t data);
189
190 host_driver_t rn42_config_driver = {
191     config_keyboard_leds,
192     config_send_keyboard,
193     config_send_mouse,
194     config_send_system,
195     config_send_consumer
196 };
197
198 static uint8_t config_keyboard_leds(void) { return 0; }
199 static void config_send_keyboard(report_keyboard_t *report) {}
200 static void config_send_mouse(report_mouse_t *report) {}
201 static void config_send_system(uint16_t data) {}
202 static void config_send_consumer(uint16_t data) {}