3      Copyright (C) Dean Camera, 2013. 
   5   dean [at] fourwalledcubicle [dot] com 
  10   Copyright 2013  Dean Camera (dean [at] fourwalledcubicle [dot] com) 
  11   Copyright 2010  Denver Gingerich (denver [at] ossguy [dot] com) 
  13   Permission to use, copy, modify, distribute, and sell this 
  14   software and its documentation for any purpose is hereby granted 
  15   without fee, provided that the above copyright notice appear in 
  16   all 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 disclaims 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 structure to hold the current keyboard interface HID report, for transmission to the host */ 
  41 static USB_KeyboardReport_Data_t KeyboardReportData
; 
  43 /** Global structure to hold the current mouse interface HID report, for transmission to the host */ 
  44 static USB_MouseReport_Data_t MouseReportData
; 
  47 /** Main program entry point. This routine configures the hardware required by the application, then 
  48  *  enters a loop to run the application tasks in sequence. 
  54         LEDs_SetAllLEDs(LEDMASK_USB_NOTREADY
); 
  55         GlobalInterruptEnable(); 
  65 /** Configures the board hardware and chip peripherals for the demo's functionality. */ 
  66 void SetupHardware(void) 
  68         /* Disable watchdog if enabled by bootloader/fuses */ 
  69         MCUSR 
&= ~(1 << WDRF
); 
  72         /* Disable clock division */ 
  73         clock_prescale_set(clock_div_1
); 
  75         /* Hardware Initialization */ 
  81 /** Event handler for the USB_Connect event. This indicates that the device is enumerating via the status LEDs and 
  82  *  starts the library USB task to begin the enumeration and USB management process. 
  84 void EVENT_USB_Device_Connect(void) 
  86         /* Indicate USB enumerating */ 
  87         LEDs_SetAllLEDs(LEDMASK_USB_ENUMERATING
); 
  90 /** Event handler for the USB_Disconnect event. This indicates that the device is no longer connected to a host via 
  91  *  the status LEDs and stops the USB management task. 
  93 void EVENT_USB_Device_Disconnect(void) 
  95         /* Indicate USB not ready */ 
  96         LEDs_SetAllLEDs(LEDMASK_USB_NOTREADY
); 
  99 /** Event handler for the USB_ConfigurationChanged event. This is fired when the host sets the current configuration 
 100  *  of the USB device after enumeration, and configures the keyboard and mouse device endpoints. 
 102 void EVENT_USB_Device_ConfigurationChanged(void) 
 104         bool ConfigSuccess 
= true; 
 106         /* Setup Keyboard HID Report Endpoints */ 
 107         ConfigSuccess 
&= Endpoint_ConfigureEndpoint(KEYBOARD_IN_EPADDR
, EP_TYPE_INTERRUPT
, HID_EPSIZE
, 1); 
 108         ConfigSuccess 
&= Endpoint_ConfigureEndpoint(KEYBOARD_OUT_EPADDR
, EP_TYPE_INTERRUPT
, HID_EPSIZE
, 1); 
 110         /* Setup Mouse HID Report Endpoint */ 
 111         ConfigSuccess 
&= Endpoint_ConfigureEndpoint(MOUSE_IN_EPADDR
, EP_TYPE_INTERRUPT
, HID_EPSIZE
, 1); 
 113         /* Indicate endpoint configuration success or failure */ 
 114         LEDs_SetAllLEDs(ConfigSuccess ? LEDMASK_USB_READY 
: LEDMASK_USB_ERROR
); 
 117 /** Event handler for the USB_ControlRequest event. This is used to catch and process control requests sent to 
 118  *  the device from the USB host before passing along unhandled control requests to the library for processing 
 121 void EVENT_USB_Device_ControlRequest(void) 
 126         /* Handle HID Class specific requests */ 
 127         switch (USB_ControlRequest
.bRequest
) 
 129                 case HID_REQ_GetReport
: 
 130                         if (USB_ControlRequest
.bmRequestType 
== (REQDIR_DEVICETOHOST 
| REQTYPE_CLASS 
| REQREC_INTERFACE
)) 
 132                                 Endpoint_ClearSETUP(); 
 134                                 /* Determine if it is the mouse or the keyboard data that is being requested */ 
 135                                 if (!(USB_ControlRequest
.wIndex
)) 
 137                                         ReportData 
= (uint8_t*)&KeyboardReportData
; 
 138                                         ReportSize 
= sizeof(KeyboardReportData
); 
 142                                         ReportData 
= (uint8_t*)&MouseReportData
; 
 143                                         ReportSize 
= sizeof(MouseReportData
); 
 146                                 /* Write the report data to the control endpoint */ 
 147                                 Endpoint_Write_Control_Stream_LE(ReportData
, ReportSize
); 
 150                                 /* Clear the report data afterwards */ 
 151                                 memset(ReportData
, 0, ReportSize
); 
 155                 case HID_REQ_SetReport
: 
 156                         if (USB_ControlRequest
.bmRequestType 
== (REQDIR_HOSTTODEVICE 
| REQTYPE_CLASS 
| REQREC_INTERFACE
)) 
 158                                 Endpoint_ClearSETUP(); 
 160                                 /* Wait until the LED report has been sent by the host */ 
 161                                 while (!(Endpoint_IsOUTReceived())) 
 163                                         if (USB_DeviceState 
== DEVICE_STATE_Unattached
) 
 167                                 /* Read in the LED report from the host */ 
 168                                 uint8_t LEDStatus 
= Endpoint_Read_8(); 
 171                                 Endpoint_ClearStatusStage(); 
 173                                 /* Process the incoming LED report */ 
 174                                 Keyboard_ProcessLEDReport(LEDStatus
); 
 181 /** Processes a given Keyboard LED report from the host, and sets the board LEDs to match. Since the Keyboard 
 182  *  LED report can be sent through either the control endpoint (via a HID SetReport request) or the HID OUT 
 183  *  endpoint, the processing code is placed here to avoid duplicating it and potentially having different 
 184  *  behavior depending on the method used to sent it. 
 186 void Keyboard_ProcessLEDReport(const uint8_t LEDStatus
) 
 188         uint8_t LEDMask 
= LEDS_LED2
; 
 190         if (LEDStatus 
& HID_KEYBOARD_LED_NUMLOCK
) 
 191           LEDMask 
|= LEDS_LED1
; 
 193         if (LEDStatus 
& HID_KEYBOARD_LED_CAPSLOCK
) 
 194           LEDMask 
|= LEDS_LED3
; 
 196         if (LEDStatus 
& HID_KEYBOARD_LED_SCROLLLOCK
) 
 197           LEDMask 
|= LEDS_LED4
; 
 199         /* Set the status LEDs to the current Keyboard LED status */ 
 200         LEDs_SetAllLEDs(LEDMask
); 
 203 /** Keyboard task. This generates the next keyboard HID report for the host, and transmits it via the 
 204  *  keyboard IN endpoint when the host is ready for more data. Additionally, it processes host LED status 
 205  *  reports sent to the device via the keyboard OUT reporting endpoint. 
 207 void Keyboard_HID_Task(void) 
 209         uint8_t JoyStatus_LCL 
= Joystick_GetStatus(); 
 211         /* Device must be connected and configured for the task to run */ 
 212         if (USB_DeviceState 
!= DEVICE_STATE_Configured
) 
 215         /* Check if board button is not pressed, if so mouse mode enabled */ 
 216         if (!(Buttons_GetStatus() & BUTTONS_BUTTON1
)) 
 218                 /* Make sent key uppercase by indicating that the left shift key is pressed */ 
 219                 KeyboardReportData
.Modifier 
= HID_KEYBOARD_MODIFIER_LEFTSHIFT
; 
 221                 if (JoyStatus_LCL 
& JOY_UP
) 
 222                   KeyboardReportData
.KeyCode
[0] = HID_KEYBOARD_SC_A
; 
 223                 else if (JoyStatus_LCL 
& JOY_DOWN
) 
 224                   KeyboardReportData
.KeyCode
[0] = HID_KEYBOARD_SC_B
; 
 226                 if (JoyStatus_LCL 
& JOY_LEFT
) 
 227                   KeyboardReportData
.KeyCode
[0] = HID_KEYBOARD_SC_C
; 
 228                 else if (JoyStatus_LCL 
& JOY_RIGHT
) 
 229                   KeyboardReportData
.KeyCode
[0] = HID_KEYBOARD_SC_D
; 
 231                 if (JoyStatus_LCL 
& JOY_PRESS
) 
 232                   KeyboardReportData
.KeyCode
[0] = HID_KEYBOARD_SC_E
; 
 235         /* Select the Keyboard Report Endpoint */ 
 236         Endpoint_SelectEndpoint(KEYBOARD_IN_EPADDR
); 
 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
), NULL
); 
 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_EPADDR
); 
 254         /* Check if Keyboard LED Endpoint Ready for Read/Write */ 
 255         if (Endpoint_IsReadWriteAllowed()) 
 257                 /* Read in and process the LED report from the host */ 
 258                 Keyboard_ProcessLEDReport(Endpoint_Read_8()); 
 260                 /* Handshake the OUT Endpoint - clear endpoint and ready for next report */ 
 265 /** Mouse task. This generates the next mouse HID report for the host, and transmits it via the 
 266  *  mouse IN endpoint when the host is ready for more data. 
 268 void Mouse_HID_Task(void) 
 270         uint8_t JoyStatus_LCL 
= Joystick_GetStatus(); 
 272         /* Device must be connected and configured for the task to run */ 
 273         if (USB_DeviceState 
!= DEVICE_STATE_Configured
) 
 276         /* Check if board button is pressed, if so mouse mode enabled */ 
 277         if (Buttons_GetStatus() & BUTTONS_BUTTON1
) 
 279                 if (JoyStatus_LCL 
& JOY_UP
) 
 280                   MouseReportData
.Y 
=  1; 
 281                 else if (JoyStatus_LCL 
& JOY_DOWN
) 
 282                   MouseReportData
.Y 
= -1; 
 284                 if (JoyStatus_LCL 
& JOY_RIGHT
) 
 285                   MouseReportData
.X 
=  1; 
 286                 else if (JoyStatus_LCL 
& JOY_LEFT
) 
 287                   MouseReportData
.X 
= -1; 
 289                 if (JoyStatus_LCL 
& JOY_PRESS
) 
 290                   MouseReportData
.Button 
|= (1 << 0); 
 293         /* Select the Mouse Report Endpoint */ 
 294         Endpoint_SelectEndpoint(MOUSE_IN_EPADDR
); 
 296         /* Check if Mouse Endpoint Ready for Read/Write */ 
 297         if (Endpoint_IsReadWriteAllowed()) 
 299                 /* Write Mouse Report Data */ 
 300                 Endpoint_Write_Stream_LE(&MouseReportData
, sizeof(MouseReportData
), NULL
); 
 302                 /* Finalize the stream transfer to send the last packet */ 
 305                 /* Clear the report data afterwards */ 
 306                 memset(&MouseReportData
, 0, sizeof(MouseReportData
));