@ -114,24 +114,37 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign
switch ( pMessage [ 0 ] )
{
case MIDI_SYSTEM_EXCLUSIVE_BEGIN :
printf ( " SysEx data length: [%d] \n " , uint16_t ( nLength ) ) ;
printf ( " SysEx data: \n " ) ;
printf ( " MIDI%u: SysEx data length: [%d]: " , nCable , uint16_t ( nLength ) ) ;
for ( uint16_t i = 0 ; i < nLength ; i + + )
{
if ( ( i % 8 ) = = 0 )
printf ( " %04d: " , i ) ;
if ( ( i % 16 ) = = 0 )
printf ( " \n %04d: " , i ) ;
printf ( " 0x%02x " , pMessage [ i ] ) ;
if ( ( i % 8 ) = = 0 )
printf ( " \n " ) ;
}
printf ( " \n " ) ;
break ;
default :
printf ( " Unhandled MIDI event type %0x02x \n " , pMessage [ 0 ] ) ;
printf ( " MIDI%u: Unhandled MIDI event type %0x02x\n " , nCable , pMessage [ 0 ] ) ;
}
break ;
}
}
// Only for debugging:
/*
if ( pMessage [ 0 ] = = MIDI_SYSTEM_EXCLUSIVE_BEGIN )
{
printf ( " MIDI%u: SysEx data length: [%d]: " , nCable , uint16_t ( nLength ) ) ;
for ( uint16_t i = 0 ; i < nLength ; i + + )
{
if ( ( i % 16 ) = = 0 )
printf ( " \n %04d: " , i ) ;
printf ( " 0x%02x " , pMessage [ i ] ) ;
}
printf ( " \n " ) ;
}
*/
// Handle MIDI Thru
if ( m_DeviceName . compare ( m_pConfig - > GetMIDIThruIn ( ) ) = = 0 )
{
@ -150,6 +163,8 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign
return ;
}
m_MIDISpinLock . Acquire ( ) ;
u8 ucStatus = pMessage [ 0 ] ;
u8 ucChannel = ucStatus & 0x0F ;
u8 ucType = ucStatus > > 4 ;
@ -157,17 +172,24 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign
// GLOBAL MIDI SYSEX
if ( pMessage [ 0 ] = = MIDI_SYSTEM_EXCLUSIVE_BEGIN & & pMessage [ 3 ] = = 0x04 & & pMessage [ 4 ] = = 0x01 & & pMessage [ nLength - 1 ] = = MIDI_SYSTEM_EXCLUSIVE_END ) // MASTER VOLUME
{
float32_t nMasterVolume = ( pMessage [ 5 ] & ( pMessage [ 6 ] < < 7 ) ) / ( 1 < < 14 ) ;
float32_t nMasterVolume = ( ( pMessage [ 5 ] & 0x7c ) & ( ( pMessage [ 6 ] & 0x7c ) < < 7 ) ) / ( 1 < < 14 ) ;
LOGNOTE ( " Master volume: %f " , nMasterVolume ) ;
// TODO: Handle global master volume
m_pSynthesizer - > setMasterVolume ( nMasterVolume ) ;
}
else
{
for ( unsigned nTG = 0 ; nTG < CConfig : : ToneGenerators ; nTG + + )
{
if ( ucStatus = = MIDI_SYSTEM_EXCLUSIVE_BEGIN )
{
// MIDI SYSEX per MIDI channel
if ( ucStatus = = MIDI_SYSTEM_EXCLUSIVE_BEGIN & & m_ChannelMap [ nTG ] = = pMessage [ 2 ] & 0x07 )
HandleSystemExclusive ( pMessage , nLength , nTG ) ;
uint8_t ucSysExChannel = ( pMessage [ 2 ] & 0x07 ) ;
if ( m_ChannelMap [ nTG ] = = ucSysExChannel | | m_ChannelMap [ nTG ] = = OmniMode )
{
LOGNOTE ( " MIDI-SYSEX: channel: %u, len: %u, TG: %u " , m_ChannelMap [ nTG ] , nLength , nTG ) ;
HandleSystemExclusive ( pMessage , nLength , nCable , nTG ) ;
}
}
else
{
if ( m_ChannelMap [ nTG ] = = ucChannel
@ -293,6 +315,7 @@ void CMIDIDevice::MIDIMessageHandler (const u8 *pMessage, size_t nLength, unsign
}
}
}
m_MIDISpinLock . Release ( ) ;
}
void CMIDIDevice : : AddDevice ( const char * pDeviceName )
@ -306,7 +329,7 @@ void CMIDIDevice::AddDevice (const char *pDeviceName)
s_DeviceMap . insert ( std : : pair < std : : string , CMIDIDevice * > ( pDeviceName , this ) ) ;
}
void CMIDIDevice : : HandleSystemExclusive ( const uint8_t * pMessage , const size_t nLength , const uint8_t nTG )
void CMIDIDevice : : HandleSystemExclusive ( const uint8_t * pMessage , const size_t nLength , const unsigned nCable , const u int8_t nTG )
{
int16_t sysex_return ;
@ -319,33 +342,33 @@ void CMIDIDevice::HandleSystemExclusive(const uint8_t* pMessage, const size_t nL
LOGERR ( " SysEx end status byte not detected. " ) ;
break ;
case - 2 :
LOGERR ( " E: SysEx vendor not Yamaha." ) ;
LOGERR ( " SysEx vendor not Yamaha. " ) ;
break ;
case - 3 :
LOGERR ( " E: Unknown SysEx parameter change." ) ;
LOGERR ( " Unknown SysEx parameter change. " ) ;
break ;
case - 4 :
LOGERR ( " Unknown SysEx voice or function." ) ;
LOGERR ( " Unknown SysEx voice or function. " ) ;
break ;
case - 5 :
LOGERR ( " E: Not a SysEx voice bulk upload." ) ;
LOGERR ( " Not a SysEx voice bulk upload. " ) ;
break ;
case - 6 :
LOGERR ( " E: Wrong length for SysEx voice bulk upload (not 155)." ) ;
LOGERR ( " Wrong length for SysEx voice bulk upload (not 155). " ) ;
break ;
case - 7 :
LOGERR ( " E: Checksum error for one voice." ) ;
LOGERR ( " Checksum error for one voice. " ) ;
break ;
case - 8 :
LOGERR ( " E: Not a SysEx bank bulk upload." ) ;
LOGERR ( " Not a SysEx bank bulk upload. " ) ;
break ;
case - 9 :
LOGERR ( " E: Wrong length for SysEx bank bulk upload (not 4096)." ) ;
LOGERR ( " Wrong length for SysEx bank bulk upload (not 4096). " ) ;
case - 10 :
LOGERR ( " E: Checksum error for bank." ) ;
LOGERR ( " Checksum error for bank. " ) ;
break ;
case - 11 :
LOGERR ( " E: Unknown SysEx message." ) ;
LOGERR ( " Unknown SysEx message. " ) ;
break ;
case 64 :
LOGDBG ( " SysEx Function parameter change: %d Value %d " , pMessage [ 4 ] , pMessage [ 5 ] ) ;
@ -407,7 +430,6 @@ void CMIDIDevice::HandleSystemExclusive(const uint8_t* pMessage, const size_t nL
// load sysex-data into voice memory
LOGDBG ( " One Voice bulk upload " ) ;
m_pSynthesizer - > loadVoiceParameters ( pMessage , nTG ) ;
break ;
case 200 :
LOGDBG ( " Bank bulk upload. " ) ;
@ -415,9 +437,41 @@ void CMIDIDevice::HandleSystemExclusive(const uint8_t* pMessage, const size_t nL
LOGNOTE ( " Currently code for storing a bulk bank upload is missing! " ) ;
break ;
default :
LOGDBG ( " SysEx voice parameter change: %d value: %d " , pMessage [ 4 ] + ( ( pMessage [ 3 ] & 0x03 ) * 128 ) , pMessage [ 5 ] ) ;
if ( sysex_return > = 300 & & sysex_return < 500 )
{
LOGDBG ( " SysEx voice parameter change: Parameter %d value: %d " , pMessage [ 4 ] + ( ( pMessage [ 3 ] & 0x03 ) * 128 ) , pMessage [ 5 ] ) ;
m_pSynthesizer - > setVoiceDataElement ( pMessage [ 4 ] + ( ( pMessage [ 3 ] & 0x03 ) * 128 ) , pMessage [ 5 ] , nTG ) ;
switch ( pMessage [ 4 ] + ( ( pMessage [ 3 ] & 0x03 ) * 128 ) )
{
case 134 :
m_pSynthesizer - > notesOff ( 0 , nTG ) ;
break ;
}
}
else if ( sysex_return > = 500 & & sysex_return < 600 )
{
LOGDBG ( " SysEx send voice %u request " , sysex_return - 500 ) ;
SendSystemExclusiveVoice ( sysex_return - 500 , nCable , nTG ) ;
}
break ;
}
}
void CMIDIDevice : : SendSystemExclusiveVoice ( uint8_t nVoice , const unsigned nCable , uint8_t nTG )
{
uint8_t voicedump [ 163 ] ;
LOGDBG ( " Sending SysEx voice %u " , nVoice ) ;
// Get voice sysex dump from TG
m_pSynthesizer - > getSysExVoiceDump ( voicedump , nTG ) ;
TDeviceMap : : const_iterator Iterator ;
// send voice dump to all MIDI interfaces
for ( Iterator = s_DeviceMap . begin ( ) ; Iterator ! = s_DeviceMap . end ( ) ; + + Iterator )
{
Iterator - > second - > Send ( voicedump , sizeof ( voicedump ) * sizeof ( uint8_t ) , nCable ) ;
LOGNOTE ( " Send SYSEX voice dump to \" %s \" \n " , Iterator - > first ) ;
}
}