mirror of https://github.com/probonopd/MiniDexed
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
233 lines
4.3 KiB
233 lines
4.3 KiB
3 years ago
|
//
|
||
|
// minidexed.cpp
|
||
|
//
|
||
|
#include "minidexed.h"
|
||
|
#include <circle/devicenameservice.h>
|
||
|
|
||
|
#define MIDI_NOTE_OFF 0b1000
|
||
|
#define MIDI_NOTE_ON 0b1001
|
||
|
|
||
|
CMiniDexed *CMiniDexed::s_pThis = 0;
|
||
|
|
||
|
bool CMiniDexed::Initialize (void)
|
||
|
{
|
||
|
if (!m_Serial.Initialize(31250))
|
||
|
{
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
m_bUseSerial = true;
|
||
|
|
||
|
activate();
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void CMiniDexed::Process(boolean bPlugAndPlayUpdated)
|
||
|
{
|
||
|
if (m_pMIDIDevice != 0)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (bPlugAndPlayUpdated)
|
||
|
{
|
||
|
m_pMIDIDevice =
|
||
|
(CUSBMIDIDevice *) CDeviceNameService::Get ()->GetDevice ("umidi1", FALSE);
|
||
|
if (m_pMIDIDevice != 0)
|
||
|
{
|
||
|
m_pMIDIDevice->RegisterRemovedHandler (USBDeviceRemovedHandler);
|
||
|
m_pMIDIDevice->RegisterPacketHandler (MIDIPacketHandler);
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
m_PCKeyboard.Process (bPlugAndPlayUpdated);
|
||
|
|
||
|
if (!m_bUseSerial)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Read serial MIDI data
|
||
|
u8 Buffer[20];
|
||
|
int nResult = m_Serial.Read (Buffer, sizeof Buffer);
|
||
|
if (nResult <= 0)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Process MIDI messages
|
||
|
// See: https://www.midi.org/specifications/item/table-1-summary-of-midi-message
|
||
|
for (int i = 0; i < nResult; i++)
|
||
|
{
|
||
|
u8 uchData = Buffer[i];
|
||
|
|
||
|
switch (m_nSerialState)
|
||
|
{
|
||
|
case 0:
|
||
|
MIDIRestart:
|
||
|
if ((uchData & 0xE0) == 0x80) // Note on or off, all channels
|
||
|
{
|
||
|
m_SerialMessage[m_nSerialState++] = uchData;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case 1:
|
||
|
case 2:
|
||
|
if (uchData & 0x80) // got status when parameter expected
|
||
|
{
|
||
|
m_nSerialState = 0;
|
||
|
|
||
|
goto MIDIRestart;
|
||
|
}
|
||
|
|
||
|
m_SerialMessage[m_nSerialState++] = uchData;
|
||
|
|
||
|
if (m_nSerialState == 3) // message is complete
|
||
|
{
|
||
|
MIDIPacketHandler (0, m_SerialMessage, sizeof m_SerialMessage);
|
||
|
|
||
|
m_nSerialState = 0;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
assert (0);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void CMiniDexed::MIDIPacketHandler (unsigned nCable, u8 *pPacket, unsigned nLength)
|
||
|
{
|
||
|
assert (s_pThis != 0);
|
||
|
|
||
|
// The packet contents are just normal MIDI data - see
|
||
|
// https://www.midi.org/specifications/item/table-1-summary-of-midi-message
|
||
|
|
||
|
if (nLength < 3)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
u8 ucStatus = pPacket[0];
|
||
|
//u8 ucChannel = ucStatus & 0x0F;
|
||
|
u8 ucType = ucStatus >> 4;
|
||
|
u8 ucKeyNumber = pPacket[1];
|
||
|
u8 ucVelocity = pPacket[2];
|
||
|
|
||
|
if (ucType == MIDI_NOTE_ON)
|
||
|
{
|
||
|
s_pThis->keydown(ucKeyNumber,ucVelocity);
|
||
|
}
|
||
|
else if (ucType == MIDI_NOTE_OFF)
|
||
|
{
|
||
|
s_pThis->keyup(ucKeyNumber);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void CMiniDexed::USBDeviceRemovedHandler (CDevice *pDevice, void *pContext)
|
||
|
{
|
||
|
if (s_pThis->m_pMIDIDevice == (CUSBMIDIDevice *) pDevice)
|
||
|
{
|
||
|
s_pThis->m_pMIDIDevice = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
bool CMiniDexedPWM::Initialize (void)
|
||
|
{
|
||
|
if (!CMiniDexed::Initialize())
|
||
|
{
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return Start ();
|
||
|
}
|
||
|
|
||
|
unsigned CMiniDexedPWM::GetChunk(u32 *pBuffer, unsigned nChunkSize)
|
||
|
{
|
||
|
unsigned nResult = nChunkSize;
|
||
|
|
||
|
int16_t int16_buf[nChunkSize/2];
|
||
|
|
||
|
getSamples(nChunkSize/2, int16_buf);
|
||
|
|
||
|
for (unsigned i = 0; nChunkSize > 0; nChunkSize -= 2) // fill the whole buffer
|
||
|
{
|
||
|
s32 nSample = int16_buf[i++];
|
||
|
nSample += 32768;
|
||
|
nSample *= GetRangeMax()/2;
|
||
|
nSample /= 32768;
|
||
|
|
||
|
*pBuffer++ = nSample; // 2 stereo channels
|
||
|
*pBuffer++ = nSample;
|
||
|
}
|
||
|
return(nResult);
|
||
|
};
|
||
|
|
||
|
bool CMiniDexedI2S::Initialize (void)
|
||
|
{
|
||
|
if (!CMiniDexed::Initialize())
|
||
|
{
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return Start ();
|
||
|
}
|
||
|
|
||
|
unsigned CMiniDexedI2S::GetChunk(u32 *pBuffer, unsigned nChunkSize)
|
||
|
{
|
||
|
unsigned nResult = nChunkSize;
|
||
|
|
||
|
int16_t int16_buf[nChunkSize/2];
|
||
|
|
||
|
getSamples(nChunkSize/2, int16_buf);
|
||
|
|
||
|
for (unsigned i = 0; nChunkSize > 0; nChunkSize -= 2) // fill the whole buffer
|
||
|
{
|
||
|
s32 nSample = int16_buf[i++];
|
||
|
nSample <<= 8;
|
||
|
|
||
|
*pBuffer++ = nSample; // 2 stereo channels
|
||
|
*pBuffer++ = nSample;
|
||
|
}
|
||
|
return(nResult);
|
||
|
};
|
||
|
|
||
|
bool CMiniDexedHDMI::Initialize (void)
|
||
|
{
|
||
|
if (!CMiniDexed::Initialize())
|
||
|
{
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return Start ();
|
||
|
}
|
||
|
|
||
|
unsigned CMiniDexedHDMI::GetChunk(u32 *pBuffer, unsigned nChunkSize)
|
||
|
{
|
||
|
unsigned nResult = nChunkSize;
|
||
|
|
||
|
int16_t int16_buf[nChunkSize/2];
|
||
|
unsigned nFrame = 0;
|
||
|
|
||
|
getSamples(nChunkSize/2, int16_buf);
|
||
|
|
||
|
for (unsigned i = 0; nChunkSize > 0; nChunkSize -= 2) // fill the whole buffer
|
||
|
{
|
||
|
s32 nSample = int16_buf[i++];
|
||
|
nSample <<= 8;
|
||
|
|
||
|
nSample = ConvertIEC958Sample (nSample, nFrame);
|
||
|
|
||
|
if (++nFrame == IEC958_FRAMES_PER_BLOCK)
|
||
|
nFrame = 0;
|
||
|
|
||
|
*pBuffer++ = nSample; // 2 stereo channels
|
||
|
*pBuffer++ = nSample;
|
||
|
}
|
||
|
return(nResult);
|
||
|
};
|