3 Copyright (C) Dean Camera, 2009.
5 dean [at] fourwalledcubicle [dot] com
6 www.fourwalledcubicle.com
10 Copyright 2009 Dean Camera (dean [at] fourwalledcubicle [dot] com)
11 Copyright 2009 Denver Gingerich (denver [at] ossguy [dot] com)
13 Permission to use, copy, modify, and distribute this software
14 and its documentation for any purpose and without fee is hereby
15 granted, provided that the above copyright notice appear in all
16 copies and that both that the copyright notice and this
17 permission notice and warranty disclaimer appear in supporting
18 documentation, and that the name of the author not be used in
19 advertising or publicity pertaining to distribution of the
20 software without specific, written prior permission.
22 The author disclaim all warranties with regard to this
23 software, including all implied warranties of merchantability
24 and fitness. In no event shall the author be liable for any
25 special, indirect or consequential damages or any damages
26 whatsoever resulting from loss of use, data or profits, whether
27 in an action of contract, negligence or other tortious action,
28 arising out of or in connection with the use or performance of
34 * Main source file for the KeyboardMouse demo. This file contains the main tasks of the demo and
35 * is responsible for the initial application hardware configuration.
38 #include "KeyboardMouse.h"
40 /* Global Variables */
41 /** Global structure to hold the current keyboard interface HID report, for transmission to the host */
42 USB_KeyboardReport_Data_t KeyboardReportData
;
44 /** Global structure to hold the current mouse interface HID report, for transmission to the host */
45 USB_MouseReport_Data_t MouseReportData
;
47 /** Main program entry point. This routine configures the hardware required by the application, then
48 * starts the scheduler to run the USB management task.
54 LEDs_SetAllLEDs(LEDMASK_USB_NOTREADY
);
64 /** Configures the board hardware and chip peripherals for the demo's functionality. */
65 void SetupHardware(void)
67 /* Disable watchdog if enabled by bootloader/fuses */
68 MCUSR
&= ~(1 << WDRF
);
71 /* Disable clock division */
72 clock_prescale_set(clock_div_1
);
74 /* Hardware Initialization */
80 /** Event handler for the USB_Connect event. This indicates that the device is enumerating via the status LEDs and
81 * starts the library USB task to begin the enumeration and USB management process.
83 void EVENT_USB_Connect(void)
85 /* Indicate USB enumerating */
86 LEDs_SetAllLEDs(LEDMASK_USB_ENUMERATING
);
89 /** Event handler for the USB_Disconnect event. This indicates that the device is no longer connected to a host via
90 * the status LEDs and stops the USB management task.
92 void EVENT_USB_Disconnect(void)
94 /* Indicate USB not ready */
95 LEDs_SetAllLEDs(LEDMASK_USB_NOTREADY
);
98 /** Event handler for the USB_ConfigurationChanged event. This is fired when the host sets the current configuration
99 * of the USB device after enumeration, and configures the keyboard and mouse device endpoints.
101 void EVENT_USB_ConfigurationChanged(void)
103 /* Indicate USB connected and ready */
104 LEDs_SetAllLEDs(LEDMASK_USB_READY
);
106 /* Setup Keyboard Report Endpoint */
107 if (!(Endpoint_ConfigureEndpoint(KEYBOARD_IN_EPNUM
, EP_TYPE_INTERRUPT
,
108 ENDPOINT_DIR_IN
, HID_EPSIZE
,
109 ENDPOINT_BANK_SINGLE
)))
111 LEDs_SetAllLEDs(LEDMASK_USB_ERROR
);
114 /* Setup Keyboard LED Report Endpoint */
115 if (!(Endpoint_ConfigureEndpoint(KEYBOARD_OUT_EPNUM
, EP_TYPE_INTERRUPT
,
116 ENDPOINT_DIR_OUT
, HID_EPSIZE
,
117 ENDPOINT_BANK_SINGLE
)))
119 LEDs_SetAllLEDs(LEDMASK_USB_ERROR
);
122 /* Setup Mouse Report Endpoint */
123 if (!(Endpoint_ConfigureEndpoint(MOUSE_IN_EPNUM
, EP_TYPE_INTERRUPT
,
124 ENDPOINT_DIR_IN
, HID_EPSIZE
,
125 ENDPOINT_BANK_SINGLE
)))
127 LEDs_SetAllLEDs(LEDMASK_USB_ERROR
);
131 /** Event handler for the USB_UnhandledControlPacket event. This is used to catch standard and class specific
132 * control requests that are not handled internally by the USB library (including the HID commands, which are
133 * all issued via the control endpoint), so that they can be handled appropriately for the application.
135 void EVENT_USB_UnhandledControlPacket(void)
140 /* Handle HID Class specific requests */
141 switch (USB_ControlRequest
.bRequest
)
144 if (USB_ControlRequest
.bmRequestType
== (REQDIR_DEVICETOHOST
| REQTYPE_CLASS
| REQREC_INTERFACE
))
146 Endpoint_ClearSETUP();
148 /* Determine if it is the mouse or the keyboard data that is being requested */
149 if (!(USB_ControlRequest
.wIndex
))
151 ReportData
= (uint8_t*)&KeyboardReportData
;
152 ReportSize
= sizeof(KeyboardReportData
);
156 ReportData
= (uint8_t*)&MouseReportData
;
157 ReportSize
= sizeof(MouseReportData
);
160 /* Write the report data to the control endpoint */
161 Endpoint_Write_Control_Stream_LE(ReportData
, ReportSize
);
163 /* Clear the report data afterwards */
164 memset(ReportData
, 0, ReportSize
);
166 /* Finalize the stream transfer to send the last packet or clear the host abort */
172 if (USB_ControlRequest
.bmRequestType
== (REQDIR_HOSTTODEVICE
| REQTYPE_CLASS
| REQREC_INTERFACE
))
174 Endpoint_ClearSETUP();
176 /* Wait until the LED report has been sent by the host */
177 while (!(Endpoint_IsOUTReceived()));
179 /* Read in the LED report from the host */
180 uint8_t LEDStatus
= Endpoint_Read_Byte();
181 uint8_t LEDMask
= LEDS_LED2
;
183 if (LEDStatus
& 0x01) // NUM Lock
184 LEDMask
|= LEDS_LED1
;
186 if (LEDStatus
& 0x02) // CAPS Lock
187 LEDMask
|= LEDS_LED3
;
189 if (LEDStatus
& 0x04) // SCROLL Lock
190 LEDMask
|= LEDS_LED4
;
192 /* Set the status LEDs to the current HID LED status */
193 LEDs_SetAllLEDs(LEDMask
);
195 /* Clear the endpoint data */
198 /* Acknowledge status stage */
199 while (!(Endpoint_IsINReady()));
207 /** Keyboard task. This generates the next keyboard HID report for the host, and transmits it via the
208 * keyboard IN endpoint when the host is ready for more data. Additionally, it processes host LED status
209 * reports sent to the device via the keyboard OUT reporting endpoint.
211 void Keyboard_HID_Task(void)
213 uint8_t JoyStatus_LCL
= Joystick_GetStatus();
215 /* Check if board button is not pressed, if so mouse mode enabled */
216 if (!(Buttons_GetStatus() & BUTTONS_BUTTON1
))
218 if (JoyStatus_LCL
& JOY_UP
)
219 KeyboardReportData
.KeyCode
[0] = 0x04; // A
220 else if (JoyStatus_LCL
& JOY_DOWN
)
221 KeyboardReportData
.KeyCode
[0] = 0x05; // B
223 if (JoyStatus_LCL
& JOY_LEFT
)
224 KeyboardReportData
.KeyCode
[0] = 0x06; // C
225 else if (JoyStatus_LCL
& JOY_RIGHT
)
226 KeyboardReportData
.KeyCode
[0] = 0x07; // D
228 if (JoyStatus_LCL
& JOY_PRESS
)
229 KeyboardReportData
.KeyCode
[0] = 0x08; // E
232 /* Check if the USB system is connected to a host and report protocol mode is enabled */
235 /* Select the Keyboard Report Endpoint */
236 Endpoint_SelectEndpoint(KEYBOARD_IN_EPNUM
);
238 /* Check if Keyboard Endpoint Ready for Read/Write */
239 if (Endpoint_IsReadWriteAllowed())
241 /* Write Keyboard Report Data */
242 Endpoint_Write_Stream_LE(&KeyboardReportData
, sizeof(KeyboardReportData
));
244 /* Finalize the stream transfer to send the last packet */
247 /* Clear the report data afterwards */
248 memset(&KeyboardReportData
, 0, sizeof(KeyboardReportData
));
251 /* Select the Keyboard LED Report Endpoint */
252 Endpoint_SelectEndpoint(KEYBOARD_OUT_EPNUM
);
254 /* Check if Keyboard LED Endpoint Ready for Read/Write */
255 if (Endpoint_IsReadWriteAllowed())
257 /* Read in the LED report from the host */
258 uint8_t LEDStatus
= Endpoint_Read_Byte();
259 uint8_t LEDMask
= LEDS_LED2
;
261 if (LEDStatus
& 0x01) // NUM Lock
262 LEDMask
|= LEDS_LED1
;
264 if (LEDStatus
& 0x02) // CAPS Lock
265 LEDMask
|= LEDS_LED3
;
267 if (LEDStatus
& 0x04) // SCROLL Lock
268 LEDMask
|= LEDS_LED4
;
270 /* Set the status LEDs to the current Keyboard LED status */
271 LEDs_SetAllLEDs(LEDMask
);
273 /* Handshake the OUT Endpoint - clear endpoint and ready for next report */
279 /** Mouse task. This generates the next mouse HID report for the host, and transmits it via the
280 * mouse IN endpoint when the host is ready for more data.
282 void Mouse_HID_Task(void)
284 uint8_t JoyStatus_LCL
= Joystick_GetStatus();
286 /* Check if board button is pressed, if so mouse mode enabled */
287 if (Buttons_GetStatus() & BUTTONS_BUTTON1
)
289 if (JoyStatus_LCL
& JOY_UP
)
290 MouseReportData
.Y
= 1;
291 else if (JoyStatus_LCL
& JOY_DOWN
)
292 MouseReportData
.Y
= -1;
294 if (JoyStatus_LCL
& JOY_RIGHT
)
295 MouseReportData
.X
= 1;
296 else if (JoyStatus_LCL
& JOY_LEFT
)
297 MouseReportData
.X
= -1;
299 if (JoyStatus_LCL
& JOY_PRESS
)
300 MouseReportData
.Button
= (1 << 0);
303 /* Check if the USB system is connected to a host and report protocol mode is enabled */
306 /* Select the Mouse Report Endpoint */
307 Endpoint_SelectEndpoint(MOUSE_IN_EPNUM
);
309 /* Check if Mouse Endpoint Ready for Read/Write */
310 if (Endpoint_IsReadWriteAllowed())
312 /* Write Mouse Report Data */
313 Endpoint_Write_Stream_LE(&MouseReportData
, sizeof(MouseReportData
));
315 /* Finalize the stream transfer to send the last packet */
318 /* Clear the report data afterwards */
319 memset(&MouseReportData
, 0, sizeof(MouseReportData
));