/* * rhmidi * * Copyright (C) 2014 Christian Pointner * * This file is part of rhmidi. * * rhmidi is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * rhmidi is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with rhmidi. If not, see . */ #include #include #include #include #include /* LUFA Library Copyright (C) Dean Camera, 2012. dean [at] fourwalledcubicle [dot] com www.lufa-lib.org */ #include #include #include "lufa-descriptor-midi.h" #include "LUFA/Drivers/Peripheral/Serial.h" /** LUFA MIDI Class driver interface configuration and state information. This structure is * passed to all MIDI Class driver functions, so that multiple instances of the same class * within a device can be differentiated from one another. */ USB_ClassInfo_MIDI_Device_t MIDI_Interface = { .Config = { .StreamingInterfaceNumber = 1, .DataINEndpointNumber = MIDI_STREAM_IN_EPNUM, .DataINEndpointSize = MIDI_STREAM_EPSIZE, .DataINEndpointDoubleBank = false, .DataOUTEndpointNumber = MIDI_STREAM_OUT_EPNUM, .DataOUTEndpointSize = MIDI_STREAM_EPSIZE, .DataOUTEndpointDoubleBank = false, }, }; /** Event handler for the library USB Configuration Changed event. */ void EVENT_USB_Device_ConfigurationChanged(void) { MIDI_Device_ConfigureEndpoints(&MIDI_Interface); } /** Event handler for the library USB Control Request reception event. */ void EVENT_USB_Device_ControlRequest(void) { MIDI_Device_ProcessControlRequest(&MIDI_Interface); } /* end LUFA MIDI Class driver specific definitions*/ #include "util.h" #include "keypad.h" #include "eventqueue.h" #include "gpio.h" static void process_incoming_midi(void) { MIDI_EventPacket_t ReceivedMIDIEvent; while(MIDI_Device_ReceiveEventPacket(&MIDI_Interface, &ReceivedMIDIEvent)) { uint8_t cmd = ReceivedMIDIEvent.Data1 & 0xF0; uint8_t note = ReceivedMIDIEvent.Data2; uint8_t value = ReceivedMIDIEvent.Data3 & 0x7F; if(note < 16 || note == 0x7F) { // TODO: hardcoded value if(cmd == MIDI_COMMAND_NOTE_ON) { if(value <= 2) { switch(value) { case 0: keypad_led_off(note); break; case 1: keypad_led_on(note); break; case 2: keypad_led_toggle(note); break; } } else if(value <= 0x7F) { keypad_led_blink(note, value); } } } } } static void process_outgoing_midi(void) { MIDI_EventPacket_t MIDIEventOn = (MIDI_EventPacket_t) { .CableNumber = 0, // TODO: hardcoded value .Command = (MIDI_COMMAND_NOTE_ON >> 4), .Data1 = MIDI_COMMAND_NOTE_ON | 0, .Data2 = 0, .Data3 = 0x7F, }; MIDI_EventPacket_t MIDIEventOff = (MIDI_EventPacket_t) { .CableNumber = 0, // TODO: hardcoded value .Command = (MIDI_COMMAND_NOTE_OFF >> 4), .Data1 = MIDI_COMMAND_NOTE_OFF | 0, .Data2 = 0, .Data3 = 0, }; uint8_t key, state; while(eventqueue_pop(&key, &state)) { if (USB_DeviceState == DEVICE_STATE_Configured) { MIDI_EventPacket_t* MIDIEvent = state ? &MIDIEventOn : &MIDIEventOff; MIDIEvent->Data2 = key; MIDI_Device_SendEventPacket(&MIDI_Interface, MIDIEvent); } } if (USB_DeviceState == DEVICE_STATE_Configured) MIDI_Device_Flush(&MIDI_Interface); } int main(void) { MCUSR &= ~(1 << WDRF); wdt_disable(); cpu_init(); eventqueue_init(); keypad_init(); gpio_init(); USB_Init(); sei(); for(;;) { process_incoming_midi(); keypad_task(); gpio_task(); process_outgoing_midi(); MIDI_Device_USBTask(&MIDI_Interface); USB_USBTask(); } }