TataconUSB/Firmware/Keyboard/Keyboard.c
Will Toohey 5d6bf93ebd Remove extraneous comments
Of note - this removes a few copyright notices. I believe my source code is now sufficiently different from the original samples that this is appropriate.
2016-10-19 12:25:02 +10:00

462 lines
14 KiB
C

#include "Keyboard.h"
#include "i2cmaster.h"
#include "Config.h"
#ifdef DEBUG
#include "usbio.h"
#include <util/delay.h>
#endif
#define LED_DIR DDRD
#define LED_PORT PORTD
#define DON_LED_PIN 6
#define KAT_LED_PIN 5
// V1 has no LEDs
#ifdef V1_BUILD
#define SET(port, pin)
#define CLEAR(port, pin)
#define TOGGLE(port, pin)
#else
#define SET(port, pin) port |= _BV(pin)
#define CLEAR(port, pin) port &= ~_BV(pin)
#define TOGGLE(port, pin) port ^= _BV(pin)
#endif
/** Buffer to hold the previously generated Keyboard HID report, for comparison purposes inside the HID class driver. */
static uint8_t PrevKeyboardHIDReportBuffer[sizeof(USB_KeyboardReport_Data_t)];
static uint8_t PrevGenericHIDReportBuffer[TATACON_CONFIG_BYTES];
/** LUFA HID Class driver interface configuration and state information. This structure is
* passed to all HID Class driver functions, so that multiple instances of the same class
* within a device can be differentiated from one another.
*/
USB_ClassInfo_HID_Device_t Keyboard_HID_Interface =
{
.Config =
{
.InterfaceNumber = INTERFACE_ID_Keyboard,
.ReportINEndpoint =
{
.Address = KEYBOARD_EPADDR,
.Size = KEYBOARD_EPSIZE,
.Banks = 1,
},
.PrevReportINBuffer = PrevKeyboardHIDReportBuffer,
.PrevReportINBufferSize = sizeof(PrevKeyboardHIDReportBuffer),
},
};
USB_ClassInfo_HID_Device_t Generic_HID_Interface =
{
.Config =
{
.InterfaceNumber = INTERFACE_ID_Generic,
.ReportINEndpoint =
{
.Address = GENERIC_EPADDR,
.Size = GENERIC_EPSIZE,
.Banks = 1,
},
.PrevReportINBuffer = PrevGenericHIDReportBuffer,
.PrevReportINBufferSize = sizeof(PrevGenericHIDReportBuffer),
},
};
typedef struct {
// optimise data sending
uint8_t state;
uint8_t lastReport;
uint8_t debounce;
} switch_t;
static switch_t switches[KB_SWITCHES];
static uint8_t switchesChanged = 1;
static uint8_t nunchuckReady = 0;
#ifdef DEBUG
#define DEBUG_DELAY_MS 10
static uint8_t debugDelay = 100; // don't do anything for 100ms
#endif
uint32_t Boot_Key ATTR_NO_INIT;
#define MAGIC_BOOT_KEY 0xDEADBE7A
// offset * word size
#define BOOTLOADER_START_ADDRESS (0x1c00 * 2)
void Bootloader_Jump_Check(void) ATTR_INIT_SECTION(3);
void Bootloader_Jump_Check(void)
{
// If the reset source was the bootloader and the key is correct, clear it and jump to the bootloader
if ((MCUSR & (1 << WDRF)) && (Boot_Key == MAGIC_BOOT_KEY))
{
Boot_Key = 0;
((void (*)(void))BOOTLOADER_START_ADDRESS)();
}
}
//todo: neater
#define NUNCHUCK_ADDR (0x52 << 1)
void Nunchuck_back(void) {
if(!nunchuckReady) {
nunchuckReady = 1;
// Turn LEDs off, it returned
CLEAR(LED_PORT, DON_LED_PIN);
CLEAR(LED_PORT, KAT_LED_PIN);
}
}
void Nunchuck_gone(void) {
i2c_stop();
if(nunchuckReady) {
nunchuckReady = 0;
// Turn LEDs on until it returns
SET(LED_PORT, DON_LED_PIN);
SET(LED_PORT, KAT_LED_PIN);
// Clear structs
for(int i = 0; i < KB_SWITCHES; i++) {
switches[i].state = 0;
if(switches[i].state != switches[i].lastReport) {
switchesChanged = 1;
}
}
}
}
void Nunchuck_Init(void) {
// try to say hello
if(!i2c_start(NUNCHUCK_ADDR | I2C_WRITE)) {
i2c_write(0xF0);
i2c_write(0x55);
i2c_stop();
_delay_ms(25);
i2c_start(NUNCHUCK_ADDR | I2C_WRITE);
i2c_write(0xFB);
i2c_write(0x00);
i2c_stop();
_delay_ms(25);
Nunchuck_back();
} else {
Nunchuck_gone();
}
}
uint8_t Nunchuck_ReadByte(uint8_t address) {
uint8_t data = 0xFF;
if(!nunchuckReady) {
Nunchuck_Init();
}
if(!i2c_start(NUNCHUCK_ADDR | I2C_WRITE)) {
i2c_write(address);
i2c_stop();
i2c_start(NUNCHUCK_ADDR | I2C_READ);
data = i2c_readNak();
i2c_stop();
Nunchuck_back();
} else {
Nunchuck_gone();
}
return data;
}
// Starting at address, read n bytes and return the last
void Nunchuck_ReadMany(uint8_t address, uint8_t *data, uint8_t count) {
if(!nunchuckReady) {
Nunchuck_Init();
}
if(!i2c_start(NUNCHUCK_ADDR | I2C_WRITE)) {
i2c_write(address);
i2c_stop();
i2c_start(NUNCHUCK_ADDR | I2C_READ);
for(uint8_t i = 0; i < count-1; i++) {
data[i] = i2c_readAck();
}
data[count-1] = i2c_readNak();
i2c_stop();
Nunchuck_back();
} else {
Nunchuck_gone();
}
}
void update_switches(void) {
uint8_t data, i;
// Data for the buttons is at this address
data = Nunchuck_ReadByte(0x05);
// Not using KB_SWITCHES, because of Tatacon limits
for(i = 0; i < 4; i++) {
// The I2C data starts at the 6th bit and goes down
uint8_t newState = !(data & _BV(6-i));
if(!switches[i].debounce && newState != switches[i].lastReport) {
switches[i].state = newState;
switches[i].debounce = tataConfig.debounce;
switchesChanged = 1;
}
}
}
/** Main program entry point. This routine contains the overall program flow, including initial
* setup of all components and the main program loop.
*/
int main(void)
{
uint8_t i;
// Clear structs
for(i = 0; i < KB_SWITCHES; i++) {
switches[i].state = 0;
switches[i].lastReport = 0;
switches[i].debounce = 0;
}
InitConfig();
SetupHardware();
GlobalInterruptEnable();
#ifdef DEBUG
printf_P(PSTR("Tatacon to USB Debug Mode\n\n"));
uint8_t tmp[6];
Nunchuck_ReadMany(0xFA, tmp, 6);
if(nunchuckReady) {
// ID debug
printf_P(PSTR("Tatacon found with ID:\n"));
for(uint8_t i = 0; i < 6; i++) {
if(tmp[i] < 0x10) {
printf("0");
}
printf("%X ", tmp[i]);
}
printf("\n\n");
// Register debug
Nunchuck_ReadMany(0x00, tmp, 6);
printf_P(PSTR("Dumping registers 0x00 to 0x05:\n"));
for(uint8_t i = 0; i < 6; i++) {
if(tmp[i] < 0x10) {
printf("0");
}
printf("%X ", tmp[i]);
}
printf("\n\n");
} else {
printf_P(PSTR("Tatacon not found or not responding\n\n"));
}
// Switch debug
printf_P(PSTR("Key order: DL, KL, CR, KR\n"));
#endif
for (;;)
{
HID_Device_USBTask(&Keyboard_HID_Interface);
HID_Device_USBTask(&Generic_HID_Interface);
USB_USBTask();
}
}
void SetupHardware()
{
/* Disable watchdog if enabled by bootloader/fuses */
MCUSR &= ~(1 << WDRF);
wdt_disable();
#ifdef V1_BUILD
CLKPR = (1 << CLKPCE); // enable a change to CLKPR
CLKPR = 0; // set the CLKDIV to 0 - was 0011b = div by 8 taking 8MHz to 1MHz
#endif
/* Hardware Initialization */
SET(LED_DIR, DON_LED_PIN);
SET(LED_DIR, KAT_LED_PIN);
#ifdef DEBUG
init_usb_stdio();
SET(LED_PORT, DON_LED_PIN);
for(int i = 0; i < 32; i++) {
TOGGLE(LED_PORT, DON_LED_PIN);
TOGGLE(LED_PORT, KAT_LED_PIN);
_delay_ms(125);
}
#endif
if(tataConfig.ledsOn) {
// Turn them on until we init with the nunchuck
SET(LED_PORT, DON_LED_PIN);
SET(LED_PORT, KAT_LED_PIN);
}
i2c_init();
Nunchuck_Init();
USB_Init();
}
/** HID class driver callback function for the creation of HID reports to the host.
*
* \param[in] HIDInterfaceInfo Pointer to the HID class interface configuration structure being referenced
* \param[in,out] ReportID Report ID requested by the host if non-zero, otherwise callback should set to the generated report ID
* \param[in] ReportType Type of the report to create, either HID_REPORT_ITEM_In or HID_REPORT_ITEM_Feature
* \param[out] ReportData Pointer to a buffer where the created report should be stored
* \param[out] ReportSize Number of bytes written in the report (or zero if no report is to be sent)
*
* \return Boolean \c true to force the sending of the report, \c false to let the library determine if it needs to be sent
*/
bool CALLBACK_HID_Device_CreateHIDReport(USB_ClassInfo_HID_Device_t* const HIDInterfaceInfo,
uint8_t* const ReportID,
const uint8_t ReportType,
void* ReportData,
uint16_t* const ReportSize)
{
if(ReportType != HID_REPORT_ITEM_In) {
*ReportSize = 0;
return false;
}
if (HIDInterfaceInfo == &Keyboard_HID_Interface) {
USB_KeyboardReport_Data_t* KeyboardReport = (USB_KeyboardReport_Data_t*)ReportData;
#ifdef DEBUG
memset(KeyboardReport, 0, sizeof(USB_KeyboardReport_Data_t));
update_switches();
if(switchesChanged) {
for(uint8_t i = 0; i < KB_SWITCHES; i++) {
printf("%d ", switches[i].state);
switches[i].lastReport = switches[i].state;
}
printf("\n");
switchesChanged = 0;
}
if(!debugDelay) {
uint8_t wroteData = make_report(KeyboardReport);
if(wroteData) {
debugDelay = DEBUG_DELAY_MS;
*ReportSize = sizeof(USB_KeyboardReport_Data_t);
return true;
} else {
*ReportSize = 0;
return false;
}
}
#else
update_switches();
if(!switchesChanged) {
*ReportSize = 0;
return false;
}
CLEAR(LED_PORT, DON_LED_PIN);
CLEAR(LED_PORT, KAT_LED_PIN);
for(uint8_t i = 0; i < KB_SWITCHES; i++) {
KeyboardReport->KeyCode[i] = switches[i].state ? tataConfig.switches[i] : 0;
switches[i].lastReport = switches[i].state;
// Update blinkenlights
if(switches[i].state) {
if(tataConfig.ledsOn) {
if(i % 2) { // odd indexes are kat, even don
SET(LED_PORT, KAT_LED_PIN);
} else {
SET(LED_PORT, DON_LED_PIN);
}
}
}
}
*ReportSize = sizeof(USB_KeyboardReport_Data_t);
switchesChanged = 0;
return true;
#endif
} else if(HIDInterfaceInfo == &Generic_HID_Interface) {
uint8_t* ConfigReport = (uint8_t*)ReportData;
memcpy(ConfigReport, &tataConfig, sizeof(tatacon_config_t));
*ReportSize = TATACON_CONFIG_BYTES;
//TOGGLE(LED_PORT, DON_LED_PIN);
return true;
}
*ReportSize = 0;
return false;
}
/** HID class driver callback function for the processing of HID reports from the host.
*
* \param[in] HIDInterfaceInfo Pointer to the HID class interface configuration structure being referenced
* \param[in] ReportID Report ID of the received report from the host
* \param[in] ReportType The type of report that the host has sent, either HID_REPORT_ITEM_Out or HID_REPORT_ITEM_Feature
* \param[in] ReportData Pointer to a buffer where the received report has been stored
* \param[in] ReportSize Size in bytes of the received HID report
*/
void CALLBACK_HID_Device_ProcessHIDReport(USB_ClassInfo_HID_Device_t* const HIDInterfaceInfo,
const uint8_t ReportID,
const uint8_t ReportType,
const void* ReportData,
const uint16_t ReportSize)
{
if(HIDInterfaceInfo == &Generic_HID_Interface && ReportType == HID_REPORT_ITEM_Out) {
uint8_t* ConfigReport = (uint8_t*)ReportData;
// So we can upgrade firmware without having to hit the button
if(ConfigReport[TATACON_CONFIG_BYTES-1] == MAGIC_RESET_NUMBER) {
// With this uncommented, reboot fails. Odd.
//USB_Disable();
cli();
// Back to the bootloader
Boot_Key = MAGIC_BOOT_KEY;
wdt_enable(WDTO_250MS);
while(1);
}
SetConfig(ConfigReport);
}
}
/** Event handler for the library USB Connection event. */
void EVENT_USB_Device_Connect(void)
{
}
/** Event handler for the library USB Disconnection event. */
void EVENT_USB_Device_Disconnect(void)
{
}
/** Event handler for the library USB Configuration Changed event. */
void EVENT_USB_Device_ConfigurationChanged(void)
{
HID_Device_ConfigureEndpoints(&Keyboard_HID_Interface);
HID_Device_ConfigureEndpoints(&Generic_HID_Interface);
USB_Device_EnableSOFEvents();
}
/** Event handler for the library USB Control Request reception event. */
void EVENT_USB_Device_ControlRequest(void)
{
HID_Device_ProcessControlRequest(&Keyboard_HID_Interface);
HID_Device_ProcessControlRequest(&Generic_HID_Interface);
}
/** Event handler for the USB device Start Of Frame event. */
void EVENT_USB_Device_StartOfFrame(void)
{
HID_Device_MillisecondElapsed(&Keyboard_HID_Interface);
HID_Device_MillisecondElapsed(&Generic_HID_Interface);
for(int i = 0; i < KB_SWITCHES; i++) {
if(switches[i].debounce) {
switches[i].debounce--;
}
}
#ifdef DEBUG
if(debugDelay) {
debugDelay--;
}
#endif
}